Giant blob of minor changes
[dotfiles/.git] / .config / coc / extensions / node_modules / coc-html / node_modules / typescript / lib / tsserverlibrary.js
index 6c6a3e40b5ac9a8930bf90ece3de90ec10243eaf..a74476c9a7e723825a899c451331635ba7447c30 100644 (file)
@@ -32,6 +32,10 @@ var __assign = (this && this.__assign) || function () {
     };
     return __assign.apply(this, arguments);
 };
+var __makeTemplateObject = (this && this.__makeTemplateObject) || function (cooked, raw) {
+    if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; }
+    return cooked;
+};
 var __generator = (this && this.__generator) || function (thisArg, body) {
     var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
     return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
@@ -59,10 +63,6 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
         if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
     }
 };
-var __makeTemplateObject = (this && this.__makeTemplateObject) || function (cooked, raw) {
-    if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; }
-    return cooked;
-};
 var __rest = (this && this.__rest) || function (s, e) {
     var t = {};
     for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
@@ -78,7 +78,7 @@ var __extends = (this && this.__extends) || (function () {
     var extendStatics = function (d, b) {
         extendStatics = Object.setPrototypeOf ||
             ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
-            function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
+            function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };
         return extendStatics(d, b);
     };
     return function (d, b) {
@@ -90,178 +90,205 @@ var __extends = (this && this.__extends) || (function () {
 /* @internal */
 var ts;
 (function (ts) {
-    function createMapShim() {
-        /** Create a MapLike with good performance. */
-        function createDictionaryObject() {
-            var map = Object.create(/*prototype*/ null); // eslint-disable-line no-null/no-null
-            // Using 'delete' on an object causes V8 to put the object in dictionary mode.
-            // This disables creation of hidden classes, which are expensive when an object is
-            // constantly changing shape.
-            map.__ = undefined;
-            delete map.__;
-            return map;
+    function createMapData() {
+        var sentinel = {};
+        sentinel.prev = sentinel;
+        return { head: sentinel, tail: sentinel, size: 0 };
+    }
+    function createMapEntry(key, value) {
+        return { key: key, value: value, next: undefined, prev: undefined };
+    }
+    function sameValueZero(x, y) {
+        // Treats -0 === 0 and NaN === NaN
+        return x === y || x !== x && y !== y;
+    }
+    function getPrev(entry) {
+        var prev = entry.prev;
+        // Entries without a 'prev' have been removed from the map.
+        // An entry whose 'prev' points to itself is the head of the list and is invalid here.
+        if (!prev || prev === entry)
+            throw new Error("Illegal state");
+        return prev;
+    }
+    function getNext(entry) {
+        while (entry) {
+            // Entries without a 'prev' have been removed from the map. Their 'next'
+            // pointer should point to the previous entry prior to deletion and
+            // that entry should be skipped to resume iteration.
+            var skipNext = !entry.prev;
+            entry = entry.next;
+            if (skipNext) {
+                continue;
+            }
+            return entry;
         }
-        var MapIterator = /** @class */ (function () {
-            function MapIterator(currentEntry, selector) {
-                this.currentEntry = currentEntry;
-                this.selector = selector;
-            }
-            MapIterator.prototype.next = function () {
-                // Navigate to the next entry.
-                while (this.currentEntry) {
-                    var skipNext = !!this.currentEntry.skipNext;
-                    this.currentEntry = this.currentEntry.nextEntry;
-                    if (!skipNext) {
-                        break;
-                    }
-                }
-                if (this.currentEntry) {
-                    return { value: this.selector(this.currentEntry.key, this.currentEntry.value), done: false };
-                }
-                else {
-                    return { value: undefined, done: true };
-                }
-            };
-            return MapIterator;
-        }());
-        return /** @class */ (function () {
-            function class_1() {
-                this.data = createDictionaryObject();
-                this.size = 0;
-                // Create a first (stub) map entry that will not contain a key
-                // and value but serves as starting point for iterators.
-                this.firstEntry = {};
-                // When the map is empty, the last entry is the same as the
-                // first one.
-                this.lastEntry = this.firstEntry;
-            }
-            class_1.prototype.get = function (key) {
-                var entry = this.data[key];
-                return entry && entry.value;
-            };
-            class_1.prototype.set = function (key, value) {
-                if (!this.has(key)) {
-                    this.size++;
-                    // Create a new entry that will be appended at the
-                    // end of the linked list.
-                    var newEntry = {
-                        key: key,
-                        value: value
-                    };
-                    this.data[key] = newEntry;
-                    // Adjust the references.
-                    var previousLastEntry = this.lastEntry;
-                    previousLastEntry.nextEntry = newEntry;
-                    newEntry.previousEntry = previousLastEntry;
-                    this.lastEntry = newEntry;
+    }
+    function getEntry(data, key) {
+        // We walk backwards from 'tail' to prioritize recently added entries.
+        // We skip 'head' because it is an empty entry used to track iteration start.
+        for (var entry = data.tail; entry !== data.head; entry = getPrev(entry)) {
+            if (sameValueZero(entry.key, key)) {
+                return entry;
+            }
+        }
+    }
+    function addOrUpdateEntry(data, key, value) {
+        var existing = getEntry(data, key);
+        if (existing) {
+            existing.value = value;
+            return;
+        }
+        var entry = createMapEntry(key, value);
+        entry.prev = data.tail;
+        data.tail.next = entry;
+        data.tail = entry;
+        data.size++;
+        return entry;
+    }
+    function deleteEntry(data, key) {
+        // We walk backwards from 'tail' to prioritize recently added entries.
+        // We skip 'head' because it is an empty entry used to track iteration start.
+        for (var entry = data.tail; entry !== data.head; entry = getPrev(entry)) {
+            // all entries in the map should have a 'prev' pointer.
+            if (entry.prev === undefined)
+                throw new Error("Illegal state");
+            if (sameValueZero(entry.key, key)) {
+                if (entry.next) {
+                    entry.next.prev = entry.prev;
                 }
                 else {
-                    this.data[key].value = value;
+                    // an entry in the map without a 'next' pointer must be the 'tail'.
+                    if (data.tail !== entry)
+                        throw new Error("Illegal state");
+                    data.tail = entry.prev;
                 }
-                return this;
-            };
-            class_1.prototype.has = function (key) {
-                // eslint-disable-next-line no-in-operator
-                return key in this.data;
-            };
-            class_1.prototype.delete = function (key) {
-                if (this.has(key)) {
-                    this.size--;
-                    var entry = this.data[key];
-                    delete this.data[key];
-                    // Adjust the linked list references of the neighbor entries.
-                    var previousEntry = entry.previousEntry;
-                    previousEntry.nextEntry = entry.nextEntry;
-                    if (entry.nextEntry) {
-                        entry.nextEntry.previousEntry = previousEntry;
-                    }
-                    // When the deleted entry was the last one, we need to
-                    // adjust the lastEntry reference.
-                    if (this.lastEntry === entry) {
-                        this.lastEntry = previousEntry;
-                    }
-                    // Adjust the forward reference of the deleted entry
-                    // in case an iterator still references it. This allows us
-                    // to throw away the entry, but when an active iterator
-                    // (which points to the current entry) continues, it will
-                    // navigate to the entry that originally came before the
-                    // current one and skip it.
-                    entry.previousEntry = undefined;
-                    entry.nextEntry = previousEntry;
-                    entry.skipNext = true;
-                    return true;
+                entry.prev.next = entry.next;
+                entry.next = entry.prev;
+                entry.prev = undefined;
+                data.size--;
+                return entry;
+            }
+        }
+    }
+    function clearEntries(data) {
+        var node = data.tail;
+        while (node !== data.head) {
+            var prev = getPrev(node);
+            node.next = data.head;
+            node.prev = undefined;
+            node = prev;
+        }
+        data.head.next = undefined;
+        data.tail = data.head;
+        data.size = 0;
+    }
+    function forEachEntry(data, action) {
+        var entry = data.head;
+        while (entry) {
+            entry = getNext(entry);
+            if (entry) {
+                action(entry.value, entry.key);
+            }
+        }
+    }
+    function forEachIteration(iterator, action) {
+        if (iterator) {
+            for (var step = iterator.next(); !step.done; step = iterator.next()) {
+                action(step.value);
+            }
+        }
+    }
+    function createIteratorData(data, selector) {
+        return { current: data.head, selector: selector };
+    }
+    function iteratorNext(data) {
+        // Navigate to the next entry.
+        data.current = getNext(data.current);
+        if (data.current) {
+            return { value: data.selector(data.current.key, data.current.value), done: false };
+        }
+        else {
+            return { value: undefined, done: true };
+        }
+    }
+    /* @internal */
+    var ShimCollections;
+    (function (ShimCollections) {
+        function createMapShim(getIterator) {
+            var MapIterator = /** @class */ (function () {
+                function MapIterator(data, selector) {
+                    this._data = createIteratorData(data, selector);
+                }
+                MapIterator.prototype.next = function () { return iteratorNext(this._data); };
+                return MapIterator;
+            }());
+            return /** @class */ (function () {
+                function Map(iterable) {
+                    var _this = this;
+                    this._mapData = createMapData();
+                    forEachIteration(getIterator(iterable), function (_a) {
+                        var key = _a[0], value = _a[1];
+                        return _this.set(key, value);
+                    });
                 }
-                return false;
-            };
-            class_1.prototype.clear = function () {
-                this.data = createDictionaryObject();
-                this.size = 0;
-                // Reset the linked list. Note that we must adjust the forward
-                // references of the deleted entries to ensure iterators stuck
-                // in the middle of the list don't continue with deleted entries,
-                // but can continue with new entries added after the clear()
-                // operation.
-                var firstEntry = this.firstEntry;
-                var currentEntry = firstEntry.nextEntry;
-                while (currentEntry) {
-                    var nextEntry = currentEntry.nextEntry;
-                    currentEntry.previousEntry = undefined;
-                    currentEntry.nextEntry = firstEntry;
-                    currentEntry.skipNext = true;
-                    currentEntry = nextEntry;
-                }
-                firstEntry.nextEntry = undefined;
-                this.lastEntry = firstEntry;
-            };
-            class_1.prototype.keys = function () {
-                return new MapIterator(this.firstEntry, function (key) { return key; });
-            };
-            class_1.prototype.values = function () {
-                return new MapIterator(this.firstEntry, function (_key, value) { return value; });
-            };
-            class_1.prototype.entries = function () {
-                return new MapIterator(this.firstEntry, function (key, value) { return [key, value]; });
-            };
-            class_1.prototype.forEach = function (action) {
-                var iterator = this.entries();
-                while (true) {
-                    var iterResult = iterator.next();
-                    if (iterResult.done) {
-                        break;
-                    }
-                    var _a = iterResult.value, key = _a[0], value = _a[1];
-                    action(value, key);
+                Object.defineProperty(Map.prototype, "size", {
+                    get: function () { return this._mapData.size; },
+                    enumerable: false,
+                    configurable: true
+                });
+                Map.prototype.get = function (key) { var _a; return (_a = getEntry(this._mapData, key)) === null || _a === void 0 ? void 0 : _a.value; };
+                Map.prototype.set = function (key, value) { return addOrUpdateEntry(this._mapData, key, value), this; };
+                Map.prototype.has = function (key) { return !!getEntry(this._mapData, key); };
+                Map.prototype.delete = function (key) { return !!deleteEntry(this._mapData, key); };
+                Map.prototype.clear = function () { clearEntries(this._mapData); };
+                Map.prototype.keys = function () { return new MapIterator(this._mapData, function (key, _value) { return key; }); };
+                Map.prototype.values = function () { return new MapIterator(this._mapData, function (_key, value) { return value; }); };
+                Map.prototype.entries = function () { return new MapIterator(this._mapData, function (key, value) { return [key, value]; }); };
+                Map.prototype.forEach = function (action) { forEachEntry(this._mapData, action); };
+                return Map;
+            }());
+        }
+        ShimCollections.createMapShim = createMapShim;
+        function createSetShim(getIterator) {
+            var SetIterator = /** @class */ (function () {
+                function SetIterator(data, selector) {
+                    this._data = createIteratorData(data, selector);
                 }
-            };
-            return class_1;
-        }());
-    }
-    ts.createMapShim = createMapShim;
+                SetIterator.prototype.next = function () { return iteratorNext(this._data); };
+                return SetIterator;
+            }());
+            return /** @class */ (function () {
+                function Set(iterable) {
+                    var _this = this;
+                    this._mapData = createMapData();
+                    forEachIteration(getIterator(iterable), function (value) { return _this.add(value); });
+                }
+                Object.defineProperty(Set.prototype, "size", {
+                    get: function () { return this._mapData.size; },
+                    enumerable: false,
+                    configurable: true
+                });
+                Set.prototype.add = function (value) { return addOrUpdateEntry(this._mapData, value, value), this; };
+                Set.prototype.has = function (value) { return !!getEntry(this._mapData, value); };
+                Set.prototype.delete = function (value) { return !!deleteEntry(this._mapData, value); };
+                Set.prototype.clear = function () { clearEntries(this._mapData); };
+                Set.prototype.keys = function () { return new SetIterator(this._mapData, function (key, _value) { return key; }); };
+                Set.prototype.values = function () { return new SetIterator(this._mapData, function (_key, value) { return value; }); };
+                Set.prototype.entries = function () { return new SetIterator(this._mapData, function (key, value) { return [key, value]; }); };
+                Set.prototype.forEach = function (action) { forEachEntry(this._mapData, action); };
+                return Set;
+            }());
+        }
+        ShimCollections.createSetShim = createSetShim;
+    })(ShimCollections = ts.ShimCollections || (ts.ShimCollections = {}));
 })(ts || (ts = {}));
 var ts;
 (function (ts) {
     // WARNING: The script `configurePrerelease.ts` uses a regexp to parse out these values.
     // If changing the text in this section, be sure to test `configurePrerelease` too.
-    ts.versionMajorMinor = "3.9";
+    ts.versionMajorMinor = "4.1";
     /** The version of the TypeScript compiler release */
-    ts.version = "3.9.5";
-    /**
-     * Returns the native Map implementation if it is available and compatible (i.e. supports iteration).
-     */
-    /* @internal */
-    function tryGetNativeMap() {
-        // eslint-disable-next-line no-in-operator
-        return typeof Map !== "undefined" && "entries" in Map.prototype ? Map : undefined;
-    }
-    ts.tryGetNativeMap = tryGetNativeMap;
-    /* @internal */
-    ts.Map = tryGetNativeMap() || (function () {
-        // NOTE: createMapShim will be defined for typescriptServices.js but not for tsc.js, so we must test for it.
-        if (typeof ts.createMapShim === "function") {
-            return ts.createMapShim();
-        }
-        throw new Error("TypeScript requires an environment that provides a compatible native Map implementation.");
-    })();
+    ts.version = "4.1.3";
     /* @internal */
     var Comparison;
     (function (Comparison) {
@@ -269,27 +296,65 @@ var ts;
         Comparison[Comparison["EqualTo"] = 0] = "EqualTo";
         Comparison[Comparison["GreaterThan"] = 1] = "GreaterThan";
     })(Comparison = ts.Comparison || (ts.Comparison = {}));
+    /* @internal */
+    var NativeCollections;
+    (function (NativeCollections) {
+        /**
+         * Returns the native Map implementation if it is available and compatible (i.e. supports iteration).
+         */
+        function tryGetNativeMap() {
+            // Internet Explorer's Map doesn't support iteration, so don't use it.
+            // eslint-disable-next-line no-in-operator
+            return typeof Map !== "undefined" && "entries" in Map.prototype && new Map([[0, 0]]).size === 1 ? Map : undefined;
+        }
+        NativeCollections.tryGetNativeMap = tryGetNativeMap;
+        /**
+         * Returns the native Set implementation if it is available and compatible (i.e. supports iteration).
+         */
+        function tryGetNativeSet() {
+            // Internet Explorer's Set doesn't support iteration, so don't use it.
+            // eslint-disable-next-line no-in-operator
+            return typeof Set !== "undefined" && "entries" in Set.prototype && new Set([0]).size === 1 ? Set : undefined;
+        }
+        NativeCollections.tryGetNativeSet = tryGetNativeSet;
+    })(NativeCollections = ts.NativeCollections || (ts.NativeCollections = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
 (function (ts) {
+    function getCollectionImplementation(name, nativeFactory, shimFactory) {
+        var _a;
+        // NOTE: ts.ShimCollections will be defined for typescriptServices.js but not for tsc.js, so we must test for it.
+        var constructor = (_a = ts.NativeCollections[nativeFactory]()) !== null && _a !== void 0 ? _a : ts.ShimCollections === null || ts.ShimCollections === void 0 ? void 0 : ts.ShimCollections[shimFactory](getIterator);
+        if (constructor)
+            return constructor;
+        throw new Error("TypeScript requires an environment that provides a compatible native " + name + " implementation.");
+    }
+    ts.Map = getCollectionImplementation("Map", "tryGetNativeMap", "createMapShim");
+    ts.Set = getCollectionImplementation("Set", "tryGetNativeSet", "createSetShim");
+    function getIterator(iterable) {
+        if (iterable) {
+            if (isArray(iterable))
+                return arrayIterator(iterable);
+            if (iterable instanceof ts.Map)
+                return iterable.entries();
+            if (iterable instanceof ts.Set)
+                return iterable.values();
+            throw new Error("Iteration not supported.");
+        }
+    }
+    ts.getIterator = getIterator;
     ts.emptyArray = [];
-    /** Create a new map. */
+    ts.emptyMap = new ts.Map();
+    ts.emptySet = new ts.Set();
     function createMap() {
         return new ts.Map();
     }
     ts.createMap = createMap;
-    /** Create a new map from an array of entries. */
-    function createMapFromEntries(entries) {
-        var map = createMap();
-        for (var _i = 0, entries_1 = entries; _i < entries_1.length; _i++) {
-            var _a = entries_1[_i], key = _a[0], value = _a[1];
-            map.set(key, value);
-        }
-        return map;
-    }
-    ts.createMapFromEntries = createMapFromEntries;
-    /** Create a new map from a template object is provided, the map will copy entries from it. */
+    /**
+     * Create a new map from a template object is provided, the map will copy entries from it.
+     * @deprecated Use `new Map(getEntries(template))` instead.
+     */
     function createMapFromTemplate(template) {
         var map = new ts.Map();
         // Copies keys/values from template. Note that for..in will not throw if
@@ -365,6 +430,16 @@ var ts;
         }
     }
     ts.firstDefinedIterator = firstDefinedIterator;
+    function reduceLeftIterator(iterator, f, initial) {
+        var result = initial;
+        if (iterator) {
+            for (var step = iterator.next(), pos = 0; !step.done; step = iterator.next(), pos++) {
+                result = f(result, step.value, pos);
+            }
+        }
+        return result;
+    }
+    ts.reduceLeftIterator = reduceLeftIterator;
     function zipWith(arrayA, arrayB, callback) {
         var result = [];
         ts.Debug.assertEqual(arrayA.length, arrayB.length);
@@ -390,13 +465,30 @@ var ts;
     ts.zipToIterator = zipToIterator;
     function zipToMap(keys, values) {
         ts.Debug.assert(keys.length === values.length);
-        var map = createMap();
+        var map = new ts.Map();
         for (var i = 0; i < keys.length; ++i) {
             map.set(keys[i], values[i]);
         }
         return map;
     }
     ts.zipToMap = zipToMap;
+    /**
+     * Creates a new array with `element` interspersed in between each element of `input`
+     * if there is more than 1 value in `input`. Otherwise, returns the existing array.
+     */
+    function intersperse(input, element) {
+        if (input.length <= 1) {
+            return input;
+        }
+        var result = [];
+        for (var i = 0, n = input.length; i < n; i++) {
+            if (i)
+                result.push(element);
+            result.push(input[i]);
+        }
+        return result;
+    }
+    ts.intersperse = intersperse;
     /**
      * Iterates through `array` by index and performs the callback on each element of array until the callback
      * returns a falsey value, then returns false.
@@ -735,18 +827,53 @@ var ts;
         };
     }
     ts.mapDefinedIterator = mapDefinedIterator;
-    function mapDefinedMap(map, mapValue, mapKey) {
-        if (mapKey === void 0) { mapKey = identity; }
-        var result = createMap();
+    function mapDefinedEntries(map, f) {
+        if (!map) {
+            return undefined;
+        }
+        var result = new ts.Map();
         map.forEach(function (value, key) {
-            var mapped = mapValue(value, key);
-            if (mapped !== undefined) {
-                result.set(mapKey(key), mapped);
+            var entry = f(key, value);
+            if (entry !== undefined) {
+                var newKey = entry[0], newValue = entry[1];
+                if (newKey !== undefined && newValue !== undefined) {
+                    result.set(newKey, newValue);
+                }
             }
         });
         return result;
     }
-    ts.mapDefinedMap = mapDefinedMap;
+    ts.mapDefinedEntries = mapDefinedEntries;
+    function mapDefinedValues(set, f) {
+        if (set) {
+            var result_1 = new ts.Set();
+            set.forEach(function (value) {
+                var newValue = f(value);
+                if (newValue !== undefined) {
+                    result_1.add(newValue);
+                }
+            });
+            return result_1;
+        }
+    }
+    ts.mapDefinedValues = mapDefinedValues;
+    function getOrUpdate(map, key, callback) {
+        if (map.has(key)) {
+            return map.get(key);
+        }
+        var value = callback();
+        map.set(key, value);
+        return value;
+    }
+    ts.getOrUpdate = getOrUpdate;
+    function tryAddToSet(set, value) {
+        if (!set.has(value)) {
+            set.add(value);
+            return true;
+        }
+        return false;
+    }
+    ts.tryAddToSet = tryAddToSet;
     ts.emptyIterator = { next: function () { return ({ value: undefined, done: true }); } };
     function singleIterator(value) {
         var done = false;
@@ -798,7 +925,7 @@ var ts;
         if (!map) {
             return undefined;
         }
-        var result = createMap();
+        var result = new ts.Map();
         map.forEach(function (value, key) {
             var _a = f(key, value), newKey = _a[0], newValue = _a[1];
             result.set(newKey, newValue);
@@ -936,6 +1063,20 @@ var ts;
         return deduplicateSorted(sort(array, comparer), equalityComparer || comparer || compareStringsCaseSensitive);
     }
     ts.sortAndDeduplicate = sortAndDeduplicate;
+    function arrayIsSorted(array, comparer) {
+        if (array.length < 2)
+            return true;
+        var prevElement = array[0];
+        for (var _i = 0, _a = array.slice(1); _i < _a.length; _i++) {
+            var element = _a[_i];
+            if (comparer(prevElement, element) === 1 /* GreaterThan */) {
+                return false;
+            }
+            prevElement = element;
+        }
+        return true;
+    }
+    ts.arrayIsSorted = arrayIsSorted;
     function arrayIsEqualTo(array1, array2, equalityComparer) {
         if (equalityComparer === void 0) { equalityComparer = equateValues; }
         if (!array1 || !array2) {
@@ -1240,7 +1381,7 @@ var ts;
         var high = array.length - 1;
         while (low <= high) {
             var middle = low + ((high - low) >> 1);
-            var midKey = keySelector(array[middle]);
+            var midKey = keySelector(array[middle], middle);
             switch (keyComparer(midKey, key)) {
                 case -1 /* LessThan */:
                     low = middle + 1;
@@ -1335,6 +1476,26 @@ var ts;
         return values;
     }
     ts.getOwnValues = getOwnValues;
+    var _entries = Object.entries || (function (obj) {
+        var keys = getOwnKeys(obj);
+        var result = Array(keys.length);
+        for (var i = 0; i < keys.length; i++) {
+            result[i] = [keys[i], obj[keys[i]]];
+        }
+        return result;
+    });
+    function getEntries(obj) {
+        return obj ? _entries(obj) : [];
+    }
+    ts.getEntries = getEntries;
+    function arrayOf(count, f) {
+        var result = new Array(count);
+        for (var i = 0; i < count; i++) {
+            result[i] = f(i);
+        }
+        return result;
+    }
+    ts.arrayOf = arrayOf;
     function arrayFrom(iterator, map) {
         var result = [];
         for (var iterResult = iterator.next(); !iterResult.done; iterResult = iterator.next()) {
@@ -1392,7 +1553,7 @@ var ts;
     ts.equalOwnProperties = equalOwnProperties;
     function arrayToMap(array, makeKey, makeValue) {
         if (makeValue === void 0) { makeValue = identity; }
-        var result = createMap();
+        var result = new ts.Map();
         for (var _i = 0, array_6 = array; _i < array_6.length; _i++) {
             var value = array_6[_i];
             var key = makeKey(value);
@@ -1469,14 +1630,8 @@ var ts;
         return fn ? fn.bind(obj) : undefined;
     }
     ts.maybeBind = maybeBind;
-    function mapMap(map, f) {
-        var result = createMap();
-        map.forEach(function (t, key) { return result.set.apply(result, (f(t, key))); });
-        return result;
-    }
-    ts.mapMap = mapMap;
     function createMultiMap() {
-        var map = createMap();
+        var map = new ts.Map();
         map.add = multiMapAdd;
         map.remove = multiMapRemove;
         return map;
@@ -1611,6 +1766,20 @@ var ts;
         };
     }
     ts.memoize = memoize;
+    /** A version of `memoize` that supports a single primitive argument */
+    function memoizeOne(callback) {
+        var map = new ts.Map();
+        return function (arg) {
+            var key = typeof arg + ":" + arg;
+            var value = map.get(key);
+            if (value === undefined && !map.has(key)) {
+                value = callback(arg);
+                map.set(key, value);
+            }
+            return value;
+        };
+    }
+    ts.memoizeOne = memoizeOne;
     function compose(a, b, c, d, e) {
         if (!!e) {
             var args_2 = [];
@@ -1708,7 +1877,7 @@ var ts;
      * Case-insensitive comparisons compare both strings one code-point at a time using the integer
      * value of each code-point after applying `toUpperCase` to each string. We always map both
      * strings to their upper-case form as some unicode characters do not properly round-trip to
-     * lowercase (such as `ẞ` (German sharp capital s)).
+     * lowercase (such as `ẞ` (German sharp capital s)).
      */
     function compareStringsCaseInsensitive(a, b) {
         if (a === b)
@@ -1780,7 +1949,7 @@ var ts;
             //
             // For case insensitive comparisons we always map both strings to their
             // upper-case form as some unicode characters do not properly round-trip to
-            // lowercase (such as `ẞ` (German sharp capital s)).
+            // lowercase (such as `Ã\83¡Ã\82ºÃ\85¾` (German sharp capital s)).
             return function (a, b) { return compareWithCallback(a, b, compareDictionaryOrder); };
             function compareDictionaryOrder(a, b) {
                 return compareStrings(a.toUpperCase(), b.toUpperCase()) || compareStrings(a, b);
@@ -2121,6 +2290,7 @@ var ts;
         var oldIndex = 0;
         var newLen = newItems.length;
         var oldLen = oldItems.length;
+        var hasChanges = false;
         while (newIndex < newLen && oldIndex < oldLen) {
             var newItem = newItems[newIndex];
             var oldItem = oldItems[oldIndex];
@@ -2128,10 +2298,12 @@ var ts;
             if (compareResult === -1 /* LessThan */) {
                 inserted(newItem);
                 newIndex++;
+                hasChanges = true;
             }
             else if (compareResult === 1 /* GreaterThan */) {
                 deleted(oldItem);
                 oldIndex++;
+                hasChanges = true;
             }
             else {
                 unchanged(oldItem, newItem);
@@ -2141,10 +2313,13 @@ var ts;
         }
         while (newIndex < newLen) {
             inserted(newItems[newIndex++]);
+            hasChanges = true;
         }
         while (oldIndex < oldLen) {
             deleted(oldItems[oldIndex++]);
+            hasChanges = true;
         }
+        return hasChanges;
     }
     ts.enumerateInsertsAndDeletes = enumerateInsertsAndDeletes;
     function fill(length, cb) {
@@ -2180,29 +2355,93 @@ var ts;
             }
         }
     }
-    function padLeft(s, length) {
-        while (s.length < length) {
-            s = " " + s;
-        }
-        return s;
+    /**
+     * Returns string left-padded with spaces or zeros until it reaches the given length.
+     *
+     * @param s String to pad.
+     * @param length Final padded length. If less than or equal to 's.length', returns 's' unchanged.
+     * @param padString Character to use as padding (default " ").
+     */
+    function padLeft(s, length, padString) {
+        if (padString === void 0) { padString = " "; }
+        return length <= s.length ? s : padString.repeat(length - s.length) + s;
     }
     ts.padLeft = padLeft;
-    function padRight(s, length) {
-        while (s.length < length) {
-            s = s + " ";
-        }
-        return s;
+    /**
+     * Returns string right-padded with spaces until it reaches the given length.
+     *
+     * @param s String to pad.
+     * @param length Final padded length. If less than or equal to 's.length', returns 's' unchanged.
+     * @param padString Character to use as padding (default " ").
+     */
+    function padRight(s, length, padString) {
+        if (padString === void 0) { padString = " "; }
+        return length <= s.length ? s : s + padString.repeat(length - s.length);
     }
     ts.padRight = padRight;
+    function takeWhile(array, predicate) {
+        var len = array.length;
+        var index = 0;
+        while (index < len && predicate(array[index])) {
+            index++;
+        }
+        return array.slice(0, index);
+    }
+    ts.takeWhile = takeWhile;
 })(ts || (ts = {}));
 /* @internal */
 var ts;
 (function (ts) {
+    var LogLevel;
+    (function (LogLevel) {
+        LogLevel[LogLevel["Off"] = 0] = "Off";
+        LogLevel[LogLevel["Error"] = 1] = "Error";
+        LogLevel[LogLevel["Warning"] = 2] = "Warning";
+        LogLevel[LogLevel["Info"] = 3] = "Info";
+        LogLevel[LogLevel["Verbose"] = 4] = "Verbose";
+    })(LogLevel = ts.LogLevel || (ts.LogLevel = {}));
     var Debug;
     (function (Debug) {
+        var typeScriptVersion;
+        /* eslint-disable prefer-const */
         var currentAssertionLevel = 0 /* None */;
-        // eslint-disable-next-line prefer-const
+        Debug.currentLogLevel = LogLevel.Warning;
         Debug.isDebugging = false;
+        function getTypeScriptVersion() {
+            return typeScriptVersion !== null && typeScriptVersion !== void 0 ? typeScriptVersion : (typeScriptVersion = new ts.Version(ts.version));
+        }
+        Debug.getTypeScriptVersion = getTypeScriptVersion;
+        function shouldLog(level) {
+            return Debug.currentLogLevel <= level;
+        }
+        Debug.shouldLog = shouldLog;
+        function logMessage(level, s) {
+            if (Debug.loggingHost && shouldLog(level)) {
+                Debug.loggingHost.log(level, s);
+            }
+        }
+        function log(s) {
+            logMessage(LogLevel.Info, s);
+        }
+        Debug.log = log;
+        (function (log_1) {
+            function error(s) {
+                logMessage(LogLevel.Error, s);
+            }
+            log_1.error = error;
+            function warn(s) {
+                logMessage(LogLevel.Warning, s);
+            }
+            log_1.warn = warn;
+            function log(s) {
+                logMessage(LogLevel.Info, s);
+            }
+            log_1.log = log;
+            function trace(s) {
+                logMessage(LogLevel.Verbose, s);
+            }
+            log_1.trace = trace;
+        })(log = Debug.log || (Debug.log = {}));
         var assertionCache = {};
         function getAssertionLevel() {
             return currentAssertionLevel;
@@ -2463,6 +2702,10 @@ var ts;
             return formatEnum(flags, ts.ObjectFlags, /*isFlags*/ true);
         }
         Debug.formatObjectFlags = formatObjectFlags;
+        function formatFlowFlags(flags) {
+            return formatEnum(flags, ts.FlowFlags, /*isFlags*/ true);
+        }
+        Debug.formatFlowFlags = formatFlowFlags;
         var isDebugInfoEnabled = false;
         var extendedDebugModule;
         function extendedDebug() {
@@ -2480,31 +2723,167 @@ var ts;
             return extendedDebug().formatControlFlowGraph(flowNode);
         }
         Debug.formatControlFlowGraph = formatControlFlowGraph;
+        var flowNodeProto;
+        function attachFlowNodeDebugInfoWorker(flowNode) {
+            if (!("__debugFlowFlags" in flowNode)) { // eslint-disable-line no-in-operator
+                Object.defineProperties(flowNode, {
+                    // for use with vscode-js-debug's new customDescriptionGenerator in launch.json
+                    __tsDebuggerDisplay: {
+                        value: function () {
+                            var flowHeader = this.flags & 2 /* Start */ ? "FlowStart" :
+                                this.flags & 4 /* BranchLabel */ ? "FlowBranchLabel" :
+                                    this.flags & 8 /* LoopLabel */ ? "FlowLoopLabel" :
+                                        this.flags & 16 /* Assignment */ ? "FlowAssignment" :
+                                            this.flags & 32 /* TrueCondition */ ? "FlowTrueCondition" :
+                                                this.flags & 64 /* FalseCondition */ ? "FlowFalseCondition" :
+                                                    this.flags & 128 /* SwitchClause */ ? "FlowSwitchClause" :
+                                                        this.flags & 256 /* ArrayMutation */ ? "FlowArrayMutation" :
+                                                            this.flags & 512 /* Call */ ? "FlowCall" :
+                                                                this.flags & 1024 /* ReduceLabel */ ? "FlowReduceLabel" :
+                                                                    this.flags & 1 /* Unreachable */ ? "FlowUnreachable" :
+                                                                        "UnknownFlow";
+                            var remainingFlags = this.flags & ~(2048 /* Referenced */ - 1);
+                            return "" + flowHeader + (remainingFlags ? " (" + formatFlowFlags(remainingFlags) + ")" : "");
+                        }
+                    },
+                    __debugFlowFlags: { get: function () { return formatEnum(this.flags, ts.FlowFlags, /*isFlags*/ true); } },
+                    __debugToString: { value: function () { return formatControlFlowGraph(this); } }
+                });
+            }
+        }
         function attachFlowNodeDebugInfo(flowNode) {
             if (isDebugInfoEnabled) {
-                if (!("__debugFlowFlags" in flowNode)) { // eslint-disable-line no-in-operator
-                    Object.defineProperties(flowNode, {
-                        __debugFlowFlags: { get: function () { return formatEnum(this.flags, ts.FlowFlags, /*isFlags*/ true); } },
-                        __debugToString: { value: function () { return formatControlFlowGraph(this); } }
-                    });
+                if (typeof Object.setPrototypeOf === "function") {
+                    // if we're in es2015, attach the method to a shared prototype for `FlowNode`
+                    // so the method doesn't show up in the watch window.
+                    if (!flowNodeProto) {
+                        flowNodeProto = Object.create(Object.prototype);
+                        attachFlowNodeDebugInfoWorker(flowNodeProto);
+                    }
+                    Object.setPrototypeOf(flowNode, flowNodeProto);
+                }
+                else {
+                    // not running in an es2015 environment, attach the method directly.
+                    attachFlowNodeDebugInfoWorker(flowNode);
                 }
             }
         }
         Debug.attachFlowNodeDebugInfo = attachFlowNodeDebugInfo;
+        var nodeArrayProto;
+        function attachNodeArrayDebugInfoWorker(array) {
+            if (!("__tsDebuggerDisplay" in array)) { // eslint-disable-line no-in-operator
+                Object.defineProperties(array, {
+                    __tsDebuggerDisplay: {
+                        value: function (defaultValue) {
+                            // An `Array` with extra properties is rendered as `[A, B, prop1: 1, prop2: 2]`. Most of
+                            // these aren't immediately useful so we trim off the `prop1: ..., prop2: ...` part from the
+                            // formatted string.
+                            defaultValue = String(defaultValue).replace(/(?:,[\s\w\d_]+:[^,]+)+\]$/, "]");
+                            return "NodeArray " + defaultValue;
+                        }
+                    }
+                });
+            }
+        }
+        function attachNodeArrayDebugInfo(array) {
+            if (isDebugInfoEnabled) {
+                if (typeof Object.setPrototypeOf === "function") {
+                    // if we're in es2015, attach the method to a shared prototype for `NodeArray`
+                    // so the method doesn't show up in the watch window.
+                    if (!nodeArrayProto) {
+                        nodeArrayProto = Object.create(Array.prototype);
+                        attachNodeArrayDebugInfoWorker(nodeArrayProto);
+                    }
+                    Object.setPrototypeOf(array, nodeArrayProto);
+                }
+                else {
+                    // not running in an es2015 environment, attach the method directly.
+                    attachNodeArrayDebugInfoWorker(array);
+                }
+            }
+        }
+        Debug.attachNodeArrayDebugInfo = attachNodeArrayDebugInfo;
         /**
          * Injects debug information into frequently used types.
          */
         function enableDebugInfo() {
             if (isDebugInfoEnabled)
                 return;
+            // avoid recomputing
+            var weakTypeTextMap;
+            var weakNodeTextMap;
+            function getWeakTypeTextMap() {
+                if (weakTypeTextMap === undefined) {
+                    if (typeof WeakMap === "function")
+                        weakTypeTextMap = new WeakMap();
+                }
+                return weakTypeTextMap;
+            }
+            function getWeakNodeTextMap() {
+                if (weakNodeTextMap === undefined) {
+                    if (typeof WeakMap === "function")
+                        weakNodeTextMap = new WeakMap();
+                }
+                return weakNodeTextMap;
+            }
             // Add additional properties in debug mode to assist with debugging.
             Object.defineProperties(ts.objectAllocator.getSymbolConstructor().prototype, {
+                // for use with vscode-js-debug's new customDescriptionGenerator in launch.json
+                __tsDebuggerDisplay: {
+                    value: function () {
+                        var symbolHeader = this.flags & 33554432 /* Transient */ ? "TransientSymbol" :
+                            "Symbol";
+                        var remainingSymbolFlags = this.flags & ~33554432 /* Transient */;
+                        return symbolHeader + " '" + ts.symbolName(this) + "'" + (remainingSymbolFlags ? " (" + formatSymbolFlags(remainingSymbolFlags) + ")" : "");
+                    }
+                },
                 __debugFlags: { get: function () { return formatSymbolFlags(this.flags); } }
             });
             Object.defineProperties(ts.objectAllocator.getTypeConstructor().prototype, {
+                // for use with vscode-js-debug's new customDescriptionGenerator in launch.json
+                __tsDebuggerDisplay: {
+                    value: function () {
+                        var typeHeader = this.flags & 98304 /* Nullable */ ? "NullableType" :
+                            this.flags & 384 /* StringOrNumberLiteral */ ? "LiteralType " + JSON.stringify(this.value) :
+                                this.flags & 2048 /* BigIntLiteral */ ? "LiteralType " + (this.value.negative ? "-" : "") + this.value.base10Value + "n" :
+                                    this.flags & 8192 /* UniqueESSymbol */ ? "UniqueESSymbolType" :
+                                        this.flags & 32 /* Enum */ ? "EnumType" :
+                                            this.flags & 67359327 /* Intrinsic */ ? "IntrinsicType " + this.intrinsicName :
+                                                this.flags & 1048576 /* Union */ ? "UnionType" :
+                                                    this.flags & 2097152 /* Intersection */ ? "IntersectionType" :
+                                                        this.flags & 4194304 /* Index */ ? "IndexType" :
+                                                            this.flags & 8388608 /* IndexedAccess */ ? "IndexedAccessType" :
+                                                                this.flags & 16777216 /* Conditional */ ? "ConditionalType" :
+                                                                    this.flags & 33554432 /* Substitution */ ? "SubstitutionType" :
+                                                                        this.flags & 262144 /* TypeParameter */ ? "TypeParameter" :
+                                                                            this.flags & 524288 /* Object */ ?
+                                                                                this.objectFlags & 3 /* ClassOrInterface */ ? "InterfaceType" :
+                                                                                    this.objectFlags & 4 /* Reference */ ? "TypeReference" :
+                                                                                        this.objectFlags & 8 /* Tuple */ ? "TupleType" :
+                                                                                            this.objectFlags & 16 /* Anonymous */ ? "AnonymousType" :
+                                                                                                this.objectFlags & 32 /* Mapped */ ? "MappedType" :
+                                                                                                    this.objectFlags & 2048 /* ReverseMapped */ ? "ReverseMappedType" :
+                                                                                                        this.objectFlags & 256 /* EvolvingArray */ ? "EvolvingArrayType" :
+                                                                                                            "ObjectType" :
+                                                                                "Type";
+                        var remainingObjectFlags = this.flags & 524288 /* Object */ ? this.objectFlags & ~2367 /* ObjectTypeKindMask */ : 0;
+                        return "" + typeHeader + (this.symbol ? " '" + ts.symbolName(this.symbol) + "'" : "") + (remainingObjectFlags ? " (" + formatObjectFlags(remainingObjectFlags) + ")" : "");
+                    }
+                },
                 __debugFlags: { get: function () { return formatTypeFlags(this.flags); } },
                 __debugObjectFlags: { get: function () { return this.flags & 524288 /* Object */ ? formatObjectFlags(this.objectFlags) : ""; } },
-                __debugTypeToString: { value: function () { return this.checker.typeToString(this); } },
+                __debugTypeToString: {
+                    value: function () {
+                        // avoid recomputing
+                        var map = getWeakTypeTextMap();
+                        var text = map === null || map === void 0 ? void 0 : map.get(this);
+                        if (text === undefined) {
+                            text = this.checker.typeToString(this);
+                            map === null || map === void 0 ? void 0 : map.set(this, text);
+                        }
+                        return text;
+                    }
+                },
             });
             var nodeConstructors = [
                 ts.objectAllocator.getNodeConstructor(),
@@ -2516,9 +2895,52 @@ var ts;
                 var ctor = nodeConstructors_1[_i];
                 if (!ctor.prototype.hasOwnProperty("__debugKind")) {
                     Object.defineProperties(ctor.prototype, {
+                        // for use with vscode-js-debug's new customDescriptionGenerator in launch.json
+                        __tsDebuggerDisplay: {
+                            value: function () {
+                                var nodeHeader = ts.isGeneratedIdentifier(this) ? "GeneratedIdentifier" :
+                                    ts.isIdentifier(this) ? "Identifier '" + ts.idText(this) + "'" :
+                                        ts.isPrivateIdentifier(this) ? "PrivateIdentifier '" + ts.idText(this) + "'" :
+                                            ts.isStringLiteral(this) ? "StringLiteral " + JSON.stringify(this.text.length < 10 ? this.text : this.text.slice(10) + "...") :
+                                                ts.isNumericLiteral(this) ? "NumericLiteral " + this.text :
+                                                    ts.isBigIntLiteral(this) ? "BigIntLiteral " + this.text + "n" :
+                                                        ts.isTypeParameterDeclaration(this) ? "TypeParameterDeclaration" :
+                                                            ts.isParameter(this) ? "ParameterDeclaration" :
+                                                                ts.isConstructorDeclaration(this) ? "ConstructorDeclaration" :
+                                                                    ts.isGetAccessorDeclaration(this) ? "GetAccessorDeclaration" :
+                                                                        ts.isSetAccessorDeclaration(this) ? "SetAccessorDeclaration" :
+                                                                            ts.isCallSignatureDeclaration(this) ? "CallSignatureDeclaration" :
+                                                                                ts.isConstructSignatureDeclaration(this) ? "ConstructSignatureDeclaration" :
+                                                                                    ts.isIndexSignatureDeclaration(this) ? "IndexSignatureDeclaration" :
+                                                                                        ts.isTypePredicateNode(this) ? "TypePredicateNode" :
+                                                                                            ts.isTypeReferenceNode(this) ? "TypeReferenceNode" :
+                                                                                                ts.isFunctionTypeNode(this) ? "FunctionTypeNode" :
+                                                                                                    ts.isConstructorTypeNode(this) ? "ConstructorTypeNode" :
+                                                                                                        ts.isTypeQueryNode(this) ? "TypeQueryNode" :
+                                                                                                            ts.isTypeLiteralNode(this) ? "TypeLiteralNode" :
+                                                                                                                ts.isArrayTypeNode(this) ? "ArrayTypeNode" :
+                                                                                                                    ts.isTupleTypeNode(this) ? "TupleTypeNode" :
+                                                                                                                        ts.isOptionalTypeNode(this) ? "OptionalTypeNode" :
+                                                                                                                            ts.isRestTypeNode(this) ? "RestTypeNode" :
+                                                                                                                                ts.isUnionTypeNode(this) ? "UnionTypeNode" :
+                                                                                                                                    ts.isIntersectionTypeNode(this) ? "IntersectionTypeNode" :
+                                                                                                                                        ts.isConditionalTypeNode(this) ? "ConditionalTypeNode" :
+                                                                                                                                            ts.isInferTypeNode(this) ? "InferTypeNode" :
+                                                                                                                                                ts.isParenthesizedTypeNode(this) ? "ParenthesizedTypeNode" :
+                                                                                                                                                    ts.isThisTypeNode(this) ? "ThisTypeNode" :
+                                                                                                                                                        ts.isTypeOperatorNode(this) ? "TypeOperatorNode" :
+                                                                                                                                                            ts.isIndexedAccessTypeNode(this) ? "IndexedAccessTypeNode" :
+                                                                                                                                                                ts.isMappedTypeNode(this) ? "MappedTypeNode" :
+                                                                                                                                                                    ts.isLiteralTypeNode(this) ? "LiteralTypeNode" :
+                                                                                                                                                                        ts.isNamedTupleMember(this) ? "NamedTupleMember" :
+                                                                                                                                                                            ts.isImportTypeNode(this) ? "ImportTypeNode" :
+                                                                                                                                                                                formatSyntaxKind(this.kind);
+                                return "" + nodeHeader + (this.flags ? " (" + formatNodeFlags(this.flags) + ")" : "");
+                            }
+                        },
                         __debugKind: { get: function () { return formatSyntaxKind(this.kind); } },
                         __debugNodeFlags: { get: function () { return formatNodeFlags(this.flags); } },
-                        __debugModifierFlags: { get: function () { return formatModifierFlags(ts.getModifierFlagsNoCache(this)); } },
+                        __debugModifierFlags: { get: function () { return formatModifierFlags(ts.getEffectiveModifierFlagsNoCache(this)); } },
                         __debugTransformFlags: { get: function () { return formatTransformFlags(this.transformFlags); } },
                         __debugIsParseTreeNode: { get: function () { return ts.isParseTreeNode(this); } },
                         __debugEmitFlags: { get: function () { return formatEmitFlags(ts.getEmitFlags(this)); } },
@@ -2526,9 +2948,16 @@ var ts;
                             value: function (includeTrivia) {
                                 if (ts.nodeIsSynthesized(this))
                                     return "";
-                                var parseNode = ts.getParseTreeNode(this);
-                                var sourceFile = parseNode && ts.getSourceFileOfNode(parseNode);
-                                return sourceFile ? ts.getSourceTextOfNodeFromSourceFile(sourceFile, parseNode, includeTrivia) : "";
+                                // avoid recomputing
+                                var map = getWeakNodeTextMap();
+                                var text = map === null || map === void 0 ? void 0 : map.get(this);
+                                if (text === undefined) {
+                                    var parseNode = ts.getParseTreeNode(this);
+                                    var sourceFile = parseNode && ts.getSourceFileOfNode(parseNode);
+                                    text = sourceFile ? ts.getSourceTextOfNodeFromSourceFile(sourceFile, parseNode, includeTrivia) : "";
+                                    map === null || map === void 0 ? void 0 : map.set(this, text);
+                                }
+                                return text;
                             }
                         }
                     });
@@ -2551,167 +2980,54 @@ var ts;
             isDebugInfoEnabled = true;
         }
         Debug.enableDebugInfo = enableDebugInfo;
-    })(Debug = ts.Debug || (ts.Debug = {}));
-})(ts || (ts = {}));
-/*@internal*/
-var ts;
-(function (ts) {
-    /** Gets a timestamp with (at least) ms resolution */
-    ts.timestamp = typeof performance !== "undefined" && performance.now ? function () { return performance.now(); } : Date.now ? Date.now : function () { return +(new Date()); };
-})(ts || (ts = {}));
-/*@internal*/
-/** Performance measurements for the compiler. */
-var ts;
-(function (ts) {
-    var performance;
-    (function (performance) {
-        // NOTE: cannot use ts.noop as core.ts loads after this
-        var profilerEvent = typeof onProfilerEvent === "function" && onProfilerEvent.profiler === true ? onProfilerEvent : function () { };
-        var enabled = false;
-        var profilerStart = 0;
-        var counts;
-        var marks;
-        var measures;
-        function createTimerIf(condition, measureName, startMarkName, endMarkName) {
-            return condition ? createTimer(measureName, startMarkName, endMarkName) : performance.nullTimer;
+        function formatDeprecationMessage(name, error, errorAfter, since, message) {
+            var deprecationMessage = error ? "DeprecationError: " : "DeprecationWarning: ";
+            deprecationMessage += "'" + name + "' ";
+            deprecationMessage += since ? "has been deprecated since v" + since : "is deprecated";
+            deprecationMessage += error ? " and can no longer be used." : errorAfter ? " and will no longer be usable after v" + errorAfter + "." : ".";
+            deprecationMessage += message ? " " + ts.formatStringFromArgs(message, [name], 0) : "";
+            return deprecationMessage;
         }
-        performance.createTimerIf = createTimerIf;
-        function createTimer(measureName, startMarkName, endMarkName) {
-            var enterCount = 0;
-            return {
-                enter: enter,
-                exit: exit
+        function createErrorDeprecation(name, errorAfter, since, message) {
+            var deprecationMessage = formatDeprecationMessage(name, /*error*/ true, errorAfter, since, message);
+            return function () {
+                throw new TypeError(deprecationMessage);
             };
-            function enter() {
-                if (++enterCount === 1) {
-                    mark(startMarkName);
-                }
-            }
-            function exit() {
-                if (--enterCount === 0) {
-                    mark(endMarkName);
-                    measure(measureName, startMarkName, endMarkName);
-                }
-                else if (enterCount < 0) {
-                    ts.Debug.fail("enter/exit count does not match.");
-                }
-            }
         }
-        performance.createTimer = createTimer;
-        performance.nullTimer = { enter: ts.noop, exit: ts.noop };
-        /**
-         * Marks a performance event.
-         *
-         * @param markName The name of the mark.
-         */
-        function mark(markName) {
-            if (enabled) {
-                marks.set(markName, ts.timestamp());
-                counts.set(markName, (counts.get(markName) || 0) + 1);
-                profilerEvent(markName);
-            }
-        }
-        performance.mark = mark;
-        /**
-         * Adds a performance measurement with the specified name.
-         *
-         * @param measureName The name of the performance measurement.
-         * @param startMarkName The name of the starting mark. If not supplied, the point at which the
-         *      profiler was enabled is used.
-         * @param endMarkName The name of the ending mark. If not supplied, the current timestamp is
-         *      used.
-         */
-        function measure(measureName, startMarkName, endMarkName) {
-            if (enabled) {
-                var end = endMarkName && marks.get(endMarkName) || ts.timestamp();
-                var start = startMarkName && marks.get(startMarkName) || profilerStart;
-                measures.set(measureName, (measures.get(measureName) || 0) + (end - start));
-            }
-        }
-        performance.measure = measure;
-        /**
-         * Gets the number of times a marker was encountered.
-         *
-         * @param markName The name of the mark.
-         */
-        function getCount(markName) {
-            return counts && counts.get(markName) || 0;
-        }
-        performance.getCount = getCount;
-        /**
-         * Gets the total duration of all measurements with the supplied name.
-         *
-         * @param measureName The name of the measure whose durations should be accumulated.
-         */
-        function getDuration(measureName) {
-            return measures && measures.get(measureName) || 0;
-        }
-        performance.getDuration = getDuration;
-        /**
-         * Iterate over each measure, performing some action
-         *
-         * @param cb The action to perform for each measure
-         */
-        function forEachMeasure(cb) {
-            measures.forEach(function (measure, key) {
-                cb(key, measure);
-            });
-        }
-        performance.forEachMeasure = forEachMeasure;
-        /** Enables (and resets) performance measurements for the compiler. */
-        function enable() {
-            counts = ts.createMap();
-            marks = ts.createMap();
-            measures = ts.createMap();
-            enabled = true;
-            profilerStart = ts.timestamp();
-        }
-        performance.enable = enable;
-        /** Disables performance measurements for the compiler. */
-        function disable() {
-            enabled = false;
+        function createWarningDeprecation(name, errorAfter, since, message) {
+            var hasWrittenDeprecation = false;
+            return function () {
+                if (!hasWrittenDeprecation) {
+                    log.warn(formatDeprecationMessage(name, /*error*/ false, errorAfter, since, message));
+                    hasWrittenDeprecation = true;
+                }
+            };
         }
-        performance.disable = disable;
-    })(performance = ts.performance || (ts.performance = {}));
-})(ts || (ts = {}));
-/* @internal */
-var ts;
-(function (ts) {
-    var nullLogger = {
-        logEvent: ts.noop,
-        logErrEvent: ts.noop,
-        logPerfEvent: ts.noop,
-        logInfoEvent: ts.noop,
-        logStartCommand: ts.noop,
-        logStopCommand: ts.noop,
-        logStartUpdateProgram: ts.noop,
-        logStopUpdateProgram: ts.noop,
-        logStartUpdateGraph: ts.noop,
-        logStopUpdateGraph: ts.noop,
-        logStartResolveModule: ts.noop,
-        logStopResolveModule: ts.noop,
-        logStartParseSourceFile: ts.noop,
-        logStopParseSourceFile: ts.noop,
-        logStartReadFile: ts.noop,
-        logStopReadFile: ts.noop,
-        logStartBindFile: ts.noop,
-        logStopBindFile: ts.noop,
-        logStartScheduledOperation: ts.noop,
-        logStopScheduledOperation: ts.noop,
-    };
-    // Load optional module to enable Event Tracing for Windows
-    // See https://github.com/microsoft/typescript-etw for more information
-    var etwModule;
-    try {
-        // require() will throw an exception if the module is not installed
-        // It may also return undefined if not installed properly
-        etwModule = require("@microsoft/typescript-etw");
-    }
-    catch (e) {
-        etwModule = undefined;
-    }
-    /** Performance logger that will generate ETW events if possible - check for `logEvent` member, as `etwModule` will be `{}` when browserified */
-    ts.perfLogger = etwModule && etwModule.logEvent ? etwModule : nullLogger;
+        function createDeprecation(name, options) {
+            var _a, _b;
+            if (options === void 0) { options = {}; }
+            var version = typeof options.typeScriptVersion === "string" ? new ts.Version(options.typeScriptVersion) : (_a = options.typeScriptVersion) !== null && _a !== void 0 ? _a : getTypeScriptVersion();
+            var errorAfter = typeof options.errorAfter === "string" ? new ts.Version(options.errorAfter) : options.errorAfter;
+            var warnAfter = typeof options.warnAfter === "string" ? new ts.Version(options.warnAfter) : options.warnAfter;
+            var since = typeof options.since === "string" ? new ts.Version(options.since) : (_b = options.since) !== null && _b !== void 0 ? _b : warnAfter;
+            var error = options.error || errorAfter && version.compareTo(errorAfter) <= 0;
+            var warn = !warnAfter || version.compareTo(warnAfter) >= 0;
+            return error ? createErrorDeprecation(name, errorAfter, since, options.message) :
+                warn ? createWarningDeprecation(name, errorAfter, since, options.message) :
+                    ts.noop;
+        }
+        function wrapFunction(deprecation, func) {
+            return function () {
+                deprecation();
+                return func.apply(this, arguments);
+            };
+        }
+        function deprecate(func, options) {
+            var deprecation = createDeprecation(getFunctionName(func), options);
+            return wrapFunction(deprecation, func);
+        }
+        Debug.deprecate = deprecate;
+    })(Debug = ts.Debug || (ts.Debug = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
@@ -2789,7 +3105,7 @@ var ts;
             return ts.compareValues(this.major, other.major)
                 || ts.compareValues(this.minor, other.minor)
                 || ts.compareValues(this.patch, other.patch)
-                || comparePrerelaseIdentifiers(this.prerelease, other.prerelease);
+                || comparePrereleaseIdentifiers(this.prerelease, other.prerelease);
         };
         Version.prototype.increment = function (field) {
             switch (field) {
@@ -2828,7 +3144,7 @@ var ts;
             build: build
         };
     }
-    function comparePrerelaseIdentifiers(left, right) {
+    function comparePrereleaseIdentifiers(left, right) {
         // https://semver.org/#spec-item-11
         // > When major, minor, and patch are equal, a pre-release version has lower precedence
         // > than a normal version.
@@ -3073,6 +3389,485 @@ var ts;
         return "" + comparator.operator + comparator.operand;
     }
 })(ts || (ts = {}));
+/*@internal*/
+var ts;
+(function (ts) {
+    // The following definitions provide the minimum compatible support for the Web Performance User Timings API
+    // between browsers and NodeJS:
+    // eslint-disable-next-line @typescript-eslint/naming-convention
+    function hasRequiredAPI(performance, PerformanceObserver) {
+        return typeof performance === "object" &&
+            typeof performance.timeOrigin === "number" &&
+            typeof performance.mark === "function" &&
+            typeof performance.measure === "function" &&
+            typeof performance.now === "function" &&
+            typeof PerformanceObserver === "function";
+    }
+    function tryGetWebPerformanceHooks() {
+        if (typeof performance === "object" &&
+            typeof PerformanceObserver === "function" &&
+            hasRequiredAPI(performance, PerformanceObserver)) {
+            return {
+                performance: performance,
+                PerformanceObserver: PerformanceObserver
+            };
+        }
+    }
+    function tryGetNodePerformanceHooks() {
+        if (typeof module === "object" && typeof require === "function") {
+            try {
+                var _a = require("perf_hooks"), performance_1 = _a.performance, PerformanceObserver_1 = _a.PerformanceObserver;
+                if (hasRequiredAPI(performance_1, PerformanceObserver_1)) {
+                    // There is a bug in Node's performance.measure prior to 12.16.3/13.13.0 that does not
+                    // match the Web Performance API specification. Node's implementation did not allow
+                    // optional `start` and `end` arguments for `performance.measure`.
+                    // See https://github.com/nodejs/node/pull/32651 for more information.
+                    var version_1 = new ts.Version(process.versions.node);
+                    var range = new ts.VersionRange("<12.16.3 || 13 <13.13");
+                    if (range.test(version_1)) {
+                        return {
+                            performance: {
+                                get timeOrigin() { return performance_1.timeOrigin; },
+                                now: function () { return performance_1.now(); },
+                                mark: function (name) { return performance_1.mark(name); },
+                                measure: function (name, start, end) {
+                                    if (start === void 0) { start = "nodeStart"; }
+                                    if (end === undefined) {
+                                        end = "__performance.measure-fix__";
+                                        performance_1.mark(end);
+                                    }
+                                    performance_1.measure(name, start, end);
+                                    if (end === "__performance.measure-fix__") {
+                                        performance_1.clearMarks("__performance.measure-fix__");
+                                    }
+                                }
+                            },
+                            PerformanceObserver: PerformanceObserver_1
+                        };
+                    }
+                    return {
+                        performance: performance_1,
+                        PerformanceObserver: PerformanceObserver_1
+                    };
+                }
+            }
+            catch (_b) {
+                // ignore errors
+            }
+        }
+    }
+    // Unlike with the native Map/Set 'tryGet' functions in corePublic.ts, we eagerly evaluate these
+    // since we will need them for `timestamp`, below.
+    var nativePerformanceHooks = tryGetWebPerformanceHooks() || tryGetNodePerformanceHooks();
+    var nativePerformance = nativePerformanceHooks === null || nativePerformanceHooks === void 0 ? void 0 : nativePerformanceHooks.performance;
+    function tryGetNativePerformanceHooks() {
+        return nativePerformanceHooks;
+    }
+    ts.tryGetNativePerformanceHooks = tryGetNativePerformanceHooks;
+    /** Gets a timestamp with (at least) ms resolution */
+    ts.timestamp = nativePerformance ? function () { return nativePerformance.now(); } :
+        Date.now ? Date.now :
+            function () { return +(new Date()); };
+})(ts || (ts = {}));
+/*@internal*/
+/** Performance measurements for the compiler. */
+var ts;
+(function (ts) {
+    var performance;
+    (function (performance) {
+        var perfHooks;
+        var perfObserver;
+        // when set, indicates the implementation of `Performance` to use for user timing.
+        // when unset, indicates user timing is unavailable or disabled.
+        var performanceImpl;
+        function createTimerIf(condition, measureName, startMarkName, endMarkName) {
+            return condition ? createTimer(measureName, startMarkName, endMarkName) : performance.nullTimer;
+        }
+        performance.createTimerIf = createTimerIf;
+        function createTimer(measureName, startMarkName, endMarkName) {
+            var enterCount = 0;
+            return {
+                enter: enter,
+                exit: exit
+            };
+            function enter() {
+                if (++enterCount === 1) {
+                    mark(startMarkName);
+                }
+            }
+            function exit() {
+                if (--enterCount === 0) {
+                    mark(endMarkName);
+                    measure(measureName, startMarkName, endMarkName);
+                }
+                else if (enterCount < 0) {
+                    ts.Debug.fail("enter/exit count does not match.");
+                }
+            }
+        }
+        performance.createTimer = createTimer;
+        performance.nullTimer = { enter: ts.noop, exit: ts.noop };
+        var counts = new ts.Map();
+        var durations = new ts.Map();
+        /**
+         * Marks a performance event.
+         *
+         * @param markName The name of the mark.
+         */
+        function mark(markName) {
+            performanceImpl === null || performanceImpl === void 0 ? void 0 : performanceImpl.mark(markName);
+        }
+        performance.mark = mark;
+        /**
+         * Adds a performance measurement with the specified name.
+         *
+         * @param measureName The name of the performance measurement.
+         * @param startMarkName The name of the starting mark. If not supplied, the point at which the
+         *      profiler was enabled is used.
+         * @param endMarkName The name of the ending mark. If not supplied, the current timestamp is
+         *      used.
+         */
+        function measure(measureName, startMarkName, endMarkName) {
+            performanceImpl === null || performanceImpl === void 0 ? void 0 : performanceImpl.measure(measureName, startMarkName, endMarkName);
+        }
+        performance.measure = measure;
+        /**
+         * Gets the number of times a marker was encountered.
+         *
+         * @param markName The name of the mark.
+         */
+        function getCount(markName) {
+            return counts.get(markName) || 0;
+        }
+        performance.getCount = getCount;
+        /**
+         * Gets the total duration of all measurements with the supplied name.
+         *
+         * @param measureName The name of the measure whose durations should be accumulated.
+         */
+        function getDuration(measureName) {
+            return durations.get(measureName) || 0;
+        }
+        performance.getDuration = getDuration;
+        /**
+         * Iterate over each measure, performing some action
+         *
+         * @param cb The action to perform for each measure
+         */
+        function forEachMeasure(cb) {
+            durations.forEach(function (duration, measureName) { return cb(measureName, duration); });
+        }
+        performance.forEachMeasure = forEachMeasure;
+        /**
+         * Indicates whether the performance API is enabled.
+         */
+        function isEnabled() {
+            return !!performanceImpl;
+        }
+        performance.isEnabled = isEnabled;
+        /** Enables (and resets) performance measurements for the compiler. */
+        function enable() {
+            if (!performanceImpl) {
+                perfHooks || (perfHooks = ts.tryGetNativePerformanceHooks());
+                if (!perfHooks)
+                    return false;
+                perfObserver || (perfObserver = new perfHooks.PerformanceObserver(updateStatisticsFromList));
+                perfObserver.observe({ entryTypes: ["mark", "measure"] });
+                performanceImpl = perfHooks.performance;
+            }
+            return true;
+        }
+        performance.enable = enable;
+        /** Disables performance measurements for the compiler. */
+        function disable() {
+            perfObserver === null || perfObserver === void 0 ? void 0 : perfObserver.disconnect();
+            performanceImpl = undefined;
+            counts.clear();
+            durations.clear();
+        }
+        performance.disable = disable;
+        function updateStatisticsFromList(list) {
+            for (var _i = 0, _a = list.getEntriesByType("mark"); _i < _a.length; _i++) {
+                var mark_1 = _a[_i];
+                counts.set(mark_1.name, (counts.get(mark_1.name) || 0) + 1);
+            }
+            for (var _b = 0, _c = list.getEntriesByType("measure"); _b < _c.length; _b++) {
+                var measure_1 = _c[_b];
+                durations.set(measure_1.name, (durations.get(measure_1.name) || 0) + measure_1.duration);
+            }
+        }
+    })(performance = ts.performance || (ts.performance = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var _a;
+    var nullLogger = {
+        logEvent: ts.noop,
+        logErrEvent: ts.noop,
+        logPerfEvent: ts.noop,
+        logInfoEvent: ts.noop,
+        logStartCommand: ts.noop,
+        logStopCommand: ts.noop,
+        logStartUpdateProgram: ts.noop,
+        logStopUpdateProgram: ts.noop,
+        logStartUpdateGraph: ts.noop,
+        logStopUpdateGraph: ts.noop,
+        logStartResolveModule: ts.noop,
+        logStopResolveModule: ts.noop,
+        logStartParseSourceFile: ts.noop,
+        logStopParseSourceFile: ts.noop,
+        logStartReadFile: ts.noop,
+        logStopReadFile: ts.noop,
+        logStartBindFile: ts.noop,
+        logStopBindFile: ts.noop,
+        logStartScheduledOperation: ts.noop,
+        logStopScheduledOperation: ts.noop,
+    };
+    // Load optional module to enable Event Tracing for Windows
+    // See https://github.com/microsoft/typescript-etw for more information
+    var etwModule;
+    try {
+        var etwModulePath = (_a = process.env.TS_ETW_MODULE_PATH) !== null && _a !== void 0 ? _a : "./node_modules/@microsoft/typescript-etw";
+        // require() will throw an exception if the module is not found
+        // It may also return undefined if not installed properly
+        etwModule = require(etwModulePath);
+    }
+    catch (e) {
+        etwModule = undefined;
+    }
+    /** Performance logger that will generate ETW events if possible - check for `logEvent` member, as `etwModule` will be `{}` when browserified */
+    ts.perfLogger = etwModule && etwModule.logEvent ? etwModule : nullLogger;
+})(ts || (ts = {}));
+/*@internal*/
+/** Tracing events for the compiler. */
+var ts;
+(function (ts) {
+    var tracing;
+    (function (tracing) {
+        var fs;
+        var traceCount = 0;
+        var traceFd;
+        var legendPath;
+        var legend = [];
+        /** Starts tracing for the given project (unless the `fs` module is unavailable). */
+        function startTracing(configFilePath, traceDir, isBuildMode) {
+            ts.Debug.assert(!traceFd, "Tracing already started");
+            if (fs === undefined) {
+                try {
+                    fs = require("fs");
+                }
+                catch (_a) {
+                    fs = false;
+                }
+            }
+            if (!fs) {
+                return;
+            }
+            if (legendPath === undefined) {
+                legendPath = ts.combinePaths(traceDir, "legend.json");
+            }
+            // Note that writing will fail later on if it exists and is not a directory
+            if (!fs.existsSync(traceDir)) {
+                fs.mkdirSync(traceDir, { recursive: true });
+            }
+            var countPart = isBuildMode ? "." + ++traceCount : "";
+            var tracePath = ts.combinePaths(traceDir, "trace" + countPart + ".json");
+            var typesPath = ts.combinePaths(traceDir, "types" + countPart + ".json");
+            legend.push({
+                configFilePath: configFilePath,
+                tracePath: tracePath,
+                typesPath: typesPath,
+            });
+            traceFd = fs.openSync(tracePath, "w");
+            // Start with a prefix that contains some metadata that the devtools profiler expects (also avoids a warning on import)
+            var meta = { cat: "__metadata", ph: "M", ts: 1000 * ts.timestamp(), pid: 1, tid: 1 };
+            fs.writeSync(traceFd, "[\n"
+                + [__assign({ name: "process_name", args: { name: "tsc" } }, meta), __assign({ name: "thread_name", args: { name: "Main" } }, meta), __assign(__assign({ name: "TracingStartedInBrowser" }, meta), { cat: "disabled-by-default-devtools.timeline" })]
+                    .map(function (v) { return JSON.stringify(v); }).join(",\n"));
+        }
+        tracing.startTracing = startTracing;
+        /** Stops tracing for the in-progress project and dumps the type catalog (unless the `fs` module is unavailable). */
+        function stopTracing(typeCatalog) {
+            if (!traceFd) {
+                ts.Debug.assert(!fs, "Tracing is not in progress");
+                return;
+            }
+            ts.Debug.assert(fs);
+            fs.writeSync(traceFd, "\n]\n");
+            fs.closeSync(traceFd);
+            traceFd = undefined;
+            if (typeCatalog) {
+                dumpTypes(typeCatalog);
+            }
+            else {
+                // We pre-computed this path for convenience, but clear it
+                // now that the file won't be created.
+                legend[legend.length - 1].typesPath = undefined;
+            }
+        }
+        tracing.stopTracing = stopTracing;
+        function isTracing() {
+            return !!traceFd;
+        }
+        tracing.isTracing = isTracing;
+        var Phase;
+        (function (Phase) {
+            Phase["Parse"] = "parse";
+            Phase["Program"] = "program";
+            Phase["Bind"] = "bind";
+            Phase["Check"] = "check";
+            Phase["Emit"] = "emit";
+        })(Phase = tracing.Phase || (tracing.Phase = {}));
+        /** Note: `push`/`pop` should be used by default.
+         * `begin`/`end` are for special cases where we need the data point even if the event never
+         * terminates (typically for reducing a scenario too big to trace to one that can be completed).
+         * In the future we might implement an exit handler to dump unfinished events which would
+         * deprecate these operations.
+         */
+        function begin(phase, name, args) {
+            if (!traceFd)
+                return;
+            writeEvent("B", phase, name, args);
+        }
+        tracing.begin = begin;
+        function end(phase, name, args) {
+            if (!traceFd)
+                return;
+            writeEvent("E", phase, name, args);
+        }
+        tracing.end = end;
+        function instant(phase, name, args) {
+            if (!traceFd)
+                return;
+            writeEvent("I", phase, name, args, "\"s\":\"g\"");
+        }
+        tracing.instant = instant;
+        // Used for "Complete" (ph:"X") events
+        var completeEvents = [];
+        function push(phase, name, args) {
+            if (!traceFd)
+                return;
+            completeEvents.push({ phase: phase, name: name, args: args, time: 1000 * ts.timestamp() });
+        }
+        tracing.push = push;
+        function pop() {
+            if (!traceFd)
+                return;
+            ts.Debug.assert(completeEvents.length > 0);
+            var _a = completeEvents.pop(), phase = _a.phase, name = _a.name, args = _a.args, time = _a.time;
+            var dur = 1000 * ts.timestamp() - time;
+            writeEvent("X", phase, name, args, "\"dur\":" + dur, time);
+        }
+        tracing.pop = pop;
+        function writeEvent(eventType, phase, name, args, extras, time) {
+            if (time === void 0) { time = 1000 * ts.timestamp(); }
+            ts.Debug.assert(traceFd);
+            ts.Debug.assert(fs);
+            ts.performance.mark("beginTracing");
+            fs.writeSync(traceFd, ",\n{\"pid\":1,\"tid\":1,\"ph\":\"" + eventType + "\",\"cat\":\"" + phase + "\",\"ts\":" + time + ",\"name\":\"" + name + "\"");
+            if (extras)
+                fs.writeSync(traceFd, "," + extras);
+            if (args)
+                fs.writeSync(traceFd, ",\"args\":" + JSON.stringify(args));
+            fs.writeSync(traceFd, "}");
+            ts.performance.mark("endTracing");
+            ts.performance.measure("Tracing", "beginTracing", "endTracing");
+        }
+        function indexFromOne(lc) {
+            return {
+                line: lc.line + 1,
+                character: lc.character + 1,
+            };
+        }
+        function dumpTypes(types) {
+            var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r;
+            ts.Debug.assert(fs);
+            ts.performance.mark("beginDumpTypes");
+            var typesPath = legend[legend.length - 1].typesPath;
+            var typesFd = fs.openSync(typesPath, "w");
+            var recursionIdentityMap = new ts.Map();
+            // Cleverness: no line break here so that the type ID will match the line number
+            fs.writeSync(typesFd, "[");
+            var numTypes = types.length;
+            for (var i = 0; i < numTypes; i++) {
+                var type = types[i];
+                var objectFlags = type.objectFlags;
+                var symbol = (_a = type.aliasSymbol) !== null && _a !== void 0 ? _a : type.symbol;
+                var firstDeclaration = (_b = symbol === null || symbol === void 0 ? void 0 : symbol.declarations) === null || _b === void 0 ? void 0 : _b[0];
+                var firstFile = firstDeclaration && ts.getSourceFileOfNode(firstDeclaration);
+                // It's slow to compute the display text, so skip it unless it's really valuable (or cheap)
+                var display = void 0;
+                if ((objectFlags & 16 /* Anonymous */) | (type.flags & 2944 /* Literal */)) {
+                    try {
+                        display = (_c = type.checker) === null || _c === void 0 ? void 0 : _c.typeToString(type);
+                    }
+                    catch (_s) {
+                        display = undefined;
+                    }
+                }
+                var indexedAccessProperties = {};
+                if (type.flags & 8388608 /* IndexedAccess */) {
+                    var indexedAccessType = type;
+                    indexedAccessProperties = {
+                        indexedAccessObjectType: (_d = indexedAccessType.objectType) === null || _d === void 0 ? void 0 : _d.id,
+                        indexedAccessIndexType: (_e = indexedAccessType.indexType) === null || _e === void 0 ? void 0 : _e.id,
+                    };
+                }
+                var referenceProperties = {};
+                if (objectFlags & 4 /* Reference */) {
+                    var referenceType = type;
+                    referenceProperties = {
+                        instantiatedType: (_f = referenceType.target) === null || _f === void 0 ? void 0 : _f.id,
+                        typeArguments: (_g = referenceType.resolvedTypeArguments) === null || _g === void 0 ? void 0 : _g.map(function (t) { return t.id; }),
+                    };
+                }
+                var conditionalProperties = {};
+                if (type.flags & 16777216 /* Conditional */) {
+                    var conditionalType = type;
+                    conditionalProperties = {
+                        conditionalCheckType: (_h = conditionalType.checkType) === null || _h === void 0 ? void 0 : _h.id,
+                        conditionalExtendsType: (_j = conditionalType.extendsType) === null || _j === void 0 ? void 0 : _j.id,
+                        conditionalTrueType: (_l = (_k = conditionalType.resolvedTrueType) === null || _k === void 0 ? void 0 : _k.id) !== null && _l !== void 0 ? _l : -1,
+                        conditionalFalseType: (_o = (_m = conditionalType.resolvedFalseType) === null || _m === void 0 ? void 0 : _m.id) !== null && _o !== void 0 ? _o : -1,
+                    };
+                }
+                // We can't print out an arbitrary object, so just assign each one a unique number.
+                // Don't call it an "id" so people don't treat it as a type id.
+                var recursionToken = void 0;
+                var recursionIdentity = type.checker.getRecursionIdentity(type);
+                if (recursionIdentity) {
+                    recursionToken = recursionIdentityMap.get(recursionIdentity);
+                    if (!recursionToken) {
+                        recursionToken = recursionIdentityMap.size;
+                        recursionIdentityMap.set(recursionIdentity, recursionToken);
+                    }
+                }
+                var descriptor = __assign(__assign(__assign(__assign({ id: type.id, intrinsicName: type.intrinsicName, symbolName: (symbol === null || symbol === void 0 ? void 0 : symbol.escapedName) && ts.unescapeLeadingUnderscores(symbol.escapedName), recursionId: recursionToken, unionTypes: (type.flags & 1048576 /* Union */) ? (_p = type.types) === null || _p === void 0 ? void 0 : _p.map(function (t) { return t.id; }) : undefined, intersectionTypes: (type.flags & 2097152 /* Intersection */) ? type.types.map(function (t) { return t.id; }) : undefined, aliasTypeArguments: (_q = type.aliasTypeArguments) === null || _q === void 0 ? void 0 : _q.map(function (t) { return t.id; }), keyofType: (type.flags & 4194304 /* Index */) ? (_r = type.type) === null || _r === void 0 ? void 0 : _r.id : undefined }, indexedAccessProperties), referenceProperties), conditionalProperties), { firstDeclaration: firstDeclaration && {
+                        path: firstFile.path,
+                        start: indexFromOne(ts.getLineAndCharacterOfPosition(firstFile, firstDeclaration.pos)),
+                        end: indexFromOne(ts.getLineAndCharacterOfPosition(ts.getSourceFileOfNode(firstDeclaration), firstDeclaration.end)),
+                    }, flags: ts.Debug.formatTypeFlags(type.flags).split("|"), display: display });
+                fs.writeSync(typesFd, JSON.stringify(descriptor));
+                if (i < numTypes - 1) {
+                    fs.writeSync(typesFd, ",\n");
+                }
+            }
+            fs.writeSync(typesFd, "]\n");
+            fs.closeSync(typesFd);
+            ts.performance.mark("endDumpTypes");
+            ts.performance.measure("Dump types", "beginDumpTypes", "endDumpTypes");
+        }
+        function dumpLegend() {
+            if (!legendPath) {
+                return;
+            }
+            ts.Debug.assert(fs);
+            fs.writeFileSync(legendPath, JSON.stringify(legend));
+        }
+        tracing.dumpLegend = dumpLegend;
+    })(tracing = ts.tracing || (ts.tracing = {}));
+})(ts || (ts = {}));
 var ts;
 (function (ts) {
     // token > SyntaxKind.Identifier => token is a keyword
@@ -3161,308 +3956,318 @@ var ts;
         SyntaxKind[SyntaxKind["GreaterThanGreaterThanGreaterThanEqualsToken"] = 71] = "GreaterThanGreaterThanGreaterThanEqualsToken";
         SyntaxKind[SyntaxKind["AmpersandEqualsToken"] = 72] = "AmpersandEqualsToken";
         SyntaxKind[SyntaxKind["BarEqualsToken"] = 73] = "BarEqualsToken";
-        SyntaxKind[SyntaxKind["CaretEqualsToken"] = 74] = "CaretEqualsToken";
+        SyntaxKind[SyntaxKind["BarBarEqualsToken"] = 74] = "BarBarEqualsToken";
+        SyntaxKind[SyntaxKind["AmpersandAmpersandEqualsToken"] = 75] = "AmpersandAmpersandEqualsToken";
+        SyntaxKind[SyntaxKind["QuestionQuestionEqualsToken"] = 76] = "QuestionQuestionEqualsToken";
+        SyntaxKind[SyntaxKind["CaretEqualsToken"] = 77] = "CaretEqualsToken";
         // Identifiers and PrivateIdentifiers
-        SyntaxKind[SyntaxKind["Identifier"] = 75] = "Identifier";
-        SyntaxKind[SyntaxKind["PrivateIdentifier"] = 76] = "PrivateIdentifier";
+        SyntaxKind[SyntaxKind["Identifier"] = 78] = "Identifier";
+        SyntaxKind[SyntaxKind["PrivateIdentifier"] = 79] = "PrivateIdentifier";
         // Reserved words
-        SyntaxKind[SyntaxKind["BreakKeyword"] = 77] = "BreakKeyword";
-        SyntaxKind[SyntaxKind["CaseKeyword"] = 78] = "CaseKeyword";
-        SyntaxKind[SyntaxKind["CatchKeyword"] = 79] = "CatchKeyword";
-        SyntaxKind[SyntaxKind["ClassKeyword"] = 80] = "ClassKeyword";
-        SyntaxKind[SyntaxKind["ConstKeyword"] = 81] = "ConstKeyword";
-        SyntaxKind[SyntaxKind["ContinueKeyword"] = 82] = "ContinueKeyword";
-        SyntaxKind[SyntaxKind["DebuggerKeyword"] = 83] = "DebuggerKeyword";
-        SyntaxKind[SyntaxKind["DefaultKeyword"] = 84] = "DefaultKeyword";
-        SyntaxKind[SyntaxKind["DeleteKeyword"] = 85] = "DeleteKeyword";
-        SyntaxKind[SyntaxKind["DoKeyword"] = 86] = "DoKeyword";
-        SyntaxKind[SyntaxKind["ElseKeyword"] = 87] = "ElseKeyword";
-        SyntaxKind[SyntaxKind["EnumKeyword"] = 88] = "EnumKeyword";
-        SyntaxKind[SyntaxKind["ExportKeyword"] = 89] = "ExportKeyword";
-        SyntaxKind[SyntaxKind["ExtendsKeyword"] = 90] = "ExtendsKeyword";
-        SyntaxKind[SyntaxKind["FalseKeyword"] = 91] = "FalseKeyword";
-        SyntaxKind[SyntaxKind["FinallyKeyword"] = 92] = "FinallyKeyword";
-        SyntaxKind[SyntaxKind["ForKeyword"] = 93] = "ForKeyword";
-        SyntaxKind[SyntaxKind["FunctionKeyword"] = 94] = "FunctionKeyword";
-        SyntaxKind[SyntaxKind["IfKeyword"] = 95] = "IfKeyword";
-        SyntaxKind[SyntaxKind["ImportKeyword"] = 96] = "ImportKeyword";
-        SyntaxKind[SyntaxKind["InKeyword"] = 97] = "InKeyword";
-        SyntaxKind[SyntaxKind["InstanceOfKeyword"] = 98] = "InstanceOfKeyword";
-        SyntaxKind[SyntaxKind["NewKeyword"] = 99] = "NewKeyword";
-        SyntaxKind[SyntaxKind["NullKeyword"] = 100] = "NullKeyword";
-        SyntaxKind[SyntaxKind["ReturnKeyword"] = 101] = "ReturnKeyword";
-        SyntaxKind[SyntaxKind["SuperKeyword"] = 102] = "SuperKeyword";
-        SyntaxKind[SyntaxKind["SwitchKeyword"] = 103] = "SwitchKeyword";
-        SyntaxKind[SyntaxKind["ThisKeyword"] = 104] = "ThisKeyword";
-        SyntaxKind[SyntaxKind["ThrowKeyword"] = 105] = "ThrowKeyword";
-        SyntaxKind[SyntaxKind["TrueKeyword"] = 106] = "TrueKeyword";
-        SyntaxKind[SyntaxKind["TryKeyword"] = 107] = "TryKeyword";
-        SyntaxKind[SyntaxKind["TypeOfKeyword"] = 108] = "TypeOfKeyword";
-        SyntaxKind[SyntaxKind["VarKeyword"] = 109] = "VarKeyword";
-        SyntaxKind[SyntaxKind["VoidKeyword"] = 110] = "VoidKeyword";
-        SyntaxKind[SyntaxKind["WhileKeyword"] = 111] = "WhileKeyword";
-        SyntaxKind[SyntaxKind["WithKeyword"] = 112] = "WithKeyword";
+        SyntaxKind[SyntaxKind["BreakKeyword"] = 80] = "BreakKeyword";
+        SyntaxKind[SyntaxKind["CaseKeyword"] = 81] = "CaseKeyword";
+        SyntaxKind[SyntaxKind["CatchKeyword"] = 82] = "CatchKeyword";
+        SyntaxKind[SyntaxKind["ClassKeyword"] = 83] = "ClassKeyword";
+        SyntaxKind[SyntaxKind["ConstKeyword"] = 84] = "ConstKeyword";
+        SyntaxKind[SyntaxKind["ContinueKeyword"] = 85] = "ContinueKeyword";
+        SyntaxKind[SyntaxKind["DebuggerKeyword"] = 86] = "DebuggerKeyword";
+        SyntaxKind[SyntaxKind["DefaultKeyword"] = 87] = "DefaultKeyword";
+        SyntaxKind[SyntaxKind["DeleteKeyword"] = 88] = "DeleteKeyword";
+        SyntaxKind[SyntaxKind["DoKeyword"] = 89] = "DoKeyword";
+        SyntaxKind[SyntaxKind["ElseKeyword"] = 90] = "ElseKeyword";
+        SyntaxKind[SyntaxKind["EnumKeyword"] = 91] = "EnumKeyword";
+        SyntaxKind[SyntaxKind["ExportKeyword"] = 92] = "ExportKeyword";
+        SyntaxKind[SyntaxKind["ExtendsKeyword"] = 93] = "ExtendsKeyword";
+        SyntaxKind[SyntaxKind["FalseKeyword"] = 94] = "FalseKeyword";
+        SyntaxKind[SyntaxKind["FinallyKeyword"] = 95] = "FinallyKeyword";
+        SyntaxKind[SyntaxKind["ForKeyword"] = 96] = "ForKeyword";
+        SyntaxKind[SyntaxKind["FunctionKeyword"] = 97] = "FunctionKeyword";
+        SyntaxKind[SyntaxKind["IfKeyword"] = 98] = "IfKeyword";
+        SyntaxKind[SyntaxKind["ImportKeyword"] = 99] = "ImportKeyword";
+        SyntaxKind[SyntaxKind["InKeyword"] = 100] = "InKeyword";
+        SyntaxKind[SyntaxKind["InstanceOfKeyword"] = 101] = "InstanceOfKeyword";
+        SyntaxKind[SyntaxKind["NewKeyword"] = 102] = "NewKeyword";
+        SyntaxKind[SyntaxKind["NullKeyword"] = 103] = "NullKeyword";
+        SyntaxKind[SyntaxKind["ReturnKeyword"] = 104] = "ReturnKeyword";
+        SyntaxKind[SyntaxKind["SuperKeyword"] = 105] = "SuperKeyword";
+        SyntaxKind[SyntaxKind["SwitchKeyword"] = 106] = "SwitchKeyword";
+        SyntaxKind[SyntaxKind["ThisKeyword"] = 107] = "ThisKeyword";
+        SyntaxKind[SyntaxKind["ThrowKeyword"] = 108] = "ThrowKeyword";
+        SyntaxKind[SyntaxKind["TrueKeyword"] = 109] = "TrueKeyword";
+        SyntaxKind[SyntaxKind["TryKeyword"] = 110] = "TryKeyword";
+        SyntaxKind[SyntaxKind["TypeOfKeyword"] = 111] = "TypeOfKeyword";
+        SyntaxKind[SyntaxKind["VarKeyword"] = 112] = "VarKeyword";
+        SyntaxKind[SyntaxKind["VoidKeyword"] = 113] = "VoidKeyword";
+        SyntaxKind[SyntaxKind["WhileKeyword"] = 114] = "WhileKeyword";
+        SyntaxKind[SyntaxKind["WithKeyword"] = 115] = "WithKeyword";
         // Strict mode reserved words
-        SyntaxKind[SyntaxKind["ImplementsKeyword"] = 113] = "ImplementsKeyword";
-        SyntaxKind[SyntaxKind["InterfaceKeyword"] = 114] = "InterfaceKeyword";
-        SyntaxKind[SyntaxKind["LetKeyword"] = 115] = "LetKeyword";
-        SyntaxKind[SyntaxKind["PackageKeyword"] = 116] = "PackageKeyword";
-        SyntaxKind[SyntaxKind["PrivateKeyword"] = 117] = "PrivateKeyword";
-        SyntaxKind[SyntaxKind["ProtectedKeyword"] = 118] = "ProtectedKeyword";
-        SyntaxKind[SyntaxKind["PublicKeyword"] = 119] = "PublicKeyword";
-        SyntaxKind[SyntaxKind["StaticKeyword"] = 120] = "StaticKeyword";
-        SyntaxKind[SyntaxKind["YieldKeyword"] = 121] = "YieldKeyword";
+        SyntaxKind[SyntaxKind["ImplementsKeyword"] = 116] = "ImplementsKeyword";
+        SyntaxKind[SyntaxKind["InterfaceKeyword"] = 117] = "InterfaceKeyword";
+        SyntaxKind[SyntaxKind["LetKeyword"] = 118] = "LetKeyword";
+        SyntaxKind[SyntaxKind["PackageKeyword"] = 119] = "PackageKeyword";
+        SyntaxKind[SyntaxKind["PrivateKeyword"] = 120] = "PrivateKeyword";
+        SyntaxKind[SyntaxKind["ProtectedKeyword"] = 121] = "ProtectedKeyword";
+        SyntaxKind[SyntaxKind["PublicKeyword"] = 122] = "PublicKeyword";
+        SyntaxKind[SyntaxKind["StaticKeyword"] = 123] = "StaticKeyword";
+        SyntaxKind[SyntaxKind["YieldKeyword"] = 124] = "YieldKeyword";
         // Contextual keywords
-        SyntaxKind[SyntaxKind["AbstractKeyword"] = 122] = "AbstractKeyword";
-        SyntaxKind[SyntaxKind["AsKeyword"] = 123] = "AsKeyword";
-        SyntaxKind[SyntaxKind["AssertsKeyword"] = 124] = "AssertsKeyword";
-        SyntaxKind[SyntaxKind["AnyKeyword"] = 125] = "AnyKeyword";
-        SyntaxKind[SyntaxKind["AsyncKeyword"] = 126] = "AsyncKeyword";
-        SyntaxKind[SyntaxKind["AwaitKeyword"] = 127] = "AwaitKeyword";
-        SyntaxKind[SyntaxKind["BooleanKeyword"] = 128] = "BooleanKeyword";
-        SyntaxKind[SyntaxKind["ConstructorKeyword"] = 129] = "ConstructorKeyword";
-        SyntaxKind[SyntaxKind["DeclareKeyword"] = 130] = "DeclareKeyword";
-        SyntaxKind[SyntaxKind["GetKeyword"] = 131] = "GetKeyword";
-        SyntaxKind[SyntaxKind["InferKeyword"] = 132] = "InferKeyword";
-        SyntaxKind[SyntaxKind["IsKeyword"] = 133] = "IsKeyword";
-        SyntaxKind[SyntaxKind["KeyOfKeyword"] = 134] = "KeyOfKeyword";
-        SyntaxKind[SyntaxKind["ModuleKeyword"] = 135] = "ModuleKeyword";
-        SyntaxKind[SyntaxKind["NamespaceKeyword"] = 136] = "NamespaceKeyword";
-        SyntaxKind[SyntaxKind["NeverKeyword"] = 137] = "NeverKeyword";
-        SyntaxKind[SyntaxKind["ReadonlyKeyword"] = 138] = "ReadonlyKeyword";
-        SyntaxKind[SyntaxKind["RequireKeyword"] = 139] = "RequireKeyword";
-        SyntaxKind[SyntaxKind["NumberKeyword"] = 140] = "NumberKeyword";
-        SyntaxKind[SyntaxKind["ObjectKeyword"] = 141] = "ObjectKeyword";
-        SyntaxKind[SyntaxKind["SetKeyword"] = 142] = "SetKeyword";
-        SyntaxKind[SyntaxKind["StringKeyword"] = 143] = "StringKeyword";
-        SyntaxKind[SyntaxKind["SymbolKeyword"] = 144] = "SymbolKeyword";
-        SyntaxKind[SyntaxKind["TypeKeyword"] = 145] = "TypeKeyword";
-        SyntaxKind[SyntaxKind["UndefinedKeyword"] = 146] = "UndefinedKeyword";
-        SyntaxKind[SyntaxKind["UniqueKeyword"] = 147] = "UniqueKeyword";
-        SyntaxKind[SyntaxKind["UnknownKeyword"] = 148] = "UnknownKeyword";
-        SyntaxKind[SyntaxKind["FromKeyword"] = 149] = "FromKeyword";
-        SyntaxKind[SyntaxKind["GlobalKeyword"] = 150] = "GlobalKeyword";
-        SyntaxKind[SyntaxKind["BigIntKeyword"] = 151] = "BigIntKeyword";
-        SyntaxKind[SyntaxKind["OfKeyword"] = 152] = "OfKeyword";
+        SyntaxKind[SyntaxKind["AbstractKeyword"] = 125] = "AbstractKeyword";
+        SyntaxKind[SyntaxKind["AsKeyword"] = 126] = "AsKeyword";
+        SyntaxKind[SyntaxKind["AssertsKeyword"] = 127] = "AssertsKeyword";
+        SyntaxKind[SyntaxKind["AnyKeyword"] = 128] = "AnyKeyword";
+        SyntaxKind[SyntaxKind["AsyncKeyword"] = 129] = "AsyncKeyword";
+        SyntaxKind[SyntaxKind["AwaitKeyword"] = 130] = "AwaitKeyword";
+        SyntaxKind[SyntaxKind["BooleanKeyword"] = 131] = "BooleanKeyword";
+        SyntaxKind[SyntaxKind["ConstructorKeyword"] = 132] = "ConstructorKeyword";
+        SyntaxKind[SyntaxKind["DeclareKeyword"] = 133] = "DeclareKeyword";
+        SyntaxKind[SyntaxKind["GetKeyword"] = 134] = "GetKeyword";
+        SyntaxKind[SyntaxKind["InferKeyword"] = 135] = "InferKeyword";
+        SyntaxKind[SyntaxKind["IntrinsicKeyword"] = 136] = "IntrinsicKeyword";
+        SyntaxKind[SyntaxKind["IsKeyword"] = 137] = "IsKeyword";
+        SyntaxKind[SyntaxKind["KeyOfKeyword"] = 138] = "KeyOfKeyword";
+        SyntaxKind[SyntaxKind["ModuleKeyword"] = 139] = "ModuleKeyword";
+        SyntaxKind[SyntaxKind["NamespaceKeyword"] = 140] = "NamespaceKeyword";
+        SyntaxKind[SyntaxKind["NeverKeyword"] = 141] = "NeverKeyword";
+        SyntaxKind[SyntaxKind["ReadonlyKeyword"] = 142] = "ReadonlyKeyword";
+        SyntaxKind[SyntaxKind["RequireKeyword"] = 143] = "RequireKeyword";
+        SyntaxKind[SyntaxKind["NumberKeyword"] = 144] = "NumberKeyword";
+        SyntaxKind[SyntaxKind["ObjectKeyword"] = 145] = "ObjectKeyword";
+        SyntaxKind[SyntaxKind["SetKeyword"] = 146] = "SetKeyword";
+        SyntaxKind[SyntaxKind["StringKeyword"] = 147] = "StringKeyword";
+        SyntaxKind[SyntaxKind["SymbolKeyword"] = 148] = "SymbolKeyword";
+        SyntaxKind[SyntaxKind["TypeKeyword"] = 149] = "TypeKeyword";
+        SyntaxKind[SyntaxKind["UndefinedKeyword"] = 150] = "UndefinedKeyword";
+        SyntaxKind[SyntaxKind["UniqueKeyword"] = 151] = "UniqueKeyword";
+        SyntaxKind[SyntaxKind["UnknownKeyword"] = 152] = "UnknownKeyword";
+        SyntaxKind[SyntaxKind["FromKeyword"] = 153] = "FromKeyword";
+        SyntaxKind[SyntaxKind["GlobalKeyword"] = 154] = "GlobalKeyword";
+        SyntaxKind[SyntaxKind["BigIntKeyword"] = 155] = "BigIntKeyword";
+        SyntaxKind[SyntaxKind["OfKeyword"] = 156] = "OfKeyword";
         // Parse tree nodes
         // Names
-        SyntaxKind[SyntaxKind["QualifiedName"] = 153] = "QualifiedName";
-        SyntaxKind[SyntaxKind["ComputedPropertyName"] = 154] = "ComputedPropertyName";
+        SyntaxKind[SyntaxKind["QualifiedName"] = 157] = "QualifiedName";
+        SyntaxKind[SyntaxKind["ComputedPropertyName"] = 158] = "ComputedPropertyName";
         // Signature elements
-        SyntaxKind[SyntaxKind["TypeParameter"] = 155] = "TypeParameter";
-        SyntaxKind[SyntaxKind["Parameter"] = 156] = "Parameter";
-        SyntaxKind[SyntaxKind["Decorator"] = 157] = "Decorator";
+        SyntaxKind[SyntaxKind["TypeParameter"] = 159] = "TypeParameter";
+        SyntaxKind[SyntaxKind["Parameter"] = 160] = "Parameter";
+        SyntaxKind[SyntaxKind["Decorator"] = 161] = "Decorator";
         // TypeMember
-        SyntaxKind[SyntaxKind["PropertySignature"] = 158] = "PropertySignature";
-        SyntaxKind[SyntaxKind["PropertyDeclaration"] = 159] = "PropertyDeclaration";
-        SyntaxKind[SyntaxKind["MethodSignature"] = 160] = "MethodSignature";
-        SyntaxKind[SyntaxKind["MethodDeclaration"] = 161] = "MethodDeclaration";
-        SyntaxKind[SyntaxKind["Constructor"] = 162] = "Constructor";
-        SyntaxKind[SyntaxKind["GetAccessor"] = 163] = "GetAccessor";
-        SyntaxKind[SyntaxKind["SetAccessor"] = 164] = "SetAccessor";
-        SyntaxKind[SyntaxKind["CallSignature"] = 165] = "CallSignature";
-        SyntaxKind[SyntaxKind["ConstructSignature"] = 166] = "ConstructSignature";
-        SyntaxKind[SyntaxKind["IndexSignature"] = 167] = "IndexSignature";
+        SyntaxKind[SyntaxKind["PropertySignature"] = 162] = "PropertySignature";
+        SyntaxKind[SyntaxKind["PropertyDeclaration"] = 163] = "PropertyDeclaration";
+        SyntaxKind[SyntaxKind["MethodSignature"] = 164] = "MethodSignature";
+        SyntaxKind[SyntaxKind["MethodDeclaration"] = 165] = "MethodDeclaration";
+        SyntaxKind[SyntaxKind["Constructor"] = 166] = "Constructor";
+        SyntaxKind[SyntaxKind["GetAccessor"] = 167] = "GetAccessor";
+        SyntaxKind[SyntaxKind["SetAccessor"] = 168] = "SetAccessor";
+        SyntaxKind[SyntaxKind["CallSignature"] = 169] = "CallSignature";
+        SyntaxKind[SyntaxKind["ConstructSignature"] = 170] = "ConstructSignature";
+        SyntaxKind[SyntaxKind["IndexSignature"] = 171] = "IndexSignature";
         // Type
-        SyntaxKind[SyntaxKind["TypePredicate"] = 168] = "TypePredicate";
-        SyntaxKind[SyntaxKind["TypeReference"] = 169] = "TypeReference";
-        SyntaxKind[SyntaxKind["FunctionType"] = 170] = "FunctionType";
-        SyntaxKind[SyntaxKind["ConstructorType"] = 171] = "ConstructorType";
-        SyntaxKind[SyntaxKind["TypeQuery"] = 172] = "TypeQuery";
-        SyntaxKind[SyntaxKind["TypeLiteral"] = 173] = "TypeLiteral";
-        SyntaxKind[SyntaxKind["ArrayType"] = 174] = "ArrayType";
-        SyntaxKind[SyntaxKind["TupleType"] = 175] = "TupleType";
-        SyntaxKind[SyntaxKind["OptionalType"] = 176] = "OptionalType";
-        SyntaxKind[SyntaxKind["RestType"] = 177] = "RestType";
-        SyntaxKind[SyntaxKind["UnionType"] = 178] = "UnionType";
-        SyntaxKind[SyntaxKind["IntersectionType"] = 179] = "IntersectionType";
-        SyntaxKind[SyntaxKind["ConditionalType"] = 180] = "ConditionalType";
-        SyntaxKind[SyntaxKind["InferType"] = 181] = "InferType";
-        SyntaxKind[SyntaxKind["ParenthesizedType"] = 182] = "ParenthesizedType";
-        SyntaxKind[SyntaxKind["ThisType"] = 183] = "ThisType";
-        SyntaxKind[SyntaxKind["TypeOperator"] = 184] = "TypeOperator";
-        SyntaxKind[SyntaxKind["IndexedAccessType"] = 185] = "IndexedAccessType";
-        SyntaxKind[SyntaxKind["MappedType"] = 186] = "MappedType";
-        SyntaxKind[SyntaxKind["LiteralType"] = 187] = "LiteralType";
-        SyntaxKind[SyntaxKind["ImportType"] = 188] = "ImportType";
+        SyntaxKind[SyntaxKind["TypePredicate"] = 172] = "TypePredicate";
+        SyntaxKind[SyntaxKind["TypeReference"] = 173] = "TypeReference";
+        SyntaxKind[SyntaxKind["FunctionType"] = 174] = "FunctionType";
+        SyntaxKind[SyntaxKind["ConstructorType"] = 175] = "ConstructorType";
+        SyntaxKind[SyntaxKind["TypeQuery"] = 176] = "TypeQuery";
+        SyntaxKind[SyntaxKind["TypeLiteral"] = 177] = "TypeLiteral";
+        SyntaxKind[SyntaxKind["ArrayType"] = 178] = "ArrayType";
+        SyntaxKind[SyntaxKind["TupleType"] = 179] = "TupleType";
+        SyntaxKind[SyntaxKind["OptionalType"] = 180] = "OptionalType";
+        SyntaxKind[SyntaxKind["RestType"] = 181] = "RestType";
+        SyntaxKind[SyntaxKind["UnionType"] = 182] = "UnionType";
+        SyntaxKind[SyntaxKind["IntersectionType"] = 183] = "IntersectionType";
+        SyntaxKind[SyntaxKind["ConditionalType"] = 184] = "ConditionalType";
+        SyntaxKind[SyntaxKind["InferType"] = 185] = "InferType";
+        SyntaxKind[SyntaxKind["ParenthesizedType"] = 186] = "ParenthesizedType";
+        SyntaxKind[SyntaxKind["ThisType"] = 187] = "ThisType";
+        SyntaxKind[SyntaxKind["TypeOperator"] = 188] = "TypeOperator";
+        SyntaxKind[SyntaxKind["IndexedAccessType"] = 189] = "IndexedAccessType";
+        SyntaxKind[SyntaxKind["MappedType"] = 190] = "MappedType";
+        SyntaxKind[SyntaxKind["LiteralType"] = 191] = "LiteralType";
+        SyntaxKind[SyntaxKind["NamedTupleMember"] = 192] = "NamedTupleMember";
+        SyntaxKind[SyntaxKind["TemplateLiteralType"] = 193] = "TemplateLiteralType";
+        SyntaxKind[SyntaxKind["TemplateLiteralTypeSpan"] = 194] = "TemplateLiteralTypeSpan";
+        SyntaxKind[SyntaxKind["ImportType"] = 195] = "ImportType";
         // Binding patterns
-        SyntaxKind[SyntaxKind["ObjectBindingPattern"] = 189] = "ObjectBindingPattern";
-        SyntaxKind[SyntaxKind["ArrayBindingPattern"] = 190] = "ArrayBindingPattern";
-        SyntaxKind[SyntaxKind["BindingElement"] = 191] = "BindingElement";
+        SyntaxKind[SyntaxKind["ObjectBindingPattern"] = 196] = "ObjectBindingPattern";
+        SyntaxKind[SyntaxKind["ArrayBindingPattern"] = 197] = "ArrayBindingPattern";
+        SyntaxKind[SyntaxKind["BindingElement"] = 198] = "BindingElement";
         // Expression
-        SyntaxKind[SyntaxKind["ArrayLiteralExpression"] = 192] = "ArrayLiteralExpression";
-        SyntaxKind[SyntaxKind["ObjectLiteralExpression"] = 193] = "ObjectLiteralExpression";
-        SyntaxKind[SyntaxKind["PropertyAccessExpression"] = 194] = "PropertyAccessExpression";
-        SyntaxKind[SyntaxKind["ElementAccessExpression"] = 195] = "ElementAccessExpression";
-        SyntaxKind[SyntaxKind["CallExpression"] = 196] = "CallExpression";
-        SyntaxKind[SyntaxKind["NewExpression"] = 197] = "NewExpression";
-        SyntaxKind[SyntaxKind["TaggedTemplateExpression"] = 198] = "TaggedTemplateExpression";
-        SyntaxKind[SyntaxKind["TypeAssertionExpression"] = 199] = "TypeAssertionExpression";
-        SyntaxKind[SyntaxKind["ParenthesizedExpression"] = 200] = "ParenthesizedExpression";
-        SyntaxKind[SyntaxKind["FunctionExpression"] = 201] = "FunctionExpression";
-        SyntaxKind[SyntaxKind["ArrowFunction"] = 202] = "ArrowFunction";
-        SyntaxKind[SyntaxKind["DeleteExpression"] = 203] = "DeleteExpression";
-        SyntaxKind[SyntaxKind["TypeOfExpression"] = 204] = "TypeOfExpression";
-        SyntaxKind[SyntaxKind["VoidExpression"] = 205] = "VoidExpression";
-        SyntaxKind[SyntaxKind["AwaitExpression"] = 206] = "AwaitExpression";
-        SyntaxKind[SyntaxKind["PrefixUnaryExpression"] = 207] = "PrefixUnaryExpression";
-        SyntaxKind[SyntaxKind["PostfixUnaryExpression"] = 208] = "PostfixUnaryExpression";
-        SyntaxKind[SyntaxKind["BinaryExpression"] = 209] = "BinaryExpression";
-        SyntaxKind[SyntaxKind["ConditionalExpression"] = 210] = "ConditionalExpression";
-        SyntaxKind[SyntaxKind["TemplateExpression"] = 211] = "TemplateExpression";
-        SyntaxKind[SyntaxKind["YieldExpression"] = 212] = "YieldExpression";
-        SyntaxKind[SyntaxKind["SpreadElement"] = 213] = "SpreadElement";
-        SyntaxKind[SyntaxKind["ClassExpression"] = 214] = "ClassExpression";
-        SyntaxKind[SyntaxKind["OmittedExpression"] = 215] = "OmittedExpression";
-        SyntaxKind[SyntaxKind["ExpressionWithTypeArguments"] = 216] = "ExpressionWithTypeArguments";
-        SyntaxKind[SyntaxKind["AsExpression"] = 217] = "AsExpression";
-        SyntaxKind[SyntaxKind["NonNullExpression"] = 218] = "NonNullExpression";
-        SyntaxKind[SyntaxKind["MetaProperty"] = 219] = "MetaProperty";
-        SyntaxKind[SyntaxKind["SyntheticExpression"] = 220] = "SyntheticExpression";
+        SyntaxKind[SyntaxKind["ArrayLiteralExpression"] = 199] = "ArrayLiteralExpression";
+        SyntaxKind[SyntaxKind["ObjectLiteralExpression"] = 200] = "ObjectLiteralExpression";
+        SyntaxKind[SyntaxKind["PropertyAccessExpression"] = 201] = "PropertyAccessExpression";
+        SyntaxKind[SyntaxKind["ElementAccessExpression"] = 202] = "ElementAccessExpression";
+        SyntaxKind[SyntaxKind["CallExpression"] = 203] = "CallExpression";
+        SyntaxKind[SyntaxKind["NewExpression"] = 204] = "NewExpression";
+        SyntaxKind[SyntaxKind["TaggedTemplateExpression"] = 205] = "TaggedTemplateExpression";
+        SyntaxKind[SyntaxKind["TypeAssertionExpression"] = 206] = "TypeAssertionExpression";
+        SyntaxKind[SyntaxKind["ParenthesizedExpression"] = 207] = "ParenthesizedExpression";
+        SyntaxKind[SyntaxKind["FunctionExpression"] = 208] = "FunctionExpression";
+        SyntaxKind[SyntaxKind["ArrowFunction"] = 209] = "ArrowFunction";
+        SyntaxKind[SyntaxKind["DeleteExpression"] = 210] = "DeleteExpression";
+        SyntaxKind[SyntaxKind["TypeOfExpression"] = 211] = "TypeOfExpression";
+        SyntaxKind[SyntaxKind["VoidExpression"] = 212] = "VoidExpression";
+        SyntaxKind[SyntaxKind["AwaitExpression"] = 213] = "AwaitExpression";
+        SyntaxKind[SyntaxKind["PrefixUnaryExpression"] = 214] = "PrefixUnaryExpression";
+        SyntaxKind[SyntaxKind["PostfixUnaryExpression"] = 215] = "PostfixUnaryExpression";
+        SyntaxKind[SyntaxKind["BinaryExpression"] = 216] = "BinaryExpression";
+        SyntaxKind[SyntaxKind["ConditionalExpression"] = 217] = "ConditionalExpression";
+        SyntaxKind[SyntaxKind["TemplateExpression"] = 218] = "TemplateExpression";
+        SyntaxKind[SyntaxKind["YieldExpression"] = 219] = "YieldExpression";
+        SyntaxKind[SyntaxKind["SpreadElement"] = 220] = "SpreadElement";
+        SyntaxKind[SyntaxKind["ClassExpression"] = 221] = "ClassExpression";
+        SyntaxKind[SyntaxKind["OmittedExpression"] = 222] = "OmittedExpression";
+        SyntaxKind[SyntaxKind["ExpressionWithTypeArguments"] = 223] = "ExpressionWithTypeArguments";
+        SyntaxKind[SyntaxKind["AsExpression"] = 224] = "AsExpression";
+        SyntaxKind[SyntaxKind["NonNullExpression"] = 225] = "NonNullExpression";
+        SyntaxKind[SyntaxKind["MetaProperty"] = 226] = "MetaProperty";
+        SyntaxKind[SyntaxKind["SyntheticExpression"] = 227] = "SyntheticExpression";
         // Misc
-        SyntaxKind[SyntaxKind["TemplateSpan"] = 221] = "TemplateSpan";
-        SyntaxKind[SyntaxKind["SemicolonClassElement"] = 222] = "SemicolonClassElement";
+        SyntaxKind[SyntaxKind["TemplateSpan"] = 228] = "TemplateSpan";
+        SyntaxKind[SyntaxKind["SemicolonClassElement"] = 229] = "SemicolonClassElement";
         // Element
-        SyntaxKind[SyntaxKind["Block"] = 223] = "Block";
-        SyntaxKind[SyntaxKind["EmptyStatement"] = 224] = "EmptyStatement";
-        SyntaxKind[SyntaxKind["VariableStatement"] = 225] = "VariableStatement";
-        SyntaxKind[SyntaxKind["ExpressionStatement"] = 226] = "ExpressionStatement";
-        SyntaxKind[SyntaxKind["IfStatement"] = 227] = "IfStatement";
-        SyntaxKind[SyntaxKind["DoStatement"] = 228] = "DoStatement";
-        SyntaxKind[SyntaxKind["WhileStatement"] = 229] = "WhileStatement";
-        SyntaxKind[SyntaxKind["ForStatement"] = 230] = "ForStatement";
-        SyntaxKind[SyntaxKind["ForInStatement"] = 231] = "ForInStatement";
-        SyntaxKind[SyntaxKind["ForOfStatement"] = 232] = "ForOfStatement";
-        SyntaxKind[SyntaxKind["ContinueStatement"] = 233] = "ContinueStatement";
-        SyntaxKind[SyntaxKind["BreakStatement"] = 234] = "BreakStatement";
-        SyntaxKind[SyntaxKind["ReturnStatement"] = 235] = "ReturnStatement";
-        SyntaxKind[SyntaxKind["WithStatement"] = 236] = "WithStatement";
-        SyntaxKind[SyntaxKind["SwitchStatement"] = 237] = "SwitchStatement";
-        SyntaxKind[SyntaxKind["LabeledStatement"] = 238] = "LabeledStatement";
-        SyntaxKind[SyntaxKind["ThrowStatement"] = 239] = "ThrowStatement";
-        SyntaxKind[SyntaxKind["TryStatement"] = 240] = "TryStatement";
-        SyntaxKind[SyntaxKind["DebuggerStatement"] = 241] = "DebuggerStatement";
-        SyntaxKind[SyntaxKind["VariableDeclaration"] = 242] = "VariableDeclaration";
-        SyntaxKind[SyntaxKind["VariableDeclarationList"] = 243] = "VariableDeclarationList";
-        SyntaxKind[SyntaxKind["FunctionDeclaration"] = 244] = "FunctionDeclaration";
-        SyntaxKind[SyntaxKind["ClassDeclaration"] = 245] = "ClassDeclaration";
-        SyntaxKind[SyntaxKind["InterfaceDeclaration"] = 246] = "InterfaceDeclaration";
-        SyntaxKind[SyntaxKind["TypeAliasDeclaration"] = 247] = "TypeAliasDeclaration";
-        SyntaxKind[SyntaxKind["EnumDeclaration"] = 248] = "EnumDeclaration";
-        SyntaxKind[SyntaxKind["ModuleDeclaration"] = 249] = "ModuleDeclaration";
-        SyntaxKind[SyntaxKind["ModuleBlock"] = 250] = "ModuleBlock";
-        SyntaxKind[SyntaxKind["CaseBlock"] = 251] = "CaseBlock";
-        SyntaxKind[SyntaxKind["NamespaceExportDeclaration"] = 252] = "NamespaceExportDeclaration";
-        SyntaxKind[SyntaxKind["ImportEqualsDeclaration"] = 253] = "ImportEqualsDeclaration";
-        SyntaxKind[SyntaxKind["ImportDeclaration"] = 254] = "ImportDeclaration";
-        SyntaxKind[SyntaxKind["ImportClause"] = 255] = "ImportClause";
-        SyntaxKind[SyntaxKind["NamespaceImport"] = 256] = "NamespaceImport";
-        SyntaxKind[SyntaxKind["NamedImports"] = 257] = "NamedImports";
-        SyntaxKind[SyntaxKind["ImportSpecifier"] = 258] = "ImportSpecifier";
-        SyntaxKind[SyntaxKind["ExportAssignment"] = 259] = "ExportAssignment";
-        SyntaxKind[SyntaxKind["ExportDeclaration"] = 260] = "ExportDeclaration";
-        SyntaxKind[SyntaxKind["NamedExports"] = 261] = "NamedExports";
-        SyntaxKind[SyntaxKind["NamespaceExport"] = 262] = "NamespaceExport";
-        SyntaxKind[SyntaxKind["ExportSpecifier"] = 263] = "ExportSpecifier";
-        SyntaxKind[SyntaxKind["MissingDeclaration"] = 264] = "MissingDeclaration";
+        SyntaxKind[SyntaxKind["Block"] = 230] = "Block";
+        SyntaxKind[SyntaxKind["EmptyStatement"] = 231] = "EmptyStatement";
+        SyntaxKind[SyntaxKind["VariableStatement"] = 232] = "VariableStatement";
+        SyntaxKind[SyntaxKind["ExpressionStatement"] = 233] = "ExpressionStatement";
+        SyntaxKind[SyntaxKind["IfStatement"] = 234] = "IfStatement";
+        SyntaxKind[SyntaxKind["DoStatement"] = 235] = "DoStatement";
+        SyntaxKind[SyntaxKind["WhileStatement"] = 236] = "WhileStatement";
+        SyntaxKind[SyntaxKind["ForStatement"] = 237] = "ForStatement";
+        SyntaxKind[SyntaxKind["ForInStatement"] = 238] = "ForInStatement";
+        SyntaxKind[SyntaxKind["ForOfStatement"] = 239] = "ForOfStatement";
+        SyntaxKind[SyntaxKind["ContinueStatement"] = 240] = "ContinueStatement";
+        SyntaxKind[SyntaxKind["BreakStatement"] = 241] = "BreakStatement";
+        SyntaxKind[SyntaxKind["ReturnStatement"] = 242] = "ReturnStatement";
+        SyntaxKind[SyntaxKind["WithStatement"] = 243] = "WithStatement";
+        SyntaxKind[SyntaxKind["SwitchStatement"] = 244] = "SwitchStatement";
+        SyntaxKind[SyntaxKind["LabeledStatement"] = 245] = "LabeledStatement";
+        SyntaxKind[SyntaxKind["ThrowStatement"] = 246] = "ThrowStatement";
+        SyntaxKind[SyntaxKind["TryStatement"] = 247] = "TryStatement";
+        SyntaxKind[SyntaxKind["DebuggerStatement"] = 248] = "DebuggerStatement";
+        SyntaxKind[SyntaxKind["VariableDeclaration"] = 249] = "VariableDeclaration";
+        SyntaxKind[SyntaxKind["VariableDeclarationList"] = 250] = "VariableDeclarationList";
+        SyntaxKind[SyntaxKind["FunctionDeclaration"] = 251] = "FunctionDeclaration";
+        SyntaxKind[SyntaxKind["ClassDeclaration"] = 252] = "ClassDeclaration";
+        SyntaxKind[SyntaxKind["InterfaceDeclaration"] = 253] = "InterfaceDeclaration";
+        SyntaxKind[SyntaxKind["TypeAliasDeclaration"] = 254] = "TypeAliasDeclaration";
+        SyntaxKind[SyntaxKind["EnumDeclaration"] = 255] = "EnumDeclaration";
+        SyntaxKind[SyntaxKind["ModuleDeclaration"] = 256] = "ModuleDeclaration";
+        SyntaxKind[SyntaxKind["ModuleBlock"] = 257] = "ModuleBlock";
+        SyntaxKind[SyntaxKind["CaseBlock"] = 258] = "CaseBlock";
+        SyntaxKind[SyntaxKind["NamespaceExportDeclaration"] = 259] = "NamespaceExportDeclaration";
+        SyntaxKind[SyntaxKind["ImportEqualsDeclaration"] = 260] = "ImportEqualsDeclaration";
+        SyntaxKind[SyntaxKind["ImportDeclaration"] = 261] = "ImportDeclaration";
+        SyntaxKind[SyntaxKind["ImportClause"] = 262] = "ImportClause";
+        SyntaxKind[SyntaxKind["NamespaceImport"] = 263] = "NamespaceImport";
+        SyntaxKind[SyntaxKind["NamedImports"] = 264] = "NamedImports";
+        SyntaxKind[SyntaxKind["ImportSpecifier"] = 265] = "ImportSpecifier";
+        SyntaxKind[SyntaxKind["ExportAssignment"] = 266] = "ExportAssignment";
+        SyntaxKind[SyntaxKind["ExportDeclaration"] = 267] = "ExportDeclaration";
+        SyntaxKind[SyntaxKind["NamedExports"] = 268] = "NamedExports";
+        SyntaxKind[SyntaxKind["NamespaceExport"] = 269] = "NamespaceExport";
+        SyntaxKind[SyntaxKind["ExportSpecifier"] = 270] = "ExportSpecifier";
+        SyntaxKind[SyntaxKind["MissingDeclaration"] = 271] = "MissingDeclaration";
         // Module references
-        SyntaxKind[SyntaxKind["ExternalModuleReference"] = 265] = "ExternalModuleReference";
+        SyntaxKind[SyntaxKind["ExternalModuleReference"] = 272] = "ExternalModuleReference";
         // JSX
-        SyntaxKind[SyntaxKind["JsxElement"] = 266] = "JsxElement";
-        SyntaxKind[SyntaxKind["JsxSelfClosingElement"] = 267] = "JsxSelfClosingElement";
-        SyntaxKind[SyntaxKind["JsxOpeningElement"] = 268] = "JsxOpeningElement";
-        SyntaxKind[SyntaxKind["JsxClosingElement"] = 269] = "JsxClosingElement";
-        SyntaxKind[SyntaxKind["JsxFragment"] = 270] = "JsxFragment";
-        SyntaxKind[SyntaxKind["JsxOpeningFragment"] = 271] = "JsxOpeningFragment";
-        SyntaxKind[SyntaxKind["JsxClosingFragment"] = 272] = "JsxClosingFragment";
-        SyntaxKind[SyntaxKind["JsxAttribute"] = 273] = "JsxAttribute";
-        SyntaxKind[SyntaxKind["JsxAttributes"] = 274] = "JsxAttributes";
-        SyntaxKind[SyntaxKind["JsxSpreadAttribute"] = 275] = "JsxSpreadAttribute";
-        SyntaxKind[SyntaxKind["JsxExpression"] = 276] = "JsxExpression";
+        SyntaxKind[SyntaxKind["JsxElement"] = 273] = "JsxElement";
+        SyntaxKind[SyntaxKind["JsxSelfClosingElement"] = 274] = "JsxSelfClosingElement";
+        SyntaxKind[SyntaxKind["JsxOpeningElement"] = 275] = "JsxOpeningElement";
+        SyntaxKind[SyntaxKind["JsxClosingElement"] = 276] = "JsxClosingElement";
+        SyntaxKind[SyntaxKind["JsxFragment"] = 277] = "JsxFragment";
+        SyntaxKind[SyntaxKind["JsxOpeningFragment"] = 278] = "JsxOpeningFragment";
+        SyntaxKind[SyntaxKind["JsxClosingFragment"] = 279] = "JsxClosingFragment";
+        SyntaxKind[SyntaxKind["JsxAttribute"] = 280] = "JsxAttribute";
+        SyntaxKind[SyntaxKind["JsxAttributes"] = 281] = "JsxAttributes";
+        SyntaxKind[SyntaxKind["JsxSpreadAttribute"] = 282] = "JsxSpreadAttribute";
+        SyntaxKind[SyntaxKind["JsxExpression"] = 283] = "JsxExpression";
         // Clauses
-        SyntaxKind[SyntaxKind["CaseClause"] = 277] = "CaseClause";
-        SyntaxKind[SyntaxKind["DefaultClause"] = 278] = "DefaultClause";
-        SyntaxKind[SyntaxKind["HeritageClause"] = 279] = "HeritageClause";
-        SyntaxKind[SyntaxKind["CatchClause"] = 280] = "CatchClause";
+        SyntaxKind[SyntaxKind["CaseClause"] = 284] = "CaseClause";
+        SyntaxKind[SyntaxKind["DefaultClause"] = 285] = "DefaultClause";
+        SyntaxKind[SyntaxKind["HeritageClause"] = 286] = "HeritageClause";
+        SyntaxKind[SyntaxKind["CatchClause"] = 287] = "CatchClause";
         // Property assignments
-        SyntaxKind[SyntaxKind["PropertyAssignment"] = 281] = "PropertyAssignment";
-        SyntaxKind[SyntaxKind["ShorthandPropertyAssignment"] = 282] = "ShorthandPropertyAssignment";
-        SyntaxKind[SyntaxKind["SpreadAssignment"] = 283] = "SpreadAssignment";
+        SyntaxKind[SyntaxKind["PropertyAssignment"] = 288] = "PropertyAssignment";
+        SyntaxKind[SyntaxKind["ShorthandPropertyAssignment"] = 289] = "ShorthandPropertyAssignment";
+        SyntaxKind[SyntaxKind["SpreadAssignment"] = 290] = "SpreadAssignment";
         // Enum
-        SyntaxKind[SyntaxKind["EnumMember"] = 284] = "EnumMember";
+        SyntaxKind[SyntaxKind["EnumMember"] = 291] = "EnumMember";
         // Unparsed
-        SyntaxKind[SyntaxKind["UnparsedPrologue"] = 285] = "UnparsedPrologue";
-        SyntaxKind[SyntaxKind["UnparsedPrepend"] = 286] = "UnparsedPrepend";
-        SyntaxKind[SyntaxKind["UnparsedText"] = 287] = "UnparsedText";
-        SyntaxKind[SyntaxKind["UnparsedInternalText"] = 288] = "UnparsedInternalText";
-        SyntaxKind[SyntaxKind["UnparsedSyntheticReference"] = 289] = "UnparsedSyntheticReference";
+        SyntaxKind[SyntaxKind["UnparsedPrologue"] = 292] = "UnparsedPrologue";
+        SyntaxKind[SyntaxKind["UnparsedPrepend"] = 293] = "UnparsedPrepend";
+        SyntaxKind[SyntaxKind["UnparsedText"] = 294] = "UnparsedText";
+        SyntaxKind[SyntaxKind["UnparsedInternalText"] = 295] = "UnparsedInternalText";
+        SyntaxKind[SyntaxKind["UnparsedSyntheticReference"] = 296] = "UnparsedSyntheticReference";
         // Top-level nodes
-        SyntaxKind[SyntaxKind["SourceFile"] = 290] = "SourceFile";
-        SyntaxKind[SyntaxKind["Bundle"] = 291] = "Bundle";
-        SyntaxKind[SyntaxKind["UnparsedSource"] = 292] = "UnparsedSource";
-        SyntaxKind[SyntaxKind["InputFiles"] = 293] = "InputFiles";
+        SyntaxKind[SyntaxKind["SourceFile"] = 297] = "SourceFile";
+        SyntaxKind[SyntaxKind["Bundle"] = 298] = "Bundle";
+        SyntaxKind[SyntaxKind["UnparsedSource"] = 299] = "UnparsedSource";
+        SyntaxKind[SyntaxKind["InputFiles"] = 300] = "InputFiles";
         // JSDoc nodes
-        SyntaxKind[SyntaxKind["JSDocTypeExpression"] = 294] = "JSDocTypeExpression";
+        SyntaxKind[SyntaxKind["JSDocTypeExpression"] = 301] = "JSDocTypeExpression";
+        SyntaxKind[SyntaxKind["JSDocNameReference"] = 302] = "JSDocNameReference";
         // The * type
-        SyntaxKind[SyntaxKind["JSDocAllType"] = 295] = "JSDocAllType";
+        SyntaxKind[SyntaxKind["JSDocAllType"] = 303] = "JSDocAllType";
         // The ? type
-        SyntaxKind[SyntaxKind["JSDocUnknownType"] = 296] = "JSDocUnknownType";
-        SyntaxKind[SyntaxKind["JSDocNullableType"] = 297] = "JSDocNullableType";
-        SyntaxKind[SyntaxKind["JSDocNonNullableType"] = 298] = "JSDocNonNullableType";
-        SyntaxKind[SyntaxKind["JSDocOptionalType"] = 299] = "JSDocOptionalType";
-        SyntaxKind[SyntaxKind["JSDocFunctionType"] = 300] = "JSDocFunctionType";
-        SyntaxKind[SyntaxKind["JSDocVariadicType"] = 301] = "JSDocVariadicType";
+        SyntaxKind[SyntaxKind["JSDocUnknownType"] = 304] = "JSDocUnknownType";
+        SyntaxKind[SyntaxKind["JSDocNullableType"] = 305] = "JSDocNullableType";
+        SyntaxKind[SyntaxKind["JSDocNonNullableType"] = 306] = "JSDocNonNullableType";
+        SyntaxKind[SyntaxKind["JSDocOptionalType"] = 307] = "JSDocOptionalType";
+        SyntaxKind[SyntaxKind["JSDocFunctionType"] = 308] = "JSDocFunctionType";
+        SyntaxKind[SyntaxKind["JSDocVariadicType"] = 309] = "JSDocVariadicType";
         // https://jsdoc.app/about-namepaths.html
-        SyntaxKind[SyntaxKind["JSDocNamepathType"] = 302] = "JSDocNamepathType";
-        SyntaxKind[SyntaxKind["JSDocComment"] = 303] = "JSDocComment";
-        SyntaxKind[SyntaxKind["JSDocTypeLiteral"] = 304] = "JSDocTypeLiteral";
-        SyntaxKind[SyntaxKind["JSDocSignature"] = 305] = "JSDocSignature";
-        SyntaxKind[SyntaxKind["JSDocTag"] = 306] = "JSDocTag";
-        SyntaxKind[SyntaxKind["JSDocAugmentsTag"] = 307] = "JSDocAugmentsTag";
-        SyntaxKind[SyntaxKind["JSDocImplementsTag"] = 308] = "JSDocImplementsTag";
-        SyntaxKind[SyntaxKind["JSDocAuthorTag"] = 309] = "JSDocAuthorTag";
-        SyntaxKind[SyntaxKind["JSDocClassTag"] = 310] = "JSDocClassTag";
-        SyntaxKind[SyntaxKind["JSDocPublicTag"] = 311] = "JSDocPublicTag";
-        SyntaxKind[SyntaxKind["JSDocPrivateTag"] = 312] = "JSDocPrivateTag";
-        SyntaxKind[SyntaxKind["JSDocProtectedTag"] = 313] = "JSDocProtectedTag";
-        SyntaxKind[SyntaxKind["JSDocReadonlyTag"] = 314] = "JSDocReadonlyTag";
-        SyntaxKind[SyntaxKind["JSDocCallbackTag"] = 315] = "JSDocCallbackTag";
-        SyntaxKind[SyntaxKind["JSDocEnumTag"] = 316] = "JSDocEnumTag";
-        SyntaxKind[SyntaxKind["JSDocParameterTag"] = 317] = "JSDocParameterTag";
-        SyntaxKind[SyntaxKind["JSDocReturnTag"] = 318] = "JSDocReturnTag";
-        SyntaxKind[SyntaxKind["JSDocThisTag"] = 319] = "JSDocThisTag";
-        SyntaxKind[SyntaxKind["JSDocTypeTag"] = 320] = "JSDocTypeTag";
-        SyntaxKind[SyntaxKind["JSDocTemplateTag"] = 321] = "JSDocTemplateTag";
-        SyntaxKind[SyntaxKind["JSDocTypedefTag"] = 322] = "JSDocTypedefTag";
-        SyntaxKind[SyntaxKind["JSDocPropertyTag"] = 323] = "JSDocPropertyTag";
+        SyntaxKind[SyntaxKind["JSDocNamepathType"] = 310] = "JSDocNamepathType";
+        SyntaxKind[SyntaxKind["JSDocComment"] = 311] = "JSDocComment";
+        SyntaxKind[SyntaxKind["JSDocTypeLiteral"] = 312] = "JSDocTypeLiteral";
+        SyntaxKind[SyntaxKind["JSDocSignature"] = 313] = "JSDocSignature";
+        SyntaxKind[SyntaxKind["JSDocTag"] = 314] = "JSDocTag";
+        SyntaxKind[SyntaxKind["JSDocAugmentsTag"] = 315] = "JSDocAugmentsTag";
+        SyntaxKind[SyntaxKind["JSDocImplementsTag"] = 316] = "JSDocImplementsTag";
+        SyntaxKind[SyntaxKind["JSDocAuthorTag"] = 317] = "JSDocAuthorTag";
+        SyntaxKind[SyntaxKind["JSDocDeprecatedTag"] = 318] = "JSDocDeprecatedTag";
+        SyntaxKind[SyntaxKind["JSDocClassTag"] = 319] = "JSDocClassTag";
+        SyntaxKind[SyntaxKind["JSDocPublicTag"] = 320] = "JSDocPublicTag";
+        SyntaxKind[SyntaxKind["JSDocPrivateTag"] = 321] = "JSDocPrivateTag";
+        SyntaxKind[SyntaxKind["JSDocProtectedTag"] = 322] = "JSDocProtectedTag";
+        SyntaxKind[SyntaxKind["JSDocReadonlyTag"] = 323] = "JSDocReadonlyTag";
+        SyntaxKind[SyntaxKind["JSDocCallbackTag"] = 324] = "JSDocCallbackTag";
+        SyntaxKind[SyntaxKind["JSDocEnumTag"] = 325] = "JSDocEnumTag";
+        SyntaxKind[SyntaxKind["JSDocParameterTag"] = 326] = "JSDocParameterTag";
+        SyntaxKind[SyntaxKind["JSDocReturnTag"] = 327] = "JSDocReturnTag";
+        SyntaxKind[SyntaxKind["JSDocThisTag"] = 328] = "JSDocThisTag";
+        SyntaxKind[SyntaxKind["JSDocTypeTag"] = 329] = "JSDocTypeTag";
+        SyntaxKind[SyntaxKind["JSDocTemplateTag"] = 330] = "JSDocTemplateTag";
+        SyntaxKind[SyntaxKind["JSDocTypedefTag"] = 331] = "JSDocTypedefTag";
+        SyntaxKind[SyntaxKind["JSDocSeeTag"] = 332] = "JSDocSeeTag";
+        SyntaxKind[SyntaxKind["JSDocPropertyTag"] = 333] = "JSDocPropertyTag";
         // Synthesized list
-        SyntaxKind[SyntaxKind["SyntaxList"] = 324] = "SyntaxList";
+        SyntaxKind[SyntaxKind["SyntaxList"] = 334] = "SyntaxList";
         // Transformation nodes
-        SyntaxKind[SyntaxKind["NotEmittedStatement"] = 325] = "NotEmittedStatement";
-        SyntaxKind[SyntaxKind["PartiallyEmittedExpression"] = 326] = "PartiallyEmittedExpression";
-        SyntaxKind[SyntaxKind["CommaListExpression"] = 327] = "CommaListExpression";
-        SyntaxKind[SyntaxKind["MergeDeclarationMarker"] = 328] = "MergeDeclarationMarker";
-        SyntaxKind[SyntaxKind["EndOfDeclarationMarker"] = 329] = "EndOfDeclarationMarker";
-        SyntaxKind[SyntaxKind["SyntheticReferenceExpression"] = 330] = "SyntheticReferenceExpression";
+        SyntaxKind[SyntaxKind["NotEmittedStatement"] = 335] = "NotEmittedStatement";
+        SyntaxKind[SyntaxKind["PartiallyEmittedExpression"] = 336] = "PartiallyEmittedExpression";
+        SyntaxKind[SyntaxKind["CommaListExpression"] = 337] = "CommaListExpression";
+        SyntaxKind[SyntaxKind["MergeDeclarationMarker"] = 338] = "MergeDeclarationMarker";
+        SyntaxKind[SyntaxKind["EndOfDeclarationMarker"] = 339] = "EndOfDeclarationMarker";
+        SyntaxKind[SyntaxKind["SyntheticReferenceExpression"] = 340] = "SyntheticReferenceExpression";
         // Enum value count
-        SyntaxKind[SyntaxKind["Count"] = 331] = "Count";
+        SyntaxKind[SyntaxKind["Count"] = 341] = "Count";
         // Markers
         SyntaxKind[SyntaxKind["FirstAssignment"] = 62] = "FirstAssignment";
-        SyntaxKind[SyntaxKind["LastAssignment"] = 74] = "LastAssignment";
+        SyntaxKind[SyntaxKind["LastAssignment"] = 77] = "LastAssignment";
         SyntaxKind[SyntaxKind["FirstCompoundAssignment"] = 63] = "FirstCompoundAssignment";
-        SyntaxKind[SyntaxKind["LastCompoundAssignment"] = 74] = "LastCompoundAssignment";
-        SyntaxKind[SyntaxKind["FirstReservedWord"] = 77] = "FirstReservedWord";
-        SyntaxKind[SyntaxKind["LastReservedWord"] = 112] = "LastReservedWord";
-        SyntaxKind[SyntaxKind["FirstKeyword"] = 77] = "FirstKeyword";
-        SyntaxKind[SyntaxKind["LastKeyword"] = 152] = "LastKeyword";
-        SyntaxKind[SyntaxKind["FirstFutureReservedWord"] = 113] = "FirstFutureReservedWord";
-        SyntaxKind[SyntaxKind["LastFutureReservedWord"] = 121] = "LastFutureReservedWord";
-        SyntaxKind[SyntaxKind["FirstTypeNode"] = 168] = "FirstTypeNode";
-        SyntaxKind[SyntaxKind["LastTypeNode"] = 188] = "LastTypeNode";
+        SyntaxKind[SyntaxKind["LastCompoundAssignment"] = 77] = "LastCompoundAssignment";
+        SyntaxKind[SyntaxKind["FirstReservedWord"] = 80] = "FirstReservedWord";
+        SyntaxKind[SyntaxKind["LastReservedWord"] = 115] = "LastReservedWord";
+        SyntaxKind[SyntaxKind["FirstKeyword"] = 80] = "FirstKeyword";
+        SyntaxKind[SyntaxKind["LastKeyword"] = 156] = "LastKeyword";
+        SyntaxKind[SyntaxKind["FirstFutureReservedWord"] = 116] = "FirstFutureReservedWord";
+        SyntaxKind[SyntaxKind["LastFutureReservedWord"] = 124] = "LastFutureReservedWord";
+        SyntaxKind[SyntaxKind["FirstTypeNode"] = 172] = "FirstTypeNode";
+        SyntaxKind[SyntaxKind["LastTypeNode"] = 195] = "LastTypeNode";
         SyntaxKind[SyntaxKind["FirstPunctuation"] = 18] = "FirstPunctuation";
-        SyntaxKind[SyntaxKind["LastPunctuation"] = 74] = "LastPunctuation";
+        SyntaxKind[SyntaxKind["LastPunctuation"] = 77] = "LastPunctuation";
         SyntaxKind[SyntaxKind["FirstToken"] = 0] = "FirstToken";
-        SyntaxKind[SyntaxKind["LastToken"] = 152] = "LastToken";
+        SyntaxKind[SyntaxKind["LastToken"] = 156] = "LastToken";
         SyntaxKind[SyntaxKind["FirstTriviaToken"] = 2] = "FirstTriviaToken";
         SyntaxKind[SyntaxKind["LastTriviaToken"] = 7] = "LastTriviaToken";
         SyntaxKind[SyntaxKind["FirstLiteralToken"] = 8] = "FirstLiteralToken";
@@ -3470,16 +4275,16 @@ var ts;
         SyntaxKind[SyntaxKind["FirstTemplateToken"] = 14] = "FirstTemplateToken";
         SyntaxKind[SyntaxKind["LastTemplateToken"] = 17] = "LastTemplateToken";
         SyntaxKind[SyntaxKind["FirstBinaryOperator"] = 29] = "FirstBinaryOperator";
-        SyntaxKind[SyntaxKind["LastBinaryOperator"] = 74] = "LastBinaryOperator";
-        SyntaxKind[SyntaxKind["FirstStatement"] = 225] = "FirstStatement";
-        SyntaxKind[SyntaxKind["LastStatement"] = 241] = "LastStatement";
-        SyntaxKind[SyntaxKind["FirstNode"] = 153] = "FirstNode";
-        SyntaxKind[SyntaxKind["FirstJSDocNode"] = 294] = "FirstJSDocNode";
-        SyntaxKind[SyntaxKind["LastJSDocNode"] = 323] = "LastJSDocNode";
-        SyntaxKind[SyntaxKind["FirstJSDocTagNode"] = 306] = "FirstJSDocTagNode";
-        SyntaxKind[SyntaxKind["LastJSDocTagNode"] = 323] = "LastJSDocTagNode";
-        /* @internal */ SyntaxKind[SyntaxKind["FirstContextualKeyword"] = 122] = "FirstContextualKeyword";
-        /* @internal */ SyntaxKind[SyntaxKind["LastContextualKeyword"] = 152] = "LastContextualKeyword";
+        SyntaxKind[SyntaxKind["LastBinaryOperator"] = 77] = "LastBinaryOperator";
+        SyntaxKind[SyntaxKind["FirstStatement"] = 232] = "FirstStatement";
+        SyntaxKind[SyntaxKind["LastStatement"] = 248] = "LastStatement";
+        SyntaxKind[SyntaxKind["FirstNode"] = 157] = "FirstNode";
+        SyntaxKind[SyntaxKind["FirstJSDocNode"] = 301] = "FirstJSDocNode";
+        SyntaxKind[SyntaxKind["LastJSDocNode"] = 333] = "LastJSDocNode";
+        SyntaxKind[SyntaxKind["FirstJSDocTagNode"] = 314] = "FirstJSDocTagNode";
+        SyntaxKind[SyntaxKind["LastJSDocTagNode"] = 333] = "LastJSDocTagNode";
+        /* @internal */ SyntaxKind[SyntaxKind["FirstContextualKeyword"] = 125] = "FirstContextualKeyword";
+        /* @internal */ SyntaxKind[SyntaxKind["LastContextualKeyword"] = 156] = "LastContextualKeyword";
     })(SyntaxKind = ts.SyntaxKind || (ts.SyntaxKind = {}));
     var NodeFlags;
     (function (NodeFlags) {
@@ -3520,6 +4325,7 @@ var ts;
         /* @internal */ NodeFlags[NodeFlags["InWithStatement"] = 16777216] = "InWithStatement";
         NodeFlags[NodeFlags["JsonFile"] = 33554432] = "JsonFile";
         /* @internal */ NodeFlags[NodeFlags["TypeCached"] = 67108864] = "TypeCached";
+        /* @internal */ NodeFlags[NodeFlags["Deprecated"] = 134217728] = "Deprecated";
         NodeFlags[NodeFlags["BlockScoped"] = 3] = "BlockScoped";
         NodeFlags[NodeFlags["ReachabilityCheckFlags"] = 768] = "ReachabilityCheckFlags";
         NodeFlags[NodeFlags["ReachabilityAndEmitFlags"] = 2816] = "ReachabilityAndEmitFlags";
@@ -3546,6 +4352,8 @@ var ts;
         ModifierFlags[ModifierFlags["Async"] = 256] = "Async";
         ModifierFlags[ModifierFlags["Default"] = 512] = "Default";
         ModifierFlags[ModifierFlags["Const"] = 2048] = "Const";
+        ModifierFlags[ModifierFlags["HasComputedJSDocModifiers"] = 4096] = "HasComputedJSDocModifiers";
+        ModifierFlags[ModifierFlags["Deprecated"] = 8192] = "Deprecated";
         ModifierFlags[ModifierFlags["HasComputedFlags"] = 536870912] = "HasComputedFlags";
         ModifierFlags[ModifierFlags["AccessibilityModifier"] = 28] = "AccessibilityModifier";
         // Accessibility modifiers and 'readonly' can be attached to a parameter in a constructor to make it a property.
@@ -3553,7 +4361,7 @@ var ts;
         ModifierFlags[ModifierFlags["NonPublicAccessibilityModifier"] = 24] = "NonPublicAccessibilityModifier";
         ModifierFlags[ModifierFlags["TypeScriptModifier"] = 2270] = "TypeScriptModifier";
         ModifierFlags[ModifierFlags["ExportDefault"] = 513] = "ExportDefault";
-        ModifierFlags[ModifierFlags["All"] = 3071] = "All";
+        ModifierFlags[ModifierFlags["All"] = 11263] = "All";
     })(ModifierFlags = ts.ModifierFlags || (ts.ModifierFlags = {}));
     var JsxFlags;
     (function (JsxFlags) {
@@ -3574,20 +4382,20 @@ var ts;
         RelationComparisonResult[RelationComparisonResult["ReportsUnreliable"] = 16] = "ReportsUnreliable";
         RelationComparisonResult[RelationComparisonResult["ReportsMask"] = 24] = "ReportsMask";
     })(RelationComparisonResult = ts.RelationComparisonResult || (ts.RelationComparisonResult = {}));
-    /*@internal*/
     var GeneratedIdentifierFlags;
     (function (GeneratedIdentifierFlags) {
         // Kinds
         GeneratedIdentifierFlags[GeneratedIdentifierFlags["None"] = 0] = "None";
-        GeneratedIdentifierFlags[GeneratedIdentifierFlags["Auto"] = 1] = "Auto";
-        GeneratedIdentifierFlags[GeneratedIdentifierFlags["Loop"] = 2] = "Loop";
-        GeneratedIdentifierFlags[GeneratedIdentifierFlags["Unique"] = 3] = "Unique";
-        GeneratedIdentifierFlags[GeneratedIdentifierFlags["Node"] = 4] = "Node";
-        GeneratedIdentifierFlags[GeneratedIdentifierFlags["KindMask"] = 7] = "KindMask";
+        /*@internal*/ GeneratedIdentifierFlags[GeneratedIdentifierFlags["Auto"] = 1] = "Auto";
+        /*@internal*/ GeneratedIdentifierFlags[GeneratedIdentifierFlags["Loop"] = 2] = "Loop";
+        /*@internal*/ GeneratedIdentifierFlags[GeneratedIdentifierFlags["Unique"] = 3] = "Unique";
+        /*@internal*/ GeneratedIdentifierFlags[GeneratedIdentifierFlags["Node"] = 4] = "Node";
+        /*@internal*/ GeneratedIdentifierFlags[GeneratedIdentifierFlags["KindMask"] = 7] = "KindMask";
         // Flags
         GeneratedIdentifierFlags[GeneratedIdentifierFlags["ReservedInNestedScopes"] = 8] = "ReservedInNestedScopes";
         GeneratedIdentifierFlags[GeneratedIdentifierFlags["Optimistic"] = 16] = "Optimistic";
         GeneratedIdentifierFlags[GeneratedIdentifierFlags["FileLevel"] = 32] = "FileLevel";
+        GeneratedIdentifierFlags[GeneratedIdentifierFlags["AllowNameSubstitution"] = 64] = "AllowNameSubstitution";
     })(GeneratedIdentifierFlags = ts.GeneratedIdentifierFlags || (ts.GeneratedIdentifierFlags = {}));
     var TokenFlags;
     (function (TokenFlags) {
@@ -3615,6 +4423,8 @@ var ts;
         TokenFlags[TokenFlags["BinaryOrOctalSpecifier"] = 384] = "BinaryOrOctalSpecifier";
         /* @internal */
         TokenFlags[TokenFlags["NumericLiteralFlags"] = 1008] = "NumericLiteralFlags";
+        /* @internal */
+        TokenFlags[TokenFlags["TemplateLiteralLikeFlags"] = 2048] = "TemplateLiteralLikeFlags";
     })(TokenFlags = ts.TokenFlags || (ts.TokenFlags = {}));
     // NOTE: Ensure this is up-to-date with src/debug/debug.ts
     var FlowFlags;
@@ -3693,6 +4503,7 @@ var ts;
         ContextFlags[ContextFlags["Signature"] = 1] = "Signature";
         ContextFlags[ContextFlags["NoConstraints"] = 2] = "NoConstraints";
         ContextFlags[ContextFlags["Completions"] = 4] = "Completions";
+        ContextFlags[ContextFlags["SkipBindingPatterns"] = 8] = "SkipBindingPatterns";
     })(ContextFlags = ts.ContextFlags || (ts.ContextFlags = {}));
     // NOTE: If modifying this enum, must modify `TypeFormatFlags` too!
     var NodeBuilderFlags;
@@ -3716,6 +4527,7 @@ var ts;
         NodeBuilderFlags[NodeBuilderFlags["UseAliasDefinedOutsideCurrentScope"] = 16384] = "UseAliasDefinedOutsideCurrentScope";
         NodeBuilderFlags[NodeBuilderFlags["UseSingleQuotesForStringLiteralType"] = 268435456] = "UseSingleQuotesForStringLiteralType";
         NodeBuilderFlags[NodeBuilderFlags["NoTypeReduction"] = 536870912] = "NoTypeReduction";
+        NodeBuilderFlags[NodeBuilderFlags["NoUndefinedOptionalParameterType"] = 1073741824] = "NoUndefinedOptionalParameterType";
         // Error handling
         NodeBuilderFlags[NodeBuilderFlags["AllowThisInObjectLiteral"] = 32768] = "AllowThisInObjectLiteral";
         NodeBuilderFlags[NodeBuilderFlags["AllowQualifedNameInPlaceOfIdentifier"] = 65536] = "AllowQualifedNameInPlaceOfIdentifier";
@@ -4027,6 +4839,8 @@ var ts;
         TypeFlags[TypeFlags["Conditional"] = 16777216] = "Conditional";
         TypeFlags[TypeFlags["Substitution"] = 33554432] = "Substitution";
         TypeFlags[TypeFlags["NonPrimitive"] = 67108864] = "NonPrimitive";
+        TypeFlags[TypeFlags["TemplateLiteral"] = 134217728] = "TemplateLiteral";
+        TypeFlags[TypeFlags["StringMapping"] = 268435456] = "StringMapping";
         /* @internal */
         TypeFlags[TypeFlags["AnyOrUnknown"] = 3] = "AnyOrUnknown";
         /* @internal */
@@ -4043,7 +4857,7 @@ var ts;
         TypeFlags[TypeFlags["Intrinsic"] = 67359327] = "Intrinsic";
         /* @internal */
         TypeFlags[TypeFlags["Primitive"] = 131068] = "Primitive";
-        TypeFlags[TypeFlags["StringLike"] = 132] = "StringLike";
+        TypeFlags[TypeFlags["StringLike"] = 402653316] = "StringLike";
         TypeFlags[TypeFlags["NumberLike"] = 296] = "NumberLike";
         TypeFlags[TypeFlags["BigIntLike"] = 2112] = "BigIntLike";
         TypeFlags[TypeFlags["BooleanLike"] = 528] = "BooleanLike";
@@ -4051,29 +4865,28 @@ var ts;
         TypeFlags[TypeFlags["ESSymbolLike"] = 12288] = "ESSymbolLike";
         TypeFlags[TypeFlags["VoidLike"] = 49152] = "VoidLike";
         /* @internal */
-        TypeFlags[TypeFlags["DisjointDomains"] = 67238908] = "DisjointDomains";
+        TypeFlags[TypeFlags["DisjointDomains"] = 469892092] = "DisjointDomains";
         TypeFlags[TypeFlags["UnionOrIntersection"] = 3145728] = "UnionOrIntersection";
         TypeFlags[TypeFlags["StructuredType"] = 3670016] = "StructuredType";
         TypeFlags[TypeFlags["TypeVariable"] = 8650752] = "TypeVariable";
         TypeFlags[TypeFlags["InstantiableNonPrimitive"] = 58982400] = "InstantiableNonPrimitive";
-        TypeFlags[TypeFlags["InstantiablePrimitive"] = 4194304] = "InstantiablePrimitive";
-        TypeFlags[TypeFlags["Instantiable"] = 63176704] = "Instantiable";
-        TypeFlags[TypeFlags["StructuredOrInstantiable"] = 66846720] = "StructuredOrInstantiable";
+        TypeFlags[TypeFlags["InstantiablePrimitive"] = 406847488] = "InstantiablePrimitive";
+        TypeFlags[TypeFlags["Instantiable"] = 465829888] = "Instantiable";
+        TypeFlags[TypeFlags["StructuredOrInstantiable"] = 469499904] = "StructuredOrInstantiable";
         /* @internal */
         TypeFlags[TypeFlags["ObjectFlagsType"] = 3899393] = "ObjectFlagsType";
         /* @internal */
         TypeFlags[TypeFlags["Simplifiable"] = 25165824] = "Simplifiable";
         /* @internal */
-        TypeFlags[TypeFlags["Substructure"] = 66584576] = "Substructure";
+        TypeFlags[TypeFlags["Substructure"] = 469237760] = "Substructure";
         // 'Narrowable' types are types where narrowing actually narrows.
         // This *should* be every type other than null, undefined, void, and never
-        TypeFlags[TypeFlags["Narrowable"] = 133970943] = "Narrowable";
-        TypeFlags[TypeFlags["NotUnionOrUnit"] = 67637251] = "NotUnionOrUnit";
+        TypeFlags[TypeFlags["Narrowable"] = 536624127] = "Narrowable";
         /* @internal */
-        TypeFlags[TypeFlags["NotPrimitiveUnion"] = 66994211] = "NotPrimitiveUnion";
+        TypeFlags[TypeFlags["NotPrimitiveUnion"] = 469647395] = "NotPrimitiveUnion";
         // The following flags are aggregated during union and intersection type construction
         /* @internal */
-        TypeFlags[TypeFlags["IncludesMask"] = 71041023] = "IncludesMask";
+        TypeFlags[TypeFlags["IncludesMask"] = 205258751] = "IncludesMask";
         // The following flags are used for different purposes during union and intersection type construction
         /* @internal */
         TypeFlags[TypeFlags["IncludesStructuredOrInstantiable"] = 262144] = "IncludesStructuredOrInstantiable";
@@ -4130,11 +4943,16 @@ var ts;
         ObjectFlags[ObjectFlags["IsNeverIntersectionComputed"] = 268435456] = "IsNeverIntersectionComputed";
         /* @internal */
         ObjectFlags[ObjectFlags["IsNeverIntersection"] = 536870912] = "IsNeverIntersection";
+        /* @internal */
+        ObjectFlags[ObjectFlags["IsClassInstanceClone"] = 1073741824] = "IsClassInstanceClone";
         ObjectFlags[ObjectFlags["ClassOrInterface"] = 3] = "ClassOrInterface";
         /* @internal */
         ObjectFlags[ObjectFlags["RequiresWidening"] = 1572864] = "RequiresWidening";
         /* @internal */
         ObjectFlags[ObjectFlags["PropagatingFlags"] = 3670016] = "PropagatingFlags";
+        // Object flags that uniquely identify the kind of ObjectType
+        /* @internal */
+        ObjectFlags[ObjectFlags["ObjectTypeKindMask"] = 2367] = "ObjectTypeKindMask";
     })(ObjectFlags = ts.ObjectFlags || (ts.ObjectFlags = {}));
     /* @internal */
     var VarianceFlags;
@@ -4149,6 +4967,14 @@ var ts;
         VarianceFlags[VarianceFlags["Unreliable"] = 16] = "Unreliable";
         VarianceFlags[VarianceFlags["AllowsStructuralFallback"] = 24] = "AllowsStructuralFallback";
     })(VarianceFlags = ts.VarianceFlags || (ts.VarianceFlags = {}));
+    var ElementFlags;
+    (function (ElementFlags) {
+        ElementFlags[ElementFlags["Required"] = 1] = "Required";
+        ElementFlags[ElementFlags["Optional"] = 2] = "Optional";
+        ElementFlags[ElementFlags["Rest"] = 4] = "Rest";
+        ElementFlags[ElementFlags["Variadic"] = 8] = "Variadic";
+        ElementFlags[ElementFlags["Variable"] = 12] = "Variable";
+    })(ElementFlags = ts.ElementFlags || (ts.ElementFlags = {}));
     /* @internal */
     var JsxReferenceKind;
     (function (JsxReferenceKind) {
@@ -4173,7 +4999,7 @@ var ts;
         // We do not propagate `IsInnerCallChain` to instantiated signatures, as that would result in us
         // attempting to add `| undefined` on each recursive call to `getReturnTypeOfSignature` when
         // instantiating the return type.
-        SignatureFlags[SignatureFlags["PropagatingFlags"] = 3] = "PropagatingFlags";
+        SignatureFlags[SignatureFlags["PropagatingFlags"] = 19] = "PropagatingFlags";
         SignatureFlags[SignatureFlags["CallChainFlags"] = 12] = "CallChainFlags";
     })(SignatureFlags = ts.SignatureFlags || (ts.SignatureFlags = {}));
     var IndexKind;
@@ -4193,16 +5019,17 @@ var ts;
     var InferencePriority;
     (function (InferencePriority) {
         InferencePriority[InferencePriority["NakedTypeVariable"] = 1] = "NakedTypeVariable";
-        InferencePriority[InferencePriority["HomomorphicMappedType"] = 2] = "HomomorphicMappedType";
-        InferencePriority[InferencePriority["PartialHomomorphicMappedType"] = 4] = "PartialHomomorphicMappedType";
-        InferencePriority[InferencePriority["MappedTypeConstraint"] = 8] = "MappedTypeConstraint";
-        InferencePriority[InferencePriority["ContravariantConditional"] = 16] = "ContravariantConditional";
-        InferencePriority[InferencePriority["ReturnType"] = 32] = "ReturnType";
-        InferencePriority[InferencePriority["LiteralKeyof"] = 64] = "LiteralKeyof";
-        InferencePriority[InferencePriority["NoConstraints"] = 128] = "NoConstraints";
-        InferencePriority[InferencePriority["AlwaysStrict"] = 256] = "AlwaysStrict";
-        InferencePriority[InferencePriority["MaxValue"] = 512] = "MaxValue";
-        InferencePriority[InferencePriority["PriorityImpliesCombination"] = 104] = "PriorityImpliesCombination";
+        InferencePriority[InferencePriority["SpeculativeTuple"] = 2] = "SpeculativeTuple";
+        InferencePriority[InferencePriority["HomomorphicMappedType"] = 4] = "HomomorphicMappedType";
+        InferencePriority[InferencePriority["PartialHomomorphicMappedType"] = 8] = "PartialHomomorphicMappedType";
+        InferencePriority[InferencePriority["MappedTypeConstraint"] = 16] = "MappedTypeConstraint";
+        InferencePriority[InferencePriority["ContravariantConditional"] = 32] = "ContravariantConditional";
+        InferencePriority[InferencePriority["ReturnType"] = 64] = "ReturnType";
+        InferencePriority[InferencePriority["LiteralKeyof"] = 128] = "LiteralKeyof";
+        InferencePriority[InferencePriority["NoConstraints"] = 256] = "NoConstraints";
+        InferencePriority[InferencePriority["AlwaysStrict"] = 512] = "AlwaysStrict";
+        InferencePriority[InferencePriority["MaxValue"] = 1024] = "MaxValue";
+        InferencePriority[InferencePriority["PriorityImpliesCombination"] = 208] = "PriorityImpliesCombination";
         InferencePriority[InferencePriority["Circularity"] = -1] = "Circularity";
     })(InferencePriority = ts.InferencePriority || (ts.InferencePriority = {}));
     /* @internal */
@@ -4215,18 +5042,18 @@ var ts;
     })(InferenceFlags = ts.InferenceFlags || (ts.InferenceFlags = {}));
     /**
      * Ternary values are defined such that
-     * x & y is False if either x or y is False.
-     * x & y is Maybe if either x or y is Maybe, but neither x or y is False.
-     * x & y is True if both x and y are True.
-     * x | y is False if both x and y are False.
-     * x | y is Maybe if either x or y is Maybe, but neither x or y is True.
-     * x | y is True if either x or y is True.
+     * x & y picks the lesser in the order False < Unknown < Maybe < True, and
+     * x | y picks the greater in the order False < Unknown < Maybe < True.
+     * Generally, Ternary.Maybe is used as the result of a relation that depends on itself, and
+     * Ternary.Unknown is used as the result of a variance check that depends on itself. We make
+     * a distinction because we don't want to cache circular variance check results.
      */
     /* @internal */
     var Ternary;
     (function (Ternary) {
         Ternary[Ternary["False"] = 0] = "False";
-        Ternary[Ternary["Maybe"] = 1] = "Maybe";
+        Ternary[Ternary["Unknown"] = 1] = "Unknown";
+        Ternary[Ternary["Maybe"] = 3] = "Maybe";
         Ternary[Ternary["True"] = -1] = "True";
     })(Ternary = ts.Ternary || (ts.Ternary = {}));
     /* @internal */
@@ -4234,6 +5061,7 @@ var ts;
     (function (AssignmentDeclarationKind) {
         AssignmentDeclarationKind[AssignmentDeclarationKind["None"] = 0] = "None";
         /// exports.name = expr
+        /// module.exports.name = expr
         AssignmentDeclarationKind[AssignmentDeclarationKind["ExportsProperty"] = 1] = "ExportsProperty";
         /// module.exports = expr
         AssignmentDeclarationKind[AssignmentDeclarationKind["ModuleExports"] = 2] = "ModuleExports";
@@ -4314,6 +5142,8 @@ var ts;
         JsxEmit[JsxEmit["Preserve"] = 1] = "Preserve";
         JsxEmit[JsxEmit["React"] = 2] = "React";
         JsxEmit[JsxEmit["ReactNative"] = 3] = "ReactNative";
+        JsxEmit[JsxEmit["ReactJSX"] = 4] = "ReactJSX";
+        JsxEmit[JsxEmit["ReactJSXDev"] = 5] = "ReactJSXDev";
     })(JsxEmit = ts.JsxEmit || (ts.JsxEmit = {}));
     var ImportsNotUsedAsValues;
     (function (ImportsNotUsedAsValues) {
@@ -4535,6 +5365,7 @@ var ts;
         TransformFlags[TransformFlags["ContainsHoistedDeclarationOrCompletion"] = 1048576] = "ContainsHoistedDeclarationOrCompletion";
         TransformFlags[TransformFlags["ContainsDynamicImport"] = 2097152] = "ContainsDynamicImport";
         TransformFlags[TransformFlags["ContainsClassFields"] = 4194304] = "ContainsClassFields";
+        TransformFlags[TransformFlags["ContainsPossibleTopLevelAwait"] = 8388608] = "ContainsPossibleTopLevelAwait";
         // Please leave this as 1 << 29.
         // It is the maximum bit we can set before we outgrow the size of a v8 small integer (SMI) on an x86 system.
         // It is a good reminder of how much room we have left
@@ -4558,13 +5389,13 @@ var ts;
         TransformFlags[TransformFlags["OuterExpressionExcludes"] = 536870912] = "OuterExpressionExcludes";
         TransformFlags[TransformFlags["PropertyAccessExcludes"] = 536870912] = "PropertyAccessExcludes";
         TransformFlags[TransformFlags["NodeExcludes"] = 536870912] = "NodeExcludes";
-        TransformFlags[TransformFlags["ArrowFunctionExcludes"] = 538920960] = "ArrowFunctionExcludes";
-        TransformFlags[TransformFlags["FunctionExcludes"] = 538925056] = "FunctionExcludes";
-        TransformFlags[TransformFlags["ConstructorExcludes"] = 538923008] = "ConstructorExcludes";
+        TransformFlags[TransformFlags["ArrowFunctionExcludes"] = 547309568] = "ArrowFunctionExcludes";
+        TransformFlags[TransformFlags["FunctionExcludes"] = 547313664] = "FunctionExcludes";
+        TransformFlags[TransformFlags["ConstructorExcludes"] = 547311616] = "ConstructorExcludes";
         TransformFlags[TransformFlags["MethodOrAccessorExcludes"] = 538923008] = "MethodOrAccessorExcludes";
         TransformFlags[TransformFlags["PropertyExcludes"] = 536875008] = "PropertyExcludes";
         TransformFlags[TransformFlags["ClassExcludes"] = 536905728] = "ClassExcludes";
-        TransformFlags[TransformFlags["ModuleExcludes"] = 537991168] = "ModuleExcludes";
+        TransformFlags[TransformFlags["ModuleExcludes"] = 546379776] = "ModuleExcludes";
         TransformFlags[TransformFlags["TypeExcludes"] = -2] = "TypeExcludes";
         TransformFlags[TransformFlags["ObjectLiteralExcludes"] = 536922112] = "ObjectLiteralExcludes";
         TransformFlags[TransformFlags["ArrayLiteralOrCallOrNewExcludes"] = 536879104] = "ArrayLiteralOrCallOrNewExcludes";
@@ -4637,12 +5468,14 @@ var ts;
         ExternalEmitHelpers[ExternalEmitHelpers["AsyncDelegator"] = 16384] = "AsyncDelegator";
         ExternalEmitHelpers[ExternalEmitHelpers["AsyncValues"] = 32768] = "AsyncValues";
         ExternalEmitHelpers[ExternalEmitHelpers["ExportStar"] = 65536] = "ExportStar";
-        ExternalEmitHelpers[ExternalEmitHelpers["MakeTemplateObject"] = 131072] = "MakeTemplateObject";
-        ExternalEmitHelpers[ExternalEmitHelpers["ClassPrivateFieldGet"] = 262144] = "ClassPrivateFieldGet";
-        ExternalEmitHelpers[ExternalEmitHelpers["ClassPrivateFieldSet"] = 524288] = "ClassPrivateFieldSet";
-        ExternalEmitHelpers[ExternalEmitHelpers["CreateBinding"] = 1048576] = "CreateBinding";
+        ExternalEmitHelpers[ExternalEmitHelpers["ImportStar"] = 131072] = "ImportStar";
+        ExternalEmitHelpers[ExternalEmitHelpers["ImportDefault"] = 262144] = "ImportDefault";
+        ExternalEmitHelpers[ExternalEmitHelpers["MakeTemplateObject"] = 524288] = "MakeTemplateObject";
+        ExternalEmitHelpers[ExternalEmitHelpers["ClassPrivateFieldGet"] = 1048576] = "ClassPrivateFieldGet";
+        ExternalEmitHelpers[ExternalEmitHelpers["ClassPrivateFieldSet"] = 2097152] = "ClassPrivateFieldSet";
+        ExternalEmitHelpers[ExternalEmitHelpers["CreateBinding"] = 4194304] = "CreateBinding";
         ExternalEmitHelpers[ExternalEmitHelpers["FirstEmitHelper"] = 1] = "FirstEmitHelper";
-        ExternalEmitHelpers[ExternalEmitHelpers["LastEmitHelper"] = 1048576] = "LastEmitHelper";
+        ExternalEmitHelpers[ExternalEmitHelpers["LastEmitHelper"] = 4194304] = "LastEmitHelper";
         // Helpers included by ES2015 for..of
         ExternalEmitHelpers[ExternalEmitHelpers["ForOfIncludes"] = 256] = "ForOfIncludes";
         // Helpers included by ES2017 for..await..of
@@ -4664,6 +5497,15 @@ var ts;
         EmitHint[EmitHint["EmbeddedStatement"] = 5] = "EmbeddedStatement";
         EmitHint[EmitHint["JsxAttributeValue"] = 6] = "JsxAttributeValue";
     })(EmitHint = ts.EmitHint || (ts.EmitHint = {}));
+    var OuterExpressionKinds;
+    (function (OuterExpressionKinds) {
+        OuterExpressionKinds[OuterExpressionKinds["Parentheses"] = 1] = "Parentheses";
+        OuterExpressionKinds[OuterExpressionKinds["TypeAssertions"] = 2] = "TypeAssertions";
+        OuterExpressionKinds[OuterExpressionKinds["NonNullAssertions"] = 4] = "NonNullAssertions";
+        OuterExpressionKinds[OuterExpressionKinds["PartiallyEmittedExpressions"] = 8] = "PartiallyEmittedExpressions";
+        OuterExpressionKinds[OuterExpressionKinds["Assertions"] = 6] = "Assertions";
+        OuterExpressionKinds[OuterExpressionKinds["All"] = 15] = "All";
+    })(OuterExpressionKinds = ts.OuterExpressionKinds || (ts.OuterExpressionKinds = {}));
     /* @internal */
     var LexicalEnvironmentFlags;
     (function (LexicalEnvironmentFlags) {
@@ -4726,7 +5568,8 @@ var ts;
         ListFormat[ListFormat["HeritageClauses"] = 512] = "HeritageClauses";
         ListFormat[ListFormat["SingleLineTypeLiteralMembers"] = 768] = "SingleLineTypeLiteralMembers";
         ListFormat[ListFormat["MultiLineTypeLiteralMembers"] = 32897] = "MultiLineTypeLiteralMembers";
-        ListFormat[ListFormat["TupleTypeElements"] = 528] = "TupleTypeElements";
+        ListFormat[ListFormat["SingleLineTupleTypeElements"] = 528] = "SingleLineTupleTypeElements";
+        ListFormat[ListFormat["MultiLineTupleTypeElements"] = 657] = "MultiLineTupleTypeElements";
         ListFormat[ListFormat["UnionTypeConstituents"] = 516] = "UnionTypeConstituents";
         ListFormat[ListFormat["IntersectionTypeConstituents"] = 520] = "IntersectionTypeConstituents";
         ListFormat[ListFormat["ObjectBindingPatternElements"] = 525136] = "ObjectBindingPatternElements";
@@ -4815,8 +5658,700 @@ var ts;
             args: [{ name: "factory" }],
             kind: 4 /* MultiLine */
         },
+        "jsxfrag": {
+            args: [{ name: "factory" }],
+            kind: 4 /* MultiLine */
+        },
+        "jsximportsource": {
+            args: [{ name: "factory" }],
+            kind: 4 /* MultiLine */
+        },
+        "jsxruntime": {
+            args: [{ name: "factory" }],
+            kind: 4 /* MultiLine */
+        },
     };
 })(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    /**
+     * Internally, we represent paths as strings with '/' as the directory separator.
+     * When we make system calls (eg: LanguageServiceHost.getDirectory()),
+     * we expect the host to correctly handle paths in our specified format.
+     */
+    ts.directorySeparator = "/";
+    var altDirectorySeparator = "\\";
+    var urlSchemeSeparator = "://";
+    var backslashRegExp = /\\/g;
+    //// Path Tests
+    /**
+     * Determines whether a charCode corresponds to `/` or `\`.
+     */
+    function isAnyDirectorySeparator(charCode) {
+        return charCode === 47 /* slash */ || charCode === 92 /* backslash */;
+    }
+    ts.isAnyDirectorySeparator = isAnyDirectorySeparator;
+    /**
+     * Determines whether a path starts with a URL scheme (e.g. starts with `http://`, `ftp://`, `file://`, etc.).
+     */
+    function isUrl(path) {
+        return getEncodedRootLength(path) < 0;
+    }
+    ts.isUrl = isUrl;
+    /**
+     * Determines whether a path is an absolute disk path (e.g. starts with `/`, or a dos path
+     * like `c:`, `c:\` or `c:/`).
+     */
+    function isRootedDiskPath(path) {
+        return getEncodedRootLength(path) > 0;
+    }
+    ts.isRootedDiskPath = isRootedDiskPath;
+    /**
+     * Determines whether a path consists only of a path root.
+     */
+    function isDiskPathRoot(path) {
+        var rootLength = getEncodedRootLength(path);
+        return rootLength > 0 && rootLength === path.length;
+    }
+    ts.isDiskPathRoot = isDiskPathRoot;
+    /**
+     * Determines whether a path starts with an absolute path component (i.e. `/`, `c:/`, `file://`, etc.).
+     *
+     * ```ts
+     * // POSIX
+     * pathIsAbsolute("/path/to/file.ext") === true
+     * // DOS
+     * pathIsAbsolute("c:/path/to/file.ext") === true
+     * // URL
+     * pathIsAbsolute("file:///path/to/file.ext") === true
+     * // Non-absolute
+     * pathIsAbsolute("path/to/file.ext") === false
+     * pathIsAbsolute("./path/to/file.ext") === false
+     * ```
+     */
+    function pathIsAbsolute(path) {
+        return getEncodedRootLength(path) !== 0;
+    }
+    ts.pathIsAbsolute = pathIsAbsolute;
+    /**
+     * Determines whether a path starts with a relative path component (i.e. `.` or `..`).
+     */
+    function pathIsRelative(path) {
+        return /^\.\.?($|[\\/])/.test(path);
+    }
+    ts.pathIsRelative = pathIsRelative;
+    /**
+     * Determines whether a path is neither relative nor absolute, e.g. "path/to/file".
+     * Also known misleadingly as "non-relative".
+     */
+    function pathIsBareSpecifier(path) {
+        return !pathIsAbsolute(path) && !pathIsRelative(path);
+    }
+    ts.pathIsBareSpecifier = pathIsBareSpecifier;
+    function hasExtension(fileName) {
+        return ts.stringContains(getBaseFileName(fileName), ".");
+    }
+    ts.hasExtension = hasExtension;
+    function fileExtensionIs(path, extension) {
+        return path.length > extension.length && ts.endsWith(path, extension);
+    }
+    ts.fileExtensionIs = fileExtensionIs;
+    function fileExtensionIsOneOf(path, extensions) {
+        for (var _i = 0, extensions_1 = extensions; _i < extensions_1.length; _i++) {
+            var extension = extensions_1[_i];
+            if (fileExtensionIs(path, extension)) {
+                return true;
+            }
+        }
+        return false;
+    }
+    ts.fileExtensionIsOneOf = fileExtensionIsOneOf;
+    /**
+     * Determines whether a path has a trailing separator (`/` or `\\`).
+     */
+    function hasTrailingDirectorySeparator(path) {
+        return path.length > 0 && isAnyDirectorySeparator(path.charCodeAt(path.length - 1));
+    }
+    ts.hasTrailingDirectorySeparator = hasTrailingDirectorySeparator;
+    //// Path Parsing
+    function isVolumeCharacter(charCode) {
+        return (charCode >= 97 /* a */ && charCode <= 122 /* z */) ||
+            (charCode >= 65 /* A */ && charCode <= 90 /* Z */);
+    }
+    function getFileUrlVolumeSeparatorEnd(url, start) {
+        var ch0 = url.charCodeAt(start);
+        if (ch0 === 58 /* colon */)
+            return start + 1;
+        if (ch0 === 37 /* percent */ && url.charCodeAt(start + 1) === 51 /* _3 */) {
+            var ch2 = url.charCodeAt(start + 2);
+            if (ch2 === 97 /* a */ || ch2 === 65 /* A */)
+                return start + 3;
+        }
+        return -1;
+    }
+    /**
+     * Returns length of the root part of a path or URL (i.e. length of "/", "x:/", "//server/share/, file:///user/files").
+     * If the root is part of a URL, the twos-complement of the root length is returned.
+     */
+    function getEncodedRootLength(path) {
+        if (!path)
+            return 0;
+        var ch0 = path.charCodeAt(0);
+        // POSIX or UNC
+        if (ch0 === 47 /* slash */ || ch0 === 92 /* backslash */) {
+            if (path.charCodeAt(1) !== ch0)
+                return 1; // POSIX: "/" (or non-normalized "\")
+            var p1 = path.indexOf(ch0 === 47 /* slash */ ? ts.directorySeparator : altDirectorySeparator, 2);
+            if (p1 < 0)
+                return path.length; // UNC: "//server" or "\\server"
+            return p1 + 1; // UNC: "//server/" or "\\server\"
+        }
+        // DOS
+        if (isVolumeCharacter(ch0) && path.charCodeAt(1) === 58 /* colon */) {
+            var ch2 = path.charCodeAt(2);
+            if (ch2 === 47 /* slash */ || ch2 === 92 /* backslash */)
+                return 3; // DOS: "c:/" or "c:\"
+            if (path.length === 2)
+                return 2; // DOS: "c:" (but not "c:d")
+        }
+        // URL
+        var schemeEnd = path.indexOf(urlSchemeSeparator);
+        if (schemeEnd !== -1) {
+            var authorityStart = schemeEnd + urlSchemeSeparator.length;
+            var authorityEnd = path.indexOf(ts.directorySeparator, authorityStart);
+            if (authorityEnd !== -1) { // URL: "file:///", "file://server/", "file://server/path"
+                // For local "file" URLs, include the leading DOS volume (if present).
+                // Per https://www.ietf.org/rfc/rfc1738.txt, a host of "" or "localhost" is a
+                // special case interpreted as "the machine from which the URL is being interpreted".
+                var scheme = path.slice(0, schemeEnd);
+                var authority = path.slice(authorityStart, authorityEnd);
+                if (scheme === "file" && (authority === "" || authority === "localhost") &&
+                    isVolumeCharacter(path.charCodeAt(authorityEnd + 1))) {
+                    var volumeSeparatorEnd = getFileUrlVolumeSeparatorEnd(path, authorityEnd + 2);
+                    if (volumeSeparatorEnd !== -1) {
+                        if (path.charCodeAt(volumeSeparatorEnd) === 47 /* slash */) {
+                            // URL: "file:///c:/", "file://localhost/c:/", "file:///c%3a/", "file://localhost/c%3a/"
+                            return ~(volumeSeparatorEnd + 1);
+                        }
+                        if (volumeSeparatorEnd === path.length) {
+                            // URL: "file:///c:", "file://localhost/c:", "file:///c$3a", "file://localhost/c%3a"
+                            // but not "file:///c:d" or "file:///c%3ad"
+                            return ~volumeSeparatorEnd;
+                        }
+                    }
+                }
+                return ~(authorityEnd + 1); // URL: "file://server/", "http://server/"
+            }
+            return ~path.length; // URL: "file://server", "http://server"
+        }
+        // relative
+        return 0;
+    }
+    /**
+     * Returns length of the root part of a path or URL (i.e. length of "/", "x:/", "//server/share/, file:///user/files").
+     *
+     * For example:
+     * ```ts
+     * getRootLength("a") === 0                   // ""
+     * getRootLength("/") === 1                   // "/"
+     * getRootLength("c:") === 2                  // "c:"
+     * getRootLength("c:d") === 0                 // ""
+     * getRootLength("c:/") === 3                 // "c:/"
+     * getRootLength("c:\\") === 3                // "c:\\"
+     * getRootLength("//server") === 7            // "//server"
+     * getRootLength("//server/share") === 8      // "//server/"
+     * getRootLength("\\\\server") === 7          // "\\\\server"
+     * getRootLength("\\\\server\\share") === 8   // "\\\\server\\"
+     * getRootLength("file:///path") === 8        // "file:///"
+     * getRootLength("file:///c:") === 10         // "file:///c:"
+     * getRootLength("file:///c:d") === 8         // "file:///"
+     * getRootLength("file:///c:/path") === 11    // "file:///c:/"
+     * getRootLength("file://server") === 13      // "file://server"
+     * getRootLength("file://server/path") === 14 // "file://server/"
+     * getRootLength("http://server") === 13      // "http://server"
+     * getRootLength("http://server/path") === 14 // "http://server/"
+     * ```
+     */
+    function getRootLength(path) {
+        var rootLength = getEncodedRootLength(path);
+        return rootLength < 0 ? ~rootLength : rootLength;
+    }
+    ts.getRootLength = getRootLength;
+    function getDirectoryPath(path) {
+        path = normalizeSlashes(path);
+        // If the path provided is itself the root, then return it.
+        var rootLength = getRootLength(path);
+        if (rootLength === path.length)
+            return path;
+        // return the leading portion of the path up to the last (non-terminal) directory separator
+        // but not including any trailing directory separator.
+        path = removeTrailingDirectorySeparator(path);
+        return path.slice(0, Math.max(rootLength, path.lastIndexOf(ts.directorySeparator)));
+    }
+    ts.getDirectoryPath = getDirectoryPath;
+    function getBaseFileName(path, extensions, ignoreCase) {
+        path = normalizeSlashes(path);
+        // if the path provided is itself the root, then it has not file name.
+        var rootLength = getRootLength(path);
+        if (rootLength === path.length)
+            return "";
+        // return the trailing portion of the path starting after the last (non-terminal) directory
+        // separator but not including any trailing directory separator.
+        path = removeTrailingDirectorySeparator(path);
+        var name = path.slice(Math.max(getRootLength(path), path.lastIndexOf(ts.directorySeparator) + 1));
+        var extension = extensions !== undefined && ignoreCase !== undefined ? getAnyExtensionFromPath(name, extensions, ignoreCase) : undefined;
+        return extension ? name.slice(0, name.length - extension.length) : name;
+    }
+    ts.getBaseFileName = getBaseFileName;
+    function tryGetExtensionFromPath(path, extension, stringEqualityComparer) {
+        if (!ts.startsWith(extension, "."))
+            extension = "." + extension;
+        if (path.length >= extension.length && path.charCodeAt(path.length - extension.length) === 46 /* dot */) {
+            var pathExtension = path.slice(path.length - extension.length);
+            if (stringEqualityComparer(pathExtension, extension)) {
+                return pathExtension;
+            }
+        }
+    }
+    function getAnyExtensionFromPathWorker(path, extensions, stringEqualityComparer) {
+        if (typeof extensions === "string") {
+            return tryGetExtensionFromPath(path, extensions, stringEqualityComparer) || "";
+        }
+        for (var _i = 0, extensions_2 = extensions; _i < extensions_2.length; _i++) {
+            var extension = extensions_2[_i];
+            var result = tryGetExtensionFromPath(path, extension, stringEqualityComparer);
+            if (result)
+                return result;
+        }
+        return "";
+    }
+    function getAnyExtensionFromPath(path, extensions, ignoreCase) {
+        // Retrieves any string from the final "." onwards from a base file name.
+        // Unlike extensionFromPath, which throws an exception on unrecognized extensions.
+        if (extensions) {
+            return getAnyExtensionFromPathWorker(removeTrailingDirectorySeparator(path), extensions, ignoreCase ? ts.equateStringsCaseInsensitive : ts.equateStringsCaseSensitive);
+        }
+        var baseFileName = getBaseFileName(path);
+        var extensionIndex = baseFileName.lastIndexOf(".");
+        if (extensionIndex >= 0) {
+            return baseFileName.substring(extensionIndex);
+        }
+        return "";
+    }
+    ts.getAnyExtensionFromPath = getAnyExtensionFromPath;
+    function pathComponents(path, rootLength) {
+        var root = path.substring(0, rootLength);
+        var rest = path.substring(rootLength).split(ts.directorySeparator);
+        if (rest.length && !ts.lastOrUndefined(rest))
+            rest.pop();
+        return __spreadArrays([root], rest);
+    }
+    /**
+     * Parse a path into an array containing a root component (at index 0) and zero or more path
+     * components (at indices > 0). The result is not normalized.
+     * If the path is relative, the root component is `""`.
+     * If the path is absolute, the root component includes the first path separator (`/`).
+     *
+     * ```ts
+     * // POSIX
+     * getPathComponents("/path/to/file.ext") === ["/", "path", "to", "file.ext"]
+     * getPathComponents("/path/to/") === ["/", "path", "to"]
+     * getPathComponents("/") === ["/"]
+     * // DOS
+     * getPathComponents("c:/path/to/file.ext") === ["c:/", "path", "to", "file.ext"]
+     * getPathComponents("c:/path/to/") === ["c:/", "path", "to"]
+     * getPathComponents("c:/") === ["c:/"]
+     * getPathComponents("c:") === ["c:"]
+     * // URL
+     * getPathComponents("http://typescriptlang.org/path/to/file.ext") === ["http://typescriptlang.org/", "path", "to", "file.ext"]
+     * getPathComponents("http://typescriptlang.org/path/to/") === ["http://typescriptlang.org/", "path", "to"]
+     * getPathComponents("http://typescriptlang.org/") === ["http://typescriptlang.org/"]
+     * getPathComponents("http://typescriptlang.org") === ["http://typescriptlang.org"]
+     * getPathComponents("file://server/path/to/file.ext") === ["file://server/", "path", "to", "file.ext"]
+     * getPathComponents("file://server/path/to/") === ["file://server/", "path", "to"]
+     * getPathComponents("file://server/") === ["file://server/"]
+     * getPathComponents("file://server") === ["file://server"]
+     * getPathComponents("file:///path/to/file.ext") === ["file:///", "path", "to", "file.ext"]
+     * getPathComponents("file:///path/to/") === ["file:///", "path", "to"]
+     * getPathComponents("file:///") === ["file:///"]
+     * getPathComponents("file://") === ["file://"]
+     */
+    function getPathComponents(path, currentDirectory) {
+        if (currentDirectory === void 0) { currentDirectory = ""; }
+        path = combinePaths(currentDirectory, path);
+        return pathComponents(path, getRootLength(path));
+    }
+    ts.getPathComponents = getPathComponents;
+    //// Path Formatting
+    /**
+     * Formats a parsed path consisting of a root component (at index 0) and zero or more path
+     * segments (at indices > 0).
+     *
+     * ```ts
+     * getPathFromPathComponents(["/", "path", "to", "file.ext"]) === "/path/to/file.ext"
+     * ```
+     */
+    function getPathFromPathComponents(pathComponents) {
+        if (pathComponents.length === 0)
+            return "";
+        var root = pathComponents[0] && ensureTrailingDirectorySeparator(pathComponents[0]);
+        return root + pathComponents.slice(1).join(ts.directorySeparator);
+    }
+    ts.getPathFromPathComponents = getPathFromPathComponents;
+    //// Path Normalization
+    /**
+     * Normalize path separators, converting `\` into `/`.
+     */
+    function normalizeSlashes(path) {
+        return path.replace(backslashRegExp, ts.directorySeparator);
+    }
+    ts.normalizeSlashes = normalizeSlashes;
+    /**
+     * Reduce an array of path components to a more simplified path by navigating any
+     * `"."` or `".."` entries in the path.
+     */
+    function reducePathComponents(components) {
+        if (!ts.some(components))
+            return [];
+        var reduced = [components[0]];
+        for (var i = 1; i < components.length; i++) {
+            var component = components[i];
+            if (!component)
+                continue;
+            if (component === ".")
+                continue;
+            if (component === "..") {
+                if (reduced.length > 1) {
+                    if (reduced[reduced.length - 1] !== "..") {
+                        reduced.pop();
+                        continue;
+                    }
+                }
+                else if (reduced[0])
+                    continue;
+            }
+            reduced.push(component);
+        }
+        return reduced;
+    }
+    ts.reducePathComponents = reducePathComponents;
+    /**
+     * Combines paths. If a path is absolute, it replaces any previous path. Relative paths are not simplified.
+     *
+     * ```ts
+     * // Non-rooted
+     * combinePaths("path", "to", "file.ext") === "path/to/file.ext"
+     * combinePaths("path", "dir", "..", "to", "file.ext") === "path/dir/../to/file.ext"
+     * // POSIX
+     * combinePaths("/path", "to", "file.ext") === "/path/to/file.ext"
+     * combinePaths("/path", "/to", "file.ext") === "/to/file.ext"
+     * // DOS
+     * combinePaths("c:/path", "to", "file.ext") === "c:/path/to/file.ext"
+     * combinePaths("c:/path", "c:/to", "file.ext") === "c:/to/file.ext"
+     * // URL
+     * combinePaths("file:///path", "to", "file.ext") === "file:///path/to/file.ext"
+     * combinePaths("file:///path", "file:///to", "file.ext") === "file:///to/file.ext"
+     * ```
+     */
+    function combinePaths(path) {
+        var paths = [];
+        for (var _i = 1; _i < arguments.length; _i++) {
+            paths[_i - 1] = arguments[_i];
+        }
+        if (path)
+            path = normalizeSlashes(path);
+        for (var _a = 0, paths_1 = paths; _a < paths_1.length; _a++) {
+            var relativePath = paths_1[_a];
+            if (!relativePath)
+                continue;
+            relativePath = normalizeSlashes(relativePath);
+            if (!path || getRootLength(relativePath) !== 0) {
+                path = relativePath;
+            }
+            else {
+                path = ensureTrailingDirectorySeparator(path) + relativePath;
+            }
+        }
+        return path;
+    }
+    ts.combinePaths = combinePaths;
+    /**
+     * Combines and resolves paths. If a path is absolute, it replaces any previous path. Any
+     * `.` and `..` path components are resolved. Trailing directory separators are preserved.
+     *
+     * ```ts
+     * resolvePath("/path", "to", "file.ext") === "path/to/file.ext"
+     * resolvePath("/path", "to", "file.ext/") === "path/to/file.ext/"
+     * resolvePath("/path", "dir", "..", "to", "file.ext") === "path/to/file.ext"
+     * ```
+     */
+    function resolvePath(path) {
+        var paths = [];
+        for (var _i = 1; _i < arguments.length; _i++) {
+            paths[_i - 1] = arguments[_i];
+        }
+        return normalizePath(ts.some(paths) ? combinePaths.apply(void 0, __spreadArrays([path], paths)) : normalizeSlashes(path));
+    }
+    ts.resolvePath = resolvePath;
+    /**
+     * Parse a path into an array containing a root component (at index 0) and zero or more path
+     * components (at indices > 0). The result is normalized.
+     * If the path is relative, the root component is `""`.
+     * If the path is absolute, the root component includes the first path separator (`/`).
+     *
+     * ```ts
+     * getNormalizedPathComponents("to/dir/../file.ext", "/path/") === ["/", "path", "to", "file.ext"]
+     * ```
+     */
+    function getNormalizedPathComponents(path, currentDirectory) {
+        return reducePathComponents(getPathComponents(path, currentDirectory));
+    }
+    ts.getNormalizedPathComponents = getNormalizedPathComponents;
+    function getNormalizedAbsolutePath(fileName, currentDirectory) {
+        return getPathFromPathComponents(getNormalizedPathComponents(fileName, currentDirectory));
+    }
+    ts.getNormalizedAbsolutePath = getNormalizedAbsolutePath;
+    function normalizePath(path) {
+        path = normalizeSlashes(path);
+        var normalized = getPathFromPathComponents(reducePathComponents(getPathComponents(path)));
+        return normalized && hasTrailingDirectorySeparator(path) ? ensureTrailingDirectorySeparator(normalized) : normalized;
+    }
+    ts.normalizePath = normalizePath;
+    function getPathWithoutRoot(pathComponents) {
+        if (pathComponents.length === 0)
+            return "";
+        return pathComponents.slice(1).join(ts.directorySeparator);
+    }
+    function getNormalizedAbsolutePathWithoutRoot(fileName, currentDirectory) {
+        return getPathWithoutRoot(getNormalizedPathComponents(fileName, currentDirectory));
+    }
+    ts.getNormalizedAbsolutePathWithoutRoot = getNormalizedAbsolutePathWithoutRoot;
+    function toPath(fileName, basePath, getCanonicalFileName) {
+        var nonCanonicalizedPath = isRootedDiskPath(fileName)
+            ? normalizePath(fileName)
+            : getNormalizedAbsolutePath(fileName, basePath);
+        return getCanonicalFileName(nonCanonicalizedPath);
+    }
+    ts.toPath = toPath;
+    function normalizePathAndParts(path) {
+        path = normalizeSlashes(path);
+        var _a = reducePathComponents(getPathComponents(path)), root = _a[0], parts = _a.slice(1);
+        if (parts.length) {
+            var joinedParts = root + parts.join(ts.directorySeparator);
+            return { path: hasTrailingDirectorySeparator(path) ? ensureTrailingDirectorySeparator(joinedParts) : joinedParts, parts: parts };
+        }
+        else {
+            return { path: root, parts: parts };
+        }
+    }
+    ts.normalizePathAndParts = normalizePathAndParts;
+    function removeTrailingDirectorySeparator(path) {
+        if (hasTrailingDirectorySeparator(path)) {
+            return path.substr(0, path.length - 1);
+        }
+        return path;
+    }
+    ts.removeTrailingDirectorySeparator = removeTrailingDirectorySeparator;
+    function ensureTrailingDirectorySeparator(path) {
+        if (!hasTrailingDirectorySeparator(path)) {
+            return path + ts.directorySeparator;
+        }
+        return path;
+    }
+    ts.ensureTrailingDirectorySeparator = ensureTrailingDirectorySeparator;
+    /**
+     * Ensures a path is either absolute (prefixed with `/` or `c:`) or dot-relative (prefixed
+     * with `./` or `../`) so as not to be confused with an unprefixed module name.
+     *
+     * ```ts
+     * ensurePathIsNonModuleName("/path/to/file.ext") === "/path/to/file.ext"
+     * ensurePathIsNonModuleName("./path/to/file.ext") === "./path/to/file.ext"
+     * ensurePathIsNonModuleName("../path/to/file.ext") === "../path/to/file.ext"
+     * ensurePathIsNonModuleName("path/to/file.ext") === "./path/to/file.ext"
+     * ```
+     */
+    function ensurePathIsNonModuleName(path) {
+        return !pathIsAbsolute(path) && !pathIsRelative(path) ? "./" + path : path;
+    }
+    ts.ensurePathIsNonModuleName = ensurePathIsNonModuleName;
+    function changeAnyExtension(path, ext, extensions, ignoreCase) {
+        var pathext = extensions !== undefined && ignoreCase !== undefined ? getAnyExtensionFromPath(path, extensions, ignoreCase) : getAnyExtensionFromPath(path);
+        return pathext ? path.slice(0, path.length - pathext.length) + (ts.startsWith(ext, ".") ? ext : "." + ext) : path;
+    }
+    ts.changeAnyExtension = changeAnyExtension;
+    //// Path Comparisons
+    // check path for these segments: '', '.'. '..'
+    var relativePathSegmentRegExp = /(^|\/)\.{0,2}($|\/)/;
+    function comparePathsWorker(a, b, componentComparer) {
+        if (a === b)
+            return 0 /* EqualTo */;
+        if (a === undefined)
+            return -1 /* LessThan */;
+        if (b === undefined)
+            return 1 /* GreaterThan */;
+        // NOTE: Performance optimization - shortcut if the root segments differ as there would be no
+        //       need to perform path reduction.
+        var aRoot = a.substring(0, getRootLength(a));
+        var bRoot = b.substring(0, getRootLength(b));
+        var result = ts.compareStringsCaseInsensitive(aRoot, bRoot);
+        if (result !== 0 /* EqualTo */) {
+            return result;
+        }
+        // NOTE: Performance optimization - shortcut if there are no relative path segments in
+        //       the non-root portion of the path
+        var aRest = a.substring(aRoot.length);
+        var bRest = b.substring(bRoot.length);
+        if (!relativePathSegmentRegExp.test(aRest) && !relativePathSegmentRegExp.test(bRest)) {
+            return componentComparer(aRest, bRest);
+        }
+        // The path contains a relative path segment. Normalize the paths and perform a slower component
+        // by component comparison.
+        var aComponents = reducePathComponents(getPathComponents(a));
+        var bComponents = reducePathComponents(getPathComponents(b));
+        var sharedLength = Math.min(aComponents.length, bComponents.length);
+        for (var i = 1; i < sharedLength; i++) {
+            var result_2 = componentComparer(aComponents[i], bComponents[i]);
+            if (result_2 !== 0 /* EqualTo */) {
+                return result_2;
+            }
+        }
+        return ts.compareValues(aComponents.length, bComponents.length);
+    }
+    /**
+     * Performs a case-sensitive comparison of two paths. Path roots are always compared case-insensitively.
+     */
+    function comparePathsCaseSensitive(a, b) {
+        return comparePathsWorker(a, b, ts.compareStringsCaseSensitive);
+    }
+    ts.comparePathsCaseSensitive = comparePathsCaseSensitive;
+    /**
+     * Performs a case-insensitive comparison of two paths.
+     */
+    function comparePathsCaseInsensitive(a, b) {
+        return comparePathsWorker(a, b, ts.compareStringsCaseInsensitive);
+    }
+    ts.comparePathsCaseInsensitive = comparePathsCaseInsensitive;
+    function comparePaths(a, b, currentDirectory, ignoreCase) {
+        if (typeof currentDirectory === "string") {
+            a = combinePaths(currentDirectory, a);
+            b = combinePaths(currentDirectory, b);
+        }
+        else if (typeof currentDirectory === "boolean") {
+            ignoreCase = currentDirectory;
+        }
+        return comparePathsWorker(a, b, ts.getStringComparer(ignoreCase));
+    }
+    ts.comparePaths = comparePaths;
+    function containsPath(parent, child, currentDirectory, ignoreCase) {
+        if (typeof currentDirectory === "string") {
+            parent = combinePaths(currentDirectory, parent);
+            child = combinePaths(currentDirectory, child);
+        }
+        else if (typeof currentDirectory === "boolean") {
+            ignoreCase = currentDirectory;
+        }
+        if (parent === undefined || child === undefined)
+            return false;
+        if (parent === child)
+            return true;
+        var parentComponents = reducePathComponents(getPathComponents(parent));
+        var childComponents = reducePathComponents(getPathComponents(child));
+        if (childComponents.length < parentComponents.length) {
+            return false;
+        }
+        var componentEqualityComparer = ignoreCase ? ts.equateStringsCaseInsensitive : ts.equateStringsCaseSensitive;
+        for (var i = 0; i < parentComponents.length; i++) {
+            var equalityComparer = i === 0 ? ts.equateStringsCaseInsensitive : componentEqualityComparer;
+            if (!equalityComparer(parentComponents[i], childComponents[i])) {
+                return false;
+            }
+        }
+        return true;
+    }
+    ts.containsPath = containsPath;
+    /**
+     * Determines whether `fileName` starts with the specified `directoryName` using the provided path canonicalization callback.
+     * Comparison is case-sensitive between the canonical paths.
+     *
+     * @deprecated Use `containsPath` if possible.
+     */
+    function startsWithDirectory(fileName, directoryName, getCanonicalFileName) {
+        var canonicalFileName = getCanonicalFileName(fileName);
+        var canonicalDirectoryName = getCanonicalFileName(directoryName);
+        return ts.startsWith(canonicalFileName, canonicalDirectoryName + "/") || ts.startsWith(canonicalFileName, canonicalDirectoryName + "\\");
+    }
+    ts.startsWithDirectory = startsWithDirectory;
+    //// Relative Paths
+    function getPathComponentsRelativeTo(from, to, stringEqualityComparer, getCanonicalFileName) {
+        var fromComponents = reducePathComponents(getPathComponents(from));
+        var toComponents = reducePathComponents(getPathComponents(to));
+        var start;
+        for (start = 0; start < fromComponents.length && start < toComponents.length; start++) {
+            var fromComponent = getCanonicalFileName(fromComponents[start]);
+            var toComponent = getCanonicalFileName(toComponents[start]);
+            var comparer = start === 0 ? ts.equateStringsCaseInsensitive : stringEqualityComparer;
+            if (!comparer(fromComponent, toComponent))
+                break;
+        }
+        if (start === 0) {
+            return toComponents;
+        }
+        var components = toComponents.slice(start);
+        var relative = [];
+        for (; start < fromComponents.length; start++) {
+            relative.push("..");
+        }
+        return __spreadArrays([""], relative, components);
+    }
+    ts.getPathComponentsRelativeTo = getPathComponentsRelativeTo;
+    function getRelativePathFromDirectory(fromDirectory, to, getCanonicalFileNameOrIgnoreCase) {
+        ts.Debug.assert((getRootLength(fromDirectory) > 0) === (getRootLength(to) > 0), "Paths must either both be absolute or both be relative");
+        var getCanonicalFileName = typeof getCanonicalFileNameOrIgnoreCase === "function" ? getCanonicalFileNameOrIgnoreCase : ts.identity;
+        var ignoreCase = typeof getCanonicalFileNameOrIgnoreCase === "boolean" ? getCanonicalFileNameOrIgnoreCase : false;
+        var pathComponents = getPathComponentsRelativeTo(fromDirectory, to, ignoreCase ? ts.equateStringsCaseInsensitive : ts.equateStringsCaseSensitive, getCanonicalFileName);
+        return getPathFromPathComponents(pathComponents);
+    }
+    ts.getRelativePathFromDirectory = getRelativePathFromDirectory;
+    function convertToRelativePath(absoluteOrRelativePath, basePath, getCanonicalFileName) {
+        return !isRootedDiskPath(absoluteOrRelativePath)
+            ? absoluteOrRelativePath
+            : getRelativePathToDirectoryOrUrl(basePath, absoluteOrRelativePath, basePath, getCanonicalFileName, /*isAbsolutePathAnUrl*/ false);
+    }
+    ts.convertToRelativePath = convertToRelativePath;
+    function getRelativePathFromFile(from, to, getCanonicalFileName) {
+        return ensurePathIsNonModuleName(getRelativePathFromDirectory(getDirectoryPath(from), to, getCanonicalFileName));
+    }
+    ts.getRelativePathFromFile = getRelativePathFromFile;
+    function getRelativePathToDirectoryOrUrl(directoryPathOrUrl, relativeOrAbsolutePath, currentDirectory, getCanonicalFileName, isAbsolutePathAnUrl) {
+        var pathComponents = getPathComponentsRelativeTo(resolvePath(currentDirectory, directoryPathOrUrl), resolvePath(currentDirectory, relativeOrAbsolutePath), ts.equateStringsCaseSensitive, getCanonicalFileName);
+        var firstComponent = pathComponents[0];
+        if (isAbsolutePathAnUrl && isRootedDiskPath(firstComponent)) {
+            var prefix = firstComponent.charAt(0) === ts.directorySeparator ? "file://" : "file:///";
+            pathComponents[0] = prefix + firstComponent;
+        }
+        return getPathFromPathComponents(pathComponents);
+    }
+    ts.getRelativePathToDirectoryOrUrl = getRelativePathToDirectoryOrUrl;
+    function forEachAncestorDirectory(directory, callback) {
+        while (true) {
+            var result = callback(directory);
+            if (result !== undefined) {
+                return result;
+            }
+            var parentPath = getDirectoryPath(directory);
+            if (parentPath === directory) {
+                return undefined;
+            }
+            directory = parentPath;
+        }
+    }
+    ts.forEachAncestorDirectory = forEachAncestorDirectory;
+    function isNodeModulesDirectory(dirPath) {
+        return ts.endsWith(dirPath, "/node_modules");
+    }
+    ts.isNodeModulesDirectory = isNodeModulesDirectory;
+})(ts || (ts = {}));
 var ts;
 (function (ts) {
     /**
@@ -5068,7 +6603,7 @@ var ts;
     function createUseFsEventsOnParentDirectoryWatchFile(fsWatch, useCaseSensitiveFileNames) {
         // One file can have multiple watchers
         var fileWatcherCallbacks = ts.createMultiMap();
-        var dirWatchers = ts.createMap();
+        var dirWatchers = new ts.Map();
         var toCanonicalName = ts.createGetCanonicalFileName(useCaseSensitiveFileNames);
         return nonPollingWatchFile;
         function nonPollingWatchFile(fileName, callback, _pollingInterval, fallbackOptions) {
@@ -5115,7 +6650,7 @@ var ts;
     }
     /* @internal */
     function createSingleFileWatcherPerName(watchFile, useCaseSensitiveFileNames) {
-        var cache = ts.createMap();
+        var cache = new ts.Map();
         var callbacksCache = ts.createMultiMap();
         var toCanonicalFileName = ts.createGetCanonicalFileName(useCaseSensitiveFileNames);
         return function (fileName, callback, pollingInterval, options) {
@@ -5185,9 +6720,9 @@ var ts;
      */
     /*@internal*/
     function createDirectoryWatcherSupportingRecursive(host) {
-        var cache = ts.createMap();
+        var cache = new ts.Map();
         var callbackCache = ts.createMultiMap();
-        var cacheToUpdateChildWatches = ts.createMap();
+        var cacheToUpdateChildWatches = new ts.Map();
         var timerToUpdateChildWatches;
         var filePathComparer = ts.getStringComparer(!host.useCaseSensitiveFileNames);
         var toCanonicalFilePath = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames);
@@ -5243,7 +6778,7 @@ var ts;
                 }
             };
         }
-        function invokeCallbacks(dirPath, fileNameOrInvokeMap) {
+        function invokeCallbacks(dirPath, fileNameOrInvokeMap, fileNames) {
             var fileName;
             var invokeMap;
             if (ts.isString(fileNameOrInvokeMap)) {
@@ -5254,11 +6789,23 @@ var ts;
             }
             // Call the actual callback
             callbackCache.forEach(function (callbacks, rootDirName) {
-                if (invokeMap && invokeMap.has(rootDirName))
+                var _a;
+                if (invokeMap && invokeMap.get(rootDirName) === true)
                     return;
                 if (rootDirName === dirPath || (ts.startsWith(dirPath, rootDirName) && dirPath[rootDirName.length] === ts.directorySeparator)) {
                     if (invokeMap) {
-                        invokeMap.set(rootDirName, true);
+                        if (fileNames) {
+                            var existing = invokeMap.get(rootDirName);
+                            if (existing) {
+                                (_a = existing).push.apply(_a, fileNames);
+                            }
+                            else {
+                                invokeMap.set(rootDirName, fileNames.slice());
+                            }
+                        }
+                        else {
+                            invokeMap.set(rootDirName, true);
+                        }
                     }
                     else {
                         callbacks.forEach(function (_a) {
@@ -5274,16 +6821,20 @@ var ts;
             var parentWatcher = cache.get(dirPath);
             if (parentWatcher && host.directoryExists(dirName)) {
                 // Schedule the update and postpone invoke for callbacks
-                scheduleUpdateChildWatches(dirName, dirPath, options);
+                scheduleUpdateChildWatches(dirName, dirPath, fileName, options);
                 return;
             }
             // Call the actual callbacks and remove child watches
             invokeCallbacks(dirPath, fileName);
             removeChildWatches(parentWatcher);
         }
-        function scheduleUpdateChildWatches(dirName, dirPath, options) {
-            if (!cacheToUpdateChildWatches.has(dirPath)) {
-                cacheToUpdateChildWatches.set(dirPath, { dirName: dirName, options: options });
+        function scheduleUpdateChildWatches(dirName, dirPath, fileName, options) {
+            var existing = cacheToUpdateChildWatches.get(dirPath);
+            if (existing) {
+                existing.fileNames.push(fileName);
+            }
+            else {
+                cacheToUpdateChildWatches.set(dirPath, { dirName: dirName, options: options, fileNames: [fileName] });
             }
             if (timerToUpdateChildWatches) {
                 host.clearTimeout(timerToUpdateChildWatches);
@@ -5295,22 +6846,28 @@ var ts;
             timerToUpdateChildWatches = undefined;
             ts.sysLog("sysLog:: onTimerToUpdateChildWatches:: " + cacheToUpdateChildWatches.size);
             var start = ts.timestamp();
-            var invokeMap = ts.createMap();
+            var invokeMap = new ts.Map();
             while (!timerToUpdateChildWatches && cacheToUpdateChildWatches.size) {
-                var _a = cacheToUpdateChildWatches.entries().next(), _b = _a.value, dirPath = _b[0], _c = _b[1], dirName = _c.dirName, options = _c.options, done = _a.done;
+                var _a = cacheToUpdateChildWatches.entries().next(), _b = _a.value, dirPath = _b[0], _c = _b[1], dirName = _c.dirName, options = _c.options, fileNames = _c.fileNames, done = _a.done;
                 ts.Debug.assert(!done);
                 cacheToUpdateChildWatches.delete(dirPath);
                 // Because the child refresh is fresh, we would need to invalidate whole root directory being watched
                 // to ensure that all the changes are reflected at this time
-                invokeCallbacks(dirPath, invokeMap);
-                updateChildWatches(dirName, dirPath, options);
+                var hasChanges = updateChildWatches(dirName, dirPath, options);
+                invokeCallbacks(dirPath, invokeMap, hasChanges ? undefined : fileNames);
             }
             ts.sysLog("sysLog:: invokingWatchers:: " + (ts.timestamp() - start) + "ms:: " + cacheToUpdateChildWatches.size);
             callbackCache.forEach(function (callbacks, rootDirName) {
-                if (invokeMap.has(rootDirName)) {
+                var existing = invokeMap.get(rootDirName);
+                if (existing) {
                     callbacks.forEach(function (_a) {
                         var callback = _a.callback, dirName = _a.dirName;
-                        return callback(dirName);
+                        if (ts.isArray(existing)) {
+                            existing.forEach(callback);
+                        }
+                        else {
+                            callback(dirName);
+                        }
                     });
                 }
             });
@@ -5328,25 +6885,20 @@ var ts;
                 removeChildWatches(cache.get(toCanonicalFilePath(childWatcher.dirName)));
             }
         }
-        function updateChildWatches(dirName, dirPath, options) {
+        function updateChildWatches(parentDir, parentDirPath, options) {
             // Iterate through existing children and update the watches if needed
-            var parentWatcher = cache.get(dirPath);
-            if (parentWatcher) {
-                parentWatcher.childWatches = watchChildDirectories(dirName, parentWatcher.childWatches, options);
-            }
-        }
-        /**
-         * Watch the directories in the parentDir
-         */
-        function watchChildDirectories(parentDir, existingChildWatches, options) {
+            var parentWatcher = cache.get(parentDirPath);
+            if (!parentWatcher)
+                return false;
             var newChildWatches;
-            ts.enumerateInsertsAndDeletes(host.directoryExists(parentDir) ? ts.mapDefined(host.getAccessibleSortedChildDirectories(parentDir), function (child) {
+            var hasChanges = ts.enumerateInsertsAndDeletes(host.directoryExists(parentDir) ? ts.mapDefined(host.getAccessibleSortedChildDirectories(parentDir), function (child) {
                 var childFullName = ts.getNormalizedAbsolutePath(child, parentDir);
                 // Filter our the symbolic link directories since those arent included in recursive watch
                 // which is same behaviour when recursive: true is passed to fs.watch
                 return !isIgnoredPath(childFullName) && filePathComparer(childFullName, ts.normalizePath(host.realpath(childFullName))) === 0 /* EqualTo */ ? childFullName : undefined;
-            }) : ts.emptyArray, existingChildWatches, function (child, childWatcher) { return filePathComparer(child, childWatcher.dirName); }, createAndAddChildDirectoryWatcher, ts.closeFileWatcher, addChildDirectoryWatcher);
-            return newChildWatches || ts.emptyArray;
+            }) : ts.emptyArray, parentWatcher.childWatches, function (child, childWatcher) { return filePathComparer(child, childWatcher.dirName); }, createAndAddChildDirectoryWatcher, ts.closeFileWatcher, addChildDirectoryWatcher);
+            parentWatcher.childWatches = newChildWatches || ts.emptyArray;
+            return hasChanges;
             /**
              * Create new childDirectoryWatcher and add it to the new ChildDirectoryWatcher list
              */
@@ -5678,7 +7230,7 @@ var ts;
                 enableCPUProfiler: enableCPUProfiler,
                 disableCPUProfiler: disableCPUProfiler,
                 realpath: realpath,
-                debugMode: !!process.env.NODE_INSPECTOR_IPC || ts.some(process.execArgv, function (arg) { return /^--(inspect|debug)(-brk)?(=\d+)?$/i.test(arg); }),
+                debugMode: !!process.env.NODE_INSPECTOR_IPC || !!process.env.VSCODE_INSPECTOR_OPTIONS || ts.some(process.execArgv, function (arg) { return /^--(inspect|debug)(-brk)?(=\d+)?$/i.test(arg); }),
                 tryEnableSourceMapsForHost: function () {
                     try {
                         require("source-map-support").install();
@@ -5742,7 +7294,7 @@ var ts;
              */
             function cleanupPaths(profile) {
                 var externalFileCounter = 0;
-                var remappedPaths = ts.createMap();
+                var remappedPaths = new ts.Map();
                 var normalizedDir = ts.normalizeSlashes(__dirname);
                 // Windows rooted dir names need an extra `/` prepended to be valid file:/// urls
                 var fileUrlRoot = "file://" + (ts.getRootLength(normalizedDir) === 1 ? "" : "/") + normalizedDir;
@@ -6004,8 +7556,8 @@ var ts;
                     var entries = _fs.readdirSync(path || ".", { withFileTypes: true });
                     var files = [];
                     var directories = [];
-                    for (var _i = 0, entries_2 = entries; _i < entries_2.length; _i++) {
-                        var dirent = entries_2[_i];
+                    for (var _i = 0, entries_1 = entries; _i < entries_1.length; _i++) {
+                        var dirent = entries_1[_i];
                         // withFileTypes is not supported before Node 10.10.
                         var entry = typeof dirent === "string" ? dirent : dirent.name;
                         // This is necessary because on some file system node fails to exclude
@@ -6045,6 +7597,10 @@ var ts;
                 return ts.matchFiles(path, extensions, excludes, includes, useCaseSensitiveFileNames, process.cwd(), depth, getAccessibleFileSystemEntries, realpath);
             }
             function fileSystemEntryExists(path, entryKind) {
+                // Since the error thrown by fs.statSync isn't used, we can avoid collecting a stack trace to improve
+                // the CPU time performance.
+                var originalStackTraceLimit = Error.stackTraceLimit;
+                Error.stackTraceLimit = 0;
                 try {
                     var stat = _fs.statSync(path);
                     switch (entryKind) {
@@ -6056,6 +7612,9 @@ var ts;
                 catch (e) {
                     return false;
                 }
+                finally {
+                    Error.stackTraceLimit = originalStackTraceLimit;
+                }
             }
             function fileExists(path) {
                 return fileSystemEntryExists(path, 0 /* File */);
@@ -6126,685 +7685,13 @@ var ts;
         ts.Debug.isDebugging = true;
     }
 })(ts || (ts = {}));
-/* @internal */
-var ts;
-(function (ts) {
-    /**
-     * Internally, we represent paths as strings with '/' as the directory separator.
-     * When we make system calls (eg: LanguageServiceHost.getDirectory()),
-     * we expect the host to correctly handle paths in our specified format.
-     */
-    ts.directorySeparator = "/";
-    var altDirectorySeparator = "\\";
-    var urlSchemeSeparator = "://";
-    var backslashRegExp = /\\/g;
-    //// Path Tests
-    /**
-     * Determines whether a charCode corresponds to `/` or `\`.
-     */
-    function isAnyDirectorySeparator(charCode) {
-        return charCode === 47 /* slash */ || charCode === 92 /* backslash */;
-    }
-    ts.isAnyDirectorySeparator = isAnyDirectorySeparator;
-    /**
-     * Determines whether a path starts with a URL scheme (e.g. starts with `http://`, `ftp://`, `file://`, etc.).
-     */
-    function isUrl(path) {
-        return getEncodedRootLength(path) < 0;
-    }
-    ts.isUrl = isUrl;
-    /**
-     * Determines whether a path is an absolute disk path (e.g. starts with `/`, or a dos path
-     * like `c:`, `c:\` or `c:/`).
-     */
-    function isRootedDiskPath(path) {
-        return getEncodedRootLength(path) > 0;
-    }
-    ts.isRootedDiskPath = isRootedDiskPath;
-    /**
-     * Determines whether a path consists only of a path root.
-     */
-    function isDiskPathRoot(path) {
-        var rootLength = getEncodedRootLength(path);
-        return rootLength > 0 && rootLength === path.length;
-    }
-    ts.isDiskPathRoot = isDiskPathRoot;
-    /**
-     * Determines whether a path starts with an absolute path component (i.e. `/`, `c:/`, `file://`, etc.).
-     *
-     * ```ts
-     * // POSIX
-     * pathIsAbsolute("/path/to/file.ext") === true
-     * // DOS
-     * pathIsAbsolute("c:/path/to/file.ext") === true
-     * // URL
-     * pathIsAbsolute("file:///path/to/file.ext") === true
-     * // Non-absolute
-     * pathIsAbsolute("path/to/file.ext") === false
-     * pathIsAbsolute("./path/to/file.ext") === false
-     * ```
-     */
-    function pathIsAbsolute(path) {
-        return getEncodedRootLength(path) !== 0;
-    }
-    ts.pathIsAbsolute = pathIsAbsolute;
-    /**
-     * Determines whether a path starts with a relative path component (i.e. `.` or `..`).
-     */
-    function pathIsRelative(path) {
-        return /^\.\.?($|[\\/])/.test(path);
-    }
-    ts.pathIsRelative = pathIsRelative;
-    function hasExtension(fileName) {
-        return ts.stringContains(getBaseFileName(fileName), ".");
-    }
-    ts.hasExtension = hasExtension;
-    function fileExtensionIs(path, extension) {
-        return path.length > extension.length && ts.endsWith(path, extension);
-    }
-    ts.fileExtensionIs = fileExtensionIs;
-    function fileExtensionIsOneOf(path, extensions) {
-        for (var _i = 0, extensions_1 = extensions; _i < extensions_1.length; _i++) {
-            var extension = extensions_1[_i];
-            if (fileExtensionIs(path, extension)) {
-                return true;
-            }
-        }
-        return false;
-    }
-    ts.fileExtensionIsOneOf = fileExtensionIsOneOf;
-    /**
-     * Determines whether a path has a trailing separator (`/` or `\\`).
-     */
-    function hasTrailingDirectorySeparator(path) {
-        return path.length > 0 && isAnyDirectorySeparator(path.charCodeAt(path.length - 1));
-    }
-    ts.hasTrailingDirectorySeparator = hasTrailingDirectorySeparator;
-    //// Path Parsing
-    function isVolumeCharacter(charCode) {
-        return (charCode >= 97 /* a */ && charCode <= 122 /* z */) ||
-            (charCode >= 65 /* A */ && charCode <= 90 /* Z */);
-    }
-    function getFileUrlVolumeSeparatorEnd(url, start) {
-        var ch0 = url.charCodeAt(start);
-        if (ch0 === 58 /* colon */)
-            return start + 1;
-        if (ch0 === 37 /* percent */ && url.charCodeAt(start + 1) === 51 /* _3 */) {
-            var ch2 = url.charCodeAt(start + 2);
-            if (ch2 === 97 /* a */ || ch2 === 65 /* A */)
-                return start + 3;
-        }
-        return -1;
-    }
-    /**
-     * Returns length of the root part of a path or URL (i.e. length of "/", "x:/", "//server/share/, file:///user/files").
-     * If the root is part of a URL, the twos-complement of the root length is returned.
-     */
-    function getEncodedRootLength(path) {
-        if (!path)
-            return 0;
-        var ch0 = path.charCodeAt(0);
-        // POSIX or UNC
-        if (ch0 === 47 /* slash */ || ch0 === 92 /* backslash */) {
-            if (path.charCodeAt(1) !== ch0)
-                return 1; // POSIX: "/" (or non-normalized "\")
-            var p1 = path.indexOf(ch0 === 47 /* slash */ ? ts.directorySeparator : altDirectorySeparator, 2);
-            if (p1 < 0)
-                return path.length; // UNC: "//server" or "\\server"
-            return p1 + 1; // UNC: "//server/" or "\\server\"
-        }
-        // DOS
-        if (isVolumeCharacter(ch0) && path.charCodeAt(1) === 58 /* colon */) {
-            var ch2 = path.charCodeAt(2);
-            if (ch2 === 47 /* slash */ || ch2 === 92 /* backslash */)
-                return 3; // DOS: "c:/" or "c:\"
-            if (path.length === 2)
-                return 2; // DOS: "c:" (but not "c:d")
-        }
-        // URL
-        var schemeEnd = path.indexOf(urlSchemeSeparator);
-        if (schemeEnd !== -1) {
-            var authorityStart = schemeEnd + urlSchemeSeparator.length;
-            var authorityEnd = path.indexOf(ts.directorySeparator, authorityStart);
-            if (authorityEnd !== -1) { // URL: "file:///", "file://server/", "file://server/path"
-                // For local "file" URLs, include the leading DOS volume (if present).
-                // Per https://www.ietf.org/rfc/rfc1738.txt, a host of "" or "localhost" is a
-                // special case interpreted as "the machine from which the URL is being interpreted".
-                var scheme = path.slice(0, schemeEnd);
-                var authority = path.slice(authorityStart, authorityEnd);
-                if (scheme === "file" && (authority === "" || authority === "localhost") &&
-                    isVolumeCharacter(path.charCodeAt(authorityEnd + 1))) {
-                    var volumeSeparatorEnd = getFileUrlVolumeSeparatorEnd(path, authorityEnd + 2);
-                    if (volumeSeparatorEnd !== -1) {
-                        if (path.charCodeAt(volumeSeparatorEnd) === 47 /* slash */) {
-                            // URL: "file:///c:/", "file://localhost/c:/", "file:///c%3a/", "file://localhost/c%3a/"
-                            return ~(volumeSeparatorEnd + 1);
-                        }
-                        if (volumeSeparatorEnd === path.length) {
-                            // URL: "file:///c:", "file://localhost/c:", "file:///c$3a", "file://localhost/c%3a"
-                            // but not "file:///c:d" or "file:///c%3ad"
-                            return ~volumeSeparatorEnd;
-                        }
-                    }
-                }
-                return ~(authorityEnd + 1); // URL: "file://server/", "http://server/"
-            }
-            return ~path.length; // URL: "file://server", "http://server"
-        }
-        // relative
-        return 0;
-    }
-    /**
-     * Returns length of the root part of a path or URL (i.e. length of "/", "x:/", "//server/share/, file:///user/files").
-     *
-     * For example:
-     * ```ts
-     * getRootLength("a") === 0                   // ""
-     * getRootLength("/") === 1                   // "/"
-     * getRootLength("c:") === 2                  // "c:"
-     * getRootLength("c:d") === 0                 // ""
-     * getRootLength("c:/") === 3                 // "c:/"
-     * getRootLength("c:\\") === 3                // "c:\\"
-     * getRootLength("//server") === 7            // "//server"
-     * getRootLength("//server/share") === 8      // "//server/"
-     * getRootLength("\\\\server") === 7          // "\\\\server"
-     * getRootLength("\\\\server\\share") === 8   // "\\\\server\\"
-     * getRootLength("file:///path") === 8        // "file:///"
-     * getRootLength("file:///c:") === 10         // "file:///c:"
-     * getRootLength("file:///c:d") === 8         // "file:///"
-     * getRootLength("file:///c:/path") === 11    // "file:///c:/"
-     * getRootLength("file://server") === 13      // "file://server"
-     * getRootLength("file://server/path") === 14 // "file://server/"
-     * getRootLength("http://server") === 13      // "http://server"
-     * getRootLength("http://server/path") === 14 // "http://server/"
-     * ```
-     */
-    function getRootLength(path) {
-        var rootLength = getEncodedRootLength(path);
-        return rootLength < 0 ? ~rootLength : rootLength;
-    }
-    ts.getRootLength = getRootLength;
-    function getDirectoryPath(path) {
-        path = normalizeSlashes(path);
-        // If the path provided is itself the root, then return it.
-        var rootLength = getRootLength(path);
-        if (rootLength === path.length)
-            return path;
-        // return the leading portion of the path up to the last (non-terminal) directory separator
-        // but not including any trailing directory separator.
-        path = removeTrailingDirectorySeparator(path);
-        return path.slice(0, Math.max(rootLength, path.lastIndexOf(ts.directorySeparator)));
-    }
-    ts.getDirectoryPath = getDirectoryPath;
-    function getBaseFileName(path, extensions, ignoreCase) {
-        path = normalizeSlashes(path);
-        // if the path provided is itself the root, then it has not file name.
-        var rootLength = getRootLength(path);
-        if (rootLength === path.length)
-            return "";
-        // return the trailing portion of the path starting after the last (non-terminal) directory
-        // separator but not including any trailing directory separator.
-        path = removeTrailingDirectorySeparator(path);
-        var name = path.slice(Math.max(getRootLength(path), path.lastIndexOf(ts.directorySeparator) + 1));
-        var extension = extensions !== undefined && ignoreCase !== undefined ? getAnyExtensionFromPath(name, extensions, ignoreCase) : undefined;
-        return extension ? name.slice(0, name.length - extension.length) : name;
-    }
-    ts.getBaseFileName = getBaseFileName;
-    function tryGetExtensionFromPath(path, extension, stringEqualityComparer) {
-        if (!ts.startsWith(extension, "."))
-            extension = "." + extension;
-        if (path.length >= extension.length && path.charCodeAt(path.length - extension.length) === 46 /* dot */) {
-            var pathExtension = path.slice(path.length - extension.length);
-            if (stringEqualityComparer(pathExtension, extension)) {
-                return pathExtension;
-            }
-        }
-    }
-    function getAnyExtensionFromPathWorker(path, extensions, stringEqualityComparer) {
-        if (typeof extensions === "string") {
-            return tryGetExtensionFromPath(path, extensions, stringEqualityComparer) || "";
-        }
-        for (var _i = 0, extensions_2 = extensions; _i < extensions_2.length; _i++) {
-            var extension = extensions_2[_i];
-            var result = tryGetExtensionFromPath(path, extension, stringEqualityComparer);
-            if (result)
-                return result;
-        }
-        return "";
-    }
-    function getAnyExtensionFromPath(path, extensions, ignoreCase) {
-        // Retrieves any string from the final "." onwards from a base file name.
-        // Unlike extensionFromPath, which throws an exception on unrecognized extensions.
-        if (extensions) {
-            return getAnyExtensionFromPathWorker(removeTrailingDirectorySeparator(path), extensions, ignoreCase ? ts.equateStringsCaseInsensitive : ts.equateStringsCaseSensitive);
-        }
-        var baseFileName = getBaseFileName(path);
-        var extensionIndex = baseFileName.lastIndexOf(".");
-        if (extensionIndex >= 0) {
-            return baseFileName.substring(extensionIndex);
-        }
-        return "";
-    }
-    ts.getAnyExtensionFromPath = getAnyExtensionFromPath;
-    function pathComponents(path, rootLength) {
-        var root = path.substring(0, rootLength);
-        var rest = path.substring(rootLength).split(ts.directorySeparator);
-        if (rest.length && !ts.lastOrUndefined(rest))
-            rest.pop();
-        return __spreadArrays([root], rest);
-    }
-    /**
-     * Parse a path into an array containing a root component (at index 0) and zero or more path
-     * components (at indices > 0). The result is not normalized.
-     * If the path is relative, the root component is `""`.
-     * If the path is absolute, the root component includes the first path separator (`/`).
-     *
-     * ```ts
-     * // POSIX
-     * getPathComponents("/path/to/file.ext") === ["/", "path", "to", "file.ext"]
-     * getPathComponents("/path/to/") === ["/", "path", "to"]
-     * getPathComponents("/") === ["/"]
-     * // DOS
-     * getPathComponents("c:/path/to/file.ext") === ["c:/", "path", "to", "file.ext"]
-     * getPathComponents("c:/path/to/") === ["c:/", "path", "to"]
-     * getPathComponents("c:/") === ["c:/"]
-     * getPathComponents("c:") === ["c:"]
-     * // URL
-     * getPathComponents("http://typescriptlang.org/path/to/file.ext") === ["http://typescriptlang.org/", "path", "to", "file.ext"]
-     * getPathComponents("http://typescriptlang.org/path/to/") === ["http://typescriptlang.org/", "path", "to"]
-     * getPathComponents("http://typescriptlang.org/") === ["http://typescriptlang.org/"]
-     * getPathComponents("http://typescriptlang.org") === ["http://typescriptlang.org"]
-     * getPathComponents("file://server/path/to/file.ext") === ["file://server/", "path", "to", "file.ext"]
-     * getPathComponents("file://server/path/to/") === ["file://server/", "path", "to"]
-     * getPathComponents("file://server/") === ["file://server/"]
-     * getPathComponents("file://server") === ["file://server"]
-     * getPathComponents("file:///path/to/file.ext") === ["file:///", "path", "to", "file.ext"]
-     * getPathComponents("file:///path/to/") === ["file:///", "path", "to"]
-     * getPathComponents("file:///") === ["file:///"]
-     * getPathComponents("file://") === ["file://"]
-     */
-    function getPathComponents(path, currentDirectory) {
-        if (currentDirectory === void 0) { currentDirectory = ""; }
-        path = combinePaths(currentDirectory, path);
-        return pathComponents(path, getRootLength(path));
-    }
-    ts.getPathComponents = getPathComponents;
-    //// Path Formatting
-    /**
-     * Formats a parsed path consisting of a root component (at index 0) and zero or more path
-     * segments (at indices > 0).
-     *
-     * ```ts
-     * getPathFromPathComponents(["/", "path", "to", "file.ext"]) === "/path/to/file.ext"
-     * ```
-     */
-    function getPathFromPathComponents(pathComponents) {
-        if (pathComponents.length === 0)
-            return "";
-        var root = pathComponents[0] && ensureTrailingDirectorySeparator(pathComponents[0]);
-        return root + pathComponents.slice(1).join(ts.directorySeparator);
-    }
-    ts.getPathFromPathComponents = getPathFromPathComponents;
-    //// Path Normalization
-    /**
-     * Normalize path separators, converting `\` into `/`.
-     */
-    function normalizeSlashes(path) {
-        return path.replace(backslashRegExp, ts.directorySeparator);
-    }
-    ts.normalizeSlashes = normalizeSlashes;
-    /**
-     * Reduce an array of path components to a more simplified path by navigating any
-     * `"."` or `".."` entries in the path.
-     */
-    function reducePathComponents(components) {
-        if (!ts.some(components))
-            return [];
-        var reduced = [components[0]];
-        for (var i = 1; i < components.length; i++) {
-            var component = components[i];
-            if (!component)
-                continue;
-            if (component === ".")
-                continue;
-            if (component === "..") {
-                if (reduced.length > 1) {
-                    if (reduced[reduced.length - 1] !== "..") {
-                        reduced.pop();
-                        continue;
-                    }
-                }
-                else if (reduced[0])
-                    continue;
-            }
-            reduced.push(component);
-        }
-        return reduced;
-    }
-    ts.reducePathComponents = reducePathComponents;
-    /**
-     * Combines paths. If a path is absolute, it replaces any previous path. Relative paths are not simplified.
-     *
-     * ```ts
-     * // Non-rooted
-     * combinePaths("path", "to", "file.ext") === "path/to/file.ext"
-     * combinePaths("path", "dir", "..", "to", "file.ext") === "path/dir/../to/file.ext"
-     * // POSIX
-     * combinePaths("/path", "to", "file.ext") === "/path/to/file.ext"
-     * combinePaths("/path", "/to", "file.ext") === "/to/file.ext"
-     * // DOS
-     * combinePaths("c:/path", "to", "file.ext") === "c:/path/to/file.ext"
-     * combinePaths("c:/path", "c:/to", "file.ext") === "c:/to/file.ext"
-     * // URL
-     * combinePaths("file:///path", "to", "file.ext") === "file:///path/to/file.ext"
-     * combinePaths("file:///path", "file:///to", "file.ext") === "file:///to/file.ext"
-     * ```
-     */
-    function combinePaths(path) {
-        var paths = [];
-        for (var _i = 1; _i < arguments.length; _i++) {
-            paths[_i - 1] = arguments[_i];
-        }
-        if (path)
-            path = normalizeSlashes(path);
-        for (var _a = 0, paths_1 = paths; _a < paths_1.length; _a++) {
-            var relativePath = paths_1[_a];
-            if (!relativePath)
-                continue;
-            relativePath = normalizeSlashes(relativePath);
-            if (!path || getRootLength(relativePath) !== 0) {
-                path = relativePath;
-            }
-            else {
-                path = ensureTrailingDirectorySeparator(path) + relativePath;
-            }
-        }
-        return path;
-    }
-    ts.combinePaths = combinePaths;
-    /**
-     * Combines and resolves paths. If a path is absolute, it replaces any previous path. Any
-     * `.` and `..` path components are resolved. Trailing directory separators are preserved.
-     *
-     * ```ts
-     * resolvePath("/path", "to", "file.ext") === "path/to/file.ext"
-     * resolvePath("/path", "to", "file.ext/") === "path/to/file.ext/"
-     * resolvePath("/path", "dir", "..", "to", "file.ext") === "path/to/file.ext"
-     * ```
-     */
-    function resolvePath(path) {
-        var paths = [];
-        for (var _i = 1; _i < arguments.length; _i++) {
-            paths[_i - 1] = arguments[_i];
-        }
-        return normalizePath(ts.some(paths) ? combinePaths.apply(void 0, __spreadArrays([path], paths)) : normalizeSlashes(path));
-    }
-    ts.resolvePath = resolvePath;
-    /**
-     * Parse a path into an array containing a root component (at index 0) and zero or more path
-     * components (at indices > 0). The result is normalized.
-     * If the path is relative, the root component is `""`.
-     * If the path is absolute, the root component includes the first path separator (`/`).
-     *
-     * ```ts
-     * getNormalizedPathComponents("to/dir/../file.ext", "/path/") === ["/", "path", "to", "file.ext"]
-     * ```
-     */
-    function getNormalizedPathComponents(path, currentDirectory) {
-        return reducePathComponents(getPathComponents(path, currentDirectory));
-    }
-    ts.getNormalizedPathComponents = getNormalizedPathComponents;
-    function getNormalizedAbsolutePath(fileName, currentDirectory) {
-        return getPathFromPathComponents(getNormalizedPathComponents(fileName, currentDirectory));
-    }
-    ts.getNormalizedAbsolutePath = getNormalizedAbsolutePath;
-    function normalizePath(path) {
-        path = normalizeSlashes(path);
-        var normalized = getPathFromPathComponents(reducePathComponents(getPathComponents(path)));
-        return normalized && hasTrailingDirectorySeparator(path) ? ensureTrailingDirectorySeparator(normalized) : normalized;
-    }
-    ts.normalizePath = normalizePath;
-    function getPathWithoutRoot(pathComponents) {
-        if (pathComponents.length === 0)
-            return "";
-        return pathComponents.slice(1).join(ts.directorySeparator);
-    }
-    function getNormalizedAbsolutePathWithoutRoot(fileName, currentDirectory) {
-        return getPathWithoutRoot(getNormalizedPathComponents(fileName, currentDirectory));
-    }
-    ts.getNormalizedAbsolutePathWithoutRoot = getNormalizedAbsolutePathWithoutRoot;
-    function toPath(fileName, basePath, getCanonicalFileName) {
-        var nonCanonicalizedPath = isRootedDiskPath(fileName)
-            ? normalizePath(fileName)
-            : getNormalizedAbsolutePath(fileName, basePath);
-        return getCanonicalFileName(nonCanonicalizedPath);
-    }
-    ts.toPath = toPath;
-    function normalizePathAndParts(path) {
-        path = normalizeSlashes(path);
-        var _a = reducePathComponents(getPathComponents(path)), root = _a[0], parts = _a.slice(1);
-        if (parts.length) {
-            var joinedParts = root + parts.join(ts.directorySeparator);
-            return { path: hasTrailingDirectorySeparator(path) ? ensureTrailingDirectorySeparator(joinedParts) : joinedParts, parts: parts };
-        }
-        else {
-            return { path: root, parts: parts };
-        }
-    }
-    ts.normalizePathAndParts = normalizePathAndParts;
-    function removeTrailingDirectorySeparator(path) {
-        if (hasTrailingDirectorySeparator(path)) {
-            return path.substr(0, path.length - 1);
-        }
-        return path;
-    }
-    ts.removeTrailingDirectorySeparator = removeTrailingDirectorySeparator;
-    function ensureTrailingDirectorySeparator(path) {
-        if (!hasTrailingDirectorySeparator(path)) {
-            return path + ts.directorySeparator;
-        }
-        return path;
-    }
-    ts.ensureTrailingDirectorySeparator = ensureTrailingDirectorySeparator;
-    /**
-     * Ensures a path is either absolute (prefixed with `/` or `c:`) or dot-relative (prefixed
-     * with `./` or `../`) so as not to be confused with an unprefixed module name.
-     *
-     * ```ts
-     * ensurePathIsNonModuleName("/path/to/file.ext") === "/path/to/file.ext"
-     * ensurePathIsNonModuleName("./path/to/file.ext") === "./path/to/file.ext"
-     * ensurePathIsNonModuleName("../path/to/file.ext") === "../path/to/file.ext"
-     * ensurePathIsNonModuleName("path/to/file.ext") === "./path/to/file.ext"
-     * ```
-     */
-    function ensurePathIsNonModuleName(path) {
-        return !pathIsAbsolute(path) && !pathIsRelative(path) ? "./" + path : path;
-    }
-    ts.ensurePathIsNonModuleName = ensurePathIsNonModuleName;
-    function changeAnyExtension(path, ext, extensions, ignoreCase) {
-        var pathext = extensions !== undefined && ignoreCase !== undefined ? getAnyExtensionFromPath(path, extensions, ignoreCase) : getAnyExtensionFromPath(path);
-        return pathext ? path.slice(0, path.length - pathext.length) + (ts.startsWith(ext, ".") ? ext : "." + ext) : path;
-    }
-    ts.changeAnyExtension = changeAnyExtension;
-    //// Path Comparisons
-    // check path for these segments: '', '.'. '..'
-    var relativePathSegmentRegExp = /(^|\/)\.{0,2}($|\/)/;
-    function comparePathsWorker(a, b, componentComparer) {
-        if (a === b)
-            return 0 /* EqualTo */;
-        if (a === undefined)
-            return -1 /* LessThan */;
-        if (b === undefined)
-            return 1 /* GreaterThan */;
-        // NOTE: Performance optimization - shortcut if the root segments differ as there would be no
-        //       need to perform path reduction.
-        var aRoot = a.substring(0, getRootLength(a));
-        var bRoot = b.substring(0, getRootLength(b));
-        var result = ts.compareStringsCaseInsensitive(aRoot, bRoot);
-        if (result !== 0 /* EqualTo */) {
-            return result;
-        }
-        // NOTE: Performance optimization - shortcut if there are no relative path segments in
-        //       the non-root portion of the path
-        var aRest = a.substring(aRoot.length);
-        var bRest = b.substring(bRoot.length);
-        if (!relativePathSegmentRegExp.test(aRest) && !relativePathSegmentRegExp.test(bRest)) {
-            return componentComparer(aRest, bRest);
-        }
-        // The path contains a relative path segment. Normalize the paths and perform a slower component
-        // by component comparison.
-        var aComponents = reducePathComponents(getPathComponents(a));
-        var bComponents = reducePathComponents(getPathComponents(b));
-        var sharedLength = Math.min(aComponents.length, bComponents.length);
-        for (var i = 1; i < sharedLength; i++) {
-            var result_1 = componentComparer(aComponents[i], bComponents[i]);
-            if (result_1 !== 0 /* EqualTo */) {
-                return result_1;
-            }
-        }
-        return ts.compareValues(aComponents.length, bComponents.length);
-    }
-    /**
-     * Performs a case-sensitive comparison of two paths. Path roots are always compared case-insensitively.
-     */
-    function comparePathsCaseSensitive(a, b) {
-        return comparePathsWorker(a, b, ts.compareStringsCaseSensitive);
-    }
-    ts.comparePathsCaseSensitive = comparePathsCaseSensitive;
-    /**
-     * Performs a case-insensitive comparison of two paths.
-     */
-    function comparePathsCaseInsensitive(a, b) {
-        return comparePathsWorker(a, b, ts.compareStringsCaseInsensitive);
-    }
-    ts.comparePathsCaseInsensitive = comparePathsCaseInsensitive;
-    function comparePaths(a, b, currentDirectory, ignoreCase) {
-        if (typeof currentDirectory === "string") {
-            a = combinePaths(currentDirectory, a);
-            b = combinePaths(currentDirectory, b);
-        }
-        else if (typeof currentDirectory === "boolean") {
-            ignoreCase = currentDirectory;
-        }
-        return comparePathsWorker(a, b, ts.getStringComparer(ignoreCase));
-    }
-    ts.comparePaths = comparePaths;
-    function containsPath(parent, child, currentDirectory, ignoreCase) {
-        if (typeof currentDirectory === "string") {
-            parent = combinePaths(currentDirectory, parent);
-            child = combinePaths(currentDirectory, child);
-        }
-        else if (typeof currentDirectory === "boolean") {
-            ignoreCase = currentDirectory;
-        }
-        if (parent === undefined || child === undefined)
-            return false;
-        if (parent === child)
-            return true;
-        var parentComponents = reducePathComponents(getPathComponents(parent));
-        var childComponents = reducePathComponents(getPathComponents(child));
-        if (childComponents.length < parentComponents.length) {
-            return false;
-        }
-        var componentEqualityComparer = ignoreCase ? ts.equateStringsCaseInsensitive : ts.equateStringsCaseSensitive;
-        for (var i = 0; i < parentComponents.length; i++) {
-            var equalityComparer = i === 0 ? ts.equateStringsCaseInsensitive : componentEqualityComparer;
-            if (!equalityComparer(parentComponents[i], childComponents[i])) {
-                return false;
-            }
-        }
-        return true;
-    }
-    ts.containsPath = containsPath;
-    /**
-     * Determines whether `fileName` starts with the specified `directoryName` using the provided path canonicalization callback.
-     * Comparison is case-sensitive between the canonical paths.
-     *
-     * @deprecated Use `containsPath` if possible.
-     */
-    function startsWithDirectory(fileName, directoryName, getCanonicalFileName) {
-        var canonicalFileName = getCanonicalFileName(fileName);
-        var canonicalDirectoryName = getCanonicalFileName(directoryName);
-        return ts.startsWith(canonicalFileName, canonicalDirectoryName + "/") || ts.startsWith(canonicalFileName, canonicalDirectoryName + "\\");
-    }
-    ts.startsWithDirectory = startsWithDirectory;
-    //// Relative Paths
-    function getPathComponentsRelativeTo(from, to, stringEqualityComparer, getCanonicalFileName) {
-        var fromComponents = reducePathComponents(getPathComponents(from));
-        var toComponents = reducePathComponents(getPathComponents(to));
-        var start;
-        for (start = 0; start < fromComponents.length && start < toComponents.length; start++) {
-            var fromComponent = getCanonicalFileName(fromComponents[start]);
-            var toComponent = getCanonicalFileName(toComponents[start]);
-            var comparer = start === 0 ? ts.equateStringsCaseInsensitive : stringEqualityComparer;
-            if (!comparer(fromComponent, toComponent))
-                break;
-        }
-        if (start === 0) {
-            return toComponents;
-        }
-        var components = toComponents.slice(start);
-        var relative = [];
-        for (; start < fromComponents.length; start++) {
-            relative.push("..");
-        }
-        return __spreadArrays([""], relative, components);
-    }
-    ts.getPathComponentsRelativeTo = getPathComponentsRelativeTo;
-    function getRelativePathFromDirectory(fromDirectory, to, getCanonicalFileNameOrIgnoreCase) {
-        ts.Debug.assert((getRootLength(fromDirectory) > 0) === (getRootLength(to) > 0), "Paths must either both be absolute or both be relative");
-        var getCanonicalFileName = typeof getCanonicalFileNameOrIgnoreCase === "function" ? getCanonicalFileNameOrIgnoreCase : ts.identity;
-        var ignoreCase = typeof getCanonicalFileNameOrIgnoreCase === "boolean" ? getCanonicalFileNameOrIgnoreCase : false;
-        var pathComponents = getPathComponentsRelativeTo(fromDirectory, to, ignoreCase ? ts.equateStringsCaseInsensitive : ts.equateStringsCaseSensitive, getCanonicalFileName);
-        return getPathFromPathComponents(pathComponents);
-    }
-    ts.getRelativePathFromDirectory = getRelativePathFromDirectory;
-    function convertToRelativePath(absoluteOrRelativePath, basePath, getCanonicalFileName) {
-        return !isRootedDiskPath(absoluteOrRelativePath)
-            ? absoluteOrRelativePath
-            : getRelativePathToDirectoryOrUrl(basePath, absoluteOrRelativePath, basePath, getCanonicalFileName, /*isAbsolutePathAnUrl*/ false);
-    }
-    ts.convertToRelativePath = convertToRelativePath;
-    function getRelativePathFromFile(from, to, getCanonicalFileName) {
-        return ensurePathIsNonModuleName(getRelativePathFromDirectory(getDirectoryPath(from), to, getCanonicalFileName));
-    }
-    ts.getRelativePathFromFile = getRelativePathFromFile;
-    function getRelativePathToDirectoryOrUrl(directoryPathOrUrl, relativeOrAbsolutePath, currentDirectory, getCanonicalFileName, isAbsolutePathAnUrl) {
-        var pathComponents = getPathComponentsRelativeTo(resolvePath(currentDirectory, directoryPathOrUrl), resolvePath(currentDirectory, relativeOrAbsolutePath), ts.equateStringsCaseSensitive, getCanonicalFileName);
-        var firstComponent = pathComponents[0];
-        if (isAbsolutePathAnUrl && isRootedDiskPath(firstComponent)) {
-            var prefix = firstComponent.charAt(0) === ts.directorySeparator ? "file://" : "file:///";
-            pathComponents[0] = prefix + firstComponent;
-        }
-        return getPathFromPathComponents(pathComponents);
-    }
-    ts.getRelativePathToDirectoryOrUrl = getRelativePathToDirectoryOrUrl;
-    function forEachAncestorDirectory(directory, callback) {
-        while (true) {
-            var result = callback(directory);
-            if (result !== undefined) {
-                return result;
-            }
-            var parentPath = getDirectoryPath(directory);
-            if (parentPath === directory) {
-                return undefined;
-            }
-            directory = parentPath;
-        }
-    }
-    ts.forEachAncestorDirectory = forEachAncestorDirectory;
-    function isNodeModulesDirectory(dirPath) {
-        return ts.endsWith(dirPath, "/node_modules");
-    }
-    ts.isNodeModulesDirectory = isNodeModulesDirectory;
-})(ts || (ts = {}));
 // <auto-generated />
 // generated from './diagnosticInformationMap.generated.ts' by 'src/compiler'
 /* @internal */
 var ts;
 (function (ts) {
-    function diag(code, category, key, message, reportsUnnecessary, elidedInCompatabilityPyramid) {
-        return { code: code, category: category, key: key, message: message, reportsUnnecessary: reportsUnnecessary, elidedInCompatabilityPyramid: elidedInCompatabilityPyramid };
+    function diag(code, category, key, message, reportsUnnecessary, elidedInCompatabilityPyramid, reportsDeprecated) {
+        return { code: code, category: category, key: key, message: message, reportsUnnecessary: reportsUnnecessary, elidedInCompatabilityPyramid: elidedInCompatabilityPyramid, reportsDeprecated: reportsDeprecated };
     }
     ts.Diagnostics = {
         Unterminated_string_literal: diag(1002, ts.DiagnosticCategory.Error, "Unterminated_string_literal_1002", "Unterminated string literal."),
@@ -6832,7 +7719,7 @@ var ts;
         Accessibility_modifier_already_seen: diag(1028, ts.DiagnosticCategory.Error, "Accessibility_modifier_already_seen_1028", "Accessibility modifier already seen."),
         _0_modifier_must_precede_1_modifier: diag(1029, ts.DiagnosticCategory.Error, "_0_modifier_must_precede_1_modifier_1029", "'{0}' modifier must precede '{1}' modifier."),
         _0_modifier_already_seen: diag(1030, ts.DiagnosticCategory.Error, "_0_modifier_already_seen_1030", "'{0}' modifier already seen."),
-        _0_modifier_cannot_appear_on_a_class_element: diag(1031, ts.DiagnosticCategory.Error, "_0_modifier_cannot_appear_on_a_class_element_1031", "'{0}' modifier cannot appear on a class element."),
+        _0_modifier_cannot_appear_on_class_elements_of_this_kind: diag(1031, ts.DiagnosticCategory.Error, "_0_modifier_cannot_appear_on_class_elements_of_this_kind_1031", "'{0}' modifier cannot appear on class elements of this kind."),
         super_must_be_followed_by_an_argument_list_or_member_access: diag(1034, ts.DiagnosticCategory.Error, "super_must_be_followed_by_an_argument_list_or_member_access_1034", "'super' must be followed by an argument list or member access."),
         Only_ambient_modules_can_use_quoted_names: diag(1035, ts.DiagnosticCategory.Error, "Only_ambient_modules_can_use_quoted_names_1035", "Only ambient modules can use quoted names."),
         Statements_are_not_allowed_in_ambient_contexts: diag(1036, ts.DiagnosticCategory.Error, "Statements_are_not_allowed_in_ambient_contexts_1036", "Statements are not allowed in ambient contexts."),
@@ -6861,7 +7748,7 @@ var ts;
         Enum_member_must_have_initializer: diag(1061, ts.DiagnosticCategory.Error, "Enum_member_must_have_initializer_1061", "Enum member must have initializer."),
         Type_is_referenced_directly_or_indirectly_in_the_fulfillment_callback_of_its_own_then_method: diag(1062, ts.DiagnosticCategory.Error, "Type_is_referenced_directly_or_indirectly_in_the_fulfillment_callback_of_its_own_then_method_1062", "Type is referenced directly or indirectly in the fulfillment callback of its own 'then' method."),
         An_export_assignment_cannot_be_used_in_a_namespace: diag(1063, ts.DiagnosticCategory.Error, "An_export_assignment_cannot_be_used_in_a_namespace_1063", "An export assignment cannot be used in a namespace."),
-        The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type: diag(1064, ts.DiagnosticCategory.Error, "The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type_1064", "The return type of an async function or method must be the global Promise<T> type."),
+        The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type_Did_you_mean_to_write_Promise_0: diag(1064, ts.DiagnosticCategory.Error, "The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type_Did_you_mean_to_wri_1064", "The return type of an async function or method must be the global Promise<T> type. Did you mean to write 'Promise<{0}>'?"),
         In_ambient_enum_declarations_member_initializer_must_be_constant_expression: diag(1066, ts.DiagnosticCategory.Error, "In_ambient_enum_declarations_member_initializer_must_be_constant_expression_1066", "In ambient enum declarations member initializer must be constant expression."),
         Unexpected_token_A_constructor_method_accessor_or_property_was_expected: diag(1068, ts.DiagnosticCategory.Error, "Unexpected_token_A_constructor_method_accessor_or_property_was_expected_1068", "Unexpected token. A constructor, method, accessor, or property was expected."),
         Unexpected_token_A_type_parameter_name_was_expected_without_curly_braces: diag(1069, ts.DiagnosticCategory.Error, "Unexpected_token_A_type_parameter_name_was_expected_without_curly_braces_1069", "Unexpected token. A type parameter name was expected without curly braces."),
@@ -6962,7 +7849,7 @@ var ts;
         An_export_declaration_cannot_have_modifiers: diag(1193, ts.DiagnosticCategory.Error, "An_export_declaration_cannot_have_modifiers_1193", "An export declaration cannot have modifiers."),
         Export_declarations_are_not_permitted_in_a_namespace: diag(1194, ts.DiagnosticCategory.Error, "Export_declarations_are_not_permitted_in_a_namespace_1194", "Export declarations are not permitted in a namespace."),
         export_Asterisk_does_not_re_export_a_default: diag(1195, ts.DiagnosticCategory.Error, "export_Asterisk_does_not_re_export_a_default_1195", "'export *' does not re-export a default."),
-        Catch_clause_variable_cannot_have_a_type_annotation: diag(1196, ts.DiagnosticCategory.Error, "Catch_clause_variable_cannot_have_a_type_annotation_1196", "Catch clause variable cannot have a type annotation."),
+        Catch_clause_variable_type_annotation_must_be_any_or_unknown_if_specified: diag(1196, ts.DiagnosticCategory.Error, "Catch_clause_variable_type_annotation_must_be_any_or_unknown_if_specified_1196", "Catch clause variable type annotation must be 'any' or 'unknown' if specified."),
         Catch_clause_variable_cannot_have_an_initializer: diag(1197, ts.DiagnosticCategory.Error, "Catch_clause_variable_cannot_have_an_initializer_1197", "Catch clause variable cannot have an initializer."),
         An_extended_Unicode_escape_value_must_be_between_0x0_and_0x10FFFF_inclusive: diag(1198, ts.DiagnosticCategory.Error, "An_extended_Unicode_escape_value_must_be_between_0x0_and_0x10FFFF_inclusive_1198", "An extended Unicode escape value must be between 0x0 and 0x10FFFF inclusive."),
         Unterminated_Unicode_escape_sequence: diag(1199, ts.DiagnosticCategory.Error, "Unterminated_Unicode_escape_sequence_1199", "Unterminated Unicode escape sequence."),
@@ -6972,7 +7859,7 @@ var ts;
         Re_exporting_a_type_when_the_isolatedModules_flag_is_provided_requires_using_export_type: diag(1205, ts.DiagnosticCategory.Error, "Re_exporting_a_type_when_the_isolatedModules_flag_is_provided_requires_using_export_type_1205", "Re-exporting a type when the '--isolatedModules' flag is provided requires using 'export type'."),
         Decorators_are_not_valid_here: diag(1206, ts.DiagnosticCategory.Error, "Decorators_are_not_valid_here_1206", "Decorators are not valid here."),
         Decorators_cannot_be_applied_to_multiple_get_Slashset_accessors_of_the_same_name: diag(1207, ts.DiagnosticCategory.Error, "Decorators_cannot_be_applied_to_multiple_get_Slashset_accessors_of_the_same_name_1207", "Decorators cannot be applied to multiple get/set accessors of the same name."),
-        All_files_must_be_modules_when_the_isolatedModules_flag_is_provided: diag(1208, ts.DiagnosticCategory.Error, "All_files_must_be_modules_when_the_isolatedModules_flag_is_provided_1208", "All files must be modules when the '--isolatedModules' flag is provided."),
+        _0_cannot_be_compiled_under_isolatedModules_because_it_is_considered_a_global_script_file_Add_an_import_export_or_an_empty_export_statement_to_make_it_a_module: diag(1208, ts.DiagnosticCategory.Error, "_0_cannot_be_compiled_under_isolatedModules_because_it_is_considered_a_global_script_file_Add_an_imp_1208", "'{0}' cannot be compiled under '--isolatedModules' because it is considered a global script file. Add an import, export, or an empty 'export {}' statement to make it a module."),
         Invalid_use_of_0_Class_definitions_are_automatically_in_strict_mode: diag(1210, ts.DiagnosticCategory.Error, "Invalid_use_of_0_Class_definitions_are_automatically_in_strict_mode_1210", "Invalid use of '{0}'. Class definitions are automatically in strict mode."),
         A_class_declaration_without_the_default_modifier_must_have_a_name: diag(1211, ts.DiagnosticCategory.Error, "A_class_declaration_without_the_default_modifier_must_have_a_name_1211", "A class declaration without the 'default' modifier must have a name."),
         Identifier_expected_0_is_a_reserved_word_in_strict_mode: diag(1212, ts.DiagnosticCategory.Error, "Identifier_expected_0_is_a_reserved_word_in_strict_mode_1212", "Identifier expected. '{0}' is a reserved word in strict mode."),
@@ -7020,13 +7907,15 @@ var ts;
         A_definite_assignment_assertion_is_not_permitted_in_this_context: diag(1255, ts.DiagnosticCategory.Error, "A_definite_assignment_assertion_is_not_permitted_in_this_context_1255", "A definite assignment assertion '!' is not permitted in this context."),
         A_rest_element_must_be_last_in_a_tuple_type: diag(1256, ts.DiagnosticCategory.Error, "A_rest_element_must_be_last_in_a_tuple_type_1256", "A rest element must be last in a tuple type."),
         A_required_element_cannot_follow_an_optional_element: diag(1257, ts.DiagnosticCategory.Error, "A_required_element_cannot_follow_an_optional_element_1257", "A required element cannot follow an optional element."),
-        Definite_assignment_assertions_can_only_be_used_along_with_a_type_annotation: diag(1258, ts.DiagnosticCategory.Error, "Definite_assignment_assertions_can_only_be_used_along_with_a_type_annotation_1258", "Definite assignment assertions can only be used along with a type annotation."),
         Module_0_can_only_be_default_imported_using_the_1_flag: diag(1259, ts.DiagnosticCategory.Error, "Module_0_can_only_be_default_imported_using_the_1_flag_1259", "Module '{0}' can only be default-imported using the '{1}' flag"),
         Keywords_cannot_contain_escape_characters: diag(1260, ts.DiagnosticCategory.Error, "Keywords_cannot_contain_escape_characters_1260", "Keywords cannot contain escape characters."),
         Already_included_file_name_0_differs_from_file_name_1_only_in_casing: diag(1261, ts.DiagnosticCategory.Error, "Already_included_file_name_0_differs_from_file_name_1_only_in_casing_1261", "Already included file name '{0}' differs from file name '{1}' only in casing."),
+        Identifier_expected_0_is_a_reserved_word_at_the_top_level_of_a_module: diag(1262, ts.DiagnosticCategory.Error, "Identifier_expected_0_is_a_reserved_word_at_the_top_level_of_a_module_1262", "Identifier expected. '{0}' is a reserved word at the top-level of a module."),
+        Declarations_with_initializers_cannot_also_have_definite_assignment_assertions: diag(1263, ts.DiagnosticCategory.Error, "Declarations_with_initializers_cannot_also_have_definite_assignment_assertions_1263", "Declarations with initializers cannot also have definite assignment assertions."),
+        Declarations_with_definite_assignment_assertions_must_also_have_type_annotations: diag(1264, ts.DiagnosticCategory.Error, "Declarations_with_definite_assignment_assertions_must_also_have_type_annotations_1264", "Declarations with definite assignment assertions must also have type annotations."),
         with_statements_are_not_allowed_in_an_async_function_block: diag(1300, ts.DiagnosticCategory.Error, "with_statements_are_not_allowed_in_an_async_function_block_1300", "'with' statements are not allowed in an async function block."),
         await_expressions_are_only_allowed_within_async_functions_and_at_the_top_levels_of_modules: diag(1308, ts.DiagnosticCategory.Error, "await_expressions_are_only_allowed_within_async_functions_and_at_the_top_levels_of_modules_1308", "'await' expressions are only allowed within async functions and at the top levels of modules."),
-        can_only_be_used_in_an_object_literal_property_inside_a_destructuring_assignment: diag(1312, ts.DiagnosticCategory.Error, "can_only_be_used_in_an_object_literal_property_inside_a_destructuring_assignment_1312", "'=' can only be used in an object literal property inside a destructuring assignment."),
+        Did_you_mean_to_use_a_Colon_An_can_only_follow_a_property_name_when_the_containing_object_literal_is_part_of_a_destructuring_pattern: diag(1312, ts.DiagnosticCategory.Error, "Did_you_mean_to_use_a_Colon_An_can_only_follow_a_property_name_when_the_containing_object_literal_is_1312", "Did you mean to use a ':'? An '=' can only follow a property name when the containing object literal is part of a destructuring pattern."),
         The_body_of_an_if_statement_cannot_be_the_empty_statement: diag(1313, ts.DiagnosticCategory.Error, "The_body_of_an_if_statement_cannot_be_the_empty_statement_1313", "The body of an 'if' statement cannot be the empty statement."),
         Global_module_exports_may_only_appear_in_module_files: diag(1314, ts.DiagnosticCategory.Error, "Global_module_exports_may_only_appear_in_module_files_1314", "Global module exports may only appear in module files."),
         Global_module_exports_may_only_appear_in_declaration_files: diag(1315, ts.DiagnosticCategory.Error, "Global_module_exports_may_only_appear_in_declaration_files_1315", "Global module exports may only appear in declaration files."),
@@ -7040,7 +7929,7 @@ var ts;
         Dynamic_imports_are_only_supported_when_the_module_flag_is_set_to_es2020_esnext_commonjs_amd_system_or_umd: diag(1323, ts.DiagnosticCategory.Error, "Dynamic_imports_are_only_supported_when_the_module_flag_is_set_to_es2020_esnext_commonjs_amd_system__1323", "Dynamic imports are only supported when the '--module' flag is set to 'es2020', 'esnext', 'commonjs', 'amd', 'system', or 'umd'."),
         Dynamic_import_must_have_one_specifier_as_an_argument: diag(1324, ts.DiagnosticCategory.Error, "Dynamic_import_must_have_one_specifier_as_an_argument_1324", "Dynamic import must have one specifier as an argument."),
         Specifier_of_dynamic_import_cannot_be_spread_element: diag(1325, ts.DiagnosticCategory.Error, "Specifier_of_dynamic_import_cannot_be_spread_element_1325", "Specifier of dynamic import cannot be spread element."),
-        Dynamic_import_cannot_have_type_arguments: diag(1326, ts.DiagnosticCategory.Error, "Dynamic_import_cannot_have_type_arguments_1326", "Dynamic import cannot have type arguments"),
+        Dynamic_import_cannot_have_type_arguments: diag(1326, ts.DiagnosticCategory.Error, "Dynamic_import_cannot_have_type_arguments_1326", "Dynamic import cannot have type arguments."),
         String_literal_with_double_quotes_expected: diag(1327, ts.DiagnosticCategory.Error, "String_literal_with_double_quotes_expected_1327", "String literal with double quotes expected."),
         Property_value_can_only_be_string_literal_numeric_literal_true_false_null_object_literal_or_array_literal: diag(1328, ts.DiagnosticCategory.Error, "Property_value_can_only_be_string_literal_numeric_literal_true_false_null_object_literal_or_array_li_1328", "Property value can only be string literal, numeric literal, 'true', 'false', 'null', object literal or array literal."),
         _0_accepts_too_few_arguments_to_be_used_as_a_decorator_here_Did_you_mean_to_call_it_first_and_write_0: diag(1329, ts.DiagnosticCategory.Error, "_0_accepts_too_few_arguments_to_be_used_as_a_decorator_here_Did_you_mean_to_call_it_first_and_write__1329", "'{0}' accepts too few arguments to be used as a decorator here. Did you mean to call it first and write '@{0}()'?"),
@@ -7056,9 +7945,9 @@ var ts;
         Module_0_does_not_refer_to_a_value_but_is_used_as_a_value_here: diag(1339, ts.DiagnosticCategory.Error, "Module_0_does_not_refer_to_a_value_but_is_used_as_a_value_here_1339", "Module '{0}' does not refer to a value, but is used as a value here."),
         Module_0_does_not_refer_to_a_type_but_is_used_as_a_type_here_Did_you_mean_typeof_import_0: diag(1340, ts.DiagnosticCategory.Error, "Module_0_does_not_refer_to_a_type_but_is_used_as_a_type_here_Did_you_mean_typeof_import_0_1340", "Module '{0}' does not refer to a type, but is used as a type here. Did you mean 'typeof import('{0}')'?"),
         Type_arguments_cannot_be_used_here: diag(1342, ts.DiagnosticCategory.Error, "Type_arguments_cannot_be_used_here_1342", "Type arguments cannot be used here."),
-        The_import_meta_meta_property_is_only_allowed_when_the_module_option_is_esnext_or_system: diag(1343, ts.DiagnosticCategory.Error, "The_import_meta_meta_property_is_only_allowed_when_the_module_option_is_esnext_or_system_1343", "The 'import.meta' meta-property is only allowed when the '--module' option is 'esnext' or 'system'."),
+        The_import_meta_meta_property_is_only_allowed_when_the_module_option_is_es2020_esnext_or_system: diag(1343, ts.DiagnosticCategory.Error, "The_import_meta_meta_property_is_only_allowed_when_the_module_option_is_es2020_esnext_or_system_1343", "The 'import.meta' meta-property is only allowed when the '--module' option is 'es2020', 'esnext', or 'system'."),
         A_label_is_not_allowed_here: diag(1344, ts.DiagnosticCategory.Error, "A_label_is_not_allowed_here_1344", "'A label is not allowed here."),
-        An_expression_of_type_void_cannot_be_tested_for_truthiness: diag(1345, ts.DiagnosticCategory.Error, "An_expression_of_type_void_cannot_be_tested_for_truthiness_1345", "An expression of type 'void' cannot be tested for truthiness"),
+        An_expression_of_type_void_cannot_be_tested_for_truthiness: diag(1345, ts.DiagnosticCategory.Error, "An_expression_of_type_void_cannot_be_tested_for_truthiness_1345", "An expression of type 'void' cannot be tested for truthiness."),
         This_parameter_is_not_allowed_with_use_strict_directive: diag(1346, ts.DiagnosticCategory.Error, "This_parameter_is_not_allowed_with_use_strict_directive_1346", "This parameter is not allowed with 'use strict' directive."),
         use_strict_directive_cannot_be_used_with_non_simple_parameter_list: diag(1347, ts.DiagnosticCategory.Error, "use_strict_directive_cannot_be_used_with_non_simple_parameter_list_1347", "'use strict' directive cannot be used with non-simple parameter list."),
         Non_simple_parameter_declared_here: diag(1348, ts.DiagnosticCategory.Error, "Non_simple_parameter_declared_here_1348", "Non-simple parameter declared here."),
@@ -7097,6 +7986,11 @@ var ts;
         Unexpected_token_Did_you_mean_or_gt: diag(1382, ts.DiagnosticCategory.Error, "Unexpected_token_Did_you_mean_or_gt_1382", "Unexpected token. Did you mean `{'>'}` or `&gt;`?"),
         Only_named_exports_may_use_export_type: diag(1383, ts.DiagnosticCategory.Error, "Only_named_exports_may_use_export_type_1383", "Only named exports may use 'export type'."),
         A_new_expression_with_type_arguments_must_always_be_followed_by_a_parenthesized_argument_list: diag(1384, ts.DiagnosticCategory.Error, "A_new_expression_with_type_arguments_must_always_be_followed_by_a_parenthesized_argument_list_1384", "A 'new' expression with type arguments must always be followed by a parenthesized argument list."),
+        Function_type_notation_must_be_parenthesized_when_used_in_a_union_type: diag(1385, ts.DiagnosticCategory.Error, "Function_type_notation_must_be_parenthesized_when_used_in_a_union_type_1385", "Function type notation must be parenthesized when used in a union type."),
+        Constructor_type_notation_must_be_parenthesized_when_used_in_a_union_type: diag(1386, ts.DiagnosticCategory.Error, "Constructor_type_notation_must_be_parenthesized_when_used_in_a_union_type_1386", "Constructor type notation must be parenthesized when used in a union type."),
+        Function_type_notation_must_be_parenthesized_when_used_in_an_intersection_type: diag(1387, ts.DiagnosticCategory.Error, "Function_type_notation_must_be_parenthesized_when_used_in_an_intersection_type_1387", "Function type notation must be parenthesized when used in an intersection type."),
+        Constructor_type_notation_must_be_parenthesized_when_used_in_an_intersection_type: diag(1388, ts.DiagnosticCategory.Error, "Constructor_type_notation_must_be_parenthesized_when_used_in_an_intersection_type_1388", "Constructor type notation must be parenthesized when used in an intersection type."),
+        _0_is_not_allowed_as_a_variable_declaration_name: diag(1389, ts.DiagnosticCategory.Error, "_0_is_not_allowed_as_a_variable_declaration_name_1389", "'{0}' is not allowed as a variable declaration name."),
         The_types_of_0_are_incompatible_between_these_types: diag(2200, ts.DiagnosticCategory.Error, "The_types_of_0_are_incompatible_between_these_types_2200", "The types of '{0}' are incompatible between these types."),
         The_types_returned_by_0_are_incompatible_between_these_types: diag(2201, ts.DiagnosticCategory.Error, "The_types_returned_by_0_are_incompatible_between_these_types_2201", "The types returned by '{0}' are incompatible between these types."),
         Call_signature_return_types_0_and_1_are_incompatible: diag(2202, ts.DiagnosticCategory.Error, "Call_signature_return_types_0_and_1_are_incompatible_2202", "Call signature return types '{0}' and '{1}' are incompatible.", /*reportsUnnecessary*/ undefined, /*elidedInCompatabilityPyramid*/ true),
@@ -7222,6 +8116,7 @@ var ts;
         Property_0_in_type_1_is_not_assignable_to_the_same_property_in_base_type_2: diag(2416, ts.DiagnosticCategory.Error, "Property_0_in_type_1_is_not_assignable_to_the_same_property_in_base_type_2_2416", "Property '{0}' in type '{1}' is not assignable to the same property in base type '{2}'."),
         Class_static_side_0_incorrectly_extends_base_class_static_side_1: diag(2417, ts.DiagnosticCategory.Error, "Class_static_side_0_incorrectly_extends_base_class_static_side_1_2417", "Class static side '{0}' incorrectly extends base class static side '{1}'."),
         Type_of_computed_property_s_value_is_0_which_is_not_assignable_to_type_1: diag(2418, ts.DiagnosticCategory.Error, "Type_of_computed_property_s_value_is_0_which_is_not_assignable_to_type_1_2418", "Type of computed property's value is '{0}', which is not assignable to type '{1}'."),
+        Types_of_construct_signatures_are_incompatible: diag(2419, ts.DiagnosticCategory.Error, "Types_of_construct_signatures_are_incompatible_2419", "Types of construct signatures are incompatible."),
         Class_0_incorrectly_implements_interface_1: diag(2420, ts.DiagnosticCategory.Error, "Class_0_incorrectly_implements_interface_1_2420", "Class '{0}' incorrectly implements interface '{1}'."),
         A_class_can_only_implement_an_object_type_or_intersection_of_object_types_with_statically_known_members: diag(2422, ts.DiagnosticCategory.Error, "A_class_can_only_implement_an_object_type_or_intersection_of_object_types_with_statically_known_memb_2422", "A class can only implement an object type or intersection of object types with statically known members."),
         Class_0_defines_instance_member_function_1_but_extended_class_2_defines_it_as_instance_member_accessor: diag(2423, ts.DiagnosticCategory.Error, "Class_0_defines_instance_member_function_1_but_extended_class_2_defines_it_as_instance_member_access_2423", "Class '{0}' defines instance member function '{1}', but extended class '{2}' defines it as instance member accessor."),
@@ -7345,6 +8240,7 @@ var ts;
         The_type_returned_by_the_0_method_of_an_async_iterator_must_be_a_promise_for_a_type_with_a_value_property: diag(2547, ts.DiagnosticCategory.Error, "The_type_returned_by_the_0_method_of_an_async_iterator_must_be_a_promise_for_a_type_with_a_value_pro_2547", "The type returned by the '{0}()' method of an async iterator must be a promise for a type with a 'value' property."),
         Type_0_is_not_an_array_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator: diag(2548, ts.DiagnosticCategory.Error, "Type_0_is_not_an_array_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator_2548", "Type '{0}' is not an array type or does not have a '[Symbol.iterator]()' method that returns an iterator."),
         Type_0_is_not_an_array_type_or_a_string_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator: diag(2549, ts.DiagnosticCategory.Error, "Type_0_is_not_an_array_type_or_a_string_type_or_does_not_have_a_Symbol_iterator_method_that_returns__2549", "Type '{0}' is not an array type or a string type or does not have a '[Symbol.iterator]()' method that returns an iterator."),
+        Property_0_does_not_exist_on_type_1_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_2_or_later: diag(2550, ts.DiagnosticCategory.Error, "Property_0_does_not_exist_on_type_1_Do_you_need_to_change_your_target_library_Try_changing_the_lib_c_2550", "Property '{0}' does not exist on type '{1}'. Do you need to change your target library? Try changing the `lib` compiler option to '{2}' or later."),
         Property_0_does_not_exist_on_type_1_Did_you_mean_2: diag(2551, ts.DiagnosticCategory.Error, "Property_0_does_not_exist_on_type_1_Did_you_mean_2_2551", "Property '{0}' does not exist on type '{1}'. Did you mean '{2}'?"),
         Cannot_find_name_0_Did_you_mean_1: diag(2552, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Did_you_mean_1_2552", "Cannot find name '{0}'. Did you mean '{1}'?"),
         Computed_values_are_not_permitted_in_an_enum_with_string_valued_members: diag(2553, ts.DiagnosticCategory.Error, "Computed_values_are_not_permitted_in_an_enum_with_string_valued_members_2553", "Computed values are not permitted in an enum with string valued members."),
@@ -7368,13 +8264,13 @@ var ts;
         Property_0_is_incompatible_with_rest_element_type: diag(2573, ts.DiagnosticCategory.Error, "Property_0_is_incompatible_with_rest_element_type_2573", "Property '{0}' is incompatible with rest element type."),
         A_rest_element_type_must_be_an_array_type: diag(2574, ts.DiagnosticCategory.Error, "A_rest_element_type_must_be_an_array_type_2574", "A rest element type must be an array type."),
         No_overload_expects_0_arguments_but_overloads_do_exist_that_expect_either_1_or_2_arguments: diag(2575, ts.DiagnosticCategory.Error, "No_overload_expects_0_arguments_but_overloads_do_exist_that_expect_either_1_or_2_arguments_2575", "No overload expects {0} arguments, but overloads do exist that expect either {1} or {2} arguments."),
-        Property_0_is_a_static_member_of_type_1: diag(2576, ts.DiagnosticCategory.Error, "Property_0_is_a_static_member_of_type_1_2576", "Property '{0}' is a static member of type '{1}'"),
+        Property_0_is_a_static_member_of_type_1: diag(2576, ts.DiagnosticCategory.Error, "Property_0_is_a_static_member_of_type_1_2576", "Property '{0}' is a static member of type '{1}'."),
         Return_type_annotation_circularly_references_itself: diag(2577, ts.DiagnosticCategory.Error, "Return_type_annotation_circularly_references_itself_2577", "Return type annotation circularly references itself."),
         Unused_ts_expect_error_directive: diag(2578, ts.DiagnosticCategory.Error, "Unused_ts_expect_error_directive_2578", "Unused '@ts-expect-error' directive."),
-        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_types_Slashnode: diag(2580, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_types_Slashnode_2580", "Cannot find name '{0}'. Do you need to install type definitions for node? Try `npm i @types/node`."),
-        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_types_Slashjquery: diag(2581, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_types_Slashjquery_2581", "Cannot find name '{0}'. Do you need to install type definitions for jQuery? Try `npm i @types/jquery`."),
-        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_types_Slashjest_or_npm_i_types_Slashmocha: diag(2582, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_types_Slashje_2582", "Cannot find name '{0}'. Do you need to install type definitions for a test runner? Try `npm i @types/jest` or `npm i @types/mocha`."),
-        Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_es2015_or_later: diag(2583, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_2583", "Cannot find name '{0}'. Do you need to change your target library? Try changing the `lib` compiler option to es2015 or later."),
+        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_save_dev_types_Slashnode: diag(2580, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_save_dev_types_Slashno_2580", "Cannot find name '{0}'. Do you need to install type definitions for node? Try `npm i --save-dev @types/node`."),
+        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slashjquery: diag(2581, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slash_2581", "Cannot find name '{0}'. Do you need to install type definitions for jQuery? Try `npm i --save-dev @types/jquery`."),
+        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_save_dev_types_Slashjest_or_npm_i_save_dev_types_Slashmocha: diag(2582, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_save_dev_type_2582", "Cannot find name '{0}'. Do you need to install type definitions for a test runner? Try `npm i --save-dev @types/jest` or `npm i --save-dev @types/mocha`."),
+        Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_1_or_later: diag(2583, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_2583", "Cannot find name '{0}'. Do you need to change your target library? Try changing the `lib` compiler option to '{1}' or later."),
         Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_include_dom: diag(2584, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_2584", "Cannot find name '{0}'. Do you need to change your target library? Try changing the `lib` compiler option to include 'dom'."),
         _0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_es2015_or_later: diag(2585, ts.DiagnosticCategory.Error, "_0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Do_you_need_to_change_your_target_library_2585", "'{0}' only refers to a type, but is being used as a value here. Do you need to change your target library? Try changing the `lib` compiler option to es2015 or later."),
         Enum_type_0_circularly_references_itself: diag(2586, ts.DiagnosticCategory.Error, "Enum_type_0_circularly_references_itself_2586", "Enum type '{0}' circularly references itself."),
@@ -7382,9 +8278,9 @@ var ts;
         Cannot_assign_to_0_because_it_is_a_constant: diag(2588, ts.DiagnosticCategory.Error, "Cannot_assign_to_0_because_it_is_a_constant_2588", "Cannot assign to '{0}' because it is a constant."),
         Type_instantiation_is_excessively_deep_and_possibly_infinite: diag(2589, ts.DiagnosticCategory.Error, "Type_instantiation_is_excessively_deep_and_possibly_infinite_2589", "Type instantiation is excessively deep and possibly infinite."),
         Expression_produces_a_union_type_that_is_too_complex_to_represent: diag(2590, ts.DiagnosticCategory.Error, "Expression_produces_a_union_type_that_is_too_complex_to_represent_2590", "Expression produces a union type that is too complex to represent."),
-        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_types_Slashnode_and_then_add_node_to_the_types_field_in_your_tsconfig: diag(2591, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_types_Slashnode_and_th_2591", "Cannot find name '{0}'. Do you need to install type definitions for node? Try `npm i @types/node` and then add `node` to the types field in your tsconfig."),
-        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_types_Slashjquery_and_then_add_jquery_to_the_types_field_in_your_tsconfig: diag(2592, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_types_Slashjquery_an_2592", "Cannot find name '{0}'. Do you need to install type definitions for jQuery? Try `npm i @types/jquery` and then add `jquery` to the types field in your tsconfig."),
-        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_types_Slashjest_or_npm_i_types_Slashmocha_and_then_add_jest_or_mocha_to_the_types_field_in_your_tsconfig: diag(2593, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_types_Slashje_2593", "Cannot find name '{0}'. Do you need to install type definitions for a test runner? Try `npm i @types/jest` or `npm i @types/mocha` and then add `jest` or `mocha` to the types field in your tsconfig."),
+        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_save_dev_types_Slashnode_and_then_add_node_to_the_types_field_in_your_tsconfig: diag(2591, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_save_dev_types_Slashno_2591", "Cannot find name '{0}'. Do you need to install type definitions for node? Try `npm i --save-dev @types/node` and then add `node` to the types field in your tsconfig."),
+        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slashjquery_and_then_add_jquery_to_the_types_field_in_your_tsconfig: diag(2592, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slash_2592", "Cannot find name '{0}'. Do you need to install type definitions for jQuery? Try `npm i --save-dev @types/jquery` and then add `jquery` to the types field in your tsconfig."),
+        Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_save_dev_types_Slashjest_or_npm_i_save_dev_types_Slashmocha_and_then_add_jest_or_mocha_to_the_types_field_in_your_tsconfig: diag(2593, ts.DiagnosticCategory.Error, "Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_save_dev_type_2593", "Cannot find name '{0}'. Do you need to install type definitions for a test runner? Try `npm i --save-dev @types/jest` or `npm i --save-dev @types/mocha` and then add `jest` or `mocha` to the types field in your tsconfig."),
         This_module_is_declared_with_using_export_and_can_only_be_used_with_a_default_import_when_using_the_0_flag: diag(2594, ts.DiagnosticCategory.Error, "This_module_is_declared_with_using_export_and_can_only_be_used_with_a_default_import_when_using_the__2594", "This module is declared with using 'export =', and can only be used with a default import when using the '{0}' flag."),
         _0_can_only_be_imported_by_using_a_default_import: diag(2595, ts.DiagnosticCategory.Error, "_0_can_only_be_imported_by_using_a_default_import_2595", "'{0}' can only be imported by using a default import."),
         _0_can_only_be_imported_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import: diag(2596, ts.DiagnosticCategory.Error, "_0_can_only_be_imported_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import_2596", "'{0}' can only be imported by turning on the 'esModuleInterop' flag and using a default import."),
@@ -7408,6 +8304,11 @@ var ts;
         Type_of_property_0_circularly_references_itself_in_mapped_type_1: diag(2615, ts.DiagnosticCategory.Error, "Type_of_property_0_circularly_references_itself_in_mapped_type_1_2615", "Type of property '{0}' circularly references itself in mapped type '{1}'."),
         _0_can_only_be_imported_by_using_import_1_require_2_or_a_default_import: diag(2616, ts.DiagnosticCategory.Error, "_0_can_only_be_imported_by_using_import_1_require_2_or_a_default_import_2616", "'{0}' can only be imported by using 'import {1} = require({2})' or a default import."),
         _0_can_only_be_imported_by_using_import_1_require_2_or_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import: diag(2617, ts.DiagnosticCategory.Error, "_0_can_only_be_imported_by_using_import_1_require_2_or_by_turning_on_the_esModuleInterop_flag_and_us_2617", "'{0}' can only be imported by using 'import {1} = require({2})' or by turning on the 'esModuleInterop' flag and using a default import."),
+        Source_has_0_element_s_but_target_requires_1: diag(2618, ts.DiagnosticCategory.Error, "Source_has_0_element_s_but_target_requires_1_2618", "Source has {0} element(s) but target requires {1}."),
+        Source_has_0_element_s_but_target_allows_only_1: diag(2619, ts.DiagnosticCategory.Error, "Source_has_0_element_s_but_target_allows_only_1_2619", "Source has {0} element(s) but target allows only {1}."),
+        Target_requires_0_element_s_but_source_may_have_fewer: diag(2620, ts.DiagnosticCategory.Error, "Target_requires_0_element_s_but_source_may_have_fewer_2620", "Target requires {0} element(s) but source may have fewer."),
+        Target_allows_only_0_element_s_but_source_may_have_more: diag(2621, ts.DiagnosticCategory.Error, "Target_allows_only_0_element_s_but_source_may_have_more_2621", "Target allows only {0} element(s) but source may have more."),
+        Element_at_index_0_is_variadic_in_one_type_but_not_in_the_other: diag(2622, ts.DiagnosticCategory.Error, "Element_at_index_0_is_variadic_in_one_type_but_not_in_the_other_2622", "Element at index {0} is variadic in one type but not in the other."),
         Cannot_augment_module_0_with_value_exports_because_it_resolves_to_a_non_module_entity: diag(2649, ts.DiagnosticCategory.Error, "Cannot_augment_module_0_with_value_exports_because_it_resolves_to_a_non_module_entity_2649", "Cannot augment module '{0}' with value exports because it resolves to a non-module entity."),
         A_member_initializer_in_a_enum_declaration_cannot_reference_members_declared_after_it_including_members_defined_in_other_enums: diag(2651, ts.DiagnosticCategory.Error, "A_member_initializer_in_a_enum_declaration_cannot_reference_members_declared_after_it_including_memb_2651", "A member initializer in a enum declaration cannot reference members declared after it, including members defined in other enums."),
         Merged_declaration_0_cannot_include_a_default_export_declaration_Consider_adding_a_separate_export_default_0_declaration_instead: diag(2652, ts.DiagnosticCategory.Error, "Merged_declaration_0_cannot_include_a_default_export_declaration_Consider_adding_a_separate_export_d_2652", "Merged declaration '{0}' cannot include a default export declaration. Consider adding a separate 'export default {0}' declaration instead."),
@@ -7447,6 +8348,7 @@ var ts;
         All_declarations_of_0_must_have_identical_modifiers: diag(2687, ts.DiagnosticCategory.Error, "All_declarations_of_0_must_have_identical_modifiers_2687", "All declarations of '{0}' must have identical modifiers."),
         Cannot_find_type_definition_file_for_0: diag(2688, ts.DiagnosticCategory.Error, "Cannot_find_type_definition_file_for_0_2688", "Cannot find type definition file for '{0}'."),
         Cannot_extend_an_interface_0_Did_you_mean_implements: diag(2689, ts.DiagnosticCategory.Error, "Cannot_extend_an_interface_0_Did_you_mean_implements_2689", "Cannot extend an interface '{0}'. Did you mean 'implements'?"),
+        _0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Did_you_mean_to_use_1_in_0: diag(2690, ts.DiagnosticCategory.Error, "_0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Did_you_mean_to_use_1_in_0_2690", "'{0}' only refers to a type, but is being used as a value here. Did you mean to use '{1} in {0}'?"),
         An_import_path_cannot_end_with_a_0_extension_Consider_importing_1_instead: diag(2691, ts.DiagnosticCategory.Error, "An_import_path_cannot_end_with_a_0_extension_Consider_importing_1_instead_2691", "An import path cannot end with a '{0}' extension. Consider importing '{1}' instead."),
         _0_is_a_primitive_but_1_is_a_wrapper_object_Prefer_using_0_when_possible: diag(2692, ts.DiagnosticCategory.Error, "_0_is_a_primitive_but_1_is_a_wrapper_object_Prefer_using_0_when_possible_2692", "'{0}' is a primitive, but '{1}' is a wrapper object. Prefer using '{0}' when possible."),
         _0_only_refers_to_a_type_but_is_being_used_as_a_value_here: diag(2693, ts.DiagnosticCategory.Error, "_0_only_refers_to_a_type_but_is_being_used_as_a_value_here_2693", "'{0}' only refers to a type, but is being used as a value here."),
@@ -7480,7 +8382,7 @@ var ts;
         Cannot_invoke_an_object_which_is_possibly_null: diag(2721, ts.DiagnosticCategory.Error, "Cannot_invoke_an_object_which_is_possibly_null_2721", "Cannot invoke an object which is possibly 'null'."),
         Cannot_invoke_an_object_which_is_possibly_undefined: diag(2722, ts.DiagnosticCategory.Error, "Cannot_invoke_an_object_which_is_possibly_undefined_2722", "Cannot invoke an object which is possibly 'undefined'."),
         Cannot_invoke_an_object_which_is_possibly_null_or_undefined: diag(2723, ts.DiagnosticCategory.Error, "Cannot_invoke_an_object_which_is_possibly_null_or_undefined_2723", "Cannot invoke an object which is possibly 'null' or 'undefined'."),
-        Module_0_has_no_exported_member_1_Did_you_mean_2: diag(2724, ts.DiagnosticCategory.Error, "Module_0_has_no_exported_member_1_Did_you_mean_2_2724", "Module '{0}' has no exported member '{1}'. Did you mean '{2}'?"),
+        _0_has_no_exported_member_named_1_Did_you_mean_2: diag(2724, ts.DiagnosticCategory.Error, "_0_has_no_exported_member_named_1_Did_you_mean_2_2724", "'{0}' has no exported member named '{1}'. Did you mean '{2}'?"),
         Class_name_cannot_be_Object_when_targeting_ES5_with_module_0: diag(2725, ts.DiagnosticCategory.Error, "Class_name_cannot_be_Object_when_targeting_ES5_with_module_0_2725", "Class name cannot be 'Object' when targeting ES5 with module {0}."),
         Cannot_find_lib_definition_for_0: diag(2726, ts.DiagnosticCategory.Error, "Cannot_find_lib_definition_for_0_2726", "Cannot find lib definition for '{0}'."),
         Cannot_find_lib_definition_for_0_Did_you_mean_1: diag(2727, ts.DiagnosticCategory.Error, "Cannot_find_lib_definition_for_0_Did_you_mean_1_2727", "Cannot find lib definition for '{0}'. Did you mean '{1}'?"),
@@ -7488,7 +8390,7 @@ var ts;
         Property_0_is_used_before_its_initialization: diag(2729, ts.DiagnosticCategory.Error, "Property_0_is_used_before_its_initialization_2729", "Property '{0}' is used before its initialization."),
         An_arrow_function_cannot_have_a_this_parameter: diag(2730, ts.DiagnosticCategory.Error, "An_arrow_function_cannot_have_a_this_parameter_2730", "An arrow function cannot have a 'this' parameter."),
         Implicit_conversion_of_a_symbol_to_a_string_will_fail_at_runtime_Consider_wrapping_this_expression_in_String: diag(2731, ts.DiagnosticCategory.Error, "Implicit_conversion_of_a_symbol_to_a_string_will_fail_at_runtime_Consider_wrapping_this_expression_i_2731", "Implicit conversion of a 'symbol' to a 'string' will fail at runtime. Consider wrapping this expression in 'String(...)'."),
-        Cannot_find_module_0_Consider_using_resolveJsonModule_to_import_module_with_json_extension: diag(2732, ts.DiagnosticCategory.Error, "Cannot_find_module_0_Consider_using_resolveJsonModule_to_import_module_with_json_extension_2732", "Cannot find module '{0}'. Consider using '--resolveJsonModule' to import module with '.json' extension"),
+        Cannot_find_module_0_Consider_using_resolveJsonModule_to_import_module_with_json_extension: diag(2732, ts.DiagnosticCategory.Error, "Cannot_find_module_0_Consider_using_resolveJsonModule_to_import_module_with_json_extension_2732", "Cannot find module '{0}'. Consider using '--resolveJsonModule' to import module with '.json' extension."),
         Property_0_was_also_declared_here: diag(2733, ts.DiagnosticCategory.Error, "Property_0_was_also_declared_here_2733", "Property '{0}' was also declared here."),
         Are_you_missing_a_semicolon: diag(2734, ts.DiagnosticCategory.Error, "Are_you_missing_a_semicolon_2734", "Are you missing a semicolon?"),
         Did_you_mean_for_0_to_be_constrained_to_type_new_args_Colon_any_1: diag(2735, ts.DiagnosticCategory.Error, "Did_you_mean_for_0_to_be_constrained_to_type_new_args_Colon_any_1_2735", "Did you mean for '{0}' to be constrained to type 'new (...args: any[]) => {1}'?"),
@@ -7546,6 +8448,13 @@ var ts;
         Its_return_type_0_is_not_a_valid_JSX_element: diag(2787, ts.DiagnosticCategory.Error, "Its_return_type_0_is_not_a_valid_JSX_element_2787", "Its return type '{0}' is not a valid JSX element."),
         Its_instance_type_0_is_not_a_valid_JSX_element: diag(2788, ts.DiagnosticCategory.Error, "Its_instance_type_0_is_not_a_valid_JSX_element_2788", "Its instance type '{0}' is not a valid JSX element."),
         Its_element_type_0_is_not_a_valid_JSX_element: diag(2789, ts.DiagnosticCategory.Error, "Its_element_type_0_is_not_a_valid_JSX_element_2789", "Its element type '{0}' is not a valid JSX element."),
+        The_operand_of_a_delete_operator_must_be_optional: diag(2790, ts.DiagnosticCategory.Error, "The_operand_of_a_delete_operator_must_be_optional_2790", "The operand of a 'delete' operator must be optional."),
+        Exponentiation_cannot_be_performed_on_bigint_values_unless_the_target_option_is_set_to_es2016_or_later: diag(2791, ts.DiagnosticCategory.Error, "Exponentiation_cannot_be_performed_on_bigint_values_unless_the_target_option_is_set_to_es2016_or_lat_2791", "Exponentiation cannot be performed on 'bigint' values unless the 'target' option is set to 'es2016' or later."),
+        Cannot_find_module_0_Did_you_mean_to_set_the_moduleResolution_option_to_node_or_to_add_aliases_to_the_paths_option: diag(2792, ts.DiagnosticCategory.Error, "Cannot_find_module_0_Did_you_mean_to_set_the_moduleResolution_option_to_node_or_to_add_aliases_to_th_2792", "Cannot find module '{0}'. Did you mean to set the 'moduleResolution' option to 'node', or to add aliases to the 'paths' option?"),
+        The_call_would_have_succeeded_against_this_implementation_but_implementation_signatures_of_overloads_are_not_externally_visible: diag(2793, ts.DiagnosticCategory.Error, "The_call_would_have_succeeded_against_this_implementation_but_implementation_signatures_of_overloads_2793", "The call would have succeeded against this implementation, but implementation signatures of overloads are not externally visible."),
+        Expected_0_arguments_but_got_1_Did_you_forget_to_include_void_in_your_type_argument_to_Promise: diag(2794, ts.DiagnosticCategory.Error, "Expected_0_arguments_but_got_1_Did_you_forget_to_include_void_in_your_type_argument_to_Promise_2794", "Expected {0} arguments, but got {1}. Did you forget to include 'void' in your type argument to 'Promise'?"),
+        The_intrinsic_keyword_can_only_be_used_to_declare_compiler_provided_intrinsic_types: diag(2795, ts.DiagnosticCategory.Error, "The_intrinsic_keyword_can_only_be_used_to_declare_compiler_provided_intrinsic_types_2795", "The 'intrinsic' keyword can only be used to declare compiler provided intrinsic types."),
+        It_is_likely_that_you_are_missing_a_comma_to_separate_these_two_template_expressions_They_form_a_tagged_template_expression_which_cannot_be_invoked: diag(2796, ts.DiagnosticCategory.Error, "It_is_likely_that_you_are_missing_a_comma_to_separate_these_two_template_expressions_They_form_a_tag_2796", "It is likely that you are missing a comma to separate these two template expressions. They form a tagged template expression which cannot be invoked."),
         Import_declaration_0_is_using_private_name_1: diag(4000, ts.DiagnosticCategory.Error, "Import_declaration_0_is_using_private_name_1_4000", "Import declaration '{0}' is using private name '{1}'."),
         Type_parameter_0_of_exported_class_has_or_is_using_private_name_1: diag(4002, ts.DiagnosticCategory.Error, "Type_parameter_0_of_exported_class_has_or_is_using_private_name_1_4002", "Type parameter '{0}' of exported class has or is using private name '{1}'."),
         Type_parameter_0_of_exported_interface_has_or_is_using_private_name_1: diag(4004, ts.DiagnosticCategory.Error, "Type_parameter_0_of_exported_interface_has_or_is_using_private_name_1_4004", "Type parameter '{0}' of exported interface has or is using private name '{1}'."),
@@ -7557,6 +8466,7 @@ var ts;
         Type_parameter_0_of_exported_function_has_or_is_using_private_name_1: diag(4016, ts.DiagnosticCategory.Error, "Type_parameter_0_of_exported_function_has_or_is_using_private_name_1_4016", "Type parameter '{0}' of exported function has or is using private name '{1}'."),
         Implements_clause_of_exported_class_0_has_or_is_using_private_name_1: diag(4019, ts.DiagnosticCategory.Error, "Implements_clause_of_exported_class_0_has_or_is_using_private_name_1_4019", "Implements clause of exported class '{0}' has or is using private name '{1}'."),
         extends_clause_of_exported_class_0_has_or_is_using_private_name_1: diag(4020, ts.DiagnosticCategory.Error, "extends_clause_of_exported_class_0_has_or_is_using_private_name_1_4020", "'extends' clause of exported class '{0}' has or is using private name '{1}'."),
+        extends_clause_of_exported_class_has_or_is_using_private_name_0: diag(4021, ts.DiagnosticCategory.Error, "extends_clause_of_exported_class_has_or_is_using_private_name_0_4021", "'extends' clause of exported class has or is using private name '{0}'."),
         extends_clause_of_exported_interface_0_has_or_is_using_private_name_1: diag(4022, ts.DiagnosticCategory.Error, "extends_clause_of_exported_interface_0_has_or_is_using_private_name_1_4022", "'extends' clause of exported interface '{0}' has or is using private name '{1}'."),
         Exported_variable_0_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named: diag(4023, ts.DiagnosticCategory.Error, "Exported_variable_0_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named_4023", "Exported variable '{0}' has or is using name '{1}' from external module {2} but cannot be named."),
         Exported_variable_0_has_or_is_using_name_1_from_private_module_2: diag(4024, ts.DiagnosticCategory.Error, "Exported_variable_0_has_or_is_using_name_1_from_private_module_2_4024", "Exported variable '{0}' has or is using name '{1}' from private module '{2}'."),
@@ -7658,7 +8568,6 @@ var ts;
         Cannot_find_a_tsconfig_json_file_at_the_specified_directory_Colon_0: diag(5057, ts.DiagnosticCategory.Error, "Cannot_find_a_tsconfig_json_file_at_the_specified_directory_Colon_0_5057", "Cannot find a tsconfig.json file at the specified directory: '{0}'."),
         The_specified_path_does_not_exist_Colon_0: diag(5058, ts.DiagnosticCategory.Error, "The_specified_path_does_not_exist_Colon_0_5058", "The specified path does not exist: '{0}'."),
         Invalid_value_for_reactNamespace_0_is_not_a_valid_identifier: diag(5059, ts.DiagnosticCategory.Error, "Invalid_value_for_reactNamespace_0_is_not_a_valid_identifier_5059", "Invalid value for '--reactNamespace'. '{0}' is not a valid identifier."),
-        Option_paths_cannot_be_used_without_specifying_baseUrl_option: diag(5060, ts.DiagnosticCategory.Error, "Option_paths_cannot_be_used_without_specifying_baseUrl_option_5060", "Option 'paths' cannot be used without specifying '--baseUrl' option."),
         Pattern_0_can_have_at_most_one_Asterisk_character: diag(5061, ts.DiagnosticCategory.Error, "Pattern_0_can_have_at_most_one_Asterisk_character_5061", "Pattern '{0}' can have at most one '*' character."),
         Substitution_0_in_pattern_1_can_have_at_most_one_Asterisk_character: diag(5062, ts.DiagnosticCategory.Error, "Substitution_0_in_pattern_1_can_have_at_most_one_Asterisk_character_5062", "Substitution '{0}' in pattern '{1}' can have at most one '*' character."),
         Substitutions_for_pattern_0_should_be_an_array: diag(5063, ts.DiagnosticCategory.Error, "Substitutions_for_pattern_0_should_be_an_array_5063", "Substitutions for pattern '{0}' should be an array."),
@@ -7682,6 +8591,13 @@ var ts;
         Cannot_find_a_tsconfig_json_file_at_the_current_directory_Colon_0: diag(5081, ts.DiagnosticCategory.Error, "Cannot_find_a_tsconfig_json_file_at_the_current_directory_Colon_0_5081", "Cannot find a tsconfig.json file at the current directory: {0}."),
         _0_could_be_instantiated_with_an_arbitrary_type_which_could_be_unrelated_to_1: diag(5082, ts.DiagnosticCategory.Error, "_0_could_be_instantiated_with_an_arbitrary_type_which_could_be_unrelated_to_1_5082", "'{0}' could be instantiated with an arbitrary type which could be unrelated to '{1}'."),
         Cannot_read_file_0: diag(5083, ts.DiagnosticCategory.Error, "Cannot_read_file_0_5083", "Cannot read file '{0}'."),
+        Tuple_members_must_all_have_names_or_all_not_have_names: diag(5084, ts.DiagnosticCategory.Error, "Tuple_members_must_all_have_names_or_all_not_have_names_5084", "Tuple members must all have names or all not have names."),
+        A_tuple_member_cannot_be_both_optional_and_rest: diag(5085, ts.DiagnosticCategory.Error, "A_tuple_member_cannot_be_both_optional_and_rest_5085", "A tuple member cannot be both optional and rest."),
+        A_labeled_tuple_element_is_declared_as_optional_with_a_question_mark_after_the_name_and_before_the_colon_rather_than_after_the_type: diag(5086, ts.DiagnosticCategory.Error, "A_labeled_tuple_element_is_declared_as_optional_with_a_question_mark_after_the_name_and_before_the_c_5086", "A labeled tuple element is declared as optional with a question mark after the name and before the colon, rather than after the type."),
+        A_labeled_tuple_element_is_declared_as_rest_with_a_before_the_name_rather_than_before_the_type: diag(5087, ts.DiagnosticCategory.Error, "A_labeled_tuple_element_is_declared_as_rest_with_a_before_the_name_rather_than_before_the_type_5087", "A labeled tuple element is declared as rest with a `...` before the name, rather than before the type."),
+        The_inferred_type_of_0_references_a_type_with_a_cyclic_structure_which_cannot_be_trivially_serialized_A_type_annotation_is_necessary: diag(5088, ts.DiagnosticCategory.Error, "The_inferred_type_of_0_references_a_type_with_a_cyclic_structure_which_cannot_be_trivially_serialize_5088", "The inferred type of '{0}' references a type with a cyclic structure which cannot be trivially serialized. A type annotation is necessary."),
+        Option_0_cannot_be_specified_when_option_jsx_is_1: diag(5089, ts.DiagnosticCategory.Error, "Option_0_cannot_be_specified_when_option_jsx_is_1_5089", "Option '{0}' cannot be specified when option 'jsx' is '{1}'."),
+        Non_relative_paths_are_not_allowed_when_baseUrl_is_not_set_Did_you_forget_a_leading_Slash: diag(5090, ts.DiagnosticCategory.Error, "Non_relative_paths_are_not_allowed_when_baseUrl_is_not_set_Did_you_forget_a_leading_Slash_5090", "Non-relative paths are not allowed when 'baseUrl' is not set. Did you forget a leading './'?"),
         Generates_a_sourcemap_for_each_corresponding_d_ts_file: diag(6000, ts.DiagnosticCategory.Message, "Generates_a_sourcemap_for_each_corresponding_d_ts_file_6000", "Generates a sourcemap for each corresponding '.d.ts' file."),
         Concatenate_and_emit_output_to_single_file: diag(6001, ts.DiagnosticCategory.Message, "Concatenate_and_emit_output_to_single_file_6001", "Concatenate and emit output to single file."),
         Generates_corresponding_d_ts_file: diag(6002, ts.DiagnosticCategory.Message, "Generates_corresponding_d_ts_file_6002", "Generates corresponding '.d.ts' file."),
@@ -7872,7 +8788,7 @@ var ts;
         Project_references_may_not_form_a_circular_graph_Cycle_detected_Colon_0: diag(6202, ts.DiagnosticCategory.Error, "Project_references_may_not_form_a_circular_graph_Cycle_detected_Colon_0_6202", "Project references may not form a circular graph. Cycle detected: {0}"),
         _0_was_also_declared_here: diag(6203, ts.DiagnosticCategory.Message, "_0_was_also_declared_here_6203", "'{0}' was also declared here."),
         and_here: diag(6204, ts.DiagnosticCategory.Message, "and_here_6204", "and here."),
-        All_type_parameters_are_unused: diag(6205, ts.DiagnosticCategory.Error, "All_type_parameters_are_unused_6205", "All type parameters are unused"),
+        All_type_parameters_are_unused: diag(6205, ts.DiagnosticCategory.Error, "All_type_parameters_are_unused_6205", "All type parameters are unused."),
         package_json_has_a_typesVersions_field_with_version_specific_path_mappings: diag(6206, ts.DiagnosticCategory.Message, "package_json_has_a_typesVersions_field_with_version_specific_path_mappings_6206", "'package.json' has a 'typesVersions' field with version-specific path mappings."),
         package_json_does_not_have_a_typesVersions_entry_that_matches_version_0: diag(6207, ts.DiagnosticCategory.Message, "package_json_does_not_have_a_typesVersions_entry_that_matches_version_0_6207", "'package.json' does not have a 'typesVersions' entry that matches version '{0}'."),
         package_json_has_a_typesVersions_entry_0_that_matches_compiler_version_1_looking_for_a_pattern_to_match_module_name_2: diag(6208, ts.DiagnosticCategory.Message, "package_json_has_a_typesVersions_entry_0_that_matches_compiler_version_1_looking_for_a_pattern_to_ma_6208", "'package.json' has a 'typesVersions' entry '{0}' that matches compiler version '{1}', looking for a pattern to match module name '{2}'."),
@@ -7901,6 +8817,11 @@ var ts;
         Could_not_resolve_the_path_0_with_the_extensions_Colon_1: diag(6231, ts.DiagnosticCategory.Error, "Could_not_resolve_the_path_0_with_the_extensions_Colon_1_6231", "Could not resolve the path '{0}' with the extensions: {1}."),
         Declaration_augments_declaration_in_another_file_This_cannot_be_serialized: diag(6232, ts.DiagnosticCategory.Error, "Declaration_augments_declaration_in_another_file_This_cannot_be_serialized_6232", "Declaration augments declaration in another file. This cannot be serialized."),
         This_is_the_declaration_being_augmented_Consider_moving_the_augmenting_declaration_into_the_same_file: diag(6233, ts.DiagnosticCategory.Error, "This_is_the_declaration_being_augmented_Consider_moving_the_augmenting_declaration_into_the_same_fil_6233", "This is the declaration being augmented. Consider moving the augmenting declaration into the same file."),
+        This_expression_is_not_callable_because_it_is_a_get_accessor_Did_you_mean_to_use_it_without: diag(6234, ts.DiagnosticCategory.Error, "This_expression_is_not_callable_because_it_is_a_get_accessor_Did_you_mean_to_use_it_without_6234", "This expression is not callable because it is a 'get' accessor. Did you mean to use it without '()'?"),
+        Disable_loading_referenced_projects: diag(6235, ts.DiagnosticCategory.Message, "Disable_loading_referenced_projects_6235", "Disable loading referenced projects."),
+        Arguments_for_the_rest_parameter_0_were_not_provided: diag(6236, ts.DiagnosticCategory.Error, "Arguments_for_the_rest_parameter_0_were_not_provided_6236", "Arguments for the rest parameter '{0}' were not provided."),
+        Generates_an_event_trace_and_a_list_of_types: diag(6237, ts.DiagnosticCategory.Message, "Generates_an_event_trace_and_a_list_of_types_6237", "Generates an event trace and a list of types."),
+        Specify_the_module_specifier_to_be_used_to_import_the_jsx_and_jsxs_factory_functions_from_eg_react: diag(6238, ts.DiagnosticCategory.Error, "Specify_the_module_specifier_to_be_used_to_import_the_jsx_and_jsxs_factory_functions_from_eg_react_6238", "Specify the module specifier to be used to import the `jsx` and `jsxs` factory functions from. eg, react"),
         Projects_to_reference: diag(6300, ts.DiagnosticCategory.Message, "Projects_to_reference_6300", "Projects to reference"),
         Enable_project_compilation: diag(6302, ts.DiagnosticCategory.Message, "Enable_project_compilation_6302", "Enable project compilation"),
         Composite_projects_may_not_disable_declaration_emit: diag(6304, ts.DiagnosticCategory.Error, "Composite_projects_may_not_disable_declaration_emit_6304", "Composite projects may not disable declaration emit."),
@@ -7909,6 +8830,7 @@ var ts;
         File_0_is_not_listed_within_the_file_list_of_project_1_Projects_must_list_all_files_or_use_an_include_pattern: diag(6307, ts.DiagnosticCategory.Error, "File_0_is_not_listed_within_the_file_list_of_project_1_Projects_must_list_all_files_or_use_an_includ_6307", "File '{0}' is not listed within the file list of project '{1}'. Projects must list all files or use an 'include' pattern."),
         Cannot_prepend_project_0_because_it_does_not_have_outFile_set: diag(6308, ts.DiagnosticCategory.Error, "Cannot_prepend_project_0_because_it_does_not_have_outFile_set_6308", "Cannot prepend project '{0}' because it does not have 'outFile' set"),
         Output_file_0_from_project_1_does_not_exist: diag(6309, ts.DiagnosticCategory.Error, "Output_file_0_from_project_1_does_not_exist_6309", "Output file '{0}' from project '{1}' does not exist"),
+        Referenced_project_0_may_not_disable_emit: diag(6310, ts.DiagnosticCategory.Error, "Referenced_project_0_may_not_disable_emit_6310", "Referenced project '{0}' may not disable emit."),
         Project_0_is_out_of_date_because_oldest_output_1_is_older_than_newest_input_2: diag(6350, ts.DiagnosticCategory.Message, "Project_0_is_out_of_date_because_oldest_output_1_is_older_than_newest_input_2_6350", "Project '{0}' is out of date because oldest output '{1}' is older than newest input '{2}'"),
         Project_0_is_up_to_date_because_newest_input_1_is_older_than_oldest_output_2: diag(6351, ts.DiagnosticCategory.Message, "Project_0_is_up_to_date_because_newest_input_1_is_older_than_oldest_output_2_6351", "Project '{0}' is up to date because newest input '{1}' is older than oldest output '{2}'"),
         Project_0_is_out_of_date_because_output_file_1_does_not_exist: diag(6352, ts.DiagnosticCategory.Message, "Project_0_is_out_of_date_because_output_file_1_does_not_exist_6352", "Project '{0}' is out of date because output file '{1}' does not exist"),
@@ -7944,11 +8866,14 @@ var ts;
         Skipping_build_of_project_0_because_its_dependency_1_was_not_built: diag(6382, ts.DiagnosticCategory.Message, "Skipping_build_of_project_0_because_its_dependency_1_was_not_built_6382", "Skipping build of project '{0}' because its dependency '{1}' was not built"),
         Project_0_can_t_be_built_because_its_dependency_1_was_not_built: diag(6383, ts.DiagnosticCategory.Message, "Project_0_can_t_be_built_because_its_dependency_1_was_not_built_6383", "Project '{0}' can't be built because its dependency '{1}' was not built"),
         Have_recompiles_in_incremental_and_watch_assume_that_changes_within_a_file_will_only_affect_files_directly_depending_on_it: diag(6384, ts.DiagnosticCategory.Message, "Have_recompiles_in_incremental_and_watch_assume_that_changes_within_a_file_will_only_affect_files_di_6384", "Have recompiles in '--incremental' and '--watch' assume that changes within a file will only affect files directly depending on it."),
+        _0_is_deprecated: diag(6385, ts.DiagnosticCategory.Suggestion, "_0_is_deprecated_6385", "'{0}' is deprecated", /*reportsUnnecessary*/ undefined, /*elidedInCompatabilityPyramid*/ undefined, /*reportsDeprecated*/ true),
+        Performance_timings_for_diagnostics_or_extendedDiagnostics_are_not_available_in_this_session_A_native_implementation_of_the_Web_Performance_API_could_not_be_found: diag(6386, ts.DiagnosticCategory.Message, "Performance_timings_for_diagnostics_or_extendedDiagnostics_are_not_available_in_this_session_A_nativ_6386", "Performance timings for '--diagnostics' or '--extendedDiagnostics' are not available in this session. A native implementation of the Web Performance API could not be found."),
         The_expected_type_comes_from_property_0_which_is_declared_here_on_type_1: diag(6500, ts.DiagnosticCategory.Message, "The_expected_type_comes_from_property_0_which_is_declared_here_on_type_1_6500", "The expected type comes from property '{0}' which is declared here on type '{1}'"),
         The_expected_type_comes_from_this_index_signature: diag(6501, ts.DiagnosticCategory.Message, "The_expected_type_comes_from_this_index_signature_6501", "The expected type comes from this index signature."),
         The_expected_type_comes_from_the_return_type_of_this_signature: diag(6502, ts.DiagnosticCategory.Message, "The_expected_type_comes_from_the_return_type_of_this_signature_6502", "The expected type comes from the return type of this signature."),
         Print_names_of_files_that_are_part_of_the_compilation_and_then_stop_processing: diag(6503, ts.DiagnosticCategory.Message, "Print_names_of_files_that_are_part_of_the_compilation_and_then_stop_processing_6503", "Print names of files that are part of the compilation and then stop processing."),
         File_0_is_a_JavaScript_file_Did_you_mean_to_enable_the_allowJs_option: diag(6504, ts.DiagnosticCategory.Error, "File_0_is_a_JavaScript_file_Did_you_mean_to_enable_the_allowJs_option_6504", "File '{0}' is a JavaScript file. Did you mean to enable the 'allowJs' option?"),
+        Include_undefined_in_index_signature_results: diag(6800, ts.DiagnosticCategory.Message, "Include_undefined_in_index_signature_results_6800", "Include 'undefined' in index signature results"),
         Variable_0_implicitly_has_an_1_type: diag(7005, ts.DiagnosticCategory.Error, "Variable_0_implicitly_has_an_1_type_7005", "Variable '{0}' implicitly has an '{1}' type."),
         Parameter_0_implicitly_has_an_1_type: diag(7006, ts.DiagnosticCategory.Error, "Parameter_0_implicitly_has_an_1_type_7006", "Parameter '{0}' implicitly has an '{1}' type."),
         Member_0_implicitly_has_an_1_type: diag(7008, ts.DiagnosticCategory.Error, "Member_0_implicitly_has_an_1_type_7008", "Member '{0}' implicitly has an '{1}' type."),
@@ -7976,7 +8901,7 @@ var ts;
         Property_0_implicitly_has_type_any_because_its_set_accessor_lacks_a_parameter_type_annotation: diag(7032, ts.DiagnosticCategory.Error, "Property_0_implicitly_has_type_any_because_its_set_accessor_lacks_a_parameter_type_annotation_7032", "Property '{0}' implicitly has type 'any', because its set accessor lacks a parameter type annotation."),
         Property_0_implicitly_has_type_any_because_its_get_accessor_lacks_a_return_type_annotation: diag(7033, ts.DiagnosticCategory.Error, "Property_0_implicitly_has_type_any_because_its_get_accessor_lacks_a_return_type_annotation_7033", "Property '{0}' implicitly has type 'any', because its get accessor lacks a return type annotation."),
         Variable_0_implicitly_has_type_1_in_some_locations_where_its_type_cannot_be_determined: diag(7034, ts.DiagnosticCategory.Error, "Variable_0_implicitly_has_type_1_in_some_locations_where_its_type_cannot_be_determined_7034", "Variable '{0}' implicitly has type '{1}' in some locations where its type cannot be determined."),
-        Try_npm_install_types_Slash_1_if_it_exists_or_add_a_new_declaration_d_ts_file_containing_declare_module_0: diag(7035, ts.DiagnosticCategory.Error, "Try_npm_install_types_Slash_1_if_it_exists_or_add_a_new_declaration_d_ts_file_containing_declare_mod_7035", "Try `npm install @types/{1}` if it exists or add a new declaration (.d.ts) file containing `declare module '{0}';`"),
+        Try_npm_i_save_dev_types_Slash_1_if_it_exists_or_add_a_new_declaration_d_ts_file_containing_declare_module_0: diag(7035, ts.DiagnosticCategory.Error, "Try_npm_i_save_dev_types_Slash_1_if_it_exists_or_add_a_new_declaration_d_ts_file_containing_declare__7035", "Try `npm i --save-dev @types/{1}` if it exists or add a new declaration (.d.ts) file containing `declare module '{0}';`"),
         Dynamic_import_s_specifier_must_be_of_type_string_but_here_has_type_0: diag(7036, ts.DiagnosticCategory.Error, "Dynamic_import_s_specifier_must_be_of_type_string_but_here_has_type_0_7036", "Dynamic import's specifier must be of type 'string', but here has type '{0}'."),
         Enables_emit_interoperability_between_CommonJS_and_ES_Modules_via_creation_of_namespace_objects_for_all_imports_Implies_allowSyntheticDefaultImports: diag(7037, ts.DiagnosticCategory.Message, "Enables_emit_interoperability_between_CommonJS_and_ES_Modules_via_creation_of_namespace_objects_for__7037", "Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'."),
         Type_originates_at_this_import_A_namespace_style_import_cannot_be_called_or_constructed_and_will_cause_a_failure_at_runtime_Consider_using_a_default_import_or_import_require_here_instead: diag(7038, ts.DiagnosticCategory.Message, "Type_originates_at_this_import_A_namespace_style_import_cannot_be_called_or_constructed_and_will_cau_7038", "Type originates at this import. A namespace-style import cannot be called or constructed, and will cause a failure at runtime. Consider using a default import or import require here instead."),
@@ -7997,6 +8922,7 @@ var ts;
         Element_implicitly_has_an_any_type_because_expression_of_type_0_can_t_be_used_to_index_type_1: diag(7053, ts.DiagnosticCategory.Error, "Element_implicitly_has_an_any_type_because_expression_of_type_0_can_t_be_used_to_index_type_1_7053", "Element implicitly has an 'any' type because expression of type '{0}' can't be used to index type '{1}'."),
         No_index_signature_with_a_parameter_of_type_0_was_found_on_type_1: diag(7054, ts.DiagnosticCategory.Error, "No_index_signature_with_a_parameter_of_type_0_was_found_on_type_1_7054", "No index signature with a parameter of type '{0}' was found on type '{1}'."),
         _0_which_lacks_return_type_annotation_implicitly_has_an_1_yield_type: diag(7055, ts.DiagnosticCategory.Error, "_0_which_lacks_return_type_annotation_implicitly_has_an_1_yield_type_7055", "'{0}', which lacks return-type annotation, implicitly has an '{1}' yield type."),
+        The_inferred_type_of_this_node_exceeds_the_maximum_length_the_compiler_will_serialize_An_explicit_type_annotation_is_needed: diag(7056, ts.DiagnosticCategory.Error, "The_inferred_type_of_this_node_exceeds_the_maximum_length_the_compiler_will_serialize_An_explicit_ty_7056", "The inferred type of this node exceeds the maximum length the compiler will serialize. An explicit type annotation is needed."),
         You_cannot_rename_this_element: diag(8000, ts.DiagnosticCategory.Error, "You_cannot_rename_this_element_8000", "You cannot rename this element."),
         You_cannot_rename_elements_that_are_defined_in_the_standard_TypeScript_library: diag(8001, ts.DiagnosticCategory.Error, "You_cannot_rename_elements_that_are_defined_in_the_standard_TypeScript_library_8001", "You cannot rename elements that are defined in the standard TypeScript library."),
         import_can_only_be_used_in_TypeScript_files: diag(8002, ts.DiagnosticCategory.Error, "import_can_only_be_used_in_TypeScript_files_8002", "'import ... =' can only be used in TypeScript files."),
@@ -8027,6 +8953,8 @@ var ts;
         The_type_of_a_function_declaration_must_match_the_function_s_signature: diag(8030, ts.DiagnosticCategory.Error, "The_type_of_a_function_declaration_must_match_the_function_s_signature_8030", "The type of a function declaration must match the function's signature."),
         You_cannot_rename_a_module_via_a_global_import: diag(8031, ts.DiagnosticCategory.Error, "You_cannot_rename_a_module_via_a_global_import_8031", "You cannot rename a module via a global import."),
         Qualified_name_0_is_not_allowed_without_a_leading_param_object_1: diag(8032, ts.DiagnosticCategory.Error, "Qualified_name_0_is_not_allowed_without_a_leading_param_object_1_8032", "Qualified name '{0}' is not allowed without a leading '@param {object} {1}'."),
+        A_JSDoc_typedef_comment_may_not_contain_multiple_type_tags: diag(8033, ts.DiagnosticCategory.Error, "A_JSDoc_typedef_comment_may_not_contain_multiple_type_tags_8033", "A JSDoc '@typedef' comment may not contain multiple '@type' tags."),
+        The_tag_was_first_specified_here: diag(8034, ts.DiagnosticCategory.Error, "The_tag_was_first_specified_here_8034", "The tag was first specified here."),
         Only_identifiers_Slashqualified_names_with_optional_type_arguments_are_currently_supported_in_a_class_extends_clause: diag(9002, ts.DiagnosticCategory.Error, "Only_identifiers_Slashqualified_names_with_optional_type_arguments_are_currently_supported_in_a_clas_9002", "Only identifiers/qualified-names with optional type arguments are currently supported in a class 'extends' clause."),
         class_expressions_are_not_currently_supported: diag(9003, ts.DiagnosticCategory.Error, "class_expressions_are_not_currently_supported_9003", "'class' expressions are not currently supported."),
         Language_service_is_disabled: diag(9004, ts.DiagnosticCategory.Error, "Language_service_is_disabled_9004", "Language service is disabled."),
@@ -8048,8 +8976,8 @@ var ts;
         Meta_property_0_is_only_allowed_in_the_body_of_a_function_declaration_function_expression_or_constructor: diag(17013, ts.DiagnosticCategory.Error, "Meta_property_0_is_only_allowed_in_the_body_of_a_function_declaration_function_expression_or_constru_17013", "Meta-property '{0}' is only allowed in the body of a function declaration, function expression, or constructor."),
         JSX_fragment_has_no_corresponding_closing_tag: diag(17014, ts.DiagnosticCategory.Error, "JSX_fragment_has_no_corresponding_closing_tag_17014", "JSX fragment has no corresponding closing tag."),
         Expected_corresponding_closing_tag_for_JSX_fragment: diag(17015, ts.DiagnosticCategory.Error, "Expected_corresponding_closing_tag_for_JSX_fragment_17015", "Expected corresponding closing tag for JSX fragment."),
-        JSX_fragment_is_not_supported_when_using_jsxFactory: diag(17016, ts.DiagnosticCategory.Error, "JSX_fragment_is_not_supported_when_using_jsxFactory_17016", "JSX fragment is not supported when using --jsxFactory"),
-        JSX_fragment_is_not_supported_when_using_an_inline_JSX_factory_pragma: diag(17017, ts.DiagnosticCategory.Error, "JSX_fragment_is_not_supported_when_using_an_inline_JSX_factory_pragma_17017", "JSX fragment is not supported when using an inline JSX factory pragma"),
+        The_jsxFragmentFactory_compiler_option_must_be_provided_to_use_JSX_fragments_with_the_jsxFactory_compiler_option: diag(17016, ts.DiagnosticCategory.Error, "The_jsxFragmentFactory_compiler_option_must_be_provided_to_use_JSX_fragments_with_the_jsxFactory_com_17016", "The 'jsxFragmentFactory' compiler option must be provided to use JSX fragments with the 'jsxFactory' compiler option."),
+        An_jsxFrag_pragma_is_required_when_using_an_jsx_pragma_with_JSX_fragments: diag(17017, ts.DiagnosticCategory.Error, "An_jsxFrag_pragma_is_required_when_using_an_jsx_pragma_with_JSX_fragments_17017", "An @jsxFrag pragma is required when using an @jsx pragma with JSX fragments."),
         Unknown_type_acquisition_option_0_Did_you_mean_1: diag(17018, ts.DiagnosticCategory.Error, "Unknown_type_acquisition_option_0_Did_you_mean_1_17018", "Unknown type acquisition option '{0}'. Did you mean '{1}'?"),
         Circularity_detected_while_resolving_configuration_Colon_0: diag(18000, ts.DiagnosticCategory.Error, "Circularity_detected_while_resolving_configuration_Colon_0_18000", "Circularity detected while resolving configuration: {0}"),
         A_path_in_an_extends_option_must_be_relative_or_rooted_but_0_is_not: diag(18001, ts.DiagnosticCategory.Error, "A_path_in_an_extends_option_must_be_relative_or_rooted_but_0_is_not_18001", "A path in an 'extends' option must be relative or rooted, but '{0}' is not."),
@@ -8071,7 +8999,6 @@ var ts;
         Implement_interface_0: diag(90006, ts.DiagnosticCategory.Message, "Implement_interface_0_90006", "Implement interface '{0}'"),
         Implement_inherited_abstract_class: diag(90007, ts.DiagnosticCategory.Message, "Implement_inherited_abstract_class_90007", "Implement inherited abstract class"),
         Add_0_to_unresolved_variable: diag(90008, ts.DiagnosticCategory.Message, "Add_0_to_unresolved_variable_90008", "Add '{0}.' to unresolved variable"),
-        Remove_destructuring: diag(90009, ts.DiagnosticCategory.Message, "Remove_destructuring_90009", "Remove destructuring"),
         Remove_variable_statement: diag(90010, ts.DiagnosticCategory.Message, "Remove_variable_statement_90010", "Remove variable statement"),
         Remove_template_tag: diag(90011, ts.DiagnosticCategory.Message, "Remove_template_tag_90011", "Remove template tag"),
         Remove_type_parameters: diag(90012, ts.DiagnosticCategory.Message, "Remove_type_parameters_90012", "Remove type parameters"),
@@ -8098,9 +9025,15 @@ var ts;
         Add_default_import_0_to_existing_import_declaration_from_1: diag(90033, ts.DiagnosticCategory.Message, "Add_default_import_0_to_existing_import_declaration_from_1_90033", "Add default import '{0}' to existing import declaration from \"{1}\""),
         Add_parameter_name: diag(90034, ts.DiagnosticCategory.Message, "Add_parameter_name_90034", "Add parameter name"),
         Declare_private_property_0: diag(90035, ts.DiagnosticCategory.Message, "Declare_private_property_0_90035", "Declare private property '{0}'"),
+        Replace_0_with_Promise_1: diag(90036, ts.DiagnosticCategory.Message, "Replace_0_with_Promise_1_90036", "Replace '{0}' with 'Promise<{1}>'"),
+        Fix_all_incorrect_return_type_of_an_async_functions: diag(90037, ts.DiagnosticCategory.Message, "Fix_all_incorrect_return_type_of_an_async_functions_90037", "Fix all incorrect return type of an async functions"),
+        Declare_private_method_0: diag(90038, ts.DiagnosticCategory.Message, "Declare_private_method_0_90038", "Declare private method '{0}'"),
+        Remove_unused_destructuring_declaration: diag(90039, ts.DiagnosticCategory.Message, "Remove_unused_destructuring_declaration_90039", "Remove unused destructuring declaration"),
+        Remove_unused_declarations_for_Colon_0: diag(90041, ts.DiagnosticCategory.Message, "Remove_unused_declarations_for_Colon_0_90041", "Remove unused declarations for: '{0}'"),
         Declare_a_private_field_named_0: diag(90053, ts.DiagnosticCategory.Message, "Declare_a_private_field_named_0_90053", "Declare a private field named '{0}'."),
         Convert_function_to_an_ES2015_class: diag(95001, ts.DiagnosticCategory.Message, "Convert_function_to_an_ES2015_class_95001", "Convert function to an ES2015 class"),
         Convert_function_0_to_class: diag(95002, ts.DiagnosticCategory.Message, "Convert_function_0_to_class_95002", "Convert function '{0}' to class"),
+        Convert_0_to_1_in_0: diag(95003, ts.DiagnosticCategory.Message, "Convert_0_to_1_in_0_95003", "Convert '{0}' to '{1} in {0}'"),
         Extract_to_0_in_1: diag(95004, ts.DiagnosticCategory.Message, "Extract_to_0_in_1_95004", "Extract to {0} in {1}"),
         Extract_function: diag(95005, ts.DiagnosticCategory.Message, "Extract_function_95005", "Extract function"),
         Extract_constant: diag(95006, ts.DiagnosticCategory.Message, "Extract_constant_95006", "Extract constant"),
@@ -8118,6 +9051,7 @@ var ts;
         Add_undefined_type_to_property_0: diag(95018, ts.DiagnosticCategory.Message, "Add_undefined_type_to_property_0_95018", "Add 'undefined' type to property '{0}'"),
         Add_initializer_to_property_0: diag(95019, ts.DiagnosticCategory.Message, "Add_initializer_to_property_0_95019", "Add initializer to property '{0}'"),
         Add_definite_assignment_assertion_to_property_0: diag(95020, ts.DiagnosticCategory.Message, "Add_definite_assignment_assertion_to_property_0_95020", "Add definite assignment assertion to property '{0}'"),
+        Convert_all_type_literals_to_mapped_type: diag(95021, ts.DiagnosticCategory.Message, "Convert_all_type_literals_to_mapped_type_95021", "Convert all type literals to mapped type"),
         Add_all_missing_members: diag(95022, ts.DiagnosticCategory.Message, "Add_all_missing_members_95022", "Add all missing members"),
         Infer_all_types_from_usage: diag(95023, ts.DiagnosticCategory.Message, "Infer_all_types_from_usage_95023", "Infer all types from usage"),
         Delete_all_unused_declarations: diag(95024, ts.DiagnosticCategory.Message, "Delete_all_unused_declarations_95024", "Delete all unused declarations"),
@@ -8208,15 +9142,43 @@ var ts;
         Wrap_all_invalid_characters_in_an_expression_container: diag(95109, ts.DiagnosticCategory.Message, "Wrap_all_invalid_characters_in_an_expression_container_95109", "Wrap all invalid characters in an expression container"),
         Visit_https_Colon_Slash_Slashaka_ms_Slashtsconfig_json_to_read_more_about_this_file: diag(95110, ts.DiagnosticCategory.Message, "Visit_https_Colon_Slash_Slashaka_ms_Slashtsconfig_json_to_read_more_about_this_file_95110", "Visit https://aka.ms/tsconfig.json to read more about this file"),
         Add_a_return_statement: diag(95111, ts.DiagnosticCategory.Message, "Add_a_return_statement_95111", "Add a return statement"),
-        Remove_block_body_braces: diag(95112, ts.DiagnosticCategory.Message, "Remove_block_body_braces_95112", "Remove block body braces"),
+        Remove_braces_from_arrow_function_body: diag(95112, ts.DiagnosticCategory.Message, "Remove_braces_from_arrow_function_body_95112", "Remove braces from arrow function body"),
         Wrap_the_following_body_with_parentheses_which_should_be_an_object_literal: diag(95113, ts.DiagnosticCategory.Message, "Wrap_the_following_body_with_parentheses_which_should_be_an_object_literal_95113", "Wrap the following body with parentheses which should be an object literal"),
         Add_all_missing_return_statement: diag(95114, ts.DiagnosticCategory.Message, "Add_all_missing_return_statement_95114", "Add all missing return statement"),
-        Remove_all_incorrect_body_block_braces: diag(95115, ts.DiagnosticCategory.Message, "Remove_all_incorrect_body_block_braces_95115", "Remove all incorrect body block braces"),
+        Remove_braces_from_all_arrow_function_bodies_with_relevant_issues: diag(95115, ts.DiagnosticCategory.Message, "Remove_braces_from_all_arrow_function_bodies_with_relevant_issues_95115", "Remove braces from all arrow function bodies with relevant issues"),
         Wrap_all_object_literal_with_parentheses: diag(95116, ts.DiagnosticCategory.Message, "Wrap_all_object_literal_with_parentheses_95116", "Wrap all object literal with parentheses"),
+        Move_labeled_tuple_element_modifiers_to_labels: diag(95117, ts.DiagnosticCategory.Message, "Move_labeled_tuple_element_modifiers_to_labels_95117", "Move labeled tuple element modifiers to labels"),
+        Convert_overload_list_to_single_signature: diag(95118, ts.DiagnosticCategory.Message, "Convert_overload_list_to_single_signature_95118", "Convert overload list to single signature"),
+        Generate_get_and_set_accessors_for_all_overriding_properties: diag(95119, ts.DiagnosticCategory.Message, "Generate_get_and_set_accessors_for_all_overriding_properties_95119", "Generate 'get' and 'set' accessors for all overriding properties"),
+        Wrap_in_JSX_fragment: diag(95120, ts.DiagnosticCategory.Message, "Wrap_in_JSX_fragment_95120", "Wrap in JSX fragment"),
+        Wrap_all_unparented_JSX_in_JSX_fragment: diag(95121, ts.DiagnosticCategory.Message, "Wrap_all_unparented_JSX_in_JSX_fragment_95121", "Wrap all unparented JSX in JSX fragment"),
+        Convert_arrow_function_or_function_expression: diag(95122, ts.DiagnosticCategory.Message, "Convert_arrow_function_or_function_expression_95122", "Convert arrow function or function expression"),
+        Convert_to_anonymous_function: diag(95123, ts.DiagnosticCategory.Message, "Convert_to_anonymous_function_95123", "Convert to anonymous function"),
+        Convert_to_named_function: diag(95124, ts.DiagnosticCategory.Message, "Convert_to_named_function_95124", "Convert to named function"),
+        Convert_to_arrow_function: diag(95125, ts.DiagnosticCategory.Message, "Convert_to_arrow_function_95125", "Convert to arrow function"),
+        Remove_parentheses: diag(95126, ts.DiagnosticCategory.Message, "Remove_parentheses_95126", "Remove parentheses"),
+        Could_not_find_a_containing_arrow_function: diag(95127, ts.DiagnosticCategory.Message, "Could_not_find_a_containing_arrow_function_95127", "Could not find a containing arrow function"),
+        Containing_function_is_not_an_arrow_function: diag(95128, ts.DiagnosticCategory.Message, "Containing_function_is_not_an_arrow_function_95128", "Containing function is not an arrow function"),
+        Could_not_find_export_statement: diag(95129, ts.DiagnosticCategory.Message, "Could_not_find_export_statement_95129", "Could not find export statement"),
+        This_file_already_has_a_default_export: diag(95130, ts.DiagnosticCategory.Message, "This_file_already_has_a_default_export_95130", "This file already has a default export"),
+        Could_not_find_import_clause: diag(95131, ts.DiagnosticCategory.Message, "Could_not_find_import_clause_95131", "Could not find import clause"),
+        Could_not_find_namespace_import_or_named_imports: diag(95132, ts.DiagnosticCategory.Message, "Could_not_find_namespace_import_or_named_imports_95132", "Could not find namespace import or named imports"),
+        Selection_is_not_a_valid_type_node: diag(95133, ts.DiagnosticCategory.Message, "Selection_is_not_a_valid_type_node_95133", "Selection is not a valid type node"),
+        No_type_could_be_extracted_from_this_type_node: diag(95134, ts.DiagnosticCategory.Message, "No_type_could_be_extracted_from_this_type_node_95134", "No type could be extracted from this type node"),
+        Could_not_find_property_for_which_to_generate_accessor: diag(95135, ts.DiagnosticCategory.Message, "Could_not_find_property_for_which_to_generate_accessor_95135", "Could not find property for which to generate accessor"),
+        Name_is_not_valid: diag(95136, ts.DiagnosticCategory.Message, "Name_is_not_valid_95136", "Name is not valid"),
+        Can_only_convert_property_with_modifier: diag(95137, ts.DiagnosticCategory.Message, "Can_only_convert_property_with_modifier_95137", "Can only convert property with modifier"),
+        Switch_each_misused_0_to_1: diag(95138, ts.DiagnosticCategory.Message, "Switch_each_misused_0_to_1_95138", "Switch each misused '{0}' to '{1}'"),
+        Convert_to_optional_chain_expression: diag(95139, ts.DiagnosticCategory.Message, "Convert_to_optional_chain_expression_95139", "Convert to optional chain expression"),
+        Could_not_find_convertible_access_expression: diag(95140, ts.DiagnosticCategory.Message, "Could_not_find_convertible_access_expression_95140", "Could not find convertible access expression"),
+        Could_not_find_matching_access_expressions: diag(95141, ts.DiagnosticCategory.Message, "Could_not_find_matching_access_expressions_95141", "Could not find matching access expressions"),
+        Can_only_convert_logical_AND_access_chains: diag(95142, ts.DiagnosticCategory.Message, "Can_only_convert_logical_AND_access_chains_95142", "Can only convert logical AND access chains"),
+        Add_void_to_Promise_resolved_without_a_value: diag(95143, ts.DiagnosticCategory.Message, "Add_void_to_Promise_resolved_without_a_value_95143", "Add 'void' to Promise resolved without a value"),
+        Add_void_to_all_Promises_resolved_without_a_value: diag(95144, ts.DiagnosticCategory.Message, "Add_void_to_all_Promises_resolved_without_a_value_95144", "Add 'void' to all Promises resolved without a value"),
         No_value_exists_in_scope_for_the_shorthand_property_0_Either_declare_one_or_provide_an_initializer: diag(18004, ts.DiagnosticCategory.Error, "No_value_exists_in_scope_for_the_shorthand_property_0_Either_declare_one_or_provide_an_initializer_18004", "No value exists in scope for the shorthand property '{0}'. Either declare one or provide an initializer."),
         Classes_may_not_have_a_field_named_constructor: diag(18006, ts.DiagnosticCategory.Error, "Classes_may_not_have_a_field_named_constructor_18006", "Classes may not have a field named 'constructor'."),
         JSX_expressions_may_not_use_the_comma_operator_Did_you_mean_to_write_an_array: diag(18007, ts.DiagnosticCategory.Error, "JSX_expressions_may_not_use_the_comma_operator_Did_you_mean_to_write_an_array_18007", "JSX expressions may not use the comma operator. Did you mean to write an array?"),
-        Private_identifiers_cannot_be_used_as_parameters: diag(18009, ts.DiagnosticCategory.Error, "Private_identifiers_cannot_be_used_as_parameters_18009", "Private identifiers cannot be used as parameters"),
+        Private_identifiers_cannot_be_used_as_parameters: diag(18009, ts.DiagnosticCategory.Error, "Private_identifiers_cannot_be_used_as_parameters_18009", "Private identifiers cannot be used as parameters."),
         An_accessibility_modifier_cannot_be_used_with_a_private_identifier: diag(18010, ts.DiagnosticCategory.Error, "An_accessibility_modifier_cannot_be_used_with_a_private_identifier_18010", "An accessibility modifier cannot be used with a private identifier."),
         The_operand_of_a_delete_operator_cannot_be_a_private_identifier: diag(18011, ts.DiagnosticCategory.Error, "The_operand_of_a_delete_operator_cannot_be_a_private_identifier_18011", "The operand of a 'delete' operator cannot be a private identifier."),
         constructor_is_a_reserved_word: diag(18012, ts.DiagnosticCategory.Error, "constructor_is_a_reserved_word_18012", "'#constructor' is a reserved word."),
@@ -8226,7 +9188,7 @@ var ts;
         Private_identifiers_are_not_allowed_outside_class_bodies: diag(18016, ts.DiagnosticCategory.Error, "Private_identifiers_are_not_allowed_outside_class_bodies_18016", "Private identifiers are not allowed outside class bodies."),
         The_shadowing_declaration_of_0_is_defined_here: diag(18017, ts.DiagnosticCategory.Error, "The_shadowing_declaration_of_0_is_defined_here_18017", "The shadowing declaration of '{0}' is defined here"),
         The_declaration_of_0_that_you_probably_intended_to_use_is_defined_here: diag(18018, ts.DiagnosticCategory.Error, "The_declaration_of_0_that_you_probably_intended_to_use_is_defined_here_18018", "The declaration of '{0}' that you probably intended to use is defined here"),
-        _0_modifier_cannot_be_used_with_a_private_identifier: diag(18019, ts.DiagnosticCategory.Error, "_0_modifier_cannot_be_used_with_a_private_identifier_18019", "'{0}' modifier cannot be used with a private identifier"),
+        _0_modifier_cannot_be_used_with_a_private_identifier: diag(18019, ts.DiagnosticCategory.Error, "_0_modifier_cannot_be_used_with_a_private_identifier_18019", "'{0}' modifier cannot be used with a private identifier."),
         A_method_cannot_be_named_with_a_private_identifier: diag(18022, ts.DiagnosticCategory.Error, "A_method_cannot_be_named_with_a_private_identifier_18022", "A method cannot be named with a private identifier."),
         An_accessor_cannot_be_named_with_a_private_identifier: diag(18023, ts.DiagnosticCategory.Error, "An_accessor_cannot_be_named_with_a_private_identifier_18023", "An accessor cannot be named with a private identifier."),
         An_enum_member_cannot_be_named_with_a_private_identifier: diag(18024, ts.DiagnosticCategory.Error, "An_enum_member_cannot_be_named_with_a_private_identifier_18024", "An enum member cannot be named with a private identifier."),
@@ -8238,6 +9200,8 @@ var ts;
         The_intersection_0_was_reduced_to_never_because_property_1_has_conflicting_types_in_some_constituents: diag(18031, ts.DiagnosticCategory.Error, "The_intersection_0_was_reduced_to_never_because_property_1_has_conflicting_types_in_some_constituent_18031", "The intersection '{0}' was reduced to 'never' because property '{1}' has conflicting types in some constituents."),
         The_intersection_0_was_reduced_to_never_because_property_1_exists_in_multiple_constituents_and_is_private_in_some: diag(18032, ts.DiagnosticCategory.Error, "The_intersection_0_was_reduced_to_never_because_property_1_exists_in_multiple_constituents_and_is_pr_18032", "The intersection '{0}' was reduced to 'never' because property '{1}' exists in multiple constituents and is private in some."),
         Only_numeric_enums_can_have_computed_members_but_this_expression_has_type_0_If_you_do_not_need_exhaustiveness_checks_consider_using_an_object_literal_instead: diag(18033, ts.DiagnosticCategory.Error, "Only_numeric_enums_can_have_computed_members_but_this_expression_has_type_0_If_you_do_not_need_exhau_18033", "Only numeric enums can have computed members, but this expression has type '{0}'. If you do not need exhaustiveness checks, consider using an object literal instead."),
+        Specify_the_JSX_fragment_factory_function_to_use_when_targeting_react_JSX_emit_with_jsxFactory_compiler_option_is_specified_e_g_Fragment: diag(18034, ts.DiagnosticCategory.Message, "Specify_the_JSX_fragment_factory_function_to_use_when_targeting_react_JSX_emit_with_jsxFactory_compi_18034", "Specify the JSX fragment factory function to use when targeting 'react' JSX emit with 'jsxFactory' compiler option is specified, e.g. 'Fragment'."),
+        Invalid_value_for_jsxFragmentFactory_0_is_not_a_valid_identifier_or_qualified_name: diag(18035, ts.DiagnosticCategory.Error, "Invalid_value_for_jsxFragmentFactory_0_is_not_a_valid_identifier_or_qualified_name_18035", "Invalid value for 'jsxFragmentFactory'. '{0}' is not a valid identifier or qualified-name."),
     };
 })(ts || (ts = {}));
 var ts;
@@ -8245,7 +9209,7 @@ var ts;
     var _a;
     /* @internal */
     function tokenIsIdentifierOrKeyword(token) {
-        return token >= 75 /* Identifier */;
+        return token >= 78 /* Identifier */;
     }
     ts.tokenIsIdentifierOrKeyword = tokenIsIdentifierOrKeyword;
     /* @internal */
@@ -8254,86 +9218,87 @@ var ts;
     }
     ts.tokenIsIdentifierOrKeywordOrGreaterThan = tokenIsIdentifierOrKeywordOrGreaterThan;
     var textToKeywordObj = (_a = {
-            abstract: 122 /* AbstractKeyword */,
-            any: 125 /* AnyKeyword */,
-            as: 123 /* AsKeyword */,
-            asserts: 124 /* AssertsKeyword */,
-            bigint: 151 /* BigIntKeyword */,
-            boolean: 128 /* BooleanKeyword */,
-            break: 77 /* BreakKeyword */,
-            case: 78 /* CaseKeyword */,
-            catch: 79 /* CatchKeyword */,
-            class: 80 /* ClassKeyword */,
-            continue: 82 /* ContinueKeyword */,
-            const: 81 /* ConstKeyword */
+            abstract: 125 /* AbstractKeyword */,
+            any: 128 /* AnyKeyword */,
+            as: 126 /* AsKeyword */,
+            asserts: 127 /* AssertsKeyword */,
+            bigint: 155 /* BigIntKeyword */,
+            boolean: 131 /* BooleanKeyword */,
+            break: 80 /* BreakKeyword */,
+            case: 81 /* CaseKeyword */,
+            catch: 82 /* CatchKeyword */,
+            class: 83 /* ClassKeyword */,
+            continue: 85 /* ContinueKeyword */,
+            const: 84 /* ConstKeyword */
         },
-        _a["" + "constructor"] = 129 /* ConstructorKeyword */,
-        _a.debugger = 83 /* DebuggerKeyword */,
-        _a.declare = 130 /* DeclareKeyword */,
-        _a.default = 84 /* DefaultKeyword */,
-        _a.delete = 85 /* DeleteKeyword */,
-        _a.do = 86 /* DoKeyword */,
-        _a.else = 87 /* ElseKeyword */,
-        _a.enum = 88 /* EnumKeyword */,
-        _a.export = 89 /* ExportKeyword */,
-        _a.extends = 90 /* ExtendsKeyword */,
-        _a.false = 91 /* FalseKeyword */,
-        _a.finally = 92 /* FinallyKeyword */,
-        _a.for = 93 /* ForKeyword */,
-        _a.from = 149 /* FromKeyword */,
-        _a.function = 94 /* FunctionKeyword */,
-        _a.get = 131 /* GetKeyword */,
-        _a.if = 95 /* IfKeyword */,
-        _a.implements = 113 /* ImplementsKeyword */,
-        _a.import = 96 /* ImportKeyword */,
-        _a.in = 97 /* InKeyword */,
-        _a.infer = 132 /* InferKeyword */,
-        _a.instanceof = 98 /* InstanceOfKeyword */,
-        _a.interface = 114 /* InterfaceKeyword */,
-        _a.is = 133 /* IsKeyword */,
-        _a.keyof = 134 /* KeyOfKeyword */,
-        _a.let = 115 /* LetKeyword */,
-        _a.module = 135 /* ModuleKeyword */,
-        _a.namespace = 136 /* NamespaceKeyword */,
-        _a.never = 137 /* NeverKeyword */,
-        _a.new = 99 /* NewKeyword */,
-        _a.null = 100 /* NullKeyword */,
-        _a.number = 140 /* NumberKeyword */,
-        _a.object = 141 /* ObjectKeyword */,
-        _a.package = 116 /* PackageKeyword */,
-        _a.private = 117 /* PrivateKeyword */,
-        _a.protected = 118 /* ProtectedKeyword */,
-        _a.public = 119 /* PublicKeyword */,
-        _a.readonly = 138 /* ReadonlyKeyword */,
-        _a.require = 139 /* RequireKeyword */,
-        _a.global = 150 /* GlobalKeyword */,
-        _a.return = 101 /* ReturnKeyword */,
-        _a.set = 142 /* SetKeyword */,
-        _a.static = 120 /* StaticKeyword */,
-        _a.string = 143 /* StringKeyword */,
-        _a.super = 102 /* SuperKeyword */,
-        _a.switch = 103 /* SwitchKeyword */,
-        _a.symbol = 144 /* SymbolKeyword */,
-        _a.this = 104 /* ThisKeyword */,
-        _a.throw = 105 /* ThrowKeyword */,
-        _a.true = 106 /* TrueKeyword */,
-        _a.try = 107 /* TryKeyword */,
-        _a.type = 145 /* TypeKeyword */,
-        _a.typeof = 108 /* TypeOfKeyword */,
-        _a.undefined = 146 /* UndefinedKeyword */,
-        _a.unique = 147 /* UniqueKeyword */,
-        _a.unknown = 148 /* UnknownKeyword */,
-        _a.var = 109 /* VarKeyword */,
-        _a.void = 110 /* VoidKeyword */,
-        _a.while = 111 /* WhileKeyword */,
-        _a.with = 112 /* WithKeyword */,
-        _a.yield = 121 /* YieldKeyword */,
-        _a.async = 126 /* AsyncKeyword */,
-        _a.await = 127 /* AwaitKeyword */,
-        _a.of = 152 /* OfKeyword */,
+        _a["" + "constructor"] = 132 /* ConstructorKeyword */,
+        _a.debugger = 86 /* DebuggerKeyword */,
+        _a.declare = 133 /* DeclareKeyword */,
+        _a.default = 87 /* DefaultKeyword */,
+        _a.delete = 88 /* DeleteKeyword */,
+        _a.do = 89 /* DoKeyword */,
+        _a.else = 90 /* ElseKeyword */,
+        _a.enum = 91 /* EnumKeyword */,
+        _a.export = 92 /* ExportKeyword */,
+        _a.extends = 93 /* ExtendsKeyword */,
+        _a.false = 94 /* FalseKeyword */,
+        _a.finally = 95 /* FinallyKeyword */,
+        _a.for = 96 /* ForKeyword */,
+        _a.from = 153 /* FromKeyword */,
+        _a.function = 97 /* FunctionKeyword */,
+        _a.get = 134 /* GetKeyword */,
+        _a.if = 98 /* IfKeyword */,
+        _a.implements = 116 /* ImplementsKeyword */,
+        _a.import = 99 /* ImportKeyword */,
+        _a.in = 100 /* InKeyword */,
+        _a.infer = 135 /* InferKeyword */,
+        _a.instanceof = 101 /* InstanceOfKeyword */,
+        _a.interface = 117 /* InterfaceKeyword */,
+        _a.intrinsic = 136 /* IntrinsicKeyword */,
+        _a.is = 137 /* IsKeyword */,
+        _a.keyof = 138 /* KeyOfKeyword */,
+        _a.let = 118 /* LetKeyword */,
+        _a.module = 139 /* ModuleKeyword */,
+        _a.namespace = 140 /* NamespaceKeyword */,
+        _a.never = 141 /* NeverKeyword */,
+        _a.new = 102 /* NewKeyword */,
+        _a.null = 103 /* NullKeyword */,
+        _a.number = 144 /* NumberKeyword */,
+        _a.object = 145 /* ObjectKeyword */,
+        _a.package = 119 /* PackageKeyword */,
+        _a.private = 120 /* PrivateKeyword */,
+        _a.protected = 121 /* ProtectedKeyword */,
+        _a.public = 122 /* PublicKeyword */,
+        _a.readonly = 142 /* ReadonlyKeyword */,
+        _a.require = 143 /* RequireKeyword */,
+        _a.global = 154 /* GlobalKeyword */,
+        _a.return = 104 /* ReturnKeyword */,
+        _a.set = 146 /* SetKeyword */,
+        _a.static = 123 /* StaticKeyword */,
+        _a.string = 147 /* StringKeyword */,
+        _a.super = 105 /* SuperKeyword */,
+        _a.switch = 106 /* SwitchKeyword */,
+        _a.symbol = 148 /* SymbolKeyword */,
+        _a.this = 107 /* ThisKeyword */,
+        _a.throw = 108 /* ThrowKeyword */,
+        _a.true = 109 /* TrueKeyword */,
+        _a.try = 110 /* TryKeyword */,
+        _a.type = 149 /* TypeKeyword */,
+        _a.typeof = 111 /* TypeOfKeyword */,
+        _a.undefined = 150 /* UndefinedKeyword */,
+        _a.unique = 151 /* UniqueKeyword */,
+        _a.unknown = 152 /* UnknownKeyword */,
+        _a.var = 112 /* VarKeyword */,
+        _a.void = 113 /* VoidKeyword */,
+        _a.while = 114 /* WhileKeyword */,
+        _a.with = 115 /* WithKeyword */,
+        _a.yield = 124 /* YieldKeyword */,
+        _a.async = 129 /* AsyncKeyword */,
+        _a.await = 130 /* AwaitKeyword */,
+        _a.of = 156 /* OfKeyword */,
         _a);
-    var textToKeyword = ts.createMapFromTemplate(textToKeywordObj);
-    var textToToken = ts.createMapFromTemplate(__assign(__assign({}, textToKeywordObj), { "{": 18 /* OpenBraceToken */, "}": 19 /* CloseBraceToken */, "(": 20 /* OpenParenToken */, ")": 21 /* CloseParenToken */, "[": 22 /* OpenBracketToken */, "]": 23 /* CloseBracketToken */, ".": 24 /* DotToken */, "...": 25 /* DotDotDotToken */, ";": 26 /* SemicolonToken */, ",": 27 /* CommaToken */, "<": 29 /* LessThanToken */, ">": 31 /* GreaterThanToken */, "<=": 32 /* LessThanEqualsToken */, ">=": 33 /* GreaterThanEqualsToken */, "==": 34 /* EqualsEqualsToken */, "!=": 35 /* ExclamationEqualsToken */, "===": 36 /* EqualsEqualsEqualsToken */, "!==": 37 /* ExclamationEqualsEqualsToken */, "=>": 38 /* EqualsGreaterThanToken */, "+": 39 /* PlusToken */, "-": 40 /* MinusToken */, "**": 42 /* AsteriskAsteriskToken */, "*": 41 /* AsteriskToken */, "/": 43 /* SlashToken */, "%": 44 /* PercentToken */, "++": 45 /* PlusPlusToken */, "--": 46 /* MinusMinusToken */, "<<": 47 /* LessThanLessThanToken */, "</": 30 /* LessThanSlashToken */, ">>": 48 /* GreaterThanGreaterThanToken */, ">>>": 49 /* GreaterThanGreaterThanGreaterThanToken */, "&": 50 /* AmpersandToken */, "|": 51 /* BarToken */, "^": 52 /* CaretToken */, "!": 53 /* ExclamationToken */, "~": 54 /* TildeToken */, "&&": 55 /* AmpersandAmpersandToken */, "||": 56 /* BarBarToken */, "?": 57 /* QuestionToken */, "??": 60 /* QuestionQuestionToken */, "?.": 28 /* QuestionDotToken */, ":": 58 /* ColonToken */, "=": 62 /* EqualsToken */, "+=": 63 /* PlusEqualsToken */, "-=": 64 /* MinusEqualsToken */, "*=": 65 /* AsteriskEqualsToken */, "**=": 66 /* AsteriskAsteriskEqualsToken */, "/=": 67 /* SlashEqualsToken */, "%=": 68 /* PercentEqualsToken */, "<<=": 69 /* LessThanLessThanEqualsToken */, ">>=": 70 /* GreaterThanGreaterThanEqualsToken */, ">>>=": 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */, "&=": 72 /* AmpersandEqualsToken */, "|=": 73 /* BarEqualsToken */, "^=": 74 /* CaretEqualsToken */, "@": 59 /* AtToken */, "`": 61 /* BacktickToken */ }));
+    var textToKeyword = new ts.Map(ts.getEntries(textToKeywordObj));
+    var textToToken = new ts.Map(ts.getEntries(__assign(__assign({}, textToKeywordObj), { "{": 18 /* OpenBraceToken */, "}": 19 /* CloseBraceToken */, "(": 20 /* OpenParenToken */, ")": 21 /* CloseParenToken */, "[": 22 /* OpenBracketToken */, "]": 23 /* CloseBracketToken */, ".": 24 /* DotToken */, "...": 25 /* DotDotDotToken */, ";": 26 /* SemicolonToken */, ",": 27 /* CommaToken */, "<": 29 /* LessThanToken */, ">": 31 /* GreaterThanToken */, "<=": 32 /* LessThanEqualsToken */, ">=": 33 /* GreaterThanEqualsToken */, "==": 34 /* EqualsEqualsToken */, "!=": 35 /* ExclamationEqualsToken */, "===": 36 /* EqualsEqualsEqualsToken */, "!==": 37 /* ExclamationEqualsEqualsToken */, "=>": 38 /* EqualsGreaterThanToken */, "+": 39 /* PlusToken */, "-": 40 /* MinusToken */, "**": 42 /* AsteriskAsteriskToken */, "*": 41 /* AsteriskToken */, "/": 43 /* SlashToken */, "%": 44 /* PercentToken */, "++": 45 /* PlusPlusToken */, "--": 46 /* MinusMinusToken */, "<<": 47 /* LessThanLessThanToken */, "</": 30 /* LessThanSlashToken */, ">>": 48 /* GreaterThanGreaterThanToken */, ">>>": 49 /* GreaterThanGreaterThanGreaterThanToken */, "&": 50 /* AmpersandToken */, "|": 51 /* BarToken */, "^": 52 /* CaretToken */, "!": 53 /* ExclamationToken */, "~": 54 /* TildeToken */, "&&": 55 /* AmpersandAmpersandToken */, "||": 56 /* BarBarToken */, "?": 57 /* QuestionToken */, "??": 60 /* QuestionQuestionToken */, "?.": 28 /* QuestionDotToken */, ":": 58 /* ColonToken */, "=": 62 /* EqualsToken */, "+=": 63 /* PlusEqualsToken */, "-=": 64 /* MinusEqualsToken */, "*=": 65 /* AsteriskEqualsToken */, "**=": 66 /* AsteriskAsteriskEqualsToken */, "/=": 67 /* SlashEqualsToken */, "%=": 68 /* PercentEqualsToken */, "<<=": 69 /* LessThanLessThanEqualsToken */, ">>=": 70 /* GreaterThanGreaterThanEqualsToken */, ">>>=": 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */, "&=": 72 /* AmpersandEqualsToken */, "|=": 73 /* BarEqualsToken */, "^=": 77 /* CaretEqualsToken */, "||=": 74 /* BarBarEqualsToken */, "&&=": 75 /* AmpersandAmpersandEqualsToken */, "??=": 76 /* QuestionQuestionEqualsToken */, "@": 59 /* AtToken */, "`": 61 /* BacktickToken */ })));
     /*
         As per ECMAScript Language Specification 3th Edition, Section 7.6: Identifiers
         IdentifierStart ::
@@ -8996,12 +9961,15 @@ var ts;
             hasUnicodeEscape: function () { return (tokenFlags & 1024 /* UnicodeEscape */) !== 0; },
             hasExtendedUnicodeEscape: function () { return (tokenFlags & 8 /* ExtendedUnicodeEscape */) !== 0; },
             hasPrecedingLineBreak: function () { return (tokenFlags & 1 /* PrecedingLineBreak */) !== 0; },
-            isIdentifier: function () { return token === 75 /* Identifier */ || token > 112 /* LastReservedWord */; },
-            isReservedWord: function () { return token >= 77 /* FirstReservedWord */ && token <= 112 /* LastReservedWord */; },
+            hasPrecedingJSDocComment: function () { return (tokenFlags & 2 /* PrecedingJSDocComment */) !== 0; },
+            isIdentifier: function () { return token === 78 /* Identifier */ || token > 115 /* LastReservedWord */; },
+            isReservedWord: function () { return token >= 80 /* FirstReservedWord */ && token <= 115 /* LastReservedWord */; },
             isUnterminated: function () { return (tokenFlags & 4 /* Unterminated */) !== 0; },
             getCommentDirectives: function () { return commentDirectives; },
+            getNumericLiteralFlags: function () { return tokenFlags & 1008 /* NumericLiteralFlags */; },
             getTokenFlags: function () { return tokenFlags; },
             reScanGreaterToken: reScanGreaterToken,
+            reScanAsteriskEqualsToken: reScanAsteriskEqualsToken,
             reScanSlashToken: reScanSlashToken,
             reScanTemplateToken: reScanTemplateToken,
             reScanTemplateHeadOrNoSubstitutionTemplate: reScanTemplateHeadOrNoSubstitutionTemplate,
@@ -9508,9 +10476,9 @@ var ts;
             return result;
         }
         function getIdentifierToken() {
-            // Reserved words are between 2 and 11 characters long and start with a lowercase letter
+            // Reserved words are between 2 and 12 characters long and start with a lowercase letter
             var len = tokenValue.length;
-            if (len >= 2 && len <= 11) {
+            if (len >= 2 && len <= 12) {
                 var ch = tokenValue.charCodeAt(0);
                 if (ch >= 97 /* a */ && ch <= 122 /* z */) {
                     var keyword = textToKeyword.get(tokenValue);
@@ -9519,7 +10487,7 @@ var ts;
                     }
                 }
             }
-            return token = 75 /* Identifier */;
+            return token = 78 /* Identifier */;
         }
         function scanBinaryOrOctalDigits(base) {
             var value = "";
@@ -9674,6 +10642,9 @@ var ts;
                         return token = 44 /* PercentToken */;
                     case 38 /* ampersand */:
                         if (text.charCodeAt(pos + 1) === 38 /* ampersand */) {
+                            if (text.charCodeAt(pos + 2) === 61 /* equals */) {
+                                return pos += 3, token = 75 /* AmpersandAmpersandEqualsToken */;
+                            }
                             return pos += 2, token = 55 /* AmpersandAmpersandToken */;
                         }
                         if (text.charCodeAt(pos + 1) === 61 /* equals */) {
@@ -9914,15 +10885,16 @@ var ts;
                         pos++;
                         return token = 31 /* GreaterThanToken */;
                     case 63 /* question */:
-                        pos++;
-                        if (text.charCodeAt(pos) === 46 /* dot */ && !isDigit(text.charCodeAt(pos + 1))) {
-                            pos++;
-                            return token = 28 /* QuestionDotToken */;
+                        if (text.charCodeAt(pos + 1) === 46 /* dot */ && !isDigit(text.charCodeAt(pos + 2))) {
+                            return pos += 2, token = 28 /* QuestionDotToken */;
                         }
-                        if (text.charCodeAt(pos) === 63 /* question */) {
-                            pos++;
-                            return token = 60 /* QuestionQuestionToken */;
+                        if (text.charCodeAt(pos + 1) === 63 /* question */) {
+                            if (text.charCodeAt(pos + 2) === 61 /* equals */) {
+                                return pos += 3, token = 76 /* QuestionQuestionEqualsToken */;
+                            }
+                            return pos += 2, token = 60 /* QuestionQuestionToken */;
                         }
+                        pos++;
                         return token = 57 /* QuestionToken */;
                     case 91 /* openBracket */:
                         pos++;
@@ -9932,7 +10904,7 @@ var ts;
                         return token = 23 /* CloseBracketToken */;
                     case 94 /* caret */:
                         if (text.charCodeAt(pos + 1) === 61 /* equals */) {
-                            return pos += 2, token = 74 /* CaretEqualsToken */;
+                            return pos += 2, token = 77 /* CaretEqualsToken */;
                         }
                         pos++;
                         return token = 52 /* CaretToken */;
@@ -9950,6 +10922,9 @@ var ts;
                             }
                         }
                         if (text.charCodeAt(pos + 1) === 124 /* bar */) {
+                            if (text.charCodeAt(pos + 2) === 61 /* equals */) {
+                                return pos += 3, token = 74 /* BarBarEqualsToken */;
+                            }
                             return pos += 2, token = 56 /* BarBarToken */;
                         }
                         if (text.charCodeAt(pos + 1) === 61 /* equals */) {
@@ -10004,7 +10979,7 @@ var ts;
                             tokenValue = "#";
                             error(ts.Diagnostics.Invalid_character);
                         }
-                        return token = 76 /* PrivateIdentifier */;
+                        return token = 79 /* PrivateIdentifier */;
                     default:
                         if (isIdentifierStart(ch, languageVersion)) {
                             pos += charSize(ch);
@@ -10053,6 +11028,11 @@ var ts;
             }
             return token;
         }
+        function reScanAsteriskEqualsToken() {
+            ts.Debug.assert(token === 65 /* AsteriskEqualsToken */, "'reScanAsteriskEqualsToken' should only be called on a '*='");
+            pos = tokenPos + 1;
+            return token = 62 /* EqualsToken */;
+        }
         function reScanSlashToken() {
             if (token === 43 /* SlashToken */ || token === 67 /* SlashEqualsToken */) {
                 var p = tokenPos + 1;
@@ -10278,8 +11258,12 @@ var ts;
                     return token = 5 /* WhitespaceTrivia */;
                 case 64 /* at */:
                     return token = 59 /* AtToken */;
-                case 10 /* lineFeed */:
                 case 13 /* carriageReturn */:
+                    if (text.charCodeAt(pos) === 10 /* lineFeed */) {
+                        pos++;
+                    }
+                // falls through
+                case 10 /* lineFeed */:
                     tokenFlags |= 1 /* PrecedingLineBreak */;
                     return token = 4 /* NewLineTrivia */;
                 case 42 /* asterisk */:
@@ -10697,9 +11681,9 @@ var ts;
     }
     ts.collapseTextChangeRangesAcrossMultipleVersions = collapseTextChangeRangesAcrossMultipleVersions;
     function getTypeParameterOwner(d) {
-        if (d && d.kind === 155 /* TypeParameter */) {
+        if (d && d.kind === 159 /* TypeParameter */) {
             for (var current = d; current; current = current.parent) {
-                if (isFunctionLike(current) || isClassLike(current) || current.kind === 246 /* InterfaceDeclaration */) {
+                if (isFunctionLike(current) || isClassLike(current) || current.kind === 253 /* InterfaceDeclaration */) {
                     return current;
                 }
             }
@@ -10707,7 +11691,7 @@ var ts;
     }
     ts.getTypeParameterOwner = getTypeParameterOwner;
     function isParameterPropertyDeclaration(node, parent) {
-        return ts.hasModifier(node, 92 /* ParameterPropertyModifier */) && parent.kind === 162 /* Constructor */;
+        return ts.hasSyntacticModifier(node, 92 /* ParameterPropertyModifier */) && parent.kind === 166 /* Constructor */;
     }
     ts.isParameterPropertyDeclaration = isParameterPropertyDeclaration;
     function isEmptyBindingPattern(node) {
@@ -10718,7 +11702,7 @@ var ts;
     }
     ts.isEmptyBindingPattern = isEmptyBindingPattern;
     function isEmptyBindingElement(node) {
-        if (isOmittedExpression(node)) {
+        if (ts.isOmittedExpression(node)) {
             return true;
         }
         return isEmptyBindingPattern(node.name);
@@ -10726,33 +11710,38 @@ var ts;
     ts.isEmptyBindingElement = isEmptyBindingElement;
     function walkUpBindingElementsAndPatterns(binding) {
         var node = binding.parent;
-        while (isBindingElement(node.parent)) {
+        while (ts.isBindingElement(node.parent)) {
             node = node.parent.parent;
         }
         return node.parent;
     }
     ts.walkUpBindingElementsAndPatterns = walkUpBindingElementsAndPatterns;
     function getCombinedFlags(node, getFlags) {
-        if (isBindingElement(node)) {
+        if (ts.isBindingElement(node)) {
             node = walkUpBindingElementsAndPatterns(node);
         }
         var flags = getFlags(node);
-        if (node.kind === 242 /* VariableDeclaration */) {
+        if (node.kind === 249 /* VariableDeclaration */) {
             node = node.parent;
         }
-        if (node && node.kind === 243 /* VariableDeclarationList */) {
+        if (node && node.kind === 250 /* VariableDeclarationList */) {
             flags |= getFlags(node);
             node = node.parent;
         }
-        if (node && node.kind === 225 /* VariableStatement */) {
+        if (node && node.kind === 232 /* VariableStatement */) {
             flags |= getFlags(node);
         }
         return flags;
     }
     function getCombinedModifierFlags(node) {
-        return getCombinedFlags(node, ts.getModifierFlags);
+        return getCombinedFlags(node, ts.getEffectiveModifierFlags);
     }
     ts.getCombinedModifierFlags = getCombinedModifierFlags;
+    /* @internal */
+    function getCombinedNodeFlagsAlwaysIncludeJSDoc(node) {
+        return getCombinedFlags(node, ts.getEffectiveModifierFlagsAlwaysIncludeJSDoc);
+    }
+    ts.getCombinedNodeFlagsAlwaysIncludeJSDoc = getCombinedNodeFlagsAlwaysIncludeJSDoc;
     // Returns the node flags for this node and all relevant parent nodes.  This is done so that
     // nodes like variable declarations and binding elements can returned a view of their flags
     // that includes the modifiers from their container.  i.e. flags like export/declare aren't
@@ -10830,6 +11819,20 @@ var ts;
         return !nodeTest || nodeTest(node) ? node : undefined;
     }
     ts.getOriginalNode = getOriginalNode;
+    function findAncestor(node, callback) {
+        while (node) {
+            var result = callback(node);
+            if (result === "quit") {
+                return undefined;
+            }
+            else if (result) {
+                return node;
+            }
+            node = node.parent;
+        }
+        return undefined;
+    }
+    ts.findAncestor = findAncestor;
     /**
      * Gets a value indicating whether a node originated in the parse tree.
      *
@@ -10843,11 +11846,13 @@ var ts;
         if (node === undefined || isParseTreeNode(node)) {
             return node;
         }
-        node = getOriginalNode(node);
-        if (isParseTreeNode(node) && (!nodeTest || nodeTest(node))) {
-            return node;
+        node = node.original;
+        while (node) {
+            if (isParseTreeNode(node)) {
+                return !nodeTest || nodeTest(node) ? node : undefined;
+            }
+            node = node.original;
         }
-        return undefined;
     }
     ts.getParseTreeNode = getParseTreeNode;
     /** Add an extra underscore to identifiers that start with two underscores to avoid issues with magic names like '__proto__' */
@@ -10893,30 +11898,30 @@ var ts;
         }
         // Covers remaining cases (returning undefined if none match).
         switch (hostNode.kind) {
-            case 225 /* VariableStatement */:
+            case 232 /* VariableStatement */:
                 if (hostNode.declarationList && hostNode.declarationList.declarations[0]) {
                     return getDeclarationIdentifier(hostNode.declarationList.declarations[0]);
                 }
                 break;
-            case 226 /* ExpressionStatement */:
+            case 233 /* ExpressionStatement */:
                 var expr = hostNode.expression;
-                if (expr.kind === 209 /* BinaryExpression */ && expr.operatorToken.kind === 62 /* EqualsToken */) {
+                if (expr.kind === 216 /* BinaryExpression */ && expr.operatorToken.kind === 62 /* EqualsToken */) {
                     expr = expr.left;
                 }
                 switch (expr.kind) {
-                    case 194 /* PropertyAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
                         return expr.name;
-                    case 195 /* ElementAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
                         var arg = expr.argumentExpression;
-                        if (isIdentifier(arg)) {
+                        if (ts.isIdentifier(arg)) {
                             return arg;
                         }
                 }
                 break;
-            case 200 /* ParenthesizedExpression */: {
+            case 207 /* ParenthesizedExpression */: {
                 return getDeclarationIdentifier(hostNode.expression);
             }
-            case 238 /* LabeledStatement */: {
+            case 245 /* LabeledStatement */: {
                 if (isDeclaration(hostNode.statement) || isExpression(hostNode.statement)) {
                     return getDeclarationIdentifier(hostNode.statement);
                 }
@@ -10926,14 +11931,14 @@ var ts;
     }
     function getDeclarationIdentifier(node) {
         var name = getNameOfDeclaration(node);
-        return name && isIdentifier(name) ? name : undefined;
+        return name && ts.isIdentifier(name) ? name : undefined;
     }
     /** @internal */
     function nodeHasName(statement, name) {
-        if (isNamedDeclaration(statement) && isIdentifier(statement.name) && idText(statement.name) === idText(name)) {
+        if (isNamedDeclaration(statement) && ts.isIdentifier(statement.name) && idText(statement.name) === idText(name)) {
             return true;
         }
-        if (isVariableStatement(statement) && ts.some(statement.declarationList.declarations, function (d) { return nodeHasName(d, name); })) {
+        if (ts.isVariableStatement(statement) && ts.some(statement.declarationList.declarations, function (d) { return nodeHasName(d, name); })) {
             return true;
         }
         return false;
@@ -10951,18 +11956,18 @@ var ts;
     /** @internal */
     function getNonAssignedNameOfDeclaration(declaration) {
         switch (declaration.kind) {
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 return declaration;
-            case 323 /* JSDocPropertyTag */:
-            case 317 /* JSDocParameterTag */: {
+            case 333 /* JSDocPropertyTag */:
+            case 326 /* JSDocParameterTag */: {
                 var name = declaration.name;
-                if (name.kind === 153 /* QualifiedName */) {
+                if (name.kind === 157 /* QualifiedName */) {
                     return name.right;
                 }
                 break;
             }
-            case 196 /* CallExpression */:
-            case 209 /* BinaryExpression */: {
+            case 203 /* CallExpression */:
+            case 216 /* BinaryExpression */: {
                 var expr_1 = declaration;
                 switch (ts.getAssignmentDeclarationKind(expr_1)) {
                     case 1 /* ExportsProperty */:
@@ -10978,15 +11983,15 @@ var ts;
                         return undefined;
                 }
             }
-            case 322 /* JSDocTypedefTag */:
+            case 331 /* JSDocTypedefTag */:
                 return getNameOfJSDocTypedef(declaration);
-            case 316 /* JSDocEnumTag */:
+            case 325 /* JSDocEnumTag */:
                 return nameForNamelessJSDocTypedef(declaration);
-            case 259 /* ExportAssignment */: {
+            case 266 /* ExportAssignment */: {
                 var expression = declaration.expression;
-                return isIdentifier(expression) ? expression : undefined;
+                return ts.isIdentifier(expression) ? expression : undefined;
             }
-            case 195 /* ElementAccessExpression */:
+            case 202 /* ElementAccessExpression */:
                 var expr = declaration;
                 if (ts.isBindableStaticElementAccessExpression(expr)) {
                     return expr.argumentExpression;
@@ -10999,50 +12004,40 @@ var ts;
         if (declaration === undefined)
             return undefined;
         return getNonAssignedNameOfDeclaration(declaration) ||
-            (isFunctionExpression(declaration) || isClassExpression(declaration) ? getAssignedName(declaration) : undefined);
+            (ts.isFunctionExpression(declaration) || ts.isClassExpression(declaration) ? getAssignedName(declaration) : undefined);
     }
     ts.getNameOfDeclaration = getNameOfDeclaration;
+    /*@internal*/
     function getAssignedName(node) {
         if (!node.parent) {
             return undefined;
         }
-        else if (isPropertyAssignment(node.parent) || isBindingElement(node.parent)) {
+        else if (ts.isPropertyAssignment(node.parent) || ts.isBindingElement(node.parent)) {
             return node.parent.name;
         }
-        else if (isBinaryExpression(node.parent) && node === node.parent.right) {
-            if (isIdentifier(node.parent.left)) {
+        else if (ts.isBinaryExpression(node.parent) && node === node.parent.right) {
+            if (ts.isIdentifier(node.parent.left)) {
                 return node.parent.left;
             }
             else if (ts.isAccessExpression(node.parent.left)) {
                 return ts.getElementOrPropertyAccessArgumentExpressionOrName(node.parent.left);
             }
         }
-        else if (isVariableDeclaration(node.parent) && isIdentifier(node.parent.name)) {
+        else if (ts.isVariableDeclaration(node.parent) && ts.isIdentifier(node.parent.name)) {
             return node.parent.name;
         }
     }
-    /**
-     * Gets the JSDoc parameter tags for the node if present.
-     *
-     * @remarks Returns any JSDoc param tag whose name matches the provided
-     * parameter, whether a param tag on a containing function
-     * expression, or a param tag on a variable declaration whose
-     * initializer is the containing function. The tags closest to the
-     * node are returned first, so in the previous example, the param
-     * tag on the containing function expression would be first.
-     *
-     * For binding patterns, parameter tags are matched by position.
-     */
-    function getJSDocParameterTags(param) {
+    ts.getAssignedName = getAssignedName;
+    function getJSDocParameterTagsWorker(param, noCache) {
         if (param.name) {
-            if (isIdentifier(param.name)) {
+            if (ts.isIdentifier(param.name)) {
                 var name_1 = param.name.escapedText;
-                return getJSDocTags(param.parent).filter(function (tag) { return isJSDocParameterTag(tag) && isIdentifier(tag.name) && tag.name.escapedText === name_1; });
+                return getJSDocTagsWorker(param.parent, noCache).filter(function (tag) { return ts.isJSDocParameterTag(tag) && ts.isIdentifier(tag.name) && tag.name.escapedText === name_1; });
             }
             else {
                 var i = param.parent.parameters.indexOf(param);
                 ts.Debug.assert(i > -1, "Parameters should always be in their parents' parameter list");
-                var paramTags = getJSDocTags(param.parent).filter(isJSDocParameterTag);
+                var paramTags = getJSDocTagsWorker(param.parent, noCache).filter(ts.isJSDocParameterTag);
                 if (i < paramTags.length) {
                     return [paramTags[i]];
                 }
@@ -11051,7 +12046,33 @@ var ts;
         // return empty array for: out-of-order binding patterns and JSDoc function syntax, which has un-named parameters
         return ts.emptyArray;
     }
+    /**
+     * Gets the JSDoc parameter tags for the node if present.
+     *
+     * @remarks Returns any JSDoc param tag whose name matches the provided
+     * parameter, whether a param tag on a containing function
+     * expression, or a param tag on a variable declaration whose
+     * initializer is the containing function. The tags closest to the
+     * node are returned first, so in the previous example, the param
+     * tag on the containing function expression would be first.
+     *
+     * For binding patterns, parameter tags are matched by position.
+     */
+    function getJSDocParameterTags(param) {
+        return getJSDocParameterTagsWorker(param, /*noCache*/ false);
+    }
     ts.getJSDocParameterTags = getJSDocParameterTags;
+    /* @internal */
+    function getJSDocParameterTagsNoCache(param) {
+        return getJSDocParameterTagsWorker(param, /*noCache*/ true);
+    }
+    ts.getJSDocParameterTagsNoCache = getJSDocParameterTagsNoCache;
+    function getJSDocTypeParameterTagsWorker(param, noCache) {
+        var name = param.name.escapedText;
+        return getJSDocTagsWorker(param.parent, noCache).filter(function (tag) {
+            return ts.isJSDocTemplateTag(tag) && tag.typeParameters.some(function (tp) { return tp.name.escapedText === name; });
+        });
+    }
     /**
      * Gets the JSDoc type parameter tags for the node if present.
      *
@@ -11063,12 +12084,14 @@ var ts;
      * tag on the containing function expression would be first.
      */
     function getJSDocTypeParameterTags(param) {
-        var name = param.name.escapedText;
-        return getJSDocTags(param.parent).filter(function (tag) {
-            return isJSDocTemplateTag(tag) && tag.typeParameters.some(function (tp) { return tp.name.escapedText === name; });
-        });
+        return getJSDocTypeParameterTagsWorker(param, /*noCache*/ false);
     }
     ts.getJSDocTypeParameterTags = getJSDocTypeParameterTags;
+    /* @internal */
+    function getJSDocTypeParameterTagsNoCache(param) {
+        return getJSDocTypeParameterTagsWorker(param, /*noCache*/ true);
+    }
+    ts.getJSDocTypeParameterTagsNoCache = getJSDocTypeParameterTagsNoCache;
     /**
      * Return true if the node has JSDoc parameter tags.
      *
@@ -11076,68 +12099,98 @@ var ts;
      * for example on a variable declaration whose initializer is a function expression.
      */
     function hasJSDocParameterTags(node) {
-        return !!getFirstJSDocTag(node, isJSDocParameterTag);
+        return !!getFirstJSDocTag(node, ts.isJSDocParameterTag);
     }
     ts.hasJSDocParameterTags = hasJSDocParameterTags;
     /** Gets the JSDoc augments tag for the node if present */
     function getJSDocAugmentsTag(node) {
-        return getFirstJSDocTag(node, isJSDocAugmentsTag);
+        return getFirstJSDocTag(node, ts.isJSDocAugmentsTag);
     }
     ts.getJSDocAugmentsTag = getJSDocAugmentsTag;
     /** Gets the JSDoc implements tags for the node if present */
     function getJSDocImplementsTags(node) {
-        return getAllJSDocTags(node, isJSDocImplementsTag);
+        return getAllJSDocTags(node, ts.isJSDocImplementsTag);
     }
     ts.getJSDocImplementsTags = getJSDocImplementsTags;
     /** Gets the JSDoc class tag for the node if present */
     function getJSDocClassTag(node) {
-        return getFirstJSDocTag(node, isJSDocClassTag);
+        return getFirstJSDocTag(node, ts.isJSDocClassTag);
     }
     ts.getJSDocClassTag = getJSDocClassTag;
     /** Gets the JSDoc public tag for the node if present */
     function getJSDocPublicTag(node) {
-        return getFirstJSDocTag(node, isJSDocPublicTag);
+        return getFirstJSDocTag(node, ts.isJSDocPublicTag);
     }
     ts.getJSDocPublicTag = getJSDocPublicTag;
+    /*@internal*/
+    function getJSDocPublicTagNoCache(node) {
+        return getFirstJSDocTag(node, ts.isJSDocPublicTag, /*noCache*/ true);
+    }
+    ts.getJSDocPublicTagNoCache = getJSDocPublicTagNoCache;
     /** Gets the JSDoc private tag for the node if present */
     function getJSDocPrivateTag(node) {
-        return getFirstJSDocTag(node, isJSDocPrivateTag);
+        return getFirstJSDocTag(node, ts.isJSDocPrivateTag);
     }
     ts.getJSDocPrivateTag = getJSDocPrivateTag;
+    /*@internal*/
+    function getJSDocPrivateTagNoCache(node) {
+        return getFirstJSDocTag(node, ts.isJSDocPrivateTag, /*noCache*/ true);
+    }
+    ts.getJSDocPrivateTagNoCache = getJSDocPrivateTagNoCache;
     /** Gets the JSDoc protected tag for the node if present */
     function getJSDocProtectedTag(node) {
-        return getFirstJSDocTag(node, isJSDocProtectedTag);
+        return getFirstJSDocTag(node, ts.isJSDocProtectedTag);
     }
     ts.getJSDocProtectedTag = getJSDocProtectedTag;
+    /*@internal*/
+    function getJSDocProtectedTagNoCache(node) {
+        return getFirstJSDocTag(node, ts.isJSDocProtectedTag, /*noCache*/ true);
+    }
+    ts.getJSDocProtectedTagNoCache = getJSDocProtectedTagNoCache;
     /** Gets the JSDoc protected tag for the node if present */
     function getJSDocReadonlyTag(node) {
-        return getFirstJSDocTag(node, isJSDocReadonlyTag);
+        return getFirstJSDocTag(node, ts.isJSDocReadonlyTag);
     }
     ts.getJSDocReadonlyTag = getJSDocReadonlyTag;
+    /*@internal*/
+    function getJSDocReadonlyTagNoCache(node) {
+        return getFirstJSDocTag(node, ts.isJSDocReadonlyTag, /*noCache*/ true);
+    }
+    ts.getJSDocReadonlyTagNoCache = getJSDocReadonlyTagNoCache;
+    /** Gets the JSDoc deprecated tag for the node if present */
+    function getJSDocDeprecatedTag(node) {
+        return getFirstJSDocTag(node, ts.isJSDocDeprecatedTag);
+    }
+    ts.getJSDocDeprecatedTag = getJSDocDeprecatedTag;
+    /*@internal */
+    function getJSDocDeprecatedTagNoCache(node) {
+        return getFirstJSDocTag(node, ts.isJSDocDeprecatedTag, /*noCache*/ true);
+    }
+    ts.getJSDocDeprecatedTagNoCache = getJSDocDeprecatedTagNoCache;
     /** Gets the JSDoc enum tag for the node if present */
     function getJSDocEnumTag(node) {
-        return getFirstJSDocTag(node, isJSDocEnumTag);
+        return getFirstJSDocTag(node, ts.isJSDocEnumTag);
     }
     ts.getJSDocEnumTag = getJSDocEnumTag;
     /** Gets the JSDoc this tag for the node if present */
     function getJSDocThisTag(node) {
-        return getFirstJSDocTag(node, isJSDocThisTag);
+        return getFirstJSDocTag(node, ts.isJSDocThisTag);
     }
     ts.getJSDocThisTag = getJSDocThisTag;
     /** Gets the JSDoc return tag for the node if present */
     function getJSDocReturnTag(node) {
-        return getFirstJSDocTag(node, isJSDocReturnTag);
+        return getFirstJSDocTag(node, ts.isJSDocReturnTag);
     }
     ts.getJSDocReturnTag = getJSDocReturnTag;
     /** Gets the JSDoc template tag for the node if present */
     function getJSDocTemplateTag(node) {
-        return getFirstJSDocTag(node, isJSDocTemplateTag);
+        return getFirstJSDocTag(node, ts.isJSDocTemplateTag);
     }
     ts.getJSDocTemplateTag = getJSDocTemplateTag;
     /** Gets the JSDoc type tag for the node if present and valid */
     function getJSDocTypeTag(node) {
         // We should have already issued an error if there were multiple type jsdocs, so just use the first one.
-        var tag = getFirstJSDocTag(node, isJSDocTypeTag);
+        var tag = getFirstJSDocTag(node, ts.isJSDocTypeTag);
         if (tag && tag.typeExpression && tag.typeExpression.type) {
             return tag;
         }
@@ -11156,8 +12209,8 @@ var ts;
      * tag directly on the node would be returned.
      */
     function getJSDocType(node) {
-        var tag = getFirstJSDocTag(node, isJSDocTypeTag);
-        if (!tag && isParameter(node)) {
+        var tag = getFirstJSDocTag(node, ts.isJSDocTypeTag);
+        if (!tag && ts.isParameter(node)) {
             tag = ts.find(getJSDocParameterTags(node), function (tag) { return !!tag.typeExpression; });
         }
         return tag && tag.typeExpression && tag.typeExpression.type;
@@ -11177,31 +12230,42 @@ var ts;
         var typeTag = getJSDocTypeTag(node);
         if (typeTag && typeTag.typeExpression) {
             var type = typeTag.typeExpression.type;
-            if (isTypeLiteralNode(type)) {
-                var sig = ts.find(type.members, isCallSignatureDeclaration);
+            if (ts.isTypeLiteralNode(type)) {
+                var sig = ts.find(type.members, ts.isCallSignatureDeclaration);
                 return sig && sig.type;
             }
-            if (isFunctionTypeNode(type) || isJSDocFunctionType(type)) {
+            if (ts.isFunctionTypeNode(type) || ts.isJSDocFunctionType(type)) {
                 return type.type;
             }
         }
     }
     ts.getJSDocReturnType = getJSDocReturnType;
-    /** Get all JSDoc tags related to a node, including those on parent nodes. */
-    function getJSDocTags(node) {
+    function getJSDocTagsWorker(node, noCache) {
         var tags = node.jsDocCache;
         // If cache is 'null', that means we did the work of searching for JSDoc tags and came up with nothing.
-        if (tags === undefined) {
-            var comments = ts.getJSDocCommentsAndTags(node);
+        if (tags === undefined || noCache) {
+            var comments = ts.getJSDocCommentsAndTags(node, noCache);
             ts.Debug.assert(comments.length < 2 || comments[0] !== comments[1]);
-            node.jsDocCache = tags = ts.flatMap(comments, function (j) { return isJSDoc(j) ? j.tags : j; });
+            tags = ts.flatMap(comments, function (j) { return ts.isJSDoc(j) ? j.tags : j; });
+            if (!noCache) {
+                node.jsDocCache = tags;
+            }
         }
         return tags;
     }
+    /** Get all JSDoc tags related to a node, including those on parent nodes. */
+    function getJSDocTags(node) {
+        return getJSDocTagsWorker(node, /*noCache*/ false);
+    }
     ts.getJSDocTags = getJSDocTags;
+    /* @internal */
+    function getJSDocTagsNoCache(node) {
+        return getJSDocTagsWorker(node, /*noCache*/ true);
+    }
+    ts.getJSDocTagsNoCache = getJSDocTagsNoCache;
     /** Get the first JSDoc tag of a specified kind, or undefined if not present. */
-    function getFirstJSDocTag(node, predicate) {
-        return ts.find(getJSDocTags(node), predicate);
+    function getFirstJSDocTag(node, predicate, noCache) {
+        return ts.find(getJSDocTagsWorker(node, noCache), predicate);
     }
     /** Gets all JSDoc tags that match a specified predicate */
     function getAllJSDocTags(node, predicate) {
@@ -11218,12 +12282,12 @@ var ts;
      * JavaScript file, gets the type parameters from the `@template` tag from JSDoc.
      */
     function getEffectiveTypeParameterDeclarations(node) {
-        if (isJSDocSignature(node)) {
+        if (ts.isJSDocSignature(node)) {
             return ts.emptyArray;
         }
         if (ts.isJSDocTypeAlias(node)) {
-            ts.Debug.assert(node.parent.kind === 303 /* JSDocComment */);
-            return ts.flatMap(node.parent.tags, function (tag) { return isJSDocTemplateTag(tag) ? tag.typeParameters : undefined; });
+            ts.Debug.assert(node.parent.kind === 311 /* JSDocComment */);
+            return ts.flatMap(node.parent.tags, function (tag) { return ts.isJSDocTemplateTag(tag) ? tag.typeParameters : undefined; });
         }
         if (node.typeParameters) {
             return node.typeParameters;
@@ -11234,7 +12298,7 @@ var ts;
                 return decls;
             }
             var typeTag = getJSDocType(node);
-            if (typeTag && isFunctionTypeNode(typeTag) && typeTag.typeParameters) {
+            if (typeTag && ts.isFunctionTypeNode(typeTag) && typeTag.typeParameters) {
                 return typeTag.typeParameters;
             }
         }
@@ -11243,265 +12307,44 @@ var ts;
     ts.getEffectiveTypeParameterDeclarations = getEffectiveTypeParameterDeclarations;
     function getEffectiveConstraintOfTypeParameter(node) {
         return node.constraint ? node.constraint :
-            isJSDocTemplateTag(node.parent) && node === node.parent.typeParameters[0] ? node.parent.constraint :
+            ts.isJSDocTemplateTag(node.parent) && node === node.parent.typeParameters[0] ? node.parent.constraint :
                 undefined;
     }
     ts.getEffectiveConstraintOfTypeParameter = getEffectiveConstraintOfTypeParameter;
     // #region
-    // Simple node tests of the form `node.kind === SyntaxKind.Foo`.
-    // Literals
-    function isNumericLiteral(node) {
-        return node.kind === 8 /* NumericLiteral */;
-    }
-    ts.isNumericLiteral = isNumericLiteral;
-    function isBigIntLiteral(node) {
-        return node.kind === 9 /* BigIntLiteral */;
-    }
-    ts.isBigIntLiteral = isBigIntLiteral;
-    function isStringLiteral(node) {
-        return node.kind === 10 /* StringLiteral */;
-    }
-    ts.isStringLiteral = isStringLiteral;
-    function isJsxText(node) {
-        return node.kind === 11 /* JsxText */;
-    }
-    ts.isJsxText = isJsxText;
-    function isRegularExpressionLiteral(node) {
-        return node.kind === 13 /* RegularExpressionLiteral */;
-    }
-    ts.isRegularExpressionLiteral = isRegularExpressionLiteral;
-    function isNoSubstitutionTemplateLiteral(node) {
-        return node.kind === 14 /* NoSubstitutionTemplateLiteral */;
-    }
-    ts.isNoSubstitutionTemplateLiteral = isNoSubstitutionTemplateLiteral;
-    // Pseudo-literals
-    function isTemplateHead(node) {
-        return node.kind === 15 /* TemplateHead */;
-    }
-    ts.isTemplateHead = isTemplateHead;
-    function isTemplateMiddle(node) {
-        return node.kind === 16 /* TemplateMiddle */;
-    }
-    ts.isTemplateMiddle = isTemplateMiddle;
-    function isTemplateTail(node) {
-        return node.kind === 17 /* TemplateTail */;
-    }
-    ts.isTemplateTail = isTemplateTail;
-    function isIdentifier(node) {
-        return node.kind === 75 /* Identifier */;
-    }
-    ts.isIdentifier = isIdentifier;
-    // Names
-    function isQualifiedName(node) {
-        return node.kind === 153 /* QualifiedName */;
-    }
-    ts.isQualifiedName = isQualifiedName;
-    function isComputedPropertyName(node) {
-        return node.kind === 154 /* ComputedPropertyName */;
-    }
-    ts.isComputedPropertyName = isComputedPropertyName;
-    function isPrivateIdentifier(node) {
-        return node.kind === 76 /* PrivateIdentifier */;
-    }
-    ts.isPrivateIdentifier = isPrivateIdentifier;
     function isIdentifierOrPrivateIdentifier(node) {
-        return node.kind === 75 /* Identifier */ || node.kind === 76 /* PrivateIdentifier */;
+        return node.kind === 78 /* Identifier */ || node.kind === 79 /* PrivateIdentifier */;
     }
     ts.isIdentifierOrPrivateIdentifier = isIdentifierOrPrivateIdentifier;
-    // Signature elements
-    function isTypeParameterDeclaration(node) {
-        return node.kind === 155 /* TypeParameter */;
-    }
-    ts.isTypeParameterDeclaration = isTypeParameterDeclaration;
-    function isParameter(node) {
-        return node.kind === 156 /* Parameter */;
-    }
-    ts.isParameter = isParameter;
-    function isDecorator(node) {
-        return node.kind === 157 /* Decorator */;
-    }
-    ts.isDecorator = isDecorator;
-    // TypeMember
-    function isPropertySignature(node) {
-        return node.kind === 158 /* PropertySignature */;
-    }
-    ts.isPropertySignature = isPropertySignature;
-    function isPropertyDeclaration(node) {
-        return node.kind === 159 /* PropertyDeclaration */;
-    }
-    ts.isPropertyDeclaration = isPropertyDeclaration;
-    function isMethodSignature(node) {
-        return node.kind === 160 /* MethodSignature */;
-    }
-    ts.isMethodSignature = isMethodSignature;
-    function isMethodDeclaration(node) {
-        return node.kind === 161 /* MethodDeclaration */;
-    }
-    ts.isMethodDeclaration = isMethodDeclaration;
-    function isConstructorDeclaration(node) {
-        return node.kind === 162 /* Constructor */;
-    }
-    ts.isConstructorDeclaration = isConstructorDeclaration;
-    function isGetAccessorDeclaration(node) {
-        return node.kind === 163 /* GetAccessor */;
-    }
-    ts.isGetAccessorDeclaration = isGetAccessorDeclaration;
-    function isSetAccessorDeclaration(node) {
-        return node.kind === 164 /* SetAccessor */;
-    }
-    ts.isSetAccessorDeclaration = isSetAccessorDeclaration;
-    function isCallSignatureDeclaration(node) {
-        return node.kind === 165 /* CallSignature */;
-    }
-    ts.isCallSignatureDeclaration = isCallSignatureDeclaration;
-    function isConstructSignatureDeclaration(node) {
-        return node.kind === 166 /* ConstructSignature */;
-    }
-    ts.isConstructSignatureDeclaration = isConstructSignatureDeclaration;
-    function isIndexSignatureDeclaration(node) {
-        return node.kind === 167 /* IndexSignature */;
-    }
-    ts.isIndexSignatureDeclaration = isIndexSignatureDeclaration;
     /* @internal */
     function isGetOrSetAccessorDeclaration(node) {
-        return node.kind === 164 /* SetAccessor */ || node.kind === 163 /* GetAccessor */;
+        return node.kind === 168 /* SetAccessor */ || node.kind === 167 /* GetAccessor */;
     }
     ts.isGetOrSetAccessorDeclaration = isGetOrSetAccessorDeclaration;
-    // Type
-    function isTypePredicateNode(node) {
-        return node.kind === 168 /* TypePredicate */;
-    }
-    ts.isTypePredicateNode = isTypePredicateNode;
-    function isTypeReferenceNode(node) {
-        return node.kind === 169 /* TypeReference */;
-    }
-    ts.isTypeReferenceNode = isTypeReferenceNode;
-    function isFunctionTypeNode(node) {
-        return node.kind === 170 /* FunctionType */;
-    }
-    ts.isFunctionTypeNode = isFunctionTypeNode;
-    function isConstructorTypeNode(node) {
-        return node.kind === 171 /* ConstructorType */;
-    }
-    ts.isConstructorTypeNode = isConstructorTypeNode;
-    function isTypeQueryNode(node) {
-        return node.kind === 172 /* TypeQuery */;
-    }
-    ts.isTypeQueryNode = isTypeQueryNode;
-    function isTypeLiteralNode(node) {
-        return node.kind === 173 /* TypeLiteral */;
-    }
-    ts.isTypeLiteralNode = isTypeLiteralNode;
-    function isArrayTypeNode(node) {
-        return node.kind === 174 /* ArrayType */;
-    }
-    ts.isArrayTypeNode = isArrayTypeNode;
-    function isTupleTypeNode(node) {
-        return node.kind === 175 /* TupleType */;
-    }
-    ts.isTupleTypeNode = isTupleTypeNode;
-    function isUnionTypeNode(node) {
-        return node.kind === 178 /* UnionType */;
-    }
-    ts.isUnionTypeNode = isUnionTypeNode;
-    function isIntersectionTypeNode(node) {
-        return node.kind === 179 /* IntersectionType */;
-    }
-    ts.isIntersectionTypeNode = isIntersectionTypeNode;
-    function isConditionalTypeNode(node) {
-        return node.kind === 180 /* ConditionalType */;
-    }
-    ts.isConditionalTypeNode = isConditionalTypeNode;
-    function isInferTypeNode(node) {
-        return node.kind === 181 /* InferType */;
-    }
-    ts.isInferTypeNode = isInferTypeNode;
-    function isParenthesizedTypeNode(node) {
-        return node.kind === 182 /* ParenthesizedType */;
-    }
-    ts.isParenthesizedTypeNode = isParenthesizedTypeNode;
-    function isThisTypeNode(node) {
-        return node.kind === 183 /* ThisType */;
-    }
-    ts.isThisTypeNode = isThisTypeNode;
-    function isTypeOperatorNode(node) {
-        return node.kind === 184 /* TypeOperator */;
-    }
-    ts.isTypeOperatorNode = isTypeOperatorNode;
-    function isIndexedAccessTypeNode(node) {
-        return node.kind === 185 /* IndexedAccessType */;
-    }
-    ts.isIndexedAccessTypeNode = isIndexedAccessTypeNode;
-    function isMappedTypeNode(node) {
-        return node.kind === 186 /* MappedType */;
-    }
-    ts.isMappedTypeNode = isMappedTypeNode;
-    function isLiteralTypeNode(node) {
-        return node.kind === 187 /* LiteralType */;
-    }
-    ts.isLiteralTypeNode = isLiteralTypeNode;
-    function isImportTypeNode(node) {
-        return node.kind === 188 /* ImportType */;
-    }
-    ts.isImportTypeNode = isImportTypeNode;
-    // Binding patterns
-    function isObjectBindingPattern(node) {
-        return node.kind === 189 /* ObjectBindingPattern */;
-    }
-    ts.isObjectBindingPattern = isObjectBindingPattern;
-    function isArrayBindingPattern(node) {
-        return node.kind === 190 /* ArrayBindingPattern */;
-    }
-    ts.isArrayBindingPattern = isArrayBindingPattern;
-    function isBindingElement(node) {
-        return node.kind === 191 /* BindingElement */;
-    }
-    ts.isBindingElement = isBindingElement;
-    // Expression
-    function isArrayLiteralExpression(node) {
-        return node.kind === 192 /* ArrayLiteralExpression */;
-    }
-    ts.isArrayLiteralExpression = isArrayLiteralExpression;
-    function isObjectLiteralExpression(node) {
-        return node.kind === 193 /* ObjectLiteralExpression */;
-    }
-    ts.isObjectLiteralExpression = isObjectLiteralExpression;
-    function isPropertyAccessExpression(node) {
-        return node.kind === 194 /* PropertyAccessExpression */;
-    }
-    ts.isPropertyAccessExpression = isPropertyAccessExpression;
     function isPropertyAccessChain(node) {
-        return isPropertyAccessExpression(node) && !!(node.flags & 32 /* OptionalChain */);
+        return ts.isPropertyAccessExpression(node) && !!(node.flags & 32 /* OptionalChain */);
     }
     ts.isPropertyAccessChain = isPropertyAccessChain;
-    function isElementAccessExpression(node) {
-        return node.kind === 195 /* ElementAccessExpression */;
-    }
-    ts.isElementAccessExpression = isElementAccessExpression;
     function isElementAccessChain(node) {
-        return isElementAccessExpression(node) && !!(node.flags & 32 /* OptionalChain */);
+        return ts.isElementAccessExpression(node) && !!(node.flags & 32 /* OptionalChain */);
     }
     ts.isElementAccessChain = isElementAccessChain;
-    function isCallExpression(node) {
-        return node.kind === 196 /* CallExpression */;
-    }
-    ts.isCallExpression = isCallExpression;
     function isCallChain(node) {
-        return isCallExpression(node) && !!(node.flags & 32 /* OptionalChain */);
+        return ts.isCallExpression(node) && !!(node.flags & 32 /* OptionalChain */);
     }
     ts.isCallChain = isCallChain;
     function isOptionalChain(node) {
         var kind = node.kind;
         return !!(node.flags & 32 /* OptionalChain */) &&
-            (kind === 194 /* PropertyAccessExpression */
-                || kind === 195 /* ElementAccessExpression */
-                || kind === 196 /* CallExpression */
-                || kind === 218 /* NonNullExpression */);
+            (kind === 201 /* PropertyAccessExpression */
+                || kind === 202 /* ElementAccessExpression */
+                || kind === 203 /* CallExpression */
+                || kind === 225 /* NonNullExpression */);
     }
     ts.isOptionalChain = isOptionalChain;
     /* @internal */
     function isOptionalChainRoot(node) {
-        return isOptionalChain(node) && !isNonNullExpression(node) && !!node.questionDotToken;
+        return isOptionalChain(node) && !ts.isNonNullExpression(node) && !!node.questionDotToken;
     }
     ts.isOptionalChainRoot = isOptionalChainRoot;
     /**
@@ -11531,406 +12374,34 @@ var ts;
     }
     ts.isOutermostOptionalChain = isOutermostOptionalChain;
     function isNullishCoalesce(node) {
-        return node.kind === 209 /* BinaryExpression */ && node.operatorToken.kind === 60 /* QuestionQuestionToken */;
+        return node.kind === 216 /* BinaryExpression */ && node.operatorToken.kind === 60 /* QuestionQuestionToken */;
     }
     ts.isNullishCoalesce = isNullishCoalesce;
-    function isNewExpression(node) {
-        return node.kind === 197 /* NewExpression */;
-    }
-    ts.isNewExpression = isNewExpression;
-    function isTaggedTemplateExpression(node) {
-        return node.kind === 198 /* TaggedTemplateExpression */;
-    }
-    ts.isTaggedTemplateExpression = isTaggedTemplateExpression;
-    function isTypeAssertion(node) {
-        return node.kind === 199 /* TypeAssertionExpression */;
-    }
-    ts.isTypeAssertion = isTypeAssertion;
     function isConstTypeReference(node) {
-        return isTypeReferenceNode(node) && isIdentifier(node.typeName) &&
+        return ts.isTypeReferenceNode(node) && ts.isIdentifier(node.typeName) &&
             node.typeName.escapedText === "const" && !node.typeArguments;
     }
     ts.isConstTypeReference = isConstTypeReference;
-    function isParenthesizedExpression(node) {
-        return node.kind === 200 /* ParenthesizedExpression */;
-    }
-    ts.isParenthesizedExpression = isParenthesizedExpression;
     function skipPartiallyEmittedExpressions(node) {
         return ts.skipOuterExpressions(node, 8 /* PartiallyEmittedExpressions */);
     }
     ts.skipPartiallyEmittedExpressions = skipPartiallyEmittedExpressions;
-    function isFunctionExpression(node) {
-        return node.kind === 201 /* FunctionExpression */;
-    }
-    ts.isFunctionExpression = isFunctionExpression;
-    function isArrowFunction(node) {
-        return node.kind === 202 /* ArrowFunction */;
-    }
-    ts.isArrowFunction = isArrowFunction;
-    function isDeleteExpression(node) {
-        return node.kind === 203 /* DeleteExpression */;
-    }
-    ts.isDeleteExpression = isDeleteExpression;
-    function isTypeOfExpression(node) {
-        return node.kind === 204 /* TypeOfExpression */;
-    }
-    ts.isTypeOfExpression = isTypeOfExpression;
-    function isVoidExpression(node) {
-        return node.kind === 205 /* VoidExpression */;
-    }
-    ts.isVoidExpression = isVoidExpression;
-    function isAwaitExpression(node) {
-        return node.kind === 206 /* AwaitExpression */;
-    }
-    ts.isAwaitExpression = isAwaitExpression;
-    function isPrefixUnaryExpression(node) {
-        return node.kind === 207 /* PrefixUnaryExpression */;
-    }
-    ts.isPrefixUnaryExpression = isPrefixUnaryExpression;
-    function isPostfixUnaryExpression(node) {
-        return node.kind === 208 /* PostfixUnaryExpression */;
-    }
-    ts.isPostfixUnaryExpression = isPostfixUnaryExpression;
-    function isBinaryExpression(node) {
-        return node.kind === 209 /* BinaryExpression */;
-    }
-    ts.isBinaryExpression = isBinaryExpression;
-    function isConditionalExpression(node) {
-        return node.kind === 210 /* ConditionalExpression */;
-    }
-    ts.isConditionalExpression = isConditionalExpression;
-    function isTemplateExpression(node) {
-        return node.kind === 211 /* TemplateExpression */;
-    }
-    ts.isTemplateExpression = isTemplateExpression;
-    function isYieldExpression(node) {
-        return node.kind === 212 /* YieldExpression */;
-    }
-    ts.isYieldExpression = isYieldExpression;
-    function isSpreadElement(node) {
-        return node.kind === 213 /* SpreadElement */;
-    }
-    ts.isSpreadElement = isSpreadElement;
-    function isClassExpression(node) {
-        return node.kind === 214 /* ClassExpression */;
-    }
-    ts.isClassExpression = isClassExpression;
-    function isOmittedExpression(node) {
-        return node.kind === 215 /* OmittedExpression */;
-    }
-    ts.isOmittedExpression = isOmittedExpression;
-    function isExpressionWithTypeArguments(node) {
-        return node.kind === 216 /* ExpressionWithTypeArguments */;
-    }
-    ts.isExpressionWithTypeArguments = isExpressionWithTypeArguments;
-    function isAsExpression(node) {
-        return node.kind === 217 /* AsExpression */;
-    }
-    ts.isAsExpression = isAsExpression;
-    function isNonNullExpression(node) {
-        return node.kind === 218 /* NonNullExpression */;
-    }
-    ts.isNonNullExpression = isNonNullExpression;
     function isNonNullChain(node) {
-        return isNonNullExpression(node) && !!(node.flags & 32 /* OptionalChain */);
+        return ts.isNonNullExpression(node) && !!(node.flags & 32 /* OptionalChain */);
     }
     ts.isNonNullChain = isNonNullChain;
-    function isMetaProperty(node) {
-        return node.kind === 219 /* MetaProperty */;
-    }
-    ts.isMetaProperty = isMetaProperty;
-    // Misc
-    function isTemplateSpan(node) {
-        return node.kind === 221 /* TemplateSpan */;
-    }
-    ts.isTemplateSpan = isTemplateSpan;
-    function isSemicolonClassElement(node) {
-        return node.kind === 222 /* SemicolonClassElement */;
-    }
-    ts.isSemicolonClassElement = isSemicolonClassElement;
-    // Block
-    function isBlock(node) {
-        return node.kind === 223 /* Block */;
-    }
-    ts.isBlock = isBlock;
-    function isVariableStatement(node) {
-        return node.kind === 225 /* VariableStatement */;
-    }
-    ts.isVariableStatement = isVariableStatement;
-    function isEmptyStatement(node) {
-        return node.kind === 224 /* EmptyStatement */;
-    }
-    ts.isEmptyStatement = isEmptyStatement;
-    function isExpressionStatement(node) {
-        return node.kind === 226 /* ExpressionStatement */;
-    }
-    ts.isExpressionStatement = isExpressionStatement;
-    function isIfStatement(node) {
-        return node.kind === 227 /* IfStatement */;
-    }
-    ts.isIfStatement = isIfStatement;
-    function isDoStatement(node) {
-        return node.kind === 228 /* DoStatement */;
-    }
-    ts.isDoStatement = isDoStatement;
-    function isWhileStatement(node) {
-        return node.kind === 229 /* WhileStatement */;
-    }
-    ts.isWhileStatement = isWhileStatement;
-    function isForStatement(node) {
-        return node.kind === 230 /* ForStatement */;
-    }
-    ts.isForStatement = isForStatement;
-    function isForInStatement(node) {
-        return node.kind === 231 /* ForInStatement */;
-    }
-    ts.isForInStatement = isForInStatement;
-    function isForOfStatement(node) {
-        return node.kind === 232 /* ForOfStatement */;
-    }
-    ts.isForOfStatement = isForOfStatement;
-    function isContinueStatement(node) {
-        return node.kind === 233 /* ContinueStatement */;
-    }
-    ts.isContinueStatement = isContinueStatement;
-    function isBreakStatement(node) {
-        return node.kind === 234 /* BreakStatement */;
-    }
-    ts.isBreakStatement = isBreakStatement;
     function isBreakOrContinueStatement(node) {
-        return node.kind === 234 /* BreakStatement */ || node.kind === 233 /* ContinueStatement */;
+        return node.kind === 241 /* BreakStatement */ || node.kind === 240 /* ContinueStatement */;
     }
     ts.isBreakOrContinueStatement = isBreakOrContinueStatement;
-    function isReturnStatement(node) {
-        return node.kind === 235 /* ReturnStatement */;
-    }
-    ts.isReturnStatement = isReturnStatement;
-    function isWithStatement(node) {
-        return node.kind === 236 /* WithStatement */;
-    }
-    ts.isWithStatement = isWithStatement;
-    function isSwitchStatement(node) {
-        return node.kind === 237 /* SwitchStatement */;
-    }
-    ts.isSwitchStatement = isSwitchStatement;
-    function isLabeledStatement(node) {
-        return node.kind === 238 /* LabeledStatement */;
-    }
-    ts.isLabeledStatement = isLabeledStatement;
-    function isThrowStatement(node) {
-        return node.kind === 239 /* ThrowStatement */;
-    }
-    ts.isThrowStatement = isThrowStatement;
-    function isTryStatement(node) {
-        return node.kind === 240 /* TryStatement */;
-    }
-    ts.isTryStatement = isTryStatement;
-    function isDebuggerStatement(node) {
-        return node.kind === 241 /* DebuggerStatement */;
-    }
-    ts.isDebuggerStatement = isDebuggerStatement;
-    function isVariableDeclaration(node) {
-        return node.kind === 242 /* VariableDeclaration */;
-    }
-    ts.isVariableDeclaration = isVariableDeclaration;
-    function isVariableDeclarationList(node) {
-        return node.kind === 243 /* VariableDeclarationList */;
-    }
-    ts.isVariableDeclarationList = isVariableDeclarationList;
-    function isFunctionDeclaration(node) {
-        return node.kind === 244 /* FunctionDeclaration */;
-    }
-    ts.isFunctionDeclaration = isFunctionDeclaration;
-    function isClassDeclaration(node) {
-        return node.kind === 245 /* ClassDeclaration */;
-    }
-    ts.isClassDeclaration = isClassDeclaration;
-    function isInterfaceDeclaration(node) {
-        return node.kind === 246 /* InterfaceDeclaration */;
-    }
-    ts.isInterfaceDeclaration = isInterfaceDeclaration;
-    function isTypeAliasDeclaration(node) {
-        return node.kind === 247 /* TypeAliasDeclaration */;
-    }
-    ts.isTypeAliasDeclaration = isTypeAliasDeclaration;
-    function isEnumDeclaration(node) {
-        return node.kind === 248 /* EnumDeclaration */;
-    }
-    ts.isEnumDeclaration = isEnumDeclaration;
-    function isModuleDeclaration(node) {
-        return node.kind === 249 /* ModuleDeclaration */;
-    }
-    ts.isModuleDeclaration = isModuleDeclaration;
-    function isModuleBlock(node) {
-        return node.kind === 250 /* ModuleBlock */;
-    }
-    ts.isModuleBlock = isModuleBlock;
-    function isCaseBlock(node) {
-        return node.kind === 251 /* CaseBlock */;
-    }
-    ts.isCaseBlock = isCaseBlock;
-    function isNamespaceExportDeclaration(node) {
-        return node.kind === 252 /* NamespaceExportDeclaration */;
-    }
-    ts.isNamespaceExportDeclaration = isNamespaceExportDeclaration;
-    function isImportEqualsDeclaration(node) {
-        return node.kind === 253 /* ImportEqualsDeclaration */;
-    }
-    ts.isImportEqualsDeclaration = isImportEqualsDeclaration;
-    function isImportDeclaration(node) {
-        return node.kind === 254 /* ImportDeclaration */;
-    }
-    ts.isImportDeclaration = isImportDeclaration;
-    function isImportClause(node) {
-        return node.kind === 255 /* ImportClause */;
-    }
-    ts.isImportClause = isImportClause;
-    function isNamespaceImport(node) {
-        return node.kind === 256 /* NamespaceImport */;
-    }
-    ts.isNamespaceImport = isNamespaceImport;
-    function isNamespaceExport(node) {
-        return node.kind === 262 /* NamespaceExport */;
-    }
-    ts.isNamespaceExport = isNamespaceExport;
     function isNamedExportBindings(node) {
-        return node.kind === 262 /* NamespaceExport */ || node.kind === 261 /* NamedExports */;
+        return node.kind === 269 /* NamespaceExport */ || node.kind === 268 /* NamedExports */;
     }
     ts.isNamedExportBindings = isNamedExportBindings;
-    function isNamedImports(node) {
-        return node.kind === 257 /* NamedImports */;
-    }
-    ts.isNamedImports = isNamedImports;
-    function isImportSpecifier(node) {
-        return node.kind === 258 /* ImportSpecifier */;
-    }
-    ts.isImportSpecifier = isImportSpecifier;
-    function isExportAssignment(node) {
-        return node.kind === 259 /* ExportAssignment */;
-    }
-    ts.isExportAssignment = isExportAssignment;
-    function isExportDeclaration(node) {
-        return node.kind === 260 /* ExportDeclaration */;
-    }
-    ts.isExportDeclaration = isExportDeclaration;
-    function isNamedExports(node) {
-        return node.kind === 261 /* NamedExports */;
-    }
-    ts.isNamedExports = isNamedExports;
-    function isExportSpecifier(node) {
-        return node.kind === 263 /* ExportSpecifier */;
-    }
-    ts.isExportSpecifier = isExportSpecifier;
-    function isMissingDeclaration(node) {
-        return node.kind === 264 /* MissingDeclaration */;
-    }
-    ts.isMissingDeclaration = isMissingDeclaration;
-    // Module References
-    function isExternalModuleReference(node) {
-        return node.kind === 265 /* ExternalModuleReference */;
-    }
-    ts.isExternalModuleReference = isExternalModuleReference;
-    // JSX
-    function isJsxElement(node) {
-        return node.kind === 266 /* JsxElement */;
-    }
-    ts.isJsxElement = isJsxElement;
-    function isJsxSelfClosingElement(node) {
-        return node.kind === 267 /* JsxSelfClosingElement */;
-    }
-    ts.isJsxSelfClosingElement = isJsxSelfClosingElement;
-    function isJsxOpeningElement(node) {
-        return node.kind === 268 /* JsxOpeningElement */;
-    }
-    ts.isJsxOpeningElement = isJsxOpeningElement;
-    function isJsxClosingElement(node) {
-        return node.kind === 269 /* JsxClosingElement */;
-    }
-    ts.isJsxClosingElement = isJsxClosingElement;
-    function isJsxFragment(node) {
-        return node.kind === 270 /* JsxFragment */;
-    }
-    ts.isJsxFragment = isJsxFragment;
-    function isJsxOpeningFragment(node) {
-        return node.kind === 271 /* JsxOpeningFragment */;
-    }
-    ts.isJsxOpeningFragment = isJsxOpeningFragment;
-    function isJsxClosingFragment(node) {
-        return node.kind === 272 /* JsxClosingFragment */;
-    }
-    ts.isJsxClosingFragment = isJsxClosingFragment;
-    function isJsxAttribute(node) {
-        return node.kind === 273 /* JsxAttribute */;
-    }
-    ts.isJsxAttribute = isJsxAttribute;
-    function isJsxAttributes(node) {
-        return node.kind === 274 /* JsxAttributes */;
-    }
-    ts.isJsxAttributes = isJsxAttributes;
-    function isJsxSpreadAttribute(node) {
-        return node.kind === 275 /* JsxSpreadAttribute */;
-    }
-    ts.isJsxSpreadAttribute = isJsxSpreadAttribute;
-    function isJsxExpression(node) {
-        return node.kind === 276 /* JsxExpression */;
-    }
-    ts.isJsxExpression = isJsxExpression;
-    // Clauses
-    function isCaseClause(node) {
-        return node.kind === 277 /* CaseClause */;
-    }
-    ts.isCaseClause = isCaseClause;
-    function isDefaultClause(node) {
-        return node.kind === 278 /* DefaultClause */;
-    }
-    ts.isDefaultClause = isDefaultClause;
-    function isHeritageClause(node) {
-        return node.kind === 279 /* HeritageClause */;
-    }
-    ts.isHeritageClause = isHeritageClause;
-    function isCatchClause(node) {
-        return node.kind === 280 /* CatchClause */;
-    }
-    ts.isCatchClause = isCatchClause;
-    // Property assignments
-    function isPropertyAssignment(node) {
-        return node.kind === 281 /* PropertyAssignment */;
-    }
-    ts.isPropertyAssignment = isPropertyAssignment;
-    function isShorthandPropertyAssignment(node) {
-        return node.kind === 282 /* ShorthandPropertyAssignment */;
-    }
-    ts.isShorthandPropertyAssignment = isShorthandPropertyAssignment;
-    function isSpreadAssignment(node) {
-        return node.kind === 283 /* SpreadAssignment */;
-    }
-    ts.isSpreadAssignment = isSpreadAssignment;
-    // Enum
-    function isEnumMember(node) {
-        return node.kind === 284 /* EnumMember */;
-    }
-    ts.isEnumMember = isEnumMember;
-    // Top-level nodes
-    function isSourceFile(node) {
-        return node.kind === 290 /* SourceFile */;
-    }
-    ts.isSourceFile = isSourceFile;
-    function isBundle(node) {
-        return node.kind === 291 /* Bundle */;
-    }
-    ts.isBundle = isBundle;
-    function isUnparsedSource(node) {
-        return node.kind === 292 /* UnparsedSource */;
-    }
-    ts.isUnparsedSource = isUnparsedSource;
-    function isUnparsedPrepend(node) {
-        return node.kind === 286 /* UnparsedPrepend */;
-    }
-    ts.isUnparsedPrepend = isUnparsedPrepend;
     function isUnparsedTextLike(node) {
         switch (node.kind) {
-            case 287 /* UnparsedText */:
-            case 288 /* UnparsedInternalText */:
+            case 294 /* UnparsedText */:
+            case 295 /* UnparsedInternalText */:
                 return true;
             default:
                 return false;
@@ -11939,127 +12410,14 @@ var ts;
     ts.isUnparsedTextLike = isUnparsedTextLike;
     function isUnparsedNode(node) {
         return isUnparsedTextLike(node) ||
-            node.kind === 285 /* UnparsedPrologue */ ||
-            node.kind === 289 /* UnparsedSyntheticReference */;
+            node.kind === 292 /* UnparsedPrologue */ ||
+            node.kind === 296 /* UnparsedSyntheticReference */;
     }
     ts.isUnparsedNode = isUnparsedNode;
-    // JSDoc
-    function isJSDocTypeExpression(node) {
-        return node.kind === 294 /* JSDocTypeExpression */;
-    }
-    ts.isJSDocTypeExpression = isJSDocTypeExpression;
-    function isJSDocAllType(node) {
-        return node.kind === 295 /* JSDocAllType */;
-    }
-    ts.isJSDocAllType = isJSDocAllType;
-    function isJSDocUnknownType(node) {
-        return node.kind === 296 /* JSDocUnknownType */;
-    }
-    ts.isJSDocUnknownType = isJSDocUnknownType;
-    function isJSDocNullableType(node) {
-        return node.kind === 297 /* JSDocNullableType */;
-    }
-    ts.isJSDocNullableType = isJSDocNullableType;
-    function isJSDocNonNullableType(node) {
-        return node.kind === 298 /* JSDocNonNullableType */;
-    }
-    ts.isJSDocNonNullableType = isJSDocNonNullableType;
-    function isJSDocOptionalType(node) {
-        return node.kind === 299 /* JSDocOptionalType */;
-    }
-    ts.isJSDocOptionalType = isJSDocOptionalType;
-    function isJSDocFunctionType(node) {
-        return node.kind === 300 /* JSDocFunctionType */;
-    }
-    ts.isJSDocFunctionType = isJSDocFunctionType;
-    function isJSDocVariadicType(node) {
-        return node.kind === 301 /* JSDocVariadicType */;
-    }
-    ts.isJSDocVariadicType = isJSDocVariadicType;
-    function isJSDoc(node) {
-        return node.kind === 303 /* JSDocComment */;
-    }
-    ts.isJSDoc = isJSDoc;
-    function isJSDocAuthorTag(node) {
-        return node.kind === 309 /* JSDocAuthorTag */;
-    }
-    ts.isJSDocAuthorTag = isJSDocAuthorTag;
-    function isJSDocAugmentsTag(node) {
-        return node.kind === 307 /* JSDocAugmentsTag */;
-    }
-    ts.isJSDocAugmentsTag = isJSDocAugmentsTag;
-    function isJSDocImplementsTag(node) {
-        return node.kind === 308 /* JSDocImplementsTag */;
-    }
-    ts.isJSDocImplementsTag = isJSDocImplementsTag;
-    function isJSDocClassTag(node) {
-        return node.kind === 310 /* JSDocClassTag */;
-    }
-    ts.isJSDocClassTag = isJSDocClassTag;
-    function isJSDocPublicTag(node) {
-        return node.kind === 311 /* JSDocPublicTag */;
-    }
-    ts.isJSDocPublicTag = isJSDocPublicTag;
-    function isJSDocPrivateTag(node) {
-        return node.kind === 312 /* JSDocPrivateTag */;
-    }
-    ts.isJSDocPrivateTag = isJSDocPrivateTag;
-    function isJSDocProtectedTag(node) {
-        return node.kind === 313 /* JSDocProtectedTag */;
-    }
-    ts.isJSDocProtectedTag = isJSDocProtectedTag;
-    function isJSDocReadonlyTag(node) {
-        return node.kind === 314 /* JSDocReadonlyTag */;
-    }
-    ts.isJSDocReadonlyTag = isJSDocReadonlyTag;
-    function isJSDocEnumTag(node) {
-        return node.kind === 316 /* JSDocEnumTag */;
-    }
-    ts.isJSDocEnumTag = isJSDocEnumTag;
-    function isJSDocThisTag(node) {
-        return node.kind === 319 /* JSDocThisTag */;
-    }
-    ts.isJSDocThisTag = isJSDocThisTag;
-    function isJSDocParameterTag(node) {
-        return node.kind === 317 /* JSDocParameterTag */;
-    }
-    ts.isJSDocParameterTag = isJSDocParameterTag;
-    function isJSDocReturnTag(node) {
-        return node.kind === 318 /* JSDocReturnTag */;
-    }
-    ts.isJSDocReturnTag = isJSDocReturnTag;
-    function isJSDocTypeTag(node) {
-        return node.kind === 320 /* JSDocTypeTag */;
-    }
-    ts.isJSDocTypeTag = isJSDocTypeTag;
-    function isJSDocTemplateTag(node) {
-        return node.kind === 321 /* JSDocTemplateTag */;
-    }
-    ts.isJSDocTemplateTag = isJSDocTemplateTag;
-    function isJSDocTypedefTag(node) {
-        return node.kind === 322 /* JSDocTypedefTag */;
-    }
-    ts.isJSDocTypedefTag = isJSDocTypedefTag;
-    function isJSDocPropertyTag(node) {
-        return node.kind === 323 /* JSDocPropertyTag */;
-    }
-    ts.isJSDocPropertyTag = isJSDocPropertyTag;
     function isJSDocPropertyLikeTag(node) {
-        return node.kind === 323 /* JSDocPropertyTag */ || node.kind === 317 /* JSDocParameterTag */;
+        return node.kind === 333 /* JSDocPropertyTag */ || node.kind === 326 /* JSDocParameterTag */;
     }
     ts.isJSDocPropertyLikeTag = isJSDocPropertyLikeTag;
-    function isJSDocTypeLiteral(node) {
-        return node.kind === 304 /* JSDocTypeLiteral */;
-    }
-    ts.isJSDocTypeLiteral = isJSDocTypeLiteral;
-    function isJSDocCallbackTag(node) {
-        return node.kind === 315 /* JSDocCallbackTag */;
-    }
-    ts.isJSDocCallbackTag = isJSDocCallbackTag;
-    function isJSDocSignature(node) {
-        return node.kind === 305 /* JSDocSignature */;
-    }
-    ts.isJSDocSignature = isJSDocSignature;
     // #endregion
     // #region
     // Node tests
@@ -12067,18 +12425,13 @@ var ts;
     // All node tests in the following list should *not* reference parent pointers so that
     // they may be used with transformations.
     /* @internal */
-    function isSyntaxList(n) {
-        return n.kind === 324 /* SyntaxList */;
-    }
-    ts.isSyntaxList = isSyntaxList;
-    /* @internal */
     function isNode(node) {
         return isNodeKind(node.kind);
     }
     ts.isNode = isNode;
     /* @internal */
     function isNodeKind(kind) {
-        return kind >= 153 /* FirstNode */;
+        return kind >= 157 /* FirstNode */;
     }
     ts.isNodeKind = isNodeKind;
     /**
@@ -12087,7 +12440,7 @@ var ts;
      * Literals are considered tokens, except TemplateLiteral, but does include TemplateHead/Middle/Tail.
      */
     function isToken(n) {
-        return n.kind >= 0 /* FirstToken */ && n.kind <= 152 /* LastToken */;
+        return n.kind >= 0 /* FirstToken */ && n.kind <= 156 /* LastToken */;
     }
     ts.isToken = isToken;
     // Node Arrays
@@ -12123,17 +12476,17 @@ var ts;
     }
     ts.isTemplateMiddleOrTemplateTail = isTemplateMiddleOrTemplateTail;
     function isImportOrExportSpecifier(node) {
-        return isImportSpecifier(node) || isExportSpecifier(node);
+        return ts.isImportSpecifier(node) || ts.isExportSpecifier(node);
     }
     ts.isImportOrExportSpecifier = isImportOrExportSpecifier;
     function isTypeOnlyImportOrExportDeclaration(node) {
         switch (node.kind) {
-            case 258 /* ImportSpecifier */:
-            case 263 /* ExportSpecifier */:
+            case 265 /* ImportSpecifier */:
+            case 270 /* ExportSpecifier */:
                 return node.parent.parent.isTypeOnly;
-            case 256 /* NamespaceImport */:
+            case 263 /* NamespaceImport */:
                 return node.parent.isTypeOnly;
-            case 255 /* ImportClause */:
+            case 262 /* ImportClause */:
                 return node.isTypeOnly;
             default:
                 return false;
@@ -12147,35 +12500,35 @@ var ts;
     // Identifiers
     /* @internal */
     function isGeneratedIdentifier(node) {
-        return isIdentifier(node) && (node.autoGenerateFlags & 7 /* KindMask */) > 0 /* None */;
+        return ts.isIdentifier(node) && (node.autoGenerateFlags & 7 /* KindMask */) > 0 /* None */;
     }
     ts.isGeneratedIdentifier = isGeneratedIdentifier;
     // Private Identifiers
     /*@internal*/
     function isPrivateIdentifierPropertyDeclaration(node) {
-        return isPropertyDeclaration(node) && isPrivateIdentifier(node.name);
+        return ts.isPropertyDeclaration(node) && ts.isPrivateIdentifier(node.name);
     }
     ts.isPrivateIdentifierPropertyDeclaration = isPrivateIdentifierPropertyDeclaration;
     /*@internal*/
     function isPrivateIdentifierPropertyAccessExpression(node) {
-        return isPropertyAccessExpression(node) && isPrivateIdentifier(node.name);
+        return ts.isPropertyAccessExpression(node) && ts.isPrivateIdentifier(node.name);
     }
     ts.isPrivateIdentifierPropertyAccessExpression = isPrivateIdentifierPropertyAccessExpression;
     // Keywords
     /* @internal */
     function isModifierKind(token) {
         switch (token) {
-            case 122 /* AbstractKeyword */:
-            case 126 /* AsyncKeyword */:
-            case 81 /* ConstKeyword */:
-            case 130 /* DeclareKeyword */:
-            case 84 /* DefaultKeyword */:
-            case 89 /* ExportKeyword */:
-            case 119 /* PublicKeyword */:
-            case 117 /* PrivateKeyword */:
-            case 118 /* ProtectedKeyword */:
-            case 138 /* ReadonlyKeyword */:
-            case 120 /* StaticKeyword */:
+            case 125 /* AbstractKeyword */:
+            case 129 /* AsyncKeyword */:
+            case 84 /* ConstKeyword */:
+            case 133 /* DeclareKeyword */:
+            case 87 /* DefaultKeyword */:
+            case 92 /* ExportKeyword */:
+            case 122 /* PublicKeyword */:
+            case 120 /* PrivateKeyword */:
+            case 121 /* ProtectedKeyword */:
+            case 142 /* ReadonlyKeyword */:
+            case 123 /* StaticKeyword */:
                 return true;
         }
         return false;
@@ -12188,7 +12541,7 @@ var ts;
     ts.isParameterPropertyModifier = isParameterPropertyModifier;
     /* @internal */
     function isClassMemberModifier(idToken) {
-        return isParameterPropertyModifier(idToken) || idToken === 120 /* StaticKeyword */;
+        return isParameterPropertyModifier(idToken) || idToken === 123 /* StaticKeyword */;
     }
     ts.isClassMemberModifier = isClassMemberModifier;
     function isModifier(node) {
@@ -12197,24 +12550,24 @@ var ts;
     ts.isModifier = isModifier;
     function isEntityName(node) {
         var kind = node.kind;
-        return kind === 153 /* QualifiedName */
-            || kind === 75 /* Identifier */;
+        return kind === 157 /* QualifiedName */
+            || kind === 78 /* Identifier */;
     }
     ts.isEntityName = isEntityName;
     function isPropertyName(node) {
         var kind = node.kind;
-        return kind === 75 /* Identifier */
-            || kind === 76 /* PrivateIdentifier */
+        return kind === 78 /* Identifier */
+            || kind === 79 /* PrivateIdentifier */
             || kind === 10 /* StringLiteral */
             || kind === 8 /* NumericLiteral */
-            || kind === 154 /* ComputedPropertyName */;
+            || kind === 158 /* ComputedPropertyName */;
     }
     ts.isPropertyName = isPropertyName;
     function isBindingName(node) {
         var kind = node.kind;
-        return kind === 75 /* Identifier */
-            || kind === 189 /* ObjectBindingPattern */
-            || kind === 190 /* ArrayBindingPattern */;
+        return kind === 78 /* Identifier */
+            || kind === 196 /* ObjectBindingPattern */
+            || kind === 197 /* ArrayBindingPattern */;
     }
     ts.isBindingName = isBindingName;
     // Functions
@@ -12229,13 +12582,13 @@ var ts;
     ts.isFunctionLikeDeclaration = isFunctionLikeDeclaration;
     function isFunctionLikeDeclarationKind(kind) {
         switch (kind) {
-            case 244 /* FunctionDeclaration */:
-            case 161 /* MethodDeclaration */:
-            case 162 /* Constructor */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
+            case 251 /* FunctionDeclaration */:
+            case 165 /* MethodDeclaration */:
+            case 166 /* Constructor */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
                 return true;
             default:
                 return false;
@@ -12244,14 +12597,14 @@ var ts;
     /* @internal */
     function isFunctionLikeKind(kind) {
         switch (kind) {
-            case 160 /* MethodSignature */:
-            case 165 /* CallSignature */:
-            case 305 /* JSDocSignature */:
-            case 166 /* ConstructSignature */:
-            case 167 /* IndexSignature */:
-            case 170 /* FunctionType */:
-            case 300 /* JSDocFunctionType */:
-            case 171 /* ConstructorType */:
+            case 164 /* MethodSignature */:
+            case 169 /* CallSignature */:
+            case 313 /* JSDocSignature */:
+            case 170 /* ConstructSignature */:
+            case 171 /* IndexSignature */:
+            case 174 /* FunctionType */:
+            case 308 /* JSDocFunctionType */:
+            case 175 /* ConstructorType */:
                 return true;
             default:
                 return isFunctionLikeDeclarationKind(kind);
@@ -12260,35 +12613,35 @@ var ts;
     ts.isFunctionLikeKind = isFunctionLikeKind;
     /* @internal */
     function isFunctionOrModuleBlock(node) {
-        return isSourceFile(node) || isModuleBlock(node) || isBlock(node) && isFunctionLike(node.parent);
+        return ts.isSourceFile(node) || ts.isModuleBlock(node) || ts.isBlock(node) && isFunctionLike(node.parent);
     }
     ts.isFunctionOrModuleBlock = isFunctionOrModuleBlock;
     // Classes
     function isClassElement(node) {
         var kind = node.kind;
-        return kind === 162 /* Constructor */
-            || kind === 159 /* PropertyDeclaration */
-            || kind === 161 /* MethodDeclaration */
-            || kind === 163 /* GetAccessor */
-            || kind === 164 /* SetAccessor */
-            || kind === 167 /* IndexSignature */
-            || kind === 222 /* SemicolonClassElement */;
+        return kind === 166 /* Constructor */
+            || kind === 163 /* PropertyDeclaration */
+            || kind === 165 /* MethodDeclaration */
+            || kind === 167 /* GetAccessor */
+            || kind === 168 /* SetAccessor */
+            || kind === 171 /* IndexSignature */
+            || kind === 229 /* SemicolonClassElement */;
     }
     ts.isClassElement = isClassElement;
     function isClassLike(node) {
-        return node && (node.kind === 245 /* ClassDeclaration */ || node.kind === 214 /* ClassExpression */);
+        return node && (node.kind === 252 /* ClassDeclaration */ || node.kind === 221 /* ClassExpression */);
     }
     ts.isClassLike = isClassLike;
     function isAccessor(node) {
-        return node && (node.kind === 163 /* GetAccessor */ || node.kind === 164 /* SetAccessor */);
+        return node && (node.kind === 167 /* GetAccessor */ || node.kind === 168 /* SetAccessor */);
     }
     ts.isAccessor = isAccessor;
     /* @internal */
     function isMethodOrAccessor(node) {
         switch (node.kind) {
-            case 161 /* MethodDeclaration */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
+            case 165 /* MethodDeclaration */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
                 return true;
             default:
                 return false;
@@ -12298,11 +12651,11 @@ var ts;
     // Type members
     function isTypeElement(node) {
         var kind = node.kind;
-        return kind === 166 /* ConstructSignature */
-            || kind === 165 /* CallSignature */
-            || kind === 158 /* PropertySignature */
-            || kind === 160 /* MethodSignature */
-            || kind === 167 /* IndexSignature */;
+        return kind === 170 /* ConstructSignature */
+            || kind === 169 /* CallSignature */
+            || kind === 162 /* PropertySignature */
+            || kind === 164 /* MethodSignature */
+            || kind === 171 /* IndexSignature */;
     }
     ts.isTypeElement = isTypeElement;
     function isClassOrTypeElement(node) {
@@ -12311,12 +12664,12 @@ var ts;
     ts.isClassOrTypeElement = isClassOrTypeElement;
     function isObjectLiteralElementLike(node) {
         var kind = node.kind;
-        return kind === 281 /* PropertyAssignment */
-            || kind === 282 /* ShorthandPropertyAssignment */
-            || kind === 283 /* SpreadAssignment */
-            || kind === 161 /* MethodDeclaration */
-            || kind === 163 /* GetAccessor */
-            || kind === 164 /* SetAccessor */;
+        return kind === 288 /* PropertyAssignment */
+            || kind === 289 /* ShorthandPropertyAssignment */
+            || kind === 290 /* SpreadAssignment */
+            || kind === 165 /* MethodDeclaration */
+            || kind === 167 /* GetAccessor */
+            || kind === 168 /* SetAccessor */;
     }
     ts.isObjectLiteralElementLike = isObjectLiteralElementLike;
     // Type
@@ -12331,8 +12684,8 @@ var ts;
     ts.isTypeNode = isTypeNode;
     function isFunctionOrConstructorTypeNode(node) {
         switch (node.kind) {
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
+            case 174 /* FunctionType */:
+            case 175 /* ConstructorType */:
                 return true;
         }
         return false;
@@ -12343,8 +12696,8 @@ var ts;
     function isBindingPattern(node) {
         if (node) {
             var kind = node.kind;
-            return kind === 190 /* ArrayBindingPattern */
-                || kind === 189 /* ObjectBindingPattern */;
+            return kind === 197 /* ArrayBindingPattern */
+                || kind === 196 /* ObjectBindingPattern */;
         }
         return false;
     }
@@ -12352,15 +12705,15 @@ var ts;
     /* @internal */
     function isAssignmentPattern(node) {
         var kind = node.kind;
-        return kind === 192 /* ArrayLiteralExpression */
-            || kind === 193 /* ObjectLiteralExpression */;
+        return kind === 199 /* ArrayLiteralExpression */
+            || kind === 200 /* ObjectLiteralExpression */;
     }
     ts.isAssignmentPattern = isAssignmentPattern;
     /* @internal */
     function isArrayBindingElement(node) {
         var kind = node.kind;
-        return kind === 191 /* BindingElement */
-            || kind === 215 /* OmittedExpression */;
+        return kind === 198 /* BindingElement */
+            || kind === 222 /* OmittedExpression */;
     }
     ts.isArrayBindingElement = isArrayBindingElement;
     /**
@@ -12369,9 +12722,9 @@ var ts;
     /* @internal */
     function isDeclarationBindingElement(bindingElement) {
         switch (bindingElement.kind) {
-            case 242 /* VariableDeclaration */:
-            case 156 /* Parameter */:
-            case 191 /* BindingElement */:
+            case 249 /* VariableDeclaration */:
+            case 160 /* Parameter */:
+            case 198 /* BindingElement */:
                 return true;
         }
         return false;
@@ -12392,8 +12745,8 @@ var ts;
     /* @internal */
     function isObjectBindingOrAssignmentPattern(node) {
         switch (node.kind) {
-            case 189 /* ObjectBindingPattern */:
-            case 193 /* ObjectLiteralExpression */:
+            case 196 /* ObjectBindingPattern */:
+            case 200 /* ObjectLiteralExpression */:
                 return true;
         }
         return false;
@@ -12405,8 +12758,8 @@ var ts;
     /* @internal */
     function isArrayBindingOrAssignmentPattern(node) {
         switch (node.kind) {
-            case 190 /* ArrayBindingPattern */:
-            case 192 /* ArrayLiteralExpression */:
+            case 197 /* ArrayBindingPattern */:
+            case 199 /* ArrayLiteralExpression */:
                 return true;
         }
         return false;
@@ -12415,26 +12768,26 @@ var ts;
     /* @internal */
     function isPropertyAccessOrQualifiedNameOrImportTypeNode(node) {
         var kind = node.kind;
-        return kind === 194 /* PropertyAccessExpression */
-            || kind === 153 /* QualifiedName */
-            || kind === 188 /* ImportType */;
+        return kind === 201 /* PropertyAccessExpression */
+            || kind === 157 /* QualifiedName */
+            || kind === 195 /* ImportType */;
     }
     ts.isPropertyAccessOrQualifiedNameOrImportTypeNode = isPropertyAccessOrQualifiedNameOrImportTypeNode;
     // Expression
     function isPropertyAccessOrQualifiedName(node) {
         var kind = node.kind;
-        return kind === 194 /* PropertyAccessExpression */
-            || kind === 153 /* QualifiedName */;
+        return kind === 201 /* PropertyAccessExpression */
+            || kind === 157 /* QualifiedName */;
     }
     ts.isPropertyAccessOrQualifiedName = isPropertyAccessOrQualifiedName;
     function isCallLikeExpression(node) {
         switch (node.kind) {
-            case 268 /* JsxOpeningElement */:
-            case 267 /* JsxSelfClosingElement */:
-            case 196 /* CallExpression */:
-            case 197 /* NewExpression */:
-            case 198 /* TaggedTemplateExpression */:
-            case 157 /* Decorator */:
+            case 275 /* JsxOpeningElement */:
+            case 274 /* JsxSelfClosingElement */:
+            case 203 /* CallExpression */:
+            case 204 /* NewExpression */:
+            case 205 /* TaggedTemplateExpression */:
+            case 161 /* Decorator */:
                 return true;
             default:
                 return false;
@@ -12442,12 +12795,12 @@ var ts;
     }
     ts.isCallLikeExpression = isCallLikeExpression;
     function isCallOrNewExpression(node) {
-        return node.kind === 196 /* CallExpression */ || node.kind === 197 /* NewExpression */;
+        return node.kind === 203 /* CallExpression */ || node.kind === 204 /* NewExpression */;
     }
     ts.isCallOrNewExpression = isCallOrNewExpression;
     function isTemplateLiteral(node) {
         var kind = node.kind;
-        return kind === 211 /* TemplateExpression */
+        return kind === 218 /* TemplateExpression */
             || kind === 14 /* NoSubstitutionTemplateLiteral */;
     }
     ts.isTemplateLiteral = isTemplateLiteral;
@@ -12458,34 +12811,34 @@ var ts;
     ts.isLeftHandSideExpression = isLeftHandSideExpression;
     function isLeftHandSideExpressionKind(kind) {
         switch (kind) {
-            case 194 /* PropertyAccessExpression */:
-            case 195 /* ElementAccessExpression */:
-            case 197 /* NewExpression */:
-            case 196 /* CallExpression */:
-            case 266 /* JsxElement */:
-            case 267 /* JsxSelfClosingElement */:
-            case 270 /* JsxFragment */:
-            case 198 /* TaggedTemplateExpression */:
-            case 192 /* ArrayLiteralExpression */:
-            case 200 /* ParenthesizedExpression */:
-            case 193 /* ObjectLiteralExpression */:
-            case 214 /* ClassExpression */:
-            case 201 /* FunctionExpression */:
-            case 75 /* Identifier */:
+            case 201 /* PropertyAccessExpression */:
+            case 202 /* ElementAccessExpression */:
+            case 204 /* NewExpression */:
+            case 203 /* CallExpression */:
+            case 273 /* JsxElement */:
+            case 274 /* JsxSelfClosingElement */:
+            case 277 /* JsxFragment */:
+            case 205 /* TaggedTemplateExpression */:
+            case 199 /* ArrayLiteralExpression */:
+            case 207 /* ParenthesizedExpression */:
+            case 200 /* ObjectLiteralExpression */:
+            case 221 /* ClassExpression */:
+            case 208 /* FunctionExpression */:
+            case 78 /* Identifier */:
             case 13 /* RegularExpressionLiteral */:
             case 8 /* NumericLiteral */:
             case 9 /* BigIntLiteral */:
             case 10 /* StringLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
-            case 211 /* TemplateExpression */:
-            case 91 /* FalseKeyword */:
-            case 100 /* NullKeyword */:
-            case 104 /* ThisKeyword */:
-            case 106 /* TrueKeyword */:
-            case 102 /* SuperKeyword */:
-            case 218 /* NonNullExpression */:
-            case 219 /* MetaProperty */:
-            case 96 /* ImportKeyword */: // technically this is only an Expression if it's in a CallExpression
+            case 218 /* TemplateExpression */:
+            case 94 /* FalseKeyword */:
+            case 103 /* NullKeyword */:
+            case 107 /* ThisKeyword */:
+            case 109 /* TrueKeyword */:
+            case 105 /* SuperKeyword */:
+            case 225 /* NonNullExpression */:
+            case 226 /* MetaProperty */:
+            case 99 /* ImportKeyword */: // technically this is only an Expression if it's in a CallExpression
                 return true;
             default:
                 return false;
@@ -12498,13 +12851,13 @@ var ts;
     ts.isUnaryExpression = isUnaryExpression;
     function isUnaryExpressionKind(kind) {
         switch (kind) {
-            case 207 /* PrefixUnaryExpression */:
-            case 208 /* PostfixUnaryExpression */:
-            case 203 /* DeleteExpression */:
-            case 204 /* TypeOfExpression */:
-            case 205 /* VoidExpression */:
-            case 206 /* AwaitExpression */:
-            case 199 /* TypeAssertionExpression */:
+            case 214 /* PrefixUnaryExpression */:
+            case 215 /* PostfixUnaryExpression */:
+            case 210 /* DeleteExpression */:
+            case 211 /* TypeOfExpression */:
+            case 212 /* VoidExpression */:
+            case 213 /* AwaitExpression */:
+            case 206 /* TypeAssertionExpression */:
                 return true;
             default:
                 return isLeftHandSideExpressionKind(kind);
@@ -12513,9 +12866,9 @@ var ts;
     /* @internal */
     function isUnaryExpressionWithWrite(expr) {
         switch (expr.kind) {
-            case 208 /* PostfixUnaryExpression */:
+            case 215 /* PostfixUnaryExpression */:
                 return true;
-            case 207 /* PrefixUnaryExpression */:
+            case 214 /* PrefixUnaryExpression */:
                 return expr.operator === 45 /* PlusPlusToken */ ||
                     expr.operator === 46 /* MinusMinusToken */;
             default:
@@ -12534,15 +12887,15 @@ var ts;
     ts.isExpression = isExpression;
     function isExpressionKind(kind) {
         switch (kind) {
-            case 210 /* ConditionalExpression */:
-            case 212 /* YieldExpression */:
-            case 202 /* ArrowFunction */:
-            case 209 /* BinaryExpression */:
-            case 213 /* SpreadElement */:
-            case 217 /* AsExpression */:
-            case 215 /* OmittedExpression */:
-            case 327 /* CommaListExpression */:
-            case 326 /* PartiallyEmittedExpression */:
+            case 217 /* ConditionalExpression */:
+            case 219 /* YieldExpression */:
+            case 209 /* ArrowFunction */:
+            case 216 /* BinaryExpression */:
+            case 220 /* SpreadElement */:
+            case 224 /* AsExpression */:
+            case 222 /* OmittedExpression */:
+            case 337 /* CommaListExpression */:
+            case 336 /* PartiallyEmittedExpression */:
                 return true;
             default:
                 return isUnaryExpressionKind(kind);
@@ -12550,40 +12903,25 @@ var ts;
     }
     function isAssertionExpression(node) {
         var kind = node.kind;
-        return kind === 199 /* TypeAssertionExpression */
-            || kind === 217 /* AsExpression */;
+        return kind === 206 /* TypeAssertionExpression */
+            || kind === 224 /* AsExpression */;
     }
     ts.isAssertionExpression = isAssertionExpression;
     /* @internal */
-    function isPartiallyEmittedExpression(node) {
-        return node.kind === 326 /* PartiallyEmittedExpression */;
-    }
-    ts.isPartiallyEmittedExpression = isPartiallyEmittedExpression;
-    /* @internal */
-    function isNotEmittedStatement(node) {
-        return node.kind === 325 /* NotEmittedStatement */;
-    }
-    ts.isNotEmittedStatement = isNotEmittedStatement;
-    /* @internal */
-    function isSyntheticReference(node) {
-        return node.kind === 330 /* SyntheticReferenceExpression */;
-    }
-    ts.isSyntheticReference = isSyntheticReference;
-    /* @internal */
     function isNotEmittedOrPartiallyEmittedNode(node) {
-        return isNotEmittedStatement(node)
-            || isPartiallyEmittedExpression(node);
+        return ts.isNotEmittedStatement(node)
+            || ts.isPartiallyEmittedExpression(node);
     }
     ts.isNotEmittedOrPartiallyEmittedNode = isNotEmittedOrPartiallyEmittedNode;
     function isIterationStatement(node, lookInLabeledStatements) {
         switch (node.kind) {
-            case 230 /* ForStatement */:
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
-            case 228 /* DoStatement */:
-            case 229 /* WhileStatement */:
+            case 237 /* ForStatement */:
+            case 238 /* ForInStatement */:
+            case 239 /* ForOfStatement */:
+            case 235 /* DoStatement */:
+            case 236 /* WhileStatement */:
                 return true;
-            case 238 /* LabeledStatement */:
+            case 245 /* LabeledStatement */:
                 return lookInLabeledStatements && isIterationStatement(node.statement, lookInLabeledStatements);
         }
         return false;
@@ -12591,7 +12929,7 @@ var ts;
     ts.isIterationStatement = isIterationStatement;
     /* @internal */
     function isScopeMarker(node) {
-        return isExportAssignment(node) || isExportDeclaration(node);
+        return ts.isExportAssignment(node) || ts.isExportDeclaration(node);
     }
     ts.isScopeMarker = isScopeMarker;
     /* @internal */
@@ -12601,149 +12939,149 @@ var ts;
     ts.hasScopeMarker = hasScopeMarker;
     /* @internal */
     function needsScopeMarker(result) {
-        return !ts.isAnyImportOrReExport(result) && !isExportAssignment(result) && !ts.hasModifier(result, 1 /* Export */) && !ts.isAmbientModule(result);
+        return !ts.isAnyImportOrReExport(result) && !ts.isExportAssignment(result) && !ts.hasSyntacticModifier(result, 1 /* Export */) && !ts.isAmbientModule(result);
     }
     ts.needsScopeMarker = needsScopeMarker;
     /* @internal */
     function isExternalModuleIndicator(result) {
         // Exported top-level member indicates moduleness
-        return ts.isAnyImportOrReExport(result) || isExportAssignment(result) || ts.hasModifier(result, 1 /* Export */);
+        return ts.isAnyImportOrReExport(result) || ts.isExportAssignment(result) || ts.hasSyntacticModifier(result, 1 /* Export */);
     }
     ts.isExternalModuleIndicator = isExternalModuleIndicator;
     /* @internal */
     function isForInOrOfStatement(node) {
-        return node.kind === 231 /* ForInStatement */ || node.kind === 232 /* ForOfStatement */;
+        return node.kind === 238 /* ForInStatement */ || node.kind === 239 /* ForOfStatement */;
     }
     ts.isForInOrOfStatement = isForInOrOfStatement;
     // Element
     /* @internal */
     function isConciseBody(node) {
-        return isBlock(node)
+        return ts.isBlock(node)
             || isExpression(node);
     }
     ts.isConciseBody = isConciseBody;
     /* @internal */
     function isFunctionBody(node) {
-        return isBlock(node);
+        return ts.isBlock(node);
     }
     ts.isFunctionBody = isFunctionBody;
     /* @internal */
     function isForInitializer(node) {
-        return isVariableDeclarationList(node)
+        return ts.isVariableDeclarationList(node)
             || isExpression(node);
     }
     ts.isForInitializer = isForInitializer;
     /* @internal */
     function isModuleBody(node) {
         var kind = node.kind;
-        return kind === 250 /* ModuleBlock */
-            || kind === 249 /* ModuleDeclaration */
-            || kind === 75 /* Identifier */;
+        return kind === 257 /* ModuleBlock */
+            || kind === 256 /* ModuleDeclaration */
+            || kind === 78 /* Identifier */;
     }
     ts.isModuleBody = isModuleBody;
     /* @internal */
     function isNamespaceBody(node) {
         var kind = node.kind;
-        return kind === 250 /* ModuleBlock */
-            || kind === 249 /* ModuleDeclaration */;
+        return kind === 257 /* ModuleBlock */
+            || kind === 256 /* ModuleDeclaration */;
     }
     ts.isNamespaceBody = isNamespaceBody;
     /* @internal */
     function isJSDocNamespaceBody(node) {
         var kind = node.kind;
-        return kind === 75 /* Identifier */
-            || kind === 249 /* ModuleDeclaration */;
+        return kind === 78 /* Identifier */
+            || kind === 256 /* ModuleDeclaration */;
     }
     ts.isJSDocNamespaceBody = isJSDocNamespaceBody;
     /* @internal */
     function isNamedImportBindings(node) {
         var kind = node.kind;
-        return kind === 257 /* NamedImports */
-            || kind === 256 /* NamespaceImport */;
+        return kind === 264 /* NamedImports */
+            || kind === 263 /* NamespaceImport */;
     }
     ts.isNamedImportBindings = isNamedImportBindings;
     /* @internal */
     function isModuleOrEnumDeclaration(node) {
-        return node.kind === 249 /* ModuleDeclaration */ || node.kind === 248 /* EnumDeclaration */;
+        return node.kind === 256 /* ModuleDeclaration */ || node.kind === 255 /* EnumDeclaration */;
     }
     ts.isModuleOrEnumDeclaration = isModuleOrEnumDeclaration;
     function isDeclarationKind(kind) {
-        return kind === 202 /* ArrowFunction */
-            || kind === 191 /* BindingElement */
-            || kind === 245 /* ClassDeclaration */
-            || kind === 214 /* ClassExpression */
-            || kind === 162 /* Constructor */
-            || kind === 248 /* EnumDeclaration */
-            || kind === 284 /* EnumMember */
-            || kind === 263 /* ExportSpecifier */
-            || kind === 244 /* FunctionDeclaration */
-            || kind === 201 /* FunctionExpression */
-            || kind === 163 /* GetAccessor */
-            || kind === 255 /* ImportClause */
-            || kind === 253 /* ImportEqualsDeclaration */
-            || kind === 258 /* ImportSpecifier */
-            || kind === 246 /* InterfaceDeclaration */
-            || kind === 273 /* JsxAttribute */
-            || kind === 161 /* MethodDeclaration */
-            || kind === 160 /* MethodSignature */
-            || kind === 249 /* ModuleDeclaration */
-            || kind === 252 /* NamespaceExportDeclaration */
-            || kind === 256 /* NamespaceImport */
-            || kind === 262 /* NamespaceExport */
-            || kind === 156 /* Parameter */
-            || kind === 281 /* PropertyAssignment */
-            || kind === 159 /* PropertyDeclaration */
-            || kind === 158 /* PropertySignature */
-            || kind === 164 /* SetAccessor */
-            || kind === 282 /* ShorthandPropertyAssignment */
-            || kind === 247 /* TypeAliasDeclaration */
-            || kind === 155 /* TypeParameter */
-            || kind === 242 /* VariableDeclaration */
-            || kind === 322 /* JSDocTypedefTag */
-            || kind === 315 /* JSDocCallbackTag */
-            || kind === 323 /* JSDocPropertyTag */;
+        return kind === 209 /* ArrowFunction */
+            || kind === 198 /* BindingElement */
+            || kind === 252 /* ClassDeclaration */
+            || kind === 221 /* ClassExpression */
+            || kind === 166 /* Constructor */
+            || kind === 255 /* EnumDeclaration */
+            || kind === 291 /* EnumMember */
+            || kind === 270 /* ExportSpecifier */
+            || kind === 251 /* FunctionDeclaration */
+            || kind === 208 /* FunctionExpression */
+            || kind === 167 /* GetAccessor */
+            || kind === 262 /* ImportClause */
+            || kind === 260 /* ImportEqualsDeclaration */
+            || kind === 265 /* ImportSpecifier */
+            || kind === 253 /* InterfaceDeclaration */
+            || kind === 280 /* JsxAttribute */
+            || kind === 165 /* MethodDeclaration */
+            || kind === 164 /* MethodSignature */
+            || kind === 256 /* ModuleDeclaration */
+            || kind === 259 /* NamespaceExportDeclaration */
+            || kind === 263 /* NamespaceImport */
+            || kind === 269 /* NamespaceExport */
+            || kind === 160 /* Parameter */
+            || kind === 288 /* PropertyAssignment */
+            || kind === 163 /* PropertyDeclaration */
+            || kind === 162 /* PropertySignature */
+            || kind === 168 /* SetAccessor */
+            || kind === 289 /* ShorthandPropertyAssignment */
+            || kind === 254 /* TypeAliasDeclaration */
+            || kind === 159 /* TypeParameter */
+            || kind === 249 /* VariableDeclaration */
+            || kind === 331 /* JSDocTypedefTag */
+            || kind === 324 /* JSDocCallbackTag */
+            || kind === 333 /* JSDocPropertyTag */;
     }
     function isDeclarationStatementKind(kind) {
-        return kind === 244 /* FunctionDeclaration */
-            || kind === 264 /* MissingDeclaration */
-            || kind === 245 /* ClassDeclaration */
-            || kind === 246 /* InterfaceDeclaration */
-            || kind === 247 /* TypeAliasDeclaration */
-            || kind === 248 /* EnumDeclaration */
-            || kind === 249 /* ModuleDeclaration */
-            || kind === 254 /* ImportDeclaration */
-            || kind === 253 /* ImportEqualsDeclaration */
-            || kind === 260 /* ExportDeclaration */
-            || kind === 259 /* ExportAssignment */
-            || kind === 252 /* NamespaceExportDeclaration */;
+        return kind === 251 /* FunctionDeclaration */
+            || kind === 271 /* MissingDeclaration */
+            || kind === 252 /* ClassDeclaration */
+            || kind === 253 /* InterfaceDeclaration */
+            || kind === 254 /* TypeAliasDeclaration */
+            || kind === 255 /* EnumDeclaration */
+            || kind === 256 /* ModuleDeclaration */
+            || kind === 261 /* ImportDeclaration */
+            || kind === 260 /* ImportEqualsDeclaration */
+            || kind === 267 /* ExportDeclaration */
+            || kind === 266 /* ExportAssignment */
+            || kind === 259 /* NamespaceExportDeclaration */;
     }
     function isStatementKindButNotDeclarationKind(kind) {
-        return kind === 234 /* BreakStatement */
-            || kind === 233 /* ContinueStatement */
-            || kind === 241 /* DebuggerStatement */
-            || kind === 228 /* DoStatement */
-            || kind === 226 /* ExpressionStatement */
-            || kind === 224 /* EmptyStatement */
-            || kind === 231 /* ForInStatement */
-            || kind === 232 /* ForOfStatement */
-            || kind === 230 /* ForStatement */
-            || kind === 227 /* IfStatement */
-            || kind === 238 /* LabeledStatement */
-            || kind === 235 /* ReturnStatement */
-            || kind === 237 /* SwitchStatement */
-            || kind === 239 /* ThrowStatement */
-            || kind === 240 /* TryStatement */
-            || kind === 225 /* VariableStatement */
-            || kind === 229 /* WhileStatement */
-            || kind === 236 /* WithStatement */
-            || kind === 325 /* NotEmittedStatement */
-            || kind === 329 /* EndOfDeclarationMarker */
-            || kind === 328 /* MergeDeclarationMarker */;
+        return kind === 241 /* BreakStatement */
+            || kind === 240 /* ContinueStatement */
+            || kind === 248 /* DebuggerStatement */
+            || kind === 235 /* DoStatement */
+            || kind === 233 /* ExpressionStatement */
+            || kind === 231 /* EmptyStatement */
+            || kind === 238 /* ForInStatement */
+            || kind === 239 /* ForOfStatement */
+            || kind === 237 /* ForStatement */
+            || kind === 234 /* IfStatement */
+            || kind === 245 /* LabeledStatement */
+            || kind === 242 /* ReturnStatement */
+            || kind === 244 /* SwitchStatement */
+            || kind === 246 /* ThrowStatement */
+            || kind === 247 /* TryStatement */
+            || kind === 232 /* VariableStatement */
+            || kind === 236 /* WhileStatement */
+            || kind === 243 /* WithStatement */
+            || kind === 335 /* NotEmittedStatement */
+            || kind === 339 /* EndOfDeclarationMarker */
+            || kind === 338 /* MergeDeclarationMarker */;
     }
     /* @internal */
     function isDeclaration(node) {
-        if (node.kind === 155 /* TypeParameter */) {
-            return (node.parent && node.parent.kind !== 321 /* JSDocTemplateTag */) || ts.isInJSFile(node);
+        if (node.kind === 159 /* TypeParameter */) {
+            return (node.parent && node.parent.kind !== 330 /* JSDocTemplateTag */) || ts.isInJSFile(node);
         }
         return isDeclarationKind(node.kind);
     }
@@ -12770,94 +13108,105 @@ var ts;
     }
     ts.isStatement = isStatement;
     function isBlockStatement(node) {
-        if (node.kind !== 223 /* Block */)
+        if (node.kind !== 230 /* Block */)
             return false;
         if (node.parent !== undefined) {
-            if (node.parent.kind === 240 /* TryStatement */ || node.parent.kind === 280 /* CatchClause */) {
+            if (node.parent.kind === 247 /* TryStatement */ || node.parent.kind === 287 /* CatchClause */) {
                 return false;
             }
         }
         return !ts.isFunctionBlock(node);
     }
+    /**
+     * NOTE: This is similar to `isStatement` but does not access parent pointers.
+     */
+    /* @internal */
+    function isStatementOrBlock(node) {
+        var kind = node.kind;
+        return isStatementKindButNotDeclarationKind(kind)
+            || isDeclarationStatementKind(kind)
+            || kind === 230 /* Block */;
+    }
+    ts.isStatementOrBlock = isStatementOrBlock;
     // Module references
     /* @internal */
     function isModuleReference(node) {
         var kind = node.kind;
-        return kind === 265 /* ExternalModuleReference */
-            || kind === 153 /* QualifiedName */
-            || kind === 75 /* Identifier */;
+        return kind === 272 /* ExternalModuleReference */
+            || kind === 157 /* QualifiedName */
+            || kind === 78 /* Identifier */;
     }
     ts.isModuleReference = isModuleReference;
     // JSX
     /* @internal */
     function isJsxTagNameExpression(node) {
         var kind = node.kind;
-        return kind === 104 /* ThisKeyword */
-            || kind === 75 /* Identifier */
-            || kind === 194 /* PropertyAccessExpression */;
+        return kind === 107 /* ThisKeyword */
+            || kind === 78 /* Identifier */
+            || kind === 201 /* PropertyAccessExpression */;
     }
     ts.isJsxTagNameExpression = isJsxTagNameExpression;
     /* @internal */
     function isJsxChild(node) {
         var kind = node.kind;
-        return kind === 266 /* JsxElement */
-            || kind === 276 /* JsxExpression */
-            || kind === 267 /* JsxSelfClosingElement */
+        return kind === 273 /* JsxElement */
+            || kind === 283 /* JsxExpression */
+            || kind === 274 /* JsxSelfClosingElement */
             || kind === 11 /* JsxText */
-            || kind === 270 /* JsxFragment */;
+            || kind === 277 /* JsxFragment */;
     }
     ts.isJsxChild = isJsxChild;
     /* @internal */
     function isJsxAttributeLike(node) {
         var kind = node.kind;
-        return kind === 273 /* JsxAttribute */
-            || kind === 275 /* JsxSpreadAttribute */;
+        return kind === 280 /* JsxAttribute */
+            || kind === 282 /* JsxSpreadAttribute */;
     }
     ts.isJsxAttributeLike = isJsxAttributeLike;
     /* @internal */
     function isStringLiteralOrJsxExpression(node) {
         var kind = node.kind;
         return kind === 10 /* StringLiteral */
-            || kind === 276 /* JsxExpression */;
+            || kind === 283 /* JsxExpression */;
     }
     ts.isStringLiteralOrJsxExpression = isStringLiteralOrJsxExpression;
     function isJsxOpeningLikeElement(node) {
         var kind = node.kind;
-        return kind === 268 /* JsxOpeningElement */
-            || kind === 267 /* JsxSelfClosingElement */;
+        return kind === 275 /* JsxOpeningElement */
+            || kind === 274 /* JsxSelfClosingElement */;
     }
     ts.isJsxOpeningLikeElement = isJsxOpeningLikeElement;
     // Clauses
     function isCaseOrDefaultClause(node) {
         var kind = node.kind;
-        return kind === 277 /* CaseClause */
-            || kind === 278 /* DefaultClause */;
+        return kind === 284 /* CaseClause */
+            || kind === 285 /* DefaultClause */;
     }
     ts.isCaseOrDefaultClause = isCaseOrDefaultClause;
     // JSDoc
     /** True if node is of some JSDoc syntax kind. */
     /* @internal */
     function isJSDocNode(node) {
-        return node.kind >= 294 /* FirstJSDocNode */ && node.kind <= 323 /* LastJSDocNode */;
+        return node.kind >= 301 /* FirstJSDocNode */ && node.kind <= 333 /* LastJSDocNode */;
     }
     ts.isJSDocNode = isJSDocNode;
     /** True if node is of a kind that may contain comment text. */
     function isJSDocCommentContainingNode(node) {
-        return node.kind === 303 /* JSDocComment */ || node.kind === 302 /* JSDocNamepathType */ || isJSDocTag(node) || isJSDocTypeLiteral(node) || isJSDocSignature(node);
+        return node.kind === 311 /* JSDocComment */ || node.kind === 310 /* JSDocNamepathType */ || isJSDocTag(node) || ts.isJSDocTypeLiteral(node) || ts.isJSDocSignature(node);
     }
     ts.isJSDocCommentContainingNode = isJSDocCommentContainingNode;
     // TODO: determine what this does before making it public.
     /* @internal */
     function isJSDocTag(node) {
-        return node.kind >= 306 /* FirstJSDocTagNode */ && node.kind <= 323 /* LastJSDocTagNode */;
+        return node.kind >= 314 /* FirstJSDocTagNode */ && node.kind <= 333 /* LastJSDocTagNode */;
     }
     ts.isJSDocTag = isJSDocTag;
     function isSetAccessor(node) {
-        return node.kind === 164 /* SetAccessor */;
+        return node.kind === 168 /* SetAccessor */;
     }
     ts.isSetAccessor = isSetAccessor;
     function isGetAccessor(node) {
-        return node.kind === 163 /* GetAccessor */;
+        return node.kind === 167 /* GetAccessor */;
     }
     ts.isGetAccessor = isGetAccessor;
     /** True if has jsdoc nodes attached to it. */
@@ -12883,13 +13232,13 @@ var ts;
     /** True if has initializer node attached to it. */
     function hasOnlyExpressionInitializer(node) {
         switch (node.kind) {
-            case 242 /* VariableDeclaration */:
-            case 156 /* Parameter */:
-            case 191 /* BindingElement */:
-            case 158 /* PropertySignature */:
-            case 159 /* PropertyDeclaration */:
-            case 281 /* PropertyAssignment */:
-            case 284 /* EnumMember */:
+            case 249 /* VariableDeclaration */:
+            case 160 /* Parameter */:
+            case 198 /* BindingElement */:
+            case 162 /* PropertySignature */:
+            case 163 /* PropertyDeclaration */:
+            case 288 /* PropertyAssignment */:
+            case 291 /* EnumMember */:
                 return true;
             default:
                 return false;
@@ -12897,12 +13246,12 @@ var ts;
     }
     ts.hasOnlyExpressionInitializer = hasOnlyExpressionInitializer;
     function isObjectLiteralElement(node) {
-        return node.kind === 273 /* JsxAttribute */ || node.kind === 275 /* JsxSpreadAttribute */ || isObjectLiteralElementLike(node);
+        return node.kind === 280 /* JsxAttribute */ || node.kind === 282 /* JsxSpreadAttribute */ || isObjectLiteralElementLike(node);
     }
     ts.isObjectLiteralElement = isObjectLiteralElement;
     /* @internal */
     function isTypeReferenceType(node) {
-        return node.kind === 169 /* TypeReference */ || node.kind === 216 /* ExpressionWithTypeArguments */;
+        return node.kind === 173 /* TypeReference */ || node.kind === 223 /* ExpressionWithTypeArguments */;
     }
     ts.isTypeReferenceType = isTypeReferenceType;
     var MAX_SMI_X86 = 1073741823;
@@ -12940,8 +13289,6 @@ var ts;
 var ts;
 (function (ts) {
     ts.resolvingEmptyArray = [];
-    ts.emptyMap = ts.createMap();
-    ts.emptyUnderscoreEscapedMap = ts.emptyMap;
     ts.externalHelpersModuleNameText = "tslib";
     ts.defaultMaximumTruncationLength = 160;
     ts.noTruncationMaximumTruncationLength = 1000000;
@@ -12958,17 +13305,23 @@ var ts;
         return undefined;
     }
     ts.getDeclarationOfKind = getDeclarationOfKind;
-    /** Create a new escaped identifier map. */
+    /**
+     * Create a new escaped identifier map.
+     * @deprecated Use `new Map<__String, T>()` instead.
+     */
     function createUnderscoreEscapedMap() {
         return new ts.Map();
     }
     ts.createUnderscoreEscapedMap = createUnderscoreEscapedMap;
+    /**
+     * @deprecated Use `!!map?.size` instead
+     */
     function hasEntries(map) {
         return !!map && !!map.size;
     }
     ts.hasEntries = hasEntries;
     function createSymbolTable(symbols) {
-        var result = ts.createMap();
+        var result = new ts.Map();
         if (symbols) {
             for (var _i = 0, symbols_1 = symbols; _i < symbols_1.length; _i++) {
                 var symbol = symbols_1[_i];
@@ -13031,20 +13384,6 @@ var ts;
         });
     }
     ts.optionsHaveModuleResolutionChanges = optionsHaveModuleResolutionChanges;
-    function findAncestor(node, callback) {
-        while (node) {
-            var result = callback(node);
-            if (result === "quit") {
-                return undefined;
-            }
-            else if (result) {
-                return node;
-            }
-            node = node.parent;
-        }
-        return undefined;
-    }
-    ts.findAncestor = findAncestor;
     function forEachAncestor(node, callback) {
         while (true) {
             var res = callback(node);
@@ -13058,6 +13397,10 @@ var ts;
         }
     }
     ts.forEachAncestor = forEachAncestor;
+    /**
+     * Calls `callback` for each entry in the map, returning the first truthy result.
+     * Use `map.forEach` instead for normal iteration.
+     */
     function forEachEntry(map, callback) {
         var iterator = map.entries();
         for (var iterResult = iterator.next(); !iterResult.done; iterResult = iterator.next()) {
@@ -13070,6 +13413,7 @@ var ts;
         return undefined;
     }
     ts.forEachEntry = forEachEntry;
+    /** `forEachEntry` for just keys. */
     function forEachKey(map, callback) {
         var iterator = map.keys();
         for (var iterResult = iterator.next(); !iterResult.done; iterResult = iterator.next()) {
@@ -13081,22 +13425,13 @@ var ts;
         return undefined;
     }
     ts.forEachKey = forEachKey;
+    /** Copy entries from `source` to `target`. */
     function copyEntries(source, target) {
         source.forEach(function (value, key) {
             target.set(key, value);
         });
     }
     ts.copyEntries = copyEntries;
-    function arrayToSet(array, makeKey) {
-        return ts.arrayToMap(array, makeKey || (function (s) { return s; }), ts.returnTrue);
-    }
-    ts.arrayToSet = arrayToSet;
-    function cloneMap(map) {
-        var clone = ts.createMap();
-        copyEntries(map, clone);
-        return clone;
-    }
-    ts.cloneMap = cloneMap;
     function usingSingleLineStringWriter(action) {
         var oldString = stringWriter.getText();
         try {
@@ -13119,14 +13454,14 @@ var ts;
     ts.getResolvedModule = getResolvedModule;
     function setResolvedModule(sourceFile, moduleNameText, resolvedModule) {
         if (!sourceFile.resolvedModules) {
-            sourceFile.resolvedModules = ts.createMap();
+            sourceFile.resolvedModules = new ts.Map();
         }
         sourceFile.resolvedModules.set(moduleNameText, resolvedModule);
     }
     ts.setResolvedModule = setResolvedModule;
     function setResolvedTypeReferenceDirective(sourceFile, typeReferenceDirectiveName, resolvedTypeReferenceDirective) {
         if (!sourceFile.resolvedTypeReferenceDirectiveNames) {
-            sourceFile.resolvedTypeReferenceDirectiveNames = ts.createMap();
+            sourceFile.resolvedTypeReferenceDirectiveNames = new ts.Map();
         }
         sourceFile.resolvedTypeReferenceDirectiveNames.set(typeReferenceDirectiveName, resolvedTypeReferenceDirective);
     }
@@ -13197,7 +13532,7 @@ var ts;
         }
     }
     function getSourceFileOfNode(node) {
-        while (node && node.kind !== 290 /* SourceFile */) {
+        while (node && node.kind !== 297 /* SourceFile */) {
             node = node.parent;
         }
         return node;
@@ -13205,11 +13540,11 @@ var ts;
     ts.getSourceFileOfNode = getSourceFileOfNode;
     function isStatementWithLocals(node) {
         switch (node.kind) {
-            case 223 /* Block */:
-            case 251 /* CaseBlock */:
-            case 230 /* ForStatement */:
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
+            case 230 /* Block */:
+            case 258 /* CaseBlock */:
+            case 237 /* ForStatement */:
+            case 238 /* ForInStatement */:
+            case 239 /* ForOfStatement */:
                 return true;
         }
         return false;
@@ -13362,11 +13697,11 @@ var ts;
     }
     ts.isPinnedComment = isPinnedComment;
     function createCommentDirectivesMap(sourceFile, commentDirectives) {
-        var directivesByLine = ts.createMapFromEntries(commentDirectives.map(function (commentDirective) { return ([
+        var directivesByLine = new ts.Map(commentDirectives.map(function (commentDirective) { return ([
             "" + ts.getLineAndCharacterOfPosition(sourceFile, commentDirective.range.end).line,
             commentDirective,
         ]); }));
-        var usedLines = ts.createMap();
+        var usedLines = new ts.Map();
         return { getUnusedExpectations: getUnusedExpectations, markUsed: markUsed };
         function getUnusedExpectations() {
             return ts.arrayFrom(directivesByLine.entries())
@@ -13394,7 +13729,8 @@ var ts;
         if (nodeIsMissing(node)) {
             return node.pos;
         }
-        if (ts.isJSDocNode(node)) {
+        if (ts.isJSDocNode(node) || node.kind === 11 /* JsxText */) {
+            // JsxText cannot actually contain comments, even though the scanner will think it sees comments
             return ts.skipTrivia((sourceFile || getSourceFileOfNode(node)).text, node.pos, /*stopAfterLineBreak*/ false, /*stopAtComments*/ true);
         }
         if (includeJsDoc && ts.hasJSDocNodes(node)) {
@@ -13404,7 +13740,7 @@ var ts;
         // the syntax list itself considers them as normal trivia. Therefore if we simply skip
         // trivia for the list, we may have skipped the JSDocComment as well. So we should process its
         // first child to determine the actual position of its first token.
-        if (node.kind === 324 /* SyntaxList */ && node._children.length > 0) {
+        if (node.kind === 334 /* SyntaxList */ && node._children.length > 0) {
             return getTokenPosOfNode(node._children[0], sourceFile, includeJsDoc);
         }
         return ts.skipTrivia((sourceFile || getSourceFileOfNode(node)).text, node.pos);
@@ -13423,8 +13759,12 @@ var ts;
     }
     ts.getSourceTextOfNodeFromSourceFile = getSourceTextOfNodeFromSourceFile;
     function isJSDocTypeExpressionOrChild(node) {
-        return !!findAncestor(node, ts.isJSDocTypeExpression);
+        return !!ts.findAncestor(node, ts.isJSDocTypeExpression);
     }
+    function isExportNamespaceAsDefaultDeclaration(node) {
+        return !!(ts.isExportDeclaration(node) && node.exportClause && ts.isNamespaceExport(node.exportClause) && node.exportClause.name.escapedText === "default");
+    }
+    ts.isExportNamespaceAsDefaultDeclaration = isExportNamespaceAsDefaultDeclaration;
     function getTextOfNodeFromSourceText(sourceText, node, includeTrivia) {
         if (includeTrivia === void 0) { includeTrivia = false; }
         if (nodeIsMissing(node)) {
@@ -13462,10 +13802,85 @@ var ts;
         return emitNode && emitNode.flags || 0;
     }
     ts.getEmitFlags = getEmitFlags;
-    function getLiteralText(node, sourceFile, neverAsciiEscape, jsxAttributeEscape) {
+    ;
+    function getScriptTargetFeatures() {
+        return {
+            es2015: {
+                Array: ["find", "findIndex", "fill", "copyWithin", "entries", "keys", "values"],
+                RegExp: ["flags", "sticky", "unicode"],
+                Reflect: ["apply", "construct", "defineProperty", "deleteProperty", "get", " getOwnPropertyDescriptor", "getPrototypeOf", "has", "isExtensible", "ownKeys", "preventExtensions", "set", "setPrototypeOf"],
+                ArrayConstructor: ["from", "of"],
+                ObjectConstructor: ["assign", "getOwnPropertySymbols", "keys", "is", "setPrototypeOf"],
+                NumberConstructor: ["isFinite", "isInteger", "isNaN", "isSafeInteger", "parseFloat", "parseInt"],
+                Math: ["clz32", "imul", "sign", "log10", "log2", "log1p", "expm1", "cosh", "sinh", "tanh", "acosh", "asinh", "atanh", "hypot", "trunc", "fround", "cbrt"],
+                Map: ["entries", "keys", "values"],
+                Set: ["entries", "keys", "values"],
+                Promise: ts.emptyArray,
+                PromiseConstructor: ["all", "race", "reject", "resolve"],
+                Symbol: ["for", "keyFor"],
+                WeakMap: ["entries", "keys", "values"],
+                WeakSet: ["entries", "keys", "values"],
+                Iterator: ts.emptyArray,
+                AsyncIterator: ts.emptyArray,
+                String: ["codePointAt", "includes", "endsWith", "normalize", "repeat", "startsWith", "anchor", "big", "blink", "bold", "fixed", "fontcolor", "fontsize", "italics", "link", "small", "strike", "sub", "sup"],
+                StringConstructor: ["fromCodePoint", "raw"]
+            },
+            es2016: {
+                Array: ["includes"]
+            },
+            es2017: {
+                Atomics: ts.emptyArray,
+                SharedArrayBuffer: ts.emptyArray,
+                String: ["padStart", "padEnd"],
+                ObjectConstructor: ["values", "entries", "getOwnPropertyDescriptors"],
+                DateTimeFormat: ["formatToParts"]
+            },
+            es2018: {
+                Promise: ["finally"],
+                RegExpMatchArray: ["groups"],
+                RegExpExecArray: ["groups"],
+                RegExp: ["dotAll"],
+                Intl: ["PluralRules"],
+                AsyncIterable: ts.emptyArray,
+                AsyncIterableIterator: ts.emptyArray,
+                AsyncGenerator: ts.emptyArray,
+                AsyncGeneratorFunction: ts.emptyArray,
+            },
+            es2019: {
+                Array: ["flat", "flatMap"],
+                ObjectConstructor: ["fromEntries"],
+                String: ["trimStart", "trimEnd", "trimLeft", "trimRight"],
+                Symbol: ["description"]
+            },
+            es2020: {
+                BigInt: ts.emptyArray,
+                BigInt64Array: ts.emptyArray,
+                BigUint64Array: ts.emptyArray,
+                PromiseConstructor: ["allSettled"],
+                SymbolConstructor: ["matchAll"],
+                String: ["matchAll"],
+                DataView: ["setBigInt64", "setBigUint64", "getBigInt64", "getBigUint64"],
+                RelativeTimeFormat: ["format", "formatToParts", "resolvedOptions"]
+            },
+            esnext: {
+                PromiseConstructor: ["any"],
+                String: ["replaceAll"],
+                NumberFormat: ["formatToParts"]
+            }
+        };
+    }
+    ts.getScriptTargetFeatures = getScriptTargetFeatures;
+    var GetLiteralTextFlags;
+    (function (GetLiteralTextFlags) {
+        GetLiteralTextFlags[GetLiteralTextFlags["None"] = 0] = "None";
+        GetLiteralTextFlags[GetLiteralTextFlags["NeverAsciiEscape"] = 1] = "NeverAsciiEscape";
+        GetLiteralTextFlags[GetLiteralTextFlags["JsxAttributeEscape"] = 2] = "JsxAttributeEscape";
+        GetLiteralTextFlags[GetLiteralTextFlags["TerminateUnterminatedLiterals"] = 4] = "TerminateUnterminatedLiterals";
+    })(GetLiteralTextFlags = ts.GetLiteralTextFlags || (ts.GetLiteralTextFlags = {}));
+    function getLiteralText(node, sourceFile, flags) {
         // If we don't need to downlevel and we can reach the original source text using
         // the node's parent reference, then simply get the text as it was originally written.
-        if (!nodeIsSynthesized(node) && node.parent && !((ts.isNumericLiteral(node) && node.numericLiteralFlags & 512 /* ContainsSeparator */) ||
+        if (!nodeIsSynthesized(node) && node.parent && !(flags & 4 /* TerminateUnterminatedLiterals */ && node.isUnterminated) && !((ts.isNumericLiteral(node) && node.numericLiteralFlags & 512 /* ContainsSeparator */) ||
             ts.isBigIntLiteral(node))) {
             return getSourceTextOfNodeFromSourceFile(sourceFile, node);
         }
@@ -13473,8 +13888,8 @@ var ts;
         // or a (possibly escaped) quoted form of the original text if it's string-like.
         switch (node.kind) {
             case 10 /* StringLiteral */: {
-                var escapeText = jsxAttributeEscape ? escapeJsxAttributeString :
-                    neverAsciiEscape || (getEmitFlags(node) & 16777216 /* NoAsciiEscaping */) ? escapeString :
+                var escapeText = flags & 2 /* JsxAttributeEscape */ ? escapeJsxAttributeString :
+                    flags & 1 /* NeverAsciiEscape */ || (getEmitFlags(node) & 16777216 /* NoAsciiEscaping */) ? escapeString :
                         escapeNonAsciiString;
                 if (node.singleQuote) {
                     return "'" + escapeText(node.text, 39 /* singleQuote */) + "'";
@@ -13489,7 +13904,7 @@ var ts;
             case 17 /* TemplateTail */: {
                 // If a NoSubstitutionTemplateLiteral appears to have a substitution in it, the original text
                 // had to include a backslash: `not \${a} substitution`.
-                var escapeText = neverAsciiEscape || (getEmitFlags(node) & 16777216 /* NoAsciiEscaping */) ? escapeString :
+                var escapeText = flags & 1 /* NeverAsciiEscape */ || (getEmitFlags(node) & 16777216 /* NoAsciiEscaping */) ? escapeString :
                     escapeNonAsciiString;
                 var rawText = node.rawText || escapeTemplateSubstitution(escapeText(node.text, 96 /* backtick */));
                 switch (node.kind) {
@@ -13506,7 +13921,11 @@ var ts;
             }
             case 8 /* NumericLiteral */:
             case 9 /* BigIntLiteral */:
+                return node.text;
             case 13 /* RegularExpressionLiteral */:
+                if (flags & 4 /* TerminateUnterminatedLiterals */ && node.isUnterminated) {
+                    return node.text + (node.text.charCodeAt(node.text.length - 1) === 92 /* backslash */ ? " /" : "/");
+                }
                 return node.text;
         }
         return ts.Debug.fail("Literal kind '" + node.kind + "' not accounted for.");
@@ -13529,7 +13948,7 @@ var ts;
     ts.isBlockOrCatchScoped = isBlockOrCatchScoped;
     function isCatchClauseVariableDeclarationOrBindingElement(declaration) {
         var node = getRootDeclaration(declaration);
-        return node.kind === 242 /* VariableDeclaration */ && node.parent.kind === 280 /* CatchClause */;
+        return node.kind === 249 /* VariableDeclaration */ && node.parent.kind === 287 /* CatchClause */;
     }
     ts.isCatchClauseVariableDeclarationOrBindingElement = isCatchClauseVariableDeclarationOrBindingElement;
     function isAmbientModule(node) {
@@ -13561,11 +13980,11 @@ var ts;
     ts.isShorthandAmbientModuleSymbol = isShorthandAmbientModuleSymbol;
     function isShorthandAmbientModule(node) {
         // The only kind of module that can be missing a body is a shorthand ambient module.
-        return node && node.kind === 249 /* ModuleDeclaration */ && (!node.body);
+        return node && node.kind === 256 /* ModuleDeclaration */ && (!node.body);
     }
     function isBlockScopedContainerTopLevel(node) {
-        return node.kind === 290 /* SourceFile */ ||
-            node.kind === 249 /* ModuleDeclaration */ ||
+        return node.kind === 297 /* SourceFile */ ||
+            node.kind === 256 /* ModuleDeclaration */ ||
             ts.isFunctionLike(node);
     }
     ts.isBlockScopedContainerTopLevel = isBlockScopedContainerTopLevel;
@@ -13582,9 +14001,9 @@ var ts;
         // - defined in the top level scope and source file is an external module
         // - defined inside ambient module declaration located in the top level scope and source file not an external module
         switch (node.parent.kind) {
-            case 290 /* SourceFile */:
+            case 297 /* SourceFile */:
                 return ts.isExternalModule(node.parent);
-            case 250 /* ModuleBlock */:
+            case 257 /* ModuleBlock */:
                 return isAmbientModule(node.parent.parent) && ts.isSourceFile(node.parent.parent.parent) && !ts.isExternalModule(node.parent.parent.parent);
         }
         return false;
@@ -13637,22 +14056,22 @@ var ts;
     ts.isEffectiveStrictModeSourceFile = isEffectiveStrictModeSourceFile;
     function isBlockScope(node, parentNode) {
         switch (node.kind) {
-            case 290 /* SourceFile */:
-            case 251 /* CaseBlock */:
-            case 280 /* CatchClause */:
-            case 249 /* ModuleDeclaration */:
-            case 230 /* ForStatement */:
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
-            case 162 /* Constructor */:
-            case 161 /* MethodDeclaration */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
+            case 297 /* SourceFile */:
+            case 258 /* CaseBlock */:
+            case 287 /* CatchClause */:
+            case 256 /* ModuleDeclaration */:
+            case 237 /* ForStatement */:
+            case 238 /* ForInStatement */:
+            case 239 /* ForOfStatement */:
+            case 166 /* Constructor */:
+            case 165 /* MethodDeclaration */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
                 return true;
-            case 223 /* Block */:
+            case 230 /* Block */:
                 // function block is not considered block-scope container
                 // see comment in binder.ts: bind(...), case for SyntaxKind.Block
                 return !ts.isFunctionLike(parentNode);
@@ -13662,9 +14081,9 @@ var ts;
     ts.isBlockScope = isBlockScope;
     function isDeclarationWithTypeParameters(node) {
         switch (node.kind) {
-            case 315 /* JSDocCallbackTag */:
-            case 322 /* JSDocTypedefTag */:
-            case 305 /* JSDocSignature */:
+            case 324 /* JSDocCallbackTag */:
+            case 331 /* JSDocTypedefTag */:
+            case 313 /* JSDocSignature */:
                 return true;
             default:
                 ts.assertType(node);
@@ -13674,25 +14093,25 @@ var ts;
     ts.isDeclarationWithTypeParameters = isDeclarationWithTypeParameters;
     function isDeclarationWithTypeParameterChildren(node) {
         switch (node.kind) {
-            case 165 /* CallSignature */:
-            case 166 /* ConstructSignature */:
-            case 160 /* MethodSignature */:
-            case 167 /* IndexSignature */:
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
-            case 300 /* JSDocFunctionType */:
-            case 245 /* ClassDeclaration */:
-            case 214 /* ClassExpression */:
-            case 246 /* InterfaceDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-            case 321 /* JSDocTemplateTag */:
-            case 244 /* FunctionDeclaration */:
-            case 161 /* MethodDeclaration */:
-            case 162 /* Constructor */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
+            case 169 /* CallSignature */:
+            case 170 /* ConstructSignature */:
+            case 164 /* MethodSignature */:
+            case 171 /* IndexSignature */:
+            case 174 /* FunctionType */:
+            case 175 /* ConstructorType */:
+            case 308 /* JSDocFunctionType */:
+            case 252 /* ClassDeclaration */:
+            case 221 /* ClassExpression */:
+            case 253 /* InterfaceDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+            case 330 /* JSDocTemplateTag */:
+            case 251 /* FunctionDeclaration */:
+            case 165 /* MethodDeclaration */:
+            case 166 /* Constructor */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
                 return true;
             default:
                 ts.assertType(node);
@@ -13702,8 +14121,8 @@ var ts;
     ts.isDeclarationWithTypeParameterChildren = isDeclarationWithTypeParameterChildren;
     function isAnyImportSyntax(node) {
         switch (node.kind) {
-            case 254 /* ImportDeclaration */:
-            case 253 /* ImportEqualsDeclaration */:
+            case 261 /* ImportDeclaration */:
+            case 260 /* ImportEqualsDeclaration */:
                 return true;
             default:
                 return false;
@@ -13712,15 +14131,15 @@ var ts;
     ts.isAnyImportSyntax = isAnyImportSyntax;
     function isLateVisibilityPaintedStatement(node) {
         switch (node.kind) {
-            case 254 /* ImportDeclaration */:
-            case 253 /* ImportEqualsDeclaration */:
-            case 225 /* VariableStatement */:
-            case 245 /* ClassDeclaration */:
-            case 244 /* FunctionDeclaration */:
-            case 249 /* ModuleDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-            case 246 /* InterfaceDeclaration */:
-            case 248 /* EnumDeclaration */:
+            case 261 /* ImportDeclaration */:
+            case 260 /* ImportEqualsDeclaration */:
+            case 232 /* VariableStatement */:
+            case 252 /* ClassDeclaration */:
+            case 251 /* FunctionDeclaration */:
+            case 256 /* ModuleDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+            case 253 /* InterfaceDeclaration */:
+            case 255 /* EnumDeclaration */:
                 return true;
             default:
                 return false;
@@ -13734,7 +14153,7 @@ var ts;
     // Gets the nearest enclosing block scope container that has the provided node
     // as a descendant, that is not the provided node.
     function getEnclosingBlockScopeContainer(node) {
-        return findAncestor(node.parent, function (current) { return isBlockScope(current, current.parent); });
+        return ts.findAncestor(node.parent, function (current) { return isBlockScope(current, current.parent); });
     }
     ts.getEnclosingBlockScopeContainer = getEnclosingBlockScopeContainer;
     // Return display name of an identifier
@@ -13749,19 +14168,19 @@ var ts;
     }
     ts.getNameFromIndexInfo = getNameFromIndexInfo;
     function isComputedNonLiteralName(name) {
-        return name.kind === 154 /* ComputedPropertyName */ && !isStringOrNumericLiteralLike(name.expression);
+        return name.kind === 158 /* ComputedPropertyName */ && !isStringOrNumericLiteralLike(name.expression);
     }
     ts.isComputedNonLiteralName = isComputedNonLiteralName;
     function getTextOfPropertyName(name) {
         switch (name.kind) {
-            case 75 /* Identifier */:
-            case 76 /* PrivateIdentifier */:
+            case 78 /* Identifier */:
+            case 79 /* PrivateIdentifier */:
                 return name.escapedText;
             case 10 /* StringLiteral */:
             case 8 /* NumericLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
                 return ts.escapeLeadingUnderscores(name.text);
-            case 154 /* ComputedPropertyName */:
+            case 158 /* ComputedPropertyName */:
                 if (isStringOrNumericLiteralLike(name.expression))
                     return ts.escapeLeadingUnderscores(name.expression.text);
                 return ts.Debug.fail("Text of property name cannot be read from non-literal-valued ComputedPropertyNames");
@@ -13772,14 +14191,14 @@ var ts;
     ts.getTextOfPropertyName = getTextOfPropertyName;
     function entityNameToString(name) {
         switch (name.kind) {
-            case 104 /* ThisKeyword */:
+            case 107 /* ThisKeyword */:
                 return "this";
-            case 76 /* PrivateIdentifier */:
-            case 75 /* Identifier */:
+            case 79 /* PrivateIdentifier */:
+            case 78 /* Identifier */:
                 return getFullWidth(name) === 0 ? ts.idText(name) : getTextOfNode(name);
-            case 153 /* QualifiedName */:
+            case 157 /* QualifiedName */:
                 return entityNameToString(name.left) + "." + entityNameToString(name.right);
-            case 194 /* PropertyAccessExpression */:
+            case 201 /* PropertyAccessExpression */:
                 if (ts.isIdentifier(name.name) || ts.isPrivateIdentifier(name.name)) {
                     return entityNameToString(name.expression) + "." + entityNameToString(name.name);
                 }
@@ -13820,6 +14239,18 @@ var ts;
         };
     }
     ts.createDiagnosticForNodeFromMessageChain = createDiagnosticForNodeFromMessageChain;
+    function createDiagnosticForFileFromMessageChain(sourceFile, messageChain, relatedInformation) {
+        return {
+            file: sourceFile,
+            start: 0,
+            length: 0,
+            code: messageChain.code,
+            category: messageChain.category,
+            messageText: messageChain.next ? messageChain : messageChain.messageText,
+            relatedInformation: relatedInformation
+        };
+    }
+    ts.createDiagnosticForFileFromMessageChain = createDiagnosticForFileFromMessageChain;
     function createDiagnosticForRange(sourceFile, range, message) {
         return {
             file: sourceFile,
@@ -13840,7 +14271,7 @@ var ts;
     ts.getSpanOfTokenAtPosition = getSpanOfTokenAtPosition;
     function getErrorSpanForArrowFunction(sourceFile, node) {
         var pos = ts.skipTrivia(sourceFile.text, node.pos);
-        if (node.body && node.body.kind === 223 /* Block */) {
+        if (node.body && node.body.kind === 230 /* Block */) {
             var startLine = ts.getLineAndCharacterOfPosition(sourceFile, node.body.pos).line;
             var endLine = ts.getLineAndCharacterOfPosition(sourceFile, node.body.end).line;
             if (startLine < endLine) {
@@ -13854,7 +14285,7 @@ var ts;
     function getErrorSpanForNode(sourceFile, node) {
         var errorNode = node;
         switch (node.kind) {
-            case 290 /* SourceFile */:
+            case 297 /* SourceFile */:
                 var pos_1 = ts.skipTrivia(sourceFile.text, 0, /*stopAfterLineBreak*/ false);
                 if (pos_1 === sourceFile.text.length) {
                     // file is empty - return span for the beginning of the file
@@ -13863,28 +14294,28 @@ var ts;
                 return getSpanOfTokenAtPosition(sourceFile, pos_1);
             // This list is a work in progress. Add missing node kinds to improve their error
             // spans.
-            case 242 /* VariableDeclaration */:
-            case 191 /* BindingElement */:
-            case 245 /* ClassDeclaration */:
-            case 214 /* ClassExpression */:
-            case 246 /* InterfaceDeclaration */:
-            case 249 /* ModuleDeclaration */:
-            case 248 /* EnumDeclaration */:
-            case 284 /* EnumMember */:
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 161 /* MethodDeclaration */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 247 /* TypeAliasDeclaration */:
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
+            case 249 /* VariableDeclaration */:
+            case 198 /* BindingElement */:
+            case 252 /* ClassDeclaration */:
+            case 221 /* ClassExpression */:
+            case 253 /* InterfaceDeclaration */:
+            case 256 /* ModuleDeclaration */:
+            case 255 /* EnumDeclaration */:
+            case 291 /* EnumMember */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 165 /* MethodDeclaration */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 254 /* TypeAliasDeclaration */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
                 errorNode = node.name;
                 break;
-            case 202 /* ArrowFunction */:
+            case 209 /* ArrowFunction */:
                 return getErrorSpanForArrowFunction(sourceFile, node);
-            case 277 /* CaseClause */:
-            case 278 /* DefaultClause */:
+            case 284 /* CaseClause */:
+            case 285 /* DefaultClause */:
                 var start = ts.skipTrivia(sourceFile.text, node.pos);
                 var end = node.statements.length > 0 ? node.statements[0].pos : node.end;
                 return ts.createTextSpanFromBounds(start, end);
@@ -13936,16 +14367,16 @@ var ts;
     }
     ts.isLet = isLet;
     function isSuperCall(n) {
-        return n.kind === 196 /* CallExpression */ && n.expression.kind === 102 /* SuperKeyword */;
+        return n.kind === 203 /* CallExpression */ && n.expression.kind === 105 /* SuperKeyword */;
     }
     ts.isSuperCall = isSuperCall;
     function isImportCall(n) {
-        return n.kind === 196 /* CallExpression */ && n.expression.kind === 96 /* ImportKeyword */;
+        return n.kind === 203 /* CallExpression */ && n.expression.kind === 99 /* ImportKeyword */;
     }
     ts.isImportCall = isImportCall;
     function isImportMeta(n) {
         return ts.isMetaProperty(n)
-            && n.keywordToken === 96 /* ImportKeyword */
+            && n.keywordToken === 99 /* ImportKeyword */
             && n.name.escapedText === "meta";
     }
     ts.isImportMeta = isImportMeta;
@@ -13954,7 +14385,7 @@ var ts;
     }
     ts.isLiteralImportTypeNode = isLiteralImportTypeNode;
     function isPrologueDirective(node) {
-        return node.kind === 226 /* ExpressionStatement */
+        return node.kind === 233 /* ExpressionStatement */
             && node.expression.kind === 10 /* StringLiteral */;
     }
     ts.isPrologueDirective = isPrologueDirective;
@@ -13982,11 +14413,11 @@ var ts;
     }
     ts.getLeadingCommentRangesOfNode = getLeadingCommentRangesOfNode;
     function getJSDocCommentRanges(node, text) {
-        var commentRanges = (node.kind === 156 /* Parameter */ ||
-            node.kind === 155 /* TypeParameter */ ||
-            node.kind === 201 /* FunctionExpression */ ||
-            node.kind === 202 /* ArrowFunction */ ||
-            node.kind === 200 /* ParenthesizedExpression */) ?
+        var commentRanges = (node.kind === 160 /* Parameter */ ||
+            node.kind === 159 /* TypeParameter */ ||
+            node.kind === 208 /* FunctionExpression */ ||
+            node.kind === 209 /* ArrowFunction */ ||
+            node.kind === 207 /* ParenthesizedExpression */) ?
             ts.concatenate(ts.getTrailingCommentRanges(text, node.pos), ts.getLeadingCommentRanges(text, node.pos)) :
             ts.getLeadingCommentRanges(text, node.pos);
         // True if the comment starts with '/**' but not if it is '/**/'
@@ -14002,48 +14433,48 @@ var ts;
     ts.fullTripleSlashAMDReferencePathRegEx = /^(\/\/\/\s*<amd-dependency\s+path\s*=\s*)('|")(.+?)\2.*?\/>/;
     var defaultLibReferenceRegEx = /^(\/\/\/\s*<reference\s+no-default-lib\s*=\s*)('|")(.+?)\2\s*\/>/;
     function isPartOfTypeNode(node) {
-        if (168 /* FirstTypeNode */ <= node.kind && node.kind <= 188 /* LastTypeNode */) {
+        if (172 /* FirstTypeNode */ <= node.kind && node.kind <= 195 /* LastTypeNode */) {
             return true;
         }
         switch (node.kind) {
-            case 125 /* AnyKeyword */:
-            case 148 /* UnknownKeyword */:
-            case 140 /* NumberKeyword */:
-            case 151 /* BigIntKeyword */:
-            case 143 /* StringKeyword */:
-            case 128 /* BooleanKeyword */:
-            case 144 /* SymbolKeyword */:
-            case 141 /* ObjectKeyword */:
-            case 146 /* UndefinedKeyword */:
-            case 137 /* NeverKeyword */:
+            case 128 /* AnyKeyword */:
+            case 152 /* UnknownKeyword */:
+            case 144 /* NumberKeyword */:
+            case 155 /* BigIntKeyword */:
+            case 147 /* StringKeyword */:
+            case 131 /* BooleanKeyword */:
+            case 148 /* SymbolKeyword */:
+            case 145 /* ObjectKeyword */:
+            case 150 /* UndefinedKeyword */:
+            case 141 /* NeverKeyword */:
                 return true;
-            case 110 /* VoidKeyword */:
-                return node.parent.kind !== 205 /* VoidExpression */;
-            case 216 /* ExpressionWithTypeArguments */:
+            case 113 /* VoidKeyword */:
+                return node.parent.kind !== 212 /* VoidExpression */;
+            case 223 /* ExpressionWithTypeArguments */:
                 return !isExpressionWithTypeArgumentsInClassExtendsClause(node);
-            case 155 /* TypeParameter */:
-                return node.parent.kind === 186 /* MappedType */ || node.parent.kind === 181 /* InferType */;
+            case 159 /* TypeParameter */:
+                return node.parent.kind === 190 /* MappedType */ || node.parent.kind === 185 /* InferType */;
             // Identifiers and qualified names may be type nodes, depending on their context. Climb
             // above them to find the lowest container
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 // If the identifier is the RHS of a qualified name, then it's a type iff its parent is.
-                if (node.parent.kind === 153 /* QualifiedName */ && node.parent.right === node) {
+                if (node.parent.kind === 157 /* QualifiedName */ && node.parent.right === node) {
                     node = node.parent;
                 }
-                else if (node.parent.kind === 194 /* PropertyAccessExpression */ && node.parent.name === node) {
+                else if (node.parent.kind === 201 /* PropertyAccessExpression */ && node.parent.name === node) {
                     node = node.parent;
                 }
                 // At this point, node is either a qualified name or an identifier
-                ts.Debug.assert(node.kind === 75 /* Identifier */ || node.kind === 153 /* QualifiedName */ || node.kind === 194 /* PropertyAccessExpression */, "'node' was expected to be a qualified name, identifier or property access in 'isPartOfTypeNode'.");
+                ts.Debug.assert(node.kind === 78 /* Identifier */ || node.kind === 157 /* QualifiedName */ || node.kind === 201 /* PropertyAccessExpression */, "'node' was expected to be a qualified name, identifier or property access in 'isPartOfTypeNode'.");
             // falls through
-            case 153 /* QualifiedName */:
-            case 194 /* PropertyAccessExpression */:
-            case 104 /* ThisKeyword */: {
+            case 157 /* QualifiedName */:
+            case 201 /* PropertyAccessExpression */:
+            case 107 /* ThisKeyword */: {
                 var parent = node.parent;
-                if (parent.kind === 172 /* TypeQuery */) {
+                if (parent.kind === 176 /* TypeQuery */) {
                     return false;
                 }
-                if (parent.kind === 188 /* ImportType */) {
+                if (parent.kind === 195 /* ImportType */) {
                     return !parent.isTypeOf;
                 }
                 // Do not recursively call isPartOfTypeNode on the parent. In the example:
@@ -14052,40 +14483,40 @@ var ts;
                 //
                 // Calling isPartOfTypeNode would consider the qualified name A.B a type node.
                 // Only C and A.B.C are type nodes.
-                if (168 /* FirstTypeNode */ <= parent.kind && parent.kind <= 188 /* LastTypeNode */) {
+                if (172 /* FirstTypeNode */ <= parent.kind && parent.kind <= 195 /* LastTypeNode */) {
                     return true;
                 }
                 switch (parent.kind) {
-                    case 216 /* ExpressionWithTypeArguments */:
+                    case 223 /* ExpressionWithTypeArguments */:
                         return !isExpressionWithTypeArgumentsInClassExtendsClause(parent);
-                    case 155 /* TypeParameter */:
+                    case 159 /* TypeParameter */:
                         return node === parent.constraint;
-                    case 321 /* JSDocTemplateTag */:
+                    case 330 /* JSDocTemplateTag */:
                         return node === parent.constraint;
-                    case 159 /* PropertyDeclaration */:
-                    case 158 /* PropertySignature */:
-                    case 156 /* Parameter */:
-                    case 242 /* VariableDeclaration */:
+                    case 163 /* PropertyDeclaration */:
+                    case 162 /* PropertySignature */:
+                    case 160 /* Parameter */:
+                    case 249 /* VariableDeclaration */:
                         return node === parent.type;
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
-                    case 202 /* ArrowFunction */:
-                    case 162 /* Constructor */:
-                    case 161 /* MethodDeclaration */:
-                    case 160 /* MethodSignature */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
+                    case 166 /* Constructor */:
+                    case 165 /* MethodDeclaration */:
+                    case 164 /* MethodSignature */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
                         return node === parent.type;
-                    case 165 /* CallSignature */:
-                    case 166 /* ConstructSignature */:
-                    case 167 /* IndexSignature */:
+                    case 169 /* CallSignature */:
+                    case 170 /* ConstructSignature */:
+                    case 171 /* IndexSignature */:
                         return node === parent.type;
-                    case 199 /* TypeAssertionExpression */:
+                    case 206 /* TypeAssertionExpression */:
                         return node === parent.type;
-                    case 196 /* CallExpression */:
-                    case 197 /* NewExpression */:
+                    case 203 /* CallExpression */:
+                    case 204 /* NewExpression */:
                         return ts.contains(parent.typeArguments, node);
-                    case 198 /* TaggedTemplateExpression */:
+                    case 205 /* TaggedTemplateExpression */:
                         // TODO (drosen): TaggedTemplateExpressions may eventually support type arguments.
                         return false;
                 }
@@ -14110,23 +14541,23 @@ var ts;
         return traverse(body);
         function traverse(node) {
             switch (node.kind) {
-                case 235 /* ReturnStatement */:
+                case 242 /* ReturnStatement */:
                     return visitor(node);
-                case 251 /* CaseBlock */:
-                case 223 /* Block */:
-                case 227 /* IfStatement */:
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
-                case 230 /* ForStatement */:
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
-                case 236 /* WithStatement */:
-                case 237 /* SwitchStatement */:
-                case 277 /* CaseClause */:
-                case 278 /* DefaultClause */:
-                case 238 /* LabeledStatement */:
-                case 240 /* TryStatement */:
-                case 280 /* CatchClause */:
+                case 258 /* CaseBlock */:
+                case 230 /* Block */:
+                case 234 /* IfStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
+                case 237 /* ForStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                case 243 /* WithStatement */:
+                case 244 /* SwitchStatement */:
+                case 284 /* CaseClause */:
+                case 285 /* DefaultClause */:
+                case 245 /* LabeledStatement */:
+                case 247 /* TryStatement */:
+                case 287 /* CatchClause */:
                     return ts.forEachChild(node, traverse);
             }
         }
@@ -14136,23 +14567,23 @@ var ts;
         return traverse(body);
         function traverse(node) {
             switch (node.kind) {
-                case 212 /* YieldExpression */:
+                case 219 /* YieldExpression */:
                     visitor(node);
                     var operand = node.expression;
                     if (operand) {
                         traverse(operand);
                     }
                     return;
-                case 248 /* EnumDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 249 /* ModuleDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
                     // These are not allowed inside a generator now, but eventually they may be allowed
                     // as local types. Regardless, skip them to avoid the work.
                     return;
                 default:
                     if (ts.isFunctionLike(node)) {
-                        if (node.name && node.name.kind === 154 /* ComputedPropertyName */) {
+                        if (node.name && node.name.kind === 158 /* ComputedPropertyName */) {
                             // Note that we will not include methods/accessors of a class because they would require
                             // first descending into the class. This is by design.
                             traverse(node.name.expression);
@@ -14175,10 +14606,10 @@ var ts;
      * @param node The type node.
      */
     function getRestParameterElementType(node) {
-        if (node && node.kind === 174 /* ArrayType */) {
+        if (node && node.kind === 178 /* ArrayType */) {
             return node.elementType;
         }
-        else if (node && node.kind === 169 /* TypeReference */) {
+        else if (node && node.kind === 173 /* TypeReference */) {
             return ts.singleOrUndefined(node.typeArguments);
         }
         else {
@@ -14188,12 +14619,12 @@ var ts;
     ts.getRestParameterElementType = getRestParameterElementType;
     function getMembersOfDeclaration(node) {
         switch (node.kind) {
-            case 246 /* InterfaceDeclaration */:
-            case 245 /* ClassDeclaration */:
-            case 214 /* ClassExpression */:
-            case 173 /* TypeLiteral */:
+            case 253 /* InterfaceDeclaration */:
+            case 252 /* ClassDeclaration */:
+            case 221 /* ClassExpression */:
+            case 177 /* TypeLiteral */:
                 return node.members;
-            case 193 /* ObjectLiteralExpression */:
+            case 200 /* ObjectLiteralExpression */:
                 return node.properties;
         }
     }
@@ -14201,14 +14632,14 @@ var ts;
     function isVariableLike(node) {
         if (node) {
             switch (node.kind) {
-                case 191 /* BindingElement */:
-                case 284 /* EnumMember */:
-                case 156 /* Parameter */:
-                case 281 /* PropertyAssignment */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                case 282 /* ShorthandPropertyAssignment */:
-                case 242 /* VariableDeclaration */:
+                case 198 /* BindingElement */:
+                case 291 /* EnumMember */:
+                case 160 /* Parameter */:
+                case 288 /* PropertyAssignment */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                case 289 /* ShorthandPropertyAssignment */:
+                case 249 /* VariableDeclaration */:
                     return true;
             }
         }
@@ -14220,25 +14651,25 @@ var ts;
     }
     ts.isVariableLikeOrAccessor = isVariableLikeOrAccessor;
     function isVariableDeclarationInVariableStatement(node) {
-        return node.parent.kind === 243 /* VariableDeclarationList */
-            && node.parent.parent.kind === 225 /* VariableStatement */;
+        return node.parent.kind === 250 /* VariableDeclarationList */
+            && node.parent.parent.kind === 232 /* VariableStatement */;
     }
     ts.isVariableDeclarationInVariableStatement = isVariableDeclarationInVariableStatement;
     function isValidESSymbolDeclaration(node) {
         return ts.isVariableDeclaration(node) ? isVarConst(node) && ts.isIdentifier(node.name) && isVariableDeclarationInVariableStatement(node) :
-            ts.isPropertyDeclaration(node) ? hasReadonlyModifier(node) && hasStaticModifier(node) :
-                ts.isPropertySignature(node) && hasReadonlyModifier(node);
+            ts.isPropertyDeclaration(node) ? hasEffectiveReadonlyModifier(node) && hasStaticModifier(node) :
+                ts.isPropertySignature(node) && hasEffectiveReadonlyModifier(node);
     }
     ts.isValidESSymbolDeclaration = isValidESSymbolDeclaration;
     function introducesArgumentsExoticObject(node) {
         switch (node.kind) {
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
-            case 162 /* Constructor */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
+            case 166 /* Constructor */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
                 return true;
         }
         return false;
@@ -14249,7 +14680,7 @@ var ts;
             if (beforeUnwrapLabelCallback) {
                 beforeUnwrapLabelCallback(node);
             }
-            if (node.statement.kind !== 238 /* LabeledStatement */) {
+            if (node.statement.kind !== 245 /* LabeledStatement */) {
                 return node.statement;
             }
             node = node.statement;
@@ -14257,17 +14688,17 @@ var ts;
     }
     ts.unwrapInnermostStatementOfLabel = unwrapInnermostStatementOfLabel;
     function isFunctionBlock(node) {
-        return node && node.kind === 223 /* Block */ && ts.isFunctionLike(node.parent);
+        return node && node.kind === 230 /* Block */ && ts.isFunctionLike(node.parent);
     }
     ts.isFunctionBlock = isFunctionBlock;
     function isObjectLiteralMethod(node) {
-        return node && node.kind === 161 /* MethodDeclaration */ && node.parent.kind === 193 /* ObjectLiteralExpression */;
+        return node && node.kind === 165 /* MethodDeclaration */ && node.parent.kind === 200 /* ObjectLiteralExpression */;
     }
     ts.isObjectLiteralMethod = isObjectLiteralMethod;
     function isObjectLiteralOrClassExpressionMethod(node) {
-        return node.kind === 161 /* MethodDeclaration */ &&
-            (node.parent.kind === 193 /* ObjectLiteralExpression */ ||
-                node.parent.kind === 214 /* ClassExpression */);
+        return node.kind === 165 /* MethodDeclaration */ &&
+            (node.parent.kind === 200 /* ObjectLiteralExpression */ ||
+                node.parent.kind === 221 /* ClassExpression */);
     }
     ts.isObjectLiteralOrClassExpressionMethod = isObjectLiteralOrClassExpressionMethod;
     function isIdentifierTypePredicate(predicate) {
@@ -14280,7 +14711,7 @@ var ts;
     ts.isThisTypePredicate = isThisTypePredicate;
     function getPropertyAssignment(objectLiteral, key, key2) {
         return objectLiteral.properties.filter(function (property) {
-            if (property.kind === 281 /* PropertyAssignment */) {
+            if (property.kind === 288 /* PropertyAssignment */) {
                 var propName = getTextOfPropertyName(property.name);
                 return key === propName || (!!key2 && key2 === propName);
             }
@@ -14309,26 +14740,26 @@ var ts;
     }
     ts.getTsConfigPropArray = getTsConfigPropArray;
     function getContainingFunction(node) {
-        return findAncestor(node.parent, ts.isFunctionLike);
+        return ts.findAncestor(node.parent, ts.isFunctionLike);
     }
     ts.getContainingFunction = getContainingFunction;
     function getContainingFunctionDeclaration(node) {
-        return findAncestor(node.parent, ts.isFunctionLikeDeclaration);
+        return ts.findAncestor(node.parent, ts.isFunctionLikeDeclaration);
     }
     ts.getContainingFunctionDeclaration = getContainingFunctionDeclaration;
     function getContainingClass(node) {
-        return findAncestor(node.parent, ts.isClassLike);
+        return ts.findAncestor(node.parent, ts.isClassLike);
     }
     ts.getContainingClass = getContainingClass;
     function getThisContainer(node, includeArrowFunctions) {
-        ts.Debug.assert(node.kind !== 290 /* SourceFile */);
+        ts.Debug.assert(node.kind !== 297 /* SourceFile */);
         while (true) {
             node = node.parent;
             if (!node) {
                 return ts.Debug.fail(); // If we never pass in a SourceFile, this should be unreachable, since we'll stop when we reach that.
             }
             switch (node.kind) {
-                case 154 /* ComputedPropertyName */:
+                case 158 /* ComputedPropertyName */:
                     // If the grandparent node is an object literal (as opposed to a class),
                     // then the computed property is not a 'this' container.
                     // A computed property name in a class needs to be a this container
@@ -14343,9 +14774,9 @@ var ts;
                     // the *body* of the container.
                     node = node.parent;
                     break;
-                case 157 /* Decorator */:
+                case 161 /* Decorator */:
                     // Decorators are always applied outside of the body of a class or method.
-                    if (node.parent.kind === 156 /* Parameter */ && ts.isClassElement(node.parent.parent)) {
+                    if (node.parent.kind === 160 /* Parameter */ && ts.isClassElement(node.parent.parent)) {
                         // If the decorator's parent is a Parameter, we resolve the this container from
                         // the grandparent class declaration.
                         node = node.parent.parent;
@@ -14356,38 +14787,47 @@ var ts;
                         node = node.parent;
                     }
                     break;
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     if (!includeArrowFunctions) {
                         continue;
                     }
                 // falls through
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 249 /* ModuleDeclaration */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 162 /* Constructor */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 167 /* IndexSignature */:
-                case 248 /* EnumDeclaration */:
-                case 290 /* SourceFile */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 256 /* ModuleDeclaration */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 166 /* Constructor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 171 /* IndexSignature */:
+                case 255 /* EnumDeclaration */:
+                case 297 /* SourceFile */:
                     return node;
             }
         }
     }
     ts.getThisContainer = getThisContainer;
+    function isInTopLevelContext(node) {
+        // The name of a class or function declaration is a BindingIdentifier in its surrounding scope.
+        if (ts.isIdentifier(node) && (ts.isClassDeclaration(node.parent) || ts.isFunctionDeclaration(node.parent)) && node.parent.name === node) {
+            node = node.parent;
+        }
+        var container = getThisContainer(node, /*includeArrowFunctions*/ true);
+        return ts.isSourceFile(container);
+    }
+    ts.isInTopLevelContext = isInTopLevelContext;
     function getNewTargetContainer(node) {
         var container = getThisContainer(node, /*includeArrowFunctions*/ false);
         if (container) {
             switch (container.kind) {
-                case 162 /* Constructor */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
+                case 166 /* Constructor */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
                     return container;
             }
         }
@@ -14409,27 +14849,27 @@ var ts;
                 return node;
             }
             switch (node.kind) {
-                case 154 /* ComputedPropertyName */:
+                case 158 /* ComputedPropertyName */:
                     node = node.parent;
                     break;
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
                     if (!stopOnFunctions) {
                         continue;
                     }
                 // falls through
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 162 /* Constructor */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 166 /* Constructor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return node;
-                case 157 /* Decorator */:
+                case 161 /* Decorator */:
                     // Decorators are always applied outside of the body of a class or method.
-                    if (node.parent.kind === 156 /* Parameter */ && ts.isClassElement(node.parent.parent)) {
+                    if (node.parent.kind === 160 /* Parameter */ && ts.isClassElement(node.parent.parent)) {
                         // If the decorator's parent is a Parameter, we resolve the this container from
                         // the grandparent class declaration.
                         node = node.parent.parent;
@@ -14445,21 +14885,21 @@ var ts;
     }
     ts.getSuperContainer = getSuperContainer;
     function getImmediatelyInvokedFunctionExpression(func) {
-        if (func.kind === 201 /* FunctionExpression */ || func.kind === 202 /* ArrowFunction */) {
+        if (func.kind === 208 /* FunctionExpression */ || func.kind === 209 /* ArrowFunction */) {
             var prev = func;
             var parent = func.parent;
-            while (parent.kind === 200 /* ParenthesizedExpression */) {
+            while (parent.kind === 207 /* ParenthesizedExpression */) {
                 prev = parent;
                 parent = parent.parent;
             }
-            if (parent.kind === 196 /* CallExpression */ && parent.expression === prev) {
+            if (parent.kind === 203 /* CallExpression */ && parent.expression === prev) {
                 return parent;
             }
         }
     }
     ts.getImmediatelyInvokedFunctionExpression = getImmediatelyInvokedFunctionExpression;
     function isSuperOrSuperProperty(node) {
-        return node.kind === 102 /* SuperKeyword */
+        return node.kind === 105 /* SuperKeyword */
             || isSuperProperty(node);
     }
     ts.isSuperOrSuperProperty = isSuperOrSuperProperty;
@@ -14468,8 +14908,8 @@ var ts;
      */
     function isSuperProperty(node) {
         var kind = node.kind;
-        return (kind === 194 /* PropertyAccessExpression */ || kind === 195 /* ElementAccessExpression */)
-            && node.expression.kind === 102 /* SuperKeyword */;
+        return (kind === 201 /* PropertyAccessExpression */ || kind === 202 /* ElementAccessExpression */)
+            && node.expression.kind === 105 /* SuperKeyword */;
     }
     ts.isSuperProperty = isSuperProperty;
     /**
@@ -14477,20 +14917,26 @@ var ts;
      */
     function isThisProperty(node) {
         var kind = node.kind;
-        return (kind === 194 /* PropertyAccessExpression */ || kind === 195 /* ElementAccessExpression */)
-            && node.expression.kind === 104 /* ThisKeyword */;
+        return (kind === 201 /* PropertyAccessExpression */ || kind === 202 /* ElementAccessExpression */)
+            && node.expression.kind === 107 /* ThisKeyword */;
     }
     ts.isThisProperty = isThisProperty;
+    function isThisInitializedDeclaration(node) {
+        var _a;
+        return !!node && ts.isVariableDeclaration(node) && ((_a = node.initializer) === null || _a === void 0 ? void 0 : _a.kind) === 107 /* ThisKeyword */;
+    }
+    ts.isThisInitializedDeclaration = isThisInitializedDeclaration;
     function getEntityNameFromTypeNode(node) {
         switch (node.kind) {
-            case 169 /* TypeReference */:
+            case 173 /* TypeReference */:
                 return node.typeName;
-            case 216 /* ExpressionWithTypeArguments */:
+            case 223 /* ExpressionWithTypeArguments */:
                 return isEntityNameExpression(node.expression)
                     ? node.expression
                     : undefined;
-            case 75 /* Identifier */:
-            case 153 /* QualifiedName */:
+            // TODO(rbuckton): These aren't valid TypeNodes, but we treat them as such because of `isPartOfTypeNode`, which returns `true` for things that aren't `TypeNode`s.
+            case 78 /* Identifier */:
+            case 157 /* QualifiedName */:
                 return node;
         }
         return undefined;
@@ -14498,10 +14944,10 @@ var ts;
     ts.getEntityNameFromTypeNode = getEntityNameFromTypeNode;
     function getInvokedExpression(node) {
         switch (node.kind) {
-            case 198 /* TaggedTemplateExpression */:
+            case 205 /* TaggedTemplateExpression */:
                 return node.tag;
-            case 268 /* JsxOpeningElement */:
-            case 267 /* JsxSelfClosingElement */:
+            case 275 /* JsxOpeningElement */:
+            case 274 /* JsxSelfClosingElement */:
                 return node.tagName;
             default:
                 return node.expression;
@@ -14514,25 +14960,25 @@ var ts;
             return false;
         }
         switch (node.kind) {
-            case 245 /* ClassDeclaration */:
+            case 252 /* ClassDeclaration */:
                 // classes are valid targets
                 return true;
-            case 159 /* PropertyDeclaration */:
+            case 163 /* PropertyDeclaration */:
                 // property declarations are valid if their parent is a class declaration.
-                return parent.kind === 245 /* ClassDeclaration */;
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 161 /* MethodDeclaration */:
+                return parent.kind === 252 /* ClassDeclaration */;
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 165 /* MethodDeclaration */:
                 // if this method has a body and its parent is a class declaration, this is a valid target.
                 return node.body !== undefined
-                    && parent.kind === 245 /* ClassDeclaration */;
-            case 156 /* Parameter */:
+                    && parent.kind === 252 /* ClassDeclaration */;
+            case 160 /* Parameter */:
                 // if the parameter's parent has a body and its grandparent is a class declaration, this is a valid target;
                 return parent.body !== undefined
-                    && (parent.kind === 162 /* Constructor */
-                        || parent.kind === 161 /* MethodDeclaration */
-                        || parent.kind === 164 /* SetAccessor */)
-                    && grandparent.kind === 245 /* ClassDeclaration */;
+                    && (parent.kind === 166 /* Constructor */
+                        || parent.kind === 165 /* MethodDeclaration */
+                        || parent.kind === 168 /* SetAccessor */)
+                    && grandparent.kind === 252 /* ClassDeclaration */;
         }
         return false;
     }
@@ -14548,10 +14994,10 @@ var ts;
     ts.nodeOrChildIsDecorated = nodeOrChildIsDecorated;
     function childIsDecorated(node, parent) {
         switch (node.kind) {
-            case 245 /* ClassDeclaration */:
+            case 252 /* ClassDeclaration */:
                 return ts.some(node.members, function (m) { return nodeOrChildIsDecorated(m, node, parent); }); // TODO: GH#18217
-            case 161 /* MethodDeclaration */:
-            case 164 /* SetAccessor */:
+            case 165 /* MethodDeclaration */:
+            case 168 /* SetAccessor */:
                 return ts.some(node.parameters, function (p) { return nodeIsDecorated(p, node, parent); }); // TODO: GH#18217
             default:
                 return false;
@@ -14560,9 +15006,9 @@ var ts;
     ts.childIsDecorated = childIsDecorated;
     function isJSXTagName(node) {
         var parent = node.parent;
-        if (parent.kind === 268 /* JsxOpeningElement */ ||
-            parent.kind === 267 /* JsxSelfClosingElement */ ||
-            parent.kind === 269 /* JsxClosingElement */) {
+        if (parent.kind === 275 /* JsxOpeningElement */ ||
+            parent.kind === 274 /* JsxSelfClosingElement */ ||
+            parent.kind === 276 /* JsxClosingElement */) {
             return parent.tagName === node;
         }
         return false;
@@ -14570,49 +15016,49 @@ var ts;
     ts.isJSXTagName = isJSXTagName;
     function isExpressionNode(node) {
         switch (node.kind) {
-            case 102 /* SuperKeyword */:
-            case 100 /* NullKeyword */:
-            case 106 /* TrueKeyword */:
-            case 91 /* FalseKeyword */:
+            case 105 /* SuperKeyword */:
+            case 103 /* NullKeyword */:
+            case 109 /* TrueKeyword */:
+            case 94 /* FalseKeyword */:
             case 13 /* RegularExpressionLiteral */:
-            case 192 /* ArrayLiteralExpression */:
-            case 193 /* ObjectLiteralExpression */:
-            case 194 /* PropertyAccessExpression */:
-            case 195 /* ElementAccessExpression */:
-            case 196 /* CallExpression */:
-            case 197 /* NewExpression */:
-            case 198 /* TaggedTemplateExpression */:
-            case 217 /* AsExpression */:
-            case 199 /* TypeAssertionExpression */:
-            case 218 /* NonNullExpression */:
-            case 200 /* ParenthesizedExpression */:
-            case 201 /* FunctionExpression */:
-            case 214 /* ClassExpression */:
-            case 202 /* ArrowFunction */:
-            case 205 /* VoidExpression */:
-            case 203 /* DeleteExpression */:
-            case 204 /* TypeOfExpression */:
-            case 207 /* PrefixUnaryExpression */:
-            case 208 /* PostfixUnaryExpression */:
-            case 209 /* BinaryExpression */:
-            case 210 /* ConditionalExpression */:
-            case 213 /* SpreadElement */:
-            case 211 /* TemplateExpression */:
-            case 215 /* OmittedExpression */:
-            case 266 /* JsxElement */:
-            case 267 /* JsxSelfClosingElement */:
-            case 270 /* JsxFragment */:
-            case 212 /* YieldExpression */:
-            case 206 /* AwaitExpression */:
-            case 219 /* MetaProperty */:
+            case 199 /* ArrayLiteralExpression */:
+            case 200 /* ObjectLiteralExpression */:
+            case 201 /* PropertyAccessExpression */:
+            case 202 /* ElementAccessExpression */:
+            case 203 /* CallExpression */:
+            case 204 /* NewExpression */:
+            case 205 /* TaggedTemplateExpression */:
+            case 224 /* AsExpression */:
+            case 206 /* TypeAssertionExpression */:
+            case 225 /* NonNullExpression */:
+            case 207 /* ParenthesizedExpression */:
+            case 208 /* FunctionExpression */:
+            case 221 /* ClassExpression */:
+            case 209 /* ArrowFunction */:
+            case 212 /* VoidExpression */:
+            case 210 /* DeleteExpression */:
+            case 211 /* TypeOfExpression */:
+            case 214 /* PrefixUnaryExpression */:
+            case 215 /* PostfixUnaryExpression */:
+            case 216 /* BinaryExpression */:
+            case 217 /* ConditionalExpression */:
+            case 220 /* SpreadElement */:
+            case 218 /* TemplateExpression */:
+            case 222 /* OmittedExpression */:
+            case 273 /* JsxElement */:
+            case 274 /* JsxSelfClosingElement */:
+            case 277 /* JsxFragment */:
+            case 219 /* YieldExpression */:
+            case 213 /* AwaitExpression */:
+            case 226 /* MetaProperty */:
                 return true;
-            case 153 /* QualifiedName */:
-                while (node.parent.kind === 153 /* QualifiedName */) {
+            case 157 /* QualifiedName */:
+                while (node.parent.kind === 157 /* QualifiedName */) {
                     node = node.parent;
                 }
-                return node.parent.kind === 172 /* TypeQuery */ || isJSXTagName(node);
-            case 75 /* Identifier */:
-                if (node.parent.kind === 172 /* TypeQuery */ || isJSXTagName(node)) {
+                return node.parent.kind === 176 /* TypeQuery */ || isJSXTagName(node);
+            case 78 /* Identifier */:
+                if (node.parent.kind === 176 /* TypeQuery */ || isJSXTagName(node)) {
                     return true;
                 }
             // falls through
@@ -14620,7 +15066,7 @@ var ts;
             case 9 /* BigIntLiteral */:
             case 10 /* StringLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
-            case 104 /* ThisKeyword */:
+            case 107 /* ThisKeyword */:
                 return isInExpressionContext(node);
             default:
                 return false;
@@ -14630,49 +15076,49 @@ var ts;
     function isInExpressionContext(node) {
         var parent = node.parent;
         switch (parent.kind) {
-            case 242 /* VariableDeclaration */:
-            case 156 /* Parameter */:
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-            case 284 /* EnumMember */:
-            case 281 /* PropertyAssignment */:
-            case 191 /* BindingElement */:
+            case 249 /* VariableDeclaration */:
+            case 160 /* Parameter */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+            case 291 /* EnumMember */:
+            case 288 /* PropertyAssignment */:
+            case 198 /* BindingElement */:
                 return parent.initializer === node;
-            case 226 /* ExpressionStatement */:
-            case 227 /* IfStatement */:
-            case 228 /* DoStatement */:
-            case 229 /* WhileStatement */:
-            case 235 /* ReturnStatement */:
-            case 236 /* WithStatement */:
-            case 237 /* SwitchStatement */:
-            case 277 /* CaseClause */:
-            case 239 /* ThrowStatement */:
+            case 233 /* ExpressionStatement */:
+            case 234 /* IfStatement */:
+            case 235 /* DoStatement */:
+            case 236 /* WhileStatement */:
+            case 242 /* ReturnStatement */:
+            case 243 /* WithStatement */:
+            case 244 /* SwitchStatement */:
+            case 284 /* CaseClause */:
+            case 246 /* ThrowStatement */:
                 return parent.expression === node;
-            case 230 /* ForStatement */:
+            case 237 /* ForStatement */:
                 var forStatement = parent;
-                return (forStatement.initializer === node && forStatement.initializer.kind !== 243 /* VariableDeclarationList */) ||
+                return (forStatement.initializer === node && forStatement.initializer.kind !== 250 /* VariableDeclarationList */) ||
                     forStatement.condition === node ||
                     forStatement.incrementor === node;
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
+            case 238 /* ForInStatement */:
+            case 239 /* ForOfStatement */:
                 var forInStatement = parent;
-                return (forInStatement.initializer === node && forInStatement.initializer.kind !== 243 /* VariableDeclarationList */) ||
+                return (forInStatement.initializer === node && forInStatement.initializer.kind !== 250 /* VariableDeclarationList */) ||
                     forInStatement.expression === node;
-            case 199 /* TypeAssertionExpression */:
-            case 217 /* AsExpression */:
+            case 206 /* TypeAssertionExpression */:
+            case 224 /* AsExpression */:
                 return node === parent.expression;
-            case 221 /* TemplateSpan */:
+            case 228 /* TemplateSpan */:
                 return node === parent.expression;
-            case 154 /* ComputedPropertyName */:
+            case 158 /* ComputedPropertyName */:
                 return node === parent.expression;
-            case 157 /* Decorator */:
-            case 276 /* JsxExpression */:
-            case 275 /* JsxSpreadAttribute */:
-            case 283 /* SpreadAssignment */:
+            case 161 /* Decorator */:
+            case 283 /* JsxExpression */:
+            case 282 /* JsxSpreadAttribute */:
+            case 290 /* SpreadAssignment */:
                 return true;
-            case 216 /* ExpressionWithTypeArguments */:
+            case 223 /* ExpressionWithTypeArguments */:
                 return parent.expression === node && isExpressionWithTypeArgumentsInClassExtendsClause(parent);
-            case 282 /* ShorthandPropertyAssignment */:
+            case 289 /* ShorthandPropertyAssignment */:
                 return parent.objectAssignmentInitializer === node;
             default:
                 return isExpressionNode(parent);
@@ -14680,14 +15126,14 @@ var ts;
     }
     ts.isInExpressionContext = isInExpressionContext;
     function isPartOfTypeQuery(node) {
-        while (node.kind === 153 /* QualifiedName */ || node.kind === 75 /* Identifier */) {
+        while (node.kind === 157 /* QualifiedName */ || node.kind === 78 /* Identifier */) {
             node = node.parent;
         }
-        return node.kind === 172 /* TypeQuery */;
+        return node.kind === 176 /* TypeQuery */;
     }
     ts.isPartOfTypeQuery = isPartOfTypeQuery;
     function isExternalModuleImportEqualsDeclaration(node) {
-        return node.kind === 253 /* ImportEqualsDeclaration */ && node.moduleReference.kind === 265 /* ExternalModuleReference */;
+        return node.kind === 260 /* ImportEqualsDeclaration */ && node.moduleReference.kind === 272 /* ExternalModuleReference */;
     }
     ts.isExternalModuleImportEqualsDeclaration = isExternalModuleImportEqualsDeclaration;
     function getExternalModuleImportEqualsDeclarationExpression(node) {
@@ -14695,8 +15141,13 @@ var ts;
         return node.moduleReference.expression;
     }
     ts.getExternalModuleImportEqualsDeclarationExpression = getExternalModuleImportEqualsDeclarationExpression;
+    function getExternalModuleRequireArgument(node) {
+        return isRequireVariableDeclaration(node, /*requireStringLiteralLikeArgument*/ true)
+            && getLeftmostAccessExpression(node.initializer).arguments[0];
+    }
+    ts.getExternalModuleRequireArgument = getExternalModuleRequireArgument;
     function isInternalModuleImportEqualsDeclaration(node) {
-        return node.kind === 253 /* ImportEqualsDeclaration */ && node.moduleReference.kind !== 265 /* ExternalModuleReference */;
+        return node.kind === 260 /* ImportEqualsDeclaration */ && node.moduleReference.kind !== 272 /* ExternalModuleReference */;
     }
     ts.isInternalModuleImportEqualsDeclaration = isInternalModuleImportEqualsDeclaration;
     function isSourceFileJS(file) {
@@ -14728,15 +15179,15 @@ var ts;
             ts.isIdentifier(node.typeName) &&
             node.typeName.escapedText === "Object" &&
             node.typeArguments && node.typeArguments.length === 2 &&
-            (node.typeArguments[0].kind === 143 /* StringKeyword */ || node.typeArguments[0].kind === 140 /* NumberKeyword */);
+            (node.typeArguments[0].kind === 147 /* StringKeyword */ || node.typeArguments[0].kind === 144 /* NumberKeyword */);
     }
     ts.isJSDocIndexSignature = isJSDocIndexSignature;
     function isRequireCall(callExpression, requireStringLiteralLikeArgument) {
-        if (callExpression.kind !== 196 /* CallExpression */) {
+        if (callExpression.kind !== 203 /* CallExpression */) {
             return false;
         }
         var _a = callExpression, expression = _a.expression, args = _a.arguments;
-        if (expression.kind !== 75 /* Identifier */ || expression.escapedText !== "require") {
+        if (expression.kind !== 78 /* Identifier */ || expression.escapedText !== "require") {
             return false;
         }
         if (args.length !== 1) {
@@ -14747,14 +15198,19 @@ var ts;
     }
     ts.isRequireCall = isRequireCall;
     function isRequireVariableDeclaration(node, requireStringLiteralLikeArgument) {
-        return ts.isVariableDeclaration(node) && !!node.initializer && isRequireCall(node.initializer, requireStringLiteralLikeArgument);
+        if (node.kind === 198 /* BindingElement */) {
+            node = node.parent.parent;
+        }
+        return ts.isVariableDeclaration(node) && !!node.initializer && isRequireCall(getLeftmostAccessExpression(node.initializer), requireStringLiteralLikeArgument);
     }
     ts.isRequireVariableDeclaration = isRequireVariableDeclaration;
-    function isRequireVariableDeclarationStatement(node, requireStringLiteralLikeArgument) {
+    function isRequireVariableStatement(node, requireStringLiteralLikeArgument) {
         if (requireStringLiteralLikeArgument === void 0) { requireStringLiteralLikeArgument = true; }
-        return ts.isVariableStatement(node) && ts.every(node.declarationList.declarations, function (decl) { return isRequireVariableDeclaration(decl, requireStringLiteralLikeArgument); });
+        return ts.isVariableStatement(node)
+            && node.declarationList.declarations.length > 0
+            && ts.every(node.declarationList.declarations, function (decl) { return isRequireVariableDeclaration(decl, requireStringLiteralLikeArgument); });
     }
-    ts.isRequireVariableDeclarationStatement = isRequireVariableDeclarationStatement;
+    ts.isRequireVariableStatement = isRequireVariableStatement;
     function isSingleOrDoubleQuote(charCode) {
         return charCode === 39 /* singleQuote */ || charCode === 34 /* doubleQuote */;
     }
@@ -14763,46 +15219,6 @@ var ts;
         return getSourceTextOfNodeFromSourceFile(sourceFile, str).charCodeAt(0) === 34 /* doubleQuote */;
     }
     ts.isStringDoubleQuoted = isStringDoubleQuoted;
-    function getDeclarationOfExpando(node) {
-        if (!node.parent) {
-            return undefined;
-        }
-        var name;
-        var decl;
-        if (ts.isVariableDeclaration(node.parent) && node.parent.initializer === node) {
-            if (!isInJSFile(node) && !isVarConst(node.parent)) {
-                return undefined;
-            }
-            name = node.parent.name;
-            decl = node.parent;
-        }
-        else if (ts.isBinaryExpression(node.parent)) {
-            var parentNode = node.parent;
-            var parentNodeOperator = node.parent.operatorToken.kind;
-            if (parentNodeOperator === 62 /* EqualsToken */ && parentNode.right === node) {
-                name = parentNode.left;
-                decl = name;
-            }
-            else if (parentNodeOperator === 56 /* BarBarToken */ || parentNodeOperator === 60 /* QuestionQuestionToken */) {
-                if (ts.isVariableDeclaration(parentNode.parent) && parentNode.parent.initializer === parentNode) {
-                    name = parentNode.parent.name;
-                    decl = parentNode.parent;
-                }
-                else if (ts.isBinaryExpression(parentNode.parent) && parentNode.parent.operatorToken.kind === 62 /* EqualsToken */ && parentNode.parent.right === parentNode) {
-                    name = parentNode.parent.left;
-                    decl = name;
-                }
-                if (!name || !isBindableStaticNameExpression(name) || !isSameEntityName(name, parentNode.left)) {
-                    return undefined;
-                }
-            }
-        }
-        if (!name || !getExpandoInitializer(node, isPrototypeAccess(name))) {
-            return undefined;
-        }
-        return decl;
-    }
-    ts.getDeclarationOfExpando = getDeclarationOfExpando;
     function isAssignmentDeclaration(decl) {
         return ts.isBinaryExpression(decl) || isAccessExpression(decl) || ts.isIdentifier(decl) || ts.isCallExpression(decl);
     }
@@ -14835,7 +15251,7 @@ var ts;
     }
     /**
      * Get the assignment 'initializer' -- the righthand side-- when the initializer is container-like (See getExpandoInitializer).
-     * We treat the right hand side of assignments with container-like initalizers as declarations.
+     * We treat the right hand side of assignments with container-like initializers as declarations.
      */
     function getAssignedExpandoInitializer(node) {
         if (node && node.parent && ts.isBinaryExpression(node.parent) && node.parent.operatorToken.kind === 62 /* EqualsToken */) {
@@ -14864,11 +15280,11 @@ var ts;
     function getExpandoInitializer(initializer, isPrototypeAssignment) {
         if (ts.isCallExpression(initializer)) {
             var e = skipParentheses(initializer.expression);
-            return e.kind === 201 /* FunctionExpression */ || e.kind === 202 /* ArrowFunction */ ? initializer : undefined;
+            return e.kind === 208 /* FunctionExpression */ || e.kind === 209 /* ArrowFunction */ ? initializer : undefined;
         }
-        if (initializer.kind === 201 /* FunctionExpression */ ||
-            initializer.kind === 214 /* ClassExpression */ ||
-            initializer.kind === 202 /* ArrowFunction */) {
+        if (initializer.kind === 208 /* FunctionExpression */ ||
+            initializer.kind === 221 /* ClassExpression */ ||
+            initializer.kind === 209 /* ArrowFunction */) {
             return initializer;
         }
         if (ts.isObjectLiteralExpression(initializer) && (initializer.properties.length === 0 || isPrototypeAssignment)) {
@@ -14923,10 +15339,10 @@ var ts;
      */
     function isSameEntityName(name, initializer) {
         if (isPropertyNameLiteral(name) && isPropertyNameLiteral(initializer)) {
-            return getTextOfIdentifierOrLiteral(name) === getTextOfIdentifierOrLiteral(name);
+            return getTextOfIdentifierOrLiteral(name) === getTextOfIdentifierOrLiteral(initializer);
         }
         if (ts.isIdentifier(name) && isLiteralLikeAccess(initializer) &&
-            (initializer.expression.kind === 104 /* ThisKeyword */ ||
+            (initializer.expression.kind === 107 /* ThisKeyword */ ||
                 ts.isIdentifier(initializer.expression) &&
                     (initializer.expression.escapedText === "window" ||
                         initializer.expression.escapedText === "self" ||
@@ -14943,6 +15359,7 @@ var ts;
         }
         return false;
     }
+    ts.isSameEntityName = isSameEntityName;
     function getRightMostAssignedExpression(node) {
         while (isAssignmentExpression(node, /*excludeCompoundAssignments*/ true)) {
             node = node.right;
@@ -14994,14 +15411,14 @@ var ts;
     ts.isLiteralLikeElementAccess = isLiteralLikeElementAccess;
     /** Any series of property and element accesses. */
     function isBindableStaticAccessExpression(node, excludeThisKeyword) {
-        return ts.isPropertyAccessExpression(node) && (!excludeThisKeyword && node.expression.kind === 104 /* ThisKeyword */ || ts.isIdentifier(node.name) && isBindableStaticNameExpression(node.expression, /*excludeThisKeyword*/ true))
+        return ts.isPropertyAccessExpression(node) && (!excludeThisKeyword && node.expression.kind === 107 /* ThisKeyword */ || ts.isIdentifier(node.name) && isBindableStaticNameExpression(node.expression, /*excludeThisKeyword*/ true))
             || isBindableStaticElementAccessExpression(node, excludeThisKeyword);
     }
     ts.isBindableStaticAccessExpression = isBindableStaticAccessExpression;
     /** Any series of property and element accesses, ending in a literal element access */
     function isBindableStaticElementAccessExpression(node, excludeThisKeyword) {
         return isLiteralLikeElementAccess(node)
-            && ((!excludeThisKeyword && node.expression.kind === 104 /* ThisKeyword */) ||
+            && ((!excludeThisKeyword && node.expression.kind === 107 /* ThisKeyword */) ||
                 isEntityNameExpression(node.expression) ||
                 isBindableStaticAccessExpression(node.expression, /*excludeThisKeyword*/ true));
     }
@@ -15031,7 +15448,7 @@ var ts;
             }
             return 7 /* ObjectDefinePropertyValue */;
         }
-        if (expr.operatorToken.kind !== 62 /* EqualsToken */ || !isAccessExpression(expr.left)) {
+        if (expr.operatorToken.kind !== 62 /* EqualsToken */ || !isAccessExpression(expr.left) || isVoidZero(getRightMostAssignedExpression(expr))) {
             return 0 /* None */;
         }
         if (isBindableStaticNameExpression(expr.left.expression, /*excludeThisKeyword*/ true) && getElementOrPropertyAccessName(expr.left) === "prototype" && ts.isObjectLiteralExpression(getInitializerOfBinaryExpression(expr))) {
@@ -15040,6 +15457,9 @@ var ts;
         }
         return getAssignmentDeclarationPropertyAccessKind(expr.left);
     }
+    function isVoidZero(node) {
+        return ts.isVoidExpression(node) && ts.isNumericLiteral(node.expression) && node.expression.text === "0";
+    }
     /**
      * Does not handle signed numeric names like `a[+0]` - handling those would require handling prefix unary expressions
      * throughout late binding handling as well, which is awkward (but ultimately probably doable if there is demand)
@@ -15073,7 +15493,7 @@ var ts;
     }
     ts.getElementOrPropertyAccessName = getElementOrPropertyAccessName;
     function getAssignmentDeclarationPropertyAccessKind(lhs) {
-        if (lhs.expression.kind === 104 /* ThisKeyword */) {
+        if (lhs.expression.kind === 107 /* ThisKeyword */) {
             return 4 /* ThisProperty */;
         }
         else if (isModuleExportsAccessExpression(lhs)) {
@@ -15118,7 +15538,7 @@ var ts;
     ts.isPrototypePropertyAssignment = isPrototypePropertyAssignment;
     function isSpecialPropertyDeclaration(expr) {
         return isInJSFile(expr) &&
-            expr.parent && expr.parent.kind === 226 /* ExpressionStatement */ &&
+            expr.parent && expr.parent.kind === 233 /* ExpressionStatement */ &&
             (!ts.isElementAccessExpression(expr) || isLiteralLikeElementAccess(expr)) &&
             !!ts.getJSDocTypeTag(expr.parent);
     }
@@ -15139,7 +15559,7 @@ var ts;
             return false;
         }
         var decl = symbol.valueDeclaration;
-        return decl.kind === 244 /* FunctionDeclaration */ || ts.isVariableDeclaration(decl) && decl.initializer && ts.isFunctionLike(decl.initializer);
+        return decl.kind === 251 /* FunctionDeclaration */ || ts.isVariableDeclaration(decl) && decl.initializer && ts.isFunctionLike(decl.initializer);
     }
     ts.isFunctionSymbol = isFunctionSymbol;
     function importFromModuleSpecifier(node) {
@@ -15148,14 +15568,14 @@ var ts;
     ts.importFromModuleSpecifier = importFromModuleSpecifier;
     function tryGetImportFromModuleSpecifier(node) {
         switch (node.parent.kind) {
-            case 254 /* ImportDeclaration */:
-            case 260 /* ExportDeclaration */:
+            case 261 /* ImportDeclaration */:
+            case 267 /* ExportDeclaration */:
                 return node.parent;
-            case 265 /* ExternalModuleReference */:
+            case 272 /* ExternalModuleReference */:
                 return node.parent.parent;
-            case 196 /* CallExpression */:
+            case 203 /* CallExpression */:
                 return isImportCall(node.parent) || isRequireCall(node.parent, /*checkArg*/ false) ? node.parent : undefined;
-            case 187 /* LiteralType */:
+            case 191 /* LiteralType */:
                 ts.Debug.assert(ts.isStringLiteral(node));
                 return ts.tryCast(node.parent.parent, ts.isImportTypeNode);
             default:
@@ -15165,12 +15585,12 @@ var ts;
     ts.tryGetImportFromModuleSpecifier = tryGetImportFromModuleSpecifier;
     function getExternalModuleName(node) {
         switch (node.kind) {
-            case 254 /* ImportDeclaration */:
-            case 260 /* ExportDeclaration */:
+            case 261 /* ImportDeclaration */:
+            case 267 /* ExportDeclaration */:
                 return node.moduleSpecifier;
-            case 253 /* ImportEqualsDeclaration */:
-                return node.moduleReference.kind === 265 /* ExternalModuleReference */ ? node.moduleReference.expression : undefined;
-            case 188 /* ImportType */:
+            case 260 /* ImportEqualsDeclaration */:
+                return node.moduleReference.kind === 272 /* ExternalModuleReference */ ? node.moduleReference.expression : undefined;
+            case 195 /* ImportType */:
                 return isLiteralImportTypeNode(node) ? node.argument.literal : undefined;
             default:
                 return ts.Debug.assertNever(node);
@@ -15179,11 +15599,11 @@ var ts;
     ts.getExternalModuleName = getExternalModuleName;
     function getNamespaceDeclarationNode(node) {
         switch (node.kind) {
-            case 254 /* ImportDeclaration */:
+            case 261 /* ImportDeclaration */:
                 return node.importClause && ts.tryCast(node.importClause.namedBindings, ts.isNamespaceImport);
-            case 253 /* ImportEqualsDeclaration */:
+            case 260 /* ImportEqualsDeclaration */:
                 return node;
-            case 260 /* ExportDeclaration */:
+            case 267 /* ExportDeclaration */:
                 return node.exportClause && ts.tryCast(node.exportClause, ts.isNamespaceExport);
             default:
                 return ts.Debug.assertNever(node);
@@ -15191,7 +15611,7 @@ var ts;
     }
     ts.getNamespaceDeclarationNode = getNamespaceDeclarationNode;
     function isDefaultImport(node) {
-        return node.kind === 254 /* ImportDeclaration */ && !!node.importClause && !!node.importClause.name;
+        return node.kind === 261 /* ImportDeclaration */ && !!node.importClause && !!node.importClause.name;
     }
     ts.isDefaultImport = isDefaultImport;
     function forEachImportClauseDeclaration(node, action) {
@@ -15212,13 +15632,13 @@ var ts;
     function hasQuestionToken(node) {
         if (node) {
             switch (node.kind) {
-                case 156 /* Parameter */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 282 /* ShorthandPropertyAssignment */:
-                case 281 /* PropertyAssignment */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
+                case 160 /* Parameter */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 289 /* ShorthandPropertyAssignment */:
+                case 288 /* PropertyAssignment */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
                     return node.questionToken !== undefined;
             }
         }
@@ -15232,7 +15652,7 @@ var ts;
     }
     ts.isJSDocConstructSignature = isJSDocConstructSignature;
     function isJSDocTypeAlias(node) {
-        return node.kind === 322 /* JSDocTypedefTag */ || node.kind === 315 /* JSDocCallbackTag */ || node.kind === 316 /* JSDocEnumTag */;
+        return node.kind === 331 /* JSDocTypedefTag */ || node.kind === 324 /* JSDocCallbackTag */ || node.kind === 325 /* JSDocEnumTag */;
     }
     ts.isJSDocTypeAlias = isJSDocTypeAlias;
     function isTypeAlias(node) {
@@ -15257,12 +15677,12 @@ var ts;
     }
     function getSingleInitializerOfVariableStatementOrPropertyDeclaration(node) {
         switch (node.kind) {
-            case 225 /* VariableStatement */:
+            case 232 /* VariableStatement */:
                 var v = getSingleVariableOfVariableStatement(node);
                 return v && v.initializer;
-            case 159 /* PropertyDeclaration */:
+            case 163 /* PropertyDeclaration */:
                 return node.initializer;
-            case 281 /* PropertyAssignment */:
+            case 288 /* PropertyAssignment */:
                 return node.initializer;
         }
     }
@@ -15270,14 +15690,15 @@ var ts;
     function getSingleVariableOfVariableStatement(node) {
         return ts.isVariableStatement(node) ? ts.firstOrUndefined(node.declarationList.declarations) : undefined;
     }
+    ts.getSingleVariableOfVariableStatement = getSingleVariableOfVariableStatement;
     function getNestedModuleDeclaration(node) {
         return ts.isModuleDeclaration(node) &&
             node.body &&
-            node.body.kind === 249 /* ModuleDeclaration */
+            node.body.kind === 256 /* ModuleDeclaration */
             ? node.body
             : undefined;
     }
-    function getJSDocCommentsAndTags(hostNode) {
+    function getJSDocCommentsAndTags(hostNode, noCache) {
         var result;
         // Pull parameter comments from declaring function as well
         if (isVariableLike(hostNode) && ts.hasInitializer(hostNode) && ts.hasJSDocNodes(hostNode.initializer)) {
@@ -15288,12 +15709,12 @@ var ts;
             if (ts.hasJSDocNodes(node)) {
                 result = ts.append(result, ts.last(node.jsDoc));
             }
-            if (node.kind === 156 /* Parameter */) {
-                result = ts.addRange(result, ts.getJSDocParameterTags(node));
+            if (node.kind === 160 /* Parameter */) {
+                result = ts.addRange(result, (noCache ? ts.getJSDocParameterTagsNoCache : ts.getJSDocParameterTags)(node));
                 break;
             }
-            if (node.kind === 155 /* TypeParameter */) {
-                result = ts.addRange(result, ts.getJSDocTypeParameterTags(node));
+            if (node.kind === 159 /* TypeParameter */) {
+                result = ts.addRange(result, (noCache ? ts.getJSDocTypeParameterTagsNoCache : ts.getJSDocTypeParameterTags)(node));
                 break;
             }
             node = getNextJSDocCommentLocation(node);
@@ -15303,10 +15724,10 @@ var ts;
     ts.getJSDocCommentsAndTags = getJSDocCommentsAndTags;
     function getNextJSDocCommentLocation(node) {
         var parent = node.parent;
-        if (parent.kind === 281 /* PropertyAssignment */ ||
-            parent.kind === 259 /* ExportAssignment */ ||
-            parent.kind === 159 /* PropertyDeclaration */ ||
-            parent.kind === 226 /* ExpressionStatement */ && node.kind === 194 /* PropertyAccessExpression */ ||
+        if (parent.kind === 288 /* PropertyAssignment */ ||
+            parent.kind === 266 /* ExportAssignment */ ||
+            parent.kind === 163 /* PropertyDeclaration */ ||
+            parent.kind === 233 /* ExpressionStatement */ && node.kind === 201 /* PropertyAccessExpression */ ||
             getNestedModuleDeclaration(parent) ||
             ts.isBinaryExpression(node) && node.operatorToken.kind === 62 /* EqualsToken */) {
             return parent;
@@ -15342,7 +15763,7 @@ var ts;
         if (!decl) {
             return undefined;
         }
-        var parameter = ts.find(decl.parameters, function (p) { return p.name.kind === 75 /* Identifier */ && p.name.escapedText === name; });
+        var parameter = ts.find(decl.parameters, function (p) { return p.name.kind === 78 /* Identifier */ && p.name.escapedText === name; });
         return parameter && parameter.symbol;
     }
     ts.getParameterSymbolFromJSDoc = getParameterSymbolFromJSDoc;
@@ -15364,7 +15785,7 @@ var ts;
     ts.getEffectiveJSDocHost = getEffectiveJSDocHost;
     /** Use getEffectiveJSDocHost if you additionally need to look for jsdoc on parent nodes, like assignments.  */
     function getJSDocHost(node) {
-        return ts.Debug.checkDefined(findAncestor(node.parent, ts.isJSDoc)).parent;
+        return ts.Debug.checkDefined(ts.findAncestor(node.parent, ts.isJSDoc)).parent;
     }
     ts.getJSDocHost = getJSDocHost;
     function getTypeParameterFromJsDoc(node) {
@@ -15380,7 +15801,7 @@ var ts;
     ts.hasRestParameter = hasRestParameter;
     function isRestParameter(node) {
         var type = ts.isJSDocParameterTag(node) ? (node.typeExpression && node.typeExpression.type) : node.type;
-        return node.dotDotDotToken !== undefined || !!type && type.kind === 301 /* JSDocVariadicType */;
+        return node.dotDotDotToken !== undefined || !!type && type.kind === 309 /* JSDocVariadicType */;
     }
     ts.isRestParameter = isRestParameter;
     function hasTypeArguments(node) {
@@ -15397,31 +15818,31 @@ var ts;
         var parent = node.parent;
         while (true) {
             switch (parent.kind) {
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     var binaryOperator = parent.operatorToken.kind;
                     return isAssignmentOperator(binaryOperator) && parent.left === node ?
-                        binaryOperator === 62 /* EqualsToken */ ? 1 /* Definite */ : 2 /* Compound */ :
+                        binaryOperator === 62 /* EqualsToken */ || isLogicalOrCoalescingAssignmentOperator(binaryOperator) ? 1 /* Definite */ : 2 /* Compound */ :
                         0 /* None */;
-                case 207 /* PrefixUnaryExpression */:
-                case 208 /* PostfixUnaryExpression */:
+                case 214 /* PrefixUnaryExpression */:
+                case 215 /* PostfixUnaryExpression */:
                     var unaryOperator = parent.operator;
                     return unaryOperator === 45 /* PlusPlusToken */ || unaryOperator === 46 /* MinusMinusToken */ ? 2 /* Compound */ : 0 /* None */;
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
                     return parent.initializer === node ? 1 /* Definite */ : 0 /* None */;
-                case 200 /* ParenthesizedExpression */:
-                case 192 /* ArrayLiteralExpression */:
-                case 213 /* SpreadElement */:
-                case 218 /* NonNullExpression */:
+                case 207 /* ParenthesizedExpression */:
+                case 199 /* ArrayLiteralExpression */:
+                case 220 /* SpreadElement */:
+                case 225 /* NonNullExpression */:
                     node = parent;
                     break;
-                case 282 /* ShorthandPropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
                     if (parent.name !== node) {
                         return 0 /* None */;
                     }
                     node = parent.parent;
                     break;
-                case 281 /* PropertyAssignment */:
+                case 288 /* PropertyAssignment */:
                     if (parent.name === node) {
                         return 0 /* None */;
                     }
@@ -15448,22 +15869,22 @@ var ts;
      */
     function isNodeWithPossibleHoistedDeclaration(node) {
         switch (node.kind) {
-            case 223 /* Block */:
-            case 225 /* VariableStatement */:
-            case 236 /* WithStatement */:
-            case 227 /* IfStatement */:
-            case 237 /* SwitchStatement */:
-            case 251 /* CaseBlock */:
-            case 277 /* CaseClause */:
-            case 278 /* DefaultClause */:
-            case 238 /* LabeledStatement */:
-            case 230 /* ForStatement */:
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
-            case 228 /* DoStatement */:
-            case 229 /* WhileStatement */:
-            case 240 /* TryStatement */:
-            case 280 /* CatchClause */:
+            case 230 /* Block */:
+            case 232 /* VariableStatement */:
+            case 243 /* WithStatement */:
+            case 234 /* IfStatement */:
+            case 244 /* SwitchStatement */:
+            case 258 /* CaseBlock */:
+            case 284 /* CaseClause */:
+            case 285 /* DefaultClause */:
+            case 245 /* LabeledStatement */:
+            case 237 /* ForStatement */:
+            case 238 /* ForInStatement */:
+            case 239 /* ForOfStatement */:
+            case 235 /* DoStatement */:
+            case 236 /* WhileStatement */:
+            case 247 /* TryStatement */:
+            case 287 /* CatchClause */:
                 return true;
         }
         return false;
@@ -15480,30 +15901,44 @@ var ts;
         return node;
     }
     function walkUpParenthesizedTypes(node) {
-        return walkUp(node, 182 /* ParenthesizedType */);
+        return walkUp(node, 186 /* ParenthesizedType */);
     }
     ts.walkUpParenthesizedTypes = walkUpParenthesizedTypes;
     function walkUpParenthesizedExpressions(node) {
-        return walkUp(node, 200 /* ParenthesizedExpression */);
+        return walkUp(node, 207 /* ParenthesizedExpression */);
     }
     ts.walkUpParenthesizedExpressions = walkUpParenthesizedExpressions;
+    /**
+     * Walks up parenthesized types.
+     * It returns both the outermost parenthesized type and its parent.
+     * If given node is not a parenthesiezd type, undefined is return as the former.
+     */
+    function walkUpParenthesizedTypesAndGetParentAndChild(node) {
+        var child;
+        while (node && node.kind === 186 /* ParenthesizedType */) {
+            child = node;
+            node = node.parent;
+        }
+        return [child, node];
+    }
+    ts.walkUpParenthesizedTypesAndGetParentAndChild = walkUpParenthesizedTypesAndGetParentAndChild;
     function skipParentheses(node) {
         return ts.skipOuterExpressions(node, 1 /* Parentheses */);
     }
     ts.skipParentheses = skipParentheses;
     function skipParenthesesUp(node) {
-        while (node.kind === 200 /* ParenthesizedExpression */) {
+        while (node.kind === 207 /* ParenthesizedExpression */) {
             node = node.parent;
         }
         return node;
     }
     // a node is delete target iff. it is PropertyAccessExpression/ElementAccessExpression with parentheses skipped
     function isDeleteTarget(node) {
-        if (node.kind !== 194 /* PropertyAccessExpression */ && node.kind !== 195 /* ElementAccessExpression */) {
+        if (node.kind !== 201 /* PropertyAccessExpression */ && node.kind !== 202 /* ElementAccessExpression */) {
             return false;
         }
         node = walkUpParenthesizedExpressions(node.parent);
-        return node && node.kind === 203 /* DeleteExpression */;
+        return node && node.kind === 210 /* DeleteExpression */;
     }
     ts.isDeleteTarget = isDeleteTarget;
     function isNodeDescendantOf(node, ancestor) {
@@ -15530,7 +15965,7 @@ var ts;
                 if (ts.isComputedPropertyName(parent))
                     return parent.parent;
             // falls through
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 if (ts.isDeclaration(parent)) {
                     return parent.name === name ? parent : undefined;
                 }
@@ -15547,7 +15982,7 @@ var ts;
                         ? binExp
                         : undefined;
                 }
-            case 76 /* PrivateIdentifier */:
+            case 79 /* PrivateIdentifier */:
                 return ts.isDeclaration(parent) && parent.name === name ? parent : undefined;
             default:
                 return undefined;
@@ -15556,7 +15991,7 @@ var ts;
     ts.getDeclarationFromName = getDeclarationFromName;
     function isLiteralComputedPropertyDeclarationName(node) {
         return isStringOrNumericLiteralLike(node) &&
-            node.parent.kind === 154 /* ComputedPropertyName */ &&
+            node.parent.kind === 158 /* ComputedPropertyName */ &&
             ts.isDeclaration(node.parent.parent);
     }
     ts.isLiteralComputedPropertyDeclarationName = isLiteralComputedPropertyDeclarationName;
@@ -15564,32 +15999,26 @@ var ts;
     function isIdentifierName(node) {
         var parent = node.parent;
         switch (parent.kind) {
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 284 /* EnumMember */:
-            case 281 /* PropertyAssignment */:
-            case 194 /* PropertyAccessExpression */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 291 /* EnumMember */:
+            case 288 /* PropertyAssignment */:
+            case 201 /* PropertyAccessExpression */:
                 // Name in member declaration or property name in property access
                 return parent.name === node;
-            case 153 /* QualifiedName */:
+            case 157 /* QualifiedName */:
                 // Name on right hand side of dot in a type query or type reference
-                if (parent.right === node) {
-                    while (parent.kind === 153 /* QualifiedName */) {
-                        parent = parent.parent;
-                    }
-                    return parent.kind === 172 /* TypeQuery */ || parent.kind === 169 /* TypeReference */;
-                }
-                return false;
-            case 191 /* BindingElement */:
-            case 258 /* ImportSpecifier */:
+                return parent.right === node;
+            case 198 /* BindingElement */:
+            case 265 /* ImportSpecifier */:
                 // Property name in binding element or import specifier
                 return parent.propertyName === node;
-            case 263 /* ExportSpecifier */:
-            case 273 /* JsxAttribute */:
+            case 270 /* ExportSpecifier */:
+            case 280 /* JsxAttribute */:
                 // Any name in an export specifier or JSX Attribute
                 return true;
         }
@@ -15609,33 +16038,33 @@ var ts;
     // {<Identifier>}
     // {name: <EntityNameExpression>}
     function isAliasSymbolDeclaration(node) {
-        return node.kind === 253 /* ImportEqualsDeclaration */ ||
-            node.kind === 252 /* NamespaceExportDeclaration */ ||
-            node.kind === 255 /* ImportClause */ && !!node.name ||
-            node.kind === 256 /* NamespaceImport */ ||
-            node.kind === 262 /* NamespaceExport */ ||
-            node.kind === 258 /* ImportSpecifier */ ||
-            node.kind === 263 /* ExportSpecifier */ ||
-            node.kind === 259 /* ExportAssignment */ && exportAssignmentIsAlias(node) ||
+        return node.kind === 260 /* ImportEqualsDeclaration */ ||
+            node.kind === 259 /* NamespaceExportDeclaration */ ||
+            node.kind === 262 /* ImportClause */ && !!node.name ||
+            node.kind === 263 /* NamespaceImport */ ||
+            node.kind === 269 /* NamespaceExport */ ||
+            node.kind === 265 /* ImportSpecifier */ ||
+            node.kind === 270 /* ExportSpecifier */ ||
+            node.kind === 266 /* ExportAssignment */ && exportAssignmentIsAlias(node) ||
             ts.isBinaryExpression(node) && getAssignmentDeclarationKind(node) === 2 /* ModuleExports */ && exportAssignmentIsAlias(node) ||
             ts.isPropertyAccessExpression(node) && ts.isBinaryExpression(node.parent) && node.parent.left === node && node.parent.operatorToken.kind === 62 /* EqualsToken */ && isAliasableExpression(node.parent.right) ||
-            node.kind === 282 /* ShorthandPropertyAssignment */ ||
-            node.kind === 281 /* PropertyAssignment */ && isAliasableExpression(node.initializer);
+            node.kind === 289 /* ShorthandPropertyAssignment */ ||
+            node.kind === 288 /* PropertyAssignment */ && isAliasableExpression(node.initializer);
     }
     ts.isAliasSymbolDeclaration = isAliasSymbolDeclaration;
     function getAliasDeclarationFromName(node) {
         switch (node.parent.kind) {
-            case 255 /* ImportClause */:
-            case 258 /* ImportSpecifier */:
-            case 256 /* NamespaceImport */:
-            case 263 /* ExportSpecifier */:
-            case 259 /* ExportAssignment */:
-            case 253 /* ImportEqualsDeclaration */:
+            case 262 /* ImportClause */:
+            case 265 /* ImportSpecifier */:
+            case 263 /* NamespaceImport */:
+            case 270 /* ExportSpecifier */:
+            case 266 /* ExportAssignment */:
+            case 260 /* ImportEqualsDeclaration */:
                 return node.parent;
-            case 153 /* QualifiedName */:
+            case 157 /* QualifiedName */:
                 do {
                     node = node.parent;
-                } while (node.parent.kind === 153 /* QualifiedName */);
+                } while (node.parent.kind === 157 /* QualifiedName */);
                 return getAliasDeclarationFromName(node);
         }
     }
@@ -15654,7 +16083,7 @@ var ts;
     }
     ts.getExportAssignmentExpression = getExportAssignmentExpression;
     function getPropertyAssignmentAliasLikeExpression(node) {
-        return node.kind === 282 /* ShorthandPropertyAssignment */ ? node.name : node.kind === 281 /* PropertyAssignment */ ? node.initializer :
+        return node.kind === 289 /* ShorthandPropertyAssignment */ ? node.name : node.kind === 288 /* PropertyAssignment */ ? node.initializer :
             node.parent.right;
     }
     ts.getPropertyAssignmentAliasLikeExpression = getPropertyAssignmentAliasLikeExpression;
@@ -15671,7 +16100,7 @@ var ts;
     }
     ts.getEffectiveBaseTypeNode = getEffectiveBaseTypeNode;
     function getClassExtendsHeritageElement(node) {
-        var heritageClause = getHeritageClause(node.heritageClauses, 90 /* ExtendsKeyword */);
+        var heritageClause = getHeritageClause(node.heritageClauses, 93 /* ExtendsKeyword */);
         return heritageClause && heritageClause.types.length > 0 ? heritageClause.types[0] : undefined;
     }
     ts.getClassExtendsHeritageElement = getClassExtendsHeritageElement;
@@ -15680,7 +16109,7 @@ var ts;
             return ts.getJSDocImplementsTags(node).map(function (n) { return n.class; });
         }
         else {
-            var heritageClause = getHeritageClause(node.heritageClauses, 113 /* ImplementsKeyword */);
+            var heritageClause = getHeritageClause(node.heritageClauses, 116 /* ImplementsKeyword */);
             return heritageClause === null || heritageClause === void 0 ? void 0 : heritageClause.types;
         }
     }
@@ -15693,7 +16122,7 @@ var ts;
     }
     ts.getAllSuperTypeNodes = getAllSuperTypeNodes;
     function getInterfaceBaseTypeNodes(node) {
-        var heritageClause = getHeritageClause(node.heritageClauses, 90 /* ExtendsKeyword */);
+        var heritageClause = getHeritageClause(node.heritageClauses, 93 /* ExtendsKeyword */);
         return heritageClause ? heritageClause.types : undefined;
     }
     ts.getInterfaceBaseTypeNodes = getInterfaceBaseTypeNodes;
@@ -15720,11 +16149,11 @@ var ts;
     }
     ts.getAncestor = getAncestor;
     function isKeyword(token) {
-        return 77 /* FirstKeyword */ <= token && token <= 152 /* LastKeyword */;
+        return 80 /* FirstKeyword */ <= token && token <= 156 /* LastKeyword */;
     }
     ts.isKeyword = isKeyword;
     function isContextualKeyword(token) {
-        return 122 /* FirstContextualKeyword */ <= token && token <= 152 /* LastContextualKeyword */;
+        return 125 /* FirstContextualKeyword */ <= token && token <= 156 /* LastContextualKeyword */;
     }
     ts.isContextualKeyword = isContextualKeyword;
     function isNonContextualKeyword(token) {
@@ -15732,7 +16161,7 @@ var ts;
     }
     ts.isNonContextualKeyword = isNonContextualKeyword;
     function isFutureReservedKeyword(token) {
-        return 113 /* FirstFutureReservedWord */ <= token && token <= 121 /* LastFutureReservedWord */;
+        return 116 /* FirstFutureReservedWord */ <= token && token <= 124 /* LastFutureReservedWord */;
     }
     ts.isFutureReservedKeyword = isFutureReservedKeyword;
     function isStringANonContextualKeyword(name) {
@@ -15768,15 +16197,15 @@ var ts;
         }
         var flags = 0 /* Normal */;
         switch (node.kind) {
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 161 /* MethodDeclaration */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 165 /* MethodDeclaration */:
                 if (node.asteriskToken) {
                     flags |= 1 /* Generator */;
                 }
             // falls through
-            case 202 /* ArrowFunction */:
-                if (hasModifier(node, 256 /* Async */)) {
+            case 209 /* ArrowFunction */:
+                if (hasSyntacticModifier(node, 256 /* Async */)) {
                     flags |= 2 /* Async */;
                 }
                 break;
@@ -15789,13 +16218,13 @@ var ts;
     ts.getFunctionFlags = getFunctionFlags;
     function isAsyncFunction(node) {
         switch (node.kind) {
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
-            case 161 /* MethodDeclaration */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
+            case 165 /* MethodDeclaration */:
                 return node.body !== undefined
                     && node.asteriskToken === undefined
-                    && hasModifier(node, 256 /* Async */);
+                    && hasSyntacticModifier(node, 256 /* Async */);
         }
         return false;
     }
@@ -15825,10 +16254,10 @@ var ts;
     }
     ts.hasDynamicName = hasDynamicName;
     function isDynamicName(name) {
-        if (!(name.kind === 154 /* ComputedPropertyName */ || name.kind === 195 /* ElementAccessExpression */)) {
+        if (!(name.kind === 158 /* ComputedPropertyName */ || name.kind === 202 /* ElementAccessExpression */)) {
             return false;
         }
-        var expr = ts.isElementAccessExpression(name) ? name.argumentExpression : name.expression;
+        var expr = ts.isElementAccessExpression(name) ? skipParentheses(name.argumentExpression) : name.expression;
         return !isStringOrNumericLiteralLike(expr) &&
             !isSignedNumericLiteral(expr) &&
             !isWellKnownSymbolSyntactically(expr);
@@ -15845,13 +16274,13 @@ var ts;
     ts.isWellKnownSymbolSyntactically = isWellKnownSymbolSyntactically;
     function getPropertyNameForPropertyNameNode(name) {
         switch (name.kind) {
-            case 75 /* Identifier */:
-            case 76 /* PrivateIdentifier */:
+            case 78 /* Identifier */:
+            case 79 /* PrivateIdentifier */:
                 return name.escapedText;
             case 10 /* StringLiteral */:
             case 8 /* NumericLiteral */:
                 return ts.escapeLeadingUnderscores(name.text);
-            case 154 /* ComputedPropertyName */:
+            case 158 /* ComputedPropertyName */:
                 var nameExpression = name.expression;
                 if (isWellKnownSymbolSyntactically(nameExpression)) {
                     return getPropertyNameForKnownSymbolName(ts.idText(nameExpression.name));
@@ -15859,6 +16288,12 @@ var ts;
                 else if (isStringOrNumericLiteralLike(nameExpression)) {
                     return ts.escapeLeadingUnderscores(nameExpression.text);
                 }
+                else if (isSignedNumericLiteral(nameExpression)) {
+                    if (nameExpression.operator === 40 /* MinusToken */) {
+                        return ts.tokenToString(nameExpression.operator) + nameExpression.operand.text;
+                    }
+                    return nameExpression.operand.text;
+                }
                 return undefined;
             default:
                 return ts.Debug.assertNever(name);
@@ -15867,7 +16302,7 @@ var ts;
     ts.getPropertyNameForPropertyNameNode = getPropertyNameForPropertyNameNode;
     function isPropertyNameLiteral(node) {
         switch (node.kind) {
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
             case 10 /* StringLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
             case 8 /* NumericLiteral */:
@@ -15905,7 +16340,7 @@ var ts;
      * Includes the word "Symbol" with unicode escapes
      */
     function isESSymbolIdentifier(node) {
-        return node.kind === 75 /* Identifier */ && node.escapedText === "Symbol";
+        return node.kind === 78 /* Identifier */ && node.escapedText === "Symbol";
     }
     ts.isESSymbolIdentifier = isESSymbolIdentifier;
     function isPushOrUnshiftIdentifier(node) {
@@ -15914,11 +16349,11 @@ var ts;
     ts.isPushOrUnshiftIdentifier = isPushOrUnshiftIdentifier;
     function isParameterDeclaration(node) {
         var root = getRootDeclaration(node);
-        return root.kind === 156 /* Parameter */;
+        return root.kind === 160 /* Parameter */;
     }
     ts.isParameterDeclaration = isParameterDeclaration;
     function getRootDeclaration(node) {
-        while (node.kind === 191 /* BindingElement */) {
+        while (node.kind === 198 /* BindingElement */) {
             node = node.parent.parent;
         }
         return node;
@@ -15926,15 +16361,15 @@ var ts;
     ts.getRootDeclaration = getRootDeclaration;
     function nodeStartsNewLexicalEnvironment(node) {
         var kind = node.kind;
-        return kind === 162 /* Constructor */
-            || kind === 201 /* FunctionExpression */
-            || kind === 244 /* FunctionDeclaration */
-            || kind === 202 /* ArrowFunction */
-            || kind === 161 /* MethodDeclaration */
-            || kind === 163 /* GetAccessor */
-            || kind === 164 /* SetAccessor */
-            || kind === 249 /* ModuleDeclaration */
-            || kind === 290 /* SourceFile */;
+        return kind === 166 /* Constructor */
+            || kind === 208 /* FunctionExpression */
+            || kind === 251 /* FunctionDeclaration */
+            || kind === 209 /* ArrowFunction */
+            || kind === 165 /* MethodDeclaration */
+            || kind === 167 /* GetAccessor */
+            || kind === 168 /* SetAccessor */
+            || kind === 256 /* ModuleDeclaration */
+            || kind === 297 /* SourceFile */;
     }
     ts.nodeStartsNewLexicalEnvironment = nodeStartsNewLexicalEnvironment;
     function nodeIsSynthesized(range) {
@@ -15953,23 +16388,23 @@ var ts;
     })(Associativity = ts.Associativity || (ts.Associativity = {}));
     function getExpressionAssociativity(expression) {
         var operator = getOperator(expression);
-        var hasArguments = expression.kind === 197 /* NewExpression */ && expression.arguments !== undefined;
+        var hasArguments = expression.kind === 204 /* NewExpression */ && expression.arguments !== undefined;
         return getOperatorAssociativity(expression.kind, operator, hasArguments);
     }
     ts.getExpressionAssociativity = getExpressionAssociativity;
     function getOperatorAssociativity(kind, operator, hasArguments) {
         switch (kind) {
-            case 197 /* NewExpression */:
+            case 204 /* NewExpression */:
                 return hasArguments ? 0 /* Left */ : 1 /* Right */;
-            case 207 /* PrefixUnaryExpression */:
-            case 204 /* TypeOfExpression */:
-            case 205 /* VoidExpression */:
-            case 203 /* DeleteExpression */:
-            case 206 /* AwaitExpression */:
-            case 210 /* ConditionalExpression */:
-            case 212 /* YieldExpression */:
+            case 214 /* PrefixUnaryExpression */:
+            case 211 /* TypeOfExpression */:
+            case 212 /* VoidExpression */:
+            case 210 /* DeleteExpression */:
+            case 213 /* AwaitExpression */:
+            case 217 /* ConditionalExpression */:
+            case 219 /* YieldExpression */:
                 return 1 /* Right */;
-            case 209 /* BinaryExpression */:
+            case 216 /* BinaryExpression */:
                 switch (operator) {
                     case 42 /* AsteriskAsteriskToken */:
                     case 62 /* EqualsToken */:
@@ -15983,8 +16418,11 @@ var ts;
                     case 70 /* GreaterThanGreaterThanEqualsToken */:
                     case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
                     case 72 /* AmpersandEqualsToken */:
-                    case 74 /* CaretEqualsToken */:
+                    case 77 /* CaretEqualsToken */:
                     case 73 /* BarEqualsToken */:
+                    case 74 /* BarBarEqualsToken */:
+                    case 75 /* AmpersandAmpersandEqualsToken */:
+                    case 76 /* QuestionQuestionEqualsToken */:
                         return 1 /* Right */;
                 }
         }
@@ -15993,15 +16431,15 @@ var ts;
     ts.getOperatorAssociativity = getOperatorAssociativity;
     function getExpressionPrecedence(expression) {
         var operator = getOperator(expression);
-        var hasArguments = expression.kind === 197 /* NewExpression */ && expression.arguments !== undefined;
+        var hasArguments = expression.kind === 204 /* NewExpression */ && expression.arguments !== undefined;
         return getOperatorPrecedence(expression.kind, operator, hasArguments);
     }
     ts.getExpressionPrecedence = getExpressionPrecedence;
     function getOperator(expression) {
-        if (expression.kind === 209 /* BinaryExpression */) {
+        if (expression.kind === 216 /* BinaryExpression */) {
             return expression.operatorToken.kind;
         }
-        else if (expression.kind === 207 /* PrefixUnaryExpression */ || expression.kind === 208 /* PostfixUnaryExpression */) {
+        else if (expression.kind === 214 /* PrefixUnaryExpression */ || expression.kind === 215 /* PostfixUnaryExpression */) {
             return expression.operator;
         }
         else {
@@ -16009,20 +16447,189 @@ var ts;
         }
     }
     ts.getOperator = getOperator;
+    var OperatorPrecedence;
+    (function (OperatorPrecedence) {
+        // Expression:
+        //     AssignmentExpression
+        //     Expression `,` AssignmentExpression
+        OperatorPrecedence[OperatorPrecedence["Comma"] = 0] = "Comma";
+        // NOTE: `Spread` is higher than `Comma` due to how it is parsed in |ElementList|
+        // SpreadElement:
+        //     `...` AssignmentExpression
+        OperatorPrecedence[OperatorPrecedence["Spread"] = 1] = "Spread";
+        // AssignmentExpression:
+        //     ConditionalExpression
+        //     YieldExpression
+        //     ArrowFunction
+        //     AsyncArrowFunction
+        //     LeftHandSideExpression `=` AssignmentExpression
+        //     LeftHandSideExpression AssignmentOperator AssignmentExpression
+        //
+        // NOTE: AssignmentExpression is broken down into several precedences due to the requirements
+        //       of the parenthesizer rules.
+        // AssignmentExpression: YieldExpression
+        // YieldExpression:
+        //     `yield`
+        //     `yield` AssignmentExpression
+        //     `yield` `*` AssignmentExpression
+        OperatorPrecedence[OperatorPrecedence["Yield"] = 2] = "Yield";
+        // AssignmentExpression: LeftHandSideExpression `=` AssignmentExpression
+        // AssignmentExpression: LeftHandSideExpression AssignmentOperator AssignmentExpression
+        // AssignmentOperator: one of
+        //     `*=` `/=` `%=` `+=` `-=` `<<=` `>>=` `>>>=` `&=` `^=` `|=` `**=`
+        OperatorPrecedence[OperatorPrecedence["Assignment"] = 3] = "Assignment";
+        // NOTE: `Conditional` is considered higher than `Assignment` here, but in reality they have
+        //       the same precedence.
+        // AssignmentExpression: ConditionalExpression
+        // ConditionalExpression:
+        //     ShortCircuitExpression
+        //     ShortCircuitExpression `?` AssignmentExpression `:` AssignmentExpression
+        // ShortCircuitExpression:
+        //     LogicalORExpression
+        //     CoalesceExpression
+        OperatorPrecedence[OperatorPrecedence["Conditional"] = 4] = "Conditional";
+        // CoalesceExpression:
+        //     CoalesceExpressionHead `??` BitwiseORExpression
+        // CoalesceExpressionHead:
+        //     CoalesceExpression
+        //     BitwiseORExpression
+        OperatorPrecedence[OperatorPrecedence["Coalesce"] = 4] = "Coalesce";
+        // LogicalORExpression:
+        //     LogicalANDExpression
+        //     LogicalORExpression `||` LogicalANDExpression
+        OperatorPrecedence[OperatorPrecedence["LogicalOR"] = 5] = "LogicalOR";
+        // LogicalANDExpression:
+        //     BitwiseORExpression
+        //     LogicalANDExprerssion `&&` BitwiseORExpression
+        OperatorPrecedence[OperatorPrecedence["LogicalAND"] = 6] = "LogicalAND";
+        // BitwiseORExpression:
+        //     BitwiseXORExpression
+        //     BitwiseORExpression `^` BitwiseXORExpression
+        OperatorPrecedence[OperatorPrecedence["BitwiseOR"] = 7] = "BitwiseOR";
+        // BitwiseXORExpression:
+        //     BitwiseANDExpression
+        //     BitwiseXORExpression `^` BitwiseANDExpression
+        OperatorPrecedence[OperatorPrecedence["BitwiseXOR"] = 8] = "BitwiseXOR";
+        // BitwiseANDExpression:
+        //     EqualityExpression
+        //     BitwiseANDExpression `^` EqualityExpression
+        OperatorPrecedence[OperatorPrecedence["BitwiseAND"] = 9] = "BitwiseAND";
+        // EqualityExpression:
+        //     RelationalExpression
+        //     EqualityExpression `==` RelationalExpression
+        //     EqualityExpression `!=` RelationalExpression
+        //     EqualityExpression `===` RelationalExpression
+        //     EqualityExpression `!==` RelationalExpression
+        OperatorPrecedence[OperatorPrecedence["Equality"] = 10] = "Equality";
+        // RelationalExpression:
+        //     ShiftExpression
+        //     RelationalExpression `<` ShiftExpression
+        //     RelationalExpression `>` ShiftExpression
+        //     RelationalExpression `<=` ShiftExpression
+        //     RelationalExpression `>=` ShiftExpression
+        //     RelationalExpression `instanceof` ShiftExpression
+        //     RelationalExpression `in` ShiftExpression
+        //     [+TypeScript] RelationalExpression `as` Type
+        OperatorPrecedence[OperatorPrecedence["Relational"] = 11] = "Relational";
+        // ShiftExpression:
+        //     AdditiveExpression
+        //     ShiftExpression `<<` AdditiveExpression
+        //     ShiftExpression `>>` AdditiveExpression
+        //     ShiftExpression `>>>` AdditiveExpression
+        OperatorPrecedence[OperatorPrecedence["Shift"] = 12] = "Shift";
+        // AdditiveExpression:
+        //     MultiplicativeExpression
+        //     AdditiveExpression `+` MultiplicativeExpression
+        //     AdditiveExpression `-` MultiplicativeExpression
+        OperatorPrecedence[OperatorPrecedence["Additive"] = 13] = "Additive";
+        // MultiplicativeExpression:
+        //     ExponentiationExpression
+        //     MultiplicativeExpression MultiplicativeOperator ExponentiationExpression
+        // MultiplicativeOperator: one of `*`, `/`, `%`
+        OperatorPrecedence[OperatorPrecedence["Multiplicative"] = 14] = "Multiplicative";
+        // ExponentiationExpression:
+        //     UnaryExpression
+        //     UpdateExpression `**` ExponentiationExpression
+        OperatorPrecedence[OperatorPrecedence["Exponentiation"] = 15] = "Exponentiation";
+        // UnaryExpression:
+        //     UpdateExpression
+        //     `delete` UnaryExpression
+        //     `void` UnaryExpression
+        //     `typeof` UnaryExpression
+        //     `+` UnaryExpression
+        //     `-` UnaryExpression
+        //     `~` UnaryExpression
+        //     `!` UnaryExpression
+        //     AwaitExpression
+        // UpdateExpression:            // TODO: Do we need to investigate the precedence here?
+        //     `++` UnaryExpression
+        //     `--` UnaryExpression
+        OperatorPrecedence[OperatorPrecedence["Unary"] = 16] = "Unary";
+        // UpdateExpression:
+        //     LeftHandSideExpression
+        //     LeftHandSideExpression `++`
+        //     LeftHandSideExpression `--`
+        OperatorPrecedence[OperatorPrecedence["Update"] = 17] = "Update";
+        // LeftHandSideExpression:
+        //     NewExpression
+        //     CallExpression
+        // NewExpression:
+        //     MemberExpression
+        //     `new` NewExpression
+        OperatorPrecedence[OperatorPrecedence["LeftHandSide"] = 18] = "LeftHandSide";
+        // CallExpression:
+        //     CoverCallExpressionAndAsyncArrowHead
+        //     SuperCall
+        //     ImportCall
+        //     CallExpression Arguments
+        //     CallExpression `[` Expression `]`
+        //     CallExpression `.` IdentifierName
+        //     CallExpression TemplateLiteral
+        // MemberExpression:
+        //     PrimaryExpression
+        //     MemberExpression `[` Expression `]`
+        //     MemberExpression `.` IdentifierName
+        //     MemberExpression TemplateLiteral
+        //     SuperProperty
+        //     MetaProperty
+        //     `new` MemberExpression Arguments
+        OperatorPrecedence[OperatorPrecedence["Member"] = 19] = "Member";
+        // TODO: JSXElement?
+        // PrimaryExpression:
+        //     `this`
+        //     IdentifierReference
+        //     Literal
+        //     ArrayLiteral
+        //     ObjectLiteral
+        //     FunctionExpression
+        //     ClassExpression
+        //     GeneratorExpression
+        //     AsyncFunctionExpression
+        //     AsyncGeneratorExpression
+        //     RegularExpressionLiteral
+        //     TemplateLiteral
+        //     CoverParenthesizedExpressionAndArrowParameterList
+        OperatorPrecedence[OperatorPrecedence["Primary"] = 20] = "Primary";
+        OperatorPrecedence[OperatorPrecedence["Highest"] = 20] = "Highest";
+        OperatorPrecedence[OperatorPrecedence["Lowest"] = 0] = "Lowest";
+        // -1 is lower than all other precedences. Returning it will cause binary expression
+        // parsing to stop.
+        OperatorPrecedence[OperatorPrecedence["Invalid"] = -1] = "Invalid";
+    })(OperatorPrecedence = ts.OperatorPrecedence || (ts.OperatorPrecedence = {}));
     function getOperatorPrecedence(nodeKind, operatorKind, hasArguments) {
         switch (nodeKind) {
-            case 327 /* CommaListExpression */:
-                return 0;
-            case 213 /* SpreadElement */:
-                return 1;
-            case 212 /* YieldExpression */:
-                return 2;
-            case 210 /* ConditionalExpression */:
-                return 4;
-            case 209 /* BinaryExpression */:
+            case 337 /* CommaListExpression */:
+                return 0 /* Comma */;
+            case 220 /* SpreadElement */:
+                return 1 /* Spread */;
+            case 219 /* YieldExpression */:
+                return 2 /* Yield */;
+            case 217 /* ConditionalExpression */:
+                return 4 /* Conditional */;
+            case 216 /* BinaryExpression */:
                 switch (operatorKind) {
                     case 27 /* CommaToken */:
-                        return 0;
+                        return 0 /* Comma */;
                     case 62 /* EqualsToken */:
                     case 63 /* PlusEqualsToken */:
                     case 64 /* MinusEqualsToken */:
@@ -16034,106 +16641,124 @@ var ts;
                     case 70 /* GreaterThanGreaterThanEqualsToken */:
                     case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
                     case 72 /* AmpersandEqualsToken */:
-                    case 74 /* CaretEqualsToken */:
+                    case 77 /* CaretEqualsToken */:
                     case 73 /* BarEqualsToken */:
-                        return 3;
+                    case 74 /* BarBarEqualsToken */:
+                    case 75 /* AmpersandAmpersandEqualsToken */:
+                    case 76 /* QuestionQuestionEqualsToken */:
+                        return 3 /* Assignment */;
                     default:
                         return getBinaryOperatorPrecedence(operatorKind);
                 }
-            case 207 /* PrefixUnaryExpression */:
-            case 204 /* TypeOfExpression */:
-            case 205 /* VoidExpression */:
-            case 203 /* DeleteExpression */:
-            case 206 /* AwaitExpression */:
-                return 16;
-            case 208 /* PostfixUnaryExpression */:
-                return 17;
-            case 196 /* CallExpression */:
-                return 18;
-            case 197 /* NewExpression */:
-                return hasArguments ? 19 : 18;
-            case 198 /* TaggedTemplateExpression */:
-            case 194 /* PropertyAccessExpression */:
-            case 195 /* ElementAccessExpression */:
-                return 19;
-            case 104 /* ThisKeyword */:
-            case 102 /* SuperKeyword */:
-            case 75 /* Identifier */:
-            case 100 /* NullKeyword */:
-            case 106 /* TrueKeyword */:
-            case 91 /* FalseKeyword */:
+            // TODO: Should prefix `++` and `--` be moved to the `Update` precedence?
+            // TODO: We are missing `TypeAssertionExpression`
+            case 214 /* PrefixUnaryExpression */:
+            case 211 /* TypeOfExpression */:
+            case 212 /* VoidExpression */:
+            case 210 /* DeleteExpression */:
+            case 213 /* AwaitExpression */:
+                return 16 /* Unary */;
+            case 215 /* PostfixUnaryExpression */:
+                return 17 /* Update */;
+            case 203 /* CallExpression */:
+                return 18 /* LeftHandSide */;
+            case 204 /* NewExpression */:
+                return hasArguments ? 19 /* Member */ : 18 /* LeftHandSide */;
+            case 205 /* TaggedTemplateExpression */:
+            case 201 /* PropertyAccessExpression */:
+            case 202 /* ElementAccessExpression */:
+                return 19 /* Member */;
+            case 107 /* ThisKeyword */:
+            case 105 /* SuperKeyword */:
+            case 78 /* Identifier */:
+            case 103 /* NullKeyword */:
+            case 109 /* TrueKeyword */:
+            case 94 /* FalseKeyword */:
             case 8 /* NumericLiteral */:
             case 9 /* BigIntLiteral */:
             case 10 /* StringLiteral */:
-            case 192 /* ArrayLiteralExpression */:
-            case 193 /* ObjectLiteralExpression */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
-            case 214 /* ClassExpression */:
-            case 266 /* JsxElement */:
-            case 267 /* JsxSelfClosingElement */:
-            case 270 /* JsxFragment */:
+            case 199 /* ArrayLiteralExpression */:
+            case 200 /* ObjectLiteralExpression */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
+            case 221 /* ClassExpression */:
             case 13 /* RegularExpressionLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
-            case 211 /* TemplateExpression */:
-            case 200 /* ParenthesizedExpression */:
-            case 215 /* OmittedExpression */:
-                return 20;
+            case 218 /* TemplateExpression */:
+            case 207 /* ParenthesizedExpression */:
+            case 222 /* OmittedExpression */:
+            case 273 /* JsxElement */:
+            case 274 /* JsxSelfClosingElement */:
+            case 277 /* JsxFragment */:
+                return 20 /* Primary */;
             default:
-                return -1;
+                return -1 /* Invalid */;
         }
     }
     ts.getOperatorPrecedence = getOperatorPrecedence;
     function getBinaryOperatorPrecedence(kind) {
         switch (kind) {
             case 60 /* QuestionQuestionToken */:
-                return 4;
+                return 4 /* Coalesce */;
             case 56 /* BarBarToken */:
-                return 5;
+                return 5 /* LogicalOR */;
             case 55 /* AmpersandAmpersandToken */:
-                return 6;
+                return 6 /* LogicalAND */;
             case 51 /* BarToken */:
-                return 7;
+                return 7 /* BitwiseOR */;
             case 52 /* CaretToken */:
-                return 8;
+                return 8 /* BitwiseXOR */;
             case 50 /* AmpersandToken */:
-                return 9;
+                return 9 /* BitwiseAND */;
             case 34 /* EqualsEqualsToken */:
             case 35 /* ExclamationEqualsToken */:
             case 36 /* EqualsEqualsEqualsToken */:
             case 37 /* ExclamationEqualsEqualsToken */:
-                return 10;
+                return 10 /* Equality */;
             case 29 /* LessThanToken */:
             case 31 /* GreaterThanToken */:
             case 32 /* LessThanEqualsToken */:
             case 33 /* GreaterThanEqualsToken */:
-            case 98 /* InstanceOfKeyword */:
-            case 97 /* InKeyword */:
-            case 123 /* AsKeyword */:
-                return 11;
+            case 101 /* InstanceOfKeyword */:
+            case 100 /* InKeyword */:
+            case 126 /* AsKeyword */:
+                return 11 /* Relational */;
             case 47 /* LessThanLessThanToken */:
             case 48 /* GreaterThanGreaterThanToken */:
             case 49 /* GreaterThanGreaterThanGreaterThanToken */:
-                return 12;
+                return 12 /* Shift */;
             case 39 /* PlusToken */:
             case 40 /* MinusToken */:
-                return 13;
+                return 13 /* Additive */;
             case 41 /* AsteriskToken */:
             case 43 /* SlashToken */:
             case 44 /* PercentToken */:
-                return 14;
+                return 14 /* Multiplicative */;
             case 42 /* AsteriskAsteriskToken */:
-                return 15;
+                return 15 /* Exponentiation */;
         }
         // -1 is lower than all other precedences.  Returning it will cause binary expression
         // parsing to stop.
         return -1;
     }
     ts.getBinaryOperatorPrecedence = getBinaryOperatorPrecedence;
+    function getSemanticJsxChildren(children) {
+        return ts.filter(children, function (i) {
+            switch (i.kind) {
+                case 283 /* JsxExpression */:
+                    return !!i.expression;
+                case 11 /* JsxText */:
+                    return !i.containsOnlyTriviaWhiteSpaces;
+                default:
+                    return true;
+            }
+        });
+    }
+    ts.getSemanticJsxChildren = getSemanticJsxChildren;
     function createDiagnosticCollection() {
         var nonFileDiagnostics = []; // See GH#19873
         var filesWithDiagnostics = [];
-        var fileDiagnostics = ts.createMap();
+        var fileDiagnostics = new ts.Map();
         var hasReadNonFileDiagnostics = false;
         return {
             add: add,
@@ -16219,7 +16844,7 @@ var ts;
     var singleQuoteEscapedCharsRegExp = /[\\\'\u0000-\u001f\t\v\f\b\r\n\u2028\u2029\u0085]/g;
     // Template strings should be preserved as much as possible
     var backtickQuoteEscapedCharsRegExp = /[\\`]/g;
-    var escapedCharsMap = ts.createMapFromTemplate({
+    var escapedCharsMap = new ts.Map(ts.getEntries({
         "\t": "\\t",
         "\v": "\\v",
         "\f": "\\f",
@@ -16233,7 +16858,7 @@ var ts;
         "\u2028": "\\u2028",
         "\u2029": "\\u2029",
         "\u0085": "\\u0085" // nextLine
-    });
+    }));
     function encodeUtf16EscapeSequence(charCode) {
         var hexCharCode = charCode.toString(16).toUpperCase();
         var paddedHexCode = ("0000" + hexCharCode).slice(-4);
@@ -16279,10 +16904,10 @@ var ts;
     // the map below must be updated.
     var jsxDoubleQuoteEscapedCharsRegExp = /[\"\u0000-\u001f\u2028\u2029\u0085]/g;
     var jsxSingleQuoteEscapedCharsRegExp = /[\'\u0000-\u001f\u2028\u2029\u0085]/g;
-    var jsxEscapedCharsMap = ts.createMapFromTemplate({
+    var jsxEscapedCharsMap = new ts.Map(ts.getEntries({
         "\"": "&quot;",
         "\'": "&apos;"
-    });
+    }));
     function encodeJsxCharacterEntity(charCode) {
         var hexCharCode = charCode.toString(16).toUpperCase();
         return "&#x" + hexCharCode + ";";
@@ -16324,8 +16949,10 @@ var ts;
     ts.isIntrinsicJsxName = isIntrinsicJsxName;
     var indentStrings = ["", "    "];
     function getIndentString(level) {
-        if (indentStrings[level] === undefined) {
-            indentStrings[level] = getIndentString(level - 1) + indentStrings[1];
+        // prepopulate cache
+        var singleLevel = indentStrings[1];
+        for (var current = indentStrings.length; current <= level; current++) {
+            indentStrings.push(indentStrings[current - 1] + singleLevel);
         }
         return indentStrings[level];
     }
@@ -16560,6 +17187,18 @@ var ts;
         return removeFileExtension(path) + ".d.ts" /* Dts */;
     }
     ts.getDeclarationEmitOutputFilePathWorker = getDeclarationEmitOutputFilePathWorker;
+    function outFile(options) {
+        return options.outFile || options.out;
+    }
+    ts.outFile = outFile;
+    /** Returns 'undefined' if and only if 'options.paths' is undefined. */
+    function getPathsBasePath(options, host) {
+        var _a, _b;
+        if (!options.paths)
+            return undefined;
+        return (_a = options.baseUrl) !== null && _a !== void 0 ? _a : ts.Debug.checkDefined(options.pathsBasePath || ((_b = host.getCurrentDirectory) === null || _b === void 0 ? void 0 : _b.call(host)), "Encountered 'paths' without a 'baseUrl', config file, or host 'getCurrentDirectory'.");
+    }
+    ts.getPathsBasePath = getPathsBasePath;
     /**
      * Gets the source files that are expected to have an emit output.
      *
@@ -16571,7 +17210,7 @@ var ts;
      */
     function getSourceFilesToEmit(host, targetSourceFile, forceDtsEmit) {
         var options = host.getCompilerOptions();
-        if (options.outFile || options.out) {
+        if (outFile(options)) {
             var moduleKind = getEmitModuleKind(options);
             var moduleEmitEnabled_1 = options.emitDeclarationOnly || moduleKind === ts.ModuleKind.AMD || moduleKind === ts.ModuleKind.System;
             // Can emit only sources that are not declaration file and are either non module code or module with --module or --target es6 specified
@@ -16673,11 +17312,11 @@ var ts;
     }
     ts.parameterIsThisKeyword = parameterIsThisKeyword;
     function isThisIdentifier(node) {
-        return !!node && node.kind === 75 /* Identifier */ && identifierIsThisKeyword(node);
+        return !!node && node.kind === 78 /* Identifier */ && identifierIsThisKeyword(node);
     }
     ts.isThisIdentifier = isThisIdentifier;
     function identifierIsThisKeyword(id) {
-        return id.originalKeywordKind === 104 /* ThisKeyword */;
+        return id.originalKeywordKind === 107 /* ThisKeyword */;
     }
     ts.identifierIsThisKeyword = identifierIsThisKeyword;
     function getAllAccessorDeclarations(declarations, accessor) {
@@ -16688,10 +17327,10 @@ var ts;
         var setAccessor;
         if (hasDynamicName(accessor)) {
             firstAccessor = accessor;
-            if (accessor.kind === 163 /* GetAccessor */) {
+            if (accessor.kind === 167 /* GetAccessor */) {
                 getAccessor = accessor;
             }
-            else if (accessor.kind === 164 /* SetAccessor */) {
+            else if (accessor.kind === 168 /* SetAccessor */) {
                 setAccessor = accessor;
             }
             else {
@@ -16701,7 +17340,7 @@ var ts;
         else {
             ts.forEach(declarations, function (member) {
                 if (ts.isAccessor(member)
-                    && hasModifier(member, 32 /* Static */) === hasModifier(accessor, 32 /* Static */)) {
+                    && hasSyntacticModifier(member, 32 /* Static */) === hasSyntacticModifier(accessor, 32 /* Static */)) {
                     var memberName = getPropertyNameForPropertyNameNode(member.name);
                     var accessorName = getPropertyNameForPropertyNameNode(accessor.name);
                     if (memberName === accessorName) {
@@ -16711,12 +17350,10 @@ var ts;
                         else if (!secondAccessor) {
                             secondAccessor = member;
                         }
-                        if (member.kind === 163 /* GetAccessor */ && !getAccessor) {
-                            // eslint-disable-next-line
+                        if (member.kind === 167 /* GetAccessor */ && !getAccessor) {
                             getAccessor = member;
                         }
-                        if (member.kind === 164 /* SetAccessor */ && !setAccessor) {
-                            // eslint-disable-next-line
+                        if (member.kind === 168 /* SetAccessor */ && !setAccessor) {
                             setAccessor = member;
                         }
                     }
@@ -16765,7 +17402,7 @@ var ts;
     ts.getJSDocTypeParameterDeclarations = getJSDocTypeParameterDeclarations;
     /** template tags are only available when a typedef isn't already using them */
     function isNonTypeAliasTemplate(tag) {
-        return ts.isJSDocTemplateTag(tag) && !(tag.parent.kind === 303 /* JSDocComment */ && tag.parent.tags.some(isJSDocTypeAlias));
+        return ts.isJSDocTemplateTag(tag) && !(tag.parent.kind === 311 /* JSDocComment */ && tag.parent.tags.some(isJSDocTypeAlias));
     }
     /**
      * Gets the effective type annotation of the value parameter of a set accessor. If the node
@@ -16968,74 +17605,136 @@ var ts;
         }
         return currentLineIndent;
     }
-    function hasModifiers(node) {
-        return getModifierFlags(node) !== 0 /* None */;
+    function hasEffectiveModifiers(node) {
+        return getEffectiveModifierFlags(node) !== 0 /* None */;
     }
-    ts.hasModifiers = hasModifiers;
-    function hasModifier(node, flags) {
-        return !!getSelectedModifierFlags(node, flags);
+    ts.hasEffectiveModifiers = hasEffectiveModifiers;
+    function hasSyntacticModifiers(node) {
+        return getSyntacticModifierFlags(node) !== 0 /* None */;
     }
-    ts.hasModifier = hasModifier;
+    ts.hasSyntacticModifiers = hasSyntacticModifiers;
+    function hasEffectiveModifier(node, flags) {
+        return !!getSelectedEffectiveModifierFlags(node, flags);
+    }
+    ts.hasEffectiveModifier = hasEffectiveModifier;
+    function hasSyntacticModifier(node, flags) {
+        return !!getSelectedSyntacticModifierFlags(node, flags);
+    }
+    ts.hasSyntacticModifier = hasSyntacticModifier;
     function hasStaticModifier(node) {
-        return hasModifier(node, 32 /* Static */);
+        return hasSyntacticModifier(node, 32 /* Static */);
     }
     ts.hasStaticModifier = hasStaticModifier;
-    function hasReadonlyModifier(node) {
-        return hasModifier(node, 64 /* Readonly */);
+    function hasEffectiveReadonlyModifier(node) {
+        return hasEffectiveModifier(node, 64 /* Readonly */);
     }
-    ts.hasReadonlyModifier = hasReadonlyModifier;
-    function getSelectedModifierFlags(node, flags) {
-        return getModifierFlags(node) & flags;
+    ts.hasEffectiveReadonlyModifier = hasEffectiveReadonlyModifier;
+    function getSelectedEffectiveModifierFlags(node, flags) {
+        return getEffectiveModifierFlags(node) & flags;
     }
-    ts.getSelectedModifierFlags = getSelectedModifierFlags;
-    function getModifierFlags(node) {
-        if (node.kind >= 0 /* FirstToken */ && node.kind <= 152 /* LastToken */) {
+    ts.getSelectedEffectiveModifierFlags = getSelectedEffectiveModifierFlags;
+    function getSelectedSyntacticModifierFlags(node, flags) {
+        return getSyntacticModifierFlags(node) & flags;
+    }
+    ts.getSelectedSyntacticModifierFlags = getSelectedSyntacticModifierFlags;
+    function getModifierFlagsWorker(node, includeJSDoc, alwaysIncludeJSDoc) {
+        if (node.kind >= 0 /* FirstToken */ && node.kind <= 156 /* LastToken */) {
             return 0 /* None */;
         }
-        if (node.modifierFlagsCache & 536870912 /* HasComputedFlags */) {
-            return node.modifierFlagsCache & ~536870912 /* HasComputedFlags */;
+        if (!(node.modifierFlagsCache & 536870912 /* HasComputedFlags */)) {
+            node.modifierFlagsCache = getSyntacticModifierFlagsNoCache(node) | 536870912 /* HasComputedFlags */;
+        }
+        if (includeJSDoc && !(node.modifierFlagsCache & 4096 /* HasComputedJSDocModifiers */) && (alwaysIncludeJSDoc || isInJSFile(node)) && node.parent) {
+            node.modifierFlagsCache |= getJSDocModifierFlagsNoCache(node) | 4096 /* HasComputedJSDocModifiers */;
+        }
+        return node.modifierFlagsCache & ~(536870912 /* HasComputedFlags */ | 4096 /* HasComputedJSDocModifiers */);
+    }
+    /**
+     * Gets the effective ModifierFlags for the provided node, including JSDoc modifiers. The modifiers will be cached on the node to improve performance.
+     *
+     * NOTE: This function may use `parent` pointers.
+     */
+    function getEffectiveModifierFlags(node) {
+        return getModifierFlagsWorker(node, /*includeJSDoc*/ true);
+    }
+    ts.getEffectiveModifierFlags = getEffectiveModifierFlags;
+    function getEffectiveModifierFlagsAlwaysIncludeJSDoc(node) {
+        return getModifierFlagsWorker(node, /*includeJSDOc*/ true, /*alwaysIncludeJSDOc*/ true);
+    }
+    ts.getEffectiveModifierFlagsAlwaysIncludeJSDoc = getEffectiveModifierFlagsAlwaysIncludeJSDoc;
+    /**
+     * Gets the ModifierFlags for syntactic modifiers on the provided node. The modifiers will be cached on the node to improve performance.
+     *
+     * NOTE: This function does not use `parent` pointers and will not include modifiers from JSDoc.
+     */
+    function getSyntacticModifierFlags(node) {
+        return getModifierFlagsWorker(node, /*includeJSDoc*/ false);
+    }
+    ts.getSyntacticModifierFlags = getSyntacticModifierFlags;
+    function getJSDocModifierFlagsNoCache(node) {
+        var flags = 0 /* None */;
+        if (!!node.parent && !ts.isParameter(node)) {
+            if (isInJSFile(node)) {
+                if (ts.getJSDocPublicTagNoCache(node))
+                    flags |= 4 /* Public */;
+                if (ts.getJSDocPrivateTagNoCache(node))
+                    flags |= 8 /* Private */;
+                if (ts.getJSDocProtectedTagNoCache(node))
+                    flags |= 16 /* Protected */;
+                if (ts.getJSDocReadonlyTagNoCache(node))
+                    flags |= 64 /* Readonly */;
+            }
+            if (ts.getJSDocDeprecatedTagNoCache(node))
+                flags |= 8192 /* Deprecated */;
+        }
+        return flags;
+    }
+    /**
+     * Gets the effective ModifierFlags for the provided node, including JSDoc modifiers. The modifier flags cache on the node is ignored.
+     *
+     * NOTE: This function may use `parent` pointers.
+     */
+    function getEffectiveModifierFlagsNoCache(node) {
+        return getSyntacticModifierFlagsNoCache(node) | getJSDocModifierFlagsNoCache(node);
+    }
+    ts.getEffectiveModifierFlagsNoCache = getEffectiveModifierFlagsNoCache;
+    /**
+     * Gets the ModifierFlags for syntactic modifiers on the provided node. The modifier flags cache on the node is ignored.
+     *
+     * NOTE: This function does not use `parent` pointers and will not include modifiers from JSDoc.
+     */
+    function getSyntacticModifierFlagsNoCache(node) {
+        var flags = modifiersToFlags(node.modifiers);
+        if (node.flags & 4 /* NestedNamespace */ || (node.kind === 78 /* Identifier */ && node.isInJSDocNamespace)) {
+            flags |= 1 /* Export */;
         }
-        var flags = getModifierFlagsNoCache(node);
-        node.modifierFlagsCache = flags | 536870912 /* HasComputedFlags */;
         return flags;
     }
-    ts.getModifierFlags = getModifierFlags;
-    function getModifierFlagsNoCache(node) {
+    ts.getSyntacticModifierFlagsNoCache = getSyntacticModifierFlagsNoCache;
+    function modifiersToFlags(modifiers) {
         var flags = 0 /* None */;
-        if (node.modifiers) {
-            for (var _i = 0, _a = node.modifiers; _i < _a.length; _i++) {
-                var modifier = _a[_i];
+        if (modifiers) {
+            for (var _i = 0, modifiers_1 = modifiers; _i < modifiers_1.length; _i++) {
+                var modifier = modifiers_1[_i];
                 flags |= modifierToFlag(modifier.kind);
             }
         }
-        if (isInJSFile(node) && !!node.parent) {
-            // getModifierFlagsNoCache should only be called when parent pointers are set,
-            // or when !(node.flags & NodeFlags.Synthesized) && node.kind !== SyntaxKind.SourceFile)
-            var tags = (ts.getJSDocPublicTag(node) ? 4 /* Public */ : 0 /* None */)
-                | (ts.getJSDocPrivateTag(node) ? 8 /* Private */ : 0 /* None */)
-                | (ts.getJSDocProtectedTag(node) ? 16 /* Protected */ : 0 /* None */)
-                | (ts.getJSDocReadonlyTag(node) ? 64 /* Readonly */ : 0 /* None */);
-            flags |= tags;
-        }
-        if (node.flags & 4 /* NestedNamespace */ || (node.kind === 75 /* Identifier */ && node.isInJSDocNamespace)) {
-            flags |= 1 /* Export */;
-        }
         return flags;
     }
-    ts.getModifierFlagsNoCache = getModifierFlagsNoCache;
+    ts.modifiersToFlags = modifiersToFlags;
     function modifierToFlag(token) {
         switch (token) {
-            case 120 /* StaticKeyword */: return 32 /* Static */;
-            case 119 /* PublicKeyword */: return 4 /* Public */;
-            case 118 /* ProtectedKeyword */: return 16 /* Protected */;
-            case 117 /* PrivateKeyword */: return 8 /* Private */;
-            case 122 /* AbstractKeyword */: return 128 /* Abstract */;
-            case 89 /* ExportKeyword */: return 1 /* Export */;
-            case 130 /* DeclareKeyword */: return 2 /* Ambient */;
-            case 81 /* ConstKeyword */: return 2048 /* Const */;
-            case 84 /* DefaultKeyword */: return 512 /* Default */;
-            case 126 /* AsyncKeyword */: return 256 /* Async */;
-            case 138 /* ReadonlyKeyword */: return 64 /* Readonly */;
+            case 123 /* StaticKeyword */: return 32 /* Static */;
+            case 122 /* PublicKeyword */: return 4 /* Public */;
+            case 121 /* ProtectedKeyword */: return 16 /* Protected */;
+            case 120 /* PrivateKeyword */: return 8 /* Private */;
+            case 125 /* AbstractKeyword */: return 128 /* Abstract */;
+            case 92 /* ExportKeyword */: return 1 /* Export */;
+            case 133 /* DeclareKeyword */: return 2 /* Ambient */;
+            case 84 /* ConstKeyword */: return 2048 /* Const */;
+            case 87 /* DefaultKeyword */: return 512 /* Default */;
+            case 129 /* AsyncKeyword */: return 256 /* Async */;
+            case 142 /* ReadonlyKeyword */: return 64 /* Readonly */;
         }
         return 0 /* None */;
     }
@@ -17046,8 +17745,18 @@ var ts;
             || token === 53 /* ExclamationToken */;
     }
     ts.isLogicalOperator = isLogicalOperator;
+    function isLogicalOrCoalescingAssignmentOperator(token) {
+        return token === 74 /* BarBarEqualsToken */
+            || token === 75 /* AmpersandAmpersandEqualsToken */
+            || token === 76 /* QuestionQuestionEqualsToken */;
+    }
+    ts.isLogicalOrCoalescingAssignmentOperator = isLogicalOrCoalescingAssignmentOperator;
+    function isLogicalOrCoalescingAssignmentExpression(expr) {
+        return isLogicalOrCoalescingAssignmentOperator(expr.operatorToken.kind);
+    }
+    ts.isLogicalOrCoalescingAssignmentExpression = isLogicalOrCoalescingAssignmentExpression;
     function isAssignmentOperator(token) {
-        return token >= 62 /* FirstAssignment */ && token <= 74 /* LastAssignment */;
+        return token >= 62 /* FirstAssignment */ && token <= 77 /* LastAssignment */;
     }
     ts.isAssignmentOperator = isAssignmentOperator;
     /** Get `C` given `N` if `N` is in the position `class C extends N` where `N` is an ExpressionWithTypeArguments. */
@@ -17060,7 +17769,7 @@ var ts;
         return ts.isExpressionWithTypeArguments(node)
             && ts.isHeritageClause(node.parent)
             && ts.isClassLike(node.parent.parent)
-            ? { class: node.parent.parent, isImplements: node.parent.token === 113 /* ImplementsKeyword */ }
+            ? { class: node.parent.parent, isImplements: node.parent.token === 116 /* ImplementsKeyword */ }
             : undefined;
     }
     ts.tryGetClassImplementingOrExtendingExpressionWithTypeArguments = tryGetClassImplementingOrExtendingExpressionWithTypeArguments;
@@ -17075,8 +17784,8 @@ var ts;
     function isDestructuringAssignment(node) {
         if (isAssignmentExpression(node, /*excludeCompoundAssignment*/ true)) {
             var kind = node.left.kind;
-            return kind === 193 /* ObjectLiteralExpression */
-                || kind === 192 /* ArrayLiteralExpression */;
+            return kind === 200 /* ObjectLiteralExpression */
+                || kind === 199 /* ArrayLiteralExpression */;
         }
         return false;
     }
@@ -17086,30 +17795,30 @@ var ts;
     }
     ts.isExpressionWithTypeArgumentsInClassExtendsClause = isExpressionWithTypeArgumentsInClassExtendsClause;
     function isEntityNameExpression(node) {
-        return node.kind === 75 /* Identifier */ || isPropertyAccessEntityNameExpression(node);
+        return node.kind === 78 /* Identifier */ || isPropertyAccessEntityNameExpression(node);
     }
     ts.isEntityNameExpression = isEntityNameExpression;
     function getFirstIdentifier(node) {
         switch (node.kind) {
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 return node;
-            case 153 /* QualifiedName */:
+            case 157 /* QualifiedName */:
                 do {
                     node = node.left;
-                } while (node.kind !== 75 /* Identifier */);
+                } while (node.kind !== 78 /* Identifier */);
                 return node;
-            case 194 /* PropertyAccessExpression */:
+            case 201 /* PropertyAccessExpression */:
                 do {
                     node = node.expression;
-                } while (node.kind !== 75 /* Identifier */);
+                } while (node.kind !== 78 /* Identifier */);
                 return node;
         }
     }
     ts.getFirstIdentifier = getFirstIdentifier;
     function isDottedName(node) {
-        return node.kind === 75 /* Identifier */ || node.kind === 104 /* ThisKeyword */ || node.kind === 102 /* SuperKeyword */ ||
-            node.kind === 194 /* PropertyAccessExpression */ && isDottedName(node.expression) ||
-            node.kind === 200 /* ParenthesizedExpression */ && isDottedName(node.expression);
+        return node.kind === 78 /* Identifier */ || node.kind === 107 /* ThisKeyword */ || node.kind === 105 /* SuperKeyword */ ||
+            node.kind === 201 /* PropertyAccessExpression */ && isDottedName(node.expression) ||
+            node.kind === 207 /* ParenthesizedExpression */ && isDottedName(node.expression);
     }
     ts.isDottedName = isDottedName;
     function isPropertyAccessEntityNameExpression(node) {
@@ -17120,7 +17829,7 @@ var ts;
         if (ts.isPropertyAccessExpression(expr)) {
             var baseStr = tryGetPropertyAccessOrIdentifierToString(expr.expression);
             if (baseStr !== undefined) {
-                return baseStr + "." + expr.name;
+                return baseStr + "." + entityNameToString(expr.name);
             }
         }
         else if (ts.isIdentifier(expr)) {
@@ -17134,26 +17843,33 @@ var ts;
     }
     ts.isPrototypeAccess = isPrototypeAccess;
     function isRightSideOfQualifiedNameOrPropertyAccess(node) {
-        return (node.parent.kind === 153 /* QualifiedName */ && node.parent.right === node) ||
-            (node.parent.kind === 194 /* PropertyAccessExpression */ && node.parent.name === node);
+        return (node.parent.kind === 157 /* QualifiedName */ && node.parent.right === node) ||
+            (node.parent.kind === 201 /* PropertyAccessExpression */ && node.parent.name === node);
     }
     ts.isRightSideOfQualifiedNameOrPropertyAccess = isRightSideOfQualifiedNameOrPropertyAccess;
     function isEmptyObjectLiteral(expression) {
-        return expression.kind === 193 /* ObjectLiteralExpression */ &&
+        return expression.kind === 200 /* ObjectLiteralExpression */ &&
             expression.properties.length === 0;
     }
     ts.isEmptyObjectLiteral = isEmptyObjectLiteral;
     function isEmptyArrayLiteral(expression) {
-        return expression.kind === 192 /* ArrayLiteralExpression */ &&
+        return expression.kind === 199 /* ArrayLiteralExpression */ &&
             expression.elements.length === 0;
     }
     ts.isEmptyArrayLiteral = isEmptyArrayLiteral;
     function getLocalSymbolForExportDefault(symbol) {
-        return isExportDefaultSymbol(symbol) ? symbol.declarations[0].localSymbol : undefined;
+        if (!isExportDefaultSymbol(symbol))
+            return undefined;
+        for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+            var decl = _a[_i];
+            if (decl.localSymbol)
+                return decl.localSymbol;
+        }
+        return undefined;
     }
     ts.getLocalSymbolForExportDefault = getLocalSymbolForExportDefault;
     function isExportDefaultSymbol(symbol) {
-        return symbol && ts.length(symbol.declarations) > 0 && hasModifier(symbol.declarations[0], 512 /* Default */);
+        return symbol && ts.length(symbol.declarations) > 0 && hasSyntacticModifier(symbol.declarations[0], 512 /* Default */);
     }
     /** Return ".ts", ".d.ts", or ".tsx", if that is the extension. */
     function tryExtractTSExtension(fileName) {
@@ -17469,8 +18185,8 @@ var ts;
         var parseNode = ts.getParseTreeNode(node);
         if (parseNode) {
             switch (parseNode.parent.kind) {
-                case 248 /* EnumDeclaration */:
-                case 249 /* ModuleDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     return parseNode === parseNode.parent.name;
             }
         }
@@ -17547,35 +18263,35 @@ var ts;
         if (!parent)
             return 0 /* Read */;
         switch (parent.kind) {
-            case 200 /* ParenthesizedExpression */:
+            case 207 /* ParenthesizedExpression */:
                 return accessKind(parent);
-            case 208 /* PostfixUnaryExpression */:
-            case 207 /* PrefixUnaryExpression */:
+            case 215 /* PostfixUnaryExpression */:
+            case 214 /* PrefixUnaryExpression */:
                 var operator = parent.operator;
                 return operator === 45 /* PlusPlusToken */ || operator === 46 /* MinusMinusToken */ ? writeOrReadWrite() : 0 /* Read */;
-            case 209 /* BinaryExpression */:
+            case 216 /* BinaryExpression */:
                 var _a = parent, left = _a.left, operatorToken = _a.operatorToken;
                 return left === node && isAssignmentOperator(operatorToken.kind) ?
                     operatorToken.kind === 62 /* EqualsToken */ ? 1 /* Write */ : writeOrReadWrite()
                     : 0 /* Read */;
-            case 194 /* PropertyAccessExpression */:
+            case 201 /* PropertyAccessExpression */:
                 return parent.name !== node ? 0 /* Read */ : accessKind(parent);
-            case 281 /* PropertyAssignment */: {
+            case 288 /* PropertyAssignment */: {
                 var parentAccess = accessKind(parent.parent);
                 // In `({ x: varname }) = { x: 1 }`, the left `x` is a read, the right `x` is a write.
                 return node === parent.name ? reverseAccessKind(parentAccess) : parentAccess;
             }
-            case 282 /* ShorthandPropertyAssignment */:
+            case 289 /* ShorthandPropertyAssignment */:
                 // Assume it's the local variable being accessed, since we don't check public properties for --noUnusedLocals.
                 return node === parent.objectAssignmentInitializer ? 0 /* Read */ : accessKind(parent.parent);
-            case 192 /* ArrayLiteralExpression */:
+            case 199 /* ArrayLiteralExpression */:
                 return accessKind(parent);
             default:
                 return 0 /* Read */;
         }
         function writeOrReadWrite() {
             // If grandparent is not an ExpressionStatement, this is used as an expression in addition to having a side effect.
-            return parent.parent && skipParenthesesUp(parent.parent).kind === 226 /* ExpressionStatement */ ? 1 /* Write */ : 2 /* ReadWrite */;
+            return parent.parent && skipParenthesesUp(parent.parent).kind === 233 /* ExpressionStatement */ ? 1 /* Write */ : 2 /* ReadWrite */;
         }
     }
     function reverseAccessKind(a) {
@@ -17662,7 +18378,7 @@ var ts;
     function isAbstractConstructorSymbol(symbol) {
         if (symbol.flags & 32 /* Class */) {
             var declaration = getClassLikeDeclarationOfSymbol(symbol);
-            return !!declaration && hasModifier(declaration, 128 /* Abstract */);
+            return !!declaration && hasSyntacticModifier(declaration, 128 /* Abstract */);
         }
         return false;
     }
@@ -17724,39 +18440,37 @@ var ts;
     }
     ts.isObjectTypeDeclaration = isObjectTypeDeclaration;
     function isTypeNodeKind(kind) {
-        return (kind >= 168 /* FirstTypeNode */ && kind <= 188 /* LastTypeNode */)
-            || kind === 125 /* AnyKeyword */
-            || kind === 148 /* UnknownKeyword */
-            || kind === 140 /* NumberKeyword */
-            || kind === 151 /* BigIntKeyword */
-            || kind === 141 /* ObjectKeyword */
-            || kind === 128 /* BooleanKeyword */
-            || kind === 143 /* StringKeyword */
-            || kind === 144 /* SymbolKeyword */
-            || kind === 104 /* ThisKeyword */
-            || kind === 110 /* VoidKeyword */
-            || kind === 146 /* UndefinedKeyword */
-            || kind === 100 /* NullKeyword */
-            || kind === 137 /* NeverKeyword */
-            || kind === 216 /* ExpressionWithTypeArguments */
-            || kind === 295 /* JSDocAllType */
-            || kind === 296 /* JSDocUnknownType */
-            || kind === 297 /* JSDocNullableType */
-            || kind === 298 /* JSDocNonNullableType */
-            || kind === 299 /* JSDocOptionalType */
-            || kind === 300 /* JSDocFunctionType */
-            || kind === 301 /* JSDocVariadicType */;
+        return (kind >= 172 /* FirstTypeNode */ && kind <= 195 /* LastTypeNode */)
+            || kind === 128 /* AnyKeyword */
+            || kind === 152 /* UnknownKeyword */
+            || kind === 144 /* NumberKeyword */
+            || kind === 155 /* BigIntKeyword */
+            || kind === 145 /* ObjectKeyword */
+            || kind === 131 /* BooleanKeyword */
+            || kind === 147 /* StringKeyword */
+            || kind === 148 /* SymbolKeyword */
+            || kind === 113 /* VoidKeyword */
+            || kind === 150 /* UndefinedKeyword */
+            || kind === 141 /* NeverKeyword */
+            || kind === 223 /* ExpressionWithTypeArguments */
+            || kind === 303 /* JSDocAllType */
+            || kind === 304 /* JSDocUnknownType */
+            || kind === 305 /* JSDocNullableType */
+            || kind === 306 /* JSDocNonNullableType */
+            || kind === 307 /* JSDocOptionalType */
+            || kind === 308 /* JSDocFunctionType */
+            || kind === 309 /* JSDocVariadicType */;
     }
     ts.isTypeNodeKind = isTypeNodeKind;
     function isAccessExpression(node) {
-        return node.kind === 194 /* PropertyAccessExpression */ || node.kind === 195 /* ElementAccessExpression */;
+        return node.kind === 201 /* PropertyAccessExpression */ || node.kind === 202 /* ElementAccessExpression */;
     }
     ts.isAccessExpression = isAccessExpression;
     function getNameOfAccessExpression(node) {
-        if (node.kind === 194 /* PropertyAccessExpression */) {
+        if (node.kind === 201 /* PropertyAccessExpression */) {
             return node.name;
         }
-        ts.Debug.assert(node.kind === 195 /* ElementAccessExpression */);
+        ts.Debug.assert(node.kind === 202 /* ElementAccessExpression */);
         return node.argumentExpression;
     }
     ts.getNameOfAccessExpression = getNameOfAccessExpression;
@@ -17771,9 +18485,48 @@ var ts;
     }
     ts.isBundleFileTextLike = isBundleFileTextLike;
     function isNamedImportsOrExports(node) {
-        return node.kind === 257 /* NamedImports */ || node.kind === 261 /* NamedExports */;
+        return node.kind === 264 /* NamedImports */ || node.kind === 268 /* NamedExports */;
     }
     ts.isNamedImportsOrExports = isNamedImportsOrExports;
+    function getLeftmostAccessExpression(expr) {
+        while (isAccessExpression(expr)) {
+            expr = expr.expression;
+        }
+        return expr;
+    }
+    ts.getLeftmostAccessExpression = getLeftmostAccessExpression;
+    function getLeftmostExpression(node, stopAtCallExpressions) {
+        while (true) {
+            switch (node.kind) {
+                case 215 /* PostfixUnaryExpression */:
+                    node = node.operand;
+                    continue;
+                case 216 /* BinaryExpression */:
+                    node = node.left;
+                    continue;
+                case 217 /* ConditionalExpression */:
+                    node = node.condition;
+                    continue;
+                case 205 /* TaggedTemplateExpression */:
+                    node = node.tag;
+                    continue;
+                case 203 /* CallExpression */:
+                    if (stopAtCallExpressions) {
+                        return node;
+                    }
+                // falls through
+                case 224 /* AsExpression */:
+                case 202 /* ElementAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
+                case 225 /* NonNullExpression */:
+                case 336 /* PartiallyEmittedExpression */:
+                    node = node.expression;
+                    continue;
+            }
+            return node;
+        }
+    }
+    ts.getLeftmostExpression = getLeftmostExpression;
     function Symbol(flags, name) {
         this.flags = flags;
         this.escapedName = name;
@@ -17785,7 +18538,7 @@ var ts;
     }
     function Type(checker, flags) {
         this.flags = flags;
-        if (ts.Debug.isDebugging) {
+        if (ts.Debug.isDebugging || ts.tracing.isTracing()) {
             this.checker = checker;
         }
     }
@@ -17861,6 +18614,71 @@ var ts;
         return ts.localizedDiagnosticMessages && ts.localizedDiagnosticMessages[message.key] || message.message;
     }
     ts.getLocaleSpecificMessage = getLocaleSpecificMessage;
+    function createDetachedDiagnostic(fileName, start, length, message) {
+        ts.Debug.assertGreaterThanOrEqual(start, 0);
+        ts.Debug.assertGreaterThanOrEqual(length, 0);
+        var text = getLocaleSpecificMessage(message);
+        if (arguments.length > 4) {
+            text = formatStringFromArgs(text, arguments, 4);
+        }
+        return {
+            file: undefined,
+            start: start,
+            length: length,
+            messageText: text,
+            category: message.category,
+            code: message.code,
+            reportsUnnecessary: message.reportsUnnecessary,
+            fileName: fileName,
+        };
+    }
+    ts.createDetachedDiagnostic = createDetachedDiagnostic;
+    function isDiagnosticWithDetachedLocation(diagnostic) {
+        return diagnostic.file === undefined
+            && diagnostic.start !== undefined
+            && diagnostic.length !== undefined
+            && typeof diagnostic.fileName === "string";
+    }
+    function attachFileToDiagnostic(diagnostic, file) {
+        var fileName = file.fileName || "";
+        var length = file.text.length;
+        ts.Debug.assertEqual(diagnostic.fileName, fileName);
+        ts.Debug.assertLessThanOrEqual(diagnostic.start, length);
+        ts.Debug.assertLessThanOrEqual(diagnostic.start + diagnostic.length, length);
+        var diagnosticWithLocation = {
+            file: file,
+            start: diagnostic.start,
+            length: diagnostic.length,
+            messageText: diagnostic.messageText,
+            category: diagnostic.category,
+            code: diagnostic.code,
+            reportsUnnecessary: diagnostic.reportsUnnecessary
+        };
+        if (diagnostic.relatedInformation) {
+            diagnosticWithLocation.relatedInformation = [];
+            for (var _i = 0, _a = diagnostic.relatedInformation; _i < _a.length; _i++) {
+                var related = _a[_i];
+                if (isDiagnosticWithDetachedLocation(related) && related.fileName === fileName) {
+                    ts.Debug.assertLessThanOrEqual(related.start, length);
+                    ts.Debug.assertLessThanOrEqual(related.start + related.length, length);
+                    diagnosticWithLocation.relatedInformation.push(attachFileToDiagnostic(related, file));
+                }
+                else {
+                    diagnosticWithLocation.relatedInformation.push(related);
+                }
+            }
+        }
+        return diagnosticWithLocation;
+    }
+    function attachFileToDiagnostics(diagnostics, file) {
+        var diagnosticsWithLocation = [];
+        for (var _i = 0, diagnostics_1 = diagnostics; _i < diagnostics_1.length; _i++) {
+            var diagnostic = diagnostics_1[_i];
+            diagnosticsWithLocation.push(attachFileToDiagnostic(diagnostic, file));
+        }
+        return diagnosticsWithLocation;
+    }
+    ts.attachFileToDiagnostics = attachFileToDiagnostics;
     function createFileDiagnostic(file, start, length, message) {
         ts.Debug.assertGreaterThanOrEqual(start, 0);
         ts.Debug.assertGreaterThanOrEqual(length, 0);
@@ -17880,6 +18698,7 @@ var ts;
             category: message.category,
             code: message.code,
             reportsUnnecessary: message.reportsUnnecessary,
+            reportsDeprecated: message.reportsDeprecated
         };
     }
     ts.createFileDiagnostic = createFileDiagnostic;
@@ -17904,6 +18723,7 @@ var ts;
             category: message.category,
             code: message.code,
             reportsUnnecessary: message.reportsUnnecessary,
+            reportsDeprecated: message.reportsDeprecated
         };
     }
     ts.createCompilerDiagnostic = createCompilerDiagnostic;
@@ -18007,6 +18827,11 @@ var ts;
         }
         return 0 /* EqualTo */;
     }
+    function getLanguageVariant(scriptKind) {
+        // .tsx and .jsx files are treated as jsx language variant.
+        return scriptKind === 4 /* TSX */ || scriptKind === 2 /* JSX */ || scriptKind === 1 /* JS */ || scriptKind === 6 /* JSON */ ? 1 /* JSX */ : 0 /* Standard */;
+    }
+    ts.getLanguageVariant = getLanguageVariant;
     function getEmitScriptTarget(compilerOptions) {
         return compilerOptions.target || 0 /* ES3 */;
     }
@@ -18070,6 +18895,10 @@ var ts;
         return compilerOptions[flag] === undefined ? !!compilerOptions.strict : !!compilerOptions[flag];
     }
     ts.getStrictOptionValue = getStrictOptionValue;
+    function getAllowJSCompilerOption(compilerOptions) {
+        return compilerOptions.allowJs === undefined ? !!compilerOptions.checkJs : compilerOptions.allowJs;
+    }
+    ts.getAllowJSCompilerOption = getAllowJSCompilerOption;
     function compilerOptionsAffectSemanticDiagnostics(newOptions, oldOptions) {
         return oldOptions !== newOptions &&
             ts.semanticDiagnosticsOptionDeclarations.some(function (option) { return !isJsonEqual(getCompilerOptionValue(oldOptions, option), getCompilerOptionValue(newOptions, option)); });
@@ -18084,6 +18913,26 @@ var ts;
         return option.strictFlag ? getStrictOptionValue(options, option.name) : options[option.name];
     }
     ts.getCompilerOptionValue = getCompilerOptionValue;
+    function getJSXTransformEnabled(options) {
+        var jsx = options.jsx;
+        return jsx === 2 /* React */ || jsx === 4 /* ReactJSX */ || jsx === 5 /* ReactJSXDev */;
+    }
+    ts.getJSXTransformEnabled = getJSXTransformEnabled;
+    function getJSXImplicitImportBase(compilerOptions, file) {
+        var jsxImportSourcePragmas = file === null || file === void 0 ? void 0 : file.pragmas.get("jsximportsource");
+        var jsxImportSourcePragma = ts.isArray(jsxImportSourcePragmas) ? jsxImportSourcePragmas[0] : jsxImportSourcePragmas;
+        return compilerOptions.jsx === 4 /* ReactJSX */ ||
+            compilerOptions.jsx === 5 /* ReactJSXDev */ ||
+            compilerOptions.jsxImportSource ||
+            jsxImportSourcePragma ?
+            (jsxImportSourcePragma === null || jsxImportSourcePragma === void 0 ? void 0 : jsxImportSourcePragma.arguments.factory) || compilerOptions.jsxImportSource || "react" :
+            undefined;
+    }
+    ts.getJSXImplicitImportBase = getJSXImplicitImportBase;
+    function getJSXRuntimeImport(base, options) {
+        return base ? base + "/" + (options.jsx === 5 /* ReactJSXDev */ ? "jsx-dev-runtime" : "jsx-runtime") : undefined;
+    }
+    ts.getJSXRuntimeImport = getJSXRuntimeImport;
     function hasZeroOrOneAsteriskCharacter(str) {
         var seenAsterisk = false;
         for (var i = 0; i < str.length; i++) {
@@ -18100,8 +18949,19 @@ var ts;
         return true;
     }
     ts.hasZeroOrOneAsteriskCharacter = hasZeroOrOneAsteriskCharacter;
+    function createSymlinkCache() {
+        var symlinkedDirectories;
+        var symlinkedFiles;
+        return {
+            getSymlinkedFiles: function () { return symlinkedFiles; },
+            getSymlinkedDirectories: function () { return symlinkedDirectories; },
+            setSymlinkedFile: function (path, real) { return (symlinkedFiles || (symlinkedFiles = new ts.Map())).set(path, real); },
+            setSymlinkedDirectory: function (path, directory) { return (symlinkedDirectories || (symlinkedDirectories = new ts.Map())).set(path, directory); },
+        };
+    }
+    ts.createSymlinkCache = createSymlinkCache;
     function discoverProbableSymlinks(files, getCanonicalFileName, cwd) {
-        var result = ts.createMap();
+        var cache = createSymlinkCache();
         var symlinks = ts.flatten(ts.mapDefined(files, function (sf) {
             return sf.resolvedModules && ts.compact(ts.arrayFrom(ts.mapIterator(sf.resolvedModules.values(), function (res) {
                 return res && res.originalPath && res.resolvedFileName !== res.originalPath ? [res.resolvedFileName, res.originalPath] : undefined;
@@ -18109,22 +18969,26 @@ var ts;
         }));
         for (var _i = 0, symlinks_1 = symlinks; _i < symlinks_1.length; _i++) {
             var _a = symlinks_1[_i], resolvedPath = _a[0], originalPath = _a[1];
-            var _b = guessDirectorySymlink(resolvedPath, originalPath, cwd, getCanonicalFileName), commonResolved = _b[0], commonOriginal = _b[1];
-            result.set(commonOriginal, commonResolved);
+            var _b = guessDirectorySymlink(resolvedPath, originalPath, cwd, getCanonicalFileName) || ts.emptyArray, commonResolved = _b[0], commonOriginal = _b[1];
+            if (commonResolved && commonOriginal) {
+                cache.setSymlinkedDirectory(ts.toPath(commonOriginal, cwd, getCanonicalFileName), { real: commonResolved, realPath: ts.toPath(commonResolved, cwd, getCanonicalFileName) });
+            }
         }
-        return result;
+        return cache;
     }
     ts.discoverProbableSymlinks = discoverProbableSymlinks;
     function guessDirectorySymlink(a, b, cwd, getCanonicalFileName) {
         var aParts = ts.getPathComponents(ts.toPath(a, cwd, getCanonicalFileName));
         var bParts = ts.getPathComponents(ts.toPath(b, cwd, getCanonicalFileName));
+        var isDirectory = false;
         while (!isNodeModulesOrScopedPackageDirectory(aParts[aParts.length - 2], getCanonicalFileName) &&
             !isNodeModulesOrScopedPackageDirectory(bParts[bParts.length - 2], getCanonicalFileName) &&
             getCanonicalFileName(aParts[aParts.length - 1]) === getCanonicalFileName(bParts[bParts.length - 1])) {
             aParts.pop();
             bParts.pop();
+            isDirectory = true;
         }
-        return [ts.getPathFromPathComponents(aParts), ts.getPathFromPathComponents(bParts)];
+        return isDirectory ? [ts.getPathFromPathComponents(aParts), ts.getPathFromPathComponents(bParts)] : undefined;
     }
     // KLUDGE: Don't assume one 'node_modules' links to another. More likely a single directory inside the node_modules is the symlink.
     // ALso, don't assume that an `@foo` directory is linked. More likely the contents of that are linked.
@@ -18313,7 +19177,7 @@ var ts;
         // Associate an array of results with each include regex. This keeps results in order of the "include" order.
         // If there are no "includes", then just put everything in results[0].
         var results = includeFileRegexes ? includeFileRegexes.map(function () { return []; }) : [[]];
-        var visited = ts.createMap();
+        var visited = new ts.Map();
         var toCanonical = ts.createGetCanonicalFileName(useCaseSensitiveFileNames);
         for (var _i = 0, _a = patterns.basePaths; _i < _a.length; _i++) {
             var basePath = _a[_i];
@@ -18448,7 +19312,7 @@ var ts;
     var allSupportedExtensions = __spreadArrays(ts.supportedTSExtensions, ts.supportedJSExtensions);
     var allSupportedExtensionsWithJson = __spreadArrays(ts.supportedTSExtensions, ts.supportedJSExtensions, [".json" /* Json */]);
     function getSupportedExtensions(options, extraFileExtensions) {
-        var needJsExtensions = options && options.allowJs;
+        var needJsExtensions = options && getAllowJSCompilerOption(options);
         if (!extraFileExtensions || extraFileExtensions.length === 0) {
             return needJsExtensions ? allSupportedExtensions : ts.supportedTSExtensions;
         }
@@ -18659,6 +19523,7 @@ var ts;
         if (!diagnostic.relatedInformation) {
             diagnostic.relatedInformation = [];
         }
+        ts.Debug.assert(diagnostic.relatedInformation !== ts.emptyArray, "Diagnostic had empty array singleton for related info, but is still being constructed!");
         (_a = diagnostic.relatedInformation).push.apply(_a, relatedInformation);
         return diagnostic;
     }
@@ -18679,62 +19544,6 @@ var ts;
         return { min: min, max: max };
     }
     ts.minAndMax = minAndMax;
-    var NodeSet = /** @class */ (function () {
-        function NodeSet() {
-            this.map = ts.createMap();
-        }
-        NodeSet.prototype.add = function (node) {
-            this.map.set(String(ts.getNodeId(node)), node);
-        };
-        NodeSet.prototype.tryAdd = function (node) {
-            if (this.has(node))
-                return false;
-            this.add(node);
-            return true;
-        };
-        NodeSet.prototype.has = function (node) {
-            return this.map.has(String(ts.getNodeId(node)));
-        };
-        NodeSet.prototype.forEach = function (cb) {
-            this.map.forEach(cb);
-        };
-        NodeSet.prototype.some = function (pred) {
-            return forEachEntry(this.map, pred) || false;
-        };
-        return NodeSet;
-    }());
-    ts.NodeSet = NodeSet;
-    var NodeMap = /** @class */ (function () {
-        function NodeMap() {
-            this.map = ts.createMap();
-        }
-        NodeMap.prototype.get = function (node) {
-            var res = this.map.get(String(ts.getNodeId(node)));
-            return res && res.value;
-        };
-        NodeMap.prototype.getOrUpdate = function (node, setValue) {
-            var res = this.get(node);
-            if (res)
-                return res;
-            var value = setValue();
-            this.set(node, value);
-            return value;
-        };
-        NodeMap.prototype.set = function (node, value) {
-            this.map.set(String(ts.getNodeId(node)), { node: node, value: value });
-        };
-        NodeMap.prototype.has = function (node) {
-            return this.map.has(String(ts.getNodeId(node)));
-        };
-        NodeMap.prototype.forEach = function (cb) {
-            this.map.forEach(function (_a) {
-                var node = _a.node, value = _a.value;
-                return cb(value, node);
-            });
-        };
-        return NodeMap;
-    }());
-    ts.NodeMap = NodeMap;
     function rangeOfNode(node) {
         return { pos: getTokenPosOfNode(node), end: node.end };
     }
@@ -18758,18 +19567,6 @@ var ts;
         return a === b || typeof a === "object" && a !== null && typeof b === "object" && b !== null && ts.equalOwnProperties(a, b, isJsonEqual);
     }
     ts.isJsonEqual = isJsonEqual;
-    function getOrUpdate(map, key, getDefault) {
-        var got = map.get(key);
-        if (got === undefined) {
-            var value = getDefault();
-            map.set(key, value);
-            return value;
-        }
-        else {
-            return got;
-        }
-    }
-    ts.getOrUpdate = getOrUpdate;
     /**
      * Converts a bigint literal string, e.g. `0x1234n`,
      * to its decimal string representation, e.g. `4660`.
@@ -18855,38 +19652,38 @@ var ts;
     }
     ts.isValidTypeOnlyAliasUseSite = isValidTypeOnlyAliasUseSite;
     function typeOnlyDeclarationIsExport(typeOnlyDeclaration) {
-        return typeOnlyDeclaration.kind === 263 /* ExportSpecifier */;
+        return typeOnlyDeclaration.kind === 270 /* ExportSpecifier */;
     }
     ts.typeOnlyDeclarationIsExport = typeOnlyDeclarationIsExport;
     function isPartOfPossiblyValidTypeOrAbstractComputedPropertyName(node) {
-        while (node.kind === 75 /* Identifier */ || node.kind === 194 /* PropertyAccessExpression */) {
+        while (node.kind === 78 /* Identifier */ || node.kind === 201 /* PropertyAccessExpression */) {
             node = node.parent;
         }
-        if (node.kind !== 154 /* ComputedPropertyName */) {
+        if (node.kind !== 158 /* ComputedPropertyName */) {
             return false;
         }
-        if (hasModifier(node.parent, 128 /* Abstract */)) {
+        if (hasSyntacticModifier(node.parent, 128 /* Abstract */)) {
             return true;
         }
         var containerKind = node.parent.parent.kind;
-        return containerKind === 246 /* InterfaceDeclaration */ || containerKind === 173 /* TypeLiteral */;
+        return containerKind === 253 /* InterfaceDeclaration */ || containerKind === 177 /* TypeLiteral */;
     }
     /** Returns true for an identifier in 1) an `implements` clause, and 2) an `extends` clause of an interface. */
     function isIdentifierInNonEmittingHeritageClause(node) {
-        if (node.kind !== 75 /* Identifier */)
+        if (node.kind !== 78 /* Identifier */)
             return false;
-        var heritageClause = findAncestor(node.parent, function (parent) {
+        var heritageClause = ts.findAncestor(node.parent, function (parent) {
             switch (parent.kind) {
-                case 279 /* HeritageClause */:
+                case 286 /* HeritageClause */:
                     return true;
-                case 194 /* PropertyAccessExpression */:
-                case 216 /* ExpressionWithTypeArguments */:
+                case 201 /* PropertyAccessExpression */:
+                case 223 /* ExpressionWithTypeArguments */:
                     return false;
                 default:
                     return "quit";
             }
         });
-        return (heritageClause === null || heritageClause === void 0 ? void 0 : heritageClause.token) === 113 /* ImplementsKeyword */ || (heritageClause === null || heritageClause === void 0 ? void 0 : heritageClause.parent.kind) === 246 /* InterfaceDeclaration */;
+        return (heritageClause === null || heritageClause === void 0 ? void 0 : heritageClause.token) === 116 /* ImplementsKeyword */ || (heritageClause === null || heritageClause === void 0 ? void 0 : heritageClause.parent.kind) === 253 /* InterfaceDeclaration */;
     }
     function isIdentifierTypeReference(node) {
         return ts.isTypeReferenceNode(node) && ts.isIdentifier(node.typeName);
@@ -18905,55864 +19702,60622 @@ var ts;
         return true;
     }
     ts.arrayIsHomogeneous = arrayIsHomogeneous;
-})(ts || (ts = {}));
-var ts;
-(function (ts) {
-    var SignatureFlags;
-    (function (SignatureFlags) {
-        SignatureFlags[SignatureFlags["None"] = 0] = "None";
-        SignatureFlags[SignatureFlags["Yield"] = 1] = "Yield";
-        SignatureFlags[SignatureFlags["Await"] = 2] = "Await";
-        SignatureFlags[SignatureFlags["Type"] = 4] = "Type";
-        SignatureFlags[SignatureFlags["IgnoreMissingOpenBrace"] = 16] = "IgnoreMissingOpenBrace";
-        SignatureFlags[SignatureFlags["JSDoc"] = 32] = "JSDoc";
-    })(SignatureFlags || (SignatureFlags = {}));
-    var NodeConstructor;
-    var TokenConstructor;
-    var IdentifierConstructor;
-    var PrivateIdentifierConstructor;
-    var SourceFileConstructor;
-    function createNode(kind, pos, end) {
-        if (kind === 290 /* SourceFile */) {
-            return new (SourceFileConstructor || (SourceFileConstructor = ts.objectAllocator.getSourceFileConstructor()))(kind, pos, end);
-        }
-        else if (kind === 75 /* Identifier */) {
-            return new (IdentifierConstructor || (IdentifierConstructor = ts.objectAllocator.getIdentifierConstructor()))(kind, pos, end);
-        }
-        else if (kind === 76 /* PrivateIdentifier */) {
-            return new (PrivateIdentifierConstructor || (PrivateIdentifierConstructor = ts.objectAllocator.getPrivateIdentifierConstructor()))(kind, pos, end);
-        }
-        else if (!ts.isNodeKind(kind)) {
-            return new (TokenConstructor || (TokenConstructor = ts.objectAllocator.getTokenConstructor()))(kind, pos, end);
-        }
-        else {
-            return new (NodeConstructor || (NodeConstructor = ts.objectAllocator.getNodeConstructor()))(kind, pos, end);
-        }
-    }
-    ts.createNode = createNode;
-    function visitNode(cbNode, node) {
-        return node && cbNode(node);
-    }
-    function visitNodes(cbNode, cbNodes, nodes) {
-        if (nodes) {
-            if (cbNodes) {
-                return cbNodes(nodes);
-            }
-            for (var _i = 0, nodes_1 = nodes; _i < nodes_1.length; _i++) {
-                var node = nodes_1[_i];
-                var result = cbNode(node);
-                if (result) {
-                    return result;
-                }
-            }
-        }
+    /**
+     * Bypasses immutability and directly sets the `pos` property of a `TextRange` or `Node`.
+     */
+    /* @internal */
+    function setTextRangePos(range, pos) {
+        range.pos = pos;
+        return range;
     }
-    /*@internal*/
-    function isJSDocLikeText(text, start) {
-        return text.charCodeAt(start + 1) === 42 /* asterisk */ &&
-            text.charCodeAt(start + 2) === 42 /* asterisk */ &&
-            text.charCodeAt(start + 3) !== 47 /* slash */;
+    ts.setTextRangePos = setTextRangePos;
+    /**
+     * Bypasses immutability and directly sets the `end` property of a `TextRange` or `Node`.
+     */
+    /* @internal */
+    function setTextRangeEnd(range, end) {
+        range.end = end;
+        return range;
     }
-    ts.isJSDocLikeText = isJSDocLikeText;
+    ts.setTextRangeEnd = setTextRangeEnd;
     /**
-     * Invokes a callback for each child of the given node. The 'cbNode' callback is invoked for all child nodes
-     * stored in properties. If a 'cbNodes' callback is specified, it is invoked for embedded arrays; otherwise,
-     * embedded arrays are flattened and the 'cbNode' callback is invoked for each element. If a callback returns
-     * a truthy value, iteration stops and that value is returned. Otherwise, undefined is returned.
-     *
-     * @param node a given node to visit its children
-     * @param cbNode a callback to be invoked for all child nodes
-     * @param cbNodes a callback to be invoked for embedded array
-     *
-     * @remarks `forEachChild` must visit the children of a node in the order
-     * that they appear in the source code. The language service depends on this property to locate nodes by position.
+     * Bypasses immutability and directly sets the `pos` and `end` properties of a `TextRange` or `Node`.
      */
-    function forEachChild(node, cbNode, cbNodes) {
-        if (!node || node.kind <= 152 /* LastToken */) {
-            return;
-        }
-        switch (node.kind) {
-            case 153 /* QualifiedName */:
-                return visitNode(cbNode, node.left) ||
-                    visitNode(cbNode, node.right);
-            case 155 /* TypeParameter */:
-                return visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.constraint) ||
-                    visitNode(cbNode, node.default) ||
-                    visitNode(cbNode, node.expression);
-            case 282 /* ShorthandPropertyAssignment */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.exclamationToken) ||
-                    visitNode(cbNode, node.equalsToken) ||
-                    visitNode(cbNode, node.objectAssignmentInitializer);
-            case 283 /* SpreadAssignment */:
-                return visitNode(cbNode, node.expression);
-            case 156 /* Parameter */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.dotDotDotToken) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.type) ||
-                    visitNode(cbNode, node.initializer);
-            case 159 /* PropertyDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.exclamationToken) ||
-                    visitNode(cbNode, node.type) ||
-                    visitNode(cbNode, node.initializer);
-            case 158 /* PropertySignature */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.type) ||
-                    visitNode(cbNode, node.initializer);
-            case 281 /* PropertyAssignment */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.initializer);
-            case 242 /* VariableDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.exclamationToken) ||
-                    visitNode(cbNode, node.type) ||
-                    visitNode(cbNode, node.initializer);
-            case 191 /* BindingElement */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.dotDotDotToken) ||
-                    visitNode(cbNode, node.propertyName) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.initializer);
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
-            case 165 /* CallSignature */:
-            case 166 /* ConstructSignature */:
-            case 167 /* IndexSignature */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
-                    visitNodes(cbNode, cbNodes, node.parameters) ||
-                    visitNode(cbNode, node.type);
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
-            case 162 /* Constructor */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 201 /* FunctionExpression */:
-            case 244 /* FunctionDeclaration */:
-            case 202 /* ArrowFunction */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.asteriskToken) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.exclamationToken) ||
-                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
-                    visitNodes(cbNode, cbNodes, node.parameters) ||
-                    visitNode(cbNode, node.type) ||
-                    visitNode(cbNode, node.equalsGreaterThanToken) ||
-                    visitNode(cbNode, node.body);
-            case 169 /* TypeReference */:
-                return visitNode(cbNode, node.typeName) ||
-                    visitNodes(cbNode, cbNodes, node.typeArguments);
-            case 168 /* TypePredicate */:
-                return visitNode(cbNode, node.assertsModifier) ||
-                    visitNode(cbNode, node.parameterName) ||
-                    visitNode(cbNode, node.type);
-            case 172 /* TypeQuery */:
-                return visitNode(cbNode, node.exprName);
-            case 173 /* TypeLiteral */:
-                return visitNodes(cbNode, cbNodes, node.members);
-            case 174 /* ArrayType */:
-                return visitNode(cbNode, node.elementType);
-            case 175 /* TupleType */:
-                return visitNodes(cbNode, cbNodes, node.elementTypes);
-            case 178 /* UnionType */:
-            case 179 /* IntersectionType */:
-                return visitNodes(cbNode, cbNodes, node.types);
-            case 180 /* ConditionalType */:
-                return visitNode(cbNode, node.checkType) ||
-                    visitNode(cbNode, node.extendsType) ||
-                    visitNode(cbNode, node.trueType) ||
-                    visitNode(cbNode, node.falseType);
-            case 181 /* InferType */:
-                return visitNode(cbNode, node.typeParameter);
-            case 188 /* ImportType */:
-                return visitNode(cbNode, node.argument) ||
-                    visitNode(cbNode, node.qualifier) ||
-                    visitNodes(cbNode, cbNodes, node.typeArguments);
-            case 182 /* ParenthesizedType */:
-            case 184 /* TypeOperator */:
-                return visitNode(cbNode, node.type);
-            case 185 /* IndexedAccessType */:
-                return visitNode(cbNode, node.objectType) ||
-                    visitNode(cbNode, node.indexType);
-            case 186 /* MappedType */:
-                return visitNode(cbNode, node.readonlyToken) ||
-                    visitNode(cbNode, node.typeParameter) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.type);
-            case 187 /* LiteralType */:
-                return visitNode(cbNode, node.literal);
-            case 189 /* ObjectBindingPattern */:
-            case 190 /* ArrayBindingPattern */:
-                return visitNodes(cbNode, cbNodes, node.elements);
-            case 192 /* ArrayLiteralExpression */:
-                return visitNodes(cbNode, cbNodes, node.elements);
-            case 193 /* ObjectLiteralExpression */:
-                return visitNodes(cbNode, cbNodes, node.properties);
-            case 194 /* PropertyAccessExpression */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.questionDotToken) ||
-                    visitNode(cbNode, node.name);
-            case 195 /* ElementAccessExpression */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.questionDotToken) ||
-                    visitNode(cbNode, node.argumentExpression);
-            case 196 /* CallExpression */:
-            case 197 /* NewExpression */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.questionDotToken) ||
-                    visitNodes(cbNode, cbNodes, node.typeArguments) ||
-                    visitNodes(cbNode, cbNodes, node.arguments);
-            case 198 /* TaggedTemplateExpression */:
-                return visitNode(cbNode, node.tag) ||
-                    visitNode(cbNode, node.questionDotToken) ||
-                    visitNodes(cbNode, cbNodes, node.typeArguments) ||
-                    visitNode(cbNode, node.template);
-            case 199 /* TypeAssertionExpression */:
-                return visitNode(cbNode, node.type) ||
-                    visitNode(cbNode, node.expression);
-            case 200 /* ParenthesizedExpression */:
-                return visitNode(cbNode, node.expression);
-            case 203 /* DeleteExpression */:
-                return visitNode(cbNode, node.expression);
-            case 204 /* TypeOfExpression */:
-                return visitNode(cbNode, node.expression);
-            case 205 /* VoidExpression */:
-                return visitNode(cbNode, node.expression);
-            case 207 /* PrefixUnaryExpression */:
-                return visitNode(cbNode, node.operand);
-            case 212 /* YieldExpression */:
-                return visitNode(cbNode, node.asteriskToken) ||
-                    visitNode(cbNode, node.expression);
-            case 206 /* AwaitExpression */:
-                return visitNode(cbNode, node.expression);
-            case 208 /* PostfixUnaryExpression */:
-                return visitNode(cbNode, node.operand);
-            case 209 /* BinaryExpression */:
-                return visitNode(cbNode, node.left) ||
-                    visitNode(cbNode, node.operatorToken) ||
-                    visitNode(cbNode, node.right);
-            case 217 /* AsExpression */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.type);
-            case 218 /* NonNullExpression */:
-                return visitNode(cbNode, node.expression);
-            case 219 /* MetaProperty */:
-                return visitNode(cbNode, node.name);
-            case 210 /* ConditionalExpression */:
-                return visitNode(cbNode, node.condition) ||
-                    visitNode(cbNode, node.questionToken) ||
-                    visitNode(cbNode, node.whenTrue) ||
-                    visitNode(cbNode, node.colonToken) ||
-                    visitNode(cbNode, node.whenFalse);
-            case 213 /* SpreadElement */:
-                return visitNode(cbNode, node.expression);
-            case 223 /* Block */:
-            case 250 /* ModuleBlock */:
-                return visitNodes(cbNode, cbNodes, node.statements);
-            case 290 /* SourceFile */:
-                return visitNodes(cbNode, cbNodes, node.statements) ||
-                    visitNode(cbNode, node.endOfFileToken);
-            case 225 /* VariableStatement */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.declarationList);
-            case 243 /* VariableDeclarationList */:
-                return visitNodes(cbNode, cbNodes, node.declarations);
-            case 226 /* ExpressionStatement */:
-                return visitNode(cbNode, node.expression);
-            case 227 /* IfStatement */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.thenStatement) ||
-                    visitNode(cbNode, node.elseStatement);
-            case 228 /* DoStatement */:
-                return visitNode(cbNode, node.statement) ||
-                    visitNode(cbNode, node.expression);
-            case 229 /* WhileStatement */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.statement);
-            case 230 /* ForStatement */:
-                return visitNode(cbNode, node.initializer) ||
-                    visitNode(cbNode, node.condition) ||
-                    visitNode(cbNode, node.incrementor) ||
-                    visitNode(cbNode, node.statement);
-            case 231 /* ForInStatement */:
-                return visitNode(cbNode, node.initializer) ||
-                    visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.statement);
-            case 232 /* ForOfStatement */:
-                return visitNode(cbNode, node.awaitModifier) ||
-                    visitNode(cbNode, node.initializer) ||
-                    visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.statement);
-            case 233 /* ContinueStatement */:
-            case 234 /* BreakStatement */:
-                return visitNode(cbNode, node.label);
-            case 235 /* ReturnStatement */:
-                return visitNode(cbNode, node.expression);
-            case 236 /* WithStatement */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.statement);
-            case 237 /* SwitchStatement */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNode(cbNode, node.caseBlock);
-            case 251 /* CaseBlock */:
-                return visitNodes(cbNode, cbNodes, node.clauses);
-            case 277 /* CaseClause */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNodes(cbNode, cbNodes, node.statements);
-            case 278 /* DefaultClause */:
-                return visitNodes(cbNode, cbNodes, node.statements);
-            case 238 /* LabeledStatement */:
-                return visitNode(cbNode, node.label) ||
-                    visitNode(cbNode, node.statement);
-            case 239 /* ThrowStatement */:
-                return visitNode(cbNode, node.expression);
-            case 240 /* TryStatement */:
-                return visitNode(cbNode, node.tryBlock) ||
-                    visitNode(cbNode, node.catchClause) ||
-                    visitNode(cbNode, node.finallyBlock);
-            case 280 /* CatchClause */:
-                return visitNode(cbNode, node.variableDeclaration) ||
-                    visitNode(cbNode, node.block);
-            case 157 /* Decorator */:
-                return visitNode(cbNode, node.expression);
-            case 245 /* ClassDeclaration */:
-            case 214 /* ClassExpression */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
-                    visitNodes(cbNode, cbNodes, node.heritageClauses) ||
-                    visitNodes(cbNode, cbNodes, node.members);
-            case 246 /* InterfaceDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
-                    visitNodes(cbNode, cbNodes, node.heritageClauses) ||
-                    visitNodes(cbNode, cbNodes, node.members);
-            case 247 /* TypeAliasDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
-                    visitNode(cbNode, node.type);
-            case 248 /* EnumDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNodes(cbNode, cbNodes, node.members);
-            case 284 /* EnumMember */:
-                return visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.initializer);
-            case 249 /* ModuleDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.body);
-            case 253 /* ImportEqualsDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.moduleReference);
-            case 254 /* ImportDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.importClause) ||
-                    visitNode(cbNode, node.moduleSpecifier);
-            case 255 /* ImportClause */:
-                return visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.namedBindings);
-            case 252 /* NamespaceExportDeclaration */:
-                return visitNode(cbNode, node.name);
-            case 256 /* NamespaceImport */:
-                return visitNode(cbNode, node.name);
-            case 262 /* NamespaceExport */:
-                return visitNode(cbNode, node.name);
-            case 257 /* NamedImports */:
-            case 261 /* NamedExports */:
-                return visitNodes(cbNode, cbNodes, node.elements);
-            case 260 /* ExportDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.exportClause) ||
-                    visitNode(cbNode, node.moduleSpecifier);
-            case 258 /* ImportSpecifier */:
-            case 263 /* ExportSpecifier */:
-                return visitNode(cbNode, node.propertyName) ||
-                    visitNode(cbNode, node.name);
-            case 259 /* ExportAssignment */:
-                return visitNodes(cbNode, cbNodes, node.decorators) ||
-                    visitNodes(cbNode, cbNodes, node.modifiers) ||
-                    visitNode(cbNode, node.expression);
-            case 211 /* TemplateExpression */:
-                return visitNode(cbNode, node.head) || visitNodes(cbNode, cbNodes, node.templateSpans);
-            case 221 /* TemplateSpan */:
-                return visitNode(cbNode, node.expression) || visitNode(cbNode, node.literal);
-            case 154 /* ComputedPropertyName */:
-                return visitNode(cbNode, node.expression);
-            case 279 /* HeritageClause */:
-                return visitNodes(cbNode, cbNodes, node.types);
-            case 216 /* ExpressionWithTypeArguments */:
-                return visitNode(cbNode, node.expression) ||
-                    visitNodes(cbNode, cbNodes, node.typeArguments);
-            case 265 /* ExternalModuleReference */:
-                return visitNode(cbNode, node.expression);
-            case 264 /* MissingDeclaration */:
-                return visitNodes(cbNode, cbNodes, node.decorators);
-            case 327 /* CommaListExpression */:
-                return visitNodes(cbNode, cbNodes, node.elements);
-            case 266 /* JsxElement */:
-                return visitNode(cbNode, node.openingElement) ||
-                    visitNodes(cbNode, cbNodes, node.children) ||
-                    visitNode(cbNode, node.closingElement);
-            case 270 /* JsxFragment */:
-                return visitNode(cbNode, node.openingFragment) ||
-                    visitNodes(cbNode, cbNodes, node.children) ||
-                    visitNode(cbNode, node.closingFragment);
-            case 267 /* JsxSelfClosingElement */:
-            case 268 /* JsxOpeningElement */:
-                return visitNode(cbNode, node.tagName) ||
-                    visitNodes(cbNode, cbNodes, node.typeArguments) ||
-                    visitNode(cbNode, node.attributes);
-            case 274 /* JsxAttributes */:
-                return visitNodes(cbNode, cbNodes, node.properties);
-            case 273 /* JsxAttribute */:
-                return visitNode(cbNode, node.name) ||
-                    visitNode(cbNode, node.initializer);
-            case 275 /* JsxSpreadAttribute */:
-                return visitNode(cbNode, node.expression);
-            case 276 /* JsxExpression */:
-                return visitNode(cbNode, node.dotDotDotToken) ||
-                    visitNode(cbNode, node.expression);
-            case 269 /* JsxClosingElement */:
-                return visitNode(cbNode, node.tagName);
-            case 176 /* OptionalType */:
-            case 177 /* RestType */:
-            case 294 /* JSDocTypeExpression */:
-            case 298 /* JSDocNonNullableType */:
-            case 297 /* JSDocNullableType */:
-            case 299 /* JSDocOptionalType */:
-            case 301 /* JSDocVariadicType */:
-                return visitNode(cbNode, node.type);
-            case 300 /* JSDocFunctionType */:
-                return visitNodes(cbNode, cbNodes, node.parameters) ||
-                    visitNode(cbNode, node.type);
-            case 303 /* JSDocComment */:
-                return visitNodes(cbNode, cbNodes, node.tags);
-            case 317 /* JSDocParameterTag */:
-            case 323 /* JSDocPropertyTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    (node.isNameFirst
-                        ? visitNode(cbNode, node.name) ||
-                            visitNode(cbNode, node.typeExpression)
-                        : visitNode(cbNode, node.typeExpression) ||
-                            visitNode(cbNode, node.name));
-            case 309 /* JSDocAuthorTag */:
-                return visitNode(cbNode, node.tagName);
-            case 308 /* JSDocImplementsTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    visitNode(cbNode, node.class);
-            case 307 /* JSDocAugmentsTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    visitNode(cbNode, node.class);
-            case 321 /* JSDocTemplateTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    visitNode(cbNode, node.constraint) ||
-                    visitNodes(cbNode, cbNodes, node.typeParameters);
-            case 322 /* JSDocTypedefTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    (node.typeExpression &&
-                        node.typeExpression.kind === 294 /* JSDocTypeExpression */
-                        ? visitNode(cbNode, node.typeExpression) ||
-                            visitNode(cbNode, node.fullName)
-                        : visitNode(cbNode, node.fullName) ||
-                            visitNode(cbNode, node.typeExpression));
-            case 315 /* JSDocCallbackTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    visitNode(cbNode, node.fullName) ||
-                    visitNode(cbNode, node.typeExpression);
-            case 318 /* JSDocReturnTag */:
-            case 320 /* JSDocTypeTag */:
-            case 319 /* JSDocThisTag */:
-            case 316 /* JSDocEnumTag */:
-                return visitNode(cbNode, node.tagName) ||
-                    visitNode(cbNode, node.typeExpression);
-            case 305 /* JSDocSignature */:
-                return ts.forEach(node.typeParameters, cbNode) ||
-                    ts.forEach(node.parameters, cbNode) ||
-                    visitNode(cbNode, node.type);
-            case 304 /* JSDocTypeLiteral */:
-                return ts.forEach(node.jsDocPropertyTags, cbNode);
-            case 306 /* JSDocTag */:
-            case 310 /* JSDocClassTag */:
-            case 311 /* JSDocPublicTag */:
-            case 312 /* JSDocPrivateTag */:
-            case 313 /* JSDocProtectedTag */:
-            case 314 /* JSDocReadonlyTag */:
-                return visitNode(cbNode, node.tagName);
-            case 326 /* PartiallyEmittedExpression */:
-                return visitNode(cbNode, node.expression);
-        }
+    /* @internal */
+    function setTextRangePosEnd(range, pos, end) {
+        return setTextRangeEnd(setTextRangePos(range, pos), end);
     }
-    ts.forEachChild = forEachChild;
-    /** @internal */
+    ts.setTextRangePosEnd = setTextRangePosEnd;
     /**
-     * Invokes a callback for each child of the given node. The 'cbNode' callback is invoked for all child nodes
-     * stored in properties. If a 'cbNodes' callback is specified, it is invoked for embedded arrays; additionally,
-     * unlike `forEachChild`, embedded arrays are flattened and the 'cbNode' callback is invoked for each element.
-     *  If a callback returns a truthy value, iteration stops and that value is returned. Otherwise, undefined is returned.
-     *
-     * @param node a given node to visit its children
-     * @param cbNode a callback to be invoked for all child nodes
-     * @param cbNodes a callback to be invoked for embedded array
-     *
-     * @remarks Unlike `forEachChild`, `forEachChildRecursively` handles recursively invoking the traversal on each child node found,
-     * and while doing so, handles traversing the structure without relying on the callstack to encode the tree structure.
+     * Bypasses immutability and directly sets the `pos` and `end` properties of a `TextRange` or `Node` from the
+     * provided position and width.
      */
-    function forEachChildRecursively(rootNode, cbNode, cbNodes) {
-        var stack = [rootNode];
-        while (stack.length) {
-            var parent = stack.pop();
-            var res = visitAllPossibleChildren(parent, gatherPossibleChildren(parent));
-            if (res) {
-                return res;
-            }
+    /* @internal */
+    function setTextRangePosWidth(range, pos, width) {
+        return setTextRangePosEnd(range, pos, pos + width);
+    }
+    ts.setTextRangePosWidth = setTextRangePosWidth;
+    function setNodeFlags(node, newFlags) {
+        if (node) {
+            node.flags = newFlags;
         }
-        return;
-        function gatherPossibleChildren(node) {
-            var children = [];
-            forEachChild(node, addWorkItem, addWorkItem); // By using a stack above and `unshift` here, we emulate a depth-first preorder traversal
-            return children;
-            function addWorkItem(n) {
-                children.unshift(n);
-            }
+        return node;
+    }
+    ts.setNodeFlags = setNodeFlags;
+    function setParent(child, parent) {
+        if (child && parent) {
+            child.parent = parent;
         }
-        function visitAllPossibleChildren(parent, children) {
+        return child;
+    }
+    ts.setParent = setParent;
+    function setEachParent(children, parent) {
+        if (children) {
             for (var _i = 0, children_1 = children; _i < children_1.length; _i++) {
                 var child = children_1[_i];
-                if (ts.isArray(child)) {
-                    if (cbNodes) {
-                        var res = cbNodes(child, parent);
-                        if (res) {
-                            if (res === "skip")
-                                continue;
-                            return res;
-                        }
-                    }
-                    for (var i = child.length - 1; i >= 0; i--) {
-                        var realChild = child[i];
-                        var res = cbNode(realChild, parent);
-                        if (res) {
-                            if (res === "skip")
-                                continue;
-                            return res;
-                        }
-                        stack.push(realChild);
-                    }
-                }
-                else {
-                    stack.push(child);
-                    var res = cbNode(child, parent);
-                    if (res) {
-                        if (res === "skip")
-                            continue;
-                        return res;
-                    }
-                }
+                setParent(child, parent);
             }
         }
+        return children;
     }
-    ts.forEachChildRecursively = forEachChildRecursively;
-    function createSourceFile(fileName, sourceText, languageVersion, setParentNodes, scriptKind) {
-        if (setParentNodes === void 0) { setParentNodes = false; }
-        ts.performance.mark("beforeParse");
-        var result;
-        ts.perfLogger.logStartParseSourceFile(fileName);
-        if (languageVersion === 100 /* JSON */) {
-            result = Parser.parseSourceFile(fileName, sourceText, languageVersion, /*syntaxCursor*/ undefined, setParentNodes, 6 /* JSON */);
+    ts.setEachParent = setEachParent;
+    function setParentRecursive(rootNode, incremental) {
+        if (!rootNode)
+            return rootNode;
+        ts.forEachChildRecursively(rootNode, ts.isJSDocNode(rootNode) ? bindParentToChildIgnoringJSDoc : bindParentToChild);
+        return rootNode;
+        function bindParentToChildIgnoringJSDoc(child, parent) {
+            if (incremental && child.parent === parent) {
+                return "skip";
+            }
+            setParent(child, parent);
         }
-        else {
-            result = Parser.parseSourceFile(fileName, sourceText, languageVersion, /*syntaxCursor*/ undefined, setParentNodes, scriptKind);
+        function bindJSDoc(child) {
+            if (ts.hasJSDocNodes(child)) {
+                for (var _i = 0, _a = child.jsDoc; _i < _a.length; _i++) {
+                    var doc = _a[_i];
+                    bindParentToChildIgnoringJSDoc(doc, child);
+                    ts.forEachChildRecursively(doc, bindParentToChildIgnoringJSDoc);
+                }
+            }
+        }
+        function bindParentToChild(child, parent) {
+            return bindParentToChildIgnoringJSDoc(child, parent) || bindJSDoc(child);
         }
-        ts.perfLogger.logStopParseSourceFile();
-        ts.performance.mark("afterParse");
-        ts.performance.measure("Parse", "beforeParse", "afterParse");
-        return result;
-    }
-    ts.createSourceFile = createSourceFile;
-    function parseIsolatedEntityName(text, languageVersion) {
-        return Parser.parseIsolatedEntityName(text, languageVersion);
     }
-    ts.parseIsolatedEntityName = parseIsolatedEntityName;
+    ts.setParentRecursive = setParentRecursive;
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
     /**
-     * Parse json text into SyntaxTree and return node and parse errors if any
-     * @param fileName
-     * @param sourceText
+     * Creates a `BaseNodeFactory` which can be used to create `Node` instances from the constructors provided by the object allocator.
      */
-    function parseJsonText(fileName, sourceText) {
-        return Parser.parseJsonText(fileName, sourceText);
-    }
-    ts.parseJsonText = parseJsonText;
-    // See also `isExternalOrCommonJsModule` in utilities.ts
-    function isExternalModule(file) {
-        return file.externalModuleIndicator !== undefined;
-    }
-    ts.isExternalModule = isExternalModule;
-    // Produces a new SourceFile for the 'newText' provided. The 'textChangeRange' parameter
-    // indicates what changed between the 'text' that this SourceFile has and the 'newText'.
-    // The SourceFile will be created with the compiler attempting to reuse as many nodes from
-    // this file as possible.
-    //
-    // Note: this function mutates nodes from this SourceFile. That means any existing nodes
-    // from this SourceFile that are being held onto may change as a result (including
-    // becoming detached from any SourceFile).  It is recommended that this SourceFile not
-    // be used once 'update' is called on it.
-    function updateSourceFile(sourceFile, newText, textChangeRange, aggressiveChecks) {
-        if (aggressiveChecks === void 0) { aggressiveChecks = false; }
-        var newSourceFile = IncrementalParser.updateSourceFile(sourceFile, newText, textChangeRange, aggressiveChecks);
-        // Because new source file node is created, it may not have the flag PossiblyContainDynamicImport. This is the case if there is no new edit to add dynamic import.
-        // We will manually port the flag to the new source file.
-        newSourceFile.flags |= (sourceFile.flags & 3145728 /* PermanentlySetIncrementalFlags */);
-        return newSourceFile;
-    }
-    ts.updateSourceFile = updateSourceFile;
-    /* @internal */
-    function parseIsolatedJSDocComment(content, start, length) {
-        var result = Parser.JSDocParser.parseIsolatedJSDocComment(content, start, length);
-        if (result && result.jsDoc) {
-            // because the jsDocComment was parsed out of the source file, it might
-            // not be covered by the fixupParentReferences.
-            Parser.fixupParentReferences(result.jsDoc);
-        }
-        return result;
-    }
-    ts.parseIsolatedJSDocComment = parseIsolatedJSDocComment;
-    /* @internal */
-    // Exposed only for testing.
-    function parseJSDocTypeExpressionForTests(content, start, length) {
-        return Parser.JSDocParser.parseJSDocTypeExpressionForTests(content, start, length);
-    }
-    ts.parseJSDocTypeExpressionForTests = parseJSDocTypeExpressionForTests;
-    // Implement the parser as a singleton module.  We do this for perf reasons because creating
-    // parser instances can actually be expensive enough to impact us on projects with many source
-    // files.
-    var Parser;
-    (function (Parser) {
-        // Share a single scanner across all calls to parse a source file.  This helps speed things
-        // up by avoiding the cost of creating/compiling scanners over and over again.
-        var scanner = ts.createScanner(99 /* Latest */, /*skipTrivia*/ true);
-        var disallowInAndDecoratorContext = 4096 /* DisallowInContext */ | 16384 /* DecoratorContext */;
-        // capture constructors in 'initializeState' to avoid null checks
+    function createBaseNodeFactory() {
+        // tslint:disable variable-name
         var NodeConstructor;
         var TokenConstructor;
         var IdentifierConstructor;
         var PrivateIdentifierConstructor;
         var SourceFileConstructor;
-        var sourceFile;
-        var parseDiagnostics;
-        var syntaxCursor;
-        var currentToken;
-        var sourceText;
-        var nodeCount;
-        var identifiers;
-        var privateIdentifiers;
-        var identifierCount;
-        var parsingContext;
-        var notParenthesizedArrow;
-        // Flags that dictate what parsing context we're in.  For example:
-        // Whether or not we are in strict parsing mode.  All that changes in strict parsing mode is
-        // that some tokens that would be considered identifiers may be considered keywords.
-        //
-        // When adding more parser context flags, consider which is the more common case that the
-        // flag will be in.  This should be the 'false' state for that flag.  The reason for this is
-        // that we don't store data in our nodes unless the value is in the *non-default* state.  So,
-        // for example, more often than code 'allows-in' (or doesn't 'disallow-in').  We opt for
-        // 'disallow-in' set to 'false'.  Otherwise, if we had 'allowsIn' set to 'true', then almost
-        // all nodes would need extra state on them to store this info.
-        //
-        // Note: 'allowIn' and 'allowYield' track 1:1 with the [in] and [yield] concepts in the ES6
-        // grammar specification.
-        //
-        // An important thing about these context concepts.  By default they are effectively inherited
-        // while parsing through every grammar production.  i.e. if you don't change them, then when
-        // you parse a sub-production, it will have the same context values as the parent production.
-        // This is great most of the time.  After all, consider all the 'expression' grammar productions
-        // and how nearly all of them pass along the 'in' and 'yield' context values:
-        //
-        // EqualityExpression[In, Yield] :
-        //      RelationalExpression[?In, ?Yield]
-        //      EqualityExpression[?In, ?Yield] == RelationalExpression[?In, ?Yield]
-        //      EqualityExpression[?In, ?Yield] != RelationalExpression[?In, ?Yield]
-        //      EqualityExpression[?In, ?Yield] === RelationalExpression[?In, ?Yield]
-        //      EqualityExpression[?In, ?Yield] !== RelationalExpression[?In, ?Yield]
-        //
-        // Where you have to be careful is then understanding what the points are in the grammar
-        // where the values are *not* passed along.  For example:
-        //
-        // SingleNameBinding[Yield,GeneratorParameter]
-        //      [+GeneratorParameter]BindingIdentifier[Yield] Initializer[In]opt
-        //      [~GeneratorParameter]BindingIdentifier[?Yield]Initializer[In, ?Yield]opt
-        //
-        // Here this is saying that if the GeneratorParameter context flag is set, that we should
-        // explicitly set the 'yield' context flag to false before calling into the BindingIdentifier
-        // and we should explicitly unset the 'yield' context flag before calling into the Initializer.
-        // production.  Conversely, if the GeneratorParameter context flag is not set, then we
-        // should leave the 'yield' context flag alone.
-        //
-        // Getting this all correct is tricky and requires careful reading of the grammar to
-        // understand when these values should be changed versus when they should be inherited.
-        //
-        // Note: it should not be necessary to save/restore these flags during speculative/lookahead
-        // parsing.  These context flags are naturally stored and restored through normal recursive
-        // descent parsing and unwinding.
-        var contextFlags;
-        // Whether or not we've had a parse error since creating the last AST node.  If we have
-        // encountered an error, it will be stored on the next AST node we create.  Parse errors
-        // can be broken down into three categories:
-        //
-        // 1) An error that occurred during scanning.  For example, an unterminated literal, or a
-        //    character that was completely not understood.
-        //
-        // 2) A token was expected, but was not present.  This type of error is commonly produced
-        //    by the 'parseExpected' function.
-        //
-        // 3) A token was present that no parsing function was able to consume.  This type of error
-        //    only occurs in the 'abortParsingListOrMoveToNextToken' function when the parser
-        //    decides to skip the token.
-        //
-        // In all of these cases, we want to mark the next node as having had an error before it.
-        // With this mark, we can know in incremental settings if this node can be reused, or if
-        // we have to reparse it.  If we don't keep this information around, we may just reuse the
-        // node.  in that event we would then not produce the same errors as we did before, causing
-        // significant confusion problems.
-        //
-        // Note: it is necessary that this value be saved/restored during speculative/lookahead
-        // parsing.  During lookahead parsing, we will often create a node.  That node will have
-        // this value attached, and then this value will be set back to 'false'.  If we decide to
-        // rewind, we must get back to the same value we had prior to the lookahead.
-        //
-        // Note: any errors at the end of the file that do not precede a regular node, should get
-        // attached to the EOF token.
-        var parseErrorBeforeNextFinishedNode = false;
-        function parseSourceFile(fileName, sourceText, languageVersion, syntaxCursor, setParentNodes, scriptKind) {
-            if (setParentNodes === void 0) { setParentNodes = false; }
-            scriptKind = ts.ensureScriptKind(fileName, scriptKind);
-            if (scriptKind === 6 /* JSON */) {
-                var result_2 = parseJsonText(fileName, sourceText, languageVersion, syntaxCursor, setParentNodes);
-                ts.convertToObjectWorker(result_2, result_2.parseDiagnostics, /*returnValue*/ false, /*knownRootOptions*/ undefined, /*jsonConversionNotifier*/ undefined);
-                result_2.referencedFiles = ts.emptyArray;
-                result_2.typeReferenceDirectives = ts.emptyArray;
-                result_2.libReferenceDirectives = ts.emptyArray;
-                result_2.amdDependencies = ts.emptyArray;
-                result_2.hasNoDefaultLib = false;
-                result_2.pragmas = ts.emptyMap;
-                return result_2;
-            }
-            initializeState(sourceText, languageVersion, syntaxCursor, scriptKind);
-            var result = parseSourceFileWorker(fileName, languageVersion, setParentNodes, scriptKind);
-            clearState();
-            return result;
+        // tslint:enable variable-name
+        return {
+            createBaseSourceFileNode: createBaseSourceFileNode,
+            createBaseIdentifierNode: createBaseIdentifierNode,
+            createBasePrivateIdentifierNode: createBasePrivateIdentifierNode,
+            createBaseTokenNode: createBaseTokenNode,
+            createBaseNode: createBaseNode
+        };
+        function createBaseSourceFileNode(kind) {
+            return new (SourceFileConstructor || (SourceFileConstructor = ts.objectAllocator.getSourceFileConstructor()))(kind, /*pos*/ -1, /*end*/ -1);
         }
-        Parser.parseSourceFile = parseSourceFile;
-        function parseIsolatedEntityName(content, languageVersion) {
-            // Choice of `isDeclarationFile` should be arbitrary
-            initializeState(content, languageVersion, /*syntaxCursor*/ undefined, 1 /* JS */);
-            // Prime the scanner.
-            nextToken();
-            var entityName = parseEntityName(/*allowReservedWords*/ true);
-            var isInvalid = token() === 1 /* EndOfFileToken */ && !parseDiagnostics.length;
-            clearState();
-            return isInvalid ? entityName : undefined;
+        function createBaseIdentifierNode(kind) {
+            return new (IdentifierConstructor || (IdentifierConstructor = ts.objectAllocator.getIdentifierConstructor()))(kind, /*pos*/ -1, /*end*/ -1);
         }
-        Parser.parseIsolatedEntityName = parseIsolatedEntityName;
-        function parseJsonText(fileName, sourceText, languageVersion, syntaxCursor, setParentNodes) {
-            if (languageVersion === void 0) { languageVersion = 2 /* ES2015 */; }
-            initializeState(sourceText, languageVersion, syntaxCursor, 6 /* JSON */);
-            // Set source file so that errors will be reported with this file name
-            sourceFile = createSourceFile(fileName, 2 /* ES2015 */, 6 /* JSON */, /*isDeclaration*/ false);
-            sourceFile.flags = contextFlags;
-            // Prime the scanner.
-            nextToken();
-            var pos = getNodePos();
-            if (token() === 1 /* EndOfFileToken */) {
-                sourceFile.statements = createNodeArray([], pos, pos);
-                sourceFile.endOfFileToken = parseTokenNode();
+        function createBasePrivateIdentifierNode(kind) {
+            return new (PrivateIdentifierConstructor || (PrivateIdentifierConstructor = ts.objectAllocator.getPrivateIdentifierConstructor()))(kind, /*pos*/ -1, /*end*/ -1);
+        }
+        function createBaseTokenNode(kind) {
+            return new (TokenConstructor || (TokenConstructor = ts.objectAllocator.getTokenConstructor()))(kind, /*pos*/ -1, /*end*/ -1);
+        }
+        function createBaseNode(kind) {
+            return new (NodeConstructor || (NodeConstructor = ts.objectAllocator.getNodeConstructor()))(kind, /*pos*/ -1, /*end*/ -1);
+        }
+    }
+    ts.createBaseNodeFactory = createBaseNodeFactory;
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    function createParenthesizerRules(factory) {
+        return {
+            parenthesizeLeftSideOfBinary: parenthesizeLeftSideOfBinary,
+            parenthesizeRightSideOfBinary: parenthesizeRightSideOfBinary,
+            parenthesizeExpressionOfComputedPropertyName: parenthesizeExpressionOfComputedPropertyName,
+            parenthesizeConditionOfConditionalExpression: parenthesizeConditionOfConditionalExpression,
+            parenthesizeBranchOfConditionalExpression: parenthesizeBranchOfConditionalExpression,
+            parenthesizeExpressionOfExportDefault: parenthesizeExpressionOfExportDefault,
+            parenthesizeExpressionOfNew: parenthesizeExpressionOfNew,
+            parenthesizeLeftSideOfAccess: parenthesizeLeftSideOfAccess,
+            parenthesizeOperandOfPostfixUnary: parenthesizeOperandOfPostfixUnary,
+            parenthesizeOperandOfPrefixUnary: parenthesizeOperandOfPrefixUnary,
+            parenthesizeExpressionsOfCommaDelimitedList: parenthesizeExpressionsOfCommaDelimitedList,
+            parenthesizeExpressionForDisallowedComma: parenthesizeExpressionForDisallowedComma,
+            parenthesizeExpressionOfExpressionStatement: parenthesizeExpressionOfExpressionStatement,
+            parenthesizeConciseBodyOfArrowFunction: parenthesizeConciseBodyOfArrowFunction,
+            parenthesizeMemberOfConditionalType: parenthesizeMemberOfConditionalType,
+            parenthesizeMemberOfElementType: parenthesizeMemberOfElementType,
+            parenthesizeElementTypeOfArrayType: parenthesizeElementTypeOfArrayType,
+            parenthesizeConstituentTypesOfUnionOrIntersectionType: parenthesizeConstituentTypesOfUnionOrIntersectionType,
+            parenthesizeTypeArguments: parenthesizeTypeArguments,
+        };
+        /**
+         * Determines whether the operand to a BinaryExpression needs to be parenthesized.
+         *
+         * @param binaryOperator The operator for the BinaryExpression.
+         * @param operand The operand for the BinaryExpression.
+         * @param isLeftSideOfBinary A value indicating whether the operand is the left side of the
+         *                           BinaryExpression.
+         */
+        function binaryOperandNeedsParentheses(binaryOperator, operand, isLeftSideOfBinary, leftOperand) {
+            // If the operand has lower precedence, then it needs to be parenthesized to preserve the
+            // intent of the expression. For example, if the operand is `a + b` and the operator is
+            // `*`, then we need to parenthesize the operand to preserve the intended order of
+            // operations: `(a + b) * x`.
+            //
+            // If the operand has higher precedence, then it does not need to be parenthesized. For
+            // example, if the operand is `a * b` and the operator is `+`, then we do not need to
+            // parenthesize to preserve the intended order of operations: `a * b + x`.
+            //
+            // If the operand has the same precedence, then we need to check the associativity of
+            // the operator based on whether this is the left or right operand of the expression.
+            //
+            // For example, if `a / d` is on the right of operator `*`, we need to parenthesize
+            // to preserve the intended order of operations: `x * (a / d)`
+            //
+            // If `a ** d` is on the left of operator `**`, we need to parenthesize to preserve
+            // the intended order of operations: `(a ** b) ** c`
+            var binaryOperatorPrecedence = ts.getOperatorPrecedence(216 /* BinaryExpression */, binaryOperator);
+            var binaryOperatorAssociativity = ts.getOperatorAssociativity(216 /* BinaryExpression */, binaryOperator);
+            var emittedOperand = ts.skipPartiallyEmittedExpressions(operand);
+            if (!isLeftSideOfBinary && operand.kind === 209 /* ArrowFunction */ && binaryOperatorPrecedence > 3 /* Assignment */) {
+                // We need to parenthesize arrow functions on the right side to avoid it being
+                // parsed as parenthesized expression: `a && (() => {})`
+                return true;
             }
-            else {
-                var statement = createNode(226 /* ExpressionStatement */);
-                switch (token()) {
-                    case 22 /* OpenBracketToken */:
-                        statement.expression = parseArrayLiteralExpression();
-                        break;
-                    case 106 /* TrueKeyword */:
-                    case 91 /* FalseKeyword */:
-                    case 100 /* NullKeyword */:
-                        statement.expression = parseTokenNode();
-                        break;
-                    case 40 /* MinusToken */:
-                        if (lookAhead(function () { return nextToken() === 8 /* NumericLiteral */ && nextToken() !== 58 /* ColonToken */; })) {
-                            statement.expression = parsePrefixUnaryExpression();
-                        }
-                        else {
-                            statement.expression = parseObjectLiteralExpression();
-                        }
-                        break;
-                    case 8 /* NumericLiteral */:
-                    case 10 /* StringLiteral */:
-                        if (lookAhead(function () { return nextToken() !== 58 /* ColonToken */; })) {
-                            statement.expression = parseLiteralNode();
-                            break;
+            var operandPrecedence = ts.getExpressionPrecedence(emittedOperand);
+            switch (ts.compareValues(operandPrecedence, binaryOperatorPrecedence)) {
+                case -1 /* LessThan */:
+                    // If the operand is the right side of a right-associative binary operation
+                    // and is a yield expression, then we do not need parentheses.
+                    if (!isLeftSideOfBinary
+                        && binaryOperatorAssociativity === 1 /* Right */
+                        && operand.kind === 219 /* YieldExpression */) {
+                        return false;
+                    }
+                    return true;
+                case 1 /* GreaterThan */:
+                    return false;
+                case 0 /* EqualTo */:
+                    if (isLeftSideOfBinary) {
+                        // No need to parenthesize the left operand when the binary operator is
+                        // left associative:
+                        //  (a*b)/x    -> a*b/x
+                        //  (a**b)/x   -> a**b/x
+                        //
+                        // Parentheses are needed for the left operand when the binary operator is
+                        // right associative:
+                        //  (a/b)**x   -> (a/b)**x
+                        //  (a**b)**x  -> (a**b)**x
+                        return binaryOperatorAssociativity === 1 /* Right */;
+                    }
+                    else {
+                        if (ts.isBinaryExpression(emittedOperand)
+                            && emittedOperand.operatorToken.kind === binaryOperator) {
+                            // No need to parenthesize the right operand when the binary operator and
+                            // operand are the same and one of the following:
+                            //  x*(a*b)     => x*a*b
+                            //  x|(a|b)     => x|a|b
+                            //  x&(a&b)     => x&a&b
+                            //  x^(a^b)     => x^a^b
+                            if (operatorHasAssociativeProperty(binaryOperator)) {
+                                return false;
+                            }
+                            // No need to parenthesize the right operand when the binary operator
+                            // is plus (+) if both the left and right operands consist solely of either
+                            // literals of the same kind or binary plus (+) expressions for literals of
+                            // the same kind (recursively).
+                            //  "a"+(1+2)       => "a"+(1+2)
+                            //  "a"+("b"+"c")   => "a"+"b"+"c"
+                            if (binaryOperator === 39 /* PlusToken */) {
+                                var leftKind = leftOperand ? getLiteralKindOfBinaryPlusOperand(leftOperand) : 0 /* Unknown */;
+                                if (ts.isLiteralKind(leftKind) && leftKind === getLiteralKindOfBinaryPlusOperand(emittedOperand)) {
+                                    return false;
+                                }
+                            }
                         }
-                    // falls through
-                    default:
-                        statement.expression = parseObjectLiteralExpression();
-                        break;
-                }
-                finishNode(statement);
-                sourceFile.statements = createNodeArray([statement], pos);
-                sourceFile.endOfFileToken = parseExpectedToken(1 /* EndOfFileToken */, ts.Diagnostics.Unexpected_token);
-            }
-            if (setParentNodes) {
-                fixupParentReferences(sourceFile);
+                        // No need to parenthesize the right operand when the operand is right
+                        // associative:
+                        //  x/(a**b)    -> x/a**b
+                        //  x**(a**b)   -> x**a**b
+                        //
+                        // Parentheses are needed for the right operand when the operand is left
+                        // associative:
+                        //  x/(a*b)     -> x/(a*b)
+                        //  x**(a/b)    -> x**(a/b)
+                        var operandAssociativity = ts.getExpressionAssociativity(emittedOperand);
+                        return operandAssociativity === 0 /* Left */;
+                    }
             }
-            sourceFile.nodeCount = nodeCount;
-            sourceFile.identifierCount = identifierCount;
-            sourceFile.identifiers = identifiers;
-            sourceFile.parseDiagnostics = parseDiagnostics;
-            var result = sourceFile;
-            clearState();
-            return result;
-        }
-        Parser.parseJsonText = parseJsonText;
-        function getLanguageVariant(scriptKind) {
-            // .tsx and .jsx files are treated as jsx language variant.
-            return scriptKind === 4 /* TSX */ || scriptKind === 2 /* JSX */ || scriptKind === 1 /* JS */ || scriptKind === 6 /* JSON */ ? 1 /* JSX */ : 0 /* Standard */;
         }
-        function initializeState(_sourceText, languageVersion, _syntaxCursor, scriptKind) {
-            NodeConstructor = ts.objectAllocator.getNodeConstructor();
-            TokenConstructor = ts.objectAllocator.getTokenConstructor();
-            IdentifierConstructor = ts.objectAllocator.getIdentifierConstructor();
-            PrivateIdentifierConstructor = ts.objectAllocator.getPrivateIdentifierConstructor();
-            SourceFileConstructor = ts.objectAllocator.getSourceFileConstructor();
-            sourceText = _sourceText;
-            syntaxCursor = _syntaxCursor;
-            parseDiagnostics = [];
-            parsingContext = 0;
-            identifiers = ts.createMap();
-            privateIdentifiers = ts.createMap();
-            identifierCount = 0;
-            nodeCount = 0;
-            switch (scriptKind) {
-                case 1 /* JS */:
-                case 2 /* JSX */:
-                    contextFlags = 131072 /* JavaScriptFile */;
-                    break;
-                case 6 /* JSON */:
-                    contextFlags = 131072 /* JavaScriptFile */ | 33554432 /* JsonFile */;
-                    break;
-                default:
-                    contextFlags = 0 /* None */;
-                    break;
+        /**
+         * Determines whether a binary operator is mathematically associative.
+         *
+         * @param binaryOperator The binary operator.
+         */
+        function operatorHasAssociativeProperty(binaryOperator) {
+            // The following operators are associative in JavaScript:
+            //  (a*b)*c     -> a*(b*c)  -> a*b*c
+            //  (a|b)|c     -> a|(b|c)  -> a|b|c
+            //  (a&b)&c     -> a&(b&c)  -> a&b&c
+            //  (a^b)^c     -> a^(b^c)  -> a^b^c
+            //
+            // While addition is associative in mathematics, JavaScript's `+` is not
+            // guaranteed to be associative as it is overloaded with string concatenation.
+            return binaryOperator === 41 /* AsteriskToken */
+                || binaryOperator === 51 /* BarToken */
+                || binaryOperator === 50 /* AmpersandToken */
+                || binaryOperator === 52 /* CaretToken */;
+        }
+        /**
+         * This function determines whether an expression consists of a homogeneous set of
+         * literal expressions or binary plus expressions that all share the same literal kind.
+         * It is used to determine whether the right-hand operand of a binary plus expression can be
+         * emitted without parentheses.
+         */
+        function getLiteralKindOfBinaryPlusOperand(node) {
+            node = ts.skipPartiallyEmittedExpressions(node);
+            if (ts.isLiteralKind(node.kind)) {
+                return node.kind;
+            }
+            if (node.kind === 216 /* BinaryExpression */ && node.operatorToken.kind === 39 /* PlusToken */) {
+                if (node.cachedLiteralKind !== undefined) {
+                    return node.cachedLiteralKind;
+                }
+                var leftKind = getLiteralKindOfBinaryPlusOperand(node.left);
+                var literalKind = ts.isLiteralKind(leftKind)
+                    && leftKind === getLiteralKindOfBinaryPlusOperand(node.right)
+                    ? leftKind
+                    : 0 /* Unknown */;
+                node.cachedLiteralKind = literalKind;
+                return literalKind;
             }
-            parseErrorBeforeNextFinishedNode = false;
-            // Initialize and prime the scanner before parsing the source elements.
-            scanner.setText(sourceText);
-            scanner.setOnError(scanError);
-            scanner.setScriptTarget(languageVersion);
-            scanner.setLanguageVariant(getLanguageVariant(scriptKind));
-        }
-        function clearState() {
-            // Clear out the text the scanner is pointing at, so it doesn't keep anything alive unnecessarily.
-            scanner.clearCommentDirectives();
-            scanner.setText("");
-            scanner.setOnError(undefined);
-            // Clear any data.  We don't want to accidentally hold onto it for too long.
-            parseDiagnostics = undefined;
-            sourceFile = undefined;
-            identifiers = undefined;
-            syntaxCursor = undefined;
-            sourceText = undefined;
-            notParenthesizedArrow = undefined;
+            return 0 /* Unknown */;
         }
-        function parseSourceFileWorker(fileName, languageVersion, setParentNodes, scriptKind) {
-            var isDeclarationFile = isDeclarationFileName(fileName);
-            if (isDeclarationFile) {
-                contextFlags |= 8388608 /* Ambient */;
+        /**
+         * Wraps the operand to a BinaryExpression in parentheses if they are needed to preserve the intended
+         * order of operations.
+         *
+         * @param binaryOperator The operator for the BinaryExpression.
+         * @param operand The operand for the BinaryExpression.
+         * @param isLeftSideOfBinary A value indicating whether the operand is the left side of the
+         *                           BinaryExpression.
+         */
+        function parenthesizeBinaryOperand(binaryOperator, operand, isLeftSideOfBinary, leftOperand) {
+            var skipped = ts.skipPartiallyEmittedExpressions(operand);
+            // If the resulting expression is already parenthesized, we do not need to do any further processing.
+            if (skipped.kind === 207 /* ParenthesizedExpression */) {
+                return operand;
+            }
+            return binaryOperandNeedsParentheses(binaryOperator, operand, isLeftSideOfBinary, leftOperand)
+                ? factory.createParenthesizedExpression(operand)
+                : operand;
+        }
+        function parenthesizeLeftSideOfBinary(binaryOperator, leftSide) {
+            return parenthesizeBinaryOperand(binaryOperator, leftSide, /*isLeftSideOfBinary*/ true);
+        }
+        function parenthesizeRightSideOfBinary(binaryOperator, leftSide, rightSide) {
+            return parenthesizeBinaryOperand(binaryOperator, rightSide, /*isLeftSideOfBinary*/ false, leftSide);
+        }
+        function parenthesizeExpressionOfComputedPropertyName(expression) {
+            return ts.isCommaSequence(expression) ? factory.createParenthesizedExpression(expression) : expression;
+        }
+        function parenthesizeConditionOfConditionalExpression(condition) {
+            var conditionalPrecedence = ts.getOperatorPrecedence(217 /* ConditionalExpression */, 57 /* QuestionToken */);
+            var emittedCondition = ts.skipPartiallyEmittedExpressions(condition);
+            var conditionPrecedence = ts.getExpressionPrecedence(emittedCondition);
+            if (ts.compareValues(conditionPrecedence, conditionalPrecedence) !== 1 /* GreaterThan */) {
+                return factory.createParenthesizedExpression(condition);
+            }
+            return condition;
+        }
+        function parenthesizeBranchOfConditionalExpression(branch) {
+            // per ES grammar both 'whenTrue' and 'whenFalse' parts of conditional expression are assignment expressions
+            // so in case when comma expression is introduced as a part of previous transformations
+            // if should be wrapped in parens since comma operator has the lowest precedence
+            var emittedExpression = ts.skipPartiallyEmittedExpressions(branch);
+            return ts.isCommaSequence(emittedExpression)
+                ? factory.createParenthesizedExpression(branch)
+                : branch;
+        }
+        /**
+         *  [Per the spec](https://tc39.github.io/ecma262/#prod-ExportDeclaration), `export default` accepts _AssigmentExpression_ but
+         *  has a lookahead restriction for `function`, `async function`, and `class`.
+         *
+         * Basically, that means we need to parenthesize in the following cases:
+         *
+         * - BinaryExpression of CommaToken
+         * - CommaList (synthetic list of multiple comma expressions)
+         * - FunctionExpression
+         * - ClassExpression
+         */
+        function parenthesizeExpressionOfExportDefault(expression) {
+            var check = ts.skipPartiallyEmittedExpressions(expression);
+            var needsParens = ts.isCommaSequence(check);
+            if (!needsParens) {
+                switch (ts.getLeftmostExpression(check, /*stopAtCallExpression*/ false).kind) {
+                    case 221 /* ClassExpression */:
+                    case 208 /* FunctionExpression */:
+                        needsParens = true;
+                }
             }
-            sourceFile = createSourceFile(fileName, languageVersion, scriptKind, isDeclarationFile);
-            sourceFile.flags = contextFlags;
-            // Prime the scanner.
-            nextToken();
-            // A member of ReadonlyArray<T> isn't assignable to a member of T[] (and prevents a direct cast) - but this is where we set up those members so they can be readonly in the future
-            processCommentPragmas(sourceFile, sourceText);
-            processPragmasIntoFields(sourceFile, reportPragmaDiagnostic);
-            sourceFile.statements = parseList(0 /* SourceElements */, parseStatement);
-            ts.Debug.assert(token() === 1 /* EndOfFileToken */);
-            sourceFile.endOfFileToken = addJSDocComment(parseTokenNode());
-            setExternalModuleIndicator(sourceFile);
-            sourceFile.commentDirectives = scanner.getCommentDirectives();
-            sourceFile.nodeCount = nodeCount;
-            sourceFile.identifierCount = identifierCount;
-            sourceFile.identifiers = identifiers;
-            sourceFile.parseDiagnostics = parseDiagnostics;
-            if (setParentNodes) {
-                fixupParentReferences(sourceFile);
+            return needsParens ? factory.createParenthesizedExpression(expression) : expression;
+        }
+        /**
+         * Wraps an expression in parentheses if it is needed in order to use the expression
+         * as the expression of a `NewExpression` node.
+         */
+        function parenthesizeExpressionOfNew(expression) {
+            var leftmostExpr = ts.getLeftmostExpression(expression, /*stopAtCallExpressions*/ true);
+            switch (leftmostExpr.kind) {
+                case 203 /* CallExpression */:
+                    return factory.createParenthesizedExpression(expression);
+                case 204 /* NewExpression */:
+                    return !leftmostExpr.arguments
+                        ? factory.createParenthesizedExpression(expression)
+                        : expression; // TODO(rbuckton): Verify this assertion holds
             }
-            return sourceFile;
-            function reportPragmaDiagnostic(pos, end, diagnostic) {
-                parseDiagnostics.push(ts.createFileDiagnostic(sourceFile, pos, end, diagnostic));
+            return parenthesizeLeftSideOfAccess(expression);
+        }
+        /**
+         * Wraps an expression in parentheses if it is needed in order to use the expression for
+         * property or element access.
+         */
+        function parenthesizeLeftSideOfAccess(expression) {
+            // isLeftHandSideExpression is almost the correct criterion for when it is not necessary
+            // to parenthesize the expression before a dot. The known exception is:
+            //
+            //    NewExpression:
+            //       new C.x        -> not the same as (new C).x
+            //
+            var emittedExpression = ts.skipPartiallyEmittedExpressions(expression);
+            if (ts.isLeftHandSideExpression(emittedExpression)
+                && (emittedExpression.kind !== 204 /* NewExpression */ || emittedExpression.arguments)) {
+                // TODO(rbuckton): Verify whether this assertion holds.
+                return expression;
             }
+            // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+            return ts.setTextRange(factory.createParenthesizedExpression(expression), expression);
         }
-        function addJSDocComment(node) {
-            ts.Debug.assert(!node.jsDoc); // Should only be called once per node
-            var jsDoc = ts.mapDefined(ts.getJSDocCommentRanges(node, sourceFile.text), function (comment) { return JSDocParser.parseJSDocComment(node, comment.pos, comment.end - comment.pos); });
-            if (jsDoc.length)
-                node.jsDoc = jsDoc;
-            return node;
+        function parenthesizeOperandOfPostfixUnary(operand) {
+            // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+            return ts.isLeftHandSideExpression(operand) ? operand : ts.setTextRange(factory.createParenthesizedExpression(operand), operand);
         }
-        function fixupParentReferences(rootNode) {
-            // normally parent references are set during binding. However, for clients that only need
-            // a syntax tree, and no semantic features, then the binding process is an unnecessary
-            // overhead.  This functions allows us to set all the parents, without all the expense of
-            // binding.
-            forEachChildRecursively(rootNode, bindParentToChild);
-            function bindParentToChild(child, parent) {
-                child.parent = parent;
-                if (ts.hasJSDocNodes(child)) {
-                    for (var _i = 0, _a = child.jsDoc; _i < _a.length; _i++) {
-                        var doc = _a[_i];
-                        bindParentToChild(doc, child);
-                        forEachChildRecursively(doc, bindParentToChild);
-                    }
-                }
-            }
+        function parenthesizeOperandOfPrefixUnary(operand) {
+            // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+            return ts.isUnaryExpression(operand) ? operand : ts.setTextRange(factory.createParenthesizedExpression(operand), operand);
         }
-        Parser.fixupParentReferences = fixupParentReferences;
-        function createSourceFile(fileName, languageVersion, scriptKind, isDeclarationFile) {
-            // code from createNode is inlined here so createNode won't have to deal with special case of creating source files
-            // this is quite rare comparing to other nodes and createNode should be as fast as possible
-            var sourceFile = new SourceFileConstructor(290 /* SourceFile */, /*pos*/ 0, /* end */ sourceText.length);
-            nodeCount++;
-            sourceFile.text = sourceText;
-            sourceFile.bindDiagnostics = [];
-            sourceFile.bindSuggestionDiagnostics = undefined;
-            sourceFile.languageVersion = languageVersion;
-            sourceFile.fileName = ts.normalizePath(fileName);
-            sourceFile.languageVariant = getLanguageVariant(scriptKind);
-            sourceFile.isDeclarationFile = isDeclarationFile;
-            sourceFile.scriptKind = scriptKind;
-            return sourceFile;
+        function parenthesizeExpressionsOfCommaDelimitedList(elements) {
+            var result = ts.sameMap(elements, parenthesizeExpressionForDisallowedComma);
+            return ts.setTextRange(factory.createNodeArray(result, elements.hasTrailingComma), elements);
         }
-        function setContextFlag(val, flag) {
-            if (val) {
-                contextFlags |= flag;
+        function parenthesizeExpressionForDisallowedComma(expression) {
+            var emittedExpression = ts.skipPartiallyEmittedExpressions(expression);
+            var expressionPrecedence = ts.getExpressionPrecedence(emittedExpression);
+            var commaPrecedence = ts.getOperatorPrecedence(216 /* BinaryExpression */, 27 /* CommaToken */);
+            // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+            return expressionPrecedence > commaPrecedence ? expression : ts.setTextRange(factory.createParenthesizedExpression(expression), expression);
+        }
+        function parenthesizeExpressionOfExpressionStatement(expression) {
+            var emittedExpression = ts.skipPartiallyEmittedExpressions(expression);
+            if (ts.isCallExpression(emittedExpression)) {
+                var callee = emittedExpression.expression;
+                var kind = ts.skipPartiallyEmittedExpressions(callee).kind;
+                if (kind === 208 /* FunctionExpression */ || kind === 209 /* ArrowFunction */) {
+                    // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+                    var updated = factory.updateCallExpression(emittedExpression, ts.setTextRange(factory.createParenthesizedExpression(callee), callee), emittedExpression.typeArguments, emittedExpression.arguments);
+                    return factory.restoreOuterExpressions(expression, updated, 8 /* PartiallyEmittedExpressions */);
+                }
             }
-            else {
-                contextFlags &= ~flag;
+            var leftmostExpressionKind = ts.getLeftmostExpression(emittedExpression, /*stopAtCallExpressions*/ false).kind;
+            if (leftmostExpressionKind === 200 /* ObjectLiteralExpression */ || leftmostExpressionKind === 208 /* FunctionExpression */) {
+                // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+                return ts.setTextRange(factory.createParenthesizedExpression(expression), expression);
             }
+            return expression;
         }
-        function setDisallowInContext(val) {
-            setContextFlag(val, 4096 /* DisallowInContext */);
+        function parenthesizeConciseBodyOfArrowFunction(body) {
+            if (!ts.isBlock(body) && (ts.isCommaSequence(body) || ts.getLeftmostExpression(body, /*stopAtCallExpressions*/ false).kind === 200 /* ObjectLiteralExpression */)) {
+                // TODO(rbuckton): Verifiy whether `setTextRange` is needed.
+                return ts.setTextRange(factory.createParenthesizedExpression(body), body);
+            }
+            return body;
         }
-        function setYieldContext(val) {
-            setContextFlag(val, 8192 /* YieldContext */);
+        function parenthesizeMemberOfConditionalType(member) {
+            return member.kind === 184 /* ConditionalType */ ? factory.createParenthesizedType(member) : member;
         }
-        function setDecoratorContext(val) {
-            setContextFlag(val, 16384 /* DecoratorContext */);
+        function parenthesizeMemberOfElementType(member) {
+            switch (member.kind) {
+                case 182 /* UnionType */:
+                case 183 /* IntersectionType */:
+                case 174 /* FunctionType */:
+                case 175 /* ConstructorType */:
+                    return factory.createParenthesizedType(member);
+            }
+            return parenthesizeMemberOfConditionalType(member);
         }
-        function setAwaitContext(val) {
-            setContextFlag(val, 32768 /* AwaitContext */);
+        function parenthesizeElementTypeOfArrayType(member) {
+            switch (member.kind) {
+                case 176 /* TypeQuery */:
+                case 188 /* TypeOperator */:
+                case 185 /* InferType */:
+                    return factory.createParenthesizedType(member);
+            }
+            return parenthesizeMemberOfElementType(member);
+        }
+        function parenthesizeConstituentTypesOfUnionOrIntersectionType(members) {
+            return factory.createNodeArray(ts.sameMap(members, parenthesizeMemberOfElementType));
+        }
+        function parenthesizeOrdinalTypeArgument(node, i) {
+            return i === 0 && ts.isFunctionOrConstructorTypeNode(node) && node.typeParameters ? factory.createParenthesizedType(node) : node;
+        }
+        function parenthesizeTypeArguments(typeArguments) {
+            if (ts.some(typeArguments)) {
+                return factory.createNodeArray(ts.sameMap(typeArguments, parenthesizeOrdinalTypeArgument));
+            }
+        }
+    }
+    ts.createParenthesizerRules = createParenthesizerRules;
+    ts.nullParenthesizerRules = {
+        parenthesizeLeftSideOfBinary: function (_binaryOperator, leftSide) { return leftSide; },
+        parenthesizeRightSideOfBinary: function (_binaryOperator, _leftSide, rightSide) { return rightSide; },
+        parenthesizeExpressionOfComputedPropertyName: ts.identity,
+        parenthesizeConditionOfConditionalExpression: ts.identity,
+        parenthesizeBranchOfConditionalExpression: ts.identity,
+        parenthesizeExpressionOfExportDefault: ts.identity,
+        parenthesizeExpressionOfNew: function (expression) { return ts.cast(expression, ts.isLeftHandSideExpression); },
+        parenthesizeLeftSideOfAccess: function (expression) { return ts.cast(expression, ts.isLeftHandSideExpression); },
+        parenthesizeOperandOfPostfixUnary: function (operand) { return ts.cast(operand, ts.isLeftHandSideExpression); },
+        parenthesizeOperandOfPrefixUnary: function (operand) { return ts.cast(operand, ts.isUnaryExpression); },
+        parenthesizeExpressionsOfCommaDelimitedList: function (nodes) { return ts.cast(nodes, ts.isNodeArray); },
+        parenthesizeExpressionForDisallowedComma: ts.identity,
+        parenthesizeExpressionOfExpressionStatement: ts.identity,
+        parenthesizeConciseBodyOfArrowFunction: ts.identity,
+        parenthesizeMemberOfConditionalType: ts.identity,
+        parenthesizeMemberOfElementType: ts.identity,
+        parenthesizeElementTypeOfArrayType: ts.identity,
+        parenthesizeConstituentTypesOfUnionOrIntersectionType: function (nodes) { return ts.cast(nodes, ts.isNodeArray); },
+        parenthesizeTypeArguments: function (nodes) { return nodes && ts.cast(nodes, ts.isNodeArray); },
+    };
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    function createNodeConverters(factory) {
+        return {
+            convertToFunctionBlock: convertToFunctionBlock,
+            convertToFunctionExpression: convertToFunctionExpression,
+            convertToArrayAssignmentElement: convertToArrayAssignmentElement,
+            convertToObjectAssignmentElement: convertToObjectAssignmentElement,
+            convertToAssignmentPattern: convertToAssignmentPattern,
+            convertToObjectAssignmentPattern: convertToObjectAssignmentPattern,
+            convertToArrayAssignmentPattern: convertToArrayAssignmentPattern,
+            convertToAssignmentElementTarget: convertToAssignmentElementTarget,
+        };
+        function convertToFunctionBlock(node, multiLine) {
+            if (ts.isBlock(node))
+                return node;
+            var returnStatement = factory.createReturnStatement(node);
+            ts.setTextRange(returnStatement, node);
+            var body = factory.createBlock([returnStatement], multiLine);
+            ts.setTextRange(body, node);
+            return body;
         }
-        function doOutsideOfContext(context, func) {
-            // contextFlagsToClear will contain only the context flags that are
-            // currently set that we need to temporarily clear
-            // We don't just blindly reset to the previous flags to ensure
-            // that we do not mutate cached flags for the incremental
-            // parser (ThisNodeHasError, ThisNodeOrAnySubNodesHasError, and
-            // HasAggregatedChildData).
-            var contextFlagsToClear = context & contextFlags;
-            if (contextFlagsToClear) {
-                // clear the requested context flags
-                setContextFlag(/*val*/ false, contextFlagsToClear);
-                var result = func();
-                // restore the context flags we just cleared
-                setContextFlag(/*val*/ true, contextFlagsToClear);
-                return result;
+        function convertToFunctionExpression(node) {
+            if (!node.body)
+                return ts.Debug.fail("Cannot convert a FunctionDeclaration without a body");
+            var updated = factory.createFunctionExpression(node.modifiers, node.asteriskToken, node.name, node.typeParameters, node.parameters, node.type, node.body);
+            ts.setOriginalNode(updated, node);
+            ts.setTextRange(updated, node);
+            if (ts.getStartsOnNewLine(node)) {
+                ts.setStartsOnNewLine(updated, /*newLine*/ true);
             }
-            // no need to do anything special as we are not in any of the requested contexts
-            return func();
+            return updated;
         }
-        function doInsideOfContext(context, func) {
-            // contextFlagsToSet will contain only the context flags that
-            // are not currently set that we need to temporarily enable.
-            // We don't just blindly reset to the previous flags to ensure
-            // that we do not mutate cached flags for the incremental
-            // parser (ThisNodeHasError, ThisNodeOrAnySubNodesHasError, and
-            // HasAggregatedChildData).
-            var contextFlagsToSet = context & ~contextFlags;
-            if (contextFlagsToSet) {
-                // set the requested context flags
-                setContextFlag(/*val*/ true, contextFlagsToSet);
-                var result = func();
-                // reset the context flags we just set
-                setContextFlag(/*val*/ false, contextFlagsToSet);
-                return result;
+        function convertToArrayAssignmentElement(element) {
+            if (ts.isBindingElement(element)) {
+                if (element.dotDotDotToken) {
+                    ts.Debug.assertNode(element.name, ts.isIdentifier);
+                    return ts.setOriginalNode(ts.setTextRange(factory.createSpreadElement(element.name), element), element);
+                }
+                var expression = convertToAssignmentElementTarget(element.name);
+                return element.initializer
+                    ? ts.setOriginalNode(ts.setTextRange(factory.createAssignment(expression, element.initializer), element), element)
+                    : expression;
             }
-            // no need to do anything special as we are already in all of the requested contexts
-            return func();
+            return ts.cast(element, ts.isExpression);
         }
-        function allowInAnd(func) {
-            return doOutsideOfContext(4096 /* DisallowInContext */, func);
+        function convertToObjectAssignmentElement(element) {
+            if (ts.isBindingElement(element)) {
+                if (element.dotDotDotToken) {
+                    ts.Debug.assertNode(element.name, ts.isIdentifier);
+                    return ts.setOriginalNode(ts.setTextRange(factory.createSpreadAssignment(element.name), element), element);
+                }
+                if (element.propertyName) {
+                    var expression = convertToAssignmentElementTarget(element.name);
+                    return ts.setOriginalNode(ts.setTextRange(factory.createPropertyAssignment(element.propertyName, element.initializer ? factory.createAssignment(expression, element.initializer) : expression), element), element);
+                }
+                ts.Debug.assertNode(element.name, ts.isIdentifier);
+                return ts.setOriginalNode(ts.setTextRange(factory.createShorthandPropertyAssignment(element.name, element.initializer), element), element);
+            }
+            return ts.cast(element, ts.isObjectLiteralElementLike);
         }
-        function disallowInAnd(func) {
-            return doInsideOfContext(4096 /* DisallowInContext */, func);
+        function convertToAssignmentPattern(node) {
+            switch (node.kind) {
+                case 197 /* ArrayBindingPattern */:
+                case 199 /* ArrayLiteralExpression */:
+                    return convertToArrayAssignmentPattern(node);
+                case 196 /* ObjectBindingPattern */:
+                case 200 /* ObjectLiteralExpression */:
+                    return convertToObjectAssignmentPattern(node);
+            }
         }
-        function doInYieldContext(func) {
-            return doInsideOfContext(8192 /* YieldContext */, func);
+        function convertToObjectAssignmentPattern(node) {
+            if (ts.isObjectBindingPattern(node)) {
+                return ts.setOriginalNode(ts.setTextRange(factory.createObjectLiteralExpression(ts.map(node.elements, convertToObjectAssignmentElement)), node), node);
+            }
+            return ts.cast(node, ts.isObjectLiteralExpression);
         }
-        function doInDecoratorContext(func) {
-            return doInsideOfContext(16384 /* DecoratorContext */, func);
+        function convertToArrayAssignmentPattern(node) {
+            if (ts.isArrayBindingPattern(node)) {
+                return ts.setOriginalNode(ts.setTextRange(factory.createArrayLiteralExpression(ts.map(node.elements, convertToArrayAssignmentElement)), node), node);
+            }
+            return ts.cast(node, ts.isArrayLiteralExpression);
         }
-        function doInAwaitContext(func) {
-            return doInsideOfContext(32768 /* AwaitContext */, func);
+        function convertToAssignmentElementTarget(node) {
+            if (ts.isBindingPattern(node)) {
+                return convertToAssignmentPattern(node);
+            }
+            return ts.cast(node, ts.isExpression);
         }
-        function doOutsideOfAwaitContext(func) {
-            return doOutsideOfContext(32768 /* AwaitContext */, func);
+    }
+    ts.createNodeConverters = createNodeConverters;
+    ts.nullNodeConverters = {
+        convertToFunctionBlock: ts.notImplemented,
+        convertToFunctionExpression: ts.notImplemented,
+        convertToArrayAssignmentElement: ts.notImplemented,
+        convertToObjectAssignmentElement: ts.notImplemented,
+        convertToAssignmentPattern: ts.notImplemented,
+        convertToObjectAssignmentPattern: ts.notImplemented,
+        convertToArrayAssignmentPattern: ts.notImplemented,
+        convertToAssignmentElementTarget: ts.notImplemented,
+    };
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    var nextAutoGenerateId = 0;
+    /* @internal */
+    var NodeFactoryFlags;
+    (function (NodeFactoryFlags) {
+        NodeFactoryFlags[NodeFactoryFlags["None"] = 0] = "None";
+        // Disables the parenthesizer rules for the factory.
+        NodeFactoryFlags[NodeFactoryFlags["NoParenthesizerRules"] = 1] = "NoParenthesizerRules";
+        // Disables the node converters for the factory.
+        NodeFactoryFlags[NodeFactoryFlags["NoNodeConverters"] = 2] = "NoNodeConverters";
+        // Ensures new `PropertyAccessExpression` nodes are created with the `NoIndentation` emit flag set.
+        NodeFactoryFlags[NodeFactoryFlags["NoIndentationOnFreshPropertyAccess"] = 4] = "NoIndentationOnFreshPropertyAccess";
+        // Do not set an `original` pointer when updating a node.
+        NodeFactoryFlags[NodeFactoryFlags["NoOriginalNode"] = 8] = "NoOriginalNode";
+    })(NodeFactoryFlags = ts.NodeFactoryFlags || (ts.NodeFactoryFlags = {}));
+    /**
+     * Creates a `NodeFactory` that can be used to create and update a syntax tree.
+     * @param flags Flags that control factory behavior.
+     * @param baseFactory A `BaseNodeFactory` used to create the base `Node` objects.
+     */
+    /* @internal */
+    function createNodeFactory(flags, baseFactory) {
+        var update = flags & 8 /* NoOriginalNode */ ? updateWithoutOriginal : updateWithOriginal;
+        // Lazily load the parenthesizer, node converters, and some factory methods until they are used.
+        var parenthesizerRules = ts.memoize(function () { return flags & 1 /* NoParenthesizerRules */ ? ts.nullParenthesizerRules : ts.createParenthesizerRules(factory); });
+        var converters = ts.memoize(function () { return flags & 2 /* NoNodeConverters */ ? ts.nullNodeConverters : ts.createNodeConverters(factory); });
+        // lazy initializaton of common operator factories
+        var getBinaryCreateFunction = ts.memoizeOne(function (operator) { return function (left, right) { return createBinaryExpression(left, operator, right); }; });
+        var getPrefixUnaryCreateFunction = ts.memoizeOne(function (operator) { return function (operand) { return createPrefixUnaryExpression(operator, operand); }; });
+        var getPostfixUnaryCreateFunction = ts.memoizeOne(function (operator) { return function (operand) { return createPostfixUnaryExpression(operand, operator); }; });
+        var getJSDocPrimaryTypeCreateFunction = ts.memoizeOne(function (kind) { return function () { return createJSDocPrimaryTypeWorker(kind); }; });
+        var getJSDocUnaryTypeCreateFunction = ts.memoizeOne(function (kind) { return function (type) { return createJSDocUnaryTypeWorker(kind, type); }; });
+        var getJSDocUnaryTypeUpdateFunction = ts.memoizeOne(function (kind) { return function (node, type) { return updateJSDocUnaryTypeWorker(kind, node, type); }; });
+        var getJSDocSimpleTagCreateFunction = ts.memoizeOne(function (kind) { return function (tagName, comment) { return createJSDocSimpleTagWorker(kind, tagName, comment); }; });
+        var getJSDocSimpleTagUpdateFunction = ts.memoizeOne(function (kind) { return function (node, tagName, comment) { return updateJSDocSimpleTagWorker(kind, node, tagName, comment); }; });
+        var getJSDocTypeLikeTagCreateFunction = ts.memoizeOne(function (kind) { return function (tagName, typeExpression, comment) { return createJSDocTypeLikeTagWorker(kind, tagName, typeExpression, comment); }; });
+        var getJSDocTypeLikeTagUpdateFunction = ts.memoizeOne(function (kind) { return function (node, tagName, typeExpression, comment) { return updateJSDocTypeLikeTagWorker(kind, node, tagName, typeExpression, comment); }; });
+        var factory = {
+            get parenthesizer() { return parenthesizerRules(); },
+            get converters() { return converters(); },
+            createNodeArray: createNodeArray,
+            createNumericLiteral: createNumericLiteral,
+            createBigIntLiteral: createBigIntLiteral,
+            createStringLiteral: createStringLiteral,
+            createStringLiteralFromNode: createStringLiteralFromNode,
+            createRegularExpressionLiteral: createRegularExpressionLiteral,
+            createLiteralLikeNode: createLiteralLikeNode,
+            createIdentifier: createIdentifier,
+            updateIdentifier: updateIdentifier,
+            createTempVariable: createTempVariable,
+            createLoopVariable: createLoopVariable,
+            createUniqueName: createUniqueName,
+            getGeneratedNameForNode: getGeneratedNameForNode,
+            createPrivateIdentifier: createPrivateIdentifier,
+            createToken: createToken,
+            createSuper: createSuper,
+            createThis: createThis,
+            createNull: createNull,
+            createTrue: createTrue,
+            createFalse: createFalse,
+            createModifier: createModifier,
+            createModifiersFromModifierFlags: createModifiersFromModifierFlags,
+            createQualifiedName: createQualifiedName,
+            updateQualifiedName: updateQualifiedName,
+            createComputedPropertyName: createComputedPropertyName,
+            updateComputedPropertyName: updateComputedPropertyName,
+            createTypeParameterDeclaration: createTypeParameterDeclaration,
+            updateTypeParameterDeclaration: updateTypeParameterDeclaration,
+            createParameterDeclaration: createParameterDeclaration,
+            updateParameterDeclaration: updateParameterDeclaration,
+            createDecorator: createDecorator,
+            updateDecorator: updateDecorator,
+            createPropertySignature: createPropertySignature,
+            updatePropertySignature: updatePropertySignature,
+            createPropertyDeclaration: createPropertyDeclaration,
+            updatePropertyDeclaration: updatePropertyDeclaration,
+            createMethodSignature: createMethodSignature,
+            updateMethodSignature: updateMethodSignature,
+            createMethodDeclaration: createMethodDeclaration,
+            updateMethodDeclaration: updateMethodDeclaration,
+            createConstructorDeclaration: createConstructorDeclaration,
+            updateConstructorDeclaration: updateConstructorDeclaration,
+            createGetAccessorDeclaration: createGetAccessorDeclaration,
+            updateGetAccessorDeclaration: updateGetAccessorDeclaration,
+            createSetAccessorDeclaration: createSetAccessorDeclaration,
+            updateSetAccessorDeclaration: updateSetAccessorDeclaration,
+            createCallSignature: createCallSignature,
+            updateCallSignature: updateCallSignature,
+            createConstructSignature: createConstructSignature,
+            updateConstructSignature: updateConstructSignature,
+            createIndexSignature: createIndexSignature,
+            updateIndexSignature: updateIndexSignature,
+            createTemplateLiteralTypeSpan: createTemplateLiteralTypeSpan,
+            updateTemplateLiteralTypeSpan: updateTemplateLiteralTypeSpan,
+            createKeywordTypeNode: createKeywordTypeNode,
+            createTypePredicateNode: createTypePredicateNode,
+            updateTypePredicateNode: updateTypePredicateNode,
+            createTypeReferenceNode: createTypeReferenceNode,
+            updateTypeReferenceNode: updateTypeReferenceNode,
+            createFunctionTypeNode: createFunctionTypeNode,
+            updateFunctionTypeNode: updateFunctionTypeNode,
+            createConstructorTypeNode: createConstructorTypeNode,
+            updateConstructorTypeNode: updateConstructorTypeNode,
+            createTypeQueryNode: createTypeQueryNode,
+            updateTypeQueryNode: updateTypeQueryNode,
+            createTypeLiteralNode: createTypeLiteralNode,
+            updateTypeLiteralNode: updateTypeLiteralNode,
+            createArrayTypeNode: createArrayTypeNode,
+            updateArrayTypeNode: updateArrayTypeNode,
+            createTupleTypeNode: createTupleTypeNode,
+            updateTupleTypeNode: updateTupleTypeNode,
+            createNamedTupleMember: createNamedTupleMember,
+            updateNamedTupleMember: updateNamedTupleMember,
+            createOptionalTypeNode: createOptionalTypeNode,
+            updateOptionalTypeNode: updateOptionalTypeNode,
+            createRestTypeNode: createRestTypeNode,
+            updateRestTypeNode: updateRestTypeNode,
+            createUnionTypeNode: createUnionTypeNode,
+            updateUnionTypeNode: updateUnionTypeNode,
+            createIntersectionTypeNode: createIntersectionTypeNode,
+            updateIntersectionTypeNode: updateIntersectionTypeNode,
+            createConditionalTypeNode: createConditionalTypeNode,
+            updateConditionalTypeNode: updateConditionalTypeNode,
+            createInferTypeNode: createInferTypeNode,
+            updateInferTypeNode: updateInferTypeNode,
+            createImportTypeNode: createImportTypeNode,
+            updateImportTypeNode: updateImportTypeNode,
+            createParenthesizedType: createParenthesizedType,
+            updateParenthesizedType: updateParenthesizedType,
+            createThisTypeNode: createThisTypeNode,
+            createTypeOperatorNode: createTypeOperatorNode,
+            updateTypeOperatorNode: updateTypeOperatorNode,
+            createIndexedAccessTypeNode: createIndexedAccessTypeNode,
+            updateIndexedAccessTypeNode: updateIndexedAccessTypeNode,
+            createMappedTypeNode: createMappedTypeNode,
+            updateMappedTypeNode: updateMappedTypeNode,
+            createLiteralTypeNode: createLiteralTypeNode,
+            updateLiteralTypeNode: updateLiteralTypeNode,
+            createTemplateLiteralType: createTemplateLiteralType,
+            updateTemplateLiteralType: updateTemplateLiteralType,
+            createObjectBindingPattern: createObjectBindingPattern,
+            updateObjectBindingPattern: updateObjectBindingPattern,
+            createArrayBindingPattern: createArrayBindingPattern,
+            updateArrayBindingPattern: updateArrayBindingPattern,
+            createBindingElement: createBindingElement,
+            updateBindingElement: updateBindingElement,
+            createArrayLiteralExpression: createArrayLiteralExpression,
+            updateArrayLiteralExpression: updateArrayLiteralExpression,
+            createObjectLiteralExpression: createObjectLiteralExpression,
+            updateObjectLiteralExpression: updateObjectLiteralExpression,
+            createPropertyAccessExpression: flags & 4 /* NoIndentationOnFreshPropertyAccess */ ?
+                function (expression, name) { return ts.setEmitFlags(createPropertyAccessExpression(expression, name), 131072 /* NoIndentation */); } :
+                createPropertyAccessExpression,
+            updatePropertyAccessExpression: updatePropertyAccessExpression,
+            createPropertyAccessChain: flags & 4 /* NoIndentationOnFreshPropertyAccess */ ?
+                function (expression, questionDotToken, name) { return ts.setEmitFlags(createPropertyAccessChain(expression, questionDotToken, name), 131072 /* NoIndentation */); } :
+                createPropertyAccessChain,
+            updatePropertyAccessChain: updatePropertyAccessChain,
+            createElementAccessExpression: createElementAccessExpression,
+            updateElementAccessExpression: updateElementAccessExpression,
+            createElementAccessChain: createElementAccessChain,
+            updateElementAccessChain: updateElementAccessChain,
+            createCallExpression: createCallExpression,
+            updateCallExpression: updateCallExpression,
+            createCallChain: createCallChain,
+            updateCallChain: updateCallChain,
+            createNewExpression: createNewExpression,
+            updateNewExpression: updateNewExpression,
+            createTaggedTemplateExpression: createTaggedTemplateExpression,
+            updateTaggedTemplateExpression: updateTaggedTemplateExpression,
+            createTypeAssertion: createTypeAssertion,
+            updateTypeAssertion: updateTypeAssertion,
+            createParenthesizedExpression: createParenthesizedExpression,
+            updateParenthesizedExpression: updateParenthesizedExpression,
+            createFunctionExpression: createFunctionExpression,
+            updateFunctionExpression: updateFunctionExpression,
+            createArrowFunction: createArrowFunction,
+            updateArrowFunction: updateArrowFunction,
+            createDeleteExpression: createDeleteExpression,
+            updateDeleteExpression: updateDeleteExpression,
+            createTypeOfExpression: createTypeOfExpression,
+            updateTypeOfExpression: updateTypeOfExpression,
+            createVoidExpression: createVoidExpression,
+            updateVoidExpression: updateVoidExpression,
+            createAwaitExpression: createAwaitExpression,
+            updateAwaitExpression: updateAwaitExpression,
+            createPrefixUnaryExpression: createPrefixUnaryExpression,
+            updatePrefixUnaryExpression: updatePrefixUnaryExpression,
+            createPostfixUnaryExpression: createPostfixUnaryExpression,
+            updatePostfixUnaryExpression: updatePostfixUnaryExpression,
+            createBinaryExpression: createBinaryExpression,
+            updateBinaryExpression: updateBinaryExpression,
+            createConditionalExpression: createConditionalExpression,
+            updateConditionalExpression: updateConditionalExpression,
+            createTemplateExpression: createTemplateExpression,
+            updateTemplateExpression: updateTemplateExpression,
+            createTemplateHead: createTemplateHead,
+            createTemplateMiddle: createTemplateMiddle,
+            createTemplateTail: createTemplateTail,
+            createNoSubstitutionTemplateLiteral: createNoSubstitutionTemplateLiteral,
+            createTemplateLiteralLikeNode: createTemplateLiteralLikeNode,
+            createYieldExpression: createYieldExpression,
+            updateYieldExpression: updateYieldExpression,
+            createSpreadElement: createSpreadElement,
+            updateSpreadElement: updateSpreadElement,
+            createClassExpression: createClassExpression,
+            updateClassExpression: updateClassExpression,
+            createOmittedExpression: createOmittedExpression,
+            createExpressionWithTypeArguments: createExpressionWithTypeArguments,
+            updateExpressionWithTypeArguments: updateExpressionWithTypeArguments,
+            createAsExpression: createAsExpression,
+            updateAsExpression: updateAsExpression,
+            createNonNullExpression: createNonNullExpression,
+            updateNonNullExpression: updateNonNullExpression,
+            createNonNullChain: createNonNullChain,
+            updateNonNullChain: updateNonNullChain,
+            createMetaProperty: createMetaProperty,
+            updateMetaProperty: updateMetaProperty,
+            createTemplateSpan: createTemplateSpan,
+            updateTemplateSpan: updateTemplateSpan,
+            createSemicolonClassElement: createSemicolonClassElement,
+            createBlock: createBlock,
+            updateBlock: updateBlock,
+            createVariableStatement: createVariableStatement,
+            updateVariableStatement: updateVariableStatement,
+            createEmptyStatement: createEmptyStatement,
+            createExpressionStatement: createExpressionStatement,
+            updateExpressionStatement: updateExpressionStatement,
+            createIfStatement: createIfStatement,
+            updateIfStatement: updateIfStatement,
+            createDoStatement: createDoStatement,
+            updateDoStatement: updateDoStatement,
+            createWhileStatement: createWhileStatement,
+            updateWhileStatement: updateWhileStatement,
+            createForStatement: createForStatement,
+            updateForStatement: updateForStatement,
+            createForInStatement: createForInStatement,
+            updateForInStatement: updateForInStatement,
+            createForOfStatement: createForOfStatement,
+            updateForOfStatement: updateForOfStatement,
+            createContinueStatement: createContinueStatement,
+            updateContinueStatement: updateContinueStatement,
+            createBreakStatement: createBreakStatement,
+            updateBreakStatement: updateBreakStatement,
+            createReturnStatement: createReturnStatement,
+            updateReturnStatement: updateReturnStatement,
+            createWithStatement: createWithStatement,
+            updateWithStatement: updateWithStatement,
+            createSwitchStatement: createSwitchStatement,
+            updateSwitchStatement: updateSwitchStatement,
+            createLabeledStatement: createLabeledStatement,
+            updateLabeledStatement: updateLabeledStatement,
+            createThrowStatement: createThrowStatement,
+            updateThrowStatement: updateThrowStatement,
+            createTryStatement: createTryStatement,
+            updateTryStatement: updateTryStatement,
+            createDebuggerStatement: createDebuggerStatement,
+            createVariableDeclaration: createVariableDeclaration,
+            updateVariableDeclaration: updateVariableDeclaration,
+            createVariableDeclarationList: createVariableDeclarationList,
+            updateVariableDeclarationList: updateVariableDeclarationList,
+            createFunctionDeclaration: createFunctionDeclaration,
+            updateFunctionDeclaration: updateFunctionDeclaration,
+            createClassDeclaration: createClassDeclaration,
+            updateClassDeclaration: updateClassDeclaration,
+            createInterfaceDeclaration: createInterfaceDeclaration,
+            updateInterfaceDeclaration: updateInterfaceDeclaration,
+            createTypeAliasDeclaration: createTypeAliasDeclaration,
+            updateTypeAliasDeclaration: updateTypeAliasDeclaration,
+            createEnumDeclaration: createEnumDeclaration,
+            updateEnumDeclaration: updateEnumDeclaration,
+            createModuleDeclaration: createModuleDeclaration,
+            updateModuleDeclaration: updateModuleDeclaration,
+            createModuleBlock: createModuleBlock,
+            updateModuleBlock: updateModuleBlock,
+            createCaseBlock: createCaseBlock,
+            updateCaseBlock: updateCaseBlock,
+            createNamespaceExportDeclaration: createNamespaceExportDeclaration,
+            updateNamespaceExportDeclaration: updateNamespaceExportDeclaration,
+            createImportEqualsDeclaration: createImportEqualsDeclaration,
+            updateImportEqualsDeclaration: updateImportEqualsDeclaration,
+            createImportDeclaration: createImportDeclaration,
+            updateImportDeclaration: updateImportDeclaration,
+            createImportClause: createImportClause,
+            updateImportClause: updateImportClause,
+            createNamespaceImport: createNamespaceImport,
+            updateNamespaceImport: updateNamespaceImport,
+            createNamespaceExport: createNamespaceExport,
+            updateNamespaceExport: updateNamespaceExport,
+            createNamedImports: createNamedImports,
+            updateNamedImports: updateNamedImports,
+            createImportSpecifier: createImportSpecifier,
+            updateImportSpecifier: updateImportSpecifier,
+            createExportAssignment: createExportAssignment,
+            updateExportAssignment: updateExportAssignment,
+            createExportDeclaration: createExportDeclaration,
+            updateExportDeclaration: updateExportDeclaration,
+            createNamedExports: createNamedExports,
+            updateNamedExports: updateNamedExports,
+            createExportSpecifier: createExportSpecifier,
+            updateExportSpecifier: updateExportSpecifier,
+            createMissingDeclaration: createMissingDeclaration,
+            createExternalModuleReference: createExternalModuleReference,
+            updateExternalModuleReference: updateExternalModuleReference,
+            // lazily load factory members for JSDoc types with similar structure
+            get createJSDocAllType() { return getJSDocPrimaryTypeCreateFunction(303 /* JSDocAllType */); },
+            get createJSDocUnknownType() { return getJSDocPrimaryTypeCreateFunction(304 /* JSDocUnknownType */); },
+            get createJSDocNonNullableType() { return getJSDocUnaryTypeCreateFunction(306 /* JSDocNonNullableType */); },
+            get updateJSDocNonNullableType() { return getJSDocUnaryTypeUpdateFunction(306 /* JSDocNonNullableType */); },
+            get createJSDocNullableType() { return getJSDocUnaryTypeCreateFunction(305 /* JSDocNullableType */); },
+            get updateJSDocNullableType() { return getJSDocUnaryTypeUpdateFunction(305 /* JSDocNullableType */); },
+            get createJSDocOptionalType() { return getJSDocUnaryTypeCreateFunction(307 /* JSDocOptionalType */); },
+            get updateJSDocOptionalType() { return getJSDocUnaryTypeUpdateFunction(307 /* JSDocOptionalType */); },
+            get createJSDocVariadicType() { return getJSDocUnaryTypeCreateFunction(309 /* JSDocVariadicType */); },
+            get updateJSDocVariadicType() { return getJSDocUnaryTypeUpdateFunction(309 /* JSDocVariadicType */); },
+            get createJSDocNamepathType() { return getJSDocUnaryTypeCreateFunction(310 /* JSDocNamepathType */); },
+            get updateJSDocNamepathType() { return getJSDocUnaryTypeUpdateFunction(310 /* JSDocNamepathType */); },
+            createJSDocFunctionType: createJSDocFunctionType,
+            updateJSDocFunctionType: updateJSDocFunctionType,
+            createJSDocTypeLiteral: createJSDocTypeLiteral,
+            updateJSDocTypeLiteral: updateJSDocTypeLiteral,
+            createJSDocTypeExpression: createJSDocTypeExpression,
+            updateJSDocTypeExpression: updateJSDocTypeExpression,
+            createJSDocSignature: createJSDocSignature,
+            updateJSDocSignature: updateJSDocSignature,
+            createJSDocTemplateTag: createJSDocTemplateTag,
+            updateJSDocTemplateTag: updateJSDocTemplateTag,
+            createJSDocTypedefTag: createJSDocTypedefTag,
+            updateJSDocTypedefTag: updateJSDocTypedefTag,
+            createJSDocParameterTag: createJSDocParameterTag,
+            updateJSDocParameterTag: updateJSDocParameterTag,
+            createJSDocPropertyTag: createJSDocPropertyTag,
+            updateJSDocPropertyTag: updateJSDocPropertyTag,
+            createJSDocCallbackTag: createJSDocCallbackTag,
+            updateJSDocCallbackTag: updateJSDocCallbackTag,
+            createJSDocAugmentsTag: createJSDocAugmentsTag,
+            updateJSDocAugmentsTag: updateJSDocAugmentsTag,
+            createJSDocImplementsTag: createJSDocImplementsTag,
+            updateJSDocImplementsTag: updateJSDocImplementsTag,
+            createJSDocSeeTag: createJSDocSeeTag,
+            updateJSDocSeeTag: updateJSDocSeeTag,
+            createJSDocNameReference: createJSDocNameReference,
+            updateJSDocNameReference: updateJSDocNameReference,
+            // lazily load factory members for JSDoc tags with similar structure
+            get createJSDocTypeTag() { return getJSDocTypeLikeTagCreateFunction(329 /* JSDocTypeTag */); },
+            get updateJSDocTypeTag() { return getJSDocTypeLikeTagUpdateFunction(329 /* JSDocTypeTag */); },
+            get createJSDocReturnTag() { return getJSDocTypeLikeTagCreateFunction(327 /* JSDocReturnTag */); },
+            get updateJSDocReturnTag() { return getJSDocTypeLikeTagUpdateFunction(327 /* JSDocReturnTag */); },
+            get createJSDocThisTag() { return getJSDocTypeLikeTagCreateFunction(328 /* JSDocThisTag */); },
+            get updateJSDocThisTag() { return getJSDocTypeLikeTagUpdateFunction(328 /* JSDocThisTag */); },
+            get createJSDocEnumTag() { return getJSDocTypeLikeTagCreateFunction(325 /* JSDocEnumTag */); },
+            get updateJSDocEnumTag() { return getJSDocTypeLikeTagUpdateFunction(325 /* JSDocEnumTag */); },
+            get createJSDocAuthorTag() { return getJSDocSimpleTagCreateFunction(317 /* JSDocAuthorTag */); },
+            get updateJSDocAuthorTag() { return getJSDocSimpleTagUpdateFunction(317 /* JSDocAuthorTag */); },
+            get createJSDocClassTag() { return getJSDocSimpleTagCreateFunction(319 /* JSDocClassTag */); },
+            get updateJSDocClassTag() { return getJSDocSimpleTagUpdateFunction(319 /* JSDocClassTag */); },
+            get createJSDocPublicTag() { return getJSDocSimpleTagCreateFunction(320 /* JSDocPublicTag */); },
+            get updateJSDocPublicTag() { return getJSDocSimpleTagUpdateFunction(320 /* JSDocPublicTag */); },
+            get createJSDocPrivateTag() { return getJSDocSimpleTagCreateFunction(321 /* JSDocPrivateTag */); },
+            get updateJSDocPrivateTag() { return getJSDocSimpleTagUpdateFunction(321 /* JSDocPrivateTag */); },
+            get createJSDocProtectedTag() { return getJSDocSimpleTagCreateFunction(322 /* JSDocProtectedTag */); },
+            get updateJSDocProtectedTag() { return getJSDocSimpleTagUpdateFunction(322 /* JSDocProtectedTag */); },
+            get createJSDocReadonlyTag() { return getJSDocSimpleTagCreateFunction(323 /* JSDocReadonlyTag */); },
+            get updateJSDocReadonlyTag() { return getJSDocSimpleTagUpdateFunction(323 /* JSDocReadonlyTag */); },
+            get createJSDocDeprecatedTag() { return getJSDocSimpleTagCreateFunction(318 /* JSDocDeprecatedTag */); },
+            get updateJSDocDeprecatedTag() { return getJSDocSimpleTagUpdateFunction(318 /* JSDocDeprecatedTag */); },
+            createJSDocUnknownTag: createJSDocUnknownTag,
+            updateJSDocUnknownTag: updateJSDocUnknownTag,
+            createJSDocComment: createJSDocComment,
+            updateJSDocComment: updateJSDocComment,
+            createJsxElement: createJsxElement,
+            updateJsxElement: updateJsxElement,
+            createJsxSelfClosingElement: createJsxSelfClosingElement,
+            updateJsxSelfClosingElement: updateJsxSelfClosingElement,
+            createJsxOpeningElement: createJsxOpeningElement,
+            updateJsxOpeningElement: updateJsxOpeningElement,
+            createJsxClosingElement: createJsxClosingElement,
+            updateJsxClosingElement: updateJsxClosingElement,
+            createJsxFragment: createJsxFragment,
+            createJsxText: createJsxText,
+            updateJsxText: updateJsxText,
+            createJsxOpeningFragment: createJsxOpeningFragment,
+            createJsxJsxClosingFragment: createJsxJsxClosingFragment,
+            updateJsxFragment: updateJsxFragment,
+            createJsxAttribute: createJsxAttribute,
+            updateJsxAttribute: updateJsxAttribute,
+            createJsxAttributes: createJsxAttributes,
+            updateJsxAttributes: updateJsxAttributes,
+            createJsxSpreadAttribute: createJsxSpreadAttribute,
+            updateJsxSpreadAttribute: updateJsxSpreadAttribute,
+            createJsxExpression: createJsxExpression,
+            updateJsxExpression: updateJsxExpression,
+            createCaseClause: createCaseClause,
+            updateCaseClause: updateCaseClause,
+            createDefaultClause: createDefaultClause,
+            updateDefaultClause: updateDefaultClause,
+            createHeritageClause: createHeritageClause,
+            updateHeritageClause: updateHeritageClause,
+            createCatchClause: createCatchClause,
+            updateCatchClause: updateCatchClause,
+            createPropertyAssignment: createPropertyAssignment,
+            updatePropertyAssignment: updatePropertyAssignment,
+            createShorthandPropertyAssignment: createShorthandPropertyAssignment,
+            updateShorthandPropertyAssignment: updateShorthandPropertyAssignment,
+            createSpreadAssignment: createSpreadAssignment,
+            updateSpreadAssignment: updateSpreadAssignment,
+            createEnumMember: createEnumMember,
+            updateEnumMember: updateEnumMember,
+            createSourceFile: createSourceFile,
+            updateSourceFile: updateSourceFile,
+            createBundle: createBundle,
+            updateBundle: updateBundle,
+            createUnparsedSource: createUnparsedSource,
+            createUnparsedPrologue: createUnparsedPrologue,
+            createUnparsedPrepend: createUnparsedPrepend,
+            createUnparsedTextLike: createUnparsedTextLike,
+            createUnparsedSyntheticReference: createUnparsedSyntheticReference,
+            createInputFiles: createInputFiles,
+            createSyntheticExpression: createSyntheticExpression,
+            createSyntaxList: createSyntaxList,
+            createNotEmittedStatement: createNotEmittedStatement,
+            createPartiallyEmittedExpression: createPartiallyEmittedExpression,
+            updatePartiallyEmittedExpression: updatePartiallyEmittedExpression,
+            createCommaListExpression: createCommaListExpression,
+            updateCommaListExpression: updateCommaListExpression,
+            createEndOfDeclarationMarker: createEndOfDeclarationMarker,
+            createMergeDeclarationMarker: createMergeDeclarationMarker,
+            createSyntheticReferenceExpression: createSyntheticReferenceExpression,
+            updateSyntheticReferenceExpression: updateSyntheticReferenceExpression,
+            cloneNode: cloneNode,
+            // Lazily load factory methods for common operator factories and utilities
+            get createComma() { return getBinaryCreateFunction(27 /* CommaToken */); },
+            get createAssignment() { return getBinaryCreateFunction(62 /* EqualsToken */); },
+            get createLogicalOr() { return getBinaryCreateFunction(56 /* BarBarToken */); },
+            get createLogicalAnd() { return getBinaryCreateFunction(55 /* AmpersandAmpersandToken */); },
+            get createBitwiseOr() { return getBinaryCreateFunction(51 /* BarToken */); },
+            get createBitwiseXor() { return getBinaryCreateFunction(52 /* CaretToken */); },
+            get createBitwiseAnd() { return getBinaryCreateFunction(50 /* AmpersandToken */); },
+            get createStrictEquality() { return getBinaryCreateFunction(36 /* EqualsEqualsEqualsToken */); },
+            get createStrictInequality() { return getBinaryCreateFunction(37 /* ExclamationEqualsEqualsToken */); },
+            get createEquality() { return getBinaryCreateFunction(34 /* EqualsEqualsToken */); },
+            get createInequality() { return getBinaryCreateFunction(35 /* ExclamationEqualsToken */); },
+            get createLessThan() { return getBinaryCreateFunction(29 /* LessThanToken */); },
+            get createLessThanEquals() { return getBinaryCreateFunction(32 /* LessThanEqualsToken */); },
+            get createGreaterThan() { return getBinaryCreateFunction(31 /* GreaterThanToken */); },
+            get createGreaterThanEquals() { return getBinaryCreateFunction(33 /* GreaterThanEqualsToken */); },
+            get createLeftShift() { return getBinaryCreateFunction(47 /* LessThanLessThanToken */); },
+            get createRightShift() { return getBinaryCreateFunction(48 /* GreaterThanGreaterThanToken */); },
+            get createUnsignedRightShift() { return getBinaryCreateFunction(49 /* GreaterThanGreaterThanGreaterThanToken */); },
+            get createAdd() { return getBinaryCreateFunction(39 /* PlusToken */); },
+            get createSubtract() { return getBinaryCreateFunction(40 /* MinusToken */); },
+            get createMultiply() { return getBinaryCreateFunction(41 /* AsteriskToken */); },
+            get createDivide() { return getBinaryCreateFunction(43 /* SlashToken */); },
+            get createModulo() { return getBinaryCreateFunction(44 /* PercentToken */); },
+            get createExponent() { return getBinaryCreateFunction(42 /* AsteriskAsteriskToken */); },
+            get createPrefixPlus() { return getPrefixUnaryCreateFunction(39 /* PlusToken */); },
+            get createPrefixMinus() { return getPrefixUnaryCreateFunction(40 /* MinusToken */); },
+            get createPrefixIncrement() { return getPrefixUnaryCreateFunction(45 /* PlusPlusToken */); },
+            get createPrefixDecrement() { return getPrefixUnaryCreateFunction(46 /* MinusMinusToken */); },
+            get createBitwiseNot() { return getPrefixUnaryCreateFunction(54 /* TildeToken */); },
+            get createLogicalNot() { return getPrefixUnaryCreateFunction(53 /* ExclamationToken */); },
+            get createPostfixIncrement() { return getPostfixUnaryCreateFunction(45 /* PlusPlusToken */); },
+            get createPostfixDecrement() { return getPostfixUnaryCreateFunction(46 /* MinusMinusToken */); },
+            // Compound nodes
+            createImmediatelyInvokedFunctionExpression: createImmediatelyInvokedFunctionExpression,
+            createImmediatelyInvokedArrowFunction: createImmediatelyInvokedArrowFunction,
+            createVoidZero: createVoidZero,
+            createExportDefault: createExportDefault,
+            createExternalModuleExport: createExternalModuleExport,
+            createTypeCheck: createTypeCheck,
+            createMethodCall: createMethodCall,
+            createGlobalMethodCall: createGlobalMethodCall,
+            createFunctionBindCall: createFunctionBindCall,
+            createFunctionCallCall: createFunctionCallCall,
+            createFunctionApplyCall: createFunctionApplyCall,
+            createArraySliceCall: createArraySliceCall,
+            createArrayConcatCall: createArrayConcatCall,
+            createObjectDefinePropertyCall: createObjectDefinePropertyCall,
+            createPropertyDescriptor: createPropertyDescriptor,
+            createCallBinding: createCallBinding,
+            // Utilities
+            inlineExpressions: inlineExpressions,
+            getInternalName: getInternalName,
+            getLocalName: getLocalName,
+            getExportName: getExportName,
+            getDeclarationName: getDeclarationName,
+            getNamespaceMemberName: getNamespaceMemberName,
+            getExternalModuleOrNamespaceExportName: getExternalModuleOrNamespaceExportName,
+            restoreOuterExpressions: restoreOuterExpressions,
+            restoreEnclosingLabel: restoreEnclosingLabel,
+            createUseStrictPrologue: createUseStrictPrologue,
+            copyPrologue: copyPrologue,
+            copyStandardPrologue: copyStandardPrologue,
+            copyCustomPrologue: copyCustomPrologue,
+            ensureUseStrict: ensureUseStrict,
+            liftToBlock: liftToBlock,
+            mergeLexicalEnvironment: mergeLexicalEnvironment,
+            updateModifiers: updateModifiers,
+        };
+        return factory;
+        // @api
+        function createNodeArray(elements, hasTrailingComma) {
+            if (elements === undefined || elements === ts.emptyArray) {
+                elements = [];
+            }
+            else if (ts.isNodeArray(elements)) {
+                // Ensure the transform flags have been aggregated for this NodeArray
+                if (elements.transformFlags === undefined) {
+                    aggregateChildrenFlags(elements);
+                }
+                ts.Debug.attachNodeArrayDebugInfo(elements);
+                return elements;
+            }
+            // Since the element list of a node array is typically created by starting with an empty array and
+            // repeatedly calling push(), the list may not have the optimal memory layout. We invoke slice() for
+            // small arrays (1 to 4 elements) to give the VM a chance to allocate an optimal representation.
+            var length = elements.length;
+            var array = (length >= 1 && length <= 4 ? elements.slice() : elements);
+            ts.setTextRangePosEnd(array, -1, -1);
+            array.hasTrailingComma = !!hasTrailingComma;
+            aggregateChildrenFlags(array);
+            ts.Debug.attachNodeArrayDebugInfo(array);
+            return array;
         }
-        function doInYieldAndAwaitContext(func) {
-            return doInsideOfContext(8192 /* YieldContext */ | 32768 /* AwaitContext */, func);
+        function createBaseNode(kind) {
+            return baseFactory.createBaseNode(kind);
+        }
+        function createBaseDeclaration(kind, decorators, modifiers) {
+            var node = createBaseNode(kind);
+            node.decorators = asNodeArray(decorators);
+            node.modifiers = asNodeArray(modifiers);
+            node.transformFlags |=
+                propagateChildrenFlags(node.decorators) |
+                    propagateChildrenFlags(node.modifiers);
+            // NOTE: The following properties are commonly set by the binder and are added here to
+            // ensure declarations have a stable shape.
+            node.symbol = undefined; // initialized by binder
+            node.localSymbol = undefined; // initialized by binder
+            node.locals = undefined; // initialized by binder
+            node.nextContainer = undefined; // initialized by binder
+            return node;
         }
-        function doOutsideOfYieldAndAwaitContext(func) {
-            return doOutsideOfContext(8192 /* YieldContext */ | 32768 /* AwaitContext */, func);
+        function createBaseNamedDeclaration(kind, decorators, modifiers, name) {
+            var node = createBaseDeclaration(kind, decorators, modifiers);
+            name = asName(name);
+            node.name = name;
+            // The PropertyName of a member is allowed to be `await`.
+            // We don't need to exclude `await` for type signatures since types
+            // don't propagate child flags.
+            if (name) {
+                switch (node.kind) {
+                    case 165 /* MethodDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 163 /* PropertyDeclaration */:
+                    case 288 /* PropertyAssignment */:
+                        if (ts.isIdentifier(name)) {
+                            node.transformFlags |= propagateIdentifierNameFlags(name);
+                            break;
+                        }
+                    // fall through
+                    default:
+                        node.transformFlags |= propagateChildFlags(name);
+                        break;
+                }
+            }
+            return node;
         }
-        function inContext(flags) {
-            return (contextFlags & flags) !== 0;
+        function createBaseGenericNamedDeclaration(kind, decorators, modifiers, name, typeParameters) {
+            var node = createBaseNamedDeclaration(kind, decorators, modifiers, name);
+            node.typeParameters = asNodeArray(typeParameters);
+            node.transformFlags |= propagateChildrenFlags(node.typeParameters);
+            if (typeParameters)
+                node.transformFlags |= 1 /* ContainsTypeScript */;
+            return node;
         }
-        function inYieldContext() {
-            return inContext(8192 /* YieldContext */);
+        function createBaseSignatureDeclaration(kind, decorators, modifiers, name, typeParameters, parameters, type) {
+            var node = createBaseGenericNamedDeclaration(kind, decorators, modifiers, name, typeParameters);
+            node.parameters = createNodeArray(parameters);
+            node.type = type;
+            node.transformFlags |=
+                propagateChildrenFlags(node.parameters) |
+                    propagateChildFlags(node.type);
+            if (type)
+                node.transformFlags |= 1 /* ContainsTypeScript */;
+            return node;
         }
-        function inDisallowInContext() {
-            return inContext(4096 /* DisallowInContext */);
+        function updateBaseSignatureDeclaration(updated, original) {
+            // copy children used only for error reporting
+            if (original.typeArguments)
+                updated.typeArguments = original.typeArguments;
+            return update(updated, original);
         }
-        function inDecoratorContext() {
-            return inContext(16384 /* DecoratorContext */);
+        function createBaseFunctionLikeDeclaration(kind, decorators, modifiers, name, typeParameters, parameters, type, body) {
+            var node = createBaseSignatureDeclaration(kind, decorators, modifiers, name, typeParameters, parameters, type);
+            node.body = body;
+            node.transformFlags |= propagateChildFlags(node.body) & ~8388608 /* ContainsPossibleTopLevelAwait */;
+            if (!body)
+                node.transformFlags |= 1 /* ContainsTypeScript */;
+            return node;
         }
-        function inAwaitContext() {
-            return inContext(32768 /* AwaitContext */);
+        function updateBaseFunctionLikeDeclaration(updated, original) {
+            // copy children used only for error reporting
+            if (original.exclamationToken)
+                updated.exclamationToken = original.exclamationToken;
+            if (original.typeArguments)
+                updated.typeArguments = original.typeArguments;
+            return updateBaseSignatureDeclaration(updated, original);
+        }
+        function createBaseInterfaceOrClassLikeDeclaration(kind, decorators, modifiers, name, typeParameters, heritageClauses) {
+            var node = createBaseGenericNamedDeclaration(kind, decorators, modifiers, name, typeParameters);
+            node.heritageClauses = asNodeArray(heritageClauses);
+            node.transformFlags |= propagateChildrenFlags(node.heritageClauses);
+            return node;
         }
-        function parseErrorAtCurrentToken(message, arg0) {
-            parseErrorAt(scanner.getTokenPos(), scanner.getTextPos(), message, arg0);
+        function createBaseClassLikeDeclaration(kind, decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            var node = createBaseInterfaceOrClassLikeDeclaration(kind, decorators, modifiers, name, typeParameters, heritageClauses);
+            node.members = createNodeArray(members);
+            node.transformFlags |= propagateChildrenFlags(node.members);
+            return node;
         }
-        function parseErrorAtPosition(start, length, message, arg0) {
-            // Don't report another error if it would just be at the same position as the last error.
-            var lastError = ts.lastOrUndefined(parseDiagnostics);
-            if (!lastError || start !== lastError.start) {
-                parseDiagnostics.push(ts.createFileDiagnostic(sourceFile, start, length, message, arg0));
-            }
-            // Mark that we've encountered an error.  We'll set an appropriate bit on the next
-            // node we finish so that it can't be reused incrementally.
-            parseErrorBeforeNextFinishedNode = true;
+        function createBaseBindingLikeDeclaration(kind, decorators, modifiers, name, initializer) {
+            var node = createBaseNamedDeclaration(kind, decorators, modifiers, name);
+            node.initializer = initializer;
+            node.transformFlags |= propagateChildFlags(node.initializer);
+            return node;
         }
-        function parseErrorAt(start, end, message, arg0) {
-            parseErrorAtPosition(start, end - start, message, arg0);
+        function createBaseVariableLikeDeclaration(kind, decorators, modifiers, name, type, initializer) {
+            var node = createBaseBindingLikeDeclaration(kind, decorators, modifiers, name, initializer);
+            node.type = type;
+            node.transformFlags |= propagateChildFlags(type);
+            if (type)
+                node.transformFlags |= 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseErrorAtRange(range, message, arg0) {
-            parseErrorAt(range.pos, range.end, message, arg0);
+        //
+        // Literals
+        //
+        function createBaseLiteral(kind, text) {
+            var node = createBaseToken(kind);
+            node.text = text;
+            return node;
         }
-        function scanError(message, length) {
-            parseErrorAtPosition(scanner.getTextPos(), length, message);
+        // @api
+        function createNumericLiteral(value, numericLiteralFlags) {
+            if (numericLiteralFlags === void 0) { numericLiteralFlags = 0 /* None */; }
+            var node = createBaseLiteral(8 /* NumericLiteral */, typeof value === "number" ? value + "" : value);
+            node.numericLiteralFlags = numericLiteralFlags;
+            if (numericLiteralFlags & 384 /* BinaryOrOctalSpecifier */)
+                node.transformFlags |= 256 /* ContainsES2015 */;
+            return node;
         }
-        function getNodePos() {
-            return scanner.getStartPos();
+        // @api
+        function createBigIntLiteral(value) {
+            var node = createBaseLiteral(9 /* BigIntLiteral */, typeof value === "string" ? value : ts.pseudoBigIntToString(value) + "n");
+            node.transformFlags |= 4 /* ContainsESNext */;
+            return node;
         }
-        // Use this function to access the current token instead of reading the currentToken
-        // variable. Since function results aren't narrowed in control flow analysis, this ensures
-        // that the type checker doesn't make wrong assumptions about the type of the current
-        // token (e.g. a call to nextToken() changes the current token but the checker doesn't
-        // reason about this side effect).  Mainstream VMs inline simple functions like this, so
-        // there is no performance penalty.
-        function token() {
-            return currentToken;
+        function createBaseStringLiteral(text, isSingleQuote) {
+            var node = createBaseLiteral(10 /* StringLiteral */, text);
+            node.singleQuote = isSingleQuote;
+            return node;
         }
-        function nextTokenWithoutCheck() {
-            return currentToken = scanner.scan();
+        // @api
+        function createStringLiteral(text, isSingleQuote, hasExtendedUnicodeEscape) {
+            var node = createBaseStringLiteral(text, isSingleQuote);
+            node.hasExtendedUnicodeEscape = hasExtendedUnicodeEscape;
+            if (hasExtendedUnicodeEscape)
+                node.transformFlags |= 256 /* ContainsES2015 */;
+            return node;
         }
-        function nextToken() {
-            // if the keyword had an escape
-            if (ts.isKeyword(currentToken) && (scanner.hasUnicodeEscape() || scanner.hasExtendedUnicodeEscape())) {
-                // issue a parse error for the escape
-                parseErrorAt(scanner.getTokenPos(), scanner.getTextPos(), ts.Diagnostics.Keywords_cannot_contain_escape_characters);
-            }
-            return nextTokenWithoutCheck();
+        // @api
+        function createStringLiteralFromNode(sourceNode) {
+            var node = createBaseStringLiteral(ts.getTextOfIdentifierOrLiteral(sourceNode), /*isSingleQuote*/ undefined);
+            node.textSourceNode = sourceNode;
+            return node;
         }
-        function nextTokenJSDoc() {
-            return currentToken = scanner.scanJsDocToken();
+        // @api
+        function createRegularExpressionLiteral(text) {
+            var node = createBaseLiteral(13 /* RegularExpressionLiteral */, text);
+            return node;
         }
-        function reScanGreaterToken() {
-            return currentToken = scanner.reScanGreaterToken();
+        // @api
+        function createLiteralLikeNode(kind, text) {
+            switch (kind) {
+                case 8 /* NumericLiteral */: return createNumericLiteral(text, /*numericLiteralFlags*/ 0);
+                case 9 /* BigIntLiteral */: return createBigIntLiteral(text);
+                case 10 /* StringLiteral */: return createStringLiteral(text, /*isSingleQuote*/ undefined);
+                case 11 /* JsxText */: return createJsxText(text, /*containsOnlyTriviaWhiteSpaces*/ false);
+                case 12 /* JsxTextAllWhiteSpaces */: return createJsxText(text, /*containsOnlyTriviaWhiteSpaces*/ true);
+                case 13 /* RegularExpressionLiteral */: return createRegularExpressionLiteral(text);
+                case 14 /* NoSubstitutionTemplateLiteral */: return createTemplateLiteralLikeNode(kind, text, /*rawText*/ undefined, /*templateFlags*/ 0);
+            }
         }
-        function reScanSlashToken() {
-            return currentToken = scanner.reScanSlashToken();
+        //
+        // Identifiers
+        //
+        function createBaseIdentifier(text, originalKeywordKind) {
+            if (originalKeywordKind === undefined && text) {
+                originalKeywordKind = ts.stringToToken(text);
+            }
+            if (originalKeywordKind === 78 /* Identifier */) {
+                originalKeywordKind = undefined;
+            }
+            var node = baseFactory.createBaseIdentifierNode(78 /* Identifier */);
+            node.originalKeywordKind = originalKeywordKind;
+            node.escapedText = ts.escapeLeadingUnderscores(text);
+            return node;
         }
-        function reScanTemplateToken(isTaggedTemplate) {
-            return currentToken = scanner.reScanTemplateToken(isTaggedTemplate);
+        function createBaseGeneratedIdentifier(text, autoGenerateFlags) {
+            var node = createBaseIdentifier(text, /*originalKeywordKind*/ undefined);
+            node.autoGenerateFlags = autoGenerateFlags;
+            node.autoGenerateId = nextAutoGenerateId;
+            nextAutoGenerateId++;
+            return node;
         }
-        function reScanTemplateHeadOrNoSubstitutionTemplate() {
-            return currentToken = scanner.reScanTemplateHeadOrNoSubstitutionTemplate();
+        // @api
+        function createIdentifier(text, typeArguments, originalKeywordKind) {
+            var node = createBaseIdentifier(text, originalKeywordKind);
+            if (typeArguments) {
+                // NOTE: we do not use `setChildren` here because typeArguments in an identifier do not contribute to transformations
+                node.typeArguments = createNodeArray(typeArguments);
+            }
+            if (node.originalKeywordKind === 130 /* AwaitKeyword */) {
+                node.transformFlags |= 8388608 /* ContainsPossibleTopLevelAwait */;
+            }
+            return node;
         }
-        function reScanLessThanToken() {
-            return currentToken = scanner.reScanLessThanToken();
+        // @api
+        function updateIdentifier(node, typeArguments) {
+            return node.typeArguments !== typeArguments
+                ? update(createIdentifier(ts.idText(node), typeArguments), node)
+                : node;
+        }
+        // @api
+        function createTempVariable(recordTempVariable, reservedInNestedScopes) {
+            var flags = 1 /* Auto */;
+            if (reservedInNestedScopes)
+                flags |= 8 /* ReservedInNestedScopes */;
+            var name = createBaseGeneratedIdentifier("", flags);
+            if (recordTempVariable) {
+                recordTempVariable(name);
+            }
+            return name;
         }
-        function scanJsxIdentifier() {
-            return currentToken = scanner.scanJsxIdentifier();
+        /** Create a unique temporary variable for use in a loop. */
+        // @api
+        function createLoopVariable() {
+            return createBaseGeneratedIdentifier("", 2 /* Loop */);
         }
-        function scanJsxText() {
-            return currentToken = scanner.scanJsxToken();
+        /** Create a unique name based on the supplied text. */
+        // @api
+        function createUniqueName(text, flags) {
+            if (flags === void 0) { flags = 0 /* None */; }
+            ts.Debug.assert(!(flags & 7 /* KindMask */), "Argument out of range: flags");
+            ts.Debug.assert((flags & (16 /* Optimistic */ | 32 /* FileLevel */)) !== 32 /* FileLevel */, "GeneratedIdentifierFlags.FileLevel cannot be set without also setting GeneratedIdentifierFlags.Optimistic");
+            return createBaseGeneratedIdentifier(text, 3 /* Unique */ | flags);
+        }
+        /** Create a unique name generated for a node. */
+        // @api
+        function getGeneratedNameForNode(node, flags) {
+            if (flags === void 0) { flags = 0; }
+            ts.Debug.assert(!(flags & 7 /* KindMask */), "Argument out of range: flags");
+            var name = createBaseGeneratedIdentifier(node && ts.isIdentifier(node) ? ts.idText(node) : "", 4 /* Node */ | flags);
+            name.original = node;
+            return name;
         }
-        function scanJsxAttributeValue() {
-            return currentToken = scanner.scanJsxAttributeValue();
+        // @api
+        function createPrivateIdentifier(text) {
+            if (!ts.startsWith(text, "#"))
+                ts.Debug.fail("First character of private identifier must be #: " + text);
+            var node = baseFactory.createBasePrivateIdentifierNode(79 /* PrivateIdentifier */);
+            node.escapedText = ts.escapeLeadingUnderscores(text);
+            node.transformFlags |= 4194304 /* ContainsClassFields */;
+            return node;
         }
-        function speculationHelper(callback, isLookAhead) {
-            // Keep track of the state we'll need to rollback to if lookahead fails (or if the
-            // caller asked us to always reset our state).
-            var saveToken = currentToken;
-            var saveParseDiagnosticsLength = parseDiagnostics.length;
-            var saveParseErrorBeforeNextFinishedNode = parseErrorBeforeNextFinishedNode;
-            // Note: it is not actually necessary to save/restore the context flags here.  That's
-            // because the saving/restoring of these flags happens naturally through the recursive
-            // descent nature of our parser.  However, we still store this here just so we can
-            // assert that invariant holds.
-            var saveContextFlags = contextFlags;
-            // If we're only looking ahead, then tell the scanner to only lookahead as well.
-            // Otherwise, if we're actually speculatively parsing, then tell the scanner to do the
-            // same.
-            var result = isLookAhead
-                ? scanner.lookAhead(callback)
-                : scanner.tryScan(callback);
-            ts.Debug.assert(saveContextFlags === contextFlags);
-            // If our callback returned something 'falsy' or we're just looking ahead,
-            // then unconditionally restore us to where we were.
-            if (!result || isLookAhead) {
-                currentToken = saveToken;
-                parseDiagnostics.length = saveParseDiagnosticsLength;
-                parseErrorBeforeNextFinishedNode = saveParseErrorBeforeNextFinishedNode;
+        //
+        // Punctuation
+        //
+        function createBaseToken(kind) {
+            return baseFactory.createBaseTokenNode(kind);
+        }
+        function createToken(token) {
+            ts.Debug.assert(token >= 0 /* FirstToken */ && token <= 156 /* LastToken */, "Invalid token");
+            ts.Debug.assert(token <= 14 /* FirstTemplateToken */ || token >= 17 /* LastTemplateToken */, "Invalid token. Use 'createTemplateLiteralLikeNode' to create template literals.");
+            ts.Debug.assert(token <= 8 /* FirstLiteralToken */ || token >= 14 /* LastLiteralToken */, "Invalid token. Use 'createLiteralLikeNode' to create literals.");
+            ts.Debug.assert(token !== 78 /* Identifier */, "Invalid token. Use 'createIdentifier' to create identifiers");
+            var node = createBaseToken(token);
+            var transformFlags = 0 /* None */;
+            switch (token) {
+                case 129 /* AsyncKeyword */:
+                    // 'async' modifier is ES2017 (async functions) or ES2018 (async generators)
+                    transformFlags =
+                        64 /* ContainsES2017 */ |
+                            32 /* ContainsES2018 */;
+                    break;
+                case 122 /* PublicKeyword */:
+                case 120 /* PrivateKeyword */:
+                case 121 /* ProtectedKeyword */:
+                case 142 /* ReadonlyKeyword */:
+                case 125 /* AbstractKeyword */:
+                case 133 /* DeclareKeyword */:
+                case 84 /* ConstKeyword */:
+                case 128 /* AnyKeyword */:
+                case 144 /* NumberKeyword */:
+                case 155 /* BigIntKeyword */:
+                case 141 /* NeverKeyword */:
+                case 145 /* ObjectKeyword */:
+                case 147 /* StringKeyword */:
+                case 131 /* BooleanKeyword */:
+                case 148 /* SymbolKeyword */:
+                case 113 /* VoidKeyword */:
+                case 152 /* UnknownKeyword */:
+                case 150 /* UndefinedKeyword */: // `undefined` is an Identifier in the expression case.
+                    transformFlags = 1 /* ContainsTypeScript */;
+                    break;
+                case 123 /* StaticKeyword */:
+                case 105 /* SuperKeyword */:
+                    transformFlags = 256 /* ContainsES2015 */;
+                    break;
+                case 107 /* ThisKeyword */:
+                    // 'this' indicates a lexical 'this'
+                    transformFlags = 4096 /* ContainsLexicalThis */;
+                    break;
             }
-            return result;
+            if (transformFlags) {
+                node.transformFlags |= transformFlags;
+            }
+            return node;
         }
-        /** Invokes the provided callback then unconditionally restores the parser to the state it
-         * was in immediately prior to invoking the callback.  The result of invoking the callback
-         * is returned from this function.
-         */
-        function lookAhead(callback) {
-            return speculationHelper(callback, /*isLookAhead*/ true);
+        //
+        // Reserved words
+        //
+        // @api
+        function createSuper() {
+            return createToken(105 /* SuperKeyword */);
         }
-        /** Invokes the provided callback.  If the callback returns something falsy, then it restores
-         * the parser to the state it was in immediately prior to invoking the callback.  If the
-         * callback returns something truthy, then the parser state is not rolled back.  The result
-         * of invoking the callback is returned from this function.
-         */
-        function tryParse(callback) {
-            return speculationHelper(callback, /*isLookAhead*/ false);
+        // @api
+        function createThis() {
+            return createToken(107 /* ThisKeyword */);
         }
-        // Ignore strict mode flag because we will report an error in type checker instead.
-        function isIdentifier() {
-            if (token() === 75 /* Identifier */) {
-                return true;
+        // @api
+        function createNull() {
+            return createToken(103 /* NullKeyword */);
+        }
+        // @api
+        function createTrue() {
+            return createToken(109 /* TrueKeyword */);
+        }
+        // @api
+        function createFalse() {
+            return createToken(94 /* FalseKeyword */);
+        }
+        //
+        // Modifiers
+        //
+        // @api
+        function createModifier(kind) {
+            return createToken(kind);
+        }
+        // @api
+        function createModifiersFromModifierFlags(flags) {
+            var result = [];
+            if (flags & 1 /* Export */) {
+                result.push(createModifier(92 /* ExportKeyword */));
             }
-            // If we have a 'yield' keyword, and we're in the [yield] context, then 'yield' is
-            // considered a keyword and is not an identifier.
-            if (token() === 121 /* YieldKeyword */ && inYieldContext()) {
-                return false;
+            if (flags & 2 /* Ambient */) {
+                result.push(createModifier(133 /* DeclareKeyword */));
             }
-            // If we have a 'await' keyword, and we're in the [Await] context, then 'await' is
-            // considered a keyword and is not an identifier.
-            if (token() === 127 /* AwaitKeyword */ && inAwaitContext()) {
-                return false;
+            if (flags & 512 /* Default */) {
+                result.push(createModifier(87 /* DefaultKeyword */));
             }
-            return token() > 112 /* LastReservedWord */;
-        }
-        function parseExpected(kind, diagnosticMessage, shouldAdvance) {
-            if (shouldAdvance === void 0) { shouldAdvance = true; }
-            if (token() === kind) {
-                if (shouldAdvance) {
-                    nextToken();
-                }
-                return true;
+            if (flags & 2048 /* Const */) {
+                result.push(createModifier(84 /* ConstKeyword */));
             }
-            // Report specific message if provided with one.  Otherwise, report generic fallback message.
-            if (diagnosticMessage) {
-                parseErrorAtCurrentToken(diagnosticMessage);
+            if (flags & 4 /* Public */) {
+                result.push(createModifier(122 /* PublicKeyword */));
             }
-            else {
-                parseErrorAtCurrentToken(ts.Diagnostics._0_expected, ts.tokenToString(kind));
+            if (flags & 8 /* Private */) {
+                result.push(createModifier(120 /* PrivateKeyword */));
             }
-            return false;
-        }
-        function parseExpectedJSDoc(kind) {
-            if (token() === kind) {
-                nextTokenJSDoc();
-                return true;
+            if (flags & 16 /* Protected */) {
+                result.push(createModifier(121 /* ProtectedKeyword */));
             }
-            parseErrorAtCurrentToken(ts.Diagnostics._0_expected, ts.tokenToString(kind));
-            return false;
-        }
-        function parseOptional(t) {
-            if (token() === t) {
-                nextToken();
-                return true;
+            if (flags & 128 /* Abstract */) {
+                result.push(createModifier(125 /* AbstractKeyword */));
             }
-            return false;
-        }
-        function parseOptionalToken(t) {
-            if (token() === t) {
-                return parseTokenNode();
+            if (flags & 32 /* Static */) {
+                result.push(createModifier(123 /* StaticKeyword */));
             }
-            return undefined;
-        }
-        function parseOptionalTokenJSDoc(t) {
-            if (token() === t) {
-                return parseTokenNodeJSDoc();
+            if (flags & 64 /* Readonly */) {
+                result.push(createModifier(142 /* ReadonlyKeyword */));
             }
-            return undefined;
-        }
-        function parseExpectedToken(t, diagnosticMessage, arg0) {
-            return parseOptionalToken(t) ||
-                createMissingNode(t, /*reportAtCurrentPosition*/ false, diagnosticMessage || ts.Diagnostics._0_expected, arg0 || ts.tokenToString(t));
+            if (flags & 256 /* Async */) {
+                result.push(createModifier(129 /* AsyncKeyword */));
+            }
+            return result;
         }
-        function parseExpectedTokenJSDoc(t) {
-            return parseOptionalTokenJSDoc(t) ||
-                createMissingNode(t, /*reportAtCurrentPosition*/ false, ts.Diagnostics._0_expected, ts.tokenToString(t));
+        //
+        // Names
+        //
+        // @api
+        function createQualifiedName(left, right) {
+            var node = createBaseNode(157 /* QualifiedName */);
+            node.left = left;
+            node.right = asName(right);
+            node.transformFlags |=
+                propagateChildFlags(node.left) |
+                    propagateIdentifierNameFlags(node.right);
+            return node;
         }
-        function parseTokenNode() {
-            var node = createNode(token());
-            nextToken();
-            return finishNode(node);
+        // @api
+        function updateQualifiedName(node, left, right) {
+            return node.left !== left
+                || node.right !== right
+                ? update(createQualifiedName(left, right), node)
+                : node;
+        }
+        // @api
+        function createComputedPropertyName(expression) {
+            var node = createBaseNode(158 /* ComputedPropertyName */);
+            node.expression = parenthesizerRules().parenthesizeExpressionOfComputedPropertyName(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    256 /* ContainsES2015 */ |
+                    32768 /* ContainsComputedPropertyName */;
+            return node;
         }
-        function parseTokenNodeJSDoc() {
-            var node = createNode(token());
-            nextTokenJSDoc();
-            return finishNode(node);
+        // @api
+        function updateComputedPropertyName(node, expression) {
+            return node.expression !== expression
+                ? update(createComputedPropertyName(expression), node)
+                : node;
         }
-        function canParseSemicolon() {
-            // If there's a real semicolon, then we can always parse it out.
-            if (token() === 26 /* SemicolonToken */) {
-                return true;
-            }
-            // We can parse out an optional semicolon in ASI cases in the following cases.
-            return token() === 19 /* CloseBraceToken */ || token() === 1 /* EndOfFileToken */ || scanner.hasPrecedingLineBreak();
+        //
+        // Signature elements
+        //
+        // @api
+        function createTypeParameterDeclaration(name, constraint, defaultType) {
+            var node = createBaseNamedDeclaration(159 /* TypeParameter */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, name);
+            node.constraint = constraint;
+            node.default = defaultType;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseSemicolon() {
-            if (canParseSemicolon()) {
-                if (token() === 26 /* SemicolonToken */) {
-                    // consume the semicolon if it was explicitly provided.
-                    nextToken();
-                }
-                return true;
+        // @api
+        function updateTypeParameterDeclaration(node, name, constraint, defaultType) {
+            return node.name !== name
+                || node.constraint !== constraint
+                || node.default !== defaultType
+                ? update(createTypeParameterDeclaration(name, constraint, defaultType), node)
+                : node;
+        }
+        // @api
+        function createParameterDeclaration(decorators, modifiers, dotDotDotToken, name, questionToken, type, initializer) {
+            var node = createBaseVariableLikeDeclaration(160 /* Parameter */, decorators, modifiers, name, type, initializer && parenthesizerRules().parenthesizeExpressionForDisallowedComma(initializer));
+            node.dotDotDotToken = dotDotDotToken;
+            node.questionToken = questionToken;
+            if (ts.isThisIdentifier(node.name)) {
+                node.transformFlags = 1 /* ContainsTypeScript */;
             }
             else {
-                return parseExpected(26 /* SemicolonToken */);
+                node.transformFlags |=
+                    propagateChildFlags(node.dotDotDotToken) |
+                        propagateChildFlags(node.questionToken);
+                if (questionToken)
+                    node.transformFlags |= 1 /* ContainsTypeScript */;
+                if (ts.modifiersToFlags(node.modifiers) & 92 /* ParameterPropertyModifier */)
+                    node.transformFlags |= 2048 /* ContainsTypeScriptClassSyntax */;
+                if (initializer || dotDotDotToken)
+                    node.transformFlags |= 256 /* ContainsES2015 */;
             }
+            return node;
         }
-        function createNode(kind, pos) {
-            nodeCount++;
-            var p = pos >= 0 ? pos : scanner.getStartPos();
-            return ts.isNodeKind(kind) || kind === 0 /* Unknown */ ? new NodeConstructor(kind, p, p) :
-                kind === 75 /* Identifier */ ? new IdentifierConstructor(kind, p, p) :
-                    kind === 76 /* PrivateIdentifier */ ? new PrivateIdentifierConstructor(kind, p, p) :
-                        new TokenConstructor(kind, p, p);
-        }
-        function createNodeWithJSDoc(kind, pos) {
-            var node = createNode(kind, pos);
-            if (scanner.getTokenFlags() & 2 /* PrecedingJSDocComment */ && (kind !== 226 /* ExpressionStatement */ || token() !== 20 /* OpenParenToken */)) {
-                addJSDocComment(node);
-            }
+        // @api
+        function updateParameterDeclaration(node, decorators, modifiers, dotDotDotToken, name, questionToken, type, initializer) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.dotDotDotToken !== dotDotDotToken
+                || node.name !== name
+                || node.questionToken !== questionToken
+                || node.type !== type
+                || node.initializer !== initializer
+                ? update(createParameterDeclaration(decorators, modifiers, dotDotDotToken, name, questionToken, type, initializer), node)
+                : node;
+        }
+        // @api
+        function createDecorator(expression) {
+            var node = createBaseNode(161 /* Decorator */);
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    1 /* ContainsTypeScript */ |
+                    2048 /* ContainsTypeScriptClassSyntax */;
             return node;
         }
-        function createNodeArray(elements, pos, end) {
-            // Since the element list of a node array is typically created by starting with an empty array and
-            // repeatedly calling push(), the list may not have the optimal memory layout. We invoke slice() for
-            // small arrays (1 to 4 elements) to give the VM a chance to allocate an optimal representation.
-            var length = elements.length;
-            var array = (length >= 1 && length <= 4 ? elements.slice() : elements);
-            array.pos = pos;
-            array.end = end === undefined ? scanner.getStartPos() : end;
-            return array;
+        // @api
+        function updateDecorator(node, expression) {
+            return node.expression !== expression
+                ? update(createDecorator(expression), node)
+                : node;
         }
-        function finishNode(node, end) {
-            node.end = end === undefined ? scanner.getStartPos() : end;
-            if (contextFlags) {
-                node.flags |= contextFlags;
-            }
-            // Keep track on the node if we encountered an error while parsing it.  If we did, then
-            // we cannot reuse the node incrementally.  Once we've marked this node, clear out the
-            // flag so that we don't mark any subsequent nodes.
-            if (parseErrorBeforeNextFinishedNode) {
-                parseErrorBeforeNextFinishedNode = false;
-                node.flags |= 65536 /* ThisNodeHasError */;
-            }
+        //
+        // Type Elements
+        //
+        // @api
+        function createPropertySignature(modifiers, name, questionToken, type) {
+            var node = createBaseNamedDeclaration(162 /* PropertySignature */, 
+            /*decorators*/ undefined, modifiers, name);
+            node.type = type;
+            node.questionToken = questionToken;
+            node.transformFlags = 1 /* ContainsTypeScript */;
             return node;
         }
-        function createMissingNode(kind, reportAtCurrentPosition, diagnosticMessage, arg0) {
-            if (reportAtCurrentPosition) {
-                parseErrorAtPosition(scanner.getStartPos(), 0, diagnosticMessage, arg0);
+        // @api
+        function updatePropertySignature(node, modifiers, name, questionToken, type) {
+            return node.modifiers !== modifiers
+                || node.name !== name
+                || node.questionToken !== questionToken
+                || node.type !== type
+                ? update(createPropertySignature(modifiers, name, questionToken, type), node)
+                : node;
+        }
+        // @api
+        function createPropertyDeclaration(decorators, modifiers, name, questionOrExclamationToken, type, initializer) {
+            var node = createBaseVariableLikeDeclaration(163 /* PropertyDeclaration */, decorators, modifiers, name, type, initializer);
+            node.questionToken = questionOrExclamationToken && ts.isQuestionToken(questionOrExclamationToken) ? questionOrExclamationToken : undefined;
+            node.exclamationToken = questionOrExclamationToken && ts.isExclamationToken(questionOrExclamationToken) ? questionOrExclamationToken : undefined;
+            node.transformFlags |=
+                propagateChildFlags(node.questionToken) |
+                    propagateChildFlags(node.exclamationToken) |
+                    4194304 /* ContainsClassFields */;
+            if (ts.isComputedPropertyName(node.name) || (ts.hasStaticModifier(node) && node.initializer)) {
+                node.transformFlags |= 2048 /* ContainsTypeScriptClassSyntax */;
+            }
+            if (questionOrExclamationToken || ts.modifiersToFlags(node.modifiers) & 2 /* Ambient */) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            else if (diagnosticMessage) {
-                parseErrorAtCurrentToken(diagnosticMessage, arg0);
-            }
-            var result = createNode(kind);
-            if (kind === 75 /* Identifier */) {
-                result.escapedText = "";
-            }
-            else if (ts.isLiteralKind(kind) || ts.isTemplateLiteralKind(kind)) {
-                result.text = "";
-            }
-            return finishNode(result);
+            return node;
         }
-        function internIdentifier(text) {
-            var identifier = identifiers.get(text);
-            if (identifier === undefined) {
-                identifiers.set(text, identifier = text);
-            }
-            return identifier;
+        // @api
+        function updatePropertyDeclaration(node, decorators, modifiers, name, questionOrExclamationToken, type, initializer) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.questionToken !== (questionOrExclamationToken !== undefined && ts.isQuestionToken(questionOrExclamationToken) ? questionOrExclamationToken : undefined)
+                || node.exclamationToken !== (questionOrExclamationToken !== undefined && ts.isExclamationToken(questionOrExclamationToken) ? questionOrExclamationToken : undefined)
+                || node.type !== type
+                || node.initializer !== initializer
+                ? update(createPropertyDeclaration(decorators, modifiers, name, questionOrExclamationToken, type, initializer), node)
+                : node;
+        }
+        // @api
+        function createMethodSignature(modifiers, name, questionToken, typeParameters, parameters, type) {
+            var node = createBaseSignatureDeclaration(164 /* MethodSignature */, 
+            /*decorators*/ undefined, modifiers, name, typeParameters, parameters, type);
+            node.questionToken = questionToken;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        // An identifier that starts with two underscores has an extra underscore character prepended to it to avoid issues
-        // with magic property names like '__proto__'. The 'identifiers' object is used to share a single string instance for
-        // each identifier in order to reduce memory consumption.
-        function createIdentifier(isIdentifier, diagnosticMessage, privateIdentifierDiagnosticMessage) {
-            identifierCount++;
-            if (isIdentifier) {
-                var node = createNode(75 /* Identifier */);
-                // Store original token kind if it is not just an Identifier so we can report appropriate error later in type checker
-                if (token() !== 75 /* Identifier */) {
-                    node.originalKeywordKind = token();
+        // @api
+        function updateMethodSignature(node, modifiers, name, questionToken, typeParameters, parameters, type) {
+            return node.modifiers !== modifiers
+                || node.name !== name
+                || node.questionToken !== questionToken
+                || node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                ? updateBaseSignatureDeclaration(createMethodSignature(modifiers, name, questionToken, typeParameters, parameters, type), node)
+                : node;
+        }
+        // @api
+        function createMethodDeclaration(decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body) {
+            var node = createBaseFunctionLikeDeclaration(165 /* MethodDeclaration */, decorators, modifiers, name, typeParameters, parameters, type, body);
+            node.asteriskToken = asteriskToken;
+            node.questionToken = questionToken;
+            node.transformFlags |=
+                propagateChildFlags(node.asteriskToken) |
+                    propagateChildFlags(node.questionToken) |
+                    256 /* ContainsES2015 */;
+            if (questionToken) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
+            }
+            if (ts.modifiersToFlags(node.modifiers) & 256 /* Async */) {
+                if (asteriskToken) {
+                    node.transformFlags |= 32 /* ContainsES2018 */;
+                }
+                else {
+                    node.transformFlags |= 64 /* ContainsES2017 */;
                 }
-                node.escapedText = ts.escapeLeadingUnderscores(internIdentifier(scanner.getTokenValue()));
-                nextTokenWithoutCheck();
-                return finishNode(node);
             }
-            if (token() === 76 /* PrivateIdentifier */) {
-                parseErrorAtCurrentToken(privateIdentifierDiagnosticMessage || ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
-                return createIdentifier(/*isIdentifier*/ true);
+            else if (asteriskToken) {
+                node.transformFlags |= 512 /* ContainsGenerator */;
             }
-            // Only for end of file because the error gets reported incorrectly on embedded script tags.
-            var reportAtCurrentPosition = token() === 1 /* EndOfFileToken */;
-            var isReservedWord = scanner.isReservedWord();
-            var msgArg = scanner.getTokenText();
-            var defaultMessage = isReservedWord ?
-                ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_that_cannot_be_used_here :
-                ts.Diagnostics.Identifier_expected;
-            return createMissingNode(75 /* Identifier */, reportAtCurrentPosition, diagnosticMessage || defaultMessage, msgArg);
+            return node;
         }
-        function parseIdentifier(diagnosticMessage, privateIdentifierDiagnosticMessage) {
-            return createIdentifier(isIdentifier(), diagnosticMessage, privateIdentifierDiagnosticMessage);
+        // @api
+        function updateMethodDeclaration(node, decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.asteriskToken !== asteriskToken
+                || node.name !== name
+                || node.questionToken !== questionToken
+                || node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createMethodDeclaration(decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body), node)
+                : node;
+        }
+        // @api
+        function createConstructorDeclaration(decorators, modifiers, parameters, body) {
+            var node = createBaseFunctionLikeDeclaration(166 /* Constructor */, decorators, modifiers, 
+            /*name*/ undefined, 
+            /*typeParameters*/ undefined, parameters, 
+            /*type*/ undefined, body);
+            node.transformFlags |= 256 /* ContainsES2015 */;
+            return node;
         }
-        function parseIdentifierName(diagnosticMessage) {
-            return createIdentifier(ts.tokenIsIdentifierOrKeyword(token()), diagnosticMessage);
+        // @api
+        function updateConstructorDeclaration(node, decorators, modifiers, parameters, body) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.parameters !== parameters
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createConstructorDeclaration(decorators, modifiers, parameters, body), node)
+                : node;
+        }
+        // @api
+        function createGetAccessorDeclaration(decorators, modifiers, name, parameters, type, body) {
+            return createBaseFunctionLikeDeclaration(167 /* GetAccessor */, decorators, modifiers, name, 
+            /*typeParameters*/ undefined, parameters, type, body);
+        }
+        // @api
+        function updateGetAccessorDeclaration(node, decorators, modifiers, name, parameters, type, body) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.parameters !== parameters
+                || node.type !== type
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createGetAccessorDeclaration(decorators, modifiers, name, parameters, type, body), node)
+                : node;
+        }
+        // @api
+        function createSetAccessorDeclaration(decorators, modifiers, name, parameters, body) {
+            return createBaseFunctionLikeDeclaration(168 /* SetAccessor */, decorators, modifiers, name, 
+            /*typeParameters*/ undefined, parameters, 
+            /*type*/ undefined, body);
         }
-        function isLiteralPropertyName() {
-            return ts.tokenIsIdentifierOrKeyword(token()) ||
-                token() === 10 /* StringLiteral */ ||
-                token() === 8 /* NumericLiteral */;
+        // @api
+        function updateSetAccessorDeclaration(node, decorators, modifiers, name, parameters, body) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.parameters !== parameters
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createSetAccessorDeclaration(decorators, modifiers, name, parameters, body), node)
+                : node;
+        }
+        // @api
+        function createCallSignature(typeParameters, parameters, type) {
+            var node = createBaseSignatureDeclaration(169 /* CallSignature */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*name*/ undefined, typeParameters, parameters, type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parsePropertyNameWorker(allowComputedPropertyNames) {
-            if (token() === 10 /* StringLiteral */ || token() === 8 /* NumericLiteral */) {
-                var node = parseLiteralNode();
-                node.text = internIdentifier(node.text);
-                return node;
-            }
-            if (allowComputedPropertyNames && token() === 22 /* OpenBracketToken */) {
-                return parseComputedPropertyName();
-            }
-            if (token() === 76 /* PrivateIdentifier */) {
-                return parsePrivateIdentifier();
-            }
-            return parseIdentifierName();
+        // @api
+        function updateCallSignature(node, typeParameters, parameters, type) {
+            return node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                ? updateBaseSignatureDeclaration(createCallSignature(typeParameters, parameters, type), node)
+                : node;
+        }
+        // @api
+        function createConstructSignature(typeParameters, parameters, type) {
+            var node = createBaseSignatureDeclaration(170 /* ConstructSignature */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*name*/ undefined, typeParameters, parameters, type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parsePropertyName() {
-            return parsePropertyNameWorker(/*allowComputedPropertyNames*/ true);
+        // @api
+        function updateConstructSignature(node, typeParameters, parameters, type) {
+            return node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                ? updateBaseSignatureDeclaration(createConstructSignature(typeParameters, parameters, type), node)
+                : node;
+        }
+        // @api
+        function createIndexSignature(decorators, modifiers, parameters, type) {
+            var node = createBaseSignatureDeclaration(171 /* IndexSignature */, decorators, modifiers, 
+            /*name*/ undefined, 
+            /*typeParameters*/ undefined, parameters, type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseComputedPropertyName() {
-            // PropertyName [Yield]:
-            //      LiteralPropertyName
-            //      ComputedPropertyName[?Yield]
-            var node = createNode(154 /* ComputedPropertyName */);
-            parseExpected(22 /* OpenBracketToken */);
-            // We parse any expression (including a comma expression). But the grammar
-            // says that only an assignment expression is allowed, so the grammar checker
-            // will error if it sees a comma expression.
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(23 /* CloseBracketToken */);
-            return finishNode(node);
+        // @api
+        function updateIndexSignature(node, decorators, modifiers, parameters, type) {
+            return node.parameters !== parameters
+                || node.type !== type
+                || node.decorators !== decorators
+                || node.modifiers !== modifiers
+                ? updateBaseSignatureDeclaration(createIndexSignature(decorators, modifiers, parameters, type), node)
+                : node;
+        }
+        // @api
+        function createTemplateLiteralTypeSpan(type, literal) {
+            var node = createBaseNode(194 /* TemplateLiteralTypeSpan */);
+            node.type = type;
+            node.literal = literal;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function internPrivateIdentifier(text) {
-            var privateIdentifier = privateIdentifiers.get(text);
-            if (privateIdentifier === undefined) {
-                privateIdentifiers.set(text, privateIdentifier = text);
-            }
-            return privateIdentifier;
+        // @api
+        function updateTemplateLiteralTypeSpan(node, type, literal) {
+            return node.type !== type
+                || node.literal !== literal
+                ? update(createTemplateLiteralTypeSpan(type, literal), node)
+                : node;
         }
-        function parsePrivateIdentifier() {
-            var node = createNode(76 /* PrivateIdentifier */);
-            node.escapedText = ts.escapeLeadingUnderscores(internPrivateIdentifier(scanner.getTokenText()));
-            nextToken();
-            return finishNode(node);
+        //
+        // Types
+        //
+        // @api
+        function createKeywordTypeNode(kind) {
+            return createToken(kind);
+        }
+        // @api
+        function createTypePredicateNode(assertsModifier, parameterName, type) {
+            var node = createBaseNode(172 /* TypePredicate */);
+            node.assertsModifier = assertsModifier;
+            node.parameterName = asName(parameterName);
+            node.type = type;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseContextualModifier(t) {
-            return token() === t && tryParse(nextTokenCanFollowModifier);
+        // @api
+        function updateTypePredicateNode(node, assertsModifier, parameterName, type) {
+            return node.assertsModifier !== assertsModifier
+                || node.parameterName !== parameterName
+                || node.type !== type
+                ? update(createTypePredicateNode(assertsModifier, parameterName, type), node)
+                : node;
+        }
+        // @api
+        function createTypeReferenceNode(typeName, typeArguments) {
+            var node = createBaseNode(173 /* TypeReference */);
+            node.typeName = asName(typeName);
+            node.typeArguments = typeArguments && parenthesizerRules().parenthesizeTypeArguments(createNodeArray(typeArguments));
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function nextTokenIsOnSameLineAndCanFollowModifier() {
-            nextToken();
-            if (scanner.hasPrecedingLineBreak()) {
-                return false;
-            }
-            return canFollowModifier();
+        // @api
+        function updateTypeReferenceNode(node, typeName, typeArguments) {
+            return node.typeName !== typeName
+                || node.typeArguments !== typeArguments
+                ? update(createTypeReferenceNode(typeName, typeArguments), node)
+                : node;
         }
-        function nextTokenCanFollowModifier() {
-            switch (token()) {
-                case 81 /* ConstKeyword */:
-                    // 'const' is only a modifier if followed by 'enum'.
-                    return nextToken() === 88 /* EnumKeyword */;
-                case 89 /* ExportKeyword */:
-                    nextToken();
-                    if (token() === 84 /* DefaultKeyword */) {
-                        return lookAhead(nextTokenCanFollowDefaultKeyword);
-                    }
-                    if (token() === 145 /* TypeKeyword */) {
-                        return lookAhead(nextTokenCanFollowExportModifier);
-                    }
-                    return canFollowExportModifier();
-                case 84 /* DefaultKeyword */:
-                    return nextTokenCanFollowDefaultKeyword();
-                case 120 /* StaticKeyword */:
-                case 131 /* GetKeyword */:
-                case 142 /* SetKeyword */:
-                    nextToken();
-                    return canFollowModifier();
-                default:
-                    return nextTokenIsOnSameLineAndCanFollowModifier();
-            }
+        // @api
+        function createFunctionTypeNode(typeParameters, parameters, type) {
+            var node = createBaseSignatureDeclaration(174 /* FunctionType */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*name*/ undefined, typeParameters, parameters, type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function canFollowExportModifier() {
-            return token() !== 41 /* AsteriskToken */
-                && token() !== 123 /* AsKeyword */
-                && token() !== 18 /* OpenBraceToken */
-                && canFollowModifier();
+        // @api
+        function updateFunctionTypeNode(node, typeParameters, parameters, type) {
+            return node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                ? updateBaseSignatureDeclaration(createFunctionTypeNode(typeParameters, parameters, type), node)
+                : node;
+        }
+        // @api
+        function createConstructorTypeNode(typeParameters, parameters, type) {
+            var node = createBaseSignatureDeclaration(175 /* ConstructorType */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*name*/ undefined, typeParameters, parameters, type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function nextTokenCanFollowExportModifier() {
-            nextToken();
-            return canFollowExportModifier();
+        // @api
+        function updateConstructorTypeNode(node, typeParameters, parameters, type) {
+            return node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                ? updateBaseSignatureDeclaration(createConstructorTypeNode(typeParameters, parameters, type), node)
+                : node;
+        }
+        // @api
+        function createTypeQueryNode(exprName) {
+            var node = createBaseNode(176 /* TypeQuery */);
+            node.exprName = exprName;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseAnyContextualModifier() {
-            return ts.isModifierKind(token()) && tryParse(nextTokenCanFollowModifier);
+        // @api
+        function updateTypeQueryNode(node, exprName) {
+            return node.exprName !== exprName
+                ? update(createTypeQueryNode(exprName), node)
+                : node;
+        }
+        // @api
+        function createTypeLiteralNode(members) {
+            var node = createBaseNode(177 /* TypeLiteral */);
+            node.members = createNodeArray(members);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function canFollowModifier() {
-            return token() === 22 /* OpenBracketToken */
-                || token() === 18 /* OpenBraceToken */
-                || token() === 41 /* AsteriskToken */
-                || token() === 25 /* DotDotDotToken */
-                || isLiteralPropertyName();
+        // @api
+        function updateTypeLiteralNode(node, members) {
+            return node.members !== members
+                ? update(createTypeLiteralNode(members), node)
+                : node;
+        }
+        // @api
+        function createArrayTypeNode(elementType) {
+            var node = createBaseNode(178 /* ArrayType */);
+            node.elementType = parenthesizerRules().parenthesizeElementTypeOfArrayType(elementType);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function nextTokenCanFollowDefaultKeyword() {
-            nextToken();
-            return token() === 80 /* ClassKeyword */ || token() === 94 /* FunctionKeyword */ ||
-                token() === 114 /* InterfaceKeyword */ ||
-                (token() === 122 /* AbstractKeyword */ && lookAhead(nextTokenIsClassKeywordOnSameLine)) ||
-                (token() === 126 /* AsyncKeyword */ && lookAhead(nextTokenIsFunctionKeywordOnSameLine));
+        // @api
+        function updateArrayTypeNode(node, elementType) {
+            return node.elementType !== elementType
+                ? update(createArrayTypeNode(elementType), node)
+                : node;
+        }
+        // @api
+        function createTupleTypeNode(elements) {
+            var node = createBaseNode(179 /* TupleType */);
+            node.elements = createNodeArray(elements);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        // True if positioned at the start of a list element
-        function isListElement(parsingContext, inErrorRecovery) {
-            var node = currentNode(parsingContext);
-            if (node) {
-                return true;
-            }
-            switch (parsingContext) {
-                case 0 /* SourceElements */:
-                case 1 /* BlockStatements */:
-                case 3 /* SwitchClauseStatements */:
-                    // If we're in error recovery, then we don't want to treat ';' as an empty statement.
-                    // The problem is that ';' can show up in far too many contexts, and if we see one
-                    // and assume it's a statement, then we may bail out inappropriately from whatever
-                    // we're parsing.  For example, if we have a semicolon in the middle of a class, then
-                    // we really don't want to assume the class is over and we're on a statement in the
-                    // outer module.  We just want to consume and move on.
-                    return !(token() === 26 /* SemicolonToken */ && inErrorRecovery) && isStartOfStatement();
-                case 2 /* SwitchClauses */:
-                    return token() === 78 /* CaseKeyword */ || token() === 84 /* DefaultKeyword */;
-                case 4 /* TypeMembers */:
-                    return lookAhead(isTypeMemberStart);
-                case 5 /* ClassMembers */:
-                    // We allow semicolons as class elements (as specified by ES6) as long as we're
-                    // not in error recovery.  If we're in error recovery, we don't want an errant
-                    // semicolon to be treated as a class member (since they're almost always used
-                    // for statements.
-                    return lookAhead(isClassMemberStart) || (token() === 26 /* SemicolonToken */ && !inErrorRecovery);
-                case 6 /* EnumMembers */:
-                    // Include open bracket computed properties. This technically also lets in indexers,
-                    // which would be a candidate for improved error reporting.
-                    return token() === 22 /* OpenBracketToken */ || isLiteralPropertyName();
-                case 12 /* ObjectLiteralMembers */:
-                    switch (token()) {
-                        case 22 /* OpenBracketToken */:
-                        case 41 /* AsteriskToken */:
-                        case 25 /* DotDotDotToken */:
-                        case 24 /* DotToken */: // Not an object literal member, but don't want to close the object (see `tests/cases/fourslash/completionsDotInObjectLiteral.ts`)
-                            return true;
-                        default:
-                            return isLiteralPropertyName();
-                    }
-                case 18 /* RestProperties */:
-                    return isLiteralPropertyName();
-                case 9 /* ObjectBindingElements */:
-                    return token() === 22 /* OpenBracketToken */ || token() === 25 /* DotDotDotToken */ || isLiteralPropertyName();
-                case 7 /* HeritageClauseElement */:
-                    // If we see `{ ... }` then only consume it as an expression if it is followed by `,` or `{`
-                    // That way we won't consume the body of a class in its heritage clause.
-                    if (token() === 18 /* OpenBraceToken */) {
-                        return lookAhead(isValidHeritageClauseObjectLiteral);
-                    }
-                    if (!inErrorRecovery) {
-                        return isStartOfLeftHandSideExpression() && !isHeritageClauseExtendsOrImplementsKeyword();
-                    }
-                    else {
-                        // If we're in error recovery we tighten up what we're willing to match.
-                        // That way we don't treat something like "this" as a valid heritage clause
-                        // element during recovery.
-                        return isIdentifier() && !isHeritageClauseExtendsOrImplementsKeyword();
-                    }
-                case 8 /* VariableDeclarations */:
-                    return isIdentifierOrPrivateIdentifierOrPattern();
-                case 10 /* ArrayBindingElements */:
-                    return token() === 27 /* CommaToken */ || token() === 25 /* DotDotDotToken */ || isIdentifierOrPrivateIdentifierOrPattern();
-                case 19 /* TypeParameters */:
-                    return isIdentifier();
-                case 15 /* ArrayLiteralMembers */:
-                    switch (token()) {
-                        case 27 /* CommaToken */:
-                        case 24 /* DotToken */: // Not an array literal member, but don't want to close the array (see `tests/cases/fourslash/completionsDotInArrayLiteralInObjectLiteral.ts`)
-                            return true;
-                    }
-                // falls through
-                case 11 /* ArgumentExpressions */:
-                    return token() === 25 /* DotDotDotToken */ || isStartOfExpression();
-                case 16 /* Parameters */:
-                    return isStartOfParameter(/*isJSDocParameter*/ false);
-                case 17 /* JSDocParameters */:
-                    return isStartOfParameter(/*isJSDocParameter*/ true);
-                case 20 /* TypeArguments */:
-                case 21 /* TupleElementTypes */:
-                    return token() === 27 /* CommaToken */ || isStartOfType();
-                case 22 /* HeritageClauses */:
-                    return isHeritageClause();
-                case 23 /* ImportOrExportSpecifiers */:
-                    return ts.tokenIsIdentifierOrKeyword(token());
-                case 13 /* JsxAttributes */:
-                    return ts.tokenIsIdentifierOrKeyword(token()) || token() === 18 /* OpenBraceToken */;
-                case 14 /* JsxChildren */:
-                    return true;
-            }
-            return ts.Debug.fail("Non-exhaustive case in 'isListElement'.");
+        // @api
+        function updateTupleTypeNode(node, elements) {
+            return node.elements !== elements
+                ? update(createTupleTypeNode(elements), node)
+                : node;
+        }
+        // @api
+        function createNamedTupleMember(dotDotDotToken, name, questionToken, type) {
+            var node = createBaseNode(192 /* NamedTupleMember */);
+            node.dotDotDotToken = dotDotDotToken;
+            node.name = name;
+            node.questionToken = questionToken;
+            node.type = type;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function isValidHeritageClauseObjectLiteral() {
-            ts.Debug.assert(token() === 18 /* OpenBraceToken */);
-            if (nextToken() === 19 /* CloseBraceToken */) {
-                // if we see "extends {}" then only treat the {} as what we're extending (and not
-                // the class body) if we have:
-                //
-                //      extends {} {
-                //      extends {},
-                //      extends {} extends
-                //      extends {} implements
-                var next = nextToken();
-                return next === 27 /* CommaToken */ || next === 18 /* OpenBraceToken */ || next === 90 /* ExtendsKeyword */ || next === 113 /* ImplementsKeyword */;
-            }
-            return true;
+        // @api
+        function updateNamedTupleMember(node, dotDotDotToken, name, questionToken, type) {
+            return node.dotDotDotToken !== dotDotDotToken
+                || node.name !== name
+                || node.questionToken !== questionToken
+                || node.type !== type
+                ? update(createNamedTupleMember(dotDotDotToken, name, questionToken, type), node)
+                : node;
+        }
+        // @api
+        function createOptionalTypeNode(type) {
+            var node = createBaseNode(180 /* OptionalType */);
+            node.type = parenthesizerRules().parenthesizeElementTypeOfArrayType(type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function nextTokenIsIdentifier() {
-            nextToken();
-            return isIdentifier();
+        // @api
+        function updateOptionalTypeNode(node, type) {
+            return node.type !== type
+                ? update(createOptionalTypeNode(type), node)
+                : node;
+        }
+        // @api
+        function createRestTypeNode(type) {
+            var node = createBaseNode(181 /* RestType */);
+            node.type = type;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function nextTokenIsIdentifierOrKeyword() {
-            nextToken();
-            return ts.tokenIsIdentifierOrKeyword(token());
+        // @api
+        function updateRestTypeNode(node, type) {
+            return node.type !== type
+                ? update(createRestTypeNode(type), node)
+                : node;
         }
-        function nextTokenIsIdentifierOrKeywordOrGreaterThan() {
-            nextToken();
-            return ts.tokenIsIdentifierOrKeywordOrGreaterThan(token());
+        function createUnionOrIntersectionTypeNode(kind, types) {
+            var node = createBaseNode(kind);
+            node.types = parenthesizerRules().parenthesizeConstituentTypesOfUnionOrIntersectionType(types);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function isHeritageClauseExtendsOrImplementsKeyword() {
-            if (token() === 113 /* ImplementsKeyword */ ||
-                token() === 90 /* ExtendsKeyword */) {
-                return lookAhead(nextTokenIsStartOfExpression);
+        function updateUnionOrIntersectionTypeNode(node, types) {
+            return node.types !== types
+                ? update(createUnionOrIntersectionTypeNode(node.kind, types), node)
+                : node;
+        }
+        // @api
+        function createUnionTypeNode(types) {
+            return createUnionOrIntersectionTypeNode(182 /* UnionType */, types);
+        }
+        // @api
+        function updateUnionTypeNode(node, types) {
+            return updateUnionOrIntersectionTypeNode(node, types);
+        }
+        // @api
+        function createIntersectionTypeNode(types) {
+            return createUnionOrIntersectionTypeNode(183 /* IntersectionType */, types);
+        }
+        // @api
+        function updateIntersectionTypeNode(node, types) {
+            return updateUnionOrIntersectionTypeNode(node, types);
+        }
+        // @api
+        function createConditionalTypeNode(checkType, extendsType, trueType, falseType) {
+            var node = createBaseNode(184 /* ConditionalType */);
+            node.checkType = parenthesizerRules().parenthesizeMemberOfConditionalType(checkType);
+            node.extendsType = parenthesizerRules().parenthesizeMemberOfConditionalType(extendsType);
+            node.trueType = trueType;
+            node.falseType = falseType;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateConditionalTypeNode(node, checkType, extendsType, trueType, falseType) {
+            return node.checkType !== checkType
+                || node.extendsType !== extendsType
+                || node.trueType !== trueType
+                || node.falseType !== falseType
+                ? update(createConditionalTypeNode(checkType, extendsType, trueType, falseType), node)
+                : node;
+        }
+        // @api
+        function createInferTypeNode(typeParameter) {
+            var node = createBaseNode(185 /* InferType */);
+            node.typeParameter = typeParameter;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateInferTypeNode(node, typeParameter) {
+            return node.typeParameter !== typeParameter
+                ? update(createInferTypeNode(typeParameter), node)
+                : node;
+        }
+        // @api
+        function createTemplateLiteralType(head, templateSpans) {
+            var node = createBaseNode(193 /* TemplateLiteralType */);
+            node.head = head;
+            node.templateSpans = createNodeArray(templateSpans);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateTemplateLiteralType(node, head, templateSpans) {
+            return node.head !== head
+                || node.templateSpans !== templateSpans
+                ? update(createTemplateLiteralType(head, templateSpans), node)
+                : node;
+        }
+        // @api
+        function createImportTypeNode(argument, qualifier, typeArguments, isTypeOf) {
+            if (isTypeOf === void 0) { isTypeOf = false; }
+            var node = createBaseNode(195 /* ImportType */);
+            node.argument = argument;
+            node.qualifier = qualifier;
+            node.typeArguments = typeArguments && parenthesizerRules().parenthesizeTypeArguments(typeArguments);
+            node.isTypeOf = isTypeOf;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateImportTypeNode(node, argument, qualifier, typeArguments, isTypeOf) {
+            if (isTypeOf === void 0) { isTypeOf = node.isTypeOf; }
+            return node.argument !== argument
+                || node.qualifier !== qualifier
+                || node.typeArguments !== typeArguments
+                || node.isTypeOf !== isTypeOf
+                ? update(createImportTypeNode(argument, qualifier, typeArguments, isTypeOf), node)
+                : node;
+        }
+        // @api
+        function createParenthesizedType(type) {
+            var node = createBaseNode(186 /* ParenthesizedType */);
+            node.type = type;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateParenthesizedType(node, type) {
+            return node.type !== type
+                ? update(createParenthesizedType(type), node)
+                : node;
+        }
+        // @api
+        function createThisTypeNode() {
+            var node = createBaseNode(187 /* ThisType */);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function createTypeOperatorNode(operator, type) {
+            var node = createBaseNode(188 /* TypeOperator */);
+            node.operator = operator;
+            node.type = parenthesizerRules().parenthesizeMemberOfElementType(type);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateTypeOperatorNode(node, type) {
+            return node.type !== type
+                ? update(createTypeOperatorNode(node.operator, type), node)
+                : node;
+        }
+        // @api
+        function createIndexedAccessTypeNode(objectType, indexType) {
+            var node = createBaseNode(189 /* IndexedAccessType */);
+            node.objectType = parenthesizerRules().parenthesizeMemberOfElementType(objectType);
+            node.indexType = indexType;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateIndexedAccessTypeNode(node, objectType, indexType) {
+            return node.objectType !== objectType
+                || node.indexType !== indexType
+                ? update(createIndexedAccessTypeNode(objectType, indexType), node)
+                : node;
+        }
+        // @api
+        function createMappedTypeNode(readonlyToken, typeParameter, nameType, questionToken, type) {
+            var node = createBaseNode(190 /* MappedType */);
+            node.readonlyToken = readonlyToken;
+            node.typeParameter = typeParameter;
+            node.nameType = nameType;
+            node.questionToken = questionToken;
+            node.type = type;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateMappedTypeNode(node, readonlyToken, typeParameter, nameType, questionToken, type) {
+            return node.readonlyToken !== readonlyToken
+                || node.typeParameter !== typeParameter
+                || node.nameType !== nameType
+                || node.questionToken !== questionToken
+                || node.type !== type
+                ? update(createMappedTypeNode(readonlyToken, typeParameter, nameType, questionToken, type), node)
+                : node;
+        }
+        // @api
+        function createLiteralTypeNode(literal) {
+            var node = createBaseNode(191 /* LiteralType */);
+            node.literal = literal;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateLiteralTypeNode(node, literal) {
+            return node.literal !== literal
+                ? update(createLiteralTypeNode(literal), node)
+                : node;
+        }
+        //
+        // Binding Patterns
+        //
+        // @api
+        function createObjectBindingPattern(elements) {
+            var node = createBaseNode(196 /* ObjectBindingPattern */);
+            node.elements = createNodeArray(elements);
+            node.transformFlags |=
+                propagateChildrenFlags(node.elements) |
+                    256 /* ContainsES2015 */ |
+                    131072 /* ContainsBindingPattern */;
+            if (node.transformFlags & 8192 /* ContainsRestOrSpread */) {
+                node.transformFlags |=
+                    32 /* ContainsES2018 */ |
+                        16384 /* ContainsObjectRestOrSpread */;
             }
-            return false;
+            return node;
         }
-        function nextTokenIsStartOfExpression() {
-            nextToken();
-            return isStartOfExpression();
+        // @api
+        function updateObjectBindingPattern(node, elements) {
+            return node.elements !== elements
+                ? update(createObjectBindingPattern(elements), node)
+                : node;
+        }
+        // @api
+        function createArrayBindingPattern(elements) {
+            var node = createBaseNode(197 /* ArrayBindingPattern */);
+            node.elements = createNodeArray(elements);
+            node.transformFlags |=
+                propagateChildrenFlags(node.elements) |
+                    256 /* ContainsES2015 */ |
+                    131072 /* ContainsBindingPattern */;
+            return node;
         }
-        function nextTokenIsStartOfType() {
-            nextToken();
-            return isStartOfType();
+        // @api
+        function updateArrayBindingPattern(node, elements) {
+            return node.elements !== elements
+                ? update(createArrayBindingPattern(elements), node)
+                : node;
         }
-        // True if positioned at a list terminator
-        function isListTerminator(kind) {
-            if (token() === 1 /* EndOfFileToken */) {
-                // Being at the end of the file ends all lists.
-                return true;
+        // @api
+        function createBindingElement(dotDotDotToken, propertyName, name, initializer) {
+            var node = createBaseBindingLikeDeclaration(198 /* BindingElement */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, name, initializer);
+            node.propertyName = asName(propertyName);
+            node.dotDotDotToken = dotDotDotToken;
+            node.transformFlags |=
+                propagateChildFlags(node.dotDotDotToken) |
+                    256 /* ContainsES2015 */;
+            if (node.propertyName) {
+                node.transformFlags |= ts.isIdentifier(node.propertyName) ?
+                    propagateIdentifierNameFlags(node.propertyName) :
+                    propagateChildFlags(node.propertyName);
             }
-            switch (kind) {
-                case 1 /* BlockStatements */:
-                case 2 /* SwitchClauses */:
-                case 4 /* TypeMembers */:
-                case 5 /* ClassMembers */:
-                case 6 /* EnumMembers */:
-                case 12 /* ObjectLiteralMembers */:
-                case 9 /* ObjectBindingElements */:
-                case 23 /* ImportOrExportSpecifiers */:
-                    return token() === 19 /* CloseBraceToken */;
-                case 3 /* SwitchClauseStatements */:
-                    return token() === 19 /* CloseBraceToken */ || token() === 78 /* CaseKeyword */ || token() === 84 /* DefaultKeyword */;
-                case 7 /* HeritageClauseElement */:
-                    return token() === 18 /* OpenBraceToken */ || token() === 90 /* ExtendsKeyword */ || token() === 113 /* ImplementsKeyword */;
-                case 8 /* VariableDeclarations */:
-                    return isVariableDeclaratorListTerminator();
-                case 19 /* TypeParameters */:
-                    // Tokens other than '>' are here for better error recovery
-                    return token() === 31 /* GreaterThanToken */ || token() === 20 /* OpenParenToken */ || token() === 18 /* OpenBraceToken */ || token() === 90 /* ExtendsKeyword */ || token() === 113 /* ImplementsKeyword */;
-                case 11 /* ArgumentExpressions */:
-                    // Tokens other than ')' are here for better error recovery
-                    return token() === 21 /* CloseParenToken */ || token() === 26 /* SemicolonToken */;
-                case 15 /* ArrayLiteralMembers */:
-                case 21 /* TupleElementTypes */:
-                case 10 /* ArrayBindingElements */:
-                    return token() === 23 /* CloseBracketToken */;
-                case 17 /* JSDocParameters */:
-                case 16 /* Parameters */:
-                case 18 /* RestProperties */:
-                    // Tokens other than ')' and ']' (the latter for index signatures) are here for better error recovery
-                    return token() === 21 /* CloseParenToken */ || token() === 23 /* CloseBracketToken */ /*|| token === SyntaxKind.OpenBraceToken*/;
-                case 20 /* TypeArguments */:
-                    // All other tokens should cause the type-argument to terminate except comma token
-                    return token() !== 27 /* CommaToken */;
-                case 22 /* HeritageClauses */:
-                    return token() === 18 /* OpenBraceToken */ || token() === 19 /* CloseBraceToken */;
-                case 13 /* JsxAttributes */:
-                    return token() === 31 /* GreaterThanToken */ || token() === 43 /* SlashToken */;
-                case 14 /* JsxChildren */:
-                    return token() === 29 /* LessThanToken */ && lookAhead(nextTokenIsSlash);
-                default:
-                    return false;
+            if (dotDotDotToken)
+                node.transformFlags |= 8192 /* ContainsRestOrSpread */;
+            return node;
+        }
+        // @api
+        function updateBindingElement(node, dotDotDotToken, propertyName, name, initializer) {
+            return node.propertyName !== propertyName
+                || node.dotDotDotToken !== dotDotDotToken
+                || node.name !== name
+                || node.initializer !== initializer
+                ? update(createBindingElement(dotDotDotToken, propertyName, name, initializer), node)
+                : node;
+        }
+        //
+        // Expression
+        //
+        function createBaseExpression(kind) {
+            var node = createBaseNode(kind);
+            // the following properties are commonly set by the checker/binder
+            return node;
+        }
+        // @api
+        function createArrayLiteralExpression(elements, multiLine) {
+            var node = createBaseExpression(199 /* ArrayLiteralExpression */);
+            node.elements = parenthesizerRules().parenthesizeExpressionsOfCommaDelimitedList(createNodeArray(elements));
+            node.multiLine = multiLine;
+            node.transformFlags |= propagateChildrenFlags(node.elements);
+            return node;
+        }
+        // @api
+        function updateArrayLiteralExpression(node, elements) {
+            return node.elements !== elements
+                ? update(createArrayLiteralExpression(elements, node.multiLine), node)
+                : node;
+        }
+        // @api
+        function createObjectLiteralExpression(properties, multiLine) {
+            var node = createBaseExpression(200 /* ObjectLiteralExpression */);
+            node.properties = createNodeArray(properties);
+            node.multiLine = multiLine;
+            node.transformFlags |= propagateChildrenFlags(node.properties);
+            return node;
+        }
+        // @api
+        function updateObjectLiteralExpression(node, properties) {
+            return node.properties !== properties
+                ? update(createObjectLiteralExpression(properties, node.multiLine), node)
+                : node;
+        }
+        // @api
+        function createPropertyAccessExpression(expression, name) {
+            var node = createBaseExpression(201 /* PropertyAccessExpression */);
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.name = asName(name);
+            node.transformFlags =
+                propagateChildFlags(node.expression) |
+                    (ts.isIdentifier(node.name) ?
+                        propagateIdentifierNameFlags(node.name) :
+                        propagateChildFlags(node.name));
+            if (ts.isSuperKeyword(expression)) {
+                // super method calls require a lexical 'this'
+                // super method calls require 'super' hoisting in ES2017 and ES2018 async functions and async generators
+                node.transformFlags |=
+                    64 /* ContainsES2017 */ |
+                        32 /* ContainsES2018 */;
             }
+            return node;
         }
-        function isVariableDeclaratorListTerminator() {
-            // If we can consume a semicolon (either explicitly, or with ASI), then consider us done
-            // with parsing the list of variable declarators.
-            if (canParseSemicolon()) {
-                return true;
+        // @api
+        function updatePropertyAccessExpression(node, expression, name) {
+            if (ts.isPropertyAccessChain(node)) {
+                return updatePropertyAccessChain(node, expression, node.questionDotToken, ts.cast(name, ts.isIdentifier));
+            }
+            return node.expression !== expression
+                || node.name !== name
+                ? update(createPropertyAccessExpression(expression, name), node)
+                : node;
+        }
+        // @api
+        function createPropertyAccessChain(expression, questionDotToken, name) {
+            var node = createBaseExpression(201 /* PropertyAccessExpression */);
+            node.flags |= 32 /* OptionalChain */;
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.questionDotToken = questionDotToken;
+            node.name = asName(name);
+            node.transformFlags |=
+                8 /* ContainsES2020 */ |
+                    propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.questionDotToken) |
+                    (ts.isIdentifier(node.name) ?
+                        propagateIdentifierNameFlags(node.name) :
+                        propagateChildFlags(node.name));
+            return node;
+        }
+        // @api
+        function updatePropertyAccessChain(node, expression, questionDotToken, name) {
+            ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a PropertyAccessExpression using updatePropertyAccessChain. Use updatePropertyAccess instead.");
+            // Because we are updating an existing PropertyAccessChain we want to inherit its emitFlags
+            // instead of using the default from createPropertyAccess
+            return node.expression !== expression
+                || node.questionDotToken !== questionDotToken
+                || node.name !== name
+                ? update(createPropertyAccessChain(expression, questionDotToken, name), node)
+                : node;
+        }
+        // @api
+        function createElementAccessExpression(expression, index) {
+            var node = createBaseExpression(202 /* ElementAccessExpression */);
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.argumentExpression = asExpression(index);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.argumentExpression);
+            if (ts.isSuperKeyword(expression)) {
+                // super method calls require a lexical 'this'
+                // super method calls require 'super' hoisting in ES2017 and ES2018 async functions and async generators
+                node.transformFlags |=
+                    64 /* ContainsES2017 */ |
+                        32 /* ContainsES2018 */;
             }
-            // in the case where we're parsing the variable declarator of a 'for-in' statement, we
-            // are done if we see an 'in' keyword in front of us. Same with for-of
-            if (isInOrOfKeyword(token())) {
-                return true;
+            return node;
+        }
+        // @api
+        function updateElementAccessExpression(node, expression, argumentExpression) {
+            if (ts.isElementAccessChain(node)) {
+                return updateElementAccessChain(node, expression, node.questionDotToken, argumentExpression);
+            }
+            return node.expression !== expression
+                || node.argumentExpression !== argumentExpression
+                ? update(createElementAccessExpression(expression, argumentExpression), node)
+                : node;
+        }
+        // @api
+        function createElementAccessChain(expression, questionDotToken, index) {
+            var node = createBaseExpression(202 /* ElementAccessExpression */);
+            node.flags |= 32 /* OptionalChain */;
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.questionDotToken = questionDotToken;
+            node.argumentExpression = asExpression(index);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.questionDotToken) |
+                    propagateChildFlags(node.argumentExpression) |
+                    8 /* ContainsES2020 */;
+            return node;
+        }
+        // @api
+        function updateElementAccessChain(node, expression, questionDotToken, argumentExpression) {
+            ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a ElementAccessExpression using updateElementAccessChain. Use updateElementAccess instead.");
+            // Because we are updating an existing ElementAccessChain we want to inherit its emitFlags
+            // instead of using the default from createElementAccess
+            return node.expression !== expression
+                || node.questionDotToken !== questionDotToken
+                || node.argumentExpression !== argumentExpression
+                ? update(createElementAccessChain(expression, questionDotToken, argumentExpression), node)
+                : node;
+        }
+        // @api
+        function createCallExpression(expression, typeArguments, argumentsArray) {
+            var node = createBaseExpression(203 /* CallExpression */);
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.typeArguments = asNodeArray(typeArguments);
+            node.arguments = parenthesizerRules().parenthesizeExpressionsOfCommaDelimitedList(createNodeArray(argumentsArray));
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    propagateChildrenFlags(node.arguments);
+            if (node.typeArguments) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            // ERROR RECOVERY TWEAK:
-            // For better error recovery, if we see an '=>' then we just stop immediately.  We've got an
-            // arrow function here and it's going to be very unlikely that we'll resynchronize and get
-            // another variable declaration.
-            if (token() === 38 /* EqualsGreaterThanToken */) {
-                return true;
+            if (ts.isImportKeyword(node.expression)) {
+                node.transformFlags |= 2097152 /* ContainsDynamicImport */;
             }
-            // Keep trying to parse out variable declarators.
-            return false;
-        }
-        // True if positioned at element or terminator of the current list or any enclosing list
-        function isInSomeParsingContext() {
-            for (var kind = 0; kind < 24 /* Count */; kind++) {
-                if (parsingContext & (1 << kind)) {
-                    if (isListElement(kind, /*inErrorRecovery*/ true) || isListTerminator(kind)) {
-                        return true;
-                    }
-                }
+            else if (ts.isSuperProperty(node.expression)) {
+                node.transformFlags |= 4096 /* ContainsLexicalThis */;
             }
-            return false;
+            return node;
         }
-        // Parses a list of elements
-        function parseList(kind, parseElement) {
-            var saveParsingContext = parsingContext;
-            parsingContext |= 1 << kind;
-            var list = [];
-            var listPos = getNodePos();
-            while (!isListTerminator(kind)) {
-                if (isListElement(kind, /*inErrorRecovery*/ false)) {
-                    var element = parseListElement(kind, parseElement);
-                    list.push(element);
-                    continue;
-                }
-                if (abortParsingListOrMoveToNextToken(kind)) {
-                    break;
-                }
+        // @api
+        function updateCallExpression(node, expression, typeArguments, argumentsArray) {
+            if (ts.isCallChain(node)) {
+                return updateCallChain(node, expression, node.questionDotToken, typeArguments, argumentsArray);
+            }
+            return node.expression !== expression
+                || node.typeArguments !== typeArguments
+                || node.arguments !== argumentsArray
+                ? update(createCallExpression(expression, typeArguments, argumentsArray), node)
+                : node;
+        }
+        // @api
+        function createCallChain(expression, questionDotToken, typeArguments, argumentsArray) {
+            var node = createBaseExpression(203 /* CallExpression */);
+            node.flags |= 32 /* OptionalChain */;
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.questionDotToken = questionDotToken;
+            node.typeArguments = asNodeArray(typeArguments);
+            node.arguments = parenthesizerRules().parenthesizeExpressionsOfCommaDelimitedList(createNodeArray(argumentsArray));
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.questionDotToken) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    propagateChildrenFlags(node.arguments) |
+                    8 /* ContainsES2020 */;
+            if (node.typeArguments) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            parsingContext = saveParsingContext;
-            return createNodeArray(list, listPos);
+            if (ts.isSuperProperty(node.expression)) {
+                node.transformFlags |= 4096 /* ContainsLexicalThis */;
+            }
+            return node;
         }
-        function parseListElement(parsingContext, parseElement) {
-            var node = currentNode(parsingContext);
-            if (node) {
-                return consumeNode(node);
+        // @api
+        function updateCallChain(node, expression, questionDotToken, typeArguments, argumentsArray) {
+            ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a CallExpression using updateCallChain. Use updateCall instead.");
+            return node.expression !== expression
+                || node.questionDotToken !== questionDotToken
+                || node.typeArguments !== typeArguments
+                || node.arguments !== argumentsArray
+                ? update(createCallChain(expression, questionDotToken, typeArguments, argumentsArray), node)
+                : node;
+        }
+        // @api
+        function createNewExpression(expression, typeArguments, argumentsArray) {
+            var node = createBaseExpression(204 /* NewExpression */);
+            node.expression = parenthesizerRules().parenthesizeExpressionOfNew(expression);
+            node.typeArguments = asNodeArray(typeArguments);
+            node.arguments = argumentsArray ? parenthesizerRules().parenthesizeExpressionsOfCommaDelimitedList(argumentsArray) : undefined;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    propagateChildrenFlags(node.arguments) |
+                    8 /* ContainsES2020 */;
+            if (node.typeArguments) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            return parseElement();
+            return node;
         }
-        function currentNode(parsingContext) {
-            // If we don't have a cursor or the parsing context isn't reusable, there's nothing to reuse.
-            //
-            // If there is an outstanding parse error that we've encountered, but not attached to
-            // some node, then we cannot get a node from the old source tree.  This is because we
-            // want to mark the next node we encounter as being unusable.
-            //
-            // Note: This may be too conservative.  Perhaps we could reuse the node and set the bit
-            // on it (or its leftmost child) as having the error.  For now though, being conservative
-            // is nice and likely won't ever affect perf.
-            if (!syntaxCursor || !isReusableParsingContext(parsingContext) || parseErrorBeforeNextFinishedNode) {
-                return undefined;
+        // @api
+        function updateNewExpression(node, expression, typeArguments, argumentsArray) {
+            return node.expression !== expression
+                || node.typeArguments !== typeArguments
+                || node.arguments !== argumentsArray
+                ? update(createNewExpression(expression, typeArguments, argumentsArray), node)
+                : node;
+        }
+        // @api
+        function createTaggedTemplateExpression(tag, typeArguments, template) {
+            var node = createBaseExpression(205 /* TaggedTemplateExpression */);
+            node.tag = parenthesizerRules().parenthesizeLeftSideOfAccess(tag);
+            node.typeArguments = asNodeArray(typeArguments);
+            node.template = template;
+            node.transformFlags |=
+                propagateChildFlags(node.tag) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    propagateChildFlags(node.template) |
+                    256 /* ContainsES2015 */;
+            if (node.typeArguments) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            var node = syntaxCursor.currentNode(scanner.getStartPos());
-            // Can't reuse a missing node.
-            // Can't reuse a node that intersected the change range.
-            // Can't reuse a node that contains a parse error.  This is necessary so that we
-            // produce the same set of errors again.
-            if (ts.nodeIsMissing(node) || node.intersectsChange || ts.containsParseError(node)) {
-                return undefined;
+            if (ts.hasInvalidEscape(node.template)) {
+                node.transformFlags |= 32 /* ContainsES2018 */;
             }
-            // We can only reuse a node if it was parsed under the same strict mode that we're
-            // currently in.  i.e. if we originally parsed a node in non-strict mode, but then
-            // the user added 'using strict' at the top of the file, then we can't use that node
-            // again as the presence of strict mode may cause us to parse the tokens in the file
-            // differently.
-            //
-            // Note: we *can* reuse tokens when the strict mode changes.  That's because tokens
-            // are unaffected by strict mode.  It's just the parser will decide what to do with it
-            // differently depending on what mode it is in.
-            //
-            // This also applies to all our other context flags as well.
-            var nodeContextFlags = node.flags & 25358336 /* ContextFlags */;
-            if (nodeContextFlags !== contextFlags) {
-                return undefined;
+            return node;
+        }
+        // @api
+        function updateTaggedTemplateExpression(node, tag, typeArguments, template) {
+            return node.tag !== tag
+                || node.typeArguments !== typeArguments
+                || node.template !== template
+                ? update(createTaggedTemplateExpression(tag, typeArguments, template), node)
+                : node;
+        }
+        // @api
+        function createTypeAssertion(type, expression) {
+            var node = createBaseExpression(206 /* TypeAssertionExpression */);
+            node.expression = parenthesizerRules().parenthesizeOperandOfPrefixUnary(expression);
+            node.type = type;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.type) |
+                    1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateTypeAssertion(node, type, expression) {
+            return node.type !== type
+                || node.expression !== expression
+                ? update(createTypeAssertion(type, expression), node)
+                : node;
+        }
+        // @api
+        function createParenthesizedExpression(expression) {
+            var node = createBaseExpression(207 /* ParenthesizedExpression */);
+            node.expression = expression;
+            node.transformFlags = propagateChildFlags(node.expression);
+            return node;
+        }
+        // @api
+        function updateParenthesizedExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createParenthesizedExpression(expression), node)
+                : node;
+        }
+        // @api
+        function createFunctionExpression(modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
+            var node = createBaseFunctionLikeDeclaration(208 /* FunctionExpression */, 
+            /*decorators*/ undefined, modifiers, name, typeParameters, parameters, type, body);
+            node.asteriskToken = asteriskToken;
+            node.transformFlags |= propagateChildFlags(node.asteriskToken);
+            if (node.typeParameters) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            // Ok, we have a node that looks like it could be reused.  Now verify that it is valid
-            // in the current list parsing context that we're currently at.
-            if (!canReuseNode(node, parsingContext)) {
-                return undefined;
+            if (ts.modifiersToFlags(node.modifiers) & 256 /* Async */) {
+                if (node.asteriskToken) {
+                    node.transformFlags |= 32 /* ContainsES2018 */;
+                }
+                else {
+                    node.transformFlags |= 64 /* ContainsES2017 */;
+                }
             }
-            if (node.jsDocCache) {
-                // jsDocCache may include tags from parent nodes, which might have been modified.
-                node.jsDocCache = undefined;
+            else if (node.asteriskToken) {
+                node.transformFlags |= 512 /* ContainsGenerator */;
             }
             return node;
         }
-        function consumeNode(node) {
-            // Move the scanner so it is after the node we just consumed.
-            scanner.setTextPos(node.end);
-            nextToken();
+        // @api
+        function updateFunctionExpression(node, modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
+            return node.name !== name
+                || node.modifiers !== modifiers
+                || node.asteriskToken !== asteriskToken
+                || node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createFunctionExpression(modifiers, asteriskToken, name, typeParameters, parameters, type, body), node)
+                : node;
+        }
+        // @api
+        function createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body) {
+            var node = createBaseFunctionLikeDeclaration(209 /* ArrowFunction */, 
+            /*decorators*/ undefined, modifiers, 
+            /*name*/ undefined, typeParameters, parameters, type, parenthesizerRules().parenthesizeConciseBodyOfArrowFunction(body));
+            node.equalsGreaterThanToken = equalsGreaterThanToken !== null && equalsGreaterThanToken !== void 0 ? equalsGreaterThanToken : createToken(38 /* EqualsGreaterThanToken */);
+            node.transformFlags |=
+                propagateChildFlags(node.equalsGreaterThanToken) |
+                    256 /* ContainsES2015 */;
+            if (ts.modifiersToFlags(node.modifiers) & 256 /* Async */) {
+                node.transformFlags |= 64 /* ContainsES2017 */;
+            }
             return node;
         }
-        function isReusableParsingContext(parsingContext) {
-            switch (parsingContext) {
-                case 5 /* ClassMembers */:
-                case 2 /* SwitchClauses */:
-                case 0 /* SourceElements */:
-                case 1 /* BlockStatements */:
-                case 3 /* SwitchClauseStatements */:
-                case 6 /* EnumMembers */:
-                case 4 /* TypeMembers */:
-                case 8 /* VariableDeclarations */:
-                case 17 /* JSDocParameters */:
-                case 16 /* Parameters */:
-                    return true;
-            }
-            return false;
+        // @api
+        function updateArrowFunction(node, modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body) {
+            return node.modifiers !== modifiers
+                || node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                || node.equalsGreaterThanToken !== equalsGreaterThanToken
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body), node)
+                : node;
+        }
+        // @api
+        function createDeleteExpression(expression) {
+            var node = createBaseExpression(210 /* DeleteExpression */);
+            node.expression = parenthesizerRules().parenthesizeOperandOfPrefixUnary(expression);
+            node.transformFlags |= propagateChildFlags(node.expression);
+            return node;
         }
-        function canReuseNode(node, parsingContext) {
-            switch (parsingContext) {
-                case 5 /* ClassMembers */:
-                    return isReusableClassMember(node);
-                case 2 /* SwitchClauses */:
-                    return isReusableSwitchClause(node);
-                case 0 /* SourceElements */:
-                case 1 /* BlockStatements */:
-                case 3 /* SwitchClauseStatements */:
-                    return isReusableStatement(node);
-                case 6 /* EnumMembers */:
-                    return isReusableEnumMember(node);
-                case 4 /* TypeMembers */:
-                    return isReusableTypeMember(node);
-                case 8 /* VariableDeclarations */:
-                    return isReusableVariableDeclaration(node);
-                case 17 /* JSDocParameters */:
-                case 16 /* Parameters */:
-                    return isReusableParameter(node);
-                // Any other lists we do not care about reusing nodes in.  But feel free to add if
-                // you can do so safely.  Danger areas involve nodes that may involve speculative
-                // parsing.  If speculative parsing is involved with the node, then the range the
-                // parser reached while looking ahead might be in the edited range (see the example
-                // in canReuseVariableDeclaratorNode for a good case of this).
-                // case ParsingContext.HeritageClauses:
-                // This would probably be safe to reuse.  There is no speculative parsing with
-                // heritage clauses.
-                // case ParsingContext.TypeParameters:
-                // This would probably be safe to reuse.  There is no speculative parsing with
-                // type parameters.  Note that that's because type *parameters* only occur in
-                // unambiguous *type* contexts.  While type *arguments* occur in very ambiguous
-                // *expression* contexts.
-                // case ParsingContext.TupleElementTypes:
-                // This would probably be safe to reuse.  There is no speculative parsing with
-                // tuple types.
-                // Technically, type argument list types are probably safe to reuse.  While
-                // speculative parsing is involved with them (since type argument lists are only
-                // produced from speculative parsing a < as a type argument list), we only have
-                // the types because speculative parsing succeeded.  Thus, the lookahead never
-                // went past the end of the list and rewound.
-                // case ParsingContext.TypeArguments:
-                // Note: these are almost certainly not safe to ever reuse.  Expressions commonly
-                // need a large amount of lookahead, and we should not reuse them as they may
-                // have actually intersected the edit.
-                // case ParsingContext.ArgumentExpressions:
-                // This is not safe to reuse for the same reason as the 'AssignmentExpression'
-                // cases.  i.e. a property assignment may end with an expression, and thus might
-                // have lookahead far beyond it's old node.
-                // case ParsingContext.ObjectLiteralMembers:
-                // This is probably not safe to reuse.  There can be speculative parsing with
-                // type names in a heritage clause.  There can be generic names in the type
-                // name list, and there can be left hand side expressions (which can have type
-                // arguments.)
-                // case ParsingContext.HeritageClauseElement:
-                // Perhaps safe to reuse, but it's unlikely we'd see more than a dozen attributes
-                // on any given element. Same for children.
-                // case ParsingContext.JsxAttributes:
-                // case ParsingContext.JsxChildren:
-            }
-            return false;
+        // @api
+        function updateDeleteExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createDeleteExpression(expression), node)
+                : node;
+        }
+        // @api
+        function createTypeOfExpression(expression) {
+            var node = createBaseExpression(211 /* TypeOfExpression */);
+            node.expression = parenthesizerRules().parenthesizeOperandOfPrefixUnary(expression);
+            node.transformFlags |= propagateChildFlags(node.expression);
+            return node;
         }
-        function isReusableClassMember(node) {
-            if (node) {
-                switch (node.kind) {
-                    case 162 /* Constructor */:
-                    case 167 /* IndexSignature */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                    case 159 /* PropertyDeclaration */:
-                    case 222 /* SemicolonClassElement */:
-                        return true;
-                    case 161 /* MethodDeclaration */:
-                        // Method declarations are not necessarily reusable.  An object-literal
-                        // may have a method calls "constructor(...)" and we must reparse that
-                        // into an actual .ConstructorDeclaration.
-                        var methodDeclaration = node;
-                        var nameIsConstructor = methodDeclaration.name.kind === 75 /* Identifier */ &&
-                            methodDeclaration.name.originalKeywordKind === 129 /* ConstructorKeyword */;
-                        return !nameIsConstructor;
-                }
-            }
-            return false;
+        // @api
+        function updateTypeOfExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createTypeOfExpression(expression), node)
+                : node;
+        }
+        // @api
+        function createVoidExpression(expression) {
+            var node = createBaseExpression(212 /* VoidExpression */);
+            node.expression = parenthesizerRules().parenthesizeOperandOfPrefixUnary(expression);
+            node.transformFlags |= propagateChildFlags(node.expression);
+            return node;
         }
-        function isReusableSwitchClause(node) {
-            if (node) {
-                switch (node.kind) {
-                    case 277 /* CaseClause */:
-                    case 278 /* DefaultClause */:
-                        return true;
-                }
+        // @api
+        function updateVoidExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createVoidExpression(expression), node)
+                : node;
+        }
+        // @api
+        function createAwaitExpression(expression) {
+            var node = createBaseExpression(213 /* AwaitExpression */);
+            node.expression = parenthesizerRules().parenthesizeOperandOfPrefixUnary(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    64 /* ContainsES2017 */ |
+                    32 /* ContainsES2018 */ |
+                    524288 /* ContainsAwait */;
+            return node;
+        }
+        // @api
+        function updateAwaitExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createAwaitExpression(expression), node)
+                : node;
+        }
+        // @api
+        function createPrefixUnaryExpression(operator, operand) {
+            var node = createBaseExpression(214 /* PrefixUnaryExpression */);
+            node.operator = operator;
+            node.operand = parenthesizerRules().parenthesizeOperandOfPrefixUnary(operand);
+            node.transformFlags |= propagateChildFlags(node.operand);
+            return node;
+        }
+        // @api
+        function updatePrefixUnaryExpression(node, operand) {
+            return node.operand !== operand
+                ? update(createPrefixUnaryExpression(node.operator, operand), node)
+                : node;
+        }
+        // @api
+        function createPostfixUnaryExpression(operand, operator) {
+            var node = createBaseExpression(215 /* PostfixUnaryExpression */);
+            node.operator = operator;
+            node.operand = parenthesizerRules().parenthesizeOperandOfPostfixUnary(operand);
+            node.transformFlags = propagateChildFlags(node.operand);
+            return node;
+        }
+        // @api
+        function updatePostfixUnaryExpression(node, operand) {
+            return node.operand !== operand
+                ? update(createPostfixUnaryExpression(operand, node.operator), node)
+                : node;
+        }
+        // @api
+        function createBinaryExpression(left, operator, right) {
+            var node = createBaseExpression(216 /* BinaryExpression */);
+            var operatorToken = asToken(operator);
+            var operatorKind = operatorToken.kind;
+            node.left = parenthesizerRules().parenthesizeLeftSideOfBinary(operatorKind, left);
+            node.operatorToken = operatorToken;
+            node.right = parenthesizerRules().parenthesizeRightSideOfBinary(operatorKind, node.left, right);
+            node.transformFlags |=
+                propagateChildFlags(node.left) |
+                    propagateChildFlags(node.operatorToken) |
+                    propagateChildFlags(node.right);
+            if (operatorKind === 60 /* QuestionQuestionToken */) {
+                node.transformFlags |= 8 /* ContainsES2020 */;
+            }
+            else if (operatorKind === 62 /* EqualsToken */) {
+                if (ts.isObjectLiteralExpression(node.left)) {
+                    node.transformFlags |=
+                        256 /* ContainsES2015 */ |
+                            32 /* ContainsES2018 */ |
+                            1024 /* ContainsDestructuringAssignment */ |
+                            propagateAssignmentPatternFlags(node.left);
+                }
+                else if (ts.isArrayLiteralExpression(node.left)) {
+                    node.transformFlags |=
+                        256 /* ContainsES2015 */ |
+                            1024 /* ContainsDestructuringAssignment */ |
+                            propagateAssignmentPatternFlags(node.left);
+                }
+            }
+            else if (operatorKind === 42 /* AsteriskAsteriskToken */ || operatorKind === 66 /* AsteriskAsteriskEqualsToken */) {
+                node.transformFlags |= 128 /* ContainsES2016 */;
+            }
+            else if (ts.isLogicalOrCoalescingAssignmentOperator(operatorKind)) {
+                node.transformFlags |= 4 /* ContainsESNext */;
             }
-            return false;
+            return node;
         }
-        function isReusableStatement(node) {
-            if (node) {
-                switch (node.kind) {
-                    case 244 /* FunctionDeclaration */:
-                    case 225 /* VariableStatement */:
-                    case 223 /* Block */:
-                    case 227 /* IfStatement */:
-                    case 226 /* ExpressionStatement */:
-                    case 239 /* ThrowStatement */:
-                    case 235 /* ReturnStatement */:
-                    case 237 /* SwitchStatement */:
-                    case 234 /* BreakStatement */:
-                    case 233 /* ContinueStatement */:
-                    case 231 /* ForInStatement */:
-                    case 232 /* ForOfStatement */:
-                    case 230 /* ForStatement */:
-                    case 229 /* WhileStatement */:
-                    case 236 /* WithStatement */:
-                    case 224 /* EmptyStatement */:
-                    case 240 /* TryStatement */:
-                    case 238 /* LabeledStatement */:
-                    case 228 /* DoStatement */:
-                    case 241 /* DebuggerStatement */:
-                    case 254 /* ImportDeclaration */:
-                    case 253 /* ImportEqualsDeclaration */:
-                    case 260 /* ExportDeclaration */:
-                    case 259 /* ExportAssignment */:
-                    case 249 /* ModuleDeclaration */:
-                    case 245 /* ClassDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 248 /* EnumDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                        return true;
+        function propagateAssignmentPatternFlags(node) {
+            if (node.transformFlags & 16384 /* ContainsObjectRestOrSpread */)
+                return 16384 /* ContainsObjectRestOrSpread */;
+            if (node.transformFlags & 32 /* ContainsES2018 */) {
+                // check for nested spread assignments, otherwise '{ x: { a, ...b } = foo } = c'
+                // will not be correctly interpreted by the ES2018 transformer
+                for (var _i = 0, _a = ts.getElementsOfBindingOrAssignmentPattern(node); _i < _a.length; _i++) {
+                    var element = _a[_i];
+                    var target = ts.getTargetOfBindingOrAssignmentElement(element);
+                    if (target && ts.isAssignmentPattern(target)) {
+                        if (target.transformFlags & 16384 /* ContainsObjectRestOrSpread */) {
+                            return 16384 /* ContainsObjectRestOrSpread */;
+                        }
+                        if (target.transformFlags & 32 /* ContainsES2018 */) {
+                            var flags_1 = propagateAssignmentPatternFlags(target);
+                            if (flags_1)
+                                return flags_1;
+                        }
+                    }
                 }
             }
-            return false;
+            return 0 /* None */;
         }
-        function isReusableEnumMember(node) {
-            return node.kind === 284 /* EnumMember */;
+        // @api
+        function updateBinaryExpression(node, left, operator, right) {
+            return node.left !== left
+                || node.operatorToken !== operator
+                || node.right !== right
+                ? update(createBinaryExpression(left, operator, right), node)
+                : node;
+        }
+        // @api
+        function createConditionalExpression(condition, questionToken, whenTrue, colonToken, whenFalse) {
+            var node = createBaseExpression(217 /* ConditionalExpression */);
+            node.condition = parenthesizerRules().parenthesizeConditionOfConditionalExpression(condition);
+            node.questionToken = questionToken !== null && questionToken !== void 0 ? questionToken : createToken(57 /* QuestionToken */);
+            node.whenTrue = parenthesizerRules().parenthesizeBranchOfConditionalExpression(whenTrue);
+            node.colonToken = colonToken !== null && colonToken !== void 0 ? colonToken : createToken(58 /* ColonToken */);
+            node.whenFalse = parenthesizerRules().parenthesizeBranchOfConditionalExpression(whenFalse);
+            node.transformFlags |=
+                propagateChildFlags(node.condition) |
+                    propagateChildFlags(node.questionToken) |
+                    propagateChildFlags(node.whenTrue) |
+                    propagateChildFlags(node.colonToken) |
+                    propagateChildFlags(node.whenFalse);
+            return node;
         }
-        function isReusableTypeMember(node) {
-            if (node) {
-                switch (node.kind) {
-                    case 166 /* ConstructSignature */:
-                    case 160 /* MethodSignature */:
-                    case 167 /* IndexSignature */:
-                    case 158 /* PropertySignature */:
-                    case 165 /* CallSignature */:
-                        return true;
+        // @api
+        function updateConditionalExpression(node, condition, questionToken, whenTrue, colonToken, whenFalse) {
+            return node.condition !== condition
+                || node.questionToken !== questionToken
+                || node.whenTrue !== whenTrue
+                || node.colonToken !== colonToken
+                || node.whenFalse !== whenFalse
+                ? update(createConditionalExpression(condition, questionToken, whenTrue, colonToken, whenFalse), node)
+                : node;
+        }
+        // @api
+        function createTemplateExpression(head, templateSpans) {
+            var node = createBaseExpression(218 /* TemplateExpression */);
+            node.head = head;
+            node.templateSpans = createNodeArray(templateSpans);
+            node.transformFlags |=
+                propagateChildFlags(node.head) |
+                    propagateChildrenFlags(node.templateSpans) |
+                    256 /* ContainsES2015 */;
+            return node;
+        }
+        // @api
+        function updateTemplateExpression(node, head, templateSpans) {
+            return node.head !== head
+                || node.templateSpans !== templateSpans
+                ? update(createTemplateExpression(head, templateSpans), node)
+                : node;
+        }
+        function createTemplateLiteralLikeNodeChecked(kind, text, rawText, templateFlags) {
+            if (templateFlags === void 0) { templateFlags = 0 /* None */; }
+            ts.Debug.assert(!(templateFlags & ~2048 /* TemplateLiteralLikeFlags */), "Unsupported template flags.");
+            // NOTE: without the assignment to `undefined`, we don't narrow the initial type of `cooked`.
+            // eslint-disable-next-line no-undef-init
+            var cooked = undefined;
+            if (rawText !== undefined && rawText !== text) {
+                cooked = getCookedText(kind, rawText);
+                if (typeof cooked === "object") {
+                    return ts.Debug.fail("Invalid raw text");
                 }
             }
-            return false;
-        }
-        function isReusableVariableDeclaration(node) {
-            if (node.kind !== 242 /* VariableDeclaration */) {
-                return false;
+            if (text === undefined) {
+                if (cooked === undefined) {
+                    return ts.Debug.fail("Arguments 'text' and 'rawText' may not both be undefined.");
+                }
+                text = cooked;
             }
-            // Very subtle incremental parsing bug.  Consider the following code:
-            //
-            //      let v = new List < A, B
-            //
-            // This is actually legal code.  It's a list of variable declarators "v = new List<A"
-            // on one side and "B" on the other. If you then change that to:
-            //
-            //      let v = new List < A, B >()
-            //
-            // then we have a problem.  "v = new List<A" doesn't intersect the change range, so we
-            // start reparsing at "B" and we completely fail to handle this properly.
-            //
-            // In order to prevent this, we do not allow a variable declarator to be reused if it
-            // has an initializer.
-            var variableDeclarator = node;
-            return variableDeclarator.initializer === undefined;
-        }
-        function isReusableParameter(node) {
-            if (node.kind !== 156 /* Parameter */) {
-                return false;
+            else if (cooked !== undefined) {
+                ts.Debug.assert(text === cooked, "Expected argument 'text' to be the normalized (i.e. 'cooked') version of argument 'rawText'.");
             }
-            // See the comment in isReusableVariableDeclaration for why we do this.
-            var parameter = node;
-            return parameter.initializer === undefined;
+            return createTemplateLiteralLikeNode(kind, text, rawText, templateFlags);
         }
-        // Returns true if we should abort parsing.
-        function abortParsingListOrMoveToNextToken(kind) {
-            parseErrorAtCurrentToken(parsingContextErrors(kind));
-            if (isInSomeParsingContext()) {
-                return true;
+        // @api
+        function createTemplateLiteralLikeNode(kind, text, rawText, templateFlags) {
+            var node = createBaseToken(kind);
+            node.text = text;
+            node.rawText = rawText;
+            node.templateFlags = templateFlags & 2048 /* TemplateLiteralLikeFlags */;
+            node.transformFlags |= 256 /* ContainsES2015 */;
+            if (node.templateFlags) {
+                node.transformFlags |= 32 /* ContainsES2018 */;
             }
-            nextToken();
-            return false;
+            return node;
         }
-        function parsingContextErrors(context) {
-            switch (context) {
-                case 0 /* SourceElements */: return ts.Diagnostics.Declaration_or_statement_expected;
-                case 1 /* BlockStatements */: return ts.Diagnostics.Declaration_or_statement_expected;
-                case 2 /* SwitchClauses */: return ts.Diagnostics.case_or_default_expected;
-                case 3 /* SwitchClauseStatements */: return ts.Diagnostics.Statement_expected;
-                case 18 /* RestProperties */: // fallthrough
-                case 4 /* TypeMembers */: return ts.Diagnostics.Property_or_signature_expected;
-                case 5 /* ClassMembers */: return ts.Diagnostics.Unexpected_token_A_constructor_method_accessor_or_property_was_expected;
-                case 6 /* EnumMembers */: return ts.Diagnostics.Enum_member_expected;
-                case 7 /* HeritageClauseElement */: return ts.Diagnostics.Expression_expected;
-                case 8 /* VariableDeclarations */: return ts.Diagnostics.Variable_declaration_expected;
-                case 9 /* ObjectBindingElements */: return ts.Diagnostics.Property_destructuring_pattern_expected;
-                case 10 /* ArrayBindingElements */: return ts.Diagnostics.Array_element_destructuring_pattern_expected;
-                case 11 /* ArgumentExpressions */: return ts.Diagnostics.Argument_expression_expected;
-                case 12 /* ObjectLiteralMembers */: return ts.Diagnostics.Property_assignment_expected;
-                case 15 /* ArrayLiteralMembers */: return ts.Diagnostics.Expression_or_comma_expected;
-                case 17 /* JSDocParameters */: return ts.Diagnostics.Parameter_declaration_expected;
-                case 16 /* Parameters */: return ts.Diagnostics.Parameter_declaration_expected;
-                case 19 /* TypeParameters */: return ts.Diagnostics.Type_parameter_declaration_expected;
-                case 20 /* TypeArguments */: return ts.Diagnostics.Type_argument_expected;
-                case 21 /* TupleElementTypes */: return ts.Diagnostics.Type_expected;
-                case 22 /* HeritageClauses */: return ts.Diagnostics.Unexpected_token_expected;
-                case 23 /* ImportOrExportSpecifiers */: return ts.Diagnostics.Identifier_expected;
-                case 13 /* JsxAttributes */: return ts.Diagnostics.Identifier_expected;
-                case 14 /* JsxChildren */: return ts.Diagnostics.Identifier_expected;
-                default: return undefined; // TODO: GH#18217 `default: Debug.assertNever(context);`
-            }
+        // @api
+        function createTemplateHead(text, rawText, templateFlags) {
+            return createTemplateLiteralLikeNodeChecked(15 /* TemplateHead */, text, rawText, templateFlags);
         }
-        // Parses a comma-delimited list of elements
-        function parseDelimitedList(kind, parseElement, considerSemicolonAsDelimiter) {
-            var saveParsingContext = parsingContext;
-            parsingContext |= 1 << kind;
-            var list = [];
-            var listPos = getNodePos();
-            var commaStart = -1; // Meaning the previous token was not a comma
-            while (true) {
-                if (isListElement(kind, /*inErrorRecovery*/ false)) {
-                    var startPos = scanner.getStartPos();
-                    list.push(parseListElement(kind, parseElement));
-                    commaStart = scanner.getTokenPos();
-                    if (parseOptional(27 /* CommaToken */)) {
-                        // No need to check for a zero length node since we know we parsed a comma
-                        continue;
-                    }
-                    commaStart = -1; // Back to the state where the last token was not a comma
-                    if (isListTerminator(kind)) {
-                        break;
-                    }
-                    // We didn't get a comma, and the list wasn't terminated, explicitly parse
-                    // out a comma so we give a good error message.
-                    parseExpected(27 /* CommaToken */, getExpectedCommaDiagnostic(kind));
-                    // If the token was a semicolon, and the caller allows that, then skip it and
-                    // continue.  This ensures we get back on track and don't result in tons of
-                    // parse errors.  For example, this can happen when people do things like use
-                    // a semicolon to delimit object literal members.   Note: we'll have already
-                    // reported an error when we called parseExpected above.
-                    if (considerSemicolonAsDelimiter && token() === 26 /* SemicolonToken */ && !scanner.hasPrecedingLineBreak()) {
-                        nextToken();
-                    }
-                    if (startPos === scanner.getStartPos()) {
-                        // What we're parsing isn't actually remotely recognizable as a element and we've consumed no tokens whatsoever
-                        // Consume a token to advance the parser in some way and avoid an infinite loop
-                        // This can happen when we're speculatively parsing parenthesized expressions which we think may be arrow functions,
-                        // or when a modifier keyword which is disallowed as a parameter name (ie, `static` in strict mode) is supplied
-                        nextToken();
-                    }
-                    continue;
-                }
-                if (isListTerminator(kind)) {
-                    break;
-                }
-                if (abortParsingListOrMoveToNextToken(kind)) {
-                    break;
-                }
-            }
-            parsingContext = saveParsingContext;
-            var result = createNodeArray(list, listPos);
-            // Recording the trailing comma is deliberately done after the previous
-            // loop, and not just if we see a list terminator. This is because the list
-            // may have ended incorrectly, but it is still important to know if there
-            // was a trailing comma.
-            // Check if the last token was a comma.
-            if (commaStart >= 0) {
-                // Always preserve a trailing comma by marking it on the NodeArray
-                result.hasTrailingComma = true;
-            }
-            return result;
+        // @api
+        function createTemplateMiddle(text, rawText, templateFlags) {
+            return createTemplateLiteralLikeNodeChecked(16 /* TemplateMiddle */, text, rawText, templateFlags);
         }
-        function getExpectedCommaDiagnostic(kind) {
-            return kind === 6 /* EnumMembers */ ? ts.Diagnostics.An_enum_member_name_must_be_followed_by_a_or : undefined;
-        }
-        function createMissingList() {
-            var list = createNodeArray([], getNodePos());
-            list.isMissingList = true;
-            return list;
+        // @api
+        function createTemplateTail(text, rawText, templateFlags) {
+            return createTemplateLiteralLikeNodeChecked(17 /* TemplateTail */, text, rawText, templateFlags);
         }
-        function isMissingList(arr) {
-            return !!arr.isMissingList;
+        // @api
+        function createNoSubstitutionTemplateLiteral(text, rawText, templateFlags) {
+            return createTemplateLiteralLikeNodeChecked(14 /* NoSubstitutionTemplateLiteral */, text, rawText, templateFlags);
         }
-        function parseBracketedList(kind, parseElement, open, close) {
-            if (parseExpected(open)) {
-                var result = parseDelimitedList(kind, parseElement);
-                parseExpected(close);
-                return result;
-            }
-            return createMissingList();
+        // @api
+        function createYieldExpression(asteriskToken, expression) {
+            ts.Debug.assert(!asteriskToken || !!expression, "A `YieldExpression` with an asteriskToken must have an expression.");
+            var node = createBaseExpression(219 /* YieldExpression */);
+            node.expression = expression && parenthesizerRules().parenthesizeExpressionForDisallowedComma(expression);
+            node.asteriskToken = asteriskToken;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.asteriskToken) |
+                    256 /* ContainsES2015 */ |
+                    32 /* ContainsES2018 */ |
+                    262144 /* ContainsYield */;
+            return node;
         }
-        function parseEntityName(allowReservedWords, diagnosticMessage) {
-            var entity = allowReservedWords ? parseIdentifierName(diagnosticMessage) : parseIdentifier(diagnosticMessage);
-            var dotPos = scanner.getStartPos();
-            while (parseOptional(24 /* DotToken */)) {
-                if (token() === 29 /* LessThanToken */) {
-                    // the entity is part of a JSDoc-style generic, so record the trailing dot for later error reporting
-                    entity.jsdocDotPos = dotPos;
-                    break;
-                }
-                dotPos = scanner.getStartPos();
-                entity = createQualifiedName(entity, parseRightSideOfDot(allowReservedWords, /* allowPrivateIdentifiers */ false));
-            }
-            return entity;
+        // @api
+        function updateYieldExpression(node, asteriskToken, expression) {
+            return node.expression !== expression
+                || node.asteriskToken !== asteriskToken
+                ? update(createYieldExpression(asteriskToken, expression), node)
+                : node;
+        }
+        // @api
+        function createSpreadElement(expression) {
+            var node = createBaseExpression(220 /* SpreadElement */);
+            node.expression = parenthesizerRules().parenthesizeExpressionForDisallowedComma(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    256 /* ContainsES2015 */ |
+                    8192 /* ContainsRestOrSpread */;
+            return node;
         }
-        function createQualifiedName(entity, name) {
-            var node = createNode(153 /* QualifiedName */, entity.pos);
-            node.left = entity;
-            node.right = name;
-            return finishNode(node);
+        // @api
+        function updateSpreadElement(node, expression) {
+            return node.expression !== expression
+                ? update(createSpreadElement(expression), node)
+                : node;
         }
-        function parseRightSideOfDot(allowIdentifierNames, allowPrivateIdentifiers) {
-            // Technically a keyword is valid here as all identifiers and keywords are identifier names.
-            // However, often we'll encounter this in error situations when the identifier or keyword
-            // is actually starting another valid construct.
-            //
-            // So, we check for the following specific case:
-            //
-            //      name.
-            //      identifierOrKeyword identifierNameOrKeyword
-            //
-            // Note: the newlines are important here.  For example, if that above code
-            // were rewritten into:
-            //
-            //      name.identifierOrKeyword
-            //      identifierNameOrKeyword
-            //
-            // Then we would consider it valid.  That's because ASI would take effect and
-            // the code would be implicitly: "name.identifierOrKeyword; identifierNameOrKeyword".
-            // In the first case though, ASI will not take effect because there is not a
-            // line terminator after the identifier or keyword.
-            if (scanner.hasPrecedingLineBreak() && ts.tokenIsIdentifierOrKeyword(token())) {
-                var matchesPattern = lookAhead(nextTokenIsIdentifierOrKeywordOnSameLine);
-                if (matchesPattern) {
-                    // Report that we need an identifier.  However, report it right after the dot,
-                    // and not on the next token.  This is because the next token might actually
-                    // be an identifier and the error would be quite confusing.
-                    return createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Identifier_expected);
-                }
-            }
-            if (token() === 76 /* PrivateIdentifier */) {
-                var node = parsePrivateIdentifier();
-                return allowPrivateIdentifiers ? node : createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Identifier_expected);
-            }
-            return allowIdentifierNames ? parseIdentifierName() : parseIdentifier();
+        // @api
+        function createClassExpression(decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            var node = createBaseClassLikeDeclaration(221 /* ClassExpression */, decorators, modifiers, name, typeParameters, heritageClauses, members);
+            node.transformFlags |= 256 /* ContainsES2015 */;
+            return node;
         }
-        function parseTemplateExpression(isTaggedTemplate) {
-            var template = createNode(211 /* TemplateExpression */);
-            template.head = parseTemplateHead(isTaggedTemplate);
-            ts.Debug.assert(template.head.kind === 15 /* TemplateHead */, "Template head has wrong token kind");
-            var list = [];
-            var listPos = getNodePos();
-            do {
-                list.push(parseTemplateSpan(isTaggedTemplate));
-            } while (ts.last(list).literal.kind === 16 /* TemplateMiddle */);
-            template.templateSpans = createNodeArray(list, listPos);
-            return finishNode(template);
+        // @api
+        function updateClassExpression(node, decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.typeParameters !== typeParameters
+                || node.heritageClauses !== heritageClauses
+                || node.members !== members
+                ? update(createClassExpression(decorators, modifiers, name, typeParameters, heritageClauses, members), node)
+                : node;
+        }
+        // @api
+        function createOmittedExpression() {
+            return createBaseExpression(222 /* OmittedExpression */);
+        }
+        // @api
+        function createExpressionWithTypeArguments(expression, typeArguments) {
+            var node = createBaseNode(223 /* ExpressionWithTypeArguments */);
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.typeArguments = typeArguments && parenthesizerRules().parenthesizeTypeArguments(typeArguments);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    256 /* ContainsES2015 */;
+            return node;
         }
-        function parseTemplateSpan(isTaggedTemplate) {
-            var span = createNode(221 /* TemplateSpan */);
-            span.expression = allowInAnd(parseExpression);
-            var literal;
-            if (token() === 19 /* CloseBraceToken */) {
-                reScanTemplateToken(isTaggedTemplate);
-                literal = parseTemplateMiddleOrTemplateTail();
-            }
-            else {
-                literal = parseExpectedToken(17 /* TemplateTail */, ts.Diagnostics._0_expected, ts.tokenToString(19 /* CloseBraceToken */));
-            }
-            span.literal = literal;
-            return finishNode(span);
+        // @api
+        function updateExpressionWithTypeArguments(node, expression, typeArguments) {
+            return node.expression !== expression
+                || node.typeArguments !== typeArguments
+                ? update(createExpressionWithTypeArguments(expression, typeArguments), node)
+                : node;
         }
-        function parseLiteralNode() {
-            return parseLiteralLikeNode(token());
+        // @api
+        function createAsExpression(expression, type) {
+            var node = createBaseExpression(224 /* AsExpression */);
+            node.expression = expression;
+            node.type = type;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.type) |
+                    1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseTemplateHead(isTaggedTemplate) {
-            if (isTaggedTemplate) {
-                reScanTemplateHeadOrNoSubstitutionTemplate();
-            }
-            var fragment = parseLiteralLikeNode(token());
-            ts.Debug.assert(fragment.kind === 15 /* TemplateHead */, "Template head has wrong token kind");
-            return fragment;
+        // @api
+        function updateAsExpression(node, expression, type) {
+            return node.expression !== expression
+                || node.type !== type
+                ? update(createAsExpression(expression, type), node)
+                : node;
+        }
+        // @api
+        function createNonNullExpression(expression) {
+            var node = createBaseExpression(225 /* NonNullExpression */);
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseTemplateMiddleOrTemplateTail() {
-            var fragment = parseLiteralLikeNode(token());
-            ts.Debug.assert(fragment.kind === 16 /* TemplateMiddle */ || fragment.kind === 17 /* TemplateTail */, "Template fragment has wrong token kind");
-            return fragment;
+        // @api
+        function updateNonNullExpression(node, expression) {
+            if (ts.isNonNullChain(node)) {
+                return updateNonNullChain(node, expression);
+            }
+            return node.expression !== expression
+                ? update(createNonNullExpression(expression), node)
+                : node;
+        }
+        // @api
+        function createNonNullChain(expression) {
+            var node = createBaseExpression(225 /* NonNullExpression */);
+            node.flags |= 32 /* OptionalChain */;
+            node.expression = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseLiteralLikeNode(kind) {
-            var node = createNode(kind);
-            node.text = scanner.getTokenValue();
-            switch (kind) {
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 15 /* TemplateHead */:
-                case 16 /* TemplateMiddle */:
-                case 17 /* TemplateTail */:
-                    var isLast = kind === 14 /* NoSubstitutionTemplateLiteral */ || kind === 17 /* TemplateTail */;
-                    var tokenText = scanner.getTokenText();
-                    node.rawText = tokenText.substring(1, tokenText.length - (scanner.isUnterminated() ? 0 : isLast ? 1 : 2));
+        // @api
+        function updateNonNullChain(node, expression) {
+            ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a NonNullExpression using updateNonNullChain. Use updateNonNullExpression instead.");
+            return node.expression !== expression
+                ? update(createNonNullChain(expression), node)
+                : node;
+        }
+        // @api
+        function createMetaProperty(keywordToken, name) {
+            var node = createBaseExpression(226 /* MetaProperty */);
+            node.keywordToken = keywordToken;
+            node.name = name;
+            node.transformFlags |= propagateChildFlags(node.name);
+            switch (keywordToken) {
+                case 102 /* NewKeyword */:
+                    node.transformFlags |= 256 /* ContainsES2015 */;
                     break;
+                case 99 /* ImportKeyword */:
+                    node.transformFlags |= 4 /* ContainsESNext */;
+                    break;
+                default:
+                    return ts.Debug.assertNever(keywordToken);
             }
-            if (scanner.hasExtendedUnicodeEscape()) {
-                node.hasExtendedUnicodeEscape = true;
-            }
-            if (scanner.isUnterminated()) {
-                node.isUnterminated = true;
-            }
-            // Octal literals are not allowed in strict mode or ES5
-            // Note that theoretically the following condition would hold true literals like 009,
-            // which is not octal.But because of how the scanner separates the tokens, we would
-            // never get a token like this. Instead, we would get 00 and 9 as two separate tokens.
-            // We also do not need to check for negatives because any prefix operator would be part of a
-            // parent unary expression.
-            if (node.kind === 8 /* NumericLiteral */) {
-                node.numericLiteralFlags = scanner.getTokenFlags() & 1008 /* NumericLiteralFlags */;
-            }
-            if (ts.isTemplateLiteralKind(node.kind)) {
-                node.templateFlags = scanner.getTokenFlags() & 2048 /* ContainsInvalidEscape */;
-            }
-            nextToken();
-            finishNode(node);
             return node;
         }
-        // TYPES
-        function parseTypeReference() {
-            var node = createNode(169 /* TypeReference */);
-            node.typeName = parseEntityName(/*allowReservedWords*/ true, ts.Diagnostics.Type_expected);
-            if (!scanner.hasPrecedingLineBreak() && reScanLessThanToken() === 29 /* LessThanToken */) {
-                node.typeArguments = parseBracketedList(20 /* TypeArguments */, parseType, 29 /* LessThanToken */, 31 /* GreaterThanToken */);
-            }
-            return finishNode(node);
+        // @api
+        function updateMetaProperty(node, name) {
+            return node.name !== name
+                ? update(createMetaProperty(node.keywordToken, name), node)
+                : node;
         }
-        // If true, we should abort parsing an error function.
-        function typeHasArrowFunctionBlockingParseError(node) {
-            switch (node.kind) {
-                case 169 /* TypeReference */:
-                    return ts.nodeIsMissing(node.typeName);
-                case 170 /* FunctionType */:
-                case 171 /* ConstructorType */: {
-                    var _a = node, parameters = _a.parameters, type = _a.type;
-                    return isMissingList(parameters) || typeHasArrowFunctionBlockingParseError(type);
-                }
-                case 182 /* ParenthesizedType */:
-                    return typeHasArrowFunctionBlockingParseError(node.type);
-                default:
-                    return false;
-            }
+        //
+        // Misc
+        //
+        // @api
+        function createTemplateSpan(expression, literal) {
+            var node = createBaseNode(228 /* TemplateSpan */);
+            node.expression = expression;
+            node.literal = literal;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.literal) |
+                    256 /* ContainsES2015 */;
+            return node;
         }
-        function parseThisTypePredicate(lhs) {
-            nextToken();
-            var node = createNode(168 /* TypePredicate */, lhs.pos);
-            node.parameterName = lhs;
-            node.type = parseType();
-            return finishNode(node);
+        // @api
+        function updateTemplateSpan(node, expression, literal) {
+            return node.expression !== expression
+                || node.literal !== literal
+                ? update(createTemplateSpan(expression, literal), node)
+                : node;
+        }
+        // @api
+        function createSemicolonClassElement() {
+            var node = createBaseNode(229 /* SemicolonClassElement */);
+            node.transformFlags |= 256 /* ContainsES2015 */;
+            return node;
         }
-        function parseThisTypeNode() {
-            var node = createNode(183 /* ThisType */);
-            nextToken();
-            return finishNode(node);
+        //
+        // Element
+        //
+        // @api
+        function createBlock(statements, multiLine) {
+            var node = createBaseNode(230 /* Block */);
+            node.statements = createNodeArray(statements);
+            node.multiLine = multiLine;
+            node.transformFlags |= propagateChildrenFlags(node.statements);
+            return node;
         }
-        function parseJSDocAllType(postFixEquals) {
-            var result = createNode(295 /* JSDocAllType */);
-            if (postFixEquals) {
-                return createPostfixType(299 /* JSDocOptionalType */, result);
+        // @api
+        function updateBlock(node, statements) {
+            return node.statements !== statements
+                ? update(createBlock(statements, node.multiLine), node)
+                : node;
+        }
+        // @api
+        function createVariableStatement(modifiers, declarationList) {
+            var node = createBaseDeclaration(232 /* VariableStatement */, /*decorators*/ undefined, modifiers);
+            node.declarationList = ts.isArray(declarationList) ? createVariableDeclarationList(declarationList) : declarationList;
+            node.transformFlags |=
+                propagateChildFlags(node.declarationList);
+            if (ts.modifiersToFlags(node.modifiers) & 2 /* Ambient */) {
+                node.transformFlags = 1 /* ContainsTypeScript */;
             }
-            else {
-                nextToken();
-            }
-            return finishNode(result);
+            return node;
         }
-        function parseJSDocNonNullableType() {
-            var result = createNode(298 /* JSDocNonNullableType */);
-            nextToken();
-            result.type = parseNonArrayType();
-            return finishNode(result);
+        // @api
+        function updateVariableStatement(node, modifiers, declarationList) {
+            return node.modifiers !== modifiers
+                || node.declarationList !== declarationList
+                ? update(createVariableStatement(modifiers, declarationList), node)
+                : node;
+        }
+        // @api
+        function createEmptyStatement() {
+            return createBaseNode(231 /* EmptyStatement */);
+        }
+        // @api
+        function createExpressionStatement(expression) {
+            var node = createBaseNode(233 /* ExpressionStatement */);
+            node.expression = parenthesizerRules().parenthesizeExpressionOfExpressionStatement(expression);
+            node.transformFlags |= propagateChildFlags(node.expression);
+            return node;
         }
-        function parseJSDocUnknownOrNullableType() {
-            var pos = scanner.getStartPos();
-            // skip the ?
-            nextToken();
-            // Need to lookahead to decide if this is a nullable or unknown type.
-            // Here are cases where we'll pick the unknown type:
-            //
-            //      Foo(?,
-            //      { a: ? }
-            //      Foo(?)
-            //      Foo<?>
-            //      Foo(?=
-            //      (?|
-            if (token() === 27 /* CommaToken */ ||
-                token() === 19 /* CloseBraceToken */ ||
-                token() === 21 /* CloseParenToken */ ||
-                token() === 31 /* GreaterThanToken */ ||
-                token() === 62 /* EqualsToken */ ||
-                token() === 51 /* BarToken */) {
-                var result = createNode(296 /* JSDocUnknownType */, pos);
-                return finishNode(result);
-            }
-            else {
-                var result = createNode(297 /* JSDocNullableType */, pos);
-                result.type = parseType();
-                return finishNode(result);
-            }
+        // @api
+        function updateExpressionStatement(node, expression) {
+            return node.expression !== expression
+                ? update(createExpressionStatement(expression), node)
+                : node;
         }
-        function parseJSDocFunctionType() {
-            if (lookAhead(nextTokenIsOpenParen)) {
-                var result = createNodeWithJSDoc(300 /* JSDocFunctionType */);
-                nextToken();
-                fillSignature(58 /* ColonToken */, 4 /* Type */ | 32 /* JSDoc */, result);
-                return finishNode(result);
-            }
-            var node = createNode(169 /* TypeReference */);
-            node.typeName = parseIdentifierName();
-            return finishNode(node);
+        // @api
+        function createIfStatement(expression, thenStatement, elseStatement) {
+            var node = createBaseNode(234 /* IfStatement */);
+            node.expression = expression;
+            node.thenStatement = asEmbeddedStatement(thenStatement);
+            node.elseStatement = asEmbeddedStatement(elseStatement);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.thenStatement) |
+                    propagateChildFlags(node.elseStatement);
+            return node;
         }
-        function parseJSDocParameter() {
-            var parameter = createNode(156 /* Parameter */);
-            if (token() === 104 /* ThisKeyword */ || token() === 99 /* NewKeyword */) {
-                parameter.name = parseIdentifierName();
-                parseExpected(58 /* ColonToken */);
-            }
-            parameter.type = parseJSDocType();
-            return finishNode(parameter);
+        // @api
+        function updateIfStatement(node, expression, thenStatement, elseStatement) {
+            return node.expression !== expression
+                || node.thenStatement !== thenStatement
+                || node.elseStatement !== elseStatement
+                ? update(createIfStatement(expression, thenStatement, elseStatement), node)
+                : node;
+        }
+        // @api
+        function createDoStatement(statement, expression) {
+            var node = createBaseNode(235 /* DoStatement */);
+            node.statement = asEmbeddedStatement(statement);
+            node.expression = expression;
+            node.transformFlags |=
+                propagateChildFlags(node.statement) |
+                    propagateChildFlags(node.expression);
+            return node;
         }
-        function parseJSDocType() {
-            scanner.setInJSDocType(true);
-            var moduleSpecifier = parseOptionalToken(135 /* ModuleKeyword */);
-            if (moduleSpecifier) {
-                var moduleTag = createNode(302 /* JSDocNamepathType */, moduleSpecifier.pos);
-                terminate: while (true) {
-                    switch (token()) {
-                        case 19 /* CloseBraceToken */:
-                        case 1 /* EndOfFileToken */:
-                        case 27 /* CommaToken */:
-                        case 5 /* WhitespaceTrivia */:
-                            break terminate;
-                        default:
-                            nextTokenJSDoc();
-                    }
-                }
-                scanner.setInJSDocType(false);
-                return finishNode(moduleTag);
-            }
-            var dotdotdot = parseOptionalToken(25 /* DotDotDotToken */);
-            var type = parseTypeOrTypePredicate();
-            scanner.setInJSDocType(false);
-            if (dotdotdot) {
-                var variadic = createNode(301 /* JSDocVariadicType */, dotdotdot.pos);
-                variadic.type = type;
-                type = finishNode(variadic);
-            }
-            if (token() === 62 /* EqualsToken */) {
-                return createPostfixType(299 /* JSDocOptionalType */, type);
-            }
-            return type;
+        // @api
+        function updateDoStatement(node, statement, expression) {
+            return node.statement !== statement
+                || node.expression !== expression
+                ? update(createDoStatement(statement, expression), node)
+                : node;
         }
-        function parseTypeQuery() {
-            var node = createNode(172 /* TypeQuery */);
-            parseExpected(108 /* TypeOfKeyword */);
-            node.exprName = parseEntityName(/*allowReservedWords*/ true);
-            return finishNode(node);
+        // @api
+        function createWhileStatement(expression, statement) {
+            var node = createBaseNode(236 /* WhileStatement */);
+            node.expression = expression;
+            node.statement = asEmbeddedStatement(statement);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.statement);
+            return node;
         }
-        function parseTypeParameter() {
-            var node = createNode(155 /* TypeParameter */);
-            node.name = parseIdentifier();
-            if (parseOptional(90 /* ExtendsKeyword */)) {
-                // It's not uncommon for people to write improper constraints to a generic.  If the
-                // user writes a constraint that is an expression and not an actual type, then parse
-                // it out as an expression (so we can recover well), but report that a type is needed
-                // instead.
-                if (isStartOfType() || !isStartOfExpression()) {
-                    node.constraint = parseType();
-                }
-                else {
-                    // It was not a type, and it looked like an expression.  Parse out an expression
-                    // here so we recover well.  Note: it is important that we call parseUnaryExpression
-                    // and not parseExpression here.  If the user has:
-                    //
-                    //      <T extends "">
-                    //
-                    // We do *not* want to consume the `>` as we're consuming the expression for "".
-                    node.expression = parseUnaryExpressionOrHigher();
-                }
-            }
-            if (parseOptional(62 /* EqualsToken */)) {
-                node.default = parseType();
-            }
-            return finishNode(node);
+        // @api
+        function updateWhileStatement(node, expression, statement) {
+            return node.expression !== expression
+                || node.statement !== statement
+                ? update(createWhileStatement(expression, statement), node)
+                : node;
+        }
+        // @api
+        function createForStatement(initializer, condition, incrementor, statement) {
+            var node = createBaseNode(237 /* ForStatement */);
+            node.initializer = initializer;
+            node.condition = condition;
+            node.incrementor = incrementor;
+            node.statement = asEmbeddedStatement(statement);
+            node.transformFlags |=
+                propagateChildFlags(node.initializer) |
+                    propagateChildFlags(node.condition) |
+                    propagateChildFlags(node.incrementor) |
+                    propagateChildFlags(node.statement);
+            return node;
         }
-        function parseTypeParameters() {
-            if (token() === 29 /* LessThanToken */) {
-                return parseBracketedList(19 /* TypeParameters */, parseTypeParameter, 29 /* LessThanToken */, 31 /* GreaterThanToken */);
-            }
+        // @api
+        function updateForStatement(node, initializer, condition, incrementor, statement) {
+            return node.initializer !== initializer
+                || node.condition !== condition
+                || node.incrementor !== incrementor
+                || node.statement !== statement
+                ? update(createForStatement(initializer, condition, incrementor, statement), node)
+                : node;
+        }
+        // @api
+        function createForInStatement(initializer, expression, statement) {
+            var node = createBaseNode(238 /* ForInStatement */);
+            node.initializer = initializer;
+            node.expression = expression;
+            node.statement = asEmbeddedStatement(statement);
+            node.transformFlags |=
+                propagateChildFlags(node.initializer) |
+                    propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.statement);
+            return node;
         }
-        function parseParameterType() {
-            if (parseOptional(58 /* ColonToken */)) {
-                return parseType();
-            }
-            return undefined;
+        // @api
+        function updateForInStatement(node, initializer, expression, statement) {
+            return node.initializer !== initializer
+                || node.expression !== expression
+                || node.statement !== statement
+                ? update(createForInStatement(initializer, expression, statement), node)
+                : node;
+        }
+        // @api
+        function createForOfStatement(awaitModifier, initializer, expression, statement) {
+            var node = createBaseNode(239 /* ForOfStatement */);
+            node.awaitModifier = awaitModifier;
+            node.initializer = initializer;
+            node.expression = parenthesizerRules().parenthesizeExpressionForDisallowedComma(expression);
+            node.statement = asEmbeddedStatement(statement);
+            node.transformFlags |=
+                propagateChildFlags(node.awaitModifier) |
+                    propagateChildFlags(node.initializer) |
+                    propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.statement) |
+                    256 /* ContainsES2015 */;
+            if (awaitModifier)
+                node.transformFlags |= 32 /* ContainsES2018 */;
+            return node;
         }
-        function isStartOfParameter(isJSDocParameter) {
-            return token() === 25 /* DotDotDotToken */ ||
-                isIdentifierOrPrivateIdentifierOrPattern() ||
-                ts.isModifierKind(token()) ||
-                token() === 59 /* AtToken */ ||
-                isStartOfType(/*inStartOfParameter*/ !isJSDocParameter);
+        // @api
+        function updateForOfStatement(node, awaitModifier, initializer, expression, statement) {
+            return node.awaitModifier !== awaitModifier
+                || node.initializer !== initializer
+                || node.expression !== expression
+                || node.statement !== statement
+                ? update(createForOfStatement(awaitModifier, initializer, expression, statement), node)
+                : node;
+        }
+        // @api
+        function createContinueStatement(label) {
+            var node = createBaseNode(240 /* ContinueStatement */);
+            node.label = asName(label);
+            node.transformFlags |=
+                propagateChildFlags(node.label) |
+                    1048576 /* ContainsHoistedDeclarationOrCompletion */;
+            return node;
         }
-        function parseParameter() {
-            var node = createNodeWithJSDoc(156 /* Parameter */);
-            if (token() === 104 /* ThisKeyword */) {
-                node.name = createIdentifier(/*isIdentifier*/ true);
-                node.type = parseParameterType();
-                return finishNode(node);
-            }
-            node.decorators = parseDecorators();
-            node.modifiers = parseModifiers();
-            node.dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
-            // FormalParameter [Yield,Await]:
-            //      BindingElement[?Yield,?Await]
-            node.name = parseIdentifierOrPattern(ts.Diagnostics.Private_identifiers_cannot_be_used_as_parameters);
-            if (ts.getFullWidth(node.name) === 0 && !node.modifiers && ts.isModifierKind(token())) {
-                // in cases like
-                // 'use strict'
-                // function foo(static)
-                // isParameter('static') === true, because of isModifier('static')
-                // however 'static' is not a legal identifier in a strict mode.
-                // so result of this function will be ParameterDeclaration (flags = 0, name = missing, type = undefined, initializer = undefined)
-                // and current token will not change => parsing of the enclosing parameter list will last till the end of time (or OOM)
-                // to avoid this we'll advance cursor to the next token.
-                nextToken();
-            }
-            node.questionToken = parseOptionalToken(57 /* QuestionToken */);
-            node.type = parseParameterType();
-            node.initializer = parseInitializer();
-            return finishNode(node);
+        // @api
+        function updateContinueStatement(node, label) {
+            return node.label !== label
+                ? update(createContinueStatement(label), node)
+                : node;
+        }
+        // @api
+        function createBreakStatement(label) {
+            var node = createBaseNode(241 /* BreakStatement */);
+            node.label = asName(label);
+            node.transformFlags |=
+                propagateChildFlags(node.label) |
+                    1048576 /* ContainsHoistedDeclarationOrCompletion */;
+            return node;
         }
-        /**
-         * Note: If returnToken is EqualsGreaterThanToken, `signature.type` will always be defined.
-         * @returns If return type parsing succeeds
-         */
-        function fillSignature(returnToken, flags, signature) {
-            if (!(flags & 32 /* JSDoc */)) {
-                signature.typeParameters = parseTypeParameters();
-            }
-            var parametersParsedSuccessfully = parseParameterList(signature, flags);
-            if (shouldParseReturnType(returnToken, !!(flags & 4 /* Type */))) {
-                signature.type = parseTypeOrTypePredicate();
-                if (typeHasArrowFunctionBlockingParseError(signature.type))
-                    return false;
-            }
-            return parametersParsedSuccessfully;
+        // @api
+        function updateBreakStatement(node, label) {
+            return node.label !== label
+                ? update(createBreakStatement(label), node)
+                : node;
         }
-        function shouldParseReturnType(returnToken, isType) {
-            if (returnToken === 38 /* EqualsGreaterThanToken */) {
-                parseExpected(returnToken);
-                return true;
-            }
-            else if (parseOptional(58 /* ColonToken */)) {
-                return true;
-            }
-            else if (isType && token() === 38 /* EqualsGreaterThanToken */) {
-                // This is easy to get backward, especially in type contexts, so parse the type anyway
-                parseErrorAtCurrentToken(ts.Diagnostics._0_expected, ts.tokenToString(58 /* ColonToken */));
-                nextToken();
-                return true;
-            }
-            return false;
+        // @api
+        function createReturnStatement(expression) {
+            var node = createBaseNode(242 /* ReturnStatement */);
+            node.expression = expression;
+            // return in an ES2018 async generator must be awaited
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    32 /* ContainsES2018 */ |
+                    1048576 /* ContainsHoistedDeclarationOrCompletion */;
+            return node;
         }
-        // Returns true on success.
-        function parseParameterList(signature, flags) {
-            // FormalParameters [Yield,Await]: (modified)
-            //      [empty]
-            //      FormalParameterList[?Yield,Await]
-            //
-            // FormalParameter[Yield,Await]: (modified)
-            //      BindingElement[?Yield,Await]
-            //
-            // BindingElement [Yield,Await]: (modified)
-            //      SingleNameBinding[?Yield,?Await]
-            //      BindingPattern[?Yield,?Await]Initializer [In, ?Yield,?Await] opt
-            //
-            // SingleNameBinding [Yield,Await]:
-            //      BindingIdentifier[?Yield,?Await]Initializer [In, ?Yield,?Await] opt
-            if (!parseExpected(20 /* OpenParenToken */)) {
-                signature.parameters = createMissingList();
-                return false;
-            }
-            var savedYieldContext = inYieldContext();
-            var savedAwaitContext = inAwaitContext();
-            setYieldContext(!!(flags & 1 /* Yield */));
-            setAwaitContext(!!(flags & 2 /* Await */));
-            signature.parameters = flags & 32 /* JSDoc */ ?
-                parseDelimitedList(17 /* JSDocParameters */, parseJSDocParameter) :
-                parseDelimitedList(16 /* Parameters */, parseParameter);
-            setYieldContext(savedYieldContext);
-            setAwaitContext(savedAwaitContext);
-            return parseExpected(21 /* CloseParenToken */);
+        // @api
+        function updateReturnStatement(node, expression) {
+            return node.expression !== expression
+                ? update(createReturnStatement(expression), node)
+                : node;
         }
-        function parseTypeMemberSemicolon() {
-            // We allow type members to be separated by commas or (possibly ASI) semicolons.
-            // First check if it was a comma.  If so, we're done with the member.
-            if (parseOptional(27 /* CommaToken */)) {
-                return;
+        // @api
+        function createWithStatement(expression, statement) {
+            var node = createBaseNode(243 /* WithStatement */);
+            node.expression = expression;
+            node.statement = asEmbeddedStatement(statement);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.statement);
+            return node;
+        }
+        // @api
+        function updateWithStatement(node, expression, statement) {
+            return node.expression !== expression
+                || node.statement !== statement
+                ? update(createWithStatement(expression, statement), node)
+                : node;
+        }
+        // @api
+        function createSwitchStatement(expression, caseBlock) {
+            var node = createBaseNode(244 /* SwitchStatement */);
+            node.expression = parenthesizerRules().parenthesizeExpressionForDisallowedComma(expression);
+            node.caseBlock = caseBlock;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.caseBlock);
+            return node;
+        }
+        // @api
+        function updateSwitchStatement(node, expression, caseBlock) {
+            return node.expression !== expression
+                || node.caseBlock !== caseBlock
+                ? update(createSwitchStatement(expression, caseBlock), node)
+                : node;
+        }
+        // @api
+        function createLabeledStatement(label, statement) {
+            var node = createBaseNode(245 /* LabeledStatement */);
+            node.label = asName(label);
+            node.statement = asEmbeddedStatement(statement);
+            node.transformFlags |=
+                propagateChildFlags(node.label) |
+                    propagateChildFlags(node.statement);
+            return node;
+        }
+        // @api
+        function updateLabeledStatement(node, label, statement) {
+            return node.label !== label
+                || node.statement !== statement
+                ? update(createLabeledStatement(label, statement), node)
+                : node;
+        }
+        // @api
+        function createThrowStatement(expression) {
+            var node = createBaseNode(246 /* ThrowStatement */);
+            node.expression = expression;
+            node.transformFlags |= propagateChildFlags(node.expression);
+            return node;
+        }
+        // @api
+        function updateThrowStatement(node, expression) {
+            return node.expression !== expression
+                ? update(createThrowStatement(expression), node)
+                : node;
+        }
+        // @api
+        function createTryStatement(tryBlock, catchClause, finallyBlock) {
+            var node = createBaseNode(247 /* TryStatement */);
+            node.tryBlock = tryBlock;
+            node.catchClause = catchClause;
+            node.finallyBlock = finallyBlock;
+            node.transformFlags |=
+                propagateChildFlags(node.tryBlock) |
+                    propagateChildFlags(node.catchClause) |
+                    propagateChildFlags(node.finallyBlock);
+            return node;
+        }
+        // @api
+        function updateTryStatement(node, tryBlock, catchClause, finallyBlock) {
+            return node.tryBlock !== tryBlock
+                || node.catchClause !== catchClause
+                || node.finallyBlock !== finallyBlock
+                ? update(createTryStatement(tryBlock, catchClause, finallyBlock), node)
+                : node;
+        }
+        // @api
+        function createDebuggerStatement() {
+            return createBaseNode(248 /* DebuggerStatement */);
+        }
+        // @api
+        function createVariableDeclaration(name, exclamationToken, type, initializer) {
+            var node = createBaseVariableLikeDeclaration(249 /* VariableDeclaration */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, name, type, initializer && parenthesizerRules().parenthesizeExpressionForDisallowedComma(initializer));
+            node.exclamationToken = exclamationToken;
+            node.transformFlags |= propagateChildFlags(node.exclamationToken);
+            if (exclamationToken) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            // Didn't have a comma.  We must have a (possible ASI) semicolon.
-            parseSemicolon();
+            return node;
         }
-        function parseSignatureMember(kind) {
-            var node = createNodeWithJSDoc(kind);
-            if (kind === 166 /* ConstructSignature */) {
-                parseExpected(99 /* NewKeyword */);
+        // @api
+        function updateVariableDeclaration(node, name, exclamationToken, type, initializer) {
+            return node.name !== name
+                || node.type !== type
+                || node.exclamationToken !== exclamationToken
+                || node.initializer !== initializer
+                ? update(createVariableDeclaration(name, exclamationToken, type, initializer), node)
+                : node;
+        }
+        // @api
+        function createVariableDeclarationList(declarations, flags) {
+            if (flags === void 0) { flags = 0 /* None */; }
+            var node = createBaseNode(250 /* VariableDeclarationList */);
+            node.flags |= flags & 3 /* BlockScoped */;
+            node.declarations = createNodeArray(declarations);
+            node.transformFlags |=
+                propagateChildrenFlags(node.declarations) |
+                    1048576 /* ContainsHoistedDeclarationOrCompletion */;
+            if (flags & 3 /* BlockScoped */) {
+                node.transformFlags |=
+                    256 /* ContainsES2015 */ |
+                        65536 /* ContainsBlockScopedBinding */;
             }
-            fillSignature(58 /* ColonToken */, 4 /* Type */, node);
-            parseTypeMemberSemicolon();
-            return finishNode(node);
+            return node;
         }
-        function isIndexSignature() {
-            return token() === 22 /* OpenBracketToken */ && lookAhead(isUnambiguouslyIndexSignature);
+        // @api
+        function updateVariableDeclarationList(node, declarations) {
+            return node.declarations !== declarations
+                ? update(createVariableDeclarationList(declarations, node.flags), node)
+                : node;
         }
-        function isUnambiguouslyIndexSignature() {
-            // The only allowed sequence is:
-            //
-            //   [id:
-            //
-            // However, for error recovery, we also check the following cases:
-            //
-            //   [...
-            //   [id,
-            //   [id?,
-            //   [id?:
-            //   [id?]
-            //   [public id
-            //   [private id
-            //   [protected id
-            //   []
-            //
-            nextToken();
-            if (token() === 25 /* DotDotDotToken */ || token() === 23 /* CloseBracketToken */) {
-                return true;
+        // @api
+        function createFunctionDeclaration(decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
+            var node = createBaseFunctionLikeDeclaration(251 /* FunctionDeclaration */, decorators, modifiers, name, typeParameters, parameters, type, body);
+            node.asteriskToken = asteriskToken;
+            if (!node.body || ts.modifiersToFlags(node.modifiers) & 2 /* Ambient */) {
+                node.transformFlags = 1 /* ContainsTypeScript */;
             }
-            if (ts.isModifierKind(token())) {
-                nextToken();
-                if (isIdentifier()) {
-                    return true;
+            else {
+                node.transformFlags |=
+                    propagateChildFlags(node.asteriskToken) |
+                        1048576 /* ContainsHoistedDeclarationOrCompletion */;
+                if (ts.modifiersToFlags(node.modifiers) & 256 /* Async */) {
+                    if (node.asteriskToken) {
+                        node.transformFlags |= 32 /* ContainsES2018 */;
+                    }
+                    else {
+                        node.transformFlags |= 64 /* ContainsES2017 */;
+                    }
+                }
+                else if (node.asteriskToken) {
+                    node.transformFlags |= 512 /* ContainsGenerator */;
                 }
             }
-            else if (!isIdentifier()) {
-                return false;
+            return node;
+        }
+        // @api
+        function updateFunctionDeclaration(node, decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.asteriskToken !== asteriskToken
+                || node.name !== name
+                || node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                || node.body !== body
+                ? updateBaseFunctionLikeDeclaration(createFunctionDeclaration(decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body), node)
+                : node;
+        }
+        // @api
+        function createClassDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            var node = createBaseClassLikeDeclaration(252 /* ClassDeclaration */, decorators, modifiers, name, typeParameters, heritageClauses, members);
+            if (ts.modifiersToFlags(node.modifiers) & 2 /* Ambient */) {
+                node.transformFlags = 1 /* ContainsTypeScript */;
             }
             else {
-                // Skip the identifier
-                nextToken();
-            }
-            // A colon signifies a well formed indexer
-            // A comma should be a badly formed indexer because comma expressions are not allowed
-            // in computed properties.
-            if (token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */) {
-                return true;
-            }
-            // Question mark could be an indexer with an optional property,
-            // or it could be a conditional expression in a computed property.
-            if (token() !== 57 /* QuestionToken */) {
-                return false;
+                node.transformFlags |= 256 /* ContainsES2015 */;
+                if (node.transformFlags & 2048 /* ContainsTypeScriptClassSyntax */) {
+                    node.transformFlags |= 1 /* ContainsTypeScript */;
+                }
             }
-            // If any of the following tokens are after the question mark, it cannot
-            // be a conditional expression, so treat it as an indexer.
-            nextToken();
-            return token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */ || token() === 23 /* CloseBracketToken */;
+            return node;
         }
-        function parseIndexSignatureDeclaration(node) {
-            node.kind = 167 /* IndexSignature */;
-            node.parameters = parseBracketedList(16 /* Parameters */, parseParameter, 22 /* OpenBracketToken */, 23 /* CloseBracketToken */);
-            node.type = parseTypeAnnotation();
-            parseTypeMemberSemicolon();
-            return finishNode(node);
+        // @api
+        function updateClassDeclaration(node, decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.typeParameters !== typeParameters
+                || node.heritageClauses !== heritageClauses
+                || node.members !== members
+                ? update(createClassDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members), node)
+                : node;
+        }
+        // @api
+        function createInterfaceDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            var node = createBaseInterfaceOrClassLikeDeclaration(253 /* InterfaceDeclaration */, decorators, modifiers, name, typeParameters, heritageClauses);
+            node.members = createNodeArray(members);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function parsePropertyOrMethodSignature(node) {
-            node.name = parsePropertyName();
-            node.questionToken = parseOptionalToken(57 /* QuestionToken */);
-            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
-                node.kind = 160 /* MethodSignature */;
-                // Method signatures don't exist in expression contexts.  So they have neither
-                // [Yield] nor [Await]
-                fillSignature(58 /* ColonToken */, 4 /* Type */, node);
+        // @api
+        function updateInterfaceDeclaration(node, decorators, modifiers, name, typeParameters, heritageClauses, members) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.typeParameters !== typeParameters
+                || node.heritageClauses !== heritageClauses
+                || node.members !== members
+                ? update(createInterfaceDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members), node)
+                : node;
+        }
+        // @api
+        function createTypeAliasDeclaration(decorators, modifiers, name, typeParameters, type) {
+            var node = createBaseGenericNamedDeclaration(254 /* TypeAliasDeclaration */, decorators, modifiers, name, typeParameters);
+            node.type = type;
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
+        }
+        // @api
+        function updateTypeAliasDeclaration(node, decorators, modifiers, name, typeParameters, type) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.typeParameters !== typeParameters
+                || node.type !== type
+                ? update(createTypeAliasDeclaration(decorators, modifiers, name, typeParameters, type), node)
+                : node;
+        }
+        // @api
+        function createEnumDeclaration(decorators, modifiers, name, members) {
+            var node = createBaseNamedDeclaration(255 /* EnumDeclaration */, decorators, modifiers, name);
+            node.members = createNodeArray(members);
+            node.transformFlags |=
+                propagateChildrenFlags(node.members) |
+                    1 /* ContainsTypeScript */;
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // Enum declarations cannot contain `await`
+            return node;
+        }
+        // @api
+        function updateEnumDeclaration(node, decorators, modifiers, name, members) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.members !== members
+                ? update(createEnumDeclaration(decorators, modifiers, name, members), node)
+                : node;
+        }
+        // @api
+        function createModuleDeclaration(decorators, modifiers, name, body, flags) {
+            if (flags === void 0) { flags = 0 /* None */; }
+            var node = createBaseDeclaration(256 /* ModuleDeclaration */, decorators, modifiers);
+            node.flags |= flags & (16 /* Namespace */ | 4 /* NestedNamespace */ | 1024 /* GlobalAugmentation */);
+            node.name = name;
+            node.body = body;
+            if (ts.modifiersToFlags(node.modifiers) & 2 /* Ambient */) {
+                node.transformFlags = 1 /* ContainsTypeScript */;
             }
             else {
-                node.kind = 158 /* PropertySignature */;
-                node.type = parseTypeAnnotation();
-                if (token() === 62 /* EqualsToken */) {
-                    // Although type literal properties cannot not have initializers, we attempt
-                    // to parse an initializer so we can report in the checker that an interface
-                    // property or type literal property cannot have an initializer.
-                    node.initializer = parseInitializer();
-                }
+                node.transformFlags |=
+                    propagateChildFlags(node.name) |
+                        propagateChildFlags(node.body) |
+                        1 /* ContainsTypeScript */;
             }
-            parseTypeMemberSemicolon();
-            return finishNode(node);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // Module declarations cannot contain `await`.
+            return node;
         }
-        function isTypeMemberStart() {
-            // Return true if we have the start of a signature member
-            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
-                return true;
-            }
-            var idToken = false;
-            // Eat up all modifiers, but hold on to the last one in case it is actually an identifier
-            while (ts.isModifierKind(token())) {
-                idToken = true;
-                nextToken();
-            }
-            // Index signatures and computed property names are type members
-            if (token() === 22 /* OpenBracketToken */) {
-                return true;
-            }
-            // Try to get the first property-like token following all modifiers
-            if (isLiteralPropertyName()) {
-                idToken = true;
-                nextToken();
-            }
-            // If we were able to get any potential identifier, check that it is
-            // the start of a member declaration
-            if (idToken) {
-                return token() === 20 /* OpenParenToken */ ||
-                    token() === 29 /* LessThanToken */ ||
-                    token() === 57 /* QuestionToken */ ||
-                    token() === 58 /* ColonToken */ ||
-                    token() === 27 /* CommaToken */ ||
-                    canParseSemicolon();
-            }
-            return false;
+        // @api
+        function updateModuleDeclaration(node, decorators, modifiers, name, body) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.body !== body
+                ? update(createModuleDeclaration(decorators, modifiers, name, body, node.flags), node)
+                : node;
+        }
+        // @api
+        function createModuleBlock(statements) {
+            var node = createBaseNode(257 /* ModuleBlock */);
+            node.statements = createNodeArray(statements);
+            node.transformFlags |= propagateChildrenFlags(node.statements);
+            return node;
         }
-        function parseTypeMember() {
-            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
-                return parseSignatureMember(165 /* CallSignature */);
-            }
-            if (token() === 99 /* NewKeyword */ && lookAhead(nextTokenIsOpenParenOrLessThan)) {
-                return parseSignatureMember(166 /* ConstructSignature */);
-            }
-            var node = createNodeWithJSDoc(0 /* Unknown */);
-            node.modifiers = parseModifiers();
-            if (isIndexSignature()) {
-                return parseIndexSignatureDeclaration(node);
-            }
-            return parsePropertyOrMethodSignature(node);
+        // @api
+        function updateModuleBlock(node, statements) {
+            return node.statements !== statements
+                ? update(createModuleBlock(statements), node)
+                : node;
+        }
+        // @api
+        function createCaseBlock(clauses) {
+            var node = createBaseNode(258 /* CaseBlock */);
+            node.clauses = createNodeArray(clauses);
+            node.transformFlags |= propagateChildrenFlags(node.clauses);
+            return node;
         }
-        function nextTokenIsOpenParenOrLessThan() {
-            nextToken();
-            return token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */;
+        // @api
+        function updateCaseBlock(node, clauses) {
+            return node.clauses !== clauses
+                ? update(createCaseBlock(clauses), node)
+                : node;
         }
-        function nextTokenIsDot() {
-            return nextToken() === 24 /* DotToken */;
+        // @api
+        function createNamespaceExportDeclaration(name) {
+            var node = createBaseNamedDeclaration(259 /* NamespaceExportDeclaration */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, name);
+            node.transformFlags = 1 /* ContainsTypeScript */;
+            return node;
         }
-        function nextTokenIsOpenParenOrLessThanOrDot() {
-            switch (nextToken()) {
-                case 20 /* OpenParenToken */:
-                case 29 /* LessThanToken */:
-                case 24 /* DotToken */:
-                    return true;
-            }
-            return false;
+        // @api
+        function updateNamespaceExportDeclaration(node, name) {
+            return node.name !== name
+                ? update(createNamespaceExportDeclaration(name), node)
+                : node;
+        }
+        // @api
+        function createImportEqualsDeclaration(decorators, modifiers, name, moduleReference) {
+            var node = createBaseNamedDeclaration(260 /* ImportEqualsDeclaration */, decorators, modifiers, name);
+            node.moduleReference = moduleReference;
+            node.transformFlags |= propagateChildFlags(node.moduleReference);
+            if (!ts.isExternalModuleReference(node.moduleReference))
+                node.transformFlags |= 1 /* ContainsTypeScript */;
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // Import= declaration is always parsed in an Await context
+            return node;
         }
-        function parseTypeLiteral() {
-            var node = createNode(173 /* TypeLiteral */);
-            node.members = parseObjectTypeMembers();
-            return finishNode(node);
+        // @api
+        function updateImportEqualsDeclaration(node, decorators, modifiers, name, moduleReference) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.name !== name
+                || node.moduleReference !== moduleReference
+                ? update(createImportEqualsDeclaration(decorators, modifiers, name, moduleReference), node)
+                : node;
+        }
+        // @api
+        function createImportDeclaration(decorators, modifiers, importClause, moduleSpecifier) {
+            var node = createBaseDeclaration(261 /* ImportDeclaration */, decorators, modifiers);
+            node.importClause = importClause;
+            node.moduleSpecifier = moduleSpecifier;
+            node.transformFlags |=
+                propagateChildFlags(node.importClause) |
+                    propagateChildFlags(node.moduleSpecifier);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseObjectTypeMembers() {
-            var members;
-            if (parseExpected(18 /* OpenBraceToken */)) {
-                members = parseList(4 /* TypeMembers */, parseTypeMember);
-                parseExpected(19 /* CloseBraceToken */);
-            }
-            else {
-                members = createMissingList();
+        // @api
+        function updateImportDeclaration(node, decorators, modifiers, importClause, moduleSpecifier) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.importClause !== importClause
+                || node.moduleSpecifier !== moduleSpecifier
+                ? update(createImportDeclaration(decorators, modifiers, importClause, moduleSpecifier), node)
+                : node;
+        }
+        // @api
+        function createImportClause(isTypeOnly, name, namedBindings) {
+            var node = createBaseNode(262 /* ImportClause */);
+            node.isTypeOnly = isTypeOnly;
+            node.name = name;
+            node.namedBindings = namedBindings;
+            node.transformFlags |=
+                propagateChildFlags(node.name) |
+                    propagateChildFlags(node.namedBindings);
+            if (isTypeOnly) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            return members;
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function isStartOfMappedType() {
-            nextToken();
-            if (token() === 39 /* PlusToken */ || token() === 40 /* MinusToken */) {
-                return nextToken() === 138 /* ReadonlyKeyword */;
-            }
-            if (token() === 138 /* ReadonlyKeyword */) {
-                nextToken();
-            }
-            return token() === 22 /* OpenBracketToken */ && nextTokenIsIdentifier() && nextToken() === 97 /* InKeyword */;
+        // @api
+        function updateImportClause(node, isTypeOnly, name, namedBindings) {
+            return node.isTypeOnly !== isTypeOnly
+                || node.name !== name
+                || node.namedBindings !== namedBindings
+                ? update(createImportClause(isTypeOnly, name, namedBindings), node)
+                : node;
+        }
+        // @api
+        function createNamespaceImport(name) {
+            var node = createBaseNode(263 /* NamespaceImport */);
+            node.name = name;
+            node.transformFlags |= propagateChildFlags(node.name);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseMappedTypeParameter() {
-            var node = createNode(155 /* TypeParameter */);
-            node.name = parseIdentifier();
-            parseExpected(97 /* InKeyword */);
-            node.constraint = parseType();
-            return finishNode(node);
+        // @api
+        function updateNamespaceImport(node, name) {
+            return node.name !== name
+                ? update(createNamespaceImport(name), node)
+                : node;
+        }
+        // @api
+        function createNamespaceExport(name) {
+            var node = createBaseNode(269 /* NamespaceExport */);
+            node.name = name;
+            node.transformFlags |=
+                propagateChildFlags(node.name) |
+                    4 /* ContainsESNext */;
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseMappedType() {
-            var node = createNode(186 /* MappedType */);
-            parseExpected(18 /* OpenBraceToken */);
-            if (token() === 138 /* ReadonlyKeyword */ || token() === 39 /* PlusToken */ || token() === 40 /* MinusToken */) {
-                node.readonlyToken = parseTokenNode();
-                if (node.readonlyToken.kind !== 138 /* ReadonlyKeyword */) {
-                    parseExpectedToken(138 /* ReadonlyKeyword */);
-                }
-            }
-            parseExpected(22 /* OpenBracketToken */);
-            node.typeParameter = parseMappedTypeParameter();
-            parseExpected(23 /* CloseBracketToken */);
-            if (token() === 57 /* QuestionToken */ || token() === 39 /* PlusToken */ || token() === 40 /* MinusToken */) {
-                node.questionToken = parseTokenNode();
-                if (node.questionToken.kind !== 57 /* QuestionToken */) {
-                    parseExpectedToken(57 /* QuestionToken */);
-                }
-            }
-            node.type = parseTypeAnnotation();
-            parseSemicolon();
-            parseExpected(19 /* CloseBraceToken */);
-            return finishNode(node);
+        // @api
+        function updateNamespaceExport(node, name) {
+            return node.name !== name
+                ? update(createNamespaceExport(name), node)
+                : node;
+        }
+        // @api
+        function createNamedImports(elements) {
+            var node = createBaseNode(264 /* NamedImports */);
+            node.elements = createNodeArray(elements);
+            node.transformFlags |= propagateChildrenFlags(node.elements);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseTupleElementType() {
-            var pos = getNodePos();
-            if (parseOptional(25 /* DotDotDotToken */)) {
-                var node = createNode(177 /* RestType */, pos);
-                node.type = parseType();
-                return finishNode(node);
-            }
-            var type = parseType();
-            if (!(contextFlags & 4194304 /* JSDoc */) && type.kind === 297 /* JSDocNullableType */ && type.pos === type.type.pos) {
-                type.kind = 176 /* OptionalType */;
-            }
-            return type;
+        // @api
+        function updateNamedImports(node, elements) {
+            return node.elements !== elements
+                ? update(createNamedImports(elements), node)
+                : node;
+        }
+        // @api
+        function createImportSpecifier(propertyName, name) {
+            var node = createBaseNode(265 /* ImportSpecifier */);
+            node.propertyName = propertyName;
+            node.name = name;
+            node.transformFlags |=
+                propagateChildFlags(node.propertyName) |
+                    propagateChildFlags(node.name);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseTupleType() {
-            var node = createNode(175 /* TupleType */);
-            node.elementTypes = parseBracketedList(21 /* TupleElementTypes */, parseTupleElementType, 22 /* OpenBracketToken */, 23 /* CloseBracketToken */);
-            return finishNode(node);
+        // @api
+        function updateImportSpecifier(node, propertyName, name) {
+            return node.propertyName !== propertyName
+                || node.name !== name
+                ? update(createImportSpecifier(propertyName, name), node)
+                : node;
+        }
+        // @api
+        function createExportAssignment(decorators, modifiers, isExportEquals, expression) {
+            var node = createBaseDeclaration(266 /* ExportAssignment */, decorators, modifiers);
+            node.isExportEquals = isExportEquals;
+            node.expression = isExportEquals
+                ? parenthesizerRules().parenthesizeRightSideOfBinary(62 /* EqualsToken */, /*leftSide*/ undefined, expression)
+                : parenthesizerRules().parenthesizeExpressionOfExportDefault(expression);
+            node.transformFlags |= propagateChildFlags(node.expression);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseParenthesizedType() {
-            var node = createNode(182 /* ParenthesizedType */);
-            parseExpected(20 /* OpenParenToken */);
-            node.type = parseType();
-            parseExpected(21 /* CloseParenToken */);
-            return finishNode(node);
+        // @api
+        function updateExportAssignment(node, decorators, modifiers, expression) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.expression !== expression
+                ? update(createExportAssignment(decorators, modifiers, node.isExportEquals, expression), node)
+                : node;
+        }
+        // @api
+        function createExportDeclaration(decorators, modifiers, isTypeOnly, exportClause, moduleSpecifier) {
+            var node = createBaseDeclaration(267 /* ExportDeclaration */, decorators, modifiers);
+            node.isTypeOnly = isTypeOnly;
+            node.exportClause = exportClause;
+            node.moduleSpecifier = moduleSpecifier;
+            node.transformFlags |=
+                propagateChildFlags(node.exportClause) |
+                    propagateChildFlags(node.moduleSpecifier);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseFunctionOrConstructorType() {
-            var pos = getNodePos();
-            var kind = parseOptional(99 /* NewKeyword */) ? 171 /* ConstructorType */ : 170 /* FunctionType */;
-            var node = createNodeWithJSDoc(kind, pos);
-            fillSignature(38 /* EqualsGreaterThanToken */, 4 /* Type */, node);
-            return finishNode(node);
+        // @api
+        function updateExportDeclaration(node, decorators, modifiers, isTypeOnly, exportClause, moduleSpecifier) {
+            return node.decorators !== decorators
+                || node.modifiers !== modifiers
+                || node.isTypeOnly !== isTypeOnly
+                || node.exportClause !== exportClause
+                || node.moduleSpecifier !== moduleSpecifier
+                ? update(createExportDeclaration(decorators, modifiers, isTypeOnly, exportClause, moduleSpecifier), node)
+                : node;
+        }
+        // @api
+        function createNamedExports(elements) {
+            var node = createBaseNode(268 /* NamedExports */);
+            node.elements = createNodeArray(elements);
+            node.transformFlags |= propagateChildrenFlags(node.elements);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseKeywordAndNoDot() {
-            var node = parseTokenNode();
-            return token() === 24 /* DotToken */ ? undefined : node;
+        // @api
+        function updateNamedExports(node, elements) {
+            return node.elements !== elements
+                ? update(createNamedExports(elements), node)
+                : node;
+        }
+        // @api
+        function createExportSpecifier(propertyName, name) {
+            var node = createBaseNode(270 /* ExportSpecifier */);
+            node.propertyName = asName(propertyName);
+            node.name = asName(name);
+            node.transformFlags |=
+                propagateChildFlags(node.propertyName) |
+                    propagateChildFlags(node.name);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
         }
-        function parseLiteralTypeNode(negative) {
-            var node = createNode(187 /* LiteralType */);
-            var unaryMinusExpression;
-            if (negative) {
-                unaryMinusExpression = createNode(207 /* PrefixUnaryExpression */);
-                unaryMinusExpression.operator = 40 /* MinusToken */;
-                nextToken();
-            }
-            var expression = token() === 106 /* TrueKeyword */ || token() === 91 /* FalseKeyword */
-                ? parseTokenNode()
-                : parseLiteralLikeNode(token());
-            if (negative) {
-                unaryMinusExpression.operand = expression;
-                finishNode(unaryMinusExpression);
-                expression = unaryMinusExpression;
-            }
-            node.literal = expression;
-            return finishNode(node);
+        // @api
+        function updateExportSpecifier(node, propertyName, name) {
+            return node.propertyName !== propertyName
+                || node.name !== name
+                ? update(createExportSpecifier(propertyName, name), node)
+                : node;
         }
-        function isStartOfTypeOfImportType() {
-            nextToken();
-            return token() === 96 /* ImportKeyword */;
+        // @api
+        function createMissingDeclaration() {
+            var node = createBaseDeclaration(271 /* MissingDeclaration */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined);
+            return node;
         }
-        function parseImportType() {
-            sourceFile.flags |= 1048576 /* PossiblyContainsDynamicImport */;
-            var node = createNode(188 /* ImportType */);
-            if (parseOptional(108 /* TypeOfKeyword */)) {
-                node.isTypeOf = true;
-            }
-            parseExpected(96 /* ImportKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.argument = parseType();
-            parseExpected(21 /* CloseParenToken */);
-            if (parseOptional(24 /* DotToken */)) {
-                node.qualifier = parseEntityName(/*allowReservedWords*/ true, ts.Diagnostics.Type_expected);
+        //
+        // Module references
+        //
+        // @api
+        function createExternalModuleReference(expression) {
+            var node = createBaseNode(272 /* ExternalModuleReference */);
+            node.expression = expression;
+            node.transformFlags |= propagateChildFlags(node.expression);
+            node.transformFlags &= ~8388608 /* ContainsPossibleTopLevelAwait */; // always parsed in an Await context
+            return node;
+        }
+        // @api
+        function updateExternalModuleReference(node, expression) {
+            return node.expression !== expression
+                ? update(createExternalModuleReference(expression), node)
+                : node;
+        }
+        //
+        // JSDoc
+        //
+        // @api
+        // createJSDocAllType
+        // createJSDocUnknownType
+        function createJSDocPrimaryTypeWorker(kind) {
+            return createBaseNode(kind);
+        }
+        // @api
+        // createJSDocNonNullableType
+        // createJSDocNullableType
+        // createJSDocOptionalType
+        // createJSDocVariadicType
+        // createJSDocNamepathType
+        function createJSDocUnaryTypeWorker(kind, type) {
+            var node = createBaseNode(kind);
+            node.type = type;
+            return node;
+        }
+        // @api
+        // updateJSDocNonNullableType
+        // updateJSDocNullableType
+        // updateJSDocOptionalType
+        // updateJSDocVariadicType
+        // updateJSDocNamepathType
+        function updateJSDocUnaryTypeWorker(kind, node, type) {
+            return node.type !== type
+                ? update(createJSDocUnaryTypeWorker(kind, type), node)
+                : node;
+        }
+        // @api
+        function createJSDocFunctionType(parameters, type) {
+            var node = createBaseSignatureDeclaration(308 /* JSDocFunctionType */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*name*/ undefined, 
+            /*typeParameters*/ undefined, parameters, type);
+            return node;
+        }
+        // @api
+        function updateJSDocFunctionType(node, parameters, type) {
+            return node.parameters !== parameters
+                || node.type !== type
+                ? update(createJSDocFunctionType(parameters, type), node)
+                : node;
+        }
+        // @api
+        function createJSDocTypeLiteral(propertyTags, isArrayType) {
+            if (isArrayType === void 0) { isArrayType = false; }
+            var node = createBaseNode(312 /* JSDocTypeLiteral */);
+            node.jsDocPropertyTags = asNodeArray(propertyTags);
+            node.isArrayType = isArrayType;
+            return node;
+        }
+        // @api
+        function updateJSDocTypeLiteral(node, propertyTags, isArrayType) {
+            return node.jsDocPropertyTags !== propertyTags
+                || node.isArrayType !== isArrayType
+                ? update(createJSDocTypeLiteral(propertyTags, isArrayType), node)
+                : node;
+        }
+        // @api
+        function createJSDocTypeExpression(type) {
+            var node = createBaseNode(301 /* JSDocTypeExpression */);
+            node.type = type;
+            return node;
+        }
+        // @api
+        function updateJSDocTypeExpression(node, type) {
+            return node.type !== type
+                ? update(createJSDocTypeExpression(type), node)
+                : node;
+        }
+        // @api
+        function createJSDocSignature(typeParameters, parameters, type) {
+            var node = createBaseNode(313 /* JSDocSignature */);
+            node.typeParameters = asNodeArray(typeParameters);
+            node.parameters = createNodeArray(parameters);
+            node.type = type;
+            return node;
+        }
+        // @api
+        function updateJSDocSignature(node, typeParameters, parameters, type) {
+            return node.typeParameters !== typeParameters
+                || node.parameters !== parameters
+                || node.type !== type
+                ? update(createJSDocSignature(typeParameters, parameters, type), node)
+                : node;
+        }
+        function getDefaultTagName(node) {
+            var defaultTagName = getDefaultTagNameForKind(node.kind);
+            return node.tagName.escapedText === ts.escapeLeadingUnderscores(defaultTagName)
+                ? node.tagName
+                : createIdentifier(defaultTagName);
+        }
+        // @api
+        function createBaseJSDocTag(kind, tagName, comment) {
+            var node = createBaseNode(kind);
+            node.tagName = tagName;
+            node.comment = comment;
+            return node;
+        }
+        // @api
+        function createJSDocTemplateTag(tagName, constraint, typeParameters, comment) {
+            var node = createBaseJSDocTag(330 /* JSDocTemplateTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("template"), comment);
+            node.constraint = constraint;
+            node.typeParameters = createNodeArray(typeParameters);
+            return node;
+        }
+        // @api
+        function updateJSDocTemplateTag(node, tagName, constraint, typeParameters, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.constraint !== constraint
+                || node.typeParameters !== typeParameters
+                || node.comment !== comment
+                ? update(createJSDocTemplateTag(tagName, constraint, typeParameters, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocTypedefTag(tagName, typeExpression, fullName, comment) {
+            var node = createBaseJSDocTag(331 /* JSDocTypedefTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("typedef"), comment);
+            node.typeExpression = typeExpression;
+            node.fullName = fullName;
+            node.name = ts.getJSDocTypeAliasName(fullName);
+            return node;
+        }
+        // @api
+        function updateJSDocTypedefTag(node, tagName, typeExpression, fullName, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.typeExpression !== typeExpression
+                || node.fullName !== fullName
+                || node.comment !== comment
+                ? update(createJSDocTypedefTag(tagName, typeExpression, fullName, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocParameterTag(tagName, name, isBracketed, typeExpression, isNameFirst, comment) {
+            var node = createBaseJSDocTag(326 /* JSDocParameterTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("param"), comment);
+            node.typeExpression = typeExpression;
+            node.name = name;
+            node.isNameFirst = !!isNameFirst;
+            node.isBracketed = isBracketed;
+            return node;
+        }
+        // @api
+        function updateJSDocParameterTag(node, tagName, name, isBracketed, typeExpression, isNameFirst, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.name !== name
+                || node.isBracketed !== isBracketed
+                || node.typeExpression !== typeExpression
+                || node.isNameFirst !== isNameFirst
+                || node.comment !== comment
+                ? update(createJSDocParameterTag(tagName, name, isBracketed, typeExpression, isNameFirst, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocPropertyTag(tagName, name, isBracketed, typeExpression, isNameFirst, comment) {
+            var node = createBaseJSDocTag(333 /* JSDocPropertyTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("prop"), comment);
+            node.typeExpression = typeExpression;
+            node.name = name;
+            node.isNameFirst = !!isNameFirst;
+            node.isBracketed = isBracketed;
+            return node;
+        }
+        // @api
+        function updateJSDocPropertyTag(node, tagName, name, isBracketed, typeExpression, isNameFirst, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.name !== name
+                || node.isBracketed !== isBracketed
+                || node.typeExpression !== typeExpression
+                || node.isNameFirst !== isNameFirst
+                || node.comment !== comment
+                ? update(createJSDocPropertyTag(tagName, name, isBracketed, typeExpression, isNameFirst, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocCallbackTag(tagName, typeExpression, fullName, comment) {
+            var node = createBaseJSDocTag(324 /* JSDocCallbackTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("callback"), comment);
+            node.typeExpression = typeExpression;
+            node.fullName = fullName;
+            node.name = ts.getJSDocTypeAliasName(fullName);
+            return node;
+        }
+        // @api
+        function updateJSDocCallbackTag(node, tagName, typeExpression, fullName, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.typeExpression !== typeExpression
+                || node.fullName !== fullName
+                || node.comment !== comment
+                ? update(createJSDocCallbackTag(tagName, typeExpression, fullName, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocAugmentsTag(tagName, className, comment) {
+            var node = createBaseJSDocTag(315 /* JSDocAugmentsTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("augments"), comment);
+            node.class = className;
+            return node;
+        }
+        // @api
+        function updateJSDocAugmentsTag(node, tagName, className, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.class !== className
+                || node.comment !== comment
+                ? update(createJSDocAugmentsTag(tagName, className, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocImplementsTag(tagName, className, comment) {
+            var node = createBaseJSDocTag(316 /* JSDocImplementsTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("implements"), comment);
+            node.class = className;
+            return node;
+        }
+        // @api
+        function createJSDocSeeTag(tagName, name, comment) {
+            var node = createBaseJSDocTag(332 /* JSDocSeeTag */, tagName !== null && tagName !== void 0 ? tagName : createIdentifier("see"), comment);
+            node.name = name;
+            return node;
+        }
+        // @api
+        function updateJSDocSeeTag(node, tagName, name, comment) {
+            return node.tagName !== tagName
+                || node.name !== name
+                || node.comment !== comment
+                ? update(createJSDocSeeTag(tagName, name, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocNameReference(name) {
+            var node = createBaseNode(302 /* JSDocNameReference */);
+            node.name = name;
+            return node;
+        }
+        // @api
+        function updateJSDocNameReference(node, name) {
+            return node.name !== name
+                ? update(createJSDocNameReference(name), node)
+                : node;
+        }
+        // @api
+        function updateJSDocImplementsTag(node, tagName, className, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.class !== className
+                || node.comment !== comment
+                ? update(createJSDocImplementsTag(tagName, className, comment), node)
+                : node;
+        }
+        // @api
+        // createJSDocAuthorTag
+        // createJSDocClassTag
+        // createJSDocPublicTag
+        // createJSDocPrivateTag
+        // createJSDocProtectedTag
+        // createJSDocReadonlyTag
+        // createJSDocDeprecatedTag
+        function createJSDocSimpleTagWorker(kind, tagName, comment) {
+            var node = createBaseJSDocTag(kind, tagName !== null && tagName !== void 0 ? tagName : createIdentifier(getDefaultTagNameForKind(kind)), comment);
+            return node;
+        }
+        // @api
+        // updateJSDocAuthorTag
+        // updateJSDocClassTag
+        // updateJSDocPublicTag
+        // updateJSDocPrivateTag
+        // updateJSDocProtectedTag
+        // updateJSDocReadonlyTag
+        // updateJSDocDeprecatedTag
+        function updateJSDocSimpleTagWorker(kind, node, tagName, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.comment !== comment
+                ? update(createJSDocSimpleTagWorker(kind, tagName, comment), node) :
+                node;
+        }
+        // @api
+        // createJSDocTypeTag
+        // createJSDocReturnTag
+        // createJSDocThisTag
+        // createJSDocEnumTag
+        function createJSDocTypeLikeTagWorker(kind, tagName, typeExpression, comment) {
+            var node = createBaseJSDocTag(kind, tagName !== null && tagName !== void 0 ? tagName : createIdentifier(getDefaultTagNameForKind(kind)), comment);
+            node.typeExpression = typeExpression;
+            return node;
+        }
+        // @api
+        // updateJSDocTypeTag
+        // updateJSDocReturnTag
+        // updateJSDocThisTag
+        // updateJSDocEnumTag
+        function updateJSDocTypeLikeTagWorker(kind, node, tagName, typeExpression, comment) {
+            if (tagName === void 0) { tagName = getDefaultTagName(node); }
+            return node.tagName !== tagName
+                || node.typeExpression !== typeExpression
+                || node.comment !== comment
+                ? update(createJSDocTypeLikeTagWorker(kind, tagName, typeExpression, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocUnknownTag(tagName, comment) {
+            var node = createBaseJSDocTag(314 /* JSDocTag */, tagName, comment);
+            return node;
+        }
+        // @api
+        function updateJSDocUnknownTag(node, tagName, comment) {
+            return node.tagName !== tagName
+                || node.comment !== comment
+                ? update(createJSDocUnknownTag(tagName, comment), node)
+                : node;
+        }
+        // @api
+        function createJSDocComment(comment, tags) {
+            var node = createBaseNode(311 /* JSDocComment */);
+            node.comment = comment;
+            node.tags = asNodeArray(tags);
+            return node;
+        }
+        // @api
+        function updateJSDocComment(node, comment, tags) {
+            return node.comment !== comment
+                || node.tags !== tags
+                ? update(createJSDocComment(comment, tags), node)
+                : node;
+        }
+        //
+        // JSX
+        //
+        // @api
+        function createJsxElement(openingElement, children, closingElement) {
+            var node = createBaseNode(273 /* JsxElement */);
+            node.openingElement = openingElement;
+            node.children = createNodeArray(children);
+            node.closingElement = closingElement;
+            node.transformFlags |=
+                propagateChildFlags(node.openingElement) |
+                    propagateChildrenFlags(node.children) |
+                    propagateChildFlags(node.closingElement) |
+                    2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function updateJsxElement(node, openingElement, children, closingElement) {
+            return node.openingElement !== openingElement
+                || node.children !== children
+                || node.closingElement !== closingElement
+                ? update(createJsxElement(openingElement, children, closingElement), node)
+                : node;
+        }
+        // @api
+        function createJsxSelfClosingElement(tagName, typeArguments, attributes) {
+            var node = createBaseNode(274 /* JsxSelfClosingElement */);
+            node.tagName = tagName;
+            node.typeArguments = asNodeArray(typeArguments);
+            node.attributes = attributes;
+            node.transformFlags |=
+                propagateChildFlags(node.tagName) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    propagateChildFlags(node.attributes) |
+                    2 /* ContainsJsx */;
+            if (node.typeArguments) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            if (!scanner.hasPrecedingLineBreak() && reScanLessThanToken() === 29 /* LessThanToken */) {
-                node.typeArguments = parseBracketedList(20 /* TypeArguments */, parseType, 29 /* LessThanToken */, 31 /* GreaterThanToken */);
+            return node;
+        }
+        // @api
+        function updateJsxSelfClosingElement(node, tagName, typeArguments, attributes) {
+            return node.tagName !== tagName
+                || node.typeArguments !== typeArguments
+                || node.attributes !== attributes
+                ? update(createJsxSelfClosingElement(tagName, typeArguments, attributes), node)
+                : node;
+        }
+        // @api
+        function createJsxOpeningElement(tagName, typeArguments, attributes) {
+            var node = createBaseNode(275 /* JsxOpeningElement */);
+            node.tagName = tagName;
+            node.typeArguments = asNodeArray(typeArguments);
+            node.attributes = attributes;
+            node.transformFlags |=
+                propagateChildFlags(node.tagName) |
+                    propagateChildrenFlags(node.typeArguments) |
+                    propagateChildFlags(node.attributes) |
+                    2 /* ContainsJsx */;
+            if (typeArguments) {
+                node.transformFlags |= 1 /* ContainsTypeScript */;
             }
-            return finishNode(node);
+            return node;
         }
-        function nextTokenIsNumericOrBigIntLiteral() {
-            nextToken();
-            return token() === 8 /* NumericLiteral */ || token() === 9 /* BigIntLiteral */;
+        // @api
+        function updateJsxOpeningElement(node, tagName, typeArguments, attributes) {
+            return node.tagName !== tagName
+                || node.typeArguments !== typeArguments
+                || node.attributes !== attributes
+                ? update(createJsxOpeningElement(tagName, typeArguments, attributes), node)
+                : node;
+        }
+        // @api
+        function createJsxClosingElement(tagName) {
+            var node = createBaseNode(276 /* JsxClosingElement */);
+            node.tagName = tagName;
+            node.transformFlags |=
+                propagateChildFlags(node.tagName) |
+                    2 /* ContainsJsx */;
+            return node;
         }
-        function parseNonArrayType() {
-            switch (token()) {
-                case 125 /* AnyKeyword */:
-                case 148 /* UnknownKeyword */:
-                case 143 /* StringKeyword */:
-                case 140 /* NumberKeyword */:
-                case 151 /* BigIntKeyword */:
-                case 144 /* SymbolKeyword */:
-                case 128 /* BooleanKeyword */:
-                case 146 /* UndefinedKeyword */:
-                case 137 /* NeverKeyword */:
-                case 141 /* ObjectKeyword */:
-                    // If these are followed by a dot, then parse these out as a dotted type reference instead.
-                    return tryParse(parseKeywordAndNoDot) || parseTypeReference();
-                case 41 /* AsteriskToken */:
-                    return parseJSDocAllType(/*postfixEquals*/ false);
-                case 65 /* AsteriskEqualsToken */:
-                    return parseJSDocAllType(/*postfixEquals*/ true);
-                case 60 /* QuestionQuestionToken */:
-                    // If there is '??', consider that is prefix '?' in JSDoc type.
-                    scanner.reScanQuestionToken();
-                // falls through
-                case 57 /* QuestionToken */:
-                    return parseJSDocUnknownOrNullableType();
-                case 94 /* FunctionKeyword */:
-                    return parseJSDocFunctionType();
-                case 53 /* ExclamationToken */:
-                    return parseJSDocNonNullableType();
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 10 /* StringLiteral */:
-                case 8 /* NumericLiteral */:
-                case 9 /* BigIntLiteral */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
-                    return parseLiteralTypeNode();
-                case 40 /* MinusToken */:
-                    return lookAhead(nextTokenIsNumericOrBigIntLiteral) ? parseLiteralTypeNode(/*negative*/ true) : parseTypeReference();
-                case 110 /* VoidKeyword */:
-                case 100 /* NullKeyword */:
-                    return parseTokenNode();
-                case 104 /* ThisKeyword */: {
-                    var thisKeyword = parseThisTypeNode();
-                    if (token() === 133 /* IsKeyword */ && !scanner.hasPrecedingLineBreak()) {
-                        return parseThisTypePredicate(thisKeyword);
-                    }
-                    else {
-                        return thisKeyword;
-                    }
-                }
-                case 108 /* TypeOfKeyword */:
-                    return lookAhead(isStartOfTypeOfImportType) ? parseImportType() : parseTypeQuery();
-                case 18 /* OpenBraceToken */:
-                    return lookAhead(isStartOfMappedType) ? parseMappedType() : parseTypeLiteral();
-                case 22 /* OpenBracketToken */:
-                    return parseTupleType();
-                case 20 /* OpenParenToken */:
-                    return parseParenthesizedType();
-                case 96 /* ImportKeyword */:
-                    return parseImportType();
-                case 124 /* AssertsKeyword */:
-                    return lookAhead(nextTokenIsIdentifierOrKeywordOnSameLine) ? parseAssertsTypePredicate() : parseTypeReference();
-                default:
-                    return parseTypeReference();
-            }
+        // @api
+        function updateJsxClosingElement(node, tagName) {
+            return node.tagName !== tagName
+                ? update(createJsxClosingElement(tagName), node)
+                : node;
+        }
+        // @api
+        function createJsxFragment(openingFragment, children, closingFragment) {
+            var node = createBaseNode(277 /* JsxFragment */);
+            node.openingFragment = openingFragment;
+            node.children = createNodeArray(children);
+            node.closingFragment = closingFragment;
+            node.transformFlags |=
+                propagateChildFlags(node.openingFragment) |
+                    propagateChildrenFlags(node.children) |
+                    propagateChildFlags(node.closingFragment) |
+                    2 /* ContainsJsx */;
+            return node;
         }
-        function isStartOfType(inStartOfParameter) {
-            switch (token()) {
-                case 125 /* AnyKeyword */:
-                case 148 /* UnknownKeyword */:
-                case 143 /* StringKeyword */:
-                case 140 /* NumberKeyword */:
-                case 151 /* BigIntKeyword */:
-                case 128 /* BooleanKeyword */:
-                case 138 /* ReadonlyKeyword */:
-                case 144 /* SymbolKeyword */:
-                case 147 /* UniqueKeyword */:
-                case 110 /* VoidKeyword */:
-                case 146 /* UndefinedKeyword */:
-                case 100 /* NullKeyword */:
-                case 104 /* ThisKeyword */:
-                case 108 /* TypeOfKeyword */:
-                case 137 /* NeverKeyword */:
-                case 18 /* OpenBraceToken */:
-                case 22 /* OpenBracketToken */:
-                case 29 /* LessThanToken */:
-                case 51 /* BarToken */:
-                case 50 /* AmpersandToken */:
-                case 99 /* NewKeyword */:
-                case 10 /* StringLiteral */:
-                case 8 /* NumericLiteral */:
-                case 9 /* BigIntLiteral */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
-                case 141 /* ObjectKeyword */:
-                case 41 /* AsteriskToken */:
-                case 57 /* QuestionToken */:
-                case 53 /* ExclamationToken */:
-                case 25 /* DotDotDotToken */:
-                case 132 /* InferKeyword */:
-                case 96 /* ImportKeyword */:
-                case 124 /* AssertsKeyword */:
-                    return true;
-                case 94 /* FunctionKeyword */:
-                    return !inStartOfParameter;
-                case 40 /* MinusToken */:
-                    return !inStartOfParameter && lookAhead(nextTokenIsNumericOrBigIntLiteral);
-                case 20 /* OpenParenToken */:
-                    // Only consider '(' the start of a type if followed by ')', '...', an identifier, a modifier,
-                    // or something that starts a type. We don't want to consider things like '(1)' a type.
-                    return !inStartOfParameter && lookAhead(isStartOfParenthesizedOrFunctionType);
+        // @api
+        function updateJsxFragment(node, openingFragment, children, closingFragment) {
+            return node.openingFragment !== openingFragment
+                || node.children !== children
+                || node.closingFragment !== closingFragment
+                ? update(createJsxFragment(openingFragment, children, closingFragment), node)
+                : node;
+        }
+        // @api
+        function createJsxText(text, containsOnlyTriviaWhiteSpaces) {
+            var node = createBaseNode(11 /* JsxText */);
+            node.text = text;
+            node.containsOnlyTriviaWhiteSpaces = !!containsOnlyTriviaWhiteSpaces;
+            node.transformFlags |= 2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function updateJsxText(node, text, containsOnlyTriviaWhiteSpaces) {
+            return node.text !== text
+                || node.containsOnlyTriviaWhiteSpaces !== containsOnlyTriviaWhiteSpaces
+                ? update(createJsxText(text, containsOnlyTriviaWhiteSpaces), node)
+                : node;
+        }
+        // @api
+        function createJsxOpeningFragment() {
+            var node = createBaseNode(278 /* JsxOpeningFragment */);
+            node.transformFlags |= 2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function createJsxJsxClosingFragment() {
+            var node = createBaseNode(279 /* JsxClosingFragment */);
+            node.transformFlags |= 2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function createJsxAttribute(name, initializer) {
+            var node = createBaseNode(280 /* JsxAttribute */);
+            node.name = name;
+            node.initializer = initializer;
+            node.transformFlags |=
+                propagateChildFlags(node.name) |
+                    propagateChildFlags(node.initializer) |
+                    2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function updateJsxAttribute(node, name, initializer) {
+            return node.name !== name
+                || node.initializer !== initializer
+                ? update(createJsxAttribute(name, initializer), node)
+                : node;
+        }
+        // @api
+        function createJsxAttributes(properties) {
+            var node = createBaseNode(281 /* JsxAttributes */);
+            node.properties = createNodeArray(properties);
+            node.transformFlags |=
+                propagateChildrenFlags(node.properties) |
+                    2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function updateJsxAttributes(node, properties) {
+            return node.properties !== properties
+                ? update(createJsxAttributes(properties), node)
+                : node;
+        }
+        // @api
+        function createJsxSpreadAttribute(expression) {
+            var node = createBaseNode(282 /* JsxSpreadAttribute */);
+            node.expression = expression;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function updateJsxSpreadAttribute(node, expression) {
+            return node.expression !== expression
+                ? update(createJsxSpreadAttribute(expression), node)
+                : node;
+        }
+        // @api
+        function createJsxExpression(dotDotDotToken, expression) {
+            var node = createBaseNode(283 /* JsxExpression */);
+            node.dotDotDotToken = dotDotDotToken;
+            node.expression = expression;
+            node.transformFlags |=
+                propagateChildFlags(node.dotDotDotToken) |
+                    propagateChildFlags(node.expression) |
+                    2 /* ContainsJsx */;
+            return node;
+        }
+        // @api
+        function updateJsxExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createJsxExpression(node.dotDotDotToken, expression), node)
+                : node;
+        }
+        //
+        // Clauses
+        //
+        // @api
+        function createCaseClause(expression, statements) {
+            var node = createBaseNode(284 /* CaseClause */);
+            node.expression = parenthesizerRules().parenthesizeExpressionForDisallowedComma(expression);
+            node.statements = createNodeArray(statements);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildrenFlags(node.statements);
+            return node;
+        }
+        // @api
+        function updateCaseClause(node, expression, statements) {
+            return node.expression !== expression
+                || node.statements !== statements
+                ? update(createCaseClause(expression, statements), node)
+                : node;
+        }
+        // @api
+        function createDefaultClause(statements) {
+            var node = createBaseNode(285 /* DefaultClause */);
+            node.statements = createNodeArray(statements);
+            node.transformFlags = propagateChildrenFlags(node.statements);
+            return node;
+        }
+        // @api
+        function updateDefaultClause(node, statements) {
+            return node.statements !== statements
+                ? update(createDefaultClause(statements), node)
+                : node;
+        }
+        // @api
+        function createHeritageClause(token, types) {
+            var node = createBaseNode(286 /* HeritageClause */);
+            node.token = token;
+            node.types = createNodeArray(types);
+            node.transformFlags |= propagateChildrenFlags(node.types);
+            switch (token) {
+                case 93 /* ExtendsKeyword */:
+                    node.transformFlags |= 256 /* ContainsES2015 */;
+                    break;
+                case 116 /* ImplementsKeyword */:
+                    node.transformFlags |= 1 /* ContainsTypeScript */;
+                    break;
                 default:
-                    return isIdentifier();
+                    return ts.Debug.assertNever(token);
             }
+            return node;
         }
-        function isStartOfParenthesizedOrFunctionType() {
-            nextToken();
-            return token() === 21 /* CloseParenToken */ || isStartOfParameter(/*isJSDocParameter*/ false) || isStartOfType();
+        // @api
+        function updateHeritageClause(node, types) {
+            return node.types !== types
+                ? update(createHeritageClause(node.token, types), node)
+                : node;
+        }
+        // @api
+        function createCatchClause(variableDeclaration, block) {
+            var node = createBaseNode(287 /* CatchClause */);
+            variableDeclaration = !ts.isString(variableDeclaration) ? variableDeclaration : createVariableDeclaration(variableDeclaration, 
+            /*exclamationToken*/ undefined, 
+            /*type*/ undefined, 
+            /*initializer*/ undefined);
+            node.variableDeclaration = variableDeclaration;
+            node.block = block;
+            node.transformFlags |=
+                propagateChildFlags(node.variableDeclaration) |
+                    propagateChildFlags(node.block);
+            if (!variableDeclaration)
+                node.transformFlags |= 16 /* ContainsES2019 */;
+            return node;
         }
-        function parsePostfixTypeOrHigher() {
-            var type = parseNonArrayType();
-            while (!scanner.hasPrecedingLineBreak()) {
-                switch (token()) {
-                    case 53 /* ExclamationToken */:
-                        type = createPostfixType(298 /* JSDocNonNullableType */, type);
-                        break;
-                    case 57 /* QuestionToken */:
-                        // If not in JSDoc and next token is start of a type we have a conditional type
-                        if (!(contextFlags & 4194304 /* JSDoc */) && lookAhead(nextTokenIsStartOfType)) {
-                            return type;
-                        }
-                        type = createPostfixType(297 /* JSDocNullableType */, type);
-                        break;
-                    case 22 /* OpenBracketToken */:
-                        parseExpected(22 /* OpenBracketToken */);
-                        if (isStartOfType()) {
-                            var node = createNode(185 /* IndexedAccessType */, type.pos);
-                            node.objectType = type;
-                            node.indexType = parseType();
-                            parseExpected(23 /* CloseBracketToken */);
-                            type = finishNode(node);
-                        }
-                        else {
-                            var node = createNode(174 /* ArrayType */, type.pos);
-                            node.elementType = type;
-                            parseExpected(23 /* CloseBracketToken */);
-                            type = finishNode(node);
-                        }
-                        break;
-                    default:
-                        return type;
-                }
-            }
-            return type;
+        // @api
+        function updateCatchClause(node, variableDeclaration, block) {
+            return node.variableDeclaration !== variableDeclaration
+                || node.block !== block
+                ? update(createCatchClause(variableDeclaration, block), node)
+                : node;
         }
-        function createPostfixType(kind, type) {
-            nextToken();
-            var postfix = createNode(kind, type.pos);
-            postfix.type = type;
-            return finishNode(postfix);
+        //
+        // Property assignments
+        //
+        // @api
+        function createPropertyAssignment(name, initializer) {
+            var node = createBaseNamedDeclaration(288 /* PropertyAssignment */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, name);
+            node.initializer = parenthesizerRules().parenthesizeExpressionForDisallowedComma(initializer);
+            node.transformFlags |=
+                propagateChildFlags(node.name) |
+                    propagateChildFlags(node.initializer);
+            return node;
         }
-        function parseTypeOperator(operator) {
-            var node = createNode(184 /* TypeOperator */);
-            parseExpected(operator);
-            node.operator = operator;
-            node.type = parseTypeOperatorOrHigher();
-            return finishNode(node);
+        function finishUpdatePropertyAssignment(updated, original) {
+            // copy children used only for error reporting
+            if (original.decorators)
+                updated.decorators = original.decorators;
+            if (original.modifiers)
+                updated.modifiers = original.modifiers;
+            if (original.questionToken)
+                updated.questionToken = original.questionToken;
+            if (original.exclamationToken)
+                updated.exclamationToken = original.exclamationToken;
+            return update(updated, original);
+        }
+        // @api
+        function updatePropertyAssignment(node, name, initializer) {
+            return node.name !== name
+                || node.initializer !== initializer
+                ? finishUpdatePropertyAssignment(createPropertyAssignment(name, initializer), node)
+                : node;
+        }
+        // @api
+        function createShorthandPropertyAssignment(name, objectAssignmentInitializer) {
+            var node = createBaseNamedDeclaration(289 /* ShorthandPropertyAssignment */, 
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, name);
+            node.objectAssignmentInitializer = objectAssignmentInitializer && parenthesizerRules().parenthesizeExpressionForDisallowedComma(objectAssignmentInitializer);
+            node.transformFlags |=
+                propagateChildFlags(node.objectAssignmentInitializer) |
+                    256 /* ContainsES2015 */;
+            return node;
         }
-        function parseInferType() {
-            var node = createNode(181 /* InferType */);
-            parseExpected(132 /* InferKeyword */);
-            var typeParameter = createNode(155 /* TypeParameter */);
-            typeParameter.name = parseIdentifier();
-            node.typeParameter = finishNode(typeParameter);
-            return finishNode(node);
+        function finishUpdateShorthandPropertyAssignment(updated, original) {
+            // copy children used only for error reporting
+            if (original.decorators)
+                updated.decorators = original.decorators;
+            if (original.modifiers)
+                updated.modifiers = original.modifiers;
+            if (original.equalsToken)
+                updated.equalsToken = original.equalsToken;
+            if (original.questionToken)
+                updated.questionToken = original.questionToken;
+            if (original.exclamationToken)
+                updated.exclamationToken = original.exclamationToken;
+            return update(updated, original);
+        }
+        // @api
+        function updateShorthandPropertyAssignment(node, name, objectAssignmentInitializer) {
+            return node.name !== name
+                || node.objectAssignmentInitializer !== objectAssignmentInitializer
+                ? finishUpdateShorthandPropertyAssignment(createShorthandPropertyAssignment(name, objectAssignmentInitializer), node)
+                : node;
+        }
+        // @api
+        function createSpreadAssignment(expression) {
+            var node = createBaseNode(290 /* SpreadAssignment */);
+            node.expression = parenthesizerRules().parenthesizeExpressionForDisallowedComma(expression);
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    32 /* ContainsES2018 */ |
+                    16384 /* ContainsObjectRestOrSpread */;
+            return node;
         }
-        function parseTypeOperatorOrHigher() {
-            var operator = token();
-            switch (operator) {
-                case 134 /* KeyOfKeyword */:
-                case 147 /* UniqueKeyword */:
-                case 138 /* ReadonlyKeyword */:
-                    return parseTypeOperator(operator);
-                case 132 /* InferKeyword */:
-                    return parseInferType();
-            }
-            return parsePostfixTypeOrHigher();
+        // @api
+        function updateSpreadAssignment(node, expression) {
+            return node.expression !== expression
+                ? update(createSpreadAssignment(expression), node)
+                : node;
         }
-        function parseUnionOrIntersectionType(kind, parseConstituentType, operator) {
-            var start = scanner.getStartPos();
-            var hasLeadingOperator = parseOptional(operator);
-            var type = parseConstituentType();
-            if (token() === operator || hasLeadingOperator) {
-                var types = [type];
-                while (parseOptional(operator)) {
-                    types.push(parseConstituentType());
-                }
-                var node = createNode(kind, start);
-                node.types = createNodeArray(types, start);
-                type = finishNode(node);
-            }
-            return type;
+        //
+        // Enum
+        //
+        // @api
+        function createEnumMember(name, initializer) {
+            var node = createBaseNode(291 /* EnumMember */);
+            node.name = asName(name);
+            node.initializer = initializer && parenthesizerRules().parenthesizeExpressionForDisallowedComma(initializer);
+            node.transformFlags |=
+                propagateChildFlags(node.name) |
+                    propagateChildFlags(node.initializer) |
+                    1 /* ContainsTypeScript */;
+            return node;
         }
-        function parseIntersectionTypeOrHigher() {
-            return parseUnionOrIntersectionType(179 /* IntersectionType */, parseTypeOperatorOrHigher, 50 /* AmpersandToken */);
+        // @api
+        function updateEnumMember(node, name, initializer) {
+            return node.name !== name
+                || node.initializer !== initializer
+                ? update(createEnumMember(name, initializer), node)
+                : node;
         }
-        function parseUnionTypeOrHigher() {
-            return parseUnionOrIntersectionType(178 /* UnionType */, parseIntersectionTypeOrHigher, 51 /* BarToken */);
+        //
+        // Top-level nodes
+        //
+        // @api
+        function createSourceFile(statements, endOfFileToken, flags) {
+            var node = baseFactory.createBaseSourceFileNode(297 /* SourceFile */);
+            node.statements = createNodeArray(statements);
+            node.endOfFileToken = endOfFileToken;
+            node.flags |= flags;
+            node.fileName = "";
+            node.text = "";
+            node.languageVersion = 0;
+            node.languageVariant = 0;
+            node.scriptKind = 0;
+            node.isDeclarationFile = false;
+            node.hasNoDefaultLib = false;
+            node.transformFlags |=
+                propagateChildrenFlags(node.statements) |
+                    propagateChildFlags(node.endOfFileToken);
+            return node;
         }
-        function isStartOfFunctionType() {
-            if (token() === 29 /* LessThanToken */) {
-                return true;
-            }
-            return token() === 20 /* OpenParenToken */ && lookAhead(isUnambiguouslyStartOfFunctionType);
+        function cloneSourceFileWithChanges(source, statements, isDeclarationFile, referencedFiles, typeReferences, hasNoDefaultLib, libReferences) {
+            var node = baseFactory.createBaseSourceFileNode(297 /* SourceFile */);
+            for (var p in source) {
+                if (p === "emitNode" || ts.hasProperty(node, p) || !ts.hasProperty(source, p))
+                    continue;
+                node[p] = source[p];
+            }
+            node.flags |= source.flags;
+            node.statements = createNodeArray(statements);
+            node.endOfFileToken = source.endOfFileToken;
+            node.isDeclarationFile = isDeclarationFile;
+            node.referencedFiles = referencedFiles;
+            node.typeReferenceDirectives = typeReferences;
+            node.hasNoDefaultLib = hasNoDefaultLib;
+            node.libReferenceDirectives = libReferences;
+            node.transformFlags =
+                propagateChildrenFlags(node.statements) |
+                    propagateChildFlags(node.endOfFileToken);
+            return node;
         }
-        function skipParameterStart() {
-            if (ts.isModifierKind(token())) {
-                // Skip modifiers
-                parseModifiers();
-            }
-            if (isIdentifier() || token() === 104 /* ThisKeyword */) {
-                nextToken();
-                return true;
-            }
-            if (token() === 22 /* OpenBracketToken */ || token() === 18 /* OpenBraceToken */) {
-                // Return true if we can parse an array or object binding pattern with no errors
-                var previousErrorCount = parseDiagnostics.length;
-                parseIdentifierOrPattern();
-                return previousErrorCount === parseDiagnostics.length;
-            }
-            return false;
+        // @api
+        function updateSourceFile(node, statements, isDeclarationFile, referencedFiles, typeReferenceDirectives, hasNoDefaultLib, libReferenceDirectives) {
+            if (isDeclarationFile === void 0) { isDeclarationFile = node.isDeclarationFile; }
+            if (referencedFiles === void 0) { referencedFiles = node.referencedFiles; }
+            if (typeReferenceDirectives === void 0) { typeReferenceDirectives = node.typeReferenceDirectives; }
+            if (hasNoDefaultLib === void 0) { hasNoDefaultLib = node.hasNoDefaultLib; }
+            if (libReferenceDirectives === void 0) { libReferenceDirectives = node.libReferenceDirectives; }
+            return node.statements !== statements
+                || node.isDeclarationFile !== isDeclarationFile
+                || node.referencedFiles !== referencedFiles
+                || node.typeReferenceDirectives !== typeReferenceDirectives
+                || node.hasNoDefaultLib !== hasNoDefaultLib
+                || node.libReferenceDirectives !== libReferenceDirectives
+                ? update(cloneSourceFileWithChanges(node, statements, isDeclarationFile, referencedFiles, typeReferenceDirectives, hasNoDefaultLib, libReferenceDirectives), node)
+                : node;
+        }
+        // @api
+        function createBundle(sourceFiles, prepends) {
+            if (prepends === void 0) { prepends = ts.emptyArray; }
+            var node = createBaseNode(298 /* Bundle */);
+            node.prepends = prepends;
+            node.sourceFiles = sourceFiles;
+            return node;
         }
-        function isUnambiguouslyStartOfFunctionType() {
-            nextToken();
-            if (token() === 21 /* CloseParenToken */ || token() === 25 /* DotDotDotToken */) {
-                // ( )
-                // ( ...
-                return true;
-            }
-            if (skipParameterStart()) {
-                // We successfully skipped modifiers (if any) and an identifier or binding pattern,
-                // now see if we have something that indicates a parameter declaration
-                if (token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */ ||
-                    token() === 57 /* QuestionToken */ || token() === 62 /* EqualsToken */) {
-                    // ( xxx :
-                    // ( xxx ,
-                    // ( xxx ?
-                    // ( xxx =
-                    return true;
+        // @api
+        function updateBundle(node, sourceFiles, prepends) {
+            if (prepends === void 0) { prepends = ts.emptyArray; }
+            return node.sourceFiles !== sourceFiles
+                || node.prepends !== prepends
+                ? update(createBundle(sourceFiles, prepends), node)
+                : node;
+        }
+        // @api
+        function createUnparsedSource(prologues, syntheticReferences, texts) {
+            var node = createBaseNode(299 /* UnparsedSource */);
+            node.prologues = prologues;
+            node.syntheticReferences = syntheticReferences;
+            node.texts = texts;
+            node.fileName = "";
+            node.text = "";
+            node.referencedFiles = ts.emptyArray;
+            node.libReferenceDirectives = ts.emptyArray;
+            node.getLineAndCharacterOfPosition = function (pos) { return ts.getLineAndCharacterOfPosition(node, pos); };
+            return node;
+        }
+        function createBaseUnparsedNode(kind, data) {
+            var node = createBaseNode(kind);
+            node.data = data;
+            return node;
+        }
+        // @api
+        function createUnparsedPrologue(data) {
+            return createBaseUnparsedNode(292 /* UnparsedPrologue */, data);
+        }
+        // @api
+        function createUnparsedPrepend(data, texts) {
+            var node = createBaseUnparsedNode(293 /* UnparsedPrepend */, data);
+            node.texts = texts;
+            return node;
+        }
+        // @api
+        function createUnparsedTextLike(data, internal) {
+            return createBaseUnparsedNode(internal ? 295 /* UnparsedInternalText */ : 294 /* UnparsedText */, data);
+        }
+        // @api
+        function createUnparsedSyntheticReference(section) {
+            var node = createBaseNode(296 /* UnparsedSyntheticReference */);
+            node.data = section.data;
+            node.section = section;
+            return node;
+        }
+        // @api
+        function createInputFiles() {
+            var node = createBaseNode(300 /* InputFiles */);
+            node.javascriptText = "";
+            node.declarationText = "";
+            return node;
+        }
+        //
+        // Synthetic Nodes (used by checker)
+        //
+        // @api
+        function createSyntheticExpression(type, isSpread, tupleNameSource) {
+            if (isSpread === void 0) { isSpread = false; }
+            var node = createBaseNode(227 /* SyntheticExpression */);
+            node.type = type;
+            node.isSpread = isSpread;
+            node.tupleNameSource = tupleNameSource;
+            return node;
+        }
+        // @api
+        function createSyntaxList(children) {
+            var node = createBaseNode(334 /* SyntaxList */);
+            node._children = children;
+            return node;
+        }
+        //
+        // Transformation nodes
+        //
+        /**
+         * Creates a synthetic statement to act as a placeholder for a not-emitted statement in
+         * order to preserve comments.
+         *
+         * @param original The original statement.
+         */
+        // @api
+        function createNotEmittedStatement(original) {
+            var node = createBaseNode(335 /* NotEmittedStatement */);
+            node.original = original;
+            ts.setTextRange(node, original);
+            return node;
+        }
+        /**
+         * Creates a synthetic expression to act as a placeholder for a not-emitted expression in
+         * order to preserve comments or sourcemap positions.
+         *
+         * @param expression The inner expression to emit.
+         * @param original The original outer expression.
+         */
+        // @api
+        function createPartiallyEmittedExpression(expression, original) {
+            var node = createBaseNode(336 /* PartiallyEmittedExpression */);
+            node.expression = expression;
+            node.original = original;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    1 /* ContainsTypeScript */;
+            ts.setTextRange(node, original);
+            return node;
+        }
+        // @api
+        function updatePartiallyEmittedExpression(node, expression) {
+            return node.expression !== expression
+                ? update(createPartiallyEmittedExpression(expression, node.original), node)
+                : node;
+        }
+        function flattenCommaElements(node) {
+            if (ts.nodeIsSynthesized(node) && !ts.isParseTreeNode(node) && !node.original && !node.emitNode && !node.id) {
+                if (ts.isCommaListExpression(node)) {
+                    return node.elements;
                 }
-                if (token() === 21 /* CloseParenToken */) {
-                    nextToken();
-                    if (token() === 38 /* EqualsGreaterThanToken */) {
-                        // ( xxx ) =>
-                        return true;
-                    }
+                if (ts.isBinaryExpression(node) && ts.isCommaToken(node.operatorToken)) {
+                    return [node.left, node.right];
                 }
             }
-            return false;
+            return node;
         }
-        function parseTypeOrTypePredicate() {
-            var typePredicateVariable = isIdentifier() && tryParse(parseTypePredicatePrefix);
-            var type = parseType();
-            if (typePredicateVariable) {
-                var node = createNode(168 /* TypePredicate */, typePredicateVariable.pos);
-                node.assertsModifier = undefined;
-                node.parameterName = typePredicateVariable;
-                node.type = type;
-                return finishNode(node);
+        // @api
+        function createCommaListExpression(elements) {
+            var node = createBaseNode(337 /* CommaListExpression */);
+            node.elements = createNodeArray(ts.sameFlatMap(elements, flattenCommaElements));
+            node.transformFlags |= propagateChildrenFlags(node.elements);
+            return node;
+        }
+        // @api
+        function updateCommaListExpression(node, elements) {
+            return node.elements !== elements
+                ? update(createCommaListExpression(elements), node)
+                : node;
+        }
+        /**
+         * Creates a synthetic element to act as a placeholder for the end of an emitted declaration in
+         * order to properly emit exports.
+         */
+        // @api
+        function createEndOfDeclarationMarker(original) {
+            var node = createBaseNode(339 /* EndOfDeclarationMarker */);
+            node.emitNode = {};
+            node.original = original;
+            return node;
+        }
+        /**
+         * Creates a synthetic element to act as a placeholder for the beginning of a merged declaration in
+         * order to properly emit exports.
+         */
+        // @api
+        function createMergeDeclarationMarker(original) {
+            var node = createBaseNode(338 /* MergeDeclarationMarker */);
+            node.emitNode = {};
+            node.original = original;
+            return node;
+        }
+        // @api
+        function createSyntheticReferenceExpression(expression, thisArg) {
+            var node = createBaseNode(340 /* SyntheticReferenceExpression */);
+            node.expression = expression;
+            node.thisArg = thisArg;
+            node.transformFlags |=
+                propagateChildFlags(node.expression) |
+                    propagateChildFlags(node.thisArg);
+            return node;
+        }
+        // @api
+        function updateSyntheticReferenceExpression(node, expression, thisArg) {
+            return node.expression !== expression
+                || node.thisArg !== thisArg
+                ? update(createSyntheticReferenceExpression(expression, thisArg), node)
+                : node;
+        }
+        function cloneNode(node) {
+            // We don't use "clone" from core.ts here, as we need to preserve the prototype chain of
+            // the original node. We also need to exclude specific properties and only include own-
+            // properties (to skip members already defined on the shared prototype).
+            if (node === undefined) {
+                return node;
             }
-            else {
-                return type;
+            var clone = ts.isSourceFile(node) ? baseFactory.createBaseSourceFileNode(297 /* SourceFile */) :
+                ts.isIdentifier(node) ? baseFactory.createBaseIdentifierNode(78 /* Identifier */) :
+                    ts.isPrivateIdentifier(node) ? baseFactory.createBasePrivateIdentifierNode(79 /* PrivateIdentifier */) :
+                        !ts.isNodeKind(node.kind) ? baseFactory.createBaseTokenNode(node.kind) :
+                            baseFactory.createBaseNode(node.kind);
+            clone.flags |= (node.flags & ~8 /* Synthesized */);
+            clone.transformFlags = node.transformFlags;
+            setOriginalNode(clone, node);
+            for (var key in node) {
+                if (clone.hasOwnProperty(key) || !node.hasOwnProperty(key)) {
+                    continue;
+                }
+                clone[key] = node[key];
             }
+            return clone;
         }
-        function parseTypePredicatePrefix() {
-            var id = parseIdentifier();
-            if (token() === 133 /* IsKeyword */ && !scanner.hasPrecedingLineBreak()) {
-                nextToken();
-                return id;
-            }
+        function createImmediatelyInvokedFunctionExpression(statements, param, paramValue) {
+            return createCallExpression(createFunctionExpression(
+            /*modifiers*/ undefined, 
+            /*asteriskToken*/ undefined, 
+            /*name*/ undefined, 
+            /*typeParameters*/ undefined, 
+            /*parameters*/ param ? [param] : [], 
+            /*type*/ undefined, createBlock(statements, /*multiLine*/ true)), 
+            /*typeArguments*/ undefined, 
+            /*argumentsArray*/ paramValue ? [paramValue] : []);
         }
-        function parseAssertsTypePredicate() {
-            var node = createNode(168 /* TypePredicate */);
-            node.assertsModifier = parseExpectedToken(124 /* AssertsKeyword */);
-            node.parameterName = token() === 104 /* ThisKeyword */ ? parseThisTypeNode() : parseIdentifier();
-            node.type = parseOptional(133 /* IsKeyword */) ? parseType() : undefined;
-            return finishNode(node);
+        function createImmediatelyInvokedArrowFunction(statements, param, paramValue) {
+            return createCallExpression(createArrowFunction(
+            /*modifiers*/ undefined, 
+            /*typeParameters*/ undefined, 
+            /*parameters*/ param ? [param] : [], 
+            /*type*/ undefined, 
+            /*equalsGreaterThanToken*/ undefined, createBlock(statements, /*multiLine*/ true)), 
+            /*typeArguments*/ undefined, 
+            /*argumentsArray*/ paramValue ? [paramValue] : []);
         }
-        function parseType() {
-            // The rules about 'yield' only apply to actual code/expression contexts.  They don't
-            // apply to 'type' contexts.  So we disable these parameters here before moving on.
-            return doOutsideOfContext(40960 /* TypeExcludesFlags */, parseTypeWorker);
+        function createVoidZero() {
+            return createVoidExpression(createNumericLiteral("0"));
         }
-        function parseTypeWorker(noConditionalTypes) {
-            if (isStartOfFunctionType() || token() === 99 /* NewKeyword */) {
-                return parseFunctionOrConstructorType();
-            }
-            var type = parseUnionTypeOrHigher();
-            if (!noConditionalTypes && !scanner.hasPrecedingLineBreak() && parseOptional(90 /* ExtendsKeyword */)) {
-                var node = createNode(180 /* ConditionalType */, type.pos);
-                node.checkType = type;
-                // The type following 'extends' is not permitted to be another conditional type
-                node.extendsType = parseTypeWorker(/*noConditionalTypes*/ true);
-                parseExpected(57 /* QuestionToken */);
-                node.trueType = parseTypeWorker();
-                parseExpected(58 /* ColonToken */);
-                node.falseType = parseTypeWorker();
-                return finishNode(node);
+        function createExportDefault(expression) {
+            return createExportAssignment(
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*isExportEquals*/ false, expression);
+        }
+        function createExternalModuleExport(exportName) {
+            return createExportDeclaration(
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*isTypeOnly*/ false, createNamedExports([
+                createExportSpecifier(/*propertyName*/ undefined, exportName)
+            ]));
+        }
+        //
+        // Utilities
+        //
+        function createTypeCheck(value, tag) {
+            return tag === "undefined"
+                ? factory.createStrictEquality(value, createVoidZero())
+                : factory.createStrictEquality(createTypeOfExpression(value), createStringLiteral(tag));
+        }
+        function createMethodCall(object, methodName, argumentsList) {
+            return createCallExpression(createPropertyAccessExpression(object, methodName), 
+            /*typeArguments*/ undefined, argumentsList);
+        }
+        function createFunctionBindCall(target, thisArg, argumentsList) {
+            return createMethodCall(target, "bind", __spreadArrays([thisArg], argumentsList));
+        }
+        function createFunctionCallCall(target, thisArg, argumentsList) {
+            return createMethodCall(target, "call", __spreadArrays([thisArg], argumentsList));
+        }
+        function createFunctionApplyCall(target, thisArg, argumentsExpression) {
+            return createMethodCall(target, "apply", [thisArg, argumentsExpression]);
+        }
+        function createGlobalMethodCall(globalObjectName, methodName, argumentsList) {
+            return createMethodCall(createIdentifier(globalObjectName), methodName, argumentsList);
+        }
+        function createArraySliceCall(array, start) {
+            return createMethodCall(array, "slice", start === undefined ? [] : [asExpression(start)]);
+        }
+        function createArrayConcatCall(array, argumentsList) {
+            return createMethodCall(array, "concat", argumentsList);
+        }
+        function createObjectDefinePropertyCall(target, propertyName, attributes) {
+            return createGlobalMethodCall("Object", "defineProperty", [target, asExpression(propertyName), attributes]);
+        }
+        function tryAddPropertyAssignment(properties, propertyName, expression) {
+            if (expression) {
+                properties.push(createPropertyAssignment(propertyName, expression));
+                return true;
             }
-            return type;
+            return false;
         }
-        function parseTypeAnnotation() {
-            return parseOptional(58 /* ColonToken */) ? parseType() : undefined;
+        function createPropertyDescriptor(attributes, singleLine) {
+            var properties = [];
+            tryAddPropertyAssignment(properties, "enumerable", asExpression(attributes.enumerable));
+            tryAddPropertyAssignment(properties, "configurable", asExpression(attributes.configurable));
+            var isData = tryAddPropertyAssignment(properties, "writable", asExpression(attributes.writable));
+            isData = tryAddPropertyAssignment(properties, "value", attributes.value) || isData;
+            var isAccessor = tryAddPropertyAssignment(properties, "get", attributes.get);
+            isAccessor = tryAddPropertyAssignment(properties, "set", attributes.set) || isAccessor;
+            ts.Debug.assert(!(isData && isAccessor), "A PropertyDescriptor may not be both an accessor descriptor and a data descriptor.");
+            return createObjectLiteralExpression(properties, !singleLine);
+        }
+        function updateOuterExpression(outerExpression, expression) {
+            switch (outerExpression.kind) {
+                case 207 /* ParenthesizedExpression */: return updateParenthesizedExpression(outerExpression, expression);
+                case 206 /* TypeAssertionExpression */: return updateTypeAssertion(outerExpression, outerExpression.type, expression);
+                case 224 /* AsExpression */: return updateAsExpression(outerExpression, expression, outerExpression.type);
+                case 225 /* NonNullExpression */: return updateNonNullExpression(outerExpression, expression);
+                case 336 /* PartiallyEmittedExpression */: return updatePartiallyEmittedExpression(outerExpression, expression);
+            }
+        }
+        /**
+         * Determines whether a node is a parenthesized expression that can be ignored when recreating outer expressions.
+         *
+         * A parenthesized expression can be ignored when all of the following are true:
+         *
+         * - It's `pos` and `end` are not -1
+         * - It does not have a custom source map range
+         * - It does not have a custom comment range
+         * - It does not have synthetic leading or trailing comments
+         *
+         * If an outermost parenthesized expression is ignored, but the containing expression requires a parentheses around
+         * the expression to maintain precedence, a new parenthesized expression should be created automatically when
+         * the containing expression is created/updated.
+         */
+        function isIgnorableParen(node) {
+            return ts.isParenthesizedExpression(node)
+                && ts.nodeIsSynthesized(node)
+                && ts.nodeIsSynthesized(ts.getSourceMapRange(node))
+                && ts.nodeIsSynthesized(ts.getCommentRange(node))
+                && !ts.some(ts.getSyntheticLeadingComments(node))
+                && !ts.some(ts.getSyntheticTrailingComments(node));
+        }
+        function restoreOuterExpressions(outerExpression, innerExpression, kinds) {
+            if (kinds === void 0) { kinds = 15 /* All */; }
+            if (outerExpression && ts.isOuterExpression(outerExpression, kinds) && !isIgnorableParen(outerExpression)) {
+                return updateOuterExpression(outerExpression, restoreOuterExpressions(outerExpression.expression, innerExpression));
+            }
+            return innerExpression;
+        }
+        function restoreEnclosingLabel(node, outermostLabeledStatement, afterRestoreLabelCallback) {
+            if (!outermostLabeledStatement) {
+                return node;
+            }
+            var updated = updateLabeledStatement(outermostLabeledStatement, outermostLabeledStatement.label, ts.isLabeledStatement(outermostLabeledStatement.statement)
+                ? restoreEnclosingLabel(node, outermostLabeledStatement.statement)
+                : node);
+            if (afterRestoreLabelCallback) {
+                afterRestoreLabelCallback(outermostLabeledStatement);
+            }
+            return updated;
         }
-        // EXPRESSIONS
-        function isStartOfLeftHandSideExpression() {
-            switch (token()) {
-                case 104 /* ThisKeyword */:
-                case 102 /* SuperKeyword */:
-                case 100 /* NullKeyword */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
+        function shouldBeCapturedInTempVariable(node, cacheIdentifiers) {
+            var target = ts.skipParentheses(node);
+            switch (target.kind) {
+                case 78 /* Identifier */:
+                    return cacheIdentifiers;
+                case 107 /* ThisKeyword */:
                 case 8 /* NumericLiteral */:
                 case 9 /* BigIntLiteral */:
                 case 10 /* StringLiteral */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 15 /* TemplateHead */:
-                case 20 /* OpenParenToken */:
-                case 22 /* OpenBracketToken */:
-                case 18 /* OpenBraceToken */:
-                case 94 /* FunctionKeyword */:
-                case 80 /* ClassKeyword */:
-                case 99 /* NewKeyword */:
-                case 43 /* SlashToken */:
-                case 67 /* SlashEqualsToken */:
-                case 75 /* Identifier */:
+                    return false;
+                case 199 /* ArrayLiteralExpression */:
+                    var elements = target.elements;
+                    if (elements.length === 0) {
+                        return false;
+                    }
                     return true;
-                case 96 /* ImportKeyword */:
-                    return lookAhead(nextTokenIsOpenParenOrLessThanOrDot);
+                case 200 /* ObjectLiteralExpression */:
+                    return target.properties.length > 0;
                 default:
-                    return isIdentifier();
+                    return true;
             }
         }
-        function isStartOfExpression() {
-            if (isStartOfLeftHandSideExpression()) {
-                return true;
+        function createCallBinding(expression, recordTempVariable, languageVersion, cacheIdentifiers) {
+            if (cacheIdentifiers === void 0) { cacheIdentifiers = false; }
+            var callee = ts.skipOuterExpressions(expression, 15 /* All */);
+            var thisArg;
+            var target;
+            if (ts.isSuperProperty(callee)) {
+                thisArg = createThis();
+                target = callee;
+            }
+            else if (ts.isSuperKeyword(callee)) {
+                thisArg = createThis();
+                target = languageVersion !== undefined && languageVersion < 2 /* ES2015 */
+                    ? ts.setTextRange(createIdentifier("_super"), callee)
+                    : callee;
+            }
+            else if (ts.getEmitFlags(callee) & 4096 /* HelperName */) {
+                thisArg = createVoidZero();
+                target = parenthesizerRules().parenthesizeLeftSideOfAccess(callee);
+            }
+            else if (ts.isPropertyAccessExpression(callee)) {
+                if (shouldBeCapturedInTempVariable(callee.expression, cacheIdentifiers)) {
+                    // for `a.b()` target is `(_a = a).b` and thisArg is `_a`
+                    thisArg = createTempVariable(recordTempVariable);
+                    target = createPropertyAccessExpression(ts.setTextRange(factory.createAssignment(thisArg, callee.expression), callee.expression), callee.name);
+                    ts.setTextRange(target, callee);
+                }
+                else {
+                    thisArg = callee.expression;
+                    target = callee;
+                }
             }
-            switch (token()) {
-                case 39 /* PlusToken */:
-                case 40 /* MinusToken */:
-                case 54 /* TildeToken */:
-                case 53 /* ExclamationToken */:
-                case 85 /* DeleteKeyword */:
-                case 108 /* TypeOfKeyword */:
-                case 110 /* VoidKeyword */:
-                case 45 /* PlusPlusToken */:
-                case 46 /* MinusMinusToken */:
-                case 29 /* LessThanToken */:
-                case 127 /* AwaitKeyword */:
-                case 121 /* YieldKeyword */:
-                case 76 /* PrivateIdentifier */:
-                    // Yield/await always starts an expression.  Either it is an identifier (in which case
-                    // it is definitely an expression).  Or it's a keyword (either because we're in
-                    // a generator or async function, or in strict mode (or both)) and it started a yield or await expression.
-                    return true;
-                default:
-                    // Error tolerance.  If we see the start of some binary operator, we consider
-                    // that the start of an expression.  That way we'll parse out a missing identifier,
-                    // give a good message about an identifier being missing, and then consume the
-                    // rest of the binary expression.
-                    if (isBinaryOperator()) {
-                        return true;
-                    }
-                    return isIdentifier();
+            else if (ts.isElementAccessExpression(callee)) {
+                if (shouldBeCapturedInTempVariable(callee.expression, cacheIdentifiers)) {
+                    // for `a[b]()` target is `(_a = a)[b]` and thisArg is `_a`
+                    thisArg = createTempVariable(recordTempVariable);
+                    target = createElementAccessExpression(ts.setTextRange(factory.createAssignment(thisArg, callee.expression), callee.expression), callee.argumentExpression);
+                    ts.setTextRange(target, callee);
+                }
+                else {
+                    thisArg = callee.expression;
+                    target = callee;
+                }
             }
+            else {
+                // for `a()` target is `a` and thisArg is `void 0`
+                thisArg = createVoidZero();
+                target = parenthesizerRules().parenthesizeLeftSideOfAccess(expression);
+            }
+            return { target: target, thisArg: thisArg };
+        }
+        function inlineExpressions(expressions) {
+            // Avoid deeply nested comma expressions as traversing them during emit can result in "Maximum call
+            // stack size exceeded" errors.
+            return expressions.length > 10
+                ? createCommaListExpression(expressions)
+                : ts.reduceLeft(expressions, factory.createComma);
+        }
+        function getName(node, allowComments, allowSourceMaps, emitFlags) {
+            if (emitFlags === void 0) { emitFlags = 0; }
+            var nodeName = ts.getNameOfDeclaration(node);
+            if (nodeName && ts.isIdentifier(nodeName) && !ts.isGeneratedIdentifier(nodeName)) {
+                // TODO(rbuckton): Does this need to be parented?
+                var name = ts.setParent(ts.setTextRange(cloneNode(nodeName), nodeName), nodeName.parent);
+                emitFlags |= ts.getEmitFlags(nodeName);
+                if (!allowSourceMaps)
+                    emitFlags |= 48 /* NoSourceMap */;
+                if (!allowComments)
+                    emitFlags |= 1536 /* NoComments */;
+                if (emitFlags)
+                    ts.setEmitFlags(name, emitFlags);
+                return name;
+            }
+            return getGeneratedNameForNode(node);
         }
-        function isStartOfExpressionStatement() {
-            // As per the grammar, none of '{' or 'function' or 'class' can start an expression statement.
-            return token() !== 18 /* OpenBraceToken */ &&
-                token() !== 94 /* FunctionKeyword */ &&
-                token() !== 80 /* ClassKeyword */ &&
-                token() !== 59 /* AtToken */ &&
-                isStartOfExpression();
+        /**
+         * Gets the internal name of a declaration. This is primarily used for declarations that can be
+         * referred to by name in the body of an ES5 class function body. An internal name will *never*
+         * be prefixed with an module or namespace export modifier like "exports." when emitted as an
+         * expression. An internal name will also *never* be renamed due to a collision with a block
+         * scoped variable.
+         *
+         * @param node The declaration.
+         * @param allowComments A value indicating whether comments may be emitted for the name.
+         * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
+         */
+        function getInternalName(node, allowComments, allowSourceMaps) {
+            return getName(node, allowComments, allowSourceMaps, 16384 /* LocalName */ | 32768 /* InternalName */);
         }
-        function parseExpression() {
-            // Expression[in]:
-            //      AssignmentExpression[in]
-            //      Expression[in] , AssignmentExpression[in]
-            // clear the decorator context when parsing Expression, as it should be unambiguous when parsing a decorator
-            var saveDecoratorContext = inDecoratorContext();
-            if (saveDecoratorContext) {
-                setDecoratorContext(/*val*/ false);
-            }
-            var expr = parseAssignmentExpressionOrHigher();
-            var operatorToken;
-            while ((operatorToken = parseOptionalToken(27 /* CommaToken */))) {
-                expr = makeBinaryExpression(expr, operatorToken, parseAssignmentExpressionOrHigher());
-            }
-            if (saveDecoratorContext) {
-                setDecoratorContext(/*val*/ true);
-            }
-            return expr;
-        }
-        function parseInitializer() {
-            return parseOptional(62 /* EqualsToken */) ? parseAssignmentExpressionOrHigher() : undefined;
-        }
-        function parseAssignmentExpressionOrHigher() {
-            //  AssignmentExpression[in,yield]:
-            //      1) ConditionalExpression[?in,?yield]
-            //      2) LeftHandSideExpression = AssignmentExpression[?in,?yield]
-            //      3) LeftHandSideExpression AssignmentOperator AssignmentExpression[?in,?yield]
-            //      4) ArrowFunctionExpression[?in,?yield]
-            //      5) AsyncArrowFunctionExpression[in,yield,await]
-            //      6) [+Yield] YieldExpression[?In]
-            //
-            // Note: for ease of implementation we treat productions '2' and '3' as the same thing.
-            // (i.e. they're both BinaryExpressions with an assignment operator in it).
-            // First, do the simple check if we have a YieldExpression (production '6').
-            if (isYieldExpression()) {
-                return parseYieldExpression();
-            }
-            // Then, check if we have an arrow function (production '4' and '5') that starts with a parenthesized
-            // parameter list or is an async arrow function.
-            // AsyncArrowFunctionExpression:
-            //      1) async[no LineTerminator here]AsyncArrowBindingIdentifier[?Yield][no LineTerminator here]=>AsyncConciseBody[?In]
-            //      2) CoverCallExpressionAndAsyncArrowHead[?Yield, ?Await][no LineTerminator here]=>AsyncConciseBody[?In]
-            // Production (1) of AsyncArrowFunctionExpression is parsed in "tryParseAsyncSimpleArrowFunctionExpression".
-            // And production (2) is parsed in "tryParseParenthesizedArrowFunctionExpression".
-            //
-            // If we do successfully parse arrow-function, we must *not* recurse for productions 1, 2 or 3. An ArrowFunction is
-            // not a LeftHandSideExpression, nor does it start a ConditionalExpression.  So we are done
-            // with AssignmentExpression if we see one.
-            var arrowExpression = tryParseParenthesizedArrowFunctionExpression() || tryParseAsyncSimpleArrowFunctionExpression();
-            if (arrowExpression) {
-                return arrowExpression;
-            }
-            // Now try to see if we're in production '1', '2' or '3'.  A conditional expression can
-            // start with a LogicalOrExpression, while the assignment productions can only start with
-            // LeftHandSideExpressions.
-            //
-            // So, first, we try to just parse out a BinaryExpression.  If we get something that is a
-            // LeftHandSide or higher, then we can try to parse out the assignment expression part.
-            // Otherwise, we try to parse out the conditional expression bit.  We want to allow any
-            // binary expression here, so we pass in the 'lowest' precedence here so that it matches
-            // and consumes anything.
-            var expr = parseBinaryExpressionOrHigher(/*precedence*/ 0);
-            // To avoid a look-ahead, we did not handle the case of an arrow function with a single un-parenthesized
-            // parameter ('x => ...') above. We handle it here by checking if the parsed expression was a single
-            // identifier and the current token is an arrow.
-            if (expr.kind === 75 /* Identifier */ && token() === 38 /* EqualsGreaterThanToken */) {
-                return parseSimpleArrowFunctionExpression(expr);
-            }
-            // Now see if we might be in cases '2' or '3'.
-            // If the expression was a LHS expression, and we have an assignment operator, then
-            // we're in '2' or '3'. Consume the assignment and return.
-            //
-            // Note: we call reScanGreaterToken so that we get an appropriately merged token
-            // for cases like `> > =` becoming `>>=`
-            if (ts.isLeftHandSideExpression(expr) && ts.isAssignmentOperator(reScanGreaterToken())) {
-                return makeBinaryExpression(expr, parseTokenNode(), parseAssignmentExpressionOrHigher());
-            }
-            // It wasn't an assignment or a lambda.  This is a conditional expression:
-            return parseConditionalExpressionRest(expr);
-        }
-        function isYieldExpression() {
-            if (token() === 121 /* YieldKeyword */) {
-                // If we have a 'yield' keyword, and this is a context where yield expressions are
-                // allowed, then definitely parse out a yield expression.
-                if (inYieldContext()) {
-                    return true;
-                }
-                // We're in a context where 'yield expr' is not allowed.  However, if we can
-                // definitely tell that the user was trying to parse a 'yield expr' and not
-                // just a normal expr that start with a 'yield' identifier, then parse out
-                // a 'yield expr'.  We can then report an error later that they are only
-                // allowed in generator expressions.
-                //
-                // for example, if we see 'yield(foo)', then we'll have to treat that as an
-                // invocation expression of something called 'yield'.  However, if we have
-                // 'yield foo' then that is not legal as a normal expression, so we can
-                // definitely recognize this as a yield expression.
-                //
-                // for now we just check if the next token is an identifier.  More heuristics
-                // can be added here later as necessary.  We just need to make sure that we
-                // don't accidentally consume something legal.
-                return lookAhead(nextTokenIsIdentifierOrKeywordOrLiteralOnSameLine);
-            }
-            return false;
-        }
-        function nextTokenIsIdentifierOnSameLine() {
-            nextToken();
-            return !scanner.hasPrecedingLineBreak() && isIdentifier();
-        }
-        function parseYieldExpression() {
-            var node = createNode(212 /* YieldExpression */);
-            // YieldExpression[In] :
-            //      yield
-            //      yield [no LineTerminator here] [Lexical goal InputElementRegExp]AssignmentExpression[?In, Yield]
-            //      yield [no LineTerminator here] * [Lexical goal InputElementRegExp]AssignmentExpression[?In, Yield]
-            nextToken();
-            if (!scanner.hasPrecedingLineBreak() &&
-                (token() === 41 /* AsteriskToken */ || isStartOfExpression())) {
-                node.asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
-                node.expression = parseAssignmentExpressionOrHigher();
-                return finishNode(node);
-            }
-            else {
-                // if the next token is not on the same line as yield.  or we don't have an '*' or
-                // the start of an expression, then this is just a simple "yield" expression.
-                return finishNode(node);
-            }
-        }
-        function parseSimpleArrowFunctionExpression(identifier, asyncModifier) {
-            ts.Debug.assert(token() === 38 /* EqualsGreaterThanToken */, "parseSimpleArrowFunctionExpression should only have been called if we had a =>");
-            var node;
-            if (asyncModifier) {
-                node = createNode(202 /* ArrowFunction */, asyncModifier.pos);
-                node.modifiers = asyncModifier;
-            }
-            else {
-                node = createNode(202 /* ArrowFunction */, identifier.pos);
-            }
-            var parameter = createNode(156 /* Parameter */, identifier.pos);
-            parameter.name = identifier;
-            finishNode(parameter);
-            node.parameters = createNodeArray([parameter], parameter.pos, parameter.end);
-            node.equalsGreaterThanToken = parseExpectedToken(38 /* EqualsGreaterThanToken */);
-            node.body = parseArrowFunctionExpressionBody(/*isAsync*/ !!asyncModifier);
-            return addJSDocComment(finishNode(node));
-        }
-        function tryParseParenthesizedArrowFunctionExpression() {
-            var triState = isParenthesizedArrowFunctionExpression();
-            if (triState === 0 /* False */) {
-                // It's definitely not a parenthesized arrow function expression.
-                return undefined;
-            }
-            // If we definitely have an arrow function, then we can just parse one, not requiring a
-            // following => or { token. Otherwise, we *might* have an arrow function.  Try to parse
-            // it out, but don't allow any ambiguity, and return 'undefined' if this could be an
-            // expression instead.
-            var arrowFunction = triState === 1 /* True */
-                ? parseParenthesizedArrowFunctionExpressionHead(/*allowAmbiguity*/ true)
-                : tryParse(parsePossibleParenthesizedArrowFunctionExpressionHead);
-            if (!arrowFunction) {
-                // Didn't appear to actually be a parenthesized arrow function.  Just bail out.
-                return undefined;
-            }
-            var isAsync = hasModifierOfKind(arrowFunction, 126 /* AsyncKeyword */);
-            // If we have an arrow, then try to parse the body. Even if not, try to parse if we
-            // have an opening brace, just in case we're in an error state.
-            var lastToken = token();
-            arrowFunction.equalsGreaterThanToken = parseExpectedToken(38 /* EqualsGreaterThanToken */);
-            arrowFunction.body = (lastToken === 38 /* EqualsGreaterThanToken */ || lastToken === 18 /* OpenBraceToken */)
-                ? parseArrowFunctionExpressionBody(isAsync)
-                : parseIdentifier();
-            return finishNode(arrowFunction);
-        }
-        //  True        -> We definitely expect a parenthesized arrow function here.
-        //  False       -> There *cannot* be a parenthesized arrow function here.
-        //  Unknown     -> There *might* be a parenthesized arrow function here.
-        //                 Speculatively look ahead to be sure, and rollback if not.
-        function isParenthesizedArrowFunctionExpression() {
-            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */ || token() === 126 /* AsyncKeyword */) {
-                return lookAhead(isParenthesizedArrowFunctionExpressionWorker);
-            }
-            if (token() === 38 /* EqualsGreaterThanToken */) {
-                // ERROR RECOVERY TWEAK:
-                // If we see a standalone => try to parse it as an arrow function expression as that's
-                // likely what the user intended to write.
-                return 1 /* True */;
-            }
-            // Definitely not a parenthesized arrow function.
-            return 0 /* False */;
-        }
-        function isParenthesizedArrowFunctionExpressionWorker() {
-            if (token() === 126 /* AsyncKeyword */) {
-                nextToken();
-                if (scanner.hasPrecedingLineBreak()) {
-                    return 0 /* False */;
-                }
-                if (token() !== 20 /* OpenParenToken */ && token() !== 29 /* LessThanToken */) {
-                    return 0 /* False */;
-                }
-            }
-            var first = token();
-            var second = nextToken();
-            if (first === 20 /* OpenParenToken */) {
-                if (second === 21 /* CloseParenToken */) {
-                    // Simple cases: "() =>", "(): ", and "() {".
-                    // This is an arrow function with no parameters.
-                    // The last one is not actually an arrow function,
-                    // but this is probably what the user intended.
-                    var third = nextToken();
-                    switch (third) {
-                        case 38 /* EqualsGreaterThanToken */:
-                        case 58 /* ColonToken */:
-                        case 18 /* OpenBraceToken */:
-                            return 1 /* True */;
-                        default:
-                            return 0 /* False */;
-                    }
-                }
-                // If encounter "([" or "({", this could be the start of a binding pattern.
-                // Examples:
-                //      ([ x ]) => { }
-                //      ({ x }) => { }
-                //      ([ x ])
-                //      ({ x })
-                if (second === 22 /* OpenBracketToken */ || second === 18 /* OpenBraceToken */) {
-                    return 2 /* Unknown */;
-                }
-                // Simple case: "(..."
-                // This is an arrow function with a rest parameter.
-                if (second === 25 /* DotDotDotToken */) {
-                    return 1 /* True */;
-                }
-                // Check for "(xxx yyy", where xxx is a modifier and yyy is an identifier. This
-                // isn't actually allowed, but we want to treat it as a lambda so we can provide
-                // a good error message.
-                if (ts.isModifierKind(second) && second !== 126 /* AsyncKeyword */ && lookAhead(nextTokenIsIdentifier)) {
-                    return 1 /* True */;
-                }
-                // If we had "(" followed by something that's not an identifier,
-                // then this definitely doesn't look like a lambda.  "this" is not
-                // valid, but we want to parse it and then give a semantic error.
-                if (!isIdentifier() && second !== 104 /* ThisKeyword */) {
-                    return 0 /* False */;
-                }
-                switch (nextToken()) {
-                    case 58 /* ColonToken */:
-                        // If we have something like "(a:", then we must have a
-                        // type-annotated parameter in an arrow function expression.
-                        return 1 /* True */;
-                    case 57 /* QuestionToken */:
-                        nextToken();
-                        // If we have "(a?:" or "(a?," or "(a?=" or "(a?)" then it is definitely a lambda.
-                        if (token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */ || token() === 62 /* EqualsToken */ || token() === 21 /* CloseParenToken */) {
-                            return 1 /* True */;
-                        }
-                        // Otherwise it is definitely not a lambda.
-                        return 0 /* False */;
-                    case 27 /* CommaToken */:
-                    case 62 /* EqualsToken */:
-                    case 21 /* CloseParenToken */:
-                        // If we have "(a," or "(a=" or "(a)" this *could* be an arrow function
-                        return 2 /* Unknown */;
-                }
-                // It is definitely not an arrow function
-                return 0 /* False */;
-            }
-            else {
-                ts.Debug.assert(first === 29 /* LessThanToken */);
-                // If we have "<" not followed by an identifier,
-                // then this definitely is not an arrow function.
-                if (!isIdentifier()) {
-                    return 0 /* False */;
-                }
-                // JSX overrides
-                if (sourceFile.languageVariant === 1 /* JSX */) {
-                    var isArrowFunctionInJsx = lookAhead(function () {
-                        var third = nextToken();
-                        if (third === 90 /* ExtendsKeyword */) {
-                            var fourth = nextToken();
-                            switch (fourth) {
-                                case 62 /* EqualsToken */:
-                                case 31 /* GreaterThanToken */:
-                                    return false;
-                                default:
-                                    return true;
-                            }
-                        }
-                        else if (third === 27 /* CommaToken */) {
-                            return true;
-                        }
-                        return false;
-                    });
-                    if (isArrowFunctionInJsx) {
-                        return 1 /* True */;
-                    }
-                    return 0 /* False */;
-                }
-                // This *could* be a parenthesized arrow function.
-                return 2 /* Unknown */;
-            }
-        }
-        function parsePossibleParenthesizedArrowFunctionExpressionHead() {
-            var tokenPos = scanner.getTokenPos();
-            if (notParenthesizedArrow && notParenthesizedArrow.has(tokenPos.toString())) {
-                return undefined;
-            }
-            var result = parseParenthesizedArrowFunctionExpressionHead(/*allowAmbiguity*/ false);
-            if (!result) {
-                (notParenthesizedArrow || (notParenthesizedArrow = ts.createMap())).set(tokenPos.toString(), true);
-            }
-            return result;
+        /**
+         * Gets the local name of a declaration. This is primarily used for declarations that can be
+         * referred to by name in the declaration's immediate scope (classes, enums, namespaces). A
+         * local name will *never* be prefixed with an module or namespace export modifier like
+         * "exports." when emitted as an expression.
+         *
+         * @param node The declaration.
+         * @param allowComments A value indicating whether comments may be emitted for the name.
+         * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
+         */
+        function getLocalName(node, allowComments, allowSourceMaps) {
+            return getName(node, allowComments, allowSourceMaps, 16384 /* LocalName */);
         }
-        function tryParseAsyncSimpleArrowFunctionExpression() {
-            // We do a check here so that we won't be doing unnecessarily call to "lookAhead"
-            if (token() === 126 /* AsyncKeyword */) {
-                if (lookAhead(isUnParenthesizedAsyncArrowFunctionWorker) === 1 /* True */) {
-                    var asyncModifier = parseModifiersForArrowFunction();
-                    var expr = parseBinaryExpressionOrHigher(/*precedence*/ 0);
-                    return parseSimpleArrowFunctionExpression(expr, asyncModifier);
-                }
-            }
-            return undefined;
+        /**
+         * Gets the export name of a declaration. This is primarily used for declarations that can be
+         * referred to by name in the declaration's immediate scope (classes, enums, namespaces). An
+         * export name will *always* be prefixed with an module or namespace export modifier like
+         * `"exports."` when emitted as an expression if the name points to an exported symbol.
+         *
+         * @param node The declaration.
+         * @param allowComments A value indicating whether comments may be emitted for the name.
+         * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
+         */
+        function getExportName(node, allowComments, allowSourceMaps) {
+            return getName(node, allowComments, allowSourceMaps, 8192 /* ExportName */);
         }
-        function isUnParenthesizedAsyncArrowFunctionWorker() {
-            // AsyncArrowFunctionExpression:
-            //      1) async[no LineTerminator here]AsyncArrowBindingIdentifier[?Yield][no LineTerminator here]=>AsyncConciseBody[?In]
-            //      2) CoverCallExpressionAndAsyncArrowHead[?Yield, ?Await][no LineTerminator here]=>AsyncConciseBody[?In]
-            if (token() === 126 /* AsyncKeyword */) {
-                nextToken();
-                // If the "async" is followed by "=>" token then it is not a beginning of an async arrow-function
-                // but instead a simple arrow-function which will be parsed inside "parseAssignmentExpressionOrHigher"
-                if (scanner.hasPrecedingLineBreak() || token() === 38 /* EqualsGreaterThanToken */) {
-                    return 0 /* False */;
-                }
-                // Check for un-parenthesized AsyncArrowFunction
-                var expr = parseBinaryExpressionOrHigher(/*precedence*/ 0);
-                if (!scanner.hasPrecedingLineBreak() && expr.kind === 75 /* Identifier */ && token() === 38 /* EqualsGreaterThanToken */) {
-                    return 1 /* True */;
-                }
-            }
-            return 0 /* False */;
+        /**
+         * Gets the name of a declaration for use in declarations.
+         *
+         * @param node The declaration.
+         * @param allowComments A value indicating whether comments may be emitted for the name.
+         * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
+         */
+        function getDeclarationName(node, allowComments, allowSourceMaps) {
+            return getName(node, allowComments, allowSourceMaps);
         }
-        function parseParenthesizedArrowFunctionExpressionHead(allowAmbiguity) {
-            var node = createNodeWithJSDoc(202 /* ArrowFunction */);
-            node.modifiers = parseModifiersForArrowFunction();
-            var isAsync = hasModifierOfKind(node, 126 /* AsyncKeyword */) ? 2 /* Await */ : 0 /* None */;
-            // Arrow functions are never generators.
-            //
-            // If we're speculatively parsing a signature for a parenthesized arrow function, then
-            // we have to have a complete parameter list.  Otherwise we might see something like
-            // a => (b => c)
-            // And think that "(b =>" was actually a parenthesized arrow function with a missing
-            // close paren.
-            if (!fillSignature(58 /* ColonToken */, isAsync, node) && !allowAmbiguity) {
-                return undefined;
-            }
-            // Parsing a signature isn't enough.
-            // Parenthesized arrow signatures often look like other valid expressions.
-            // For instance:
-            //  - "(x = 10)" is an assignment expression parsed as a signature with a default parameter value.
-            //  - "(x,y)" is a comma expression parsed as a signature with two parameters.
-            //  - "a ? (b): c" will have "(b):" parsed as a signature with a return type annotation.
-            //  - "a ? (b): function() {}" will too, since function() is a valid JSDoc function type.
-            //
-            // So we need just a bit of lookahead to ensure that it can only be a signature.
-            var hasJSDocFunctionType = node.type && ts.isJSDocFunctionType(node.type);
-            if (!allowAmbiguity && token() !== 38 /* EqualsGreaterThanToken */ && (hasJSDocFunctionType || token() !== 18 /* OpenBraceToken */)) {
-                // Returning undefined here will cause our caller to rewind to where we started from.
-                return undefined;
-            }
-            return node;
+        /**
+         * Gets a namespace-qualified name for use in expressions.
+         *
+         * @param ns The namespace identifier.
+         * @param name The name.
+         * @param allowComments A value indicating whether comments may be emitted for the name.
+         * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
+         */
+        function getNamespaceMemberName(ns, name, allowComments, allowSourceMaps) {
+            var qualifiedName = createPropertyAccessExpression(ns, ts.nodeIsSynthesized(name) ? name : cloneNode(name));
+            ts.setTextRange(qualifiedName, name);
+            var emitFlags = 0;
+            if (!allowSourceMaps)
+                emitFlags |= 48 /* NoSourceMap */;
+            if (!allowComments)
+                emitFlags |= 1536 /* NoComments */;
+            if (emitFlags)
+                ts.setEmitFlags(qualifiedName, emitFlags);
+            return qualifiedName;
         }
-        function parseArrowFunctionExpressionBody(isAsync) {
-            if (token() === 18 /* OpenBraceToken */) {
-                return parseFunctionBlock(isAsync ? 2 /* Await */ : 0 /* None */);
-            }
-            if (token() !== 26 /* SemicolonToken */ &&
-                token() !== 94 /* FunctionKeyword */ &&
-                token() !== 80 /* ClassKeyword */ &&
-                isStartOfStatement() &&
-                !isStartOfExpressionStatement()) {
-                // Check if we got a plain statement (i.e. no expression-statements, no function/class expressions/declarations)
-                //
-                // Here we try to recover from a potential error situation in the case where the
-                // user meant to supply a block. For example, if the user wrote:
-                //
-                //  a =>
-                //      let v = 0;
-                //  }
-                //
-                // they may be missing an open brace.  Check to see if that's the case so we can
-                // try to recover better.  If we don't do this, then the next close curly we see may end
-                // up preemptively closing the containing construct.
-                //
-                // Note: even when 'IgnoreMissingOpenBrace' is passed, parseBody will still error.
-                return parseFunctionBlock(16 /* IgnoreMissingOpenBrace */ | (isAsync ? 2 /* Await */ : 0 /* None */));
+        /**
+         * Gets the exported name of a declaration for use in expressions.
+         *
+         * An exported name will *always* be prefixed with an module or namespace export modifier like
+         * "exports." if the name points to an exported symbol.
+         *
+         * @param ns The namespace identifier.
+         * @param node The declaration.
+         * @param allowComments A value indicating whether comments may be emitted for the name.
+         * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
+         */
+        function getExternalModuleOrNamespaceExportName(ns, node, allowComments, allowSourceMaps) {
+            if (ns && ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                return getNamespaceMemberName(ns, getName(node), allowComments, allowSourceMaps);
             }
-            return isAsync
-                ? doInAwaitContext(parseAssignmentExpressionOrHigher)
-                : doOutsideOfAwaitContext(parseAssignmentExpressionOrHigher);
+            return getExportName(node, allowComments, allowSourceMaps);
         }
-        function parseConditionalExpressionRest(leftOperand) {
-            // Note: we are passed in an expression which was produced from parseBinaryExpressionOrHigher.
-            var questionToken = parseOptionalToken(57 /* QuestionToken */);
-            if (!questionToken) {
-                return leftOperand;
-            }
-            // Note: we explicitly 'allowIn' in the whenTrue part of the condition expression, and
-            // we do not that for the 'whenFalse' part.
-            var node = createNode(210 /* ConditionalExpression */, leftOperand.pos);
-            node.condition = leftOperand;
-            node.questionToken = questionToken;
-            node.whenTrue = doOutsideOfContext(disallowInAndDecoratorContext, parseAssignmentExpressionOrHigher);
-            node.colonToken = parseExpectedToken(58 /* ColonToken */);
-            node.whenFalse = ts.nodeIsPresent(node.colonToken)
-                ? parseAssignmentExpressionOrHigher()
-                : createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ false, ts.Diagnostics._0_expected, ts.tokenToString(58 /* ColonToken */));
-            return finishNode(node);
+        /**
+         * Copies any necessary standard and custom prologue-directives into target array.
+         * @param source origin statements array
+         * @param target result statements array
+         * @param ensureUseStrict boolean determining whether the function need to add prologue-directives
+         * @param visitor Optional callback used to visit any custom prologue directives.
+         */
+        function copyPrologue(source, target, ensureUseStrict, visitor) {
+            var offset = copyStandardPrologue(source, target, ensureUseStrict);
+            return copyCustomPrologue(source, target, offset, visitor);
         }
-        function parseBinaryExpressionOrHigher(precedence) {
-            var leftOperand = parseUnaryExpressionOrHigher();
-            return parseBinaryExpressionRest(precedence, leftOperand);
+        function isUseStrictPrologue(node) {
+            return ts.isStringLiteral(node.expression) && node.expression.text === "use strict";
         }
-        function isInOrOfKeyword(t) {
-            return t === 97 /* InKeyword */ || t === 152 /* OfKeyword */;
+        function createUseStrictPrologue() {
+            return ts.startOnNewLine(createExpressionStatement(createStringLiteral("use strict")));
         }
-        function parseBinaryExpressionRest(precedence, leftOperand) {
-            while (true) {
-                // We either have a binary operator here, or we're finished.  We call
-                // reScanGreaterToken so that we merge token sequences like > and = into >=
-                reScanGreaterToken();
-                var newPrecedence = ts.getBinaryOperatorPrecedence(token());
-                // Check the precedence to see if we should "take" this operator
-                // - For left associative operator (all operator but **), consume the operator,
-                //   recursively call the function below, and parse binaryExpression as a rightOperand
-                //   of the caller if the new precedence of the operator is greater then or equal to the current precedence.
-                //   For example:
-                //      a - b - c;
-                //            ^token; leftOperand = b. Return b to the caller as a rightOperand
-                //      a * b - c
-                //            ^token; leftOperand = b. Return b to the caller as a rightOperand
-                //      a - b * c;
-                //            ^token; leftOperand = b. Return b * c to the caller as a rightOperand
-                // - For right associative operator (**), consume the operator, recursively call the function
-                //   and parse binaryExpression as a rightOperand of the caller if the new precedence of
-                //   the operator is strictly grater than the current precedence
-                //   For example:
-                //      a ** b ** c;
-                //             ^^token; leftOperand = b. Return b ** c to the caller as a rightOperand
-                //      a - b ** c;
-                //            ^^token; leftOperand = b. Return b ** c to the caller as a rightOperand
-                //      a ** b - c
-                //             ^token; leftOperand = b. Return b to the caller as a rightOperand
-                var consumeCurrentOperator = token() === 42 /* AsteriskAsteriskToken */ ?
-                    newPrecedence >= precedence :
-                    newPrecedence > precedence;
-                if (!consumeCurrentOperator) {
-                    break;
-                }
-                if (token() === 97 /* InKeyword */ && inDisallowInContext()) {
-                    break;
-                }
-                if (token() === 123 /* AsKeyword */) {
-                    // Make sure we *do* perform ASI for constructs like this:
-                    //    var x = foo
-                    //    as (Bar)
-                    // This should be parsed as an initialized variable, followed
-                    // by a function call to 'as' with the argument 'Bar'
-                    if (scanner.hasPrecedingLineBreak()) {
-                        break;
-                    }
-                    else {
-                        nextToken();
-                        leftOperand = makeAsExpression(leftOperand, parseType());
+        /**
+         * Copies only the standard (string-expression) prologue-directives into the target statement-array.
+         * @param source origin statements array
+         * @param target result statements array
+         * @param ensureUseStrict boolean determining whether the function need to add prologue-directives
+         */
+        function copyStandardPrologue(source, target, ensureUseStrict) {
+            ts.Debug.assert(target.length === 0, "Prologue directives should be at the first statement in the target statements array");
+            var foundUseStrict = false;
+            var statementOffset = 0;
+            var numStatements = source.length;
+            while (statementOffset < numStatements) {
+                var statement = source[statementOffset];
+                if (ts.isPrologueDirective(statement)) {
+                    if (isUseStrictPrologue(statement)) {
+                        foundUseStrict = true;
                     }
+                    target.push(statement);
                 }
                 else {
-                    leftOperand = makeBinaryExpression(leftOperand, parseTokenNode(), parseBinaryExpressionOrHigher(newPrecedence));
+                    break;
                 }
+                statementOffset++;
             }
-            return leftOperand;
-        }
-        function isBinaryOperator() {
-            if (inDisallowInContext() && token() === 97 /* InKeyword */) {
-                return false;
-            }
-            return ts.getBinaryOperatorPrecedence(token()) > 0;
-        }
-        function makeBinaryExpression(left, operatorToken, right) {
-            var node = createNode(209 /* BinaryExpression */, left.pos);
-            node.left = left;
-            node.operatorToken = operatorToken;
-            node.right = right;
-            return finishNode(node);
-        }
-        function makeAsExpression(left, right) {
-            var node = createNode(217 /* AsExpression */, left.pos);
-            node.expression = left;
-            node.type = right;
-            return finishNode(node);
-        }
-        function parsePrefixUnaryExpression() {
-            var node = createNode(207 /* PrefixUnaryExpression */);
-            node.operator = token();
-            nextToken();
-            node.operand = parseSimpleUnaryExpression();
-            return finishNode(node);
-        }
-        function parseDeleteExpression() {
-            var node = createNode(203 /* DeleteExpression */);
-            nextToken();
-            node.expression = parseSimpleUnaryExpression();
-            return finishNode(node);
-        }
-        function parseTypeOfExpression() {
-            var node = createNode(204 /* TypeOfExpression */);
-            nextToken();
-            node.expression = parseSimpleUnaryExpression();
-            return finishNode(node);
-        }
-        function parseVoidExpression() {
-            var node = createNode(205 /* VoidExpression */);
-            nextToken();
-            node.expression = parseSimpleUnaryExpression();
-            return finishNode(node);
-        }
-        function isAwaitExpression() {
-            if (token() === 127 /* AwaitKeyword */) {
-                if (inAwaitContext()) {
-                    return true;
-                }
-                // here we are using similar heuristics as 'isYieldExpression'
-                return lookAhead(nextTokenIsIdentifierOrKeywordOrLiteralOnSameLine);
+            if (ensureUseStrict && !foundUseStrict) {
+                target.push(createUseStrictPrologue());
             }
-            return false;
-        }
-        function parseAwaitExpression() {
-            var node = createNode(206 /* AwaitExpression */);
-            nextToken();
-            node.expression = parseSimpleUnaryExpression();
-            return finishNode(node);
+            return statementOffset;
         }
-        /**
-         * Parse ES7 exponential expression and await expression
-         *
-         * ES7 ExponentiationExpression:
-         *      1) UnaryExpression[?Yield]
-         *      2) UpdateExpression[?Yield] ** ExponentiationExpression[?Yield]
-         *
-         */
-        function parseUnaryExpressionOrHigher() {
-            /**
-             * ES7 UpdateExpression:
-             *      1) LeftHandSideExpression[?Yield]
-             *      2) LeftHandSideExpression[?Yield][no LineTerminator here]++
-             *      3) LeftHandSideExpression[?Yield][no LineTerminator here]--
-             *      4) ++UnaryExpression[?Yield]
-             *      5) --UnaryExpression[?Yield]
-             */
-            if (isUpdateExpression()) {
-                var updateExpression = parseUpdateExpression();
-                return token() === 42 /* AsteriskAsteriskToken */ ?
-                    parseBinaryExpressionRest(ts.getBinaryOperatorPrecedence(token()), updateExpression) :
-                    updateExpression;
-            }
-            /**
-             * ES7 UnaryExpression:
-             *      1) UpdateExpression[?yield]
-             *      2) delete UpdateExpression[?yield]
-             *      3) void UpdateExpression[?yield]
-             *      4) typeof UpdateExpression[?yield]
-             *      5) + UpdateExpression[?yield]
-             *      6) - UpdateExpression[?yield]
-             *      7) ~ UpdateExpression[?yield]
-             *      8) ! UpdateExpression[?yield]
-             */
-            var unaryOperator = token();
-            var simpleUnaryExpression = parseSimpleUnaryExpression();
-            if (token() === 42 /* AsteriskAsteriskToken */) {
-                var pos = ts.skipTrivia(sourceText, simpleUnaryExpression.pos);
-                var end = simpleUnaryExpression.end;
-                if (simpleUnaryExpression.kind === 199 /* TypeAssertionExpression */) {
-                    parseErrorAt(pos, end, ts.Diagnostics.A_type_assertion_expression_is_not_allowed_in_the_left_hand_side_of_an_exponentiation_expression_Consider_enclosing_the_expression_in_parentheses);
+        function copyCustomPrologue(source, target, statementOffset, visitor, filter) {
+            if (filter === void 0) { filter = ts.returnTrue; }
+            var numStatements = source.length;
+            while (statementOffset !== undefined && statementOffset < numStatements) {
+                var statement = source[statementOffset];
+                if (ts.getEmitFlags(statement) & 1048576 /* CustomPrologue */ && filter(statement)) {
+                    ts.append(target, visitor ? ts.visitNode(statement, visitor, ts.isStatement) : statement);
                 }
                 else {
-                    parseErrorAt(pos, end, ts.Diagnostics.An_unary_expression_with_the_0_operator_is_not_allowed_in_the_left_hand_side_of_an_exponentiation_expression_Consider_enclosing_the_expression_in_parentheses, ts.tokenToString(unaryOperator));
+                    break;
                 }
+                statementOffset++;
             }
-            return simpleUnaryExpression;
+            return statementOffset;
         }
         /**
-         * Parse ES7 simple-unary expression or higher:
+         * Ensures "use strict" directive is added
          *
-         * ES7 UnaryExpression:
-         *      1) UpdateExpression[?yield]
-         *      2) delete UnaryExpression[?yield]
-         *      3) void UnaryExpression[?yield]
-         *      4) typeof UnaryExpression[?yield]
-         *      5) + UnaryExpression[?yield]
-         *      6) - UnaryExpression[?yield]
-         *      7) ~ UnaryExpression[?yield]
-         *      8) ! UnaryExpression[?yield]
-         *      9) [+Await] await UnaryExpression[?yield]
+         * @param statements An array of statements
          */
-        function parseSimpleUnaryExpression() {
-            switch (token()) {
-                case 39 /* PlusToken */:
-                case 40 /* MinusToken */:
-                case 54 /* TildeToken */:
-                case 53 /* ExclamationToken */:
-                    return parsePrefixUnaryExpression();
-                case 85 /* DeleteKeyword */:
-                    return parseDeleteExpression();
-                case 108 /* TypeOfKeyword */:
-                    return parseTypeOfExpression();
-                case 110 /* VoidKeyword */:
-                    return parseVoidExpression();
-                case 29 /* LessThanToken */:
-                    // This is modified UnaryExpression grammar in TypeScript
-                    //  UnaryExpression (modified):
-                    //      < type > UnaryExpression
-                    return parseTypeAssertion();
-                case 127 /* AwaitKeyword */:
-                    if (isAwaitExpression()) {
-                        return parseAwaitExpression();
-                    }
-                // falls through
-                default:
-                    return parseUpdateExpression();
+        function ensureUseStrict(statements) {
+            var foundUseStrict = ts.findUseStrictPrologue(statements);
+            if (!foundUseStrict) {
+                return ts.setTextRange(createNodeArray(__spreadArrays([createUseStrictPrologue()], statements)), statements);
             }
+            return statements;
         }
         /**
-         * Check if the current token can possibly be an ES7 increment expression.
+         * Lifts a NodeArray containing only Statement nodes to a block.
          *
-         * ES7 UpdateExpression:
-         *      LeftHandSideExpression[?Yield]
-         *      LeftHandSideExpression[?Yield][no LineTerminator here]++
-         *      LeftHandSideExpression[?Yield][no LineTerminator here]--
-         *      ++LeftHandSideExpression[?Yield]
-         *      --LeftHandSideExpression[?Yield]
+         * @param nodes The NodeArray.
          */
-        function isUpdateExpression() {
-            // This function is called inside parseUnaryExpression to decide
-            // whether to call parseSimpleUnaryExpression or call parseUpdateExpression directly
-            switch (token()) {
-                case 39 /* PlusToken */:
-                case 40 /* MinusToken */:
-                case 54 /* TildeToken */:
-                case 53 /* ExclamationToken */:
-                case 85 /* DeleteKeyword */:
-                case 108 /* TypeOfKeyword */:
-                case 110 /* VoidKeyword */:
-                case 127 /* AwaitKeyword */:
-                    return false;
-                case 29 /* LessThanToken */:
-                    // If we are not in JSX context, we are parsing TypeAssertion which is an UnaryExpression
-                    if (sourceFile.languageVariant !== 1 /* JSX */) {
-                        return false;
-                    }
-                // We are in JSX context and the token is part of JSXElement.
-                // falls through
-                default:
-                    return true;
-            }
+        function liftToBlock(nodes) {
+            ts.Debug.assert(ts.every(nodes, ts.isStatementOrBlock), "Cannot lift nodes to a Block.");
+            return ts.singleOrUndefined(nodes) || createBlock(nodes);
         }
-        /**
-         * Parse ES7 UpdateExpression. UpdateExpression is used instead of ES6's PostFixExpression.
-         *
-         * ES7 UpdateExpression[yield]:
-         *      1) LeftHandSideExpression[?yield]
-         *      2) LeftHandSideExpression[?yield] [[no LineTerminator here]]++
-         *      3) LeftHandSideExpression[?yield] [[no LineTerminator here]]--
-         *      4) ++LeftHandSideExpression[?yield]
-         *      5) --LeftHandSideExpression[?yield]
-         * In TypeScript (2), (3) are parsed as PostfixUnaryExpression. (4), (5) are parsed as PrefixUnaryExpression
-         */
-        function parseUpdateExpression() {
-            if (token() === 45 /* PlusPlusToken */ || token() === 46 /* MinusMinusToken */) {
-                var node = createNode(207 /* PrefixUnaryExpression */);
-                node.operator = token();
-                nextToken();
-                node.operand = parseLeftHandSideExpressionOrHigher();
-                return finishNode(node);
-            }
-            else if (sourceFile.languageVariant === 1 /* JSX */ && token() === 29 /* LessThanToken */ && lookAhead(nextTokenIsIdentifierOrKeywordOrGreaterThan)) {
-                // JSXElement is part of primaryExpression
-                return parseJsxElementOrSelfClosingElementOrFragment(/*inExpressionContext*/ true);
-            }
-            var expression = parseLeftHandSideExpressionOrHigher();
-            ts.Debug.assert(ts.isLeftHandSideExpression(expression));
-            if ((token() === 45 /* PlusPlusToken */ || token() === 46 /* MinusMinusToken */) && !scanner.hasPrecedingLineBreak()) {
-                var node = createNode(208 /* PostfixUnaryExpression */, expression.pos);
-                node.operand = expression;
-                node.operator = token();
-                nextToken();
-                return finishNode(node);
+        function findSpanEnd(array, test, start) {
+            var i = start;
+            while (i < array.length && test(array[i])) {
+                i++;
             }
-            return expression;
+            return i;
         }
-        function parseLeftHandSideExpressionOrHigher() {
-            // Original Ecma:
-            // LeftHandSideExpression: See 11.2
-            //      NewExpression
-            //      CallExpression
-            //
-            // Our simplification:
+        function mergeLexicalEnvironment(statements, declarations) {
+            if (!ts.some(declarations)) {
+                return statements;
+            }
+            // When we merge new lexical statements into an existing statement list, we merge them in the following manner:
             //
-            // LeftHandSideExpression: See 11.2
-            //      MemberExpression
-            //      CallExpression
+            // Given:
             //
-            // See comment in parseMemberExpressionOrHigher on how we replaced NewExpression with
-            // MemberExpression to make our lives easier.
+            // | Left                               | Right                               |
+            // |------------------------------------|-------------------------------------|
+            // | [standard prologues (left)]        | [standard prologues (right)]        |
+            // | [hoisted functions (left)]         | [hoisted functions (right)]         |
+            // | [hoisted variables (left)]         | [hoisted variables (right)]         |
+            // | [lexical init statements (left)]   | [lexical init statements (right)]   |
+            // | [other statements (left)]          |                                     |
             //
-            // to best understand the below code, it's important to see how CallExpression expands
-            // out into its own productions:
+            // The resulting statement list will be:
             //
-            // CallExpression:
-            //      MemberExpression Arguments
-            //      CallExpression Arguments
-            //      CallExpression[Expression]
-            //      CallExpression.IdentifierName
-            //      import (AssignmentExpression)
-            //      super Arguments
-            //      super.IdentifierName
+            // | Result                              |
+            // |-------------------------------------|
+            // | [standard prologues (right)]        |
+            // | [standard prologues (left)]         |
+            // | [hoisted functions (right)]         |
+            // | [hoisted functions (left)]          |
+            // | [hoisted variables (right)]         |
+            // | [hoisted variables (left)]          |
+            // | [lexical init statements (right)]   |
+            // | [lexical init statements (left)]    |
+            // | [other statements (left)]           |
             //
-            // Because of the recursion in these calls, we need to bottom out first. There are three
-            // bottom out states we can run into: 1) We see 'super' which must start either of
-            // the last two CallExpression productions. 2) We see 'import' which must start import call.
-            // 3)we have a MemberExpression which either completes the LeftHandSideExpression,
-            // or starts the beginning of the first four CallExpression productions.
-            var expression;
-            if (token() === 96 /* ImportKeyword */) {
-                if (lookAhead(nextTokenIsOpenParenOrLessThan)) {
-                    // We don't want to eagerly consume all import keyword as import call expression so we look ahead to find "("
-                    // For example:
-                    //      var foo3 = require("subfolder
-                    //      import * as foo1 from "module-from-node
-                    // We want this import to be a statement rather than import call expression
-                    sourceFile.flags |= 1048576 /* PossiblyContainsDynamicImport */;
-                    expression = parseTokenNode();
-                }
-                else if (lookAhead(nextTokenIsDot)) {
-                    // This is an 'import.*' metaproperty (i.e. 'import.meta')
-                    var fullStart = scanner.getStartPos();
-                    nextToken(); // advance past the 'import'
-                    nextToken(); // advance past the dot
-                    var node = createNode(219 /* MetaProperty */, fullStart);
-                    node.keywordToken = 96 /* ImportKeyword */;
-                    node.name = parseIdentifierName();
-                    expression = finishNode(node);
-                    sourceFile.flags |= 2097152 /* PossiblyContainsImportMeta */;
+            // NOTE: It is expected that new lexical init statements must be evaluated before existing lexical init statements,
+            // as the prior transformation may depend on the evaluation of the lexical init statements to be in the correct state.
+            // find standard prologues on left in the following order: standard directives, hoisted functions, hoisted variables, other custom
+            var leftStandardPrologueEnd = findSpanEnd(statements, ts.isPrologueDirective, 0);
+            var leftHoistedFunctionsEnd = findSpanEnd(statements, ts.isHoistedFunction, leftStandardPrologueEnd);
+            var leftHoistedVariablesEnd = findSpanEnd(statements, ts.isHoistedVariableStatement, leftHoistedFunctionsEnd);
+            // find standard prologues on right in the following order: standard directives, hoisted functions, hoisted variables, other custom
+            var rightStandardPrologueEnd = findSpanEnd(declarations, ts.isPrologueDirective, 0);
+            var rightHoistedFunctionsEnd = findSpanEnd(declarations, ts.isHoistedFunction, rightStandardPrologueEnd);
+            var rightHoistedVariablesEnd = findSpanEnd(declarations, ts.isHoistedVariableStatement, rightHoistedFunctionsEnd);
+            var rightCustomPrologueEnd = findSpanEnd(declarations, ts.isCustomPrologue, rightHoistedVariablesEnd);
+            ts.Debug.assert(rightCustomPrologueEnd === declarations.length, "Expected declarations to be valid standard or custom prologues");
+            // splice prologues from the right into the left. We do this in reverse order
+            // so that we don't need to recompute the index on the left when we insert items.
+            var left = ts.isNodeArray(statements) ? statements.slice() : statements;
+            // splice other custom prologues from right into left
+            if (rightCustomPrologueEnd > rightHoistedVariablesEnd) {
+                left.splice.apply(left, __spreadArrays([leftHoistedVariablesEnd, 0], declarations.slice(rightHoistedVariablesEnd, rightCustomPrologueEnd)));
+            }
+            // splice hoisted variables from right into left
+            if (rightHoistedVariablesEnd > rightHoistedFunctionsEnd) {
+                left.splice.apply(left, __spreadArrays([leftHoistedFunctionsEnd, 0], declarations.slice(rightHoistedFunctionsEnd, rightHoistedVariablesEnd)));
+            }
+            // splice hoisted functions from right into left
+            if (rightHoistedFunctionsEnd > rightStandardPrologueEnd) {
+                left.splice.apply(left, __spreadArrays([leftStandardPrologueEnd, 0], declarations.slice(rightStandardPrologueEnd, rightHoistedFunctionsEnd)));
+            }
+            // splice standard prologues from right into left (that are not already in left)
+            if (rightStandardPrologueEnd > 0) {
+                if (leftStandardPrologueEnd === 0) {
+                    left.splice.apply(left, __spreadArrays([0, 0], declarations.slice(0, rightStandardPrologueEnd)));
                 }
                 else {
-                    expression = parseMemberExpressionOrHigher();
+                    var leftPrologues = new ts.Map();
+                    for (var i = 0; i < leftStandardPrologueEnd; i++) {
+                        var leftPrologue = statements[i];
+                        leftPrologues.set(leftPrologue.expression.text, true);
+                    }
+                    for (var i = rightStandardPrologueEnd - 1; i >= 0; i--) {
+                        var rightPrologue = declarations[i];
+                        if (!leftPrologues.has(rightPrologue.expression.text)) {
+                            left.unshift(rightPrologue);
+                        }
+                    }
                 }
             }
-            else {
-                expression = token() === 102 /* SuperKeyword */ ? parseSuperExpression() : parseMemberExpressionOrHigher();
+            if (ts.isNodeArray(statements)) {
+                return ts.setTextRange(createNodeArray(left, statements.hasTrailingComma), statements);
             }
-            // Now, we *may* be complete.  However, we might have consumed the start of a
-            // CallExpression or OptionalExpression.  As such, we need to consume the rest
-            // of it here to be complete.
-            return parseCallExpressionRest(expression);
-        }
-        function parseMemberExpressionOrHigher() {
-            // Note: to make our lives simpler, we decompose the NewExpression productions and
-            // place ObjectCreationExpression and FunctionExpression into PrimaryExpression.
-            // like so:
-            //
-            //   PrimaryExpression : See 11.1
-            //      this
-            //      Identifier
-            //      Literal
-            //      ArrayLiteral
-            //      ObjectLiteral
-            //      (Expression)
-            //      FunctionExpression
-            //      new MemberExpression Arguments?
-            //
-            //   MemberExpression : See 11.2
-            //      PrimaryExpression
-            //      MemberExpression[Expression]
-            //      MemberExpression.IdentifierName
-            //
-            //   CallExpression : See 11.2
-            //      MemberExpression
-            //      CallExpression Arguments
-            //      CallExpression[Expression]
-            //      CallExpression.IdentifierName
-            //
-            // Technically this is ambiguous.  i.e. CallExpression defines:
-            //
-            //   CallExpression:
-            //      CallExpression Arguments
-            //
-            // If you see: "new Foo()"
-            //
-            // Then that could be treated as a single ObjectCreationExpression, or it could be
-            // treated as the invocation of "new Foo".  We disambiguate that in code (to match
-            // the original grammar) by making sure that if we see an ObjectCreationExpression
-            // we always consume arguments if they are there. So we treat "new Foo()" as an
-            // object creation only, and not at all as an invocation.  Another way to think
-            // about this is that for every "new" that we see, we will consume an argument list if
-            // it is there as part of the *associated* object creation node.  Any additional
-            // argument lists we see, will become invocation expressions.
-            //
-            // Because there are no other places in the grammar now that refer to FunctionExpression
-            // or ObjectCreationExpression, it is safe to push down into the PrimaryExpression
-            // production.
-            //
-            // Because CallExpression and MemberExpression are left recursive, we need to bottom out
-            // of the recursion immediately.  So we parse out a primary expression to start with.
-            var expression = parsePrimaryExpression();
-            return parseMemberExpressionRest(expression, /*allowOptionalChain*/ true);
+            return statements;
         }
-        function parseSuperExpression() {
-            var expression = parseTokenNode();
-            if (token() === 29 /* LessThanToken */) {
-                var startPos = getNodePos();
-                var typeArguments = tryParse(parseTypeArgumentsInExpression);
-                if (typeArguments !== undefined) {
-                    parseErrorAt(startPos, getNodePos(), ts.Diagnostics.super_may_not_use_type_arguments);
-                }
-            }
-            if (token() === 20 /* OpenParenToken */ || token() === 24 /* DotToken */ || token() === 22 /* OpenBracketToken */) {
-                return expression;
-            }
-            // If we have seen "super" it must be followed by '(' or '.'.
-            // If it wasn't then just try to parse out a '.' and report an error.
-            var node = createNode(194 /* PropertyAccessExpression */, expression.pos);
-            node.expression = expression;
-            parseExpectedToken(24 /* DotToken */, ts.Diagnostics.super_must_be_followed_by_an_argument_list_or_member_access);
-            // private names will never work with `super` (`super.#foo`), but that's a semantic error, not syntactic
-            node.name = parseRightSideOfDot(/*allowIdentifierNames*/ true, /*allowPrivateIdentifiers*/ true);
-            return finishNode(node);
+        function updateModifiers(node, modifiers) {
+            var _a;
+            if (typeof modifiers === "number") {
+                modifiers = createModifiersFromModifierFlags(modifiers);
+            }
+            return ts.isParameter(node) ? updateParameterDeclaration(node, node.decorators, modifiers, node.dotDotDotToken, node.name, node.questionToken, node.type, node.initializer) :
+                ts.isPropertySignature(node) ? updatePropertySignature(node, modifiers, node.name, node.questionToken, node.type) :
+                    ts.isPropertyDeclaration(node) ? updatePropertyDeclaration(node, node.decorators, modifiers, node.name, (_a = node.questionToken) !== null && _a !== void 0 ? _a : node.exclamationToken, node.type, node.initializer) :
+                        ts.isMethodSignature(node) ? updateMethodSignature(node, modifiers, node.name, node.questionToken, node.typeParameters, node.parameters, node.type) :
+                            ts.isMethodDeclaration(node) ? updateMethodDeclaration(node, node.decorators, modifiers, node.asteriskToken, node.name, node.questionToken, node.typeParameters, node.parameters, node.type, node.body) :
+                                ts.isConstructorDeclaration(node) ? updateConstructorDeclaration(node, node.decorators, modifiers, node.parameters, node.body) :
+                                    ts.isGetAccessorDeclaration(node) ? updateGetAccessorDeclaration(node, node.decorators, modifiers, node.name, node.parameters, node.type, node.body) :
+                                        ts.isSetAccessorDeclaration(node) ? updateSetAccessorDeclaration(node, node.decorators, modifiers, node.name, node.parameters, node.body) :
+                                            ts.isIndexSignatureDeclaration(node) ? updateIndexSignature(node, node.decorators, modifiers, node.parameters, node.type) :
+                                                ts.isFunctionExpression(node) ? updateFunctionExpression(node, modifiers, node.asteriskToken, node.name, node.typeParameters, node.parameters, node.type, node.body) :
+                                                    ts.isArrowFunction(node) ? updateArrowFunction(node, modifiers, node.typeParameters, node.parameters, node.type, node.equalsGreaterThanToken, node.body) :
+                                                        ts.isClassExpression(node) ? updateClassExpression(node, node.decorators, modifiers, node.name, node.typeParameters, node.heritageClauses, node.members) :
+                                                            ts.isVariableStatement(node) ? updateVariableStatement(node, modifiers, node.declarationList) :
+                                                                ts.isFunctionDeclaration(node) ? updateFunctionDeclaration(node, node.decorators, modifiers, node.asteriskToken, node.name, node.typeParameters, node.parameters, node.type, node.body) :
+                                                                    ts.isClassDeclaration(node) ? updateClassDeclaration(node, node.decorators, modifiers, node.name, node.typeParameters, node.heritageClauses, node.members) :
+                                                                        ts.isInterfaceDeclaration(node) ? updateInterfaceDeclaration(node, node.decorators, modifiers, node.name, node.typeParameters, node.heritageClauses, node.members) :
+                                                                            ts.isTypeAliasDeclaration(node) ? updateTypeAliasDeclaration(node, node.decorators, modifiers, node.name, node.typeParameters, node.type) :
+                                                                                ts.isEnumDeclaration(node) ? updateEnumDeclaration(node, node.decorators, modifiers, node.name, node.members) :
+                                                                                    ts.isModuleDeclaration(node) ? updateModuleDeclaration(node, node.decorators, modifiers, node.name, node.body) :
+                                                                                        ts.isImportEqualsDeclaration(node) ? updateImportEqualsDeclaration(node, node.decorators, modifiers, node.name, node.moduleReference) :
+                                                                                            ts.isImportDeclaration(node) ? updateImportDeclaration(node, node.decorators, modifiers, node.importClause, node.moduleSpecifier) :
+                                                                                                ts.isExportAssignment(node) ? updateExportAssignment(node, node.decorators, modifiers, node.expression) :
+                                                                                                    ts.isExportDeclaration(node) ? updateExportDeclaration(node, node.decorators, modifiers, node.isTypeOnly, node.exportClause, node.moduleSpecifier) :
+                                                                                                        ts.Debug.assertNever(node);
+        }
+        function asNodeArray(array) {
+            return array ? createNodeArray(array) : undefined;
+        }
+        function asName(name) {
+            return typeof name === "string" ? createIdentifier(name) :
+                name;
+        }
+        function asExpression(value) {
+            return typeof value === "string" ? createStringLiteral(value) :
+                typeof value === "number" ? createNumericLiteral(value) :
+                    typeof value === "boolean" ? value ? createTrue() : createFalse() :
+                        value;
+        }
+        function asToken(value) {
+            return typeof value === "number" ? createToken(value) : value;
+        }
+        function asEmbeddedStatement(statement) {
+            return statement && ts.isNotEmittedStatement(statement) ? ts.setTextRange(setOriginalNode(createEmptyStatement(), statement), statement) : statement;
+        }
+    }
+    ts.createNodeFactory = createNodeFactory;
+    function updateWithoutOriginal(updated, original) {
+        if (updated !== original) {
+            ts.setTextRange(updated, original);
         }
-        function parseJsxElementOrSelfClosingElementOrFragment(inExpressionContext) {
-            var opening = parseJsxOpeningOrSelfClosingElementOrOpeningFragment(inExpressionContext);
-            var result;
-            if (opening.kind === 268 /* JsxOpeningElement */) {
-                var node = createNode(266 /* JsxElement */, opening.pos);
-                node.openingElement = opening;
-                node.children = parseJsxChildren(node.openingElement);
-                node.closingElement = parseJsxClosingElement(inExpressionContext);
-                if (!tagNamesAreEquivalent(node.openingElement.tagName, node.closingElement.tagName)) {
-                    parseErrorAtRange(node.closingElement, ts.Diagnostics.Expected_corresponding_JSX_closing_tag_for_0, ts.getTextOfNodeFromSourceText(sourceText, node.openingElement.tagName));
-                }
-                result = finishNode(node);
-            }
-            else if (opening.kind === 271 /* JsxOpeningFragment */) {
-                var node = createNode(270 /* JsxFragment */, opening.pos);
-                node.openingFragment = opening;
-                node.children = parseJsxChildren(node.openingFragment);
-                node.closingFragment = parseJsxClosingFragment(inExpressionContext);
-                result = finishNode(node);
-            }
-            else {
-                ts.Debug.assert(opening.kind === 267 /* JsxSelfClosingElement */);
-                // Nothing else to do for self-closing elements
-                result = opening;
-            }
-            // If the user writes the invalid code '<div></div><div></div>' in an expression context (i.e. not wrapped in
-            // an enclosing tag), we'll naively try to parse   ^ this as a 'less than' operator and the remainder of the tag
-            // as garbage, which will cause the formatter to badly mangle the JSX. Perform a speculative parse of a JSX
-            // element if we see a < token so that we can wrap it in a synthetic binary expression so the formatter
-            // does less damage and we can report a better error.
-            // Since JSX elements are invalid < operands anyway, this lookahead parse will only occur in error scenarios
-            // of one sort or another.
-            if (inExpressionContext && token() === 29 /* LessThanToken */) {
-                var invalidElement = tryParse(function () { return parseJsxElementOrSelfClosingElementOrFragment(/*inExpressionContext*/ true); });
-                if (invalidElement) {
-                    parseErrorAtCurrentToken(ts.Diagnostics.JSX_expressions_must_have_one_parent_element);
-                    var badNode = createNode(209 /* BinaryExpression */, result.pos);
-                    badNode.end = invalidElement.end;
-                    badNode.left = result;
-                    badNode.right = invalidElement;
-                    badNode.operatorToken = createMissingNode(27 /* CommaToken */, /*reportAtCurrentPosition*/ false);
-                    badNode.operatorToken.pos = badNode.operatorToken.end = badNode.right.pos;
-                    return badNode;
-                }
-            }
-            return result;
+        return updated;
+    }
+    function updateWithOriginal(updated, original) {
+        if (updated !== original) {
+            setOriginalNode(updated, original);
+            ts.setTextRange(updated, original);
         }
-        function parseJsxText() {
-            var node = createNode(11 /* JsxText */);
-            node.text = scanner.getTokenValue();
-            node.containsOnlyTriviaWhiteSpaces = currentToken === 12 /* JsxTextAllWhiteSpaces */;
-            currentToken = scanner.scanJsxToken();
-            return finishNode(node);
+        return updated;
+    }
+    function getDefaultTagNameForKind(kind) {
+        switch (kind) {
+            case 329 /* JSDocTypeTag */: return "type";
+            case 327 /* JSDocReturnTag */: return "returns";
+            case 328 /* JSDocThisTag */: return "this";
+            case 325 /* JSDocEnumTag */: return "enum";
+            case 317 /* JSDocAuthorTag */: return "author";
+            case 319 /* JSDocClassTag */: return "class";
+            case 320 /* JSDocPublicTag */: return "public";
+            case 321 /* JSDocPrivateTag */: return "private";
+            case 322 /* JSDocProtectedTag */: return "protected";
+            case 323 /* JSDocReadonlyTag */: return "readonly";
+            case 330 /* JSDocTemplateTag */: return "template";
+            case 331 /* JSDocTypedefTag */: return "typedef";
+            case 326 /* JSDocParameterTag */: return "param";
+            case 333 /* JSDocPropertyTag */: return "prop";
+            case 324 /* JSDocCallbackTag */: return "callback";
+            case 315 /* JSDocAugmentsTag */: return "augments";
+            case 316 /* JSDocImplementsTag */: return "implements";
+            default:
+                return ts.Debug.fail("Unsupported kind: " + ts.Debug.formatSyntaxKind(kind));
         }
-        function parseJsxChild(openingTag, token) {
-            switch (token) {
-                case 1 /* EndOfFileToken */:
-                    // If we hit EOF, issue the error at the tag that lacks the closing element
-                    // rather than at the end of the file (which is useless)
-                    if (ts.isJsxOpeningFragment(openingTag)) {
-                        parseErrorAtRange(openingTag, ts.Diagnostics.JSX_fragment_has_no_corresponding_closing_tag);
-                    }
-                    else {
-                        // We want the error span to cover only 'Foo.Bar' in < Foo.Bar >
-                        // or to cover only 'Foo' in < Foo >
-                        var tag = openingTag.tagName;
-                        var start = ts.skipTrivia(sourceText, tag.pos);
-                        parseErrorAt(start, tag.end, ts.Diagnostics.JSX_element_0_has_no_corresponding_closing_tag, ts.getTextOfNodeFromSourceText(sourceText, openingTag.tagName));
-                    }
-                    return undefined;
-                case 30 /* LessThanSlashToken */:
-                case 7 /* ConflictMarkerTrivia */:
-                    return undefined;
-                case 11 /* JsxText */:
-                case 12 /* JsxTextAllWhiteSpaces */:
-                    return parseJsxText();
-                case 18 /* OpenBraceToken */:
-                    return parseJsxExpression(/*inExpressionContext*/ false);
-                case 29 /* LessThanToken */:
-                    return parseJsxElementOrSelfClosingElementOrFragment(/*inExpressionContext*/ false);
-                default:
-                    return ts.Debug.assertNever(token);
-            }
+    }
+    var rawTextScanner;
+    var invalidValueSentinel = {};
+    function getCookedText(kind, rawText) {
+        if (!rawTextScanner) {
+            rawTextScanner = ts.createScanner(99 /* Latest */, /*skipTrivia*/ false, 0 /* Standard */);
         }
-        function parseJsxChildren(openingTag) {
-            var list = [];
-            var listPos = getNodePos();
-            var saveParsingContext = parsingContext;
-            parsingContext |= 1 << 14 /* JsxChildren */;
-            while (true) {
-                var child = parseJsxChild(openingTag, currentToken = scanner.reScanJsxToken());
-                if (!child)
-                    break;
-                list.push(child);
-            }
-            parsingContext = saveParsingContext;
-            return createNodeArray(list, listPos);
+        switch (kind) {
+            case 14 /* NoSubstitutionTemplateLiteral */:
+                rawTextScanner.setText("`" + rawText + "`");
+                break;
+            case 15 /* TemplateHead */:
+                // tslint:disable-next-line no-invalid-template-strings
+                rawTextScanner.setText("`" + rawText + "${");
+                break;
+            case 16 /* TemplateMiddle */:
+                // tslint:disable-next-line no-invalid-template-strings
+                rawTextScanner.setText("}" + rawText + "${");
+                break;
+            case 17 /* TemplateTail */:
+                rawTextScanner.setText("}" + rawText + "`");
+                break;
         }
-        function parseJsxAttributes() {
-            var jsxAttributes = createNode(274 /* JsxAttributes */);
-            jsxAttributes.properties = parseList(13 /* JsxAttributes */, parseJsxAttribute);
-            return finishNode(jsxAttributes);
+        var token = rawTextScanner.scan();
+        if (token === 23 /* CloseBracketToken */) {
+            token = rawTextScanner.reScanTemplateToken(/*isTaggedTemplate*/ false);
         }
-        function parseJsxOpeningOrSelfClosingElementOrOpeningFragment(inExpressionContext) {
-            var fullStart = scanner.getStartPos();
-            parseExpected(29 /* LessThanToken */);
-            if (token() === 31 /* GreaterThanToken */) {
-                // See below for explanation of scanJsxText
-                var node_1 = createNode(271 /* JsxOpeningFragment */, fullStart);
-                scanJsxText();
-                return finishNode(node_1);
-            }
-            var tagName = parseJsxElementName();
-            var typeArguments = tryParseTypeArguments();
-            var attributes = parseJsxAttributes();
-            var node;
-            if (token() === 31 /* GreaterThanToken */) {
-                // Closing tag, so scan the immediately-following text with the JSX scanning instead
-                // of regular scanning to avoid treating illegal characters (e.g. '#') as immediate
-                // scanning errors
-                node = createNode(268 /* JsxOpeningElement */, fullStart);
-                scanJsxText();
-            }
-            else {
-                parseExpected(43 /* SlashToken */);
-                if (inExpressionContext) {
-                    parseExpected(31 /* GreaterThanToken */);
-                }
-                else {
-                    parseExpected(31 /* GreaterThanToken */, /*diagnostic*/ undefined, /*shouldAdvance*/ false);
-                    scanJsxText();
-                }
-                node = createNode(267 /* JsxSelfClosingElement */, fullStart);
-            }
-            node.tagName = tagName;
-            node.typeArguments = typeArguments;
-            node.attributes = attributes;
-            return finishNode(node);
+        if (rawTextScanner.isUnterminated()) {
+            rawTextScanner.setText(undefined);
+            return invalidValueSentinel;
         }
-        function parseJsxElementName() {
-            scanJsxIdentifier();
-            // JsxElement can have name in the form of
-            //      propertyAccessExpression
-            //      primaryExpression in the form of an identifier and "this" keyword
-            // We can't just simply use parseLeftHandSideExpressionOrHigher because then we will start consider class,function etc as a keyword
-            // We only want to consider "this" as a primaryExpression
-            var expression = token() === 104 /* ThisKeyword */ ?
-                parseTokenNode() : parseIdentifierName();
-            while (parseOptional(24 /* DotToken */)) {
-                var propertyAccess = createNode(194 /* PropertyAccessExpression */, expression.pos);
-                propertyAccess.expression = expression;
-                propertyAccess.name = parseRightSideOfDot(/*allowIdentifierNames*/ true, /*allowPrivateIdentifiers*/ false);
-                expression = finishNode(propertyAccess);
-            }
-            return expression;
+        var tokenValue;
+        switch (token) {
+            case 14 /* NoSubstitutionTemplateLiteral */:
+            case 15 /* TemplateHead */:
+            case 16 /* TemplateMiddle */:
+            case 17 /* TemplateTail */:
+                tokenValue = rawTextScanner.getTokenValue();
+                break;
         }
-        function parseJsxExpression(inExpressionContext) {
-            var node = createNode(276 /* JsxExpression */);
-            if (!parseExpected(18 /* OpenBraceToken */)) {
-                return undefined;
-            }
-            if (token() !== 19 /* CloseBraceToken */) {
-                node.dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
-                // Only an AssignmentExpression is valid here per the JSX spec,
-                // but we can unambiguously parse a comma sequence and provide
-                // a better error message in grammar checking.
-                node.expression = parseExpression();
-            }
-            if (inExpressionContext) {
-                parseExpected(19 /* CloseBraceToken */);
-            }
-            else {
-                if (parseExpected(19 /* CloseBraceToken */, /*message*/ undefined, /*shouldAdvance*/ false)) {
-                    scanJsxText();
-                }
-            }
-            return finishNode(node);
+        if (tokenValue === undefined || rawTextScanner.scan() !== 1 /* EndOfFileToken */) {
+            rawTextScanner.setText(undefined);
+            return invalidValueSentinel;
         }
-        function parseJsxAttribute() {
-            if (token() === 18 /* OpenBraceToken */) {
-                return parseJsxSpreadAttribute();
-            }
-            scanJsxIdentifier();
-            var node = createNode(273 /* JsxAttribute */);
-            node.name = parseIdentifierName();
-            if (token() === 62 /* EqualsToken */) {
-                switch (scanJsxAttributeValue()) {
-                    case 10 /* StringLiteral */:
-                        node.initializer = parseLiteralNode();
-                        break;
-                    default:
-                        node.initializer = parseJsxExpression(/*inExpressionContext*/ true);
-                        break;
-                }
-            }
-            return finishNode(node);
+        rawTextScanner.setText(undefined);
+        return tokenValue;
+    }
+    function propagateIdentifierNameFlags(node) {
+        // An IdentifierName is allowed to be `await`
+        return propagateChildFlags(node) & ~8388608 /* ContainsPossibleTopLevelAwait */;
+    }
+    function propagatePropertyNameFlagsOfChild(node, transformFlags) {
+        return transformFlags | (node.transformFlags & 4096 /* PropertyNamePropagatingFlags */);
+    }
+    function propagateChildFlags(child) {
+        if (!child)
+            return 0 /* None */;
+        var childFlags = child.transformFlags & ~getTransformFlagsSubtreeExclusions(child.kind);
+        return ts.isNamedDeclaration(child) && ts.isPropertyName(child.name) ? propagatePropertyNameFlagsOfChild(child.name, childFlags) : childFlags;
+    }
+    function propagateChildrenFlags(children) {
+        return children ? children.transformFlags : 0 /* None */;
+    }
+    function aggregateChildrenFlags(children) {
+        var subtreeFlags = 0 /* None */;
+        for (var _i = 0, children_2 = children; _i < children_2.length; _i++) {
+            var child = children_2[_i];
+            subtreeFlags |= propagateChildFlags(child);
         }
-        function parseJsxSpreadAttribute() {
-            var node = createNode(275 /* JsxSpreadAttribute */);
-            parseExpected(18 /* OpenBraceToken */);
-            parseExpected(25 /* DotDotDotToken */);
-            node.expression = parseExpression();
-            parseExpected(19 /* CloseBraceToken */);
-            return finishNode(node);
+        children.transformFlags = subtreeFlags;
+    }
+    /**
+     * Gets the transform flags to exclude when unioning the transform flags of a subtree.
+     */
+    /* @internal */
+    function getTransformFlagsSubtreeExclusions(kind) {
+        if (kind >= 172 /* FirstTypeNode */ && kind <= 195 /* LastTypeNode */) {
+            return -2 /* TypeExcludes */;
         }
-        function parseJsxClosingElement(inExpressionContext) {
-            var node = createNode(269 /* JsxClosingElement */);
-            parseExpected(30 /* LessThanSlashToken */);
-            node.tagName = parseJsxElementName();
-            if (inExpressionContext) {
-                parseExpected(31 /* GreaterThanToken */);
-            }
-            else {
-                parseExpected(31 /* GreaterThanToken */, /*diagnostic*/ undefined, /*shouldAdvance*/ false);
-                scanJsxText();
-            }
-            return finishNode(node);
+        switch (kind) {
+            case 203 /* CallExpression */:
+            case 204 /* NewExpression */:
+            case 199 /* ArrayLiteralExpression */:
+                return 536879104 /* ArrayLiteralOrCallOrNewExcludes */;
+            case 256 /* ModuleDeclaration */:
+                return 546379776 /* ModuleExcludes */;
+            case 160 /* Parameter */:
+                return 536870912 /* ParameterExcludes */;
+            case 209 /* ArrowFunction */:
+                return 547309568 /* ArrowFunctionExcludes */;
+            case 208 /* FunctionExpression */:
+            case 251 /* FunctionDeclaration */:
+                return 547313664 /* FunctionExcludes */;
+            case 250 /* VariableDeclarationList */:
+                return 537018368 /* VariableDeclarationListExcludes */;
+            case 252 /* ClassDeclaration */:
+            case 221 /* ClassExpression */:
+                return 536905728 /* ClassExcludes */;
+            case 166 /* Constructor */:
+                return 547311616 /* ConstructorExcludes */;
+            case 163 /* PropertyDeclaration */:
+                return 536875008 /* PropertyExcludes */;
+            case 165 /* MethodDeclaration */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+                return 538923008 /* MethodOrAccessorExcludes */;
+            case 128 /* AnyKeyword */:
+            case 144 /* NumberKeyword */:
+            case 155 /* BigIntKeyword */:
+            case 141 /* NeverKeyword */:
+            case 147 /* StringKeyword */:
+            case 145 /* ObjectKeyword */:
+            case 131 /* BooleanKeyword */:
+            case 148 /* SymbolKeyword */:
+            case 113 /* VoidKeyword */:
+            case 159 /* TypeParameter */:
+            case 162 /* PropertySignature */:
+            case 164 /* MethodSignature */:
+            case 169 /* CallSignature */:
+            case 170 /* ConstructSignature */:
+            case 171 /* IndexSignature */:
+            case 253 /* InterfaceDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+                return -2 /* TypeExcludes */;
+            case 200 /* ObjectLiteralExpression */:
+                return 536922112 /* ObjectLiteralExcludes */;
+            case 287 /* CatchClause */:
+                return 536887296 /* CatchClauseExcludes */;
+            case 196 /* ObjectBindingPattern */:
+            case 197 /* ArrayBindingPattern */:
+                return 536879104 /* BindingPatternExcludes */;
+            case 206 /* TypeAssertionExpression */:
+            case 224 /* AsExpression */:
+            case 336 /* PartiallyEmittedExpression */:
+            case 207 /* ParenthesizedExpression */:
+            case 105 /* SuperKeyword */:
+                return 536870912 /* OuterExpressionExcludes */;
+            case 201 /* PropertyAccessExpression */:
+            case 202 /* ElementAccessExpression */:
+                return 536870912 /* PropertyAccessExcludes */;
+            default:
+                return 536870912 /* NodeExcludes */;
         }
-        function parseJsxClosingFragment(inExpressionContext) {
-            var node = createNode(272 /* JsxClosingFragment */);
-            parseExpected(30 /* LessThanSlashToken */);
-            if (ts.tokenIsIdentifierOrKeyword(token())) {
-                parseErrorAtRange(parseJsxElementName(), ts.Diagnostics.Expected_corresponding_closing_tag_for_JSX_fragment);
-            }
-            if (inExpressionContext) {
-                parseExpected(31 /* GreaterThanToken */);
-            }
-            else {
-                parseExpected(31 /* GreaterThanToken */, /*diagnostic*/ undefined, /*shouldAdvance*/ false);
-                scanJsxText();
+    }
+    ts.getTransformFlagsSubtreeExclusions = getTransformFlagsSubtreeExclusions;
+    var baseFactory = ts.createBaseNodeFactory();
+    function makeSynthetic(node) {
+        node.flags |= 8 /* Synthesized */;
+        return node;
+    }
+    var syntheticFactory = {
+        createBaseSourceFileNode: function (kind) { return makeSynthetic(baseFactory.createBaseSourceFileNode(kind)); },
+        createBaseIdentifierNode: function (kind) { return makeSynthetic(baseFactory.createBaseIdentifierNode(kind)); },
+        createBasePrivateIdentifierNode: function (kind) { return makeSynthetic(baseFactory.createBasePrivateIdentifierNode(kind)); },
+        createBaseTokenNode: function (kind) { return makeSynthetic(baseFactory.createBaseTokenNode(kind)); },
+        createBaseNode: function (kind) { return makeSynthetic(baseFactory.createBaseNode(kind)); },
+    };
+    ts.factory = createNodeFactory(4 /* NoIndentationOnFreshPropertyAccess */, syntheticFactory);
+    function createUnparsedSourceFile(textOrInputFiles, mapPathOrType, mapTextOrStripInternal) {
+        var stripInternal;
+        var bundleFileInfo;
+        var fileName;
+        var text;
+        var length;
+        var sourceMapPath;
+        var sourceMapText;
+        var getText;
+        var getSourceMapText;
+        var oldFileOfCurrentEmit;
+        if (!ts.isString(textOrInputFiles)) {
+            ts.Debug.assert(mapPathOrType === "js" || mapPathOrType === "dts");
+            fileName = (mapPathOrType === "js" ? textOrInputFiles.javascriptPath : textOrInputFiles.declarationPath) || "";
+            sourceMapPath = mapPathOrType === "js" ? textOrInputFiles.javascriptMapPath : textOrInputFiles.declarationMapPath;
+            getText = function () { return mapPathOrType === "js" ? textOrInputFiles.javascriptText : textOrInputFiles.declarationText; };
+            getSourceMapText = function () { return mapPathOrType === "js" ? textOrInputFiles.javascriptMapText : textOrInputFiles.declarationMapText; };
+            length = function () { return getText().length; };
+            if (textOrInputFiles.buildInfo && textOrInputFiles.buildInfo.bundle) {
+                ts.Debug.assert(mapTextOrStripInternal === undefined || typeof mapTextOrStripInternal === "boolean");
+                stripInternal = mapTextOrStripInternal;
+                bundleFileInfo = mapPathOrType === "js" ? textOrInputFiles.buildInfo.bundle.js : textOrInputFiles.buildInfo.bundle.dts;
+                oldFileOfCurrentEmit = textOrInputFiles.oldFileOfCurrentEmit;
             }
-            return finishNode(node);
-        }
-        function parseTypeAssertion() {
-            var node = createNode(199 /* TypeAssertionExpression */);
-            parseExpected(29 /* LessThanToken */);
-            node.type = parseType();
-            parseExpected(31 /* GreaterThanToken */);
-            node.expression = parseSimpleUnaryExpression();
-            return finishNode(node);
         }
-        function nextTokenIsIdentifierOrKeywordOrOpenBracketOrTemplate() {
-            nextToken();
-            return ts.tokenIsIdentifierOrKeyword(token())
-                || token() === 22 /* OpenBracketToken */
-                || isTemplateStartOfTaggedTemplate();
+        else {
+            fileName = "";
+            text = textOrInputFiles;
+            length = textOrInputFiles.length;
+            sourceMapPath = mapPathOrType;
+            sourceMapText = mapTextOrStripInternal;
+        }
+        var node = oldFileOfCurrentEmit ?
+            parseOldFileOfCurrentEmit(ts.Debug.assertDefined(bundleFileInfo)) :
+            parseUnparsedSourceFile(bundleFileInfo, stripInternal, length);
+        node.fileName = fileName;
+        node.sourceMapPath = sourceMapPath;
+        node.oldFileOfCurrentEmit = oldFileOfCurrentEmit;
+        if (getText && getSourceMapText) {
+            Object.defineProperty(node, "text", { get: getText });
+            Object.defineProperty(node, "sourceMapText", { get: getSourceMapText });
         }
-        function isStartOfOptionalPropertyOrElementAccessChain() {
-            return token() === 28 /* QuestionDotToken */
-                && lookAhead(nextTokenIsIdentifierOrKeywordOrOpenBracketOrTemplate);
+        else {
+            ts.Debug.assert(!oldFileOfCurrentEmit);
+            node.text = text !== null && text !== void 0 ? text : "";
+            node.sourceMapText = sourceMapText;
         }
-        function tryReparseOptionalChain(node) {
-            if (node.flags & 32 /* OptionalChain */) {
-                return true;
-            }
-            // check for an optional chain in a non-null expression
-            if (ts.isNonNullExpression(node)) {
-                var expr = node.expression;
-                while (ts.isNonNullExpression(expr) && !(expr.flags & 32 /* OptionalChain */)) {
-                    expr = expr.expression;
-                }
-                if (expr.flags & 32 /* OptionalChain */) {
-                    // this is part of an optional chain. Walk down from `node` to `expression` and set the flag.
-                    while (ts.isNonNullExpression(node)) {
-                        node.flags |= 32 /* OptionalChain */;
-                        node = node.expression;
+        return node;
+    }
+    ts.createUnparsedSourceFile = createUnparsedSourceFile;
+    function parseUnparsedSourceFile(bundleFileInfo, stripInternal, length) {
+        var prologues;
+        var helpers;
+        var referencedFiles;
+        var typeReferenceDirectives;
+        var libReferenceDirectives;
+        var prependChildren;
+        var texts;
+        var hasNoDefaultLib;
+        for (var _i = 0, _a = bundleFileInfo ? bundleFileInfo.sections : ts.emptyArray; _i < _a.length; _i++) {
+            var section = _a[_i];
+            switch (section.kind) {
+                case "prologue" /* Prologue */:
+                    prologues = ts.append(prologues, ts.setTextRange(ts.factory.createUnparsedPrologue(section.data), section));
+                    break;
+                case "emitHelpers" /* EmitHelpers */:
+                    helpers = ts.append(helpers, ts.getAllUnscopedEmitHelpers().get(section.data));
+                    break;
+                case "no-default-lib" /* NoDefaultLib */:
+                    hasNoDefaultLib = true;
+                    break;
+                case "reference" /* Reference */:
+                    referencedFiles = ts.append(referencedFiles, { pos: -1, end: -1, fileName: section.data });
+                    break;
+                case "type" /* Type */:
+                    typeReferenceDirectives = ts.append(typeReferenceDirectives, section.data);
+                    break;
+                case "lib" /* Lib */:
+                    libReferenceDirectives = ts.append(libReferenceDirectives, { pos: -1, end: -1, fileName: section.data });
+                    break;
+                case "prepend" /* Prepend */:
+                    var prependTexts = void 0;
+                    for (var _b = 0, _c = section.texts; _b < _c.length; _b++) {
+                        var text = _c[_b];
+                        if (!stripInternal || text.kind !== "internal" /* Internal */) {
+                            prependTexts = ts.append(prependTexts, ts.setTextRange(ts.factory.createUnparsedTextLike(text.data, text.kind === "internal" /* Internal */), text));
+                        }
                     }
-                    return true;
-                }
+                    prependChildren = ts.addRange(prependChildren, prependTexts);
+                    texts = ts.append(texts, ts.factory.createUnparsedPrepend(section.data, prependTexts !== null && prependTexts !== void 0 ? prependTexts : ts.emptyArray));
+                    break;
+                case "internal" /* Internal */:
+                    if (stripInternal) {
+                        if (!texts)
+                            texts = [];
+                        break;
+                    }
+                // falls through
+                case "text" /* Text */:
+                    texts = ts.append(texts, ts.setTextRange(ts.factory.createUnparsedTextLike(section.data, section.kind === "internal" /* Internal */), section));
+                    break;
+                default:
+                    ts.Debug.assertNever(section);
             }
-            return false;
         }
-        function parsePropertyAccessExpressionRest(expression, questionDotToken) {
-            var propertyAccess = createNode(194 /* PropertyAccessExpression */, expression.pos);
-            propertyAccess.expression = expression;
-            propertyAccess.questionDotToken = questionDotToken;
-            propertyAccess.name = parseRightSideOfDot(/*allowIdentifierNames*/ true, /*allowPrivateIdentifiers*/ true);
-            if (questionDotToken || tryReparseOptionalChain(expression)) {
-                propertyAccess.flags |= 32 /* OptionalChain */;
-                if (ts.isPrivateIdentifier(propertyAccess.name)) {
-                    parseErrorAtRange(propertyAccess.name, ts.Diagnostics.An_optional_chain_cannot_contain_private_identifiers);
-                }
-            }
-            return finishNode(propertyAccess);
+        if (!texts) {
+            var textNode = ts.factory.createUnparsedTextLike(/*data*/ undefined, /*internal*/ false);
+            ts.setTextRangePosWidth(textNode, 0, typeof length === "function" ? length() : length);
+            texts = [textNode];
         }
-        function parseElementAccessExpressionRest(expression, questionDotToken) {
-            var indexedAccess = createNode(195 /* ElementAccessExpression */, expression.pos);
-            indexedAccess.expression = expression;
-            indexedAccess.questionDotToken = questionDotToken;
-            if (token() === 23 /* CloseBracketToken */) {
-                indexedAccess.argumentExpression = createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.An_element_access_expression_should_take_an_argument);
-            }
-            else {
-                var argument = allowInAnd(parseExpression);
-                if (ts.isStringOrNumericLiteralLike(argument)) {
-                    argument.text = internIdentifier(argument.text);
-                }
-                indexedAccess.argumentExpression = argument;
-            }
-            parseExpected(23 /* CloseBracketToken */);
-            if (questionDotToken || tryReparseOptionalChain(expression)) {
-                indexedAccess.flags |= 32 /* OptionalChain */;
-            }
-            return finishNode(indexedAccess);
-        }
-        function parseMemberExpressionRest(expression, allowOptionalChain) {
-            while (true) {
-                var questionDotToken = void 0;
-                var isPropertyAccess = false;
-                if (allowOptionalChain && isStartOfOptionalPropertyOrElementAccessChain()) {
-                    questionDotToken = parseExpectedToken(28 /* QuestionDotToken */);
-                    isPropertyAccess = ts.tokenIsIdentifierOrKeyword(token());
-                }
-                else {
-                    isPropertyAccess = parseOptional(24 /* DotToken */);
-                }
-                if (isPropertyAccess) {
-                    expression = parsePropertyAccessExpressionRest(expression, questionDotToken);
-                    continue;
-                }
-                if (!questionDotToken && token() === 53 /* ExclamationToken */ && !scanner.hasPrecedingLineBreak()) {
-                    nextToken();
-                    var nonNullExpression = createNode(218 /* NonNullExpression */, expression.pos);
-                    nonNullExpression.expression = expression;
-                    expression = finishNode(nonNullExpression);
-                    continue;
-                }
-                // when in the [Decorator] context, we do not parse ElementAccess as it could be part of a ComputedPropertyName
-                if ((questionDotToken || !inDecoratorContext()) && parseOptional(22 /* OpenBracketToken */)) {
-                    expression = parseElementAccessExpressionRest(expression, questionDotToken);
-                    continue;
-                }
-                if (isTemplateStartOfTaggedTemplate()) {
-                    expression = parseTaggedTemplateRest(expression, questionDotToken, /*typeArguments*/ undefined);
-                    continue;
-                }
-                return expression;
-            }
-        }
-        function isTemplateStartOfTaggedTemplate() {
-            return token() === 14 /* NoSubstitutionTemplateLiteral */ || token() === 15 /* TemplateHead */;
-        }
-        function parseTaggedTemplateRest(tag, questionDotToken, typeArguments) {
-            var tagExpression = createNode(198 /* TaggedTemplateExpression */, tag.pos);
-            tagExpression.tag = tag;
-            tagExpression.questionDotToken = questionDotToken;
-            tagExpression.typeArguments = typeArguments;
-            tagExpression.template = token() === 14 /* NoSubstitutionTemplateLiteral */
-                ? (reScanTemplateHeadOrNoSubstitutionTemplate(), parseLiteralNode())
-                : parseTemplateExpression(/*isTaggedTemplate*/ true);
-            if (questionDotToken || tag.flags & 32 /* OptionalChain */) {
-                tagExpression.flags |= 32 /* OptionalChain */;
-            }
-            return finishNode(tagExpression);
-        }
-        function parseCallExpressionRest(expression) {
-            while (true) {
-                expression = parseMemberExpressionRest(expression, /*allowOptionalChain*/ true);
-                var questionDotToken = parseOptionalToken(28 /* QuestionDotToken */);
-                // handle 'foo<<T>()'
-                if (token() === 29 /* LessThanToken */ || token() === 47 /* LessThanLessThanToken */) {
-                    // See if this is the start of a generic invocation.  If so, consume it and
-                    // keep checking for postfix expressions.  Otherwise, it's just a '<' that's
-                    // part of an arithmetic expression.  Break out so we consume it higher in the
-                    // stack.
-                    var typeArguments = tryParse(parseTypeArgumentsInExpression);
-                    if (typeArguments) {
-                        if (isTemplateStartOfTaggedTemplate()) {
-                            expression = parseTaggedTemplateRest(expression, questionDotToken, typeArguments);
-                            continue;
-                        }
-                        var callExpr = createNode(196 /* CallExpression */, expression.pos);
-                        callExpr.expression = expression;
-                        callExpr.questionDotToken = questionDotToken;
-                        callExpr.typeArguments = typeArguments;
-                        callExpr.arguments = parseArgumentList();
-                        if (questionDotToken || tryReparseOptionalChain(expression)) {
-                            callExpr.flags |= 32 /* OptionalChain */;
-                        }
-                        expression = finishNode(callExpr);
-                        continue;
-                    }
-                }
-                else if (token() === 20 /* OpenParenToken */) {
-                    var callExpr = createNode(196 /* CallExpression */, expression.pos);
-                    callExpr.expression = expression;
-                    callExpr.questionDotToken = questionDotToken;
-                    callExpr.arguments = parseArgumentList();
-                    if (questionDotToken || tryReparseOptionalChain(expression)) {
-                        callExpr.flags |= 32 /* OptionalChain */;
-                    }
-                    expression = finishNode(callExpr);
-                    continue;
-                }
-                if (questionDotToken) {
-                    // We failed to parse anything, so report a missing identifier here.
-                    var propertyAccess = createNode(194 /* PropertyAccessExpression */, expression.pos);
-                    propertyAccess.expression = expression;
-                    propertyAccess.questionDotToken = questionDotToken;
-                    propertyAccess.name = createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ false, ts.Diagnostics.Identifier_expected);
-                    propertyAccess.flags |= 32 /* OptionalChain */;
-                    expression = finishNode(propertyAccess);
-                }
-                break;
-            }
-            return expression;
-        }
-        function parseArgumentList() {
-            parseExpected(20 /* OpenParenToken */);
-            var result = parseDelimitedList(11 /* ArgumentExpressions */, parseArgumentExpression);
-            parseExpected(21 /* CloseParenToken */);
-            return result;
-        }
-        function parseTypeArgumentsInExpression() {
-            if (reScanLessThanToken() !== 29 /* LessThanToken */) {
-                return undefined;
-            }
-            nextToken();
-            var typeArguments = parseDelimitedList(20 /* TypeArguments */, parseType);
-            if (!parseExpected(31 /* GreaterThanToken */)) {
-                // If it doesn't have the closing `>` then it's definitely not an type argument list.
-                return undefined;
-            }
-            // If we have a '<', then only parse this as a argument list if the type arguments
-            // are complete and we have an open paren.  if we don't, rewind and return nothing.
-            return typeArguments && canFollowTypeArgumentsInExpression()
-                ? typeArguments
-                : undefined;
-        }
-        function canFollowTypeArgumentsInExpression() {
-            switch (token()) {
-                case 20 /* OpenParenToken */: // foo<x>(
-                case 14 /* NoSubstitutionTemplateLiteral */: // foo<T> `...`
-                case 15 /* TemplateHead */: // foo<T> `...${100}...`
-                // these are the only tokens can legally follow a type argument
-                // list. So we definitely want to treat them as type arg lists.
-                // falls through
-                case 24 /* DotToken */: // foo<x>.
-                case 21 /* CloseParenToken */: // foo<x>)
-                case 23 /* CloseBracketToken */: // foo<x>]
-                case 58 /* ColonToken */: // foo<x>:
-                case 26 /* SemicolonToken */: // foo<x>;
-                case 57 /* QuestionToken */: // foo<x>?
-                case 34 /* EqualsEqualsToken */: // foo<x> ==
-                case 36 /* EqualsEqualsEqualsToken */: // foo<x> ===
-                case 35 /* ExclamationEqualsToken */: // foo<x> !=
-                case 37 /* ExclamationEqualsEqualsToken */: // foo<x> !==
-                case 55 /* AmpersandAmpersandToken */: // foo<x> &&
-                case 56 /* BarBarToken */: // foo<x> ||
-                case 60 /* QuestionQuestionToken */: // foo<x> ??
-                case 52 /* CaretToken */: // foo<x> ^
-                case 50 /* AmpersandToken */: // foo<x> &
-                case 51 /* BarToken */: // foo<x> |
-                case 19 /* CloseBraceToken */: // foo<x> }
-                case 1 /* EndOfFileToken */: // foo<x>
-                    // these cases can't legally follow a type arg list.  However, they're not legal
-                    // expressions either.  The user is probably in the middle of a generic type. So
-                    // treat it as such.
-                    return true;
-                case 27 /* CommaToken */: // foo<x>,
-                case 18 /* OpenBraceToken */: // foo<x> {
-                // We don't want to treat these as type arguments.  Otherwise we'll parse this
-                // as an invocation expression.  Instead, we want to parse out the expression
-                // in isolation from the type arguments.
-                // falls through
-                default:
-                    // Anything else treat as an expression.
-                    return false;
-            }
-        }
-        function parsePrimaryExpression() {
-            switch (token()) {
-                case 8 /* NumericLiteral */:
-                case 9 /* BigIntLiteral */:
-                case 10 /* StringLiteral */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                    return parseLiteralNode();
-                case 104 /* ThisKeyword */:
-                case 102 /* SuperKeyword */:
-                case 100 /* NullKeyword */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
-                    return parseTokenNode();
-                case 20 /* OpenParenToken */:
-                    return parseParenthesizedExpression();
-                case 22 /* OpenBracketToken */:
-                    return parseArrayLiteralExpression();
-                case 18 /* OpenBraceToken */:
-                    return parseObjectLiteralExpression();
-                case 126 /* AsyncKeyword */:
-                    // Async arrow functions are parsed earlier in parseAssignmentExpressionOrHigher.
-                    // If we encounter `async [no LineTerminator here] function` then this is an async
-                    // function; otherwise, its an identifier.
-                    if (!lookAhead(nextTokenIsFunctionKeywordOnSameLine)) {
-                        break;
-                    }
-                    return parseFunctionExpression();
-                case 80 /* ClassKeyword */:
-                    return parseClassExpression();
-                case 94 /* FunctionKeyword */:
-                    return parseFunctionExpression();
-                case 99 /* NewKeyword */:
-                    return parseNewExpressionOrNewDotTarget();
-                case 43 /* SlashToken */:
-                case 67 /* SlashEqualsToken */:
-                    if (reScanSlashToken() === 13 /* RegularExpressionLiteral */) {
-                        return parseLiteralNode();
-                    }
+        var node = ts.parseNodeFactory.createUnparsedSource(prologues !== null && prologues !== void 0 ? prologues : ts.emptyArray, /*syntheticReferences*/ undefined, texts);
+        ts.setEachParent(prologues, node);
+        ts.setEachParent(texts, node);
+        ts.setEachParent(prependChildren, node);
+        node.hasNoDefaultLib = hasNoDefaultLib;
+        node.helpers = helpers;
+        node.referencedFiles = referencedFiles || ts.emptyArray;
+        node.typeReferenceDirectives = typeReferenceDirectives;
+        node.libReferenceDirectives = libReferenceDirectives || ts.emptyArray;
+        return node;
+    }
+    function parseOldFileOfCurrentEmit(bundleFileInfo) {
+        var texts;
+        var syntheticReferences;
+        for (var _i = 0, _a = bundleFileInfo.sections; _i < _a.length; _i++) {
+            var section = _a[_i];
+            switch (section.kind) {
+                case "internal" /* Internal */:
+                case "text" /* Text */:
+                    texts = ts.append(texts, ts.setTextRange(ts.factory.createUnparsedTextLike(section.data, section.kind === "internal" /* Internal */), section));
                     break;
-                case 15 /* TemplateHead */:
-                    return parseTemplateExpression(/* isTaggedTemplate */ false);
-            }
-            return parseIdentifier(ts.Diagnostics.Expression_expected);
-        }
-        function parseParenthesizedExpression() {
-            var node = createNodeWithJSDoc(200 /* ParenthesizedExpression */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(21 /* CloseParenToken */);
-            return finishNode(node);
-        }
-        function parseSpreadElement() {
-            var node = createNode(213 /* SpreadElement */);
-            parseExpected(25 /* DotDotDotToken */);
-            node.expression = parseAssignmentExpressionOrHigher();
-            return finishNode(node);
-        }
-        function parseArgumentOrArrayLiteralElement() {
-            return token() === 25 /* DotDotDotToken */ ? parseSpreadElement() :
-                token() === 27 /* CommaToken */ ? createNode(215 /* OmittedExpression */) :
-                    parseAssignmentExpressionOrHigher();
-        }
-        function parseArgumentExpression() {
-            return doOutsideOfContext(disallowInAndDecoratorContext, parseArgumentOrArrayLiteralElement);
-        }
-        function parseArrayLiteralExpression() {
-            var node = createNode(192 /* ArrayLiteralExpression */);
-            parseExpected(22 /* OpenBracketToken */);
-            if (scanner.hasPrecedingLineBreak()) {
-                node.multiLine = true;
+                case "no-default-lib" /* NoDefaultLib */:
+                case "reference" /* Reference */:
+                case "type" /* Type */:
+                case "lib" /* Lib */:
+                    syntheticReferences = ts.append(syntheticReferences, ts.setTextRange(ts.factory.createUnparsedSyntheticReference(section), section));
+                    break;
+                // Ignore
+                case "prologue" /* Prologue */:
+                case "emitHelpers" /* EmitHelpers */:
+                case "prepend" /* Prepend */:
+                    break;
+                default:
+                    ts.Debug.assertNever(section);
             }
-            node.elements = parseDelimitedList(15 /* ArrayLiteralMembers */, parseArgumentOrArrayLiteralElement);
-            parseExpected(23 /* CloseBracketToken */);
-            return finishNode(node);
         }
-        function parseObjectLiteralElement() {
-            var node = createNodeWithJSDoc(0 /* Unknown */);
-            if (parseOptionalToken(25 /* DotDotDotToken */)) {
-                node.kind = 283 /* SpreadAssignment */;
-                node.expression = parseAssignmentExpressionOrHigher();
-                return finishNode(node);
-            }
-            node.decorators = parseDecorators();
-            node.modifiers = parseModifiers();
-            if (parseContextualModifier(131 /* GetKeyword */)) {
-                return parseAccessorDeclaration(node, 163 /* GetAccessor */);
-            }
-            if (parseContextualModifier(142 /* SetKeyword */)) {
-                return parseAccessorDeclaration(node, 164 /* SetAccessor */);
-            }
-            var asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
-            var tokenIsIdentifier = isIdentifier();
-            node.name = parsePropertyName();
-            // Disallowing of optional property assignments and definite assignment assertion happens in the grammar checker.
-            node.questionToken = parseOptionalToken(57 /* QuestionToken */);
-            node.exclamationToken = parseOptionalToken(53 /* ExclamationToken */);
-            if (asteriskToken || token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
-                return parseMethodDeclaration(node, asteriskToken);
-            }
-            // check if it is short-hand property assignment or normal property assignment
-            // NOTE: if token is EqualsToken it is interpreted as CoverInitializedName production
-            // CoverInitializedName[Yield] :
-            //     IdentifierReference[?Yield] Initializer[In, ?Yield]
-            // this is necessary because ObjectLiteral productions are also used to cover grammar for ObjectAssignmentPattern
-            var isShorthandPropertyAssignment = tokenIsIdentifier && (token() !== 58 /* ColonToken */);
-            if (isShorthandPropertyAssignment) {
-                node.kind = 282 /* ShorthandPropertyAssignment */;
-                var equalsToken = parseOptionalToken(62 /* EqualsToken */);
-                if (equalsToken) {
-                    node.equalsToken = equalsToken;
-                    node.objectAssignmentInitializer = allowInAnd(parseAssignmentExpressionOrHigher);
+        var node = ts.factory.createUnparsedSource(ts.emptyArray, syntheticReferences, texts !== null && texts !== void 0 ? texts : ts.emptyArray);
+        ts.setEachParent(syntheticReferences, node);
+        ts.setEachParent(texts, node);
+        node.helpers = ts.map(bundleFileInfo.sources && bundleFileInfo.sources.helpers, function (name) { return ts.getAllUnscopedEmitHelpers().get(name); });
+        return node;
+    }
+    function createInputFiles(javascriptTextOrReadFileText, declarationTextOrJavascriptPath, javascriptMapPath, javascriptMapTextOrDeclarationPath, declarationMapPath, declarationMapTextOrBuildInfoPath, javascriptPath, declarationPath, buildInfoPath, buildInfo, oldFileOfCurrentEmit) {
+        var node = ts.parseNodeFactory.createInputFiles();
+        if (!ts.isString(javascriptTextOrReadFileText)) {
+            var cache_1 = new ts.Map();
+            var textGetter_1 = function (path) {
+                if (path === undefined)
+                    return undefined;
+                var value = cache_1.get(path);
+                if (value === undefined) {
+                    value = javascriptTextOrReadFileText(path);
+                    cache_1.set(path, value !== undefined ? value : false);
                 }
-            }
-            else {
-                node.kind = 281 /* PropertyAssignment */;
-                parseExpected(58 /* ColonToken */);
-                node.initializer = allowInAnd(parseAssignmentExpressionOrHigher);
-            }
-            return finishNode(node);
-        }
-        function parseObjectLiteralExpression() {
-            var node = createNode(193 /* ObjectLiteralExpression */);
-            var openBracePosition = scanner.getTokenPos();
-            parseExpected(18 /* OpenBraceToken */);
-            if (scanner.hasPrecedingLineBreak()) {
-                node.multiLine = true;
-            }
-            node.properties = parseDelimitedList(12 /* ObjectLiteralMembers */, parseObjectLiteralElement, /*considerSemicolonAsDelimiter*/ true);
-            if (!parseExpected(19 /* CloseBraceToken */)) {
-                var lastError = ts.lastOrUndefined(parseDiagnostics);
-                if (lastError && lastError.code === ts.Diagnostics._0_expected.code) {
-                    ts.addRelatedInfo(lastError, ts.createFileDiagnostic(sourceFile, openBracePosition, 1, ts.Diagnostics.The_parser_expected_to_find_a_to_match_the_token_here));
+                return value !== false ? value : undefined;
+            };
+            var definedTextGetter_1 = function (path) {
+                var result = textGetter_1(path);
+                return result !== undefined ? result : "/* Input file " + path + " was missing */\r\n";
+            };
+            var buildInfo_1;
+            var getAndCacheBuildInfo_1 = function (getText) {
+                if (buildInfo_1 === undefined) {
+                    var result = getText();
+                    buildInfo_1 = result !== undefined ? ts.getBuildInfo(result) : false;
                 }
-            }
-            return finishNode(node);
-        }
-        function parseFunctionExpression() {
-            // GeneratorExpression:
-            //      function* BindingIdentifier [Yield][opt](FormalParameters[Yield]){ GeneratorBody }
-            //
-            // FunctionExpression:
-            //      function BindingIdentifier[opt](FormalParameters){ FunctionBody }
-            var saveDecoratorContext = inDecoratorContext();
-            if (saveDecoratorContext) {
-                setDecoratorContext(/*val*/ false);
-            }
-            var node = createNodeWithJSDoc(201 /* FunctionExpression */);
-            node.modifiers = parseModifiers();
-            parseExpected(94 /* FunctionKeyword */);
-            node.asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
-            var isGenerator = node.asteriskToken ? 1 /* Yield */ : 0 /* None */;
-            var isAsync = hasModifierOfKind(node, 126 /* AsyncKeyword */) ? 2 /* Await */ : 0 /* None */;
-            node.name =
-                isGenerator && isAsync ? doInYieldAndAwaitContext(parseOptionalIdentifier) :
-                    isGenerator ? doInYieldContext(parseOptionalIdentifier) :
-                        isAsync ? doInAwaitContext(parseOptionalIdentifier) :
-                            parseOptionalIdentifier();
-            fillSignature(58 /* ColonToken */, isGenerator | isAsync, node);
-            node.body = parseFunctionBlock(isGenerator | isAsync);
-            if (saveDecoratorContext) {
-                setDecoratorContext(/*val*/ true);
-            }
-            return finishNode(node);
-        }
-        function parseOptionalIdentifier() {
-            return isIdentifier() ? parseIdentifier() : undefined;
+                return buildInfo_1 || undefined;
+            };
+            node.javascriptPath = declarationTextOrJavascriptPath;
+            node.javascriptMapPath = javascriptMapPath;
+            node.declarationPath = ts.Debug.assertDefined(javascriptMapTextOrDeclarationPath);
+            node.declarationMapPath = declarationMapPath;
+            node.buildInfoPath = declarationMapTextOrBuildInfoPath;
+            Object.defineProperties(node, {
+                javascriptText: { get: function () { return definedTextGetter_1(declarationTextOrJavascriptPath); } },
+                javascriptMapText: { get: function () { return textGetter_1(javascriptMapPath); } },
+                declarationText: { get: function () { return definedTextGetter_1(ts.Debug.assertDefined(javascriptMapTextOrDeclarationPath)); } },
+                declarationMapText: { get: function () { return textGetter_1(declarationMapPath); } },
+                buildInfo: { get: function () { return getAndCacheBuildInfo_1(function () { return textGetter_1(declarationMapTextOrBuildInfoPath); }); } }
+            });
         }
-        function parseNewExpressionOrNewDotTarget() {
-            var fullStart = scanner.getStartPos();
-            parseExpected(99 /* NewKeyword */);
-            if (parseOptional(24 /* DotToken */)) {
-                var node_2 = createNode(219 /* MetaProperty */, fullStart);
-                node_2.keywordToken = 99 /* NewKeyword */;
-                node_2.name = parseIdentifierName();
-                return finishNode(node_2);
-            }
-            var expression = parsePrimaryExpression();
-            var typeArguments;
-            while (true) {
-                expression = parseMemberExpressionRest(expression, /*allowOptionalChain*/ false);
-                typeArguments = tryParse(parseTypeArgumentsInExpression);
-                if (isTemplateStartOfTaggedTemplate()) {
-                    ts.Debug.assert(!!typeArguments, "Expected a type argument list; all plain tagged template starts should be consumed in 'parseMemberExpressionRest'");
-                    expression = parseTaggedTemplateRest(expression, /*optionalChain*/ undefined, typeArguments);
-                    typeArguments = undefined;
-                }
-                break;
-            }
-            var node = createNode(197 /* NewExpression */, fullStart);
-            node.expression = expression;
-            node.typeArguments = typeArguments;
-            if (token() === 20 /* OpenParenToken */) {
-                node.arguments = parseArgumentList();
-            }
-            else if (node.typeArguments) {
-                parseErrorAt(fullStart, scanner.getStartPos(), ts.Diagnostics.A_new_expression_with_type_arguments_must_always_be_followed_by_a_parenthesized_argument_list);
-            }
-            return finishNode(node);
+        else {
+            node.javascriptText = javascriptTextOrReadFileText;
+            node.javascriptMapPath = javascriptMapPath;
+            node.javascriptMapText = javascriptMapTextOrDeclarationPath;
+            node.declarationText = declarationTextOrJavascriptPath;
+            node.declarationMapPath = declarationMapPath;
+            node.declarationMapText = declarationMapTextOrBuildInfoPath;
+            node.javascriptPath = javascriptPath;
+            node.declarationPath = declarationPath;
+            node.buildInfoPath = buildInfoPath;
+            node.buildInfo = buildInfo;
+            node.oldFileOfCurrentEmit = oldFileOfCurrentEmit;
         }
-        // STATEMENTS
-        function parseBlock(ignoreMissingOpenBrace, diagnosticMessage) {
-            var node = createNode(223 /* Block */);
-            var openBracePosition = scanner.getTokenPos();
-            if (parseExpected(18 /* OpenBraceToken */, diagnosticMessage) || ignoreMissingOpenBrace) {
-                if (scanner.hasPrecedingLineBreak()) {
-                    node.multiLine = true;
-                }
-                node.statements = parseList(1 /* BlockStatements */, parseStatement);
-                if (!parseExpected(19 /* CloseBraceToken */)) {
-                    var lastError = ts.lastOrUndefined(parseDiagnostics);
-                    if (lastError && lastError.code === ts.Diagnostics._0_expected.code) {
-                        ts.addRelatedInfo(lastError, ts.createFileDiagnostic(sourceFile, openBracePosition, 1, ts.Diagnostics.The_parser_expected_to_find_a_to_match_the_token_here));
-                    }
-                }
-            }
-            else {
-                node.statements = createMissingList();
-            }
-            return finishNode(node);
+        return node;
+    }
+    ts.createInputFiles = createInputFiles;
+    // tslint:disable-next-line variable-name
+    var SourceMapSource;
+    /**
+     * Create an external source map source file reference
+     */
+    function createSourceMapSource(fileName, text, skipTrivia) {
+        return new (SourceMapSource || (SourceMapSource = ts.objectAllocator.getSourceMapSourceConstructor()))(fileName, text, skipTrivia);
+    }
+    ts.createSourceMapSource = createSourceMapSource;
+    // Utilities
+    function setOriginalNode(node, original) {
+        node.original = original;
+        if (original) {
+            var emitNode = original.emitNode;
+            if (emitNode)
+                node.emitNode = mergeEmitNode(emitNode, node.emitNode);
         }
-        function parseFunctionBlock(flags, diagnosticMessage) {
-            var savedYieldContext = inYieldContext();
-            setYieldContext(!!(flags & 1 /* Yield */));
-            var savedAwaitContext = inAwaitContext();
-            setAwaitContext(!!(flags & 2 /* Await */));
-            // We may be in a [Decorator] context when parsing a function expression or
-            // arrow function. The body of the function is not in [Decorator] context.
-            var saveDecoratorContext = inDecoratorContext();
-            if (saveDecoratorContext) {
-                setDecoratorContext(/*val*/ false);
-            }
-            var block = parseBlock(!!(flags & 16 /* IgnoreMissingOpenBrace */), diagnosticMessage);
-            if (saveDecoratorContext) {
-                setDecoratorContext(/*val*/ true);
+        return node;
+    }
+    ts.setOriginalNode = setOriginalNode;
+    function mergeEmitNode(sourceEmitNode, destEmitNode) {
+        var flags = sourceEmitNode.flags, leadingComments = sourceEmitNode.leadingComments, trailingComments = sourceEmitNode.trailingComments, commentRange = sourceEmitNode.commentRange, sourceMapRange = sourceEmitNode.sourceMapRange, tokenSourceMapRanges = sourceEmitNode.tokenSourceMapRanges, constantValue = sourceEmitNode.constantValue, helpers = sourceEmitNode.helpers, startsOnNewLine = sourceEmitNode.startsOnNewLine;
+        if (!destEmitNode)
+            destEmitNode = {};
+        // We are using `.slice()` here in case `destEmitNode.leadingComments` is pushed to later.
+        if (leadingComments)
+            destEmitNode.leadingComments = ts.addRange(leadingComments.slice(), destEmitNode.leadingComments);
+        if (trailingComments)
+            destEmitNode.trailingComments = ts.addRange(trailingComments.slice(), destEmitNode.trailingComments);
+        if (flags)
+            destEmitNode.flags = flags;
+        if (commentRange)
+            destEmitNode.commentRange = commentRange;
+        if (sourceMapRange)
+            destEmitNode.sourceMapRange = sourceMapRange;
+        if (tokenSourceMapRanges)
+            destEmitNode.tokenSourceMapRanges = mergeTokenSourceMapRanges(tokenSourceMapRanges, destEmitNode.tokenSourceMapRanges);
+        if (constantValue !== undefined)
+            destEmitNode.constantValue = constantValue;
+        if (helpers) {
+            for (var _i = 0, helpers_1 = helpers; _i < helpers_1.length; _i++) {
+                var helper = helpers_1[_i];
+                destEmitNode.helpers = ts.appendIfUnique(destEmitNode.helpers, helper);
             }
-            setYieldContext(savedYieldContext);
-            setAwaitContext(savedAwaitContext);
-            return block;
-        }
-        function parseEmptyStatement() {
-            var node = createNode(224 /* EmptyStatement */);
-            parseExpected(26 /* SemicolonToken */);
-            return finishNode(node);
-        }
-        function parseIfStatement() {
-            var node = createNode(227 /* IfStatement */);
-            parseExpected(95 /* IfKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(21 /* CloseParenToken */);
-            node.thenStatement = parseStatement();
-            node.elseStatement = parseOptional(87 /* ElseKeyword */) ? parseStatement() : undefined;
-            return finishNode(node);
-        }
-        function parseDoStatement() {
-            var node = createNode(228 /* DoStatement */);
-            parseExpected(86 /* DoKeyword */);
-            node.statement = parseStatement();
-            parseExpected(111 /* WhileKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(21 /* CloseParenToken */);
-            // From: https://mail.mozilla.org/pipermail/es-discuss/2011-August/016188.html
-            // 157 min --- All allen at wirfs-brock.com CONF --- "do{;}while(false)false" prohibited in
-            // spec but allowed in consensus reality. Approved -- this is the de-facto standard whereby
-            //  do;while(0)x will have a semicolon inserted before x.
-            parseOptional(26 /* SemicolonToken */);
-            return finishNode(node);
         }
-        function parseWhileStatement() {
-            var node = createNode(229 /* WhileStatement */);
-            parseExpected(111 /* WhileKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(21 /* CloseParenToken */);
-            node.statement = parseStatement();
-            return finishNode(node);
+        if (startsOnNewLine !== undefined)
+            destEmitNode.startsOnNewLine = startsOnNewLine;
+        return destEmitNode;
+    }
+    function mergeTokenSourceMapRanges(sourceRanges, destRanges) {
+        if (!destRanges)
+            destRanges = [];
+        for (var key in sourceRanges) {
+            destRanges[key] = sourceRanges[key];
         }
-        function parseForOrForInOrForOfStatement() {
-            var pos = getNodePos();
-            parseExpected(93 /* ForKeyword */);
-            var awaitToken = parseOptionalToken(127 /* AwaitKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            var initializer;
-            if (token() !== 26 /* SemicolonToken */) {
-                if (token() === 109 /* VarKeyword */ || token() === 115 /* LetKeyword */ || token() === 81 /* ConstKeyword */) {
-                    initializer = parseVariableDeclarationList(/*inForStatementInitializer*/ true);
-                }
-                else {
-                    initializer = disallowInAnd(parseExpression);
-                }
-            }
-            var forOrForInOrForOfStatement;
-            if (awaitToken ? parseExpected(152 /* OfKeyword */) : parseOptional(152 /* OfKeyword */)) {
-                var forOfStatement = createNode(232 /* ForOfStatement */, pos);
-                forOfStatement.awaitModifier = awaitToken;
-                forOfStatement.initializer = initializer;
-                forOfStatement.expression = allowInAnd(parseAssignmentExpressionOrHigher);
-                parseExpected(21 /* CloseParenToken */);
-                forOrForInOrForOfStatement = forOfStatement;
-            }
-            else if (parseOptional(97 /* InKeyword */)) {
-                var forInStatement = createNode(231 /* ForInStatement */, pos);
-                forInStatement.initializer = initializer;
-                forInStatement.expression = allowInAnd(parseExpression);
-                parseExpected(21 /* CloseParenToken */);
-                forOrForInOrForOfStatement = forInStatement;
-            }
-            else {
-                var forStatement = createNode(230 /* ForStatement */, pos);
-                forStatement.initializer = initializer;
-                parseExpected(26 /* SemicolonToken */);
-                if (token() !== 26 /* SemicolonToken */ && token() !== 21 /* CloseParenToken */) {
-                    forStatement.condition = allowInAnd(parseExpression);
-                }
-                parseExpected(26 /* SemicolonToken */);
-                if (token() !== 21 /* CloseParenToken */) {
-                    forStatement.incrementor = allowInAnd(parseExpression);
+        return destRanges;
+    }
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    /**
+     * Associates a node with the current transformation, initializing
+     * various transient transformation properties.
+     * @internal
+     */
+    function getOrCreateEmitNode(node) {
+        var _a;
+        if (!node.emitNode) {
+            if (ts.isParseTreeNode(node)) {
+                // To avoid holding onto transformation artifacts, we keep track of any
+                // parse tree node we are annotating. This allows us to clean them up after
+                // all transformations have completed.
+                if (node.kind === 297 /* SourceFile */) {
+                    return node.emitNode = { annotatedNodes: [node] };
                 }
-                parseExpected(21 /* CloseParenToken */);
-                forOrForInOrForOfStatement = forStatement;
-            }
-            forOrForInOrForOfStatement.statement = parseStatement();
-            return finishNode(forOrForInOrForOfStatement);
-        }
-        function parseBreakOrContinueStatement(kind) {
-            var node = createNode(kind);
-            parseExpected(kind === 234 /* BreakStatement */ ? 77 /* BreakKeyword */ : 82 /* ContinueKeyword */);
-            if (!canParseSemicolon()) {
-                node.label = parseIdentifier();
-            }
-            parseSemicolon();
-            return finishNode(node);
-        }
-        function parseReturnStatement() {
-            var node = createNode(235 /* ReturnStatement */);
-            parseExpected(101 /* ReturnKeyword */);
-            if (!canParseSemicolon()) {
-                node.expression = allowInAnd(parseExpression);
+                var sourceFile = (_a = ts.getSourceFileOfNode(ts.getParseTreeNode(ts.getSourceFileOfNode(node)))) !== null && _a !== void 0 ? _a : ts.Debug.fail("Could not determine parsed source file.");
+                getOrCreateEmitNode(sourceFile).annotatedNodes.push(node);
             }
-            parseSemicolon();
-            return finishNode(node);
-        }
-        function parseWithStatement() {
-            var node = createNode(236 /* WithStatement */);
-            parseExpected(112 /* WithKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(21 /* CloseParenToken */);
-            node.statement = doInsideOfContext(16777216 /* InWithStatement */, parseStatement);
-            return finishNode(node);
-        }
-        function parseCaseClause() {
-            var node = createNode(277 /* CaseClause */);
-            parseExpected(78 /* CaseKeyword */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(58 /* ColonToken */);
-            node.statements = parseList(3 /* SwitchClauseStatements */, parseStatement);
-            return finishNode(node);
-        }
-        function parseDefaultClause() {
-            var node = createNode(278 /* DefaultClause */);
-            parseExpected(84 /* DefaultKeyword */);
-            parseExpected(58 /* ColonToken */);
-            node.statements = parseList(3 /* SwitchClauseStatements */, parseStatement);
-            return finishNode(node);
-        }
-        function parseCaseOrDefaultClause() {
-            return token() === 78 /* CaseKeyword */ ? parseCaseClause() : parseDefaultClause();
-        }
-        function parseSwitchStatement() {
-            var node = createNode(237 /* SwitchStatement */);
-            parseExpected(103 /* SwitchKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = allowInAnd(parseExpression);
-            parseExpected(21 /* CloseParenToken */);
-            var caseBlock = createNode(251 /* CaseBlock */);
-            parseExpected(18 /* OpenBraceToken */);
-            caseBlock.clauses = parseList(2 /* SwitchClauses */, parseCaseOrDefaultClause);
-            parseExpected(19 /* CloseBraceToken */);
-            node.caseBlock = finishNode(caseBlock);
-            return finishNode(node);
-        }
-        function parseThrowStatement() {
-            // ThrowStatement[Yield] :
-            //      throw [no LineTerminator here]Expression[In, ?Yield];
-            // Because of automatic semicolon insertion, we need to report error if this
-            // throw could be terminated with a semicolon.  Note: we can't call 'parseExpression'
-            // directly as that might consume an expression on the following line.
-            // We just return 'undefined' in that case.  The actual error will be reported in the
-            // grammar walker.
-            var node = createNode(239 /* ThrowStatement */);
-            parseExpected(105 /* ThrowKeyword */);
-            node.expression = scanner.hasPrecedingLineBreak() ? undefined : allowInAnd(parseExpression);
-            parseSemicolon();
-            return finishNode(node);
+            node.emitNode = {};
         }
-        // TODO: Review for error recovery
-        function parseTryStatement() {
-            var node = createNode(240 /* TryStatement */);
-            parseExpected(107 /* TryKeyword */);
-            node.tryBlock = parseBlock(/*ignoreMissingOpenBrace*/ false);
-            node.catchClause = token() === 79 /* CatchKeyword */ ? parseCatchClause() : undefined;
-            // If we don't have a catch clause, then we must have a finally clause.  Try to parse
-            // one out no matter what.
-            if (!node.catchClause || token() === 92 /* FinallyKeyword */) {
-                parseExpected(92 /* FinallyKeyword */);
-                node.finallyBlock = parseBlock(/*ignoreMissingOpenBrace*/ false);
+        return node.emitNode;
+    }
+    ts.getOrCreateEmitNode = getOrCreateEmitNode;
+    /**
+     * Clears any `EmitNode` entries from parse-tree nodes.
+     * @param sourceFile A source file.
+     */
+    function disposeEmitNodes(sourceFile) {
+        var _a, _b;
+        // During transformation we may need to annotate a parse tree node with transient
+        // transformation properties. As parse tree nodes live longer than transformation
+        // nodes, we need to make sure we reclaim any memory allocated for custom ranges
+        // from these nodes to ensure we do not hold onto entire subtrees just for position
+        // information. We also need to reset these nodes to a pre-transformation state
+        // for incremental parsing scenarios so that we do not impact later emit.
+        var annotatedNodes = (_b = (_a = ts.getSourceFileOfNode(ts.getParseTreeNode(sourceFile))) === null || _a === void 0 ? void 0 : _a.emitNode) === null || _b === void 0 ? void 0 : _b.annotatedNodes;
+        if (annotatedNodes) {
+            for (var _i = 0, annotatedNodes_1 = annotatedNodes; _i < annotatedNodes_1.length; _i++) {
+                var node = annotatedNodes_1[_i];
+                node.emitNode = undefined;
             }
-            return finishNode(node);
         }
-        function parseCatchClause() {
-            var result = createNode(280 /* CatchClause */);
-            parseExpected(79 /* CatchKeyword */);
-            if (parseOptional(20 /* OpenParenToken */)) {
-                result.variableDeclaration = parseVariableDeclaration();
-                parseExpected(21 /* CloseParenToken */);
-            }
-            else {
-                // Keep shape of node to avoid degrading performance.
-                result.variableDeclaration = undefined;
+    }
+    ts.disposeEmitNodes = disposeEmitNodes;
+    /**
+     * Sets `EmitFlags.NoComments` on a node and removes any leading and trailing synthetic comments.
+     * @internal
+     */
+    function removeAllComments(node) {
+        var emitNode = getOrCreateEmitNode(node);
+        emitNode.flags |= 1536 /* NoComments */;
+        emitNode.leadingComments = undefined;
+        emitNode.trailingComments = undefined;
+        return node;
+    }
+    ts.removeAllComments = removeAllComments;
+    /**
+     * Sets flags that control emit behavior of a node.
+     */
+    function setEmitFlags(node, emitFlags) {
+        getOrCreateEmitNode(node).flags = emitFlags;
+        return node;
+    }
+    ts.setEmitFlags = setEmitFlags;
+    /**
+     * Sets flags that control emit behavior of a node.
+     */
+    /* @internal */
+    function addEmitFlags(node, emitFlags) {
+        var emitNode = getOrCreateEmitNode(node);
+        emitNode.flags = emitNode.flags | emitFlags;
+        return node;
+    }
+    ts.addEmitFlags = addEmitFlags;
+    /**
+     * Gets a custom text range to use when emitting source maps.
+     */
+    function getSourceMapRange(node) {
+        var _a, _b;
+        return (_b = (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.sourceMapRange) !== null && _b !== void 0 ? _b : node;
+    }
+    ts.getSourceMapRange = getSourceMapRange;
+    /**
+     * Sets a custom text range to use when emitting source maps.
+     */
+    function setSourceMapRange(node, range) {
+        getOrCreateEmitNode(node).sourceMapRange = range;
+        return node;
+    }
+    ts.setSourceMapRange = setSourceMapRange;
+    /**
+     * Gets the TextRange to use for source maps for a token of a node.
+     */
+    function getTokenSourceMapRange(node, token) {
+        var _a, _b;
+        return (_b = (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.tokenSourceMapRanges) === null || _b === void 0 ? void 0 : _b[token];
+    }
+    ts.getTokenSourceMapRange = getTokenSourceMapRange;
+    /**
+     * Sets the TextRange to use for source maps for a token of a node.
+     */
+    function setTokenSourceMapRange(node, token, range) {
+        var _a;
+        var emitNode = getOrCreateEmitNode(node);
+        var tokenSourceMapRanges = (_a = emitNode.tokenSourceMapRanges) !== null && _a !== void 0 ? _a : (emitNode.tokenSourceMapRanges = []);
+        tokenSourceMapRanges[token] = range;
+        return node;
+    }
+    ts.setTokenSourceMapRange = setTokenSourceMapRange;
+    /**
+     * Gets a custom text range to use when emitting comments.
+     */
+    /*@internal*/
+    function getStartsOnNewLine(node) {
+        var _a;
+        return (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.startsOnNewLine;
+    }
+    ts.getStartsOnNewLine = getStartsOnNewLine;
+    /**
+     * Sets a custom text range to use when emitting comments.
+     */
+    /*@internal*/
+    function setStartsOnNewLine(node, newLine) {
+        getOrCreateEmitNode(node).startsOnNewLine = newLine;
+        return node;
+    }
+    ts.setStartsOnNewLine = setStartsOnNewLine;
+    /**
+     * Gets a custom text range to use when emitting comments.
+     */
+    function getCommentRange(node) {
+        var _a, _b;
+        return (_b = (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.commentRange) !== null && _b !== void 0 ? _b : node;
+    }
+    ts.getCommentRange = getCommentRange;
+    /**
+     * Sets a custom text range to use when emitting comments.
+     */
+    function setCommentRange(node, range) {
+        getOrCreateEmitNode(node).commentRange = range;
+        return node;
+    }
+    ts.setCommentRange = setCommentRange;
+    function getSyntheticLeadingComments(node) {
+        var _a;
+        return (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.leadingComments;
+    }
+    ts.getSyntheticLeadingComments = getSyntheticLeadingComments;
+    function setSyntheticLeadingComments(node, comments) {
+        getOrCreateEmitNode(node).leadingComments = comments;
+        return node;
+    }
+    ts.setSyntheticLeadingComments = setSyntheticLeadingComments;
+    function addSyntheticLeadingComment(node, kind, text, hasTrailingNewLine) {
+        return setSyntheticLeadingComments(node, ts.append(getSyntheticLeadingComments(node), { kind: kind, pos: -1, end: -1, hasTrailingNewLine: hasTrailingNewLine, text: text }));
+    }
+    ts.addSyntheticLeadingComment = addSyntheticLeadingComment;
+    function getSyntheticTrailingComments(node) {
+        var _a;
+        return (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.trailingComments;
+    }
+    ts.getSyntheticTrailingComments = getSyntheticTrailingComments;
+    function setSyntheticTrailingComments(node, comments) {
+        getOrCreateEmitNode(node).trailingComments = comments;
+        return node;
+    }
+    ts.setSyntheticTrailingComments = setSyntheticTrailingComments;
+    function addSyntheticTrailingComment(node, kind, text, hasTrailingNewLine) {
+        return setSyntheticTrailingComments(node, ts.append(getSyntheticTrailingComments(node), { kind: kind, pos: -1, end: -1, hasTrailingNewLine: hasTrailingNewLine, text: text }));
+    }
+    ts.addSyntheticTrailingComment = addSyntheticTrailingComment;
+    function moveSyntheticComments(node, original) {
+        setSyntheticLeadingComments(node, getSyntheticLeadingComments(original));
+        setSyntheticTrailingComments(node, getSyntheticTrailingComments(original));
+        var emit = getOrCreateEmitNode(original);
+        emit.leadingComments = undefined;
+        emit.trailingComments = undefined;
+        return node;
+    }
+    ts.moveSyntheticComments = moveSyntheticComments;
+    /**
+     * Gets the constant value to emit for an expression representing an enum.
+     */
+    function getConstantValue(node) {
+        var _a;
+        return (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.constantValue;
+    }
+    ts.getConstantValue = getConstantValue;
+    /**
+     * Sets the constant value to emit for an expression.
+     */
+    function setConstantValue(node, value) {
+        var emitNode = getOrCreateEmitNode(node);
+        emitNode.constantValue = value;
+        return node;
+    }
+    ts.setConstantValue = setConstantValue;
+    /**
+     * Adds an EmitHelper to a node.
+     */
+    function addEmitHelper(node, helper) {
+        var emitNode = getOrCreateEmitNode(node);
+        emitNode.helpers = ts.append(emitNode.helpers, helper);
+        return node;
+    }
+    ts.addEmitHelper = addEmitHelper;
+    /**
+     * Add EmitHelpers to a node.
+     */
+    function addEmitHelpers(node, helpers) {
+        if (ts.some(helpers)) {
+            var emitNode = getOrCreateEmitNode(node);
+            for (var _i = 0, helpers_2 = helpers; _i < helpers_2.length; _i++) {
+                var helper = helpers_2[_i];
+                emitNode.helpers = ts.appendIfUnique(emitNode.helpers, helper);
             }
-            result.block = parseBlock(/*ignoreMissingOpenBrace*/ false);
-            return finishNode(result);
         }
-        function parseDebuggerStatement() {
-            var node = createNode(241 /* DebuggerStatement */);
-            parseExpected(83 /* DebuggerKeyword */);
-            parseSemicolon();
-            return finishNode(node);
+        return node;
+    }
+    ts.addEmitHelpers = addEmitHelpers;
+    /**
+     * Removes an EmitHelper from a node.
+     */
+    function removeEmitHelper(node, helper) {
+        var _a;
+        var helpers = (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.helpers;
+        if (helpers) {
+            return ts.orderedRemoveItem(helpers, helper);
         }
-        function parseExpressionOrLabeledStatement() {
-            // Avoiding having to do the lookahead for a labeled statement by just trying to parse
-            // out an expression, seeing if it is identifier and then seeing if it is followed by
-            // a colon.
-            var node = createNodeWithJSDoc(token() === 75 /* Identifier */ ? 0 /* Unknown */ : 226 /* ExpressionStatement */);
-            var expression = allowInAnd(parseExpression);
-            if (expression.kind === 75 /* Identifier */ && parseOptional(58 /* ColonToken */)) {
-                node.kind = 238 /* LabeledStatement */;
-                node.label = expression;
-                node.statement = parseStatement();
+        return false;
+    }
+    ts.removeEmitHelper = removeEmitHelper;
+    /**
+     * Gets the EmitHelpers of a node.
+     */
+    function getEmitHelpers(node) {
+        var _a;
+        return (_a = node.emitNode) === null || _a === void 0 ? void 0 : _a.helpers;
+    }
+    ts.getEmitHelpers = getEmitHelpers;
+    /**
+     * Moves matching emit helpers from a source node to a target node.
+     */
+    function moveEmitHelpers(source, target, predicate) {
+        var sourceEmitNode = source.emitNode;
+        var sourceEmitHelpers = sourceEmitNode && sourceEmitNode.helpers;
+        if (!ts.some(sourceEmitHelpers))
+            return;
+        var targetEmitNode = getOrCreateEmitNode(target);
+        var helpersRemoved = 0;
+        for (var i = 0; i < sourceEmitHelpers.length; i++) {
+            var helper = sourceEmitHelpers[i];
+            if (predicate(helper)) {
+                helpersRemoved++;
+                targetEmitNode.helpers = ts.appendIfUnique(targetEmitNode.helpers, helper);
             }
-            else {
-                node.kind = 226 /* ExpressionStatement */;
-                node.expression = expression;
-                parseSemicolon();
+            else if (helpersRemoved > 0) {
+                sourceEmitHelpers[i - helpersRemoved] = helper;
             }
-            return finishNode(node);
-        }
-        function nextTokenIsIdentifierOrKeywordOnSameLine() {
-            nextToken();
-            return ts.tokenIsIdentifierOrKeyword(token()) && !scanner.hasPrecedingLineBreak();
         }
-        function nextTokenIsClassKeywordOnSameLine() {
-            nextToken();
-            return token() === 80 /* ClassKeyword */ && !scanner.hasPrecedingLineBreak();
-        }
-        function nextTokenIsFunctionKeywordOnSameLine() {
-            nextToken();
-            return token() === 94 /* FunctionKeyword */ && !scanner.hasPrecedingLineBreak();
+        if (helpersRemoved > 0) {
+            sourceEmitHelpers.length -= helpersRemoved;
         }
-        function nextTokenIsIdentifierOrKeywordOrLiteralOnSameLine() {
-            nextToken();
-            return (ts.tokenIsIdentifierOrKeyword(token()) || token() === 8 /* NumericLiteral */ || token() === 9 /* BigIntLiteral */ || token() === 10 /* StringLiteral */) && !scanner.hasPrecedingLineBreak();
+    }
+    ts.moveEmitHelpers = moveEmitHelpers;
+    /* @internal */
+    function ignoreSourceNewlines(node) {
+        getOrCreateEmitNode(node).flags |= 134217728 /* IgnoreSourceNewlines */;
+        return node;
+    }
+    ts.ignoreSourceNewlines = ignoreSourceNewlines;
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    function createEmitHelperFactory(context) {
+        var factory = context.factory;
+        return {
+            getUnscopedHelperName: getUnscopedHelperName,
+            // TypeScript Helpers
+            createDecorateHelper: createDecorateHelper,
+            createMetadataHelper: createMetadataHelper,
+            createParamHelper: createParamHelper,
+            // ES2018 Helpers
+            createAssignHelper: createAssignHelper,
+            createAwaitHelper: createAwaitHelper,
+            createAsyncGeneratorHelper: createAsyncGeneratorHelper,
+            createAsyncDelegatorHelper: createAsyncDelegatorHelper,
+            createAsyncValuesHelper: createAsyncValuesHelper,
+            // ES2018 Destructuring Helpers
+            createRestHelper: createRestHelper,
+            // ES2017 Helpers
+            createAwaiterHelper: createAwaiterHelper,
+            // ES2015 Helpers
+            createExtendsHelper: createExtendsHelper,
+            createTemplateObjectHelper: createTemplateObjectHelper,
+            createSpreadHelper: createSpreadHelper,
+            createSpreadArraysHelper: createSpreadArraysHelper,
+            // ES2015 Destructuring Helpers
+            createValuesHelper: createValuesHelper,
+            createReadHelper: createReadHelper,
+            // ES2015 Generator Helpers
+            createGeneratorHelper: createGeneratorHelper,
+            // ES Module Helpers
+            createCreateBindingHelper: createCreateBindingHelper,
+            createImportStarHelper: createImportStarHelper,
+            createImportStarCallbackHelper: createImportStarCallbackHelper,
+            createImportDefaultHelper: createImportDefaultHelper,
+            createExportStarHelper: createExportStarHelper,
+            // Class Fields Helpers
+            createClassPrivateFieldGetHelper: createClassPrivateFieldGetHelper,
+            createClassPrivateFieldSetHelper: createClassPrivateFieldSetHelper,
+        };
+        /**
+         * Gets an identifier for the name of an *unscoped* emit helper.
+         */
+        function getUnscopedHelperName(name) {
+            return ts.setEmitFlags(factory.createIdentifier(name), 4096 /* HelperName */ | 2 /* AdviseOnEmitNode */);
         }
-        function isDeclaration() {
-            while (true) {
-                switch (token()) {
-                    case 109 /* VarKeyword */:
-                    case 115 /* LetKeyword */:
-                    case 81 /* ConstKeyword */:
-                    case 94 /* FunctionKeyword */:
-                    case 80 /* ClassKeyword */:
-                    case 88 /* EnumKeyword */:
-                        return true;
-                    // 'declare', 'module', 'namespace', 'interface'* and 'type' are all legal JavaScript identifiers;
-                    // however, an identifier cannot be followed by another identifier on the same line. This is what we
-                    // count on to parse out the respective declarations. For instance, we exploit this to say that
-                    //
-                    //    namespace n
-                    //
-                    // can be none other than the beginning of a namespace declaration, but need to respect that JavaScript sees
-                    //
-                    //    namespace
-                    //    n
-                    //
-                    // as the identifier 'namespace' on one line followed by the identifier 'n' on another.
-                    // We need to look one token ahead to see if it permissible to try parsing a declaration.
-                    //
-                    // *Note*: 'interface' is actually a strict mode reserved word. So while
-                    //
-                    //   "use strict"
-                    //   interface
-                    //   I {}
-                    //
-                    // could be legal, it would add complexity for very little gain.
-                    case 114 /* InterfaceKeyword */:
-                    case 145 /* TypeKeyword */:
-                        return nextTokenIsIdentifierOnSameLine();
-                    case 135 /* ModuleKeyword */:
-                    case 136 /* NamespaceKeyword */:
-                        return nextTokenIsIdentifierOrStringLiteralOnSameLine();
-                    case 122 /* AbstractKeyword */:
-                    case 126 /* AsyncKeyword */:
-                    case 130 /* DeclareKeyword */:
-                    case 117 /* PrivateKeyword */:
-                    case 118 /* ProtectedKeyword */:
-                    case 119 /* PublicKeyword */:
-                    case 138 /* ReadonlyKeyword */:
-                        nextToken();
-                        // ASI takes effect for this modifier.
-                        if (scanner.hasPrecedingLineBreak()) {
-                            return false;
-                        }
-                        continue;
-                    case 150 /* GlobalKeyword */:
-                        nextToken();
-                        return token() === 18 /* OpenBraceToken */ || token() === 75 /* Identifier */ || token() === 89 /* ExportKeyword */;
-                    case 96 /* ImportKeyword */:
-                        nextToken();
-                        return token() === 10 /* StringLiteral */ || token() === 41 /* AsteriskToken */ ||
-                            token() === 18 /* OpenBraceToken */ || ts.tokenIsIdentifierOrKeyword(token());
-                    case 89 /* ExportKeyword */:
-                        var currentToken_1 = nextToken();
-                        if (currentToken_1 === 145 /* TypeKeyword */) {
-                            currentToken_1 = lookAhead(nextToken);
-                        }
-                        if (currentToken_1 === 62 /* EqualsToken */ || currentToken_1 === 41 /* AsteriskToken */ ||
-                            currentToken_1 === 18 /* OpenBraceToken */ || currentToken_1 === 84 /* DefaultKeyword */ ||
-                            currentToken_1 === 123 /* AsKeyword */) {
-                            return true;
-                        }
-                        continue;
-                    case 120 /* StaticKeyword */:
-                        nextToken();
-                        continue;
-                    default:
-                        return false;
+        // TypeScript Helpers
+        function createDecorateHelper(decoratorExpressions, target, memberName, descriptor) {
+            context.requestEmitHelper(ts.decorateHelper);
+            var argumentsArray = [];
+            argumentsArray.push(factory.createArrayLiteralExpression(decoratorExpressions, /*multiLine*/ true));
+            argumentsArray.push(target);
+            if (memberName) {
+                argumentsArray.push(memberName);
+                if (descriptor) {
+                    argumentsArray.push(descriptor);
                 }
             }
+            return factory.createCallExpression(getUnscopedHelperName("__decorate"), 
+            /*typeArguments*/ undefined, argumentsArray);
         }
-        function isStartOfDeclaration() {
-            return lookAhead(isDeclaration);
-        }
-        function isStartOfStatement() {
-            switch (token()) {
-                case 59 /* AtToken */:
-                case 26 /* SemicolonToken */:
-                case 18 /* OpenBraceToken */:
-                case 109 /* VarKeyword */:
-                case 115 /* LetKeyword */:
-                case 94 /* FunctionKeyword */:
-                case 80 /* ClassKeyword */:
-                case 88 /* EnumKeyword */:
-                case 95 /* IfKeyword */:
-                case 86 /* DoKeyword */:
-                case 111 /* WhileKeyword */:
-                case 93 /* ForKeyword */:
-                case 82 /* ContinueKeyword */:
-                case 77 /* BreakKeyword */:
-                case 101 /* ReturnKeyword */:
-                case 112 /* WithKeyword */:
-                case 103 /* SwitchKeyword */:
-                case 105 /* ThrowKeyword */:
-                case 107 /* TryKeyword */:
-                case 83 /* DebuggerKeyword */:
-                // 'catch' and 'finally' do not actually indicate that the code is part of a statement,
-                // however, we say they are here so that we may gracefully parse them and error later.
-                // falls through
-                case 79 /* CatchKeyword */:
-                case 92 /* FinallyKeyword */:
-                    return true;
-                case 96 /* ImportKeyword */:
-                    return isStartOfDeclaration() || lookAhead(nextTokenIsOpenParenOrLessThanOrDot);
-                case 81 /* ConstKeyword */:
-                case 89 /* ExportKeyword */:
-                    return isStartOfDeclaration();
-                case 126 /* AsyncKeyword */:
-                case 130 /* DeclareKeyword */:
-                case 114 /* InterfaceKeyword */:
-                case 135 /* ModuleKeyword */:
-                case 136 /* NamespaceKeyword */:
-                case 145 /* TypeKeyword */:
-                case 150 /* GlobalKeyword */:
-                    // When these don't start a declaration, they're an identifier in an expression statement
-                    return true;
-                case 119 /* PublicKeyword */:
-                case 117 /* PrivateKeyword */:
-                case 118 /* ProtectedKeyword */:
-                case 120 /* StaticKeyword */:
-                case 138 /* ReadonlyKeyword */:
-                    // When these don't start a declaration, they may be the start of a class member if an identifier
-                    // immediately follows. Otherwise they're an identifier in an expression statement.
-                    return isStartOfDeclaration() || !lookAhead(nextTokenIsIdentifierOrKeywordOnSameLine);
-                default:
-                    return isStartOfExpression();
-            }
-        }
-        function nextTokenIsIdentifierOrStartOfDestructuring() {
-            nextToken();
-            return isIdentifier() || token() === 18 /* OpenBraceToken */ || token() === 22 /* OpenBracketToken */;
+        function createMetadataHelper(metadataKey, metadataValue) {
+            context.requestEmitHelper(ts.metadataHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__metadata"), 
+            /*typeArguments*/ undefined, [
+                factory.createStringLiteral(metadataKey),
+                metadataValue
+            ]);
         }
-        function isLetDeclaration() {
-            // In ES6 'let' always starts a lexical declaration if followed by an identifier or {
-            // or [.
-            return lookAhead(nextTokenIsIdentifierOrStartOfDestructuring);
+        function createParamHelper(expression, parameterOffset, location) {
+            context.requestEmitHelper(ts.paramHelper);
+            return ts.setTextRange(factory.createCallExpression(getUnscopedHelperName("__param"), 
+            /*typeArguments*/ undefined, [
+                factory.createNumericLiteral(parameterOffset + ""),
+                expression
+            ]), location);
+        }
+        // ES2018 Helpers
+        function createAssignHelper(attributesSegments) {
+            if (context.getCompilerOptions().target >= 2 /* ES2015 */) {
+                return factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("Object"), "assign"), 
+                /*typeArguments*/ undefined, attributesSegments);
+            }
+            context.requestEmitHelper(ts.assignHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__assign"), 
+            /*typeArguments*/ undefined, attributesSegments);
+        }
+        function createAwaitHelper(expression) {
+            context.requestEmitHelper(ts.awaitHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__await"), /*typeArguments*/ undefined, [expression]);
+        }
+        function createAsyncGeneratorHelper(generatorFunc, hasLexicalThis) {
+            context.requestEmitHelper(ts.awaitHelper);
+            context.requestEmitHelper(ts.asyncGeneratorHelper);
+            // Mark this node as originally an async function
+            (generatorFunc.emitNode || (generatorFunc.emitNode = {})).flags |= 262144 /* AsyncFunctionBody */ | 524288 /* ReuseTempVariableScope */;
+            return factory.createCallExpression(getUnscopedHelperName("__asyncGenerator"), 
+            /*typeArguments*/ undefined, [
+                hasLexicalThis ? factory.createThis() : factory.createVoidZero(),
+                factory.createIdentifier("arguments"),
+                generatorFunc
+            ]);
         }
-        function parseStatement() {
-            switch (token()) {
-                case 26 /* SemicolonToken */:
-                    return parseEmptyStatement();
-                case 18 /* OpenBraceToken */:
-                    return parseBlock(/*ignoreMissingOpenBrace*/ false);
-                case 109 /* VarKeyword */:
-                    return parseVariableStatement(createNodeWithJSDoc(242 /* VariableDeclaration */));
-                case 115 /* LetKeyword */:
-                    if (isLetDeclaration()) {
-                        return parseVariableStatement(createNodeWithJSDoc(242 /* VariableDeclaration */));
+        function createAsyncDelegatorHelper(expression) {
+            context.requestEmitHelper(ts.awaitHelper);
+            context.requestEmitHelper(ts.asyncDelegator);
+            return factory.createCallExpression(getUnscopedHelperName("__asyncDelegator"), 
+            /*typeArguments*/ undefined, [expression]);
+        }
+        function createAsyncValuesHelper(expression) {
+            context.requestEmitHelper(ts.asyncValues);
+            return factory.createCallExpression(getUnscopedHelperName("__asyncValues"), 
+            /*typeArguments*/ undefined, [expression]);
+        }
+        // ES2018 Destructuring Helpers
+        /** Given value: o, propName: p, pattern: { a, b, ...p } from the original statement
+         * `{ a, b, ...p } = o`, create `p = __rest(o, ["a", "b"]);`
+         */
+        function createRestHelper(value, elements, computedTempVariables, location) {
+            context.requestEmitHelper(ts.restHelper);
+            var propertyNames = [];
+            var computedTempVariableOffset = 0;
+            for (var i = 0; i < elements.length - 1; i++) {
+                var propertyName = ts.getPropertyNameOfBindingOrAssignmentElement(elements[i]);
+                if (propertyName) {
+                    if (ts.isComputedPropertyName(propertyName)) {
+                        ts.Debug.assertIsDefined(computedTempVariables, "Encountered computed property name but 'computedTempVariables' argument was not provided.");
+                        var temp = computedTempVariables[computedTempVariableOffset];
+                        computedTempVariableOffset++;
+                        // typeof _tmp === "symbol" ? _tmp : _tmp + ""
+                        propertyNames.push(factory.createConditionalExpression(factory.createTypeCheck(temp, "symbol"), 
+                        /*questionToken*/ undefined, temp, 
+                        /*colonToken*/ undefined, factory.createAdd(temp, factory.createStringLiteral(""))));
                     }
-                    break;
-                case 94 /* FunctionKeyword */:
-                    return parseFunctionDeclaration(createNodeWithJSDoc(244 /* FunctionDeclaration */));
-                case 80 /* ClassKeyword */:
-                    return parseClassDeclaration(createNodeWithJSDoc(245 /* ClassDeclaration */));
-                case 95 /* IfKeyword */:
-                    return parseIfStatement();
-                case 86 /* DoKeyword */:
-                    return parseDoStatement();
-                case 111 /* WhileKeyword */:
-                    return parseWhileStatement();
-                case 93 /* ForKeyword */:
-                    return parseForOrForInOrForOfStatement();
-                case 82 /* ContinueKeyword */:
-                    return parseBreakOrContinueStatement(233 /* ContinueStatement */);
-                case 77 /* BreakKeyword */:
-                    return parseBreakOrContinueStatement(234 /* BreakStatement */);
-                case 101 /* ReturnKeyword */:
-                    return parseReturnStatement();
-                case 112 /* WithKeyword */:
-                    return parseWithStatement();
-                case 103 /* SwitchKeyword */:
-                    return parseSwitchStatement();
-                case 105 /* ThrowKeyword */:
-                    return parseThrowStatement();
-                case 107 /* TryKeyword */:
-                // Include 'catch' and 'finally' for error recovery.
-                // falls through
-                case 79 /* CatchKeyword */:
-                case 92 /* FinallyKeyword */:
-                    return parseTryStatement();
-                case 83 /* DebuggerKeyword */:
-                    return parseDebuggerStatement();
-                case 59 /* AtToken */:
-                    return parseDeclaration();
-                case 126 /* AsyncKeyword */:
-                case 114 /* InterfaceKeyword */:
-                case 145 /* TypeKeyword */:
-                case 135 /* ModuleKeyword */:
-                case 136 /* NamespaceKeyword */:
-                case 130 /* DeclareKeyword */:
-                case 81 /* ConstKeyword */:
-                case 88 /* EnumKeyword */:
-                case 89 /* ExportKeyword */:
-                case 96 /* ImportKeyword */:
-                case 117 /* PrivateKeyword */:
-                case 118 /* ProtectedKeyword */:
-                case 119 /* PublicKeyword */:
-                case 122 /* AbstractKeyword */:
-                case 120 /* StaticKeyword */:
-                case 138 /* ReadonlyKeyword */:
-                case 150 /* GlobalKeyword */:
-                    if (isStartOfDeclaration()) {
-                        return parseDeclaration();
+                    else {
+                        propertyNames.push(factory.createStringLiteralFromNode(propertyName));
                     }
-                    break;
-            }
-            return parseExpressionOrLabeledStatement();
-        }
-        function isDeclareModifier(modifier) {
-            return modifier.kind === 130 /* DeclareKeyword */;
-        }
-        function parseDeclaration() {
-            var modifiers = lookAhead(function () { return (parseDecorators(), parseModifiers()); });
-            // `parseListElement` attempted to get the reused node at this position,
-            // but the ambient context flag was not yet set, so the node appeared
-            // not reusable in that context.
-            var isAmbient = ts.some(modifiers, isDeclareModifier);
-            if (isAmbient) {
-                var node_3 = tryReuseAmbientDeclaration();
-                if (node_3) {
-                    return node_3;
-                }
-            }
-            var node = createNodeWithJSDoc(0 /* Unknown */);
-            node.decorators = parseDecorators();
-            node.modifiers = parseModifiers();
-            if (isAmbient) {
-                for (var _i = 0, _a = node.modifiers; _i < _a.length; _i++) {
-                    var m = _a[_i];
-                    m.flags |= 8388608 /* Ambient */;
-                }
-                return doInsideOfContext(8388608 /* Ambient */, function () { return parseDeclarationWorker(node); });
-            }
-            else {
-                return parseDeclarationWorker(node);
-            }
-        }
-        function tryReuseAmbientDeclaration() {
-            return doInsideOfContext(8388608 /* Ambient */, function () {
-                var node = currentNode(parsingContext);
-                if (node) {
-                    return consumeNode(node);
                 }
-            });
-        }
-        function parseDeclarationWorker(node) {
-            switch (token()) {
-                case 109 /* VarKeyword */:
-                case 115 /* LetKeyword */:
-                case 81 /* ConstKeyword */:
-                    return parseVariableStatement(node);
-                case 94 /* FunctionKeyword */:
-                    return parseFunctionDeclaration(node);
-                case 80 /* ClassKeyword */:
-                    return parseClassDeclaration(node);
-                case 114 /* InterfaceKeyword */:
-                    return parseInterfaceDeclaration(node);
-                case 145 /* TypeKeyword */:
-                    return parseTypeAliasDeclaration(node);
-                case 88 /* EnumKeyword */:
-                    return parseEnumDeclaration(node);
-                case 150 /* GlobalKeyword */:
-                case 135 /* ModuleKeyword */:
-                case 136 /* NamespaceKeyword */:
-                    return parseModuleDeclaration(node);
-                case 96 /* ImportKeyword */:
-                    return parseImportDeclarationOrImportEqualsDeclaration(node);
-                case 89 /* ExportKeyword */:
-                    nextToken();
-                    switch (token()) {
-                        case 84 /* DefaultKeyword */:
-                        case 62 /* EqualsToken */:
-                            return parseExportAssignment(node);
-                        case 123 /* AsKeyword */:
-                            return parseNamespaceExportDeclaration(node);
-                        default:
-                            return parseExportDeclaration(node);
-                    }
-                default:
-                    if (node.decorators || node.modifiers) {
-                        // We reached this point because we encountered decorators and/or modifiers and assumed a declaration
-                        // would follow. For recovery and error reporting purposes, return an incomplete declaration.
-                        var missing = createMissingNode(264 /* MissingDeclaration */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Declaration_expected);
-                        missing.pos = node.pos;
-                        missing.decorators = node.decorators;
-                        missing.modifiers = node.modifiers;
-                        return finishNode(missing);
-                    }
-                    return undefined; // TODO: GH#18217
-            }
-        }
-        function nextTokenIsIdentifierOrStringLiteralOnSameLine() {
-            nextToken();
-            return !scanner.hasPrecedingLineBreak() && (isIdentifier() || token() === 10 /* StringLiteral */);
-        }
-        function parseFunctionBlockOrSemicolon(flags, diagnosticMessage) {
-            if (token() !== 18 /* OpenBraceToken */ && canParseSemicolon()) {
-                parseSemicolon();
-                return;
-            }
-            return parseFunctionBlock(flags, diagnosticMessage);
-        }
-        // DECLARATIONS
-        function parseArrayBindingElement() {
-            if (token() === 27 /* CommaToken */) {
-                return createNode(215 /* OmittedExpression */);
-            }
-            var node = createNode(191 /* BindingElement */);
-            node.dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
-            node.name = parseIdentifierOrPattern();
-            node.initializer = parseInitializer();
-            return finishNode(node);
-        }
-        function parseObjectBindingElement() {
-            var node = createNode(191 /* BindingElement */);
-            node.dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
-            var tokenIsIdentifier = isIdentifier();
-            var propertyName = parsePropertyName();
-            if (tokenIsIdentifier && token() !== 58 /* ColonToken */) {
-                node.name = propertyName;
-            }
-            else {
-                parseExpected(58 /* ColonToken */);
-                node.propertyName = propertyName;
-                node.name = parseIdentifierOrPattern();
-            }
-            node.initializer = parseInitializer();
-            return finishNode(node);
-        }
-        function parseObjectBindingPattern() {
-            var node = createNode(189 /* ObjectBindingPattern */);
-            parseExpected(18 /* OpenBraceToken */);
-            node.elements = parseDelimitedList(9 /* ObjectBindingElements */, parseObjectBindingElement);
-            parseExpected(19 /* CloseBraceToken */);
-            return finishNode(node);
-        }
-        function parseArrayBindingPattern() {
-            var node = createNode(190 /* ArrayBindingPattern */);
-            parseExpected(22 /* OpenBracketToken */);
-            node.elements = parseDelimitedList(10 /* ArrayBindingElements */, parseArrayBindingElement);
-            parseExpected(23 /* CloseBracketToken */);
-            return finishNode(node);
-        }
-        function isIdentifierOrPrivateIdentifierOrPattern() {
-            return token() === 18 /* OpenBraceToken */
-                || token() === 22 /* OpenBracketToken */
-                || token() === 76 /* PrivateIdentifier */
-                || isIdentifier();
-        }
-        function parseIdentifierOrPattern(privateIdentifierDiagnosticMessage) {
-            if (token() === 22 /* OpenBracketToken */) {
-                return parseArrayBindingPattern();
-            }
-            if (token() === 18 /* OpenBraceToken */) {
-                return parseObjectBindingPattern();
-            }
-            return parseIdentifier(/*diagnosticMessage*/ undefined, privateIdentifierDiagnosticMessage);
-        }
-        function parseVariableDeclarationAllowExclamation() {
-            return parseVariableDeclaration(/*allowExclamation*/ true);
-        }
-        function parseVariableDeclaration(allowExclamation) {
-            var node = createNode(242 /* VariableDeclaration */);
-            node.name = parseIdentifierOrPattern(ts.Diagnostics.Private_identifiers_are_not_allowed_in_variable_declarations);
-            if (allowExclamation && node.name.kind === 75 /* Identifier */ &&
-                token() === 53 /* ExclamationToken */ && !scanner.hasPrecedingLineBreak()) {
-                node.exclamationToken = parseTokenNode();
-            }
-            node.type = parseTypeAnnotation();
-            if (!isInOrOfKeyword(token())) {
-                node.initializer = parseInitializer();
-            }
-            return finishNode(node);
-        }
-        function parseVariableDeclarationList(inForStatementInitializer) {
-            var node = createNode(243 /* VariableDeclarationList */);
-            switch (token()) {
-                case 109 /* VarKeyword */:
-                    break;
-                case 115 /* LetKeyword */:
-                    node.flags |= 1 /* Let */;
-                    break;
-                case 81 /* ConstKeyword */:
-                    node.flags |= 2 /* Const */;
-                    break;
-                default:
-                    ts.Debug.fail();
-            }
-            nextToken();
-            // The user may have written the following:
-            //
-            //    for (let of X) { }
-            //
-            // In this case, we want to parse an empty declaration list, and then parse 'of'
-            // as a keyword. The reason this is not automatic is that 'of' is a valid identifier.
-            // So we need to look ahead to determine if 'of' should be treated as a keyword in
-            // this context.
-            // The checker will then give an error that there is an empty declaration list.
-            if (token() === 152 /* OfKeyword */ && lookAhead(canFollowContextualOfKeyword)) {
-                node.declarations = createMissingList();
             }
-            else {
-                var savedDisallowIn = inDisallowInContext();
-                setDisallowInContext(inForStatementInitializer);
-                node.declarations = parseDelimitedList(8 /* VariableDeclarations */, inForStatementInitializer ? parseVariableDeclaration : parseVariableDeclarationAllowExclamation);
-                setDisallowInContext(savedDisallowIn);
-            }
-            return finishNode(node);
+            return factory.createCallExpression(getUnscopedHelperName("__rest"), 
+            /*typeArguments*/ undefined, [
+                value,
+                ts.setTextRange(factory.createArrayLiteralExpression(propertyNames), location)
+            ]);
         }
-        function canFollowContextualOfKeyword() {
-            return nextTokenIsIdentifier() && nextToken() === 21 /* CloseParenToken */;
+        // ES2017 Helpers
+        function createAwaiterHelper(hasLexicalThis, hasLexicalArguments, promiseConstructor, body) {
+            context.requestEmitHelper(ts.awaiterHelper);
+            var generatorFunc = factory.createFunctionExpression(
+            /*modifiers*/ undefined, factory.createToken(41 /* AsteriskToken */), 
+            /*name*/ undefined, 
+            /*typeParameters*/ undefined, 
+            /*parameters*/ [], 
+            /*type*/ undefined, body);
+            // Mark this node as originally an async function
+            (generatorFunc.emitNode || (generatorFunc.emitNode = {})).flags |= 262144 /* AsyncFunctionBody */ | 524288 /* ReuseTempVariableScope */;
+            return factory.createCallExpression(getUnscopedHelperName("__awaiter"), 
+            /*typeArguments*/ undefined, [
+                hasLexicalThis ? factory.createThis() : factory.createVoidZero(),
+                hasLexicalArguments ? factory.createIdentifier("arguments") : factory.createVoidZero(),
+                promiseConstructor ? ts.createExpressionFromEntityName(factory, promiseConstructor) : factory.createVoidZero(),
+                generatorFunc
+            ]);
         }
-        function parseVariableStatement(node) {
-            node.kind = 225 /* VariableStatement */;
-            node.declarationList = parseVariableDeclarationList(/*inForStatementInitializer*/ false);
-            parseSemicolon();
-            return finishNode(node);
-        }
-        function parseFunctionDeclaration(node) {
-            node.kind = 244 /* FunctionDeclaration */;
-            parseExpected(94 /* FunctionKeyword */);
-            node.asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
-            node.name = hasModifierOfKind(node, 84 /* DefaultKeyword */) ? parseOptionalIdentifier() : parseIdentifier();
-            var isGenerator = node.asteriskToken ? 1 /* Yield */ : 0 /* None */;
-            var isAsync = hasModifierOfKind(node, 126 /* AsyncKeyword */) ? 2 /* Await */ : 0 /* None */;
-            fillSignature(58 /* ColonToken */, isGenerator | isAsync, node);
-            node.body = parseFunctionBlockOrSemicolon(isGenerator | isAsync, ts.Diagnostics.or_expected);
-            return finishNode(node);
+        // ES2015 Helpers
+        function createExtendsHelper(name) {
+            context.requestEmitHelper(ts.extendsHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__extends"), 
+            /*typeArguments*/ undefined, [name, factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */)]);
+        }
+        function createTemplateObjectHelper(cooked, raw) {
+            context.requestEmitHelper(ts.templateObjectHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__makeTemplateObject"), 
+            /*typeArguments*/ undefined, [cooked, raw]);
+        }
+        function createSpreadHelper(argumentList) {
+            context.requestEmitHelper(ts.readHelper);
+            context.requestEmitHelper(ts.spreadHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__spread"), 
+            /*typeArguments*/ undefined, argumentList);
+        }
+        function createSpreadArraysHelper(argumentList) {
+            context.requestEmitHelper(ts.spreadArraysHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__spreadArrays"), 
+            /*typeArguments*/ undefined, argumentList);
+        }
+        // ES2015 Destructuring Helpers
+        function createValuesHelper(expression) {
+            context.requestEmitHelper(ts.valuesHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__values"), 
+            /*typeArguments*/ undefined, [expression]);
+        }
+        function createReadHelper(iteratorRecord, count) {
+            context.requestEmitHelper(ts.readHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__read"), 
+            /*typeArguments*/ undefined, count !== undefined
+                ? [iteratorRecord, factory.createNumericLiteral(count + "")]
+                : [iteratorRecord]);
+        }
+        // ES2015 Generator Helpers
+        function createGeneratorHelper(body) {
+            context.requestEmitHelper(ts.generatorHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__generator"), 
+            /*typeArguments*/ undefined, [factory.createThis(), body]);
+        }
+        // ES Module Helpers
+        function createCreateBindingHelper(module, inputName, outputName) {
+            context.requestEmitHelper(ts.createBindingHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__createBinding"), 
+            /*typeArguments*/ undefined, __spreadArrays([factory.createIdentifier("exports"), module, inputName], (outputName ? [outputName] : [])));
+        }
+        function createImportStarHelper(expression) {
+            context.requestEmitHelper(ts.importStarHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__importStar"), 
+            /*typeArguments*/ undefined, [expression]);
+        }
+        function createImportStarCallbackHelper() {
+            context.requestEmitHelper(ts.importStarHelper);
+            return getUnscopedHelperName("__importStar");
+        }
+        function createImportDefaultHelper(expression) {
+            context.requestEmitHelper(ts.importDefaultHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__importDefault"), 
+            /*typeArguments*/ undefined, [expression]);
+        }
+        function createExportStarHelper(moduleExpression, exportsExpression) {
+            if (exportsExpression === void 0) { exportsExpression = factory.createIdentifier("exports"); }
+            context.requestEmitHelper(ts.exportStarHelper);
+            context.requestEmitHelper(ts.createBindingHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__exportStar"), 
+            /*typeArguments*/ undefined, [moduleExpression, exportsExpression]);
+        }
+        // Class Fields Helpers
+        function createClassPrivateFieldGetHelper(receiver, privateField) {
+            context.requestEmitHelper(ts.classPrivateFieldGetHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__classPrivateFieldGet"), /*typeArguments*/ undefined, [receiver, privateField]);
+        }
+        function createClassPrivateFieldSetHelper(receiver, privateField, value) {
+            context.requestEmitHelper(ts.classPrivateFieldSetHelper);
+            return factory.createCallExpression(getUnscopedHelperName("__classPrivateFieldSet"), /*typeArguments*/ undefined, [receiver, privateField, value]);
+        }
+    }
+    ts.createEmitHelperFactory = createEmitHelperFactory;
+    /* @internal */
+    function compareEmitHelpers(x, y) {
+        if (x === y)
+            return 0 /* EqualTo */;
+        if (x.priority === y.priority)
+            return 0 /* EqualTo */;
+        if (x.priority === undefined)
+            return 1 /* GreaterThan */;
+        if (y.priority === undefined)
+            return -1 /* LessThan */;
+        return ts.compareValues(x.priority, y.priority);
+    }
+    ts.compareEmitHelpers = compareEmitHelpers;
+    /**
+     * @param input Template string input strings
+     * @param args Names which need to be made file-level unique
+     */
+    function helperString(input) {
+        var args = [];
+        for (var _i = 1; _i < arguments.length; _i++) {
+            args[_i - 1] = arguments[_i];
         }
-        function parseConstructorName() {
-            if (token() === 129 /* ConstructorKeyword */) {
-                return parseExpected(129 /* ConstructorKeyword */);
-            }
-            if (token() === 10 /* StringLiteral */ && lookAhead(nextToken) === 20 /* OpenParenToken */) {
-                return tryParse(function () {
-                    var literalNode = parseLiteralNode();
-                    return literalNode.text === "constructor" ? literalNode : undefined;
-                });
+        return function (uniqueName) {
+            var result = "";
+            for (var i = 0; i < args.length; i++) {
+                result += input[i];
+                result += uniqueName(args[i]);
             }
+            result += input[input.length - 1];
+            return result;
+        };
+    }
+    ts.helperString = helperString;
+    // TypeScript Helpers
+    ts.decorateHelper = {
+        name: "typescript:decorate",
+        importName: "__decorate",
+        scoped: false,
+        priority: 2,
+        text: "\n            var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {\n                var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\n                if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\n                else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\n                return c > 3 && r && Object.defineProperty(target, key, r), r;\n            };"
+    };
+    ts.metadataHelper = {
+        name: "typescript:metadata",
+        importName: "__metadata",
+        scoped: false,
+        priority: 3,
+        text: "\n            var __metadata = (this && this.__metadata) || function (k, v) {\n                if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(k, v);\n            };"
+    };
+    ts.paramHelper = {
+        name: "typescript:param",
+        importName: "__param",
+        scoped: false,
+        priority: 4,
+        text: "\n            var __param = (this && this.__param) || function (paramIndex, decorator) {\n                return function (target, key) { decorator(target, key, paramIndex); }\n            };"
+    };
+    // ES2018 Helpers
+    ts.assignHelper = {
+        name: "typescript:assign",
+        importName: "__assign",
+        scoped: false,
+        priority: 1,
+        text: "\n            var __assign = (this && this.__assign) || function () {\n                __assign = Object.assign || function(t) {\n                    for (var s, i = 1, n = arguments.length; i < n; i++) {\n                        s = arguments[i];\n                        for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n                            t[p] = s[p];\n                    }\n                    return t;\n                };\n                return __assign.apply(this, arguments);\n            };"
+    };
+    ts.awaitHelper = {
+        name: "typescript:await",
+        importName: "__await",
+        scoped: false,
+        text: "\n            var __await = (this && this.__await) || function (v) { return this instanceof __await ? (this.v = v, this) : new __await(v); }"
+    };
+    ts.asyncGeneratorHelper = {
+        name: "typescript:asyncGenerator",
+        importName: "__asyncGenerator",
+        scoped: false,
+        dependencies: [ts.awaitHelper],
+        text: "\n            var __asyncGenerator = (this && this.__asyncGenerator) || function (thisArg, _arguments, generator) {\n                if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n                var g = generator.apply(thisArg, _arguments || []), i, q = [];\n                return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\n                function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\n                function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\n                function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\n                function fulfill(value) { resume(\"next\", value); }\n                function reject(value) { resume(\"throw\", value); }\n                function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\n            };"
+    };
+    ts.asyncDelegator = {
+        name: "typescript:asyncDelegator",
+        importName: "__asyncDelegator",
+        scoped: false,
+        dependencies: [ts.awaitHelper],
+        text: "\n            var __asyncDelegator = (this && this.__asyncDelegator) || function (o) {\n                var i, p;\n                return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\n                function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: n === \"return\" } : f ? f(v) : v; } : f; }\n            };"
+    };
+    ts.asyncValues = {
+        name: "typescript:asyncValues",
+        importName: "__asyncValues",
+        scoped: false,
+        text: "\n            var __asyncValues = (this && this.__asyncValues) || function (o) {\n                if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n                var m = o[Symbol.asyncIterator], i;\n                return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\n                function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\n                function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\n            };"
+    };
+    // ES2018 Destructuring Helpers
+    ts.restHelper = {
+        name: "typescript:rest",
+        importName: "__rest",
+        scoped: false,
+        text: "\n            var __rest = (this && this.__rest) || function (s, e) {\n                var t = {};\n                for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\n                    t[p] = s[p];\n                if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\n                    for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\n                        if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\n                            t[p[i]] = s[p[i]];\n                    }\n                return t;\n            };"
+    };
+    // ES2017 Helpers
+    ts.awaiterHelper = {
+        name: "typescript:awaiter",
+        importName: "__awaiter",
+        scoped: false,
+        priority: 5,
+        text: "\n            var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n                function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n                return new (P || (P = Promise))(function (resolve, reject) {\n                    function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n                    function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n                    function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n                    step((generator = generator.apply(thisArg, _arguments || [])).next());\n                });\n            };"
+    };
+    // ES2015 Helpers
+    ts.extendsHelper = {
+        name: "typescript:extends",
+        importName: "__extends",
+        scoped: false,
+        priority: 0,
+        text: "\n            var __extends = (this && this.__extends) || (function () {\n                var extendStatics = function (d, b) {\n                    extendStatics = Object.setPrototypeOf ||\n                        ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\n                        function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\n                    return extendStatics(d, b);\n                };\n\n                return function (d, b) {\n                    extendStatics(d, b);\n                    function __() { this.constructor = d; }\n                    d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\n                };\n            })();"
+    };
+    ts.templateObjectHelper = {
+        name: "typescript:makeTemplateObject",
+        importName: "__makeTemplateObject",
+        scoped: false,
+        priority: 0,
+        text: "\n            var __makeTemplateObject = (this && this.__makeTemplateObject) || function (cooked, raw) {\n                if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\n                return cooked;\n            };"
+    };
+    ts.readHelper = {
+        name: "typescript:read",
+        importName: "__read",
+        scoped: false,
+        text: "\n            var __read = (this && this.__read) || function (o, n) {\n                var m = typeof Symbol === \"function\" && o[Symbol.iterator];\n                if (!m) return o;\n                var i = m.call(o), r, ar = [], e;\n                try {\n                    while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\n                }\n                catch (error) { e = { error: error }; }\n                finally {\n                    try {\n                        if (r && !r.done && (m = i[\"return\"])) m.call(i);\n                    }\n                    finally { if (e) throw e.error; }\n                }\n                return ar;\n            };"
+    };
+    ts.spreadHelper = {
+        name: "typescript:spread",
+        importName: "__spread",
+        scoped: false,
+        dependencies: [ts.readHelper],
+        text: "\n            var __spread = (this && this.__spread) || function () {\n                for (var ar = [], i = 0; i < arguments.length; i++) ar = ar.concat(__read(arguments[i]));\n                return ar;\n            };"
+    };
+    ts.spreadArraysHelper = {
+        name: "typescript:spreadArrays",
+        importName: "__spreadArrays",
+        scoped: false,
+        text: "\n            var __spreadArrays = (this && this.__spreadArrays) || function () {\n                for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\n                for (var r = Array(s), k = 0, i = 0; i < il; i++)\n                    for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\n                        r[k] = a[j];\n                return r;\n            };"
+    };
+    // ES2015 Destructuring Helpers
+    ts.valuesHelper = {
+        name: "typescript:values",
+        importName: "__values",
+        scoped: false,
+        text: "\n            var __values = (this && this.__values) || function(o) {\n                var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\n                if (m) return m.call(o);\n                if (o && typeof o.length === \"number\") return {\n                    next: function () {\n                        if (o && i >= o.length) o = void 0;\n                        return { value: o && o[i++], done: !o };\n                    }\n                };\n                throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\n            };"
+    };
+    // ES2015 Generator Helpers
+    // The __generator helper is used by down-level transformations to emulate the runtime
+    // semantics of an ES2015 generator function. When called, this helper returns an
+    // object that implements the Iterator protocol, in that it has `next`, `return`, and
+    // `throw` methods that step through the generator when invoked.
+    //
+    // parameters:
+    //  @param thisArg  The value to use as the `this` binding for the transformed generator body.
+    //  @param body     A function that acts as the transformed generator body.
+    //
+    // variables:
+    //  _       Persistent state for the generator that is shared between the helper and the
+    //          generator body. The state object has the following members:
+    //            sent() - A method that returns or throws the current completion value.
+    //            label  - The next point at which to resume evaluation of the generator body.
+    //            trys   - A stack of protected regions (try/catch/finally blocks).
+    //            ops    - A stack of pending instructions when inside of a finally block.
+    //  f       A value indicating whether the generator is executing.
+    //  y       An iterator to delegate for a yield*.
+    //  t       A temporary variable that holds one of the following values (note that these
+    //          cases do not overlap):
+    //          - The completion value when resuming from a `yield` or `yield*`.
+    //          - The error value for a catch block.
+    //          - The current protected region (array of try/catch/finally/end labels).
+    //          - The verb (`next`, `throw`, or `return` method) to delegate to the expression
+    //            of a `yield*`.
+    //          - The result of evaluating the verb delegated to the expression of a `yield*`.
+    //
+    // functions:
+    //  verb(n)     Creates a bound callback to the `step` function for opcode `n`.
+    //  step(op)    Evaluates opcodes in a generator body until execution is suspended or
+    //              completed.
+    //
+    // The __generator helper understands a limited set of instructions:
+    //  0: next(value?)     - Start or resume the generator with the specified value.
+    //  1: throw(error)     - Resume the generator with an exception. If the generator is
+    //                        suspended inside of one or more protected regions, evaluates
+    //                        any intervening finally blocks between the current label and
+    //                        the nearest catch block or function boundary. If uncaught, the
+    //                        exception is thrown to the caller.
+    //  2: return(value?)   - Resume the generator as if with a return. If the generator is
+    //                        suspended inside of one or more protected regions, evaluates any
+    //                        intervening finally blocks.
+    //  3: break(label)     - Jump to the specified label. If the label is outside of the
+    //                        current protected region, evaluates any intervening finally
+    //                        blocks.
+    //  4: yield(value?)    - Yield execution to the caller with an optional value. When
+    //                        resumed, the generator will continue at the next label.
+    //  5: yield*(value)    - Delegates evaluation to the supplied iterator. When
+    //                        delegation completes, the generator will continue at the next
+    //                        label.
+    //  6: catch(error)     - Handles an exception thrown from within the generator body. If
+    //                        the current label is inside of one or more protected regions,
+    //                        evaluates any intervening finally blocks between the current
+    //                        label and the nearest catch block or function boundary. If
+    //                        uncaught, the exception is thrown to the caller.
+    //  7: endfinally       - Ends a finally block, resuming the last instruction prior to
+    //                        entering a finally block.
+    //
+    // For examples of how these are used, see the comments in ./transformers/generators.ts
+    ts.generatorHelper = {
+        name: "typescript:generator",
+        importName: "__generator",
+        scoped: false,
+        priority: 6,
+        text: "\n            var __generator = (this && this.__generator) || function (thisArg, body) {\n                var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\n                return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n                function verb(n) { return function (v) { return step([n, v]); }; }\n                function step(op) {\n                    if (f) throw new TypeError(\"Generator is already executing.\");\n                    while (_) try {\n                        if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n                        if (y = 0, t) op = [op[0] & 2, t.value];\n                        switch (op[0]) {\n                            case 0: case 1: t = op; break;\n                            case 4: _.label++; return { value: op[1], done: false };\n                            case 5: _.label++; y = op[1]; op = [0]; continue;\n                            case 7: op = _.ops.pop(); _.trys.pop(); continue;\n                            default:\n                                if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n                                if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n                                if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n                                if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n                                if (t[2]) _.ops.pop();\n                                _.trys.pop(); continue;\n                        }\n                        op = body.call(thisArg, _);\n                    } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n                    if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n                }\n            };"
+    };
+    // ES Module Helpers
+    ts.createBindingHelper = {
+        name: "typescript:commonjscreatebinding",
+        importName: "__createBinding",
+        scoped: false,
+        priority: 1,
+        text: "\n            var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {\n                if (k2 === undefined) k2 = k;\n                Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });\n            }) : (function(o, m, k, k2) {\n                if (k2 === undefined) k2 = k;\n                o[k2] = m[k];\n            }));"
+    };
+    ts.setModuleDefaultHelper = {
+        name: "typescript:commonjscreatevalue",
+        importName: "__setModuleDefault",
+        scoped: false,
+        priority: 1,
+        text: "\n            var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {\n                Object.defineProperty(o, \"default\", { enumerable: true, value: v });\n            }) : function(o, v) {\n                o[\"default\"] = v;\n            });"
+    };
+    // emit helper for `import * as Name from "foo"`
+    ts.importStarHelper = {
+        name: "typescript:commonjsimportstar",
+        importName: "__importStar",
+        scoped: false,
+        dependencies: [ts.createBindingHelper, ts.setModuleDefaultHelper],
+        priority: 2,
+        text: "\n            var __importStar = (this && this.__importStar) || function (mod) {\n                if (mod && mod.__esModule) return mod;\n                var result = {};\n                if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\n                __setModuleDefault(result, mod);\n                return result;\n            };"
+    };
+    // emit helper for `import Name from "foo"`
+    ts.importDefaultHelper = {
+        name: "typescript:commonjsimportdefault",
+        importName: "__importDefault",
+        scoped: false,
+        text: "\n            var __importDefault = (this && this.__importDefault) || function (mod) {\n                return (mod && mod.__esModule) ? mod : { \"default\": mod };\n            };"
+    };
+    // emit output for the __export helper function
+    ts.exportStarHelper = {
+        name: "typescript:export-star",
+        importName: "__exportStar",
+        scoped: false,
+        dependencies: [ts.createBindingHelper],
+        priority: 2,
+        text: "\n            var __exportStar = (this && this.__exportStar) || function(m, exports) {\n                for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);\n            };"
+    };
+    // Class fields helpers
+    ts.classPrivateFieldGetHelper = {
+        name: "typescript:classPrivateFieldGet",
+        importName: "__classPrivateFieldGet",
+        scoped: false,
+        text: "\n            var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, privateMap) {\n                if (!privateMap.has(receiver)) {\n                    throw new TypeError(\"attempted to get private field on non-instance\");\n                }\n                return privateMap.get(receiver);\n            };"
+    };
+    ts.classPrivateFieldSetHelper = {
+        name: "typescript:classPrivateFieldSet",
+        importName: "__classPrivateFieldSet",
+        scoped: false,
+        text: "\n            var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, privateMap, value) {\n                if (!privateMap.has(receiver)) {\n                    throw new TypeError(\"attempted to set private field on non-instance\");\n                }\n                privateMap.set(receiver, value);\n                return value;\n            };"
+    };
+    var allUnscopedEmitHelpers;
+    function getAllUnscopedEmitHelpers() {
+        return allUnscopedEmitHelpers || (allUnscopedEmitHelpers = ts.arrayToMap([
+            ts.decorateHelper,
+            ts.metadataHelper,
+            ts.paramHelper,
+            ts.assignHelper,
+            ts.awaitHelper,
+            ts.asyncGeneratorHelper,
+            ts.asyncDelegator,
+            ts.asyncValues,
+            ts.restHelper,
+            ts.awaiterHelper,
+            ts.extendsHelper,
+            ts.templateObjectHelper,
+            ts.spreadHelper,
+            ts.spreadArraysHelper,
+            ts.valuesHelper,
+            ts.readHelper,
+            ts.generatorHelper,
+            ts.importStarHelper,
+            ts.importDefaultHelper,
+            ts.exportStarHelper,
+            ts.classPrivateFieldGetHelper,
+            ts.classPrivateFieldSetHelper,
+            ts.createBindingHelper,
+            ts.setModuleDefaultHelper
+        ], function (helper) { return helper.name; }));
+    }
+    ts.getAllUnscopedEmitHelpers = getAllUnscopedEmitHelpers;
+    ts.asyncSuperHelper = {
+        name: "typescript:async-super",
+        scoped: true,
+        text: helperString(__makeTemplateObject(["\n            const ", " = name => super[name];"], ["\n            const ", " = name => super[name];"]), "_superIndex")
+    };
+    ts.advancedAsyncSuperHelper = {
+        name: "typescript:advanced-async-super",
+        scoped: true,
+        text: helperString(__makeTemplateObject(["\n            const ", " = (function (geti, seti) {\n                const cache = Object.create(null);\n                return name => cache[name] || (cache[name] = { get value() { return geti(name); }, set value(v) { seti(name, v); } });\n            })(name => super[name], (name, value) => super[name] = value);"], ["\n            const ", " = (function (geti, seti) {\n                const cache = Object.create(null);\n                return name => cache[name] || (cache[name] = { get value() { return geti(name); }, set value(v) { seti(name, v); } });\n            })(name => super[name], (name, value) => super[name] = value);"]), "_superIndex")
+    };
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    // Literals
+    function isNumericLiteral(node) {
+        return node.kind === 8 /* NumericLiteral */;
+    }
+    ts.isNumericLiteral = isNumericLiteral;
+    function isBigIntLiteral(node) {
+        return node.kind === 9 /* BigIntLiteral */;
+    }
+    ts.isBigIntLiteral = isBigIntLiteral;
+    function isStringLiteral(node) {
+        return node.kind === 10 /* StringLiteral */;
+    }
+    ts.isStringLiteral = isStringLiteral;
+    function isJsxText(node) {
+        return node.kind === 11 /* JsxText */;
+    }
+    ts.isJsxText = isJsxText;
+    function isRegularExpressionLiteral(node) {
+        return node.kind === 13 /* RegularExpressionLiteral */;
+    }
+    ts.isRegularExpressionLiteral = isRegularExpressionLiteral;
+    function isNoSubstitutionTemplateLiteral(node) {
+        return node.kind === 14 /* NoSubstitutionTemplateLiteral */;
+    }
+    ts.isNoSubstitutionTemplateLiteral = isNoSubstitutionTemplateLiteral;
+    // Pseudo-literals
+    function isTemplateHead(node) {
+        return node.kind === 15 /* TemplateHead */;
+    }
+    ts.isTemplateHead = isTemplateHead;
+    function isTemplateMiddle(node) {
+        return node.kind === 16 /* TemplateMiddle */;
+    }
+    ts.isTemplateMiddle = isTemplateMiddle;
+    function isTemplateTail(node) {
+        return node.kind === 17 /* TemplateTail */;
+    }
+    ts.isTemplateTail = isTemplateTail;
+    // Identifiers
+    function isIdentifier(node) {
+        return node.kind === 78 /* Identifier */;
+    }
+    ts.isIdentifier = isIdentifier;
+    // Names
+    function isQualifiedName(node) {
+        return node.kind === 157 /* QualifiedName */;
+    }
+    ts.isQualifiedName = isQualifiedName;
+    function isComputedPropertyName(node) {
+        return node.kind === 158 /* ComputedPropertyName */;
+    }
+    ts.isComputedPropertyName = isComputedPropertyName;
+    function isPrivateIdentifier(node) {
+        return node.kind === 79 /* PrivateIdentifier */;
+    }
+    ts.isPrivateIdentifier = isPrivateIdentifier;
+    // Tokens
+    /*@internal*/
+    function isSuperKeyword(node) {
+        return node.kind === 105 /* SuperKeyword */;
+    }
+    ts.isSuperKeyword = isSuperKeyword;
+    /*@internal*/
+    function isImportKeyword(node) {
+        return node.kind === 99 /* ImportKeyword */;
+    }
+    ts.isImportKeyword = isImportKeyword;
+    /*@internal*/
+    function isCommaToken(node) {
+        return node.kind === 27 /* CommaToken */;
+    }
+    ts.isCommaToken = isCommaToken;
+    /*@internal*/
+    function isQuestionToken(node) {
+        return node.kind === 57 /* QuestionToken */;
+    }
+    ts.isQuestionToken = isQuestionToken;
+    /*@internal*/
+    function isExclamationToken(node) {
+        return node.kind === 53 /* ExclamationToken */;
+    }
+    ts.isExclamationToken = isExclamationToken;
+    // Signature elements
+    function isTypeParameterDeclaration(node) {
+        return node.kind === 159 /* TypeParameter */;
+    }
+    ts.isTypeParameterDeclaration = isTypeParameterDeclaration;
+    // TODO(rbuckton): Rename to 'isParameterDeclaration'
+    function isParameter(node) {
+        return node.kind === 160 /* Parameter */;
+    }
+    ts.isParameter = isParameter;
+    function isDecorator(node) {
+        return node.kind === 161 /* Decorator */;
+    }
+    ts.isDecorator = isDecorator;
+    // TypeMember
+    function isPropertySignature(node) {
+        return node.kind === 162 /* PropertySignature */;
+    }
+    ts.isPropertySignature = isPropertySignature;
+    function isPropertyDeclaration(node) {
+        return node.kind === 163 /* PropertyDeclaration */;
+    }
+    ts.isPropertyDeclaration = isPropertyDeclaration;
+    function isMethodSignature(node) {
+        return node.kind === 164 /* MethodSignature */;
+    }
+    ts.isMethodSignature = isMethodSignature;
+    function isMethodDeclaration(node) {
+        return node.kind === 165 /* MethodDeclaration */;
+    }
+    ts.isMethodDeclaration = isMethodDeclaration;
+    function isConstructorDeclaration(node) {
+        return node.kind === 166 /* Constructor */;
+    }
+    ts.isConstructorDeclaration = isConstructorDeclaration;
+    function isGetAccessorDeclaration(node) {
+        return node.kind === 167 /* GetAccessor */;
+    }
+    ts.isGetAccessorDeclaration = isGetAccessorDeclaration;
+    function isSetAccessorDeclaration(node) {
+        return node.kind === 168 /* SetAccessor */;
+    }
+    ts.isSetAccessorDeclaration = isSetAccessorDeclaration;
+    function isCallSignatureDeclaration(node) {
+        return node.kind === 169 /* CallSignature */;
+    }
+    ts.isCallSignatureDeclaration = isCallSignatureDeclaration;
+    function isConstructSignatureDeclaration(node) {
+        return node.kind === 170 /* ConstructSignature */;
+    }
+    ts.isConstructSignatureDeclaration = isConstructSignatureDeclaration;
+    function isIndexSignatureDeclaration(node) {
+        return node.kind === 171 /* IndexSignature */;
+    }
+    ts.isIndexSignatureDeclaration = isIndexSignatureDeclaration;
+    // Type
+    function isTypePredicateNode(node) {
+        return node.kind === 172 /* TypePredicate */;
+    }
+    ts.isTypePredicateNode = isTypePredicateNode;
+    function isTypeReferenceNode(node) {
+        return node.kind === 173 /* TypeReference */;
+    }
+    ts.isTypeReferenceNode = isTypeReferenceNode;
+    function isFunctionTypeNode(node) {
+        return node.kind === 174 /* FunctionType */;
+    }
+    ts.isFunctionTypeNode = isFunctionTypeNode;
+    function isConstructorTypeNode(node) {
+        return node.kind === 175 /* ConstructorType */;
+    }
+    ts.isConstructorTypeNode = isConstructorTypeNode;
+    function isTypeQueryNode(node) {
+        return node.kind === 176 /* TypeQuery */;
+    }
+    ts.isTypeQueryNode = isTypeQueryNode;
+    function isTypeLiteralNode(node) {
+        return node.kind === 177 /* TypeLiteral */;
+    }
+    ts.isTypeLiteralNode = isTypeLiteralNode;
+    function isArrayTypeNode(node) {
+        return node.kind === 178 /* ArrayType */;
+    }
+    ts.isArrayTypeNode = isArrayTypeNode;
+    function isTupleTypeNode(node) {
+        return node.kind === 179 /* TupleType */;
+    }
+    ts.isTupleTypeNode = isTupleTypeNode;
+    function isNamedTupleMember(node) {
+        return node.kind === 192 /* NamedTupleMember */;
+    }
+    ts.isNamedTupleMember = isNamedTupleMember;
+    function isOptionalTypeNode(node) {
+        return node.kind === 180 /* OptionalType */;
+    }
+    ts.isOptionalTypeNode = isOptionalTypeNode;
+    function isRestTypeNode(node) {
+        return node.kind === 181 /* RestType */;
+    }
+    ts.isRestTypeNode = isRestTypeNode;
+    function isUnionTypeNode(node) {
+        return node.kind === 182 /* UnionType */;
+    }
+    ts.isUnionTypeNode = isUnionTypeNode;
+    function isIntersectionTypeNode(node) {
+        return node.kind === 183 /* IntersectionType */;
+    }
+    ts.isIntersectionTypeNode = isIntersectionTypeNode;
+    function isConditionalTypeNode(node) {
+        return node.kind === 184 /* ConditionalType */;
+    }
+    ts.isConditionalTypeNode = isConditionalTypeNode;
+    function isInferTypeNode(node) {
+        return node.kind === 185 /* InferType */;
+    }
+    ts.isInferTypeNode = isInferTypeNode;
+    function isParenthesizedTypeNode(node) {
+        return node.kind === 186 /* ParenthesizedType */;
+    }
+    ts.isParenthesizedTypeNode = isParenthesizedTypeNode;
+    function isThisTypeNode(node) {
+        return node.kind === 187 /* ThisType */;
+    }
+    ts.isThisTypeNode = isThisTypeNode;
+    function isTypeOperatorNode(node) {
+        return node.kind === 188 /* TypeOperator */;
+    }
+    ts.isTypeOperatorNode = isTypeOperatorNode;
+    function isIndexedAccessTypeNode(node) {
+        return node.kind === 189 /* IndexedAccessType */;
+    }
+    ts.isIndexedAccessTypeNode = isIndexedAccessTypeNode;
+    function isMappedTypeNode(node) {
+        return node.kind === 190 /* MappedType */;
+    }
+    ts.isMappedTypeNode = isMappedTypeNode;
+    function isLiteralTypeNode(node) {
+        return node.kind === 191 /* LiteralType */;
+    }
+    ts.isLiteralTypeNode = isLiteralTypeNode;
+    function isImportTypeNode(node) {
+        return node.kind === 195 /* ImportType */;
+    }
+    ts.isImportTypeNode = isImportTypeNode;
+    function isTemplateLiteralTypeSpan(node) {
+        return node.kind === 194 /* TemplateLiteralTypeSpan */;
+    }
+    ts.isTemplateLiteralTypeSpan = isTemplateLiteralTypeSpan;
+    function isTemplateLiteralTypeNode(node) {
+        return node.kind === 193 /* TemplateLiteralType */;
+    }
+    ts.isTemplateLiteralTypeNode = isTemplateLiteralTypeNode;
+    // Binding patterns
+    function isObjectBindingPattern(node) {
+        return node.kind === 196 /* ObjectBindingPattern */;
+    }
+    ts.isObjectBindingPattern = isObjectBindingPattern;
+    function isArrayBindingPattern(node) {
+        return node.kind === 197 /* ArrayBindingPattern */;
+    }
+    ts.isArrayBindingPattern = isArrayBindingPattern;
+    function isBindingElement(node) {
+        return node.kind === 198 /* BindingElement */;
+    }
+    ts.isBindingElement = isBindingElement;
+    // Expression
+    function isArrayLiteralExpression(node) {
+        return node.kind === 199 /* ArrayLiteralExpression */;
+    }
+    ts.isArrayLiteralExpression = isArrayLiteralExpression;
+    function isObjectLiteralExpression(node) {
+        return node.kind === 200 /* ObjectLiteralExpression */;
+    }
+    ts.isObjectLiteralExpression = isObjectLiteralExpression;
+    function isPropertyAccessExpression(node) {
+        return node.kind === 201 /* PropertyAccessExpression */;
+    }
+    ts.isPropertyAccessExpression = isPropertyAccessExpression;
+    function isElementAccessExpression(node) {
+        return node.kind === 202 /* ElementAccessExpression */;
+    }
+    ts.isElementAccessExpression = isElementAccessExpression;
+    function isCallExpression(node) {
+        return node.kind === 203 /* CallExpression */;
+    }
+    ts.isCallExpression = isCallExpression;
+    function isNewExpression(node) {
+        return node.kind === 204 /* NewExpression */;
+    }
+    ts.isNewExpression = isNewExpression;
+    function isTaggedTemplateExpression(node) {
+        return node.kind === 205 /* TaggedTemplateExpression */;
+    }
+    ts.isTaggedTemplateExpression = isTaggedTemplateExpression;
+    function isTypeAssertionExpression(node) {
+        return node.kind === 206 /* TypeAssertionExpression */;
+    }
+    ts.isTypeAssertionExpression = isTypeAssertionExpression;
+    function isParenthesizedExpression(node) {
+        return node.kind === 207 /* ParenthesizedExpression */;
+    }
+    ts.isParenthesizedExpression = isParenthesizedExpression;
+    function isFunctionExpression(node) {
+        return node.kind === 208 /* FunctionExpression */;
+    }
+    ts.isFunctionExpression = isFunctionExpression;
+    function isArrowFunction(node) {
+        return node.kind === 209 /* ArrowFunction */;
+    }
+    ts.isArrowFunction = isArrowFunction;
+    function isDeleteExpression(node) {
+        return node.kind === 210 /* DeleteExpression */;
+    }
+    ts.isDeleteExpression = isDeleteExpression;
+    function isTypeOfExpression(node) {
+        return node.kind === 211 /* TypeOfExpression */;
+    }
+    ts.isTypeOfExpression = isTypeOfExpression;
+    function isVoidExpression(node) {
+        return node.kind === 212 /* VoidExpression */;
+    }
+    ts.isVoidExpression = isVoidExpression;
+    function isAwaitExpression(node) {
+        return node.kind === 213 /* AwaitExpression */;
+    }
+    ts.isAwaitExpression = isAwaitExpression;
+    function isPrefixUnaryExpression(node) {
+        return node.kind === 214 /* PrefixUnaryExpression */;
+    }
+    ts.isPrefixUnaryExpression = isPrefixUnaryExpression;
+    function isPostfixUnaryExpression(node) {
+        return node.kind === 215 /* PostfixUnaryExpression */;
+    }
+    ts.isPostfixUnaryExpression = isPostfixUnaryExpression;
+    function isBinaryExpression(node) {
+        return node.kind === 216 /* BinaryExpression */;
+    }
+    ts.isBinaryExpression = isBinaryExpression;
+    function isConditionalExpression(node) {
+        return node.kind === 217 /* ConditionalExpression */;
+    }
+    ts.isConditionalExpression = isConditionalExpression;
+    function isTemplateExpression(node) {
+        return node.kind === 218 /* TemplateExpression */;
+    }
+    ts.isTemplateExpression = isTemplateExpression;
+    function isYieldExpression(node) {
+        return node.kind === 219 /* YieldExpression */;
+    }
+    ts.isYieldExpression = isYieldExpression;
+    function isSpreadElement(node) {
+        return node.kind === 220 /* SpreadElement */;
+    }
+    ts.isSpreadElement = isSpreadElement;
+    function isClassExpression(node) {
+        return node.kind === 221 /* ClassExpression */;
+    }
+    ts.isClassExpression = isClassExpression;
+    function isOmittedExpression(node) {
+        return node.kind === 222 /* OmittedExpression */;
+    }
+    ts.isOmittedExpression = isOmittedExpression;
+    function isExpressionWithTypeArguments(node) {
+        return node.kind === 223 /* ExpressionWithTypeArguments */;
+    }
+    ts.isExpressionWithTypeArguments = isExpressionWithTypeArguments;
+    function isAsExpression(node) {
+        return node.kind === 224 /* AsExpression */;
+    }
+    ts.isAsExpression = isAsExpression;
+    function isNonNullExpression(node) {
+        return node.kind === 225 /* NonNullExpression */;
+    }
+    ts.isNonNullExpression = isNonNullExpression;
+    function isMetaProperty(node) {
+        return node.kind === 226 /* MetaProperty */;
+    }
+    ts.isMetaProperty = isMetaProperty;
+    function isSyntheticExpression(node) {
+        return node.kind === 227 /* SyntheticExpression */;
+    }
+    ts.isSyntheticExpression = isSyntheticExpression;
+    function isPartiallyEmittedExpression(node) {
+        return node.kind === 336 /* PartiallyEmittedExpression */;
+    }
+    ts.isPartiallyEmittedExpression = isPartiallyEmittedExpression;
+    function isCommaListExpression(node) {
+        return node.kind === 337 /* CommaListExpression */;
+    }
+    ts.isCommaListExpression = isCommaListExpression;
+    // Misc
+    function isTemplateSpan(node) {
+        return node.kind === 228 /* TemplateSpan */;
+    }
+    ts.isTemplateSpan = isTemplateSpan;
+    function isSemicolonClassElement(node) {
+        return node.kind === 229 /* SemicolonClassElement */;
+    }
+    ts.isSemicolonClassElement = isSemicolonClassElement;
+    // Elements
+    function isBlock(node) {
+        return node.kind === 230 /* Block */;
+    }
+    ts.isBlock = isBlock;
+    function isVariableStatement(node) {
+        return node.kind === 232 /* VariableStatement */;
+    }
+    ts.isVariableStatement = isVariableStatement;
+    function isEmptyStatement(node) {
+        return node.kind === 231 /* EmptyStatement */;
+    }
+    ts.isEmptyStatement = isEmptyStatement;
+    function isExpressionStatement(node) {
+        return node.kind === 233 /* ExpressionStatement */;
+    }
+    ts.isExpressionStatement = isExpressionStatement;
+    function isIfStatement(node) {
+        return node.kind === 234 /* IfStatement */;
+    }
+    ts.isIfStatement = isIfStatement;
+    function isDoStatement(node) {
+        return node.kind === 235 /* DoStatement */;
+    }
+    ts.isDoStatement = isDoStatement;
+    function isWhileStatement(node) {
+        return node.kind === 236 /* WhileStatement */;
+    }
+    ts.isWhileStatement = isWhileStatement;
+    function isForStatement(node) {
+        return node.kind === 237 /* ForStatement */;
+    }
+    ts.isForStatement = isForStatement;
+    function isForInStatement(node) {
+        return node.kind === 238 /* ForInStatement */;
+    }
+    ts.isForInStatement = isForInStatement;
+    function isForOfStatement(node) {
+        return node.kind === 239 /* ForOfStatement */;
+    }
+    ts.isForOfStatement = isForOfStatement;
+    function isContinueStatement(node) {
+        return node.kind === 240 /* ContinueStatement */;
+    }
+    ts.isContinueStatement = isContinueStatement;
+    function isBreakStatement(node) {
+        return node.kind === 241 /* BreakStatement */;
+    }
+    ts.isBreakStatement = isBreakStatement;
+    function isReturnStatement(node) {
+        return node.kind === 242 /* ReturnStatement */;
+    }
+    ts.isReturnStatement = isReturnStatement;
+    function isWithStatement(node) {
+        return node.kind === 243 /* WithStatement */;
+    }
+    ts.isWithStatement = isWithStatement;
+    function isSwitchStatement(node) {
+        return node.kind === 244 /* SwitchStatement */;
+    }
+    ts.isSwitchStatement = isSwitchStatement;
+    function isLabeledStatement(node) {
+        return node.kind === 245 /* LabeledStatement */;
+    }
+    ts.isLabeledStatement = isLabeledStatement;
+    function isThrowStatement(node) {
+        return node.kind === 246 /* ThrowStatement */;
+    }
+    ts.isThrowStatement = isThrowStatement;
+    function isTryStatement(node) {
+        return node.kind === 247 /* TryStatement */;
+    }
+    ts.isTryStatement = isTryStatement;
+    function isDebuggerStatement(node) {
+        return node.kind === 248 /* DebuggerStatement */;
+    }
+    ts.isDebuggerStatement = isDebuggerStatement;
+    function isVariableDeclaration(node) {
+        return node.kind === 249 /* VariableDeclaration */;
+    }
+    ts.isVariableDeclaration = isVariableDeclaration;
+    function isVariableDeclarationList(node) {
+        return node.kind === 250 /* VariableDeclarationList */;
+    }
+    ts.isVariableDeclarationList = isVariableDeclarationList;
+    function isFunctionDeclaration(node) {
+        return node.kind === 251 /* FunctionDeclaration */;
+    }
+    ts.isFunctionDeclaration = isFunctionDeclaration;
+    function isClassDeclaration(node) {
+        return node.kind === 252 /* ClassDeclaration */;
+    }
+    ts.isClassDeclaration = isClassDeclaration;
+    function isInterfaceDeclaration(node) {
+        return node.kind === 253 /* InterfaceDeclaration */;
+    }
+    ts.isInterfaceDeclaration = isInterfaceDeclaration;
+    function isTypeAliasDeclaration(node) {
+        return node.kind === 254 /* TypeAliasDeclaration */;
+    }
+    ts.isTypeAliasDeclaration = isTypeAliasDeclaration;
+    function isEnumDeclaration(node) {
+        return node.kind === 255 /* EnumDeclaration */;
+    }
+    ts.isEnumDeclaration = isEnumDeclaration;
+    function isModuleDeclaration(node) {
+        return node.kind === 256 /* ModuleDeclaration */;
+    }
+    ts.isModuleDeclaration = isModuleDeclaration;
+    function isModuleBlock(node) {
+        return node.kind === 257 /* ModuleBlock */;
+    }
+    ts.isModuleBlock = isModuleBlock;
+    function isCaseBlock(node) {
+        return node.kind === 258 /* CaseBlock */;
+    }
+    ts.isCaseBlock = isCaseBlock;
+    function isNamespaceExportDeclaration(node) {
+        return node.kind === 259 /* NamespaceExportDeclaration */;
+    }
+    ts.isNamespaceExportDeclaration = isNamespaceExportDeclaration;
+    function isImportEqualsDeclaration(node) {
+        return node.kind === 260 /* ImportEqualsDeclaration */;
+    }
+    ts.isImportEqualsDeclaration = isImportEqualsDeclaration;
+    function isImportDeclaration(node) {
+        return node.kind === 261 /* ImportDeclaration */;
+    }
+    ts.isImportDeclaration = isImportDeclaration;
+    function isImportClause(node) {
+        return node.kind === 262 /* ImportClause */;
+    }
+    ts.isImportClause = isImportClause;
+    function isNamespaceImport(node) {
+        return node.kind === 263 /* NamespaceImport */;
+    }
+    ts.isNamespaceImport = isNamespaceImport;
+    function isNamespaceExport(node) {
+        return node.kind === 269 /* NamespaceExport */;
+    }
+    ts.isNamespaceExport = isNamespaceExport;
+    function isNamedImports(node) {
+        return node.kind === 264 /* NamedImports */;
+    }
+    ts.isNamedImports = isNamedImports;
+    function isImportSpecifier(node) {
+        return node.kind === 265 /* ImportSpecifier */;
+    }
+    ts.isImportSpecifier = isImportSpecifier;
+    function isExportAssignment(node) {
+        return node.kind === 266 /* ExportAssignment */;
+    }
+    ts.isExportAssignment = isExportAssignment;
+    function isExportDeclaration(node) {
+        return node.kind === 267 /* ExportDeclaration */;
+    }
+    ts.isExportDeclaration = isExportDeclaration;
+    function isNamedExports(node) {
+        return node.kind === 268 /* NamedExports */;
+    }
+    ts.isNamedExports = isNamedExports;
+    function isExportSpecifier(node) {
+        return node.kind === 270 /* ExportSpecifier */;
+    }
+    ts.isExportSpecifier = isExportSpecifier;
+    function isMissingDeclaration(node) {
+        return node.kind === 271 /* MissingDeclaration */;
+    }
+    ts.isMissingDeclaration = isMissingDeclaration;
+    function isNotEmittedStatement(node) {
+        return node.kind === 335 /* NotEmittedStatement */;
+    }
+    ts.isNotEmittedStatement = isNotEmittedStatement;
+    /* @internal */
+    function isSyntheticReference(node) {
+        return node.kind === 340 /* SyntheticReferenceExpression */;
+    }
+    ts.isSyntheticReference = isSyntheticReference;
+    /* @internal */
+    function isMergeDeclarationMarker(node) {
+        return node.kind === 338 /* MergeDeclarationMarker */;
+    }
+    ts.isMergeDeclarationMarker = isMergeDeclarationMarker;
+    /* @internal */
+    function isEndOfDeclarationMarker(node) {
+        return node.kind === 339 /* EndOfDeclarationMarker */;
+    }
+    ts.isEndOfDeclarationMarker = isEndOfDeclarationMarker;
+    // Module References
+    function isExternalModuleReference(node) {
+        return node.kind === 272 /* ExternalModuleReference */;
+    }
+    ts.isExternalModuleReference = isExternalModuleReference;
+    // JSX
+    function isJsxElement(node) {
+        return node.kind === 273 /* JsxElement */;
+    }
+    ts.isJsxElement = isJsxElement;
+    function isJsxSelfClosingElement(node) {
+        return node.kind === 274 /* JsxSelfClosingElement */;
+    }
+    ts.isJsxSelfClosingElement = isJsxSelfClosingElement;
+    function isJsxOpeningElement(node) {
+        return node.kind === 275 /* JsxOpeningElement */;
+    }
+    ts.isJsxOpeningElement = isJsxOpeningElement;
+    function isJsxClosingElement(node) {
+        return node.kind === 276 /* JsxClosingElement */;
+    }
+    ts.isJsxClosingElement = isJsxClosingElement;
+    function isJsxFragment(node) {
+        return node.kind === 277 /* JsxFragment */;
+    }
+    ts.isJsxFragment = isJsxFragment;
+    function isJsxOpeningFragment(node) {
+        return node.kind === 278 /* JsxOpeningFragment */;
+    }
+    ts.isJsxOpeningFragment = isJsxOpeningFragment;
+    function isJsxClosingFragment(node) {
+        return node.kind === 279 /* JsxClosingFragment */;
+    }
+    ts.isJsxClosingFragment = isJsxClosingFragment;
+    function isJsxAttribute(node) {
+        return node.kind === 280 /* JsxAttribute */;
+    }
+    ts.isJsxAttribute = isJsxAttribute;
+    function isJsxAttributes(node) {
+        return node.kind === 281 /* JsxAttributes */;
+    }
+    ts.isJsxAttributes = isJsxAttributes;
+    function isJsxSpreadAttribute(node) {
+        return node.kind === 282 /* JsxSpreadAttribute */;
+    }
+    ts.isJsxSpreadAttribute = isJsxSpreadAttribute;
+    function isJsxExpression(node) {
+        return node.kind === 283 /* JsxExpression */;
+    }
+    ts.isJsxExpression = isJsxExpression;
+    // Clauses
+    function isCaseClause(node) {
+        return node.kind === 284 /* CaseClause */;
+    }
+    ts.isCaseClause = isCaseClause;
+    function isDefaultClause(node) {
+        return node.kind === 285 /* DefaultClause */;
+    }
+    ts.isDefaultClause = isDefaultClause;
+    function isHeritageClause(node) {
+        return node.kind === 286 /* HeritageClause */;
+    }
+    ts.isHeritageClause = isHeritageClause;
+    function isCatchClause(node) {
+        return node.kind === 287 /* CatchClause */;
+    }
+    ts.isCatchClause = isCatchClause;
+    // Property assignments
+    function isPropertyAssignment(node) {
+        return node.kind === 288 /* PropertyAssignment */;
+    }
+    ts.isPropertyAssignment = isPropertyAssignment;
+    function isShorthandPropertyAssignment(node) {
+        return node.kind === 289 /* ShorthandPropertyAssignment */;
+    }
+    ts.isShorthandPropertyAssignment = isShorthandPropertyAssignment;
+    function isSpreadAssignment(node) {
+        return node.kind === 290 /* SpreadAssignment */;
+    }
+    ts.isSpreadAssignment = isSpreadAssignment;
+    // Enum
+    function isEnumMember(node) {
+        return node.kind === 291 /* EnumMember */;
+    }
+    ts.isEnumMember = isEnumMember;
+    // Unparsed
+    // TODO(rbuckton): isUnparsedPrologue
+    function isUnparsedPrepend(node) {
+        return node.kind === 293 /* UnparsedPrepend */;
+    }
+    ts.isUnparsedPrepend = isUnparsedPrepend;
+    // TODO(rbuckton): isUnparsedText
+    // TODO(rbuckton): isUnparsedInternalText
+    // TODO(rbuckton): isUnparsedSyntheticReference
+    // Top-level nodes
+    function isSourceFile(node) {
+        return node.kind === 297 /* SourceFile */;
+    }
+    ts.isSourceFile = isSourceFile;
+    function isBundle(node) {
+        return node.kind === 298 /* Bundle */;
+    }
+    ts.isBundle = isBundle;
+    function isUnparsedSource(node) {
+        return node.kind === 299 /* UnparsedSource */;
+    }
+    ts.isUnparsedSource = isUnparsedSource;
+    // TODO(rbuckton): isInputFiles
+    // JSDoc Elements
+    function isJSDocTypeExpression(node) {
+        return node.kind === 301 /* JSDocTypeExpression */;
+    }
+    ts.isJSDocTypeExpression = isJSDocTypeExpression;
+    function isJSDocNameReference(node) {
+        return node.kind === 302 /* JSDocNameReference */;
+    }
+    ts.isJSDocNameReference = isJSDocNameReference;
+    function isJSDocAllType(node) {
+        return node.kind === 303 /* JSDocAllType */;
+    }
+    ts.isJSDocAllType = isJSDocAllType;
+    function isJSDocUnknownType(node) {
+        return node.kind === 304 /* JSDocUnknownType */;
+    }
+    ts.isJSDocUnknownType = isJSDocUnknownType;
+    function isJSDocNullableType(node) {
+        return node.kind === 305 /* JSDocNullableType */;
+    }
+    ts.isJSDocNullableType = isJSDocNullableType;
+    function isJSDocNonNullableType(node) {
+        return node.kind === 306 /* JSDocNonNullableType */;
+    }
+    ts.isJSDocNonNullableType = isJSDocNonNullableType;
+    function isJSDocOptionalType(node) {
+        return node.kind === 307 /* JSDocOptionalType */;
+    }
+    ts.isJSDocOptionalType = isJSDocOptionalType;
+    function isJSDocFunctionType(node) {
+        return node.kind === 308 /* JSDocFunctionType */;
+    }
+    ts.isJSDocFunctionType = isJSDocFunctionType;
+    function isJSDocVariadicType(node) {
+        return node.kind === 309 /* JSDocVariadicType */;
+    }
+    ts.isJSDocVariadicType = isJSDocVariadicType;
+    function isJSDocNamepathType(node) {
+        return node.kind === 310 /* JSDocNamepathType */;
+    }
+    ts.isJSDocNamepathType = isJSDocNamepathType;
+    function isJSDoc(node) {
+        return node.kind === 311 /* JSDocComment */;
+    }
+    ts.isJSDoc = isJSDoc;
+    function isJSDocTypeLiteral(node) {
+        return node.kind === 312 /* JSDocTypeLiteral */;
+    }
+    ts.isJSDocTypeLiteral = isJSDocTypeLiteral;
+    function isJSDocSignature(node) {
+        return node.kind === 313 /* JSDocSignature */;
+    }
+    ts.isJSDocSignature = isJSDocSignature;
+    // JSDoc Tags
+    function isJSDocAugmentsTag(node) {
+        return node.kind === 315 /* JSDocAugmentsTag */;
+    }
+    ts.isJSDocAugmentsTag = isJSDocAugmentsTag;
+    function isJSDocAuthorTag(node) {
+        return node.kind === 317 /* JSDocAuthorTag */;
+    }
+    ts.isJSDocAuthorTag = isJSDocAuthorTag;
+    function isJSDocClassTag(node) {
+        return node.kind === 319 /* JSDocClassTag */;
+    }
+    ts.isJSDocClassTag = isJSDocClassTag;
+    function isJSDocCallbackTag(node) {
+        return node.kind === 324 /* JSDocCallbackTag */;
+    }
+    ts.isJSDocCallbackTag = isJSDocCallbackTag;
+    function isJSDocPublicTag(node) {
+        return node.kind === 320 /* JSDocPublicTag */;
+    }
+    ts.isJSDocPublicTag = isJSDocPublicTag;
+    function isJSDocPrivateTag(node) {
+        return node.kind === 321 /* JSDocPrivateTag */;
+    }
+    ts.isJSDocPrivateTag = isJSDocPrivateTag;
+    function isJSDocProtectedTag(node) {
+        return node.kind === 322 /* JSDocProtectedTag */;
+    }
+    ts.isJSDocProtectedTag = isJSDocProtectedTag;
+    function isJSDocReadonlyTag(node) {
+        return node.kind === 323 /* JSDocReadonlyTag */;
+    }
+    ts.isJSDocReadonlyTag = isJSDocReadonlyTag;
+    function isJSDocDeprecatedTag(node) {
+        return node.kind === 318 /* JSDocDeprecatedTag */;
+    }
+    ts.isJSDocDeprecatedTag = isJSDocDeprecatedTag;
+    function isJSDocEnumTag(node) {
+        return node.kind === 325 /* JSDocEnumTag */;
+    }
+    ts.isJSDocEnumTag = isJSDocEnumTag;
+    function isJSDocParameterTag(node) {
+        return node.kind === 326 /* JSDocParameterTag */;
+    }
+    ts.isJSDocParameterTag = isJSDocParameterTag;
+    function isJSDocReturnTag(node) {
+        return node.kind === 327 /* JSDocReturnTag */;
+    }
+    ts.isJSDocReturnTag = isJSDocReturnTag;
+    function isJSDocThisTag(node) {
+        return node.kind === 328 /* JSDocThisTag */;
+    }
+    ts.isJSDocThisTag = isJSDocThisTag;
+    function isJSDocTypeTag(node) {
+        return node.kind === 329 /* JSDocTypeTag */;
+    }
+    ts.isJSDocTypeTag = isJSDocTypeTag;
+    function isJSDocTemplateTag(node) {
+        return node.kind === 330 /* JSDocTemplateTag */;
+    }
+    ts.isJSDocTemplateTag = isJSDocTemplateTag;
+    function isJSDocTypedefTag(node) {
+        return node.kind === 331 /* JSDocTypedefTag */;
+    }
+    ts.isJSDocTypedefTag = isJSDocTypedefTag;
+    function isJSDocUnknownTag(node) {
+        return node.kind === 314 /* JSDocTag */;
+    }
+    ts.isJSDocUnknownTag = isJSDocUnknownTag;
+    function isJSDocPropertyTag(node) {
+        return node.kind === 333 /* JSDocPropertyTag */;
+    }
+    ts.isJSDocPropertyTag = isJSDocPropertyTag;
+    function isJSDocImplementsTag(node) {
+        return node.kind === 316 /* JSDocImplementsTag */;
+    }
+    ts.isJSDocImplementsTag = isJSDocImplementsTag;
+    // Synthesized list
+    /* @internal */
+    function isSyntaxList(n) {
+        return n.kind === 334 /* SyntaxList */;
+    }
+    ts.isSyntaxList = isSyntaxList;
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    // Compound nodes
+    function createEmptyExports(factory) {
+        return factory.createExportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*isTypeOnly*/ false, factory.createNamedExports([]), /*moduleSpecifier*/ undefined);
+    }
+    ts.createEmptyExports = createEmptyExports;
+    function createMemberAccessForPropertyName(factory, target, memberName, location) {
+        if (ts.isComputedPropertyName(memberName)) {
+            return ts.setTextRange(factory.createElementAccessExpression(target, memberName.expression), location);
         }
-        function tryParseConstructorDeclaration(node) {
-            return tryParse(function () {
-                if (parseConstructorName()) {
-                    node.kind = 162 /* Constructor */;
-                    fillSignature(58 /* ColonToken */, 0 /* None */, node);
-                    node.body = parseFunctionBlockOrSemicolon(0 /* None */, ts.Diagnostics.or_expected);
-                    return finishNode(node);
-                }
-            });
+        else {
+            var expression = ts.setTextRange(ts.isIdentifierOrPrivateIdentifier(memberName)
+                ? factory.createPropertyAccessExpression(target, memberName)
+                : factory.createElementAccessExpression(target, memberName), memberName);
+            ts.getOrCreateEmitNode(expression).flags |= 64 /* NoNestedSourceMaps */;
+            return expression;
         }
-        function parseMethodDeclaration(node, asteriskToken, diagnosticMessage) {
-            node.kind = 161 /* MethodDeclaration */;
-            node.asteriskToken = asteriskToken;
-            var isGenerator = asteriskToken ? 1 /* Yield */ : 0 /* None */;
-            var isAsync = hasModifierOfKind(node, 126 /* AsyncKeyword */) ? 2 /* Await */ : 0 /* None */;
-            fillSignature(58 /* ColonToken */, isGenerator | isAsync, node);
-            node.body = parseFunctionBlockOrSemicolon(isGenerator | isAsync, diagnosticMessage);
-            return finishNode(node);
-        }
-        function parsePropertyDeclaration(node) {
-            node.kind = 159 /* PropertyDeclaration */;
-            if (!node.questionToken && token() === 53 /* ExclamationToken */ && !scanner.hasPrecedingLineBreak()) {
-                node.exclamationToken = parseTokenNode();
-            }
-            node.type = parseTypeAnnotation();
-            node.initializer = doOutsideOfContext(8192 /* YieldContext */ | 32768 /* AwaitContext */ | 4096 /* DisallowInContext */, parseInitializer);
-            parseSemicolon();
-            return finishNode(node);
+    }
+    ts.createMemberAccessForPropertyName = createMemberAccessForPropertyName;
+    function createReactNamespace(reactNamespace, parent) {
+        // To ensure the emit resolver can properly resolve the namespace, we need to
+        // treat this identifier as if it were a source tree node by clearing the `Synthesized`
+        // flag and setting a parent node.
+        var react = ts.parseNodeFactory.createIdentifier(reactNamespace || "React");
+        // Set the parent that is in parse tree
+        // this makes sure that parent chain is intact for checker to traverse complete scope tree
+        ts.setParent(react, ts.getParseTreeNode(parent));
+        return react;
+    }
+    function createJsxFactoryExpressionFromEntityName(factory, jsxFactory, parent) {
+        if (ts.isQualifiedName(jsxFactory)) {
+            var left = createJsxFactoryExpressionFromEntityName(factory, jsxFactory.left, parent);
+            var right = factory.createIdentifier(ts.idText(jsxFactory.right));
+            right.escapedText = jsxFactory.right.escapedText;
+            return factory.createPropertyAccessExpression(left, right);
         }
-        function parsePropertyOrMethodDeclaration(node) {
-            var asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
-            node.name = parsePropertyName();
-            // Note: this is not legal as per the grammar.  But we allow it in the parser and
-            // report an error in the grammar checker.
-            node.questionToken = parseOptionalToken(57 /* QuestionToken */);
-            if (asteriskToken || token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
-                return parseMethodDeclaration(node, asteriskToken, ts.Diagnostics.or_expected);
-            }
-            return parsePropertyDeclaration(node);
+        else {
+            return createReactNamespace(ts.idText(jsxFactory), parent);
         }
-        function parseAccessorDeclaration(node, kind) {
-            node.kind = kind;
-            node.name = parsePropertyName();
-            fillSignature(58 /* ColonToken */, 0 /* None */, node);
-            node.body = parseFunctionBlockOrSemicolon(0 /* None */);
-            return finishNode(node);
+    }
+    function createJsxFactoryExpression(factory, jsxFactoryEntity, reactNamespace, parent) {
+        return jsxFactoryEntity ?
+            createJsxFactoryExpressionFromEntityName(factory, jsxFactoryEntity, parent) :
+            factory.createPropertyAccessExpression(createReactNamespace(reactNamespace, parent), "createElement");
+    }
+    ts.createJsxFactoryExpression = createJsxFactoryExpression;
+    function createJsxFragmentFactoryExpression(factory, jsxFragmentFactoryEntity, reactNamespace, parent) {
+        return jsxFragmentFactoryEntity ?
+            createJsxFactoryExpressionFromEntityName(factory, jsxFragmentFactoryEntity, parent) :
+            factory.createPropertyAccessExpression(createReactNamespace(reactNamespace, parent), "Fragment");
+    }
+    function createExpressionForJsxElement(factory, callee, tagName, props, children, location) {
+        var argumentsList = [tagName];
+        if (props) {
+            argumentsList.push(props);
         }
-        function isClassMemberStart() {
-            var idToken;
-            if (token() === 59 /* AtToken */) {
-                return true;
+        if (children && children.length > 0) {
+            if (!props) {
+                argumentsList.push(factory.createNull());
             }
-            // Eat up all modifiers, but hold on to the last one in case it is actually an identifier.
-            while (ts.isModifierKind(token())) {
-                idToken = token();
-                // If the idToken is a class modifier (protected, private, public, and static), it is
-                // certain that we are starting to parse class member. This allows better error recovery
-                // Example:
-                //      public foo() ...     // true
-                //      public @dec blah ... // true; we will then report an error later
-                //      export public ...    // true; we will then report an error later
-                if (ts.isClassMemberModifier(idToken)) {
-                    return true;
+            if (children.length > 1) {
+                for (var _i = 0, children_3 = children; _i < children_3.length; _i++) {
+                    var child = children_3[_i];
+                    startOnNewLine(child);
+                    argumentsList.push(child);
                 }
-                nextToken();
-            }
-            if (token() === 41 /* AsteriskToken */) {
-                return true;
-            }
-            // Try to get the first property-like token following all modifiers.
-            // This can either be an identifier or the 'get' or 'set' keywords.
-            if (isLiteralPropertyName()) {
-                idToken = token();
-                nextToken();
             }
-            // Index signatures and computed properties are class members; we can parse.
-            if (token() === 22 /* OpenBracketToken */) {
-                return true;
-            }
-            // If we were able to get any potential identifier...
-            if (idToken !== undefined) {
-                // If we have a non-keyword identifier, or if we have an accessor, then it's safe to parse.
-                if (!ts.isKeyword(idToken) || idToken === 142 /* SetKeyword */ || idToken === 131 /* GetKeyword */) {
-                    return true;
-                }
-                // If it *is* a keyword, but not an accessor, check a little farther along
-                // to see if it should actually be parsed as a class member.
-                switch (token()) {
-                    case 20 /* OpenParenToken */: // Method declaration
-                    case 29 /* LessThanToken */: // Generic Method declaration
-                    case 53 /* ExclamationToken */: // Non-null assertion on property name
-                    case 58 /* ColonToken */: // Type Annotation for declaration
-                    case 62 /* EqualsToken */: // Initializer for declaration
-                    case 57 /* QuestionToken */: // Not valid, but permitted so that it gets caught later on.
-                        return true;
-                    default:
-                        // Covers
-                        //  - Semicolons     (declaration termination)
-                        //  - Closing braces (end-of-class, must be declaration)
-                        //  - End-of-files   (not valid, but permitted so that it gets caught later on)
-                        //  - Line-breaks    (enabling *automatic semicolon insertion*)
-                        return canParseSemicolon();
-                }
+            else {
+                argumentsList.push(children[0]);
             }
-            return false;
         }
-        function parseDecorators() {
-            var list;
-            var listPos = getNodePos();
-            while (true) {
-                var decoratorStart = getNodePos();
-                if (!parseOptional(59 /* AtToken */)) {
-                    break;
+        return ts.setTextRange(factory.createCallExpression(callee, 
+        /*typeArguments*/ undefined, argumentsList), location);
+    }
+    ts.createExpressionForJsxElement = createExpressionForJsxElement;
+    function createExpressionForJsxFragment(factory, jsxFactoryEntity, jsxFragmentFactoryEntity, reactNamespace, children, parentElement, location) {
+        var tagName = createJsxFragmentFactoryExpression(factory, jsxFragmentFactoryEntity, reactNamespace, parentElement);
+        var argumentsList = [tagName, factory.createNull()];
+        if (children && children.length > 0) {
+            if (children.length > 1) {
+                for (var _i = 0, children_4 = children; _i < children_4.length; _i++) {
+                    var child = children_4[_i];
+                    startOnNewLine(child);
+                    argumentsList.push(child);
                 }
-                var decorator = createNode(157 /* Decorator */, decoratorStart);
-                decorator.expression = doInDecoratorContext(parseLeftHandSideExpressionOrHigher);
-                finishNode(decorator);
-                (list || (list = [])).push(decorator);
             }
-            return list && createNodeArray(list, listPos);
-        }
-        /*
-         * There are situations in which a modifier like 'const' will appear unexpectedly, such as on a class member.
-         * In those situations, if we are entirely sure that 'const' is not valid on its own (such as when ASI takes effect
-         * and turns it into a standalone declaration), then it is better to parse it and report an error later.
-         *
-         * In such situations, 'permitInvalidConstAsModifier' should be set to true.
-         */
-        function parseModifiers(permitInvalidConstAsModifier) {
-            var list;
-            var listPos = getNodePos();
-            while (true) {
-                var modifierStart = scanner.getStartPos();
-                var modifierKind = token();
-                if (token() === 81 /* ConstKeyword */ && permitInvalidConstAsModifier) {
-                    // We need to ensure that any subsequent modifiers appear on the same line
-                    // so that when 'const' is a standalone declaration, we don't issue an error.
-                    if (!tryParse(nextTokenIsOnSameLineAndCanFollowModifier)) {
-                        break;
-                    }
-                }
-                else {
-                    if (!parseAnyContextualModifier()) {
-                        break;
-                    }
-                }
-                var modifier = finishNode(createNode(modifierKind, modifierStart));
-                (list || (list = [])).push(modifier);
+            else {
+                argumentsList.push(children[0]);
             }
-            return list && createNodeArray(list, listPos);
         }
-        function parseModifiersForArrowFunction() {
-            var modifiers;
-            if (token() === 126 /* AsyncKeyword */) {
-                var modifierStart = scanner.getStartPos();
-                var modifierKind = token();
-                nextToken();
-                var modifier = finishNode(createNode(modifierKind, modifierStart));
-                modifiers = createNodeArray([modifier], modifierStart);
-            }
-            return modifiers;
+        return ts.setTextRange(factory.createCallExpression(createJsxFactoryExpression(factory, jsxFactoryEntity, reactNamespace, parentElement), 
+        /*typeArguments*/ undefined, argumentsList), location);
+    }
+    ts.createExpressionForJsxFragment = createExpressionForJsxFragment;
+    // Utilities
+    function createForOfBindingStatement(factory, node, boundValue) {
+        if (ts.isVariableDeclarationList(node)) {
+            var firstDeclaration = ts.first(node.declarations);
+            var updatedDeclaration = factory.updateVariableDeclaration(firstDeclaration, firstDeclaration.name, 
+            /*exclamationToken*/ undefined, 
+            /*type*/ undefined, boundValue);
+            return ts.setTextRange(factory.createVariableStatement(
+            /*modifiers*/ undefined, factory.updateVariableDeclarationList(node, [updatedDeclaration])), 
+            /*location*/ node);
         }
-        function parseClassElement() {
-            if (token() === 26 /* SemicolonToken */) {
-                var result = createNode(222 /* SemicolonClassElement */);
-                nextToken();
-                return finishNode(result);
-            }
-            var node = createNodeWithJSDoc(0 /* Unknown */);
-            node.decorators = parseDecorators();
-            node.modifiers = parseModifiers(/*permitInvalidConstAsModifier*/ true);
-            if (parseContextualModifier(131 /* GetKeyword */)) {
-                return parseAccessorDeclaration(node, 163 /* GetAccessor */);
-            }
-            if (parseContextualModifier(142 /* SetKeyword */)) {
-                return parseAccessorDeclaration(node, 164 /* SetAccessor */);
-            }
-            if (token() === 129 /* ConstructorKeyword */ || token() === 10 /* StringLiteral */) {
-                var constructorDeclaration = tryParseConstructorDeclaration(node);
-                if (constructorDeclaration) {
-                    return constructorDeclaration;
-                }
-            }
-            if (isIndexSignature()) {
-                return parseIndexSignatureDeclaration(node);
-            }
-            // It is very important that we check this *after* checking indexers because
-            // the [ token can start an index signature or a computed property name
-            if (ts.tokenIsIdentifierOrKeyword(token()) ||
-                token() === 10 /* StringLiteral */ ||
-                token() === 8 /* NumericLiteral */ ||
-                token() === 41 /* AsteriskToken */ ||
-                token() === 22 /* OpenBracketToken */) {
-                var isAmbient = node.modifiers && ts.some(node.modifiers, isDeclareModifier);
-                if (isAmbient) {
-                    for (var _i = 0, _a = node.modifiers; _i < _a.length; _i++) {
-                        var m = _a[_i];
-                        m.flags |= 8388608 /* Ambient */;
-                    }
-                    return doInsideOfContext(8388608 /* Ambient */, function () { return parsePropertyOrMethodDeclaration(node); });
-                }
-                else {
-                    return parsePropertyOrMethodDeclaration(node);
-                }
-            }
-            if (node.decorators || node.modifiers) {
-                // treat this as a property declaration with a missing name.
-                node.name = createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Declaration_expected);
-                return parsePropertyDeclaration(node);
-            }
-            // 'isClassMemberStart' should have hinted not to attempt parsing.
-            return ts.Debug.fail("Should not have attempted to parse class member declaration.");
+        else {
+            var updatedExpression = ts.setTextRange(factory.createAssignment(node, boundValue), /*location*/ node);
+            return ts.setTextRange(factory.createExpressionStatement(updatedExpression), /*location*/ node);
         }
-        function parseClassExpression() {
-            return parseClassDeclarationOrExpression(createNodeWithJSDoc(0 /* Unknown */), 214 /* ClassExpression */);
+    }
+    ts.createForOfBindingStatement = createForOfBindingStatement;
+    function insertLeadingStatement(factory, dest, source) {
+        if (ts.isBlock(dest)) {
+            return factory.updateBlock(dest, ts.setTextRange(factory.createNodeArray(__spreadArrays([source], dest.statements)), dest.statements));
         }
-        function parseClassDeclaration(node) {
-            return parseClassDeclarationOrExpression(node, 245 /* ClassDeclaration */);
+        else {
+            return factory.createBlock(factory.createNodeArray([dest, source]), /*multiLine*/ true);
         }
-        function parseClassDeclarationOrExpression(node, kind) {
-            node.kind = kind;
-            parseExpected(80 /* ClassKeyword */);
-            node.name = parseNameOfClassDeclarationOrExpression();
-            node.typeParameters = parseTypeParameters();
-            node.heritageClauses = parseHeritageClauses();
-            if (parseExpected(18 /* OpenBraceToken */)) {
-                // ClassTail[Yield,Await] : (Modified) See 14.5
-                //      ClassHeritage[?Yield,?Await]opt { ClassBody[?Yield,?Await]opt }
-                node.members = parseClassMembers();
-                parseExpected(19 /* CloseBraceToken */);
-            }
-            else {
-                node.members = createMissingList();
-            }
-            return finishNode(node);
-        }
-        function parseNameOfClassDeclarationOrExpression() {
-            // implements is a future reserved word so
-            // 'class implements' might mean either
-            // - class expression with omitted name, 'implements' starts heritage clause
-            // - class with name 'implements'
-            // 'isImplementsClause' helps to disambiguate between these two cases
-            return isIdentifier() && !isImplementsClause()
-                ? parseIdentifier()
-                : undefined;
-        }
-        function isImplementsClause() {
-            return token() === 113 /* ImplementsKeyword */ && lookAhead(nextTokenIsIdentifierOrKeyword);
+    }
+    ts.insertLeadingStatement = insertLeadingStatement;
+    function createExpressionFromEntityName(factory, node) {
+        if (ts.isQualifiedName(node)) {
+            var left = createExpressionFromEntityName(factory, node.left);
+            // TODO(rbuckton): Does this need to be parented?
+            var right = ts.setParent(ts.setTextRange(factory.cloneNode(node.right), node.right), node.right.parent);
+            return ts.setTextRange(factory.createPropertyAccessExpression(left, right), node);
         }
-        function parseHeritageClauses() {
-            // ClassTail[Yield,Await] : (Modified) See 14.5
-            //      ClassHeritage[?Yield,?Await]opt { ClassBody[?Yield,?Await]opt }
-            if (isHeritageClause()) {
-                return parseList(22 /* HeritageClauses */, parseHeritageClause);
-            }
-            return undefined;
+        else {
+            // TODO(rbuckton): Does this need to be parented?
+            return ts.setParent(ts.setTextRange(factory.cloneNode(node), node), node.parent);
         }
-        function parseHeritageClause() {
-            var tok = token();
-            ts.Debug.assert(tok === 90 /* ExtendsKeyword */ || tok === 113 /* ImplementsKeyword */); // isListElement() should ensure this.
-            var node = createNode(279 /* HeritageClause */);
-            node.token = tok;
-            nextToken();
-            node.types = parseDelimitedList(7 /* HeritageClauseElement */, parseExpressionWithTypeArguments);
-            return finishNode(node);
+    }
+    ts.createExpressionFromEntityName = createExpressionFromEntityName;
+    function createExpressionForPropertyName(factory, memberName) {
+        if (ts.isIdentifier(memberName)) {
+            return factory.createStringLiteralFromNode(memberName);
         }
-        function parseExpressionWithTypeArguments() {
-            var node = createNode(216 /* ExpressionWithTypeArguments */);
-            node.expression = parseLeftHandSideExpressionOrHigher();
-            node.typeArguments = tryParseTypeArguments();
-            return finishNode(node);
+        else if (ts.isComputedPropertyName(memberName)) {
+            // TODO(rbuckton): Does this need to be parented?
+            return ts.setParent(ts.setTextRange(factory.cloneNode(memberName.expression), memberName.expression), memberName.expression.parent);
         }
-        function tryParseTypeArguments() {
-            return token() === 29 /* LessThanToken */ ?
-                parseBracketedList(20 /* TypeArguments */, parseType, 29 /* LessThanToken */, 31 /* GreaterThanToken */) : undefined;
+        else {
+            // TODO(rbuckton): Does this need to be parented?
+            return ts.setParent(ts.setTextRange(factory.cloneNode(memberName), memberName), memberName.parent);
         }
-        function isHeritageClause() {
-            return token() === 90 /* ExtendsKeyword */ || token() === 113 /* ImplementsKeyword */;
+    }
+    ts.createExpressionForPropertyName = createExpressionForPropertyName;
+    function createExpressionForAccessorDeclaration(factory, properties, property, receiver, multiLine) {
+        var _a = ts.getAllAccessorDeclarations(properties, property), firstAccessor = _a.firstAccessor, getAccessor = _a.getAccessor, setAccessor = _a.setAccessor;
+        if (property === firstAccessor) {
+            return ts.setTextRange(factory.createObjectDefinePropertyCall(receiver, createExpressionForPropertyName(factory, property.name), factory.createPropertyDescriptor({
+                enumerable: factory.createFalse(),
+                configurable: true,
+                get: getAccessor && ts.setTextRange(ts.setOriginalNode(factory.createFunctionExpression(getAccessor.modifiers, 
+                /*asteriskToken*/ undefined, 
+                /*name*/ undefined, 
+                /*typeParameters*/ undefined, getAccessor.parameters, 
+                /*type*/ undefined, getAccessor.body // TODO: GH#18217
+                ), getAccessor), getAccessor),
+                set: setAccessor && ts.setTextRange(ts.setOriginalNode(factory.createFunctionExpression(setAccessor.modifiers, 
+                /*asteriskToken*/ undefined, 
+                /*name*/ undefined, 
+                /*typeParameters*/ undefined, setAccessor.parameters, 
+                /*type*/ undefined, setAccessor.body // TODO: GH#18217
+                ), setAccessor), setAccessor)
+            }, !multiLine)), firstAccessor);
         }
-        function parseClassMembers() {
-            return parseList(5 /* ClassMembers */, parseClassElement);
+        return undefined;
+    }
+    function createExpressionForPropertyAssignment(factory, property, receiver) {
+        return ts.setOriginalNode(ts.setTextRange(factory.createAssignment(createMemberAccessForPropertyName(factory, receiver, property.name, /*location*/ property.name), property.initializer), property), property);
+    }
+    function createExpressionForShorthandPropertyAssignment(factory, property, receiver) {
+        return ts.setOriginalNode(ts.setTextRange(factory.createAssignment(createMemberAccessForPropertyName(factory, receiver, property.name, /*location*/ property.name), factory.cloneNode(property.name)), 
+        /*location*/ property), 
+        /*original*/ property);
+    }
+    function createExpressionForMethodDeclaration(factory, method, receiver) {
+        return ts.setOriginalNode(ts.setTextRange(factory.createAssignment(createMemberAccessForPropertyName(factory, receiver, method.name, /*location*/ method.name), ts.setOriginalNode(ts.setTextRange(factory.createFunctionExpression(method.modifiers, method.asteriskToken, 
+        /*name*/ undefined, 
+        /*typeParameters*/ undefined, method.parameters, 
+        /*type*/ undefined, method.body // TODO: GH#18217
+        ), 
+        /*location*/ method), 
+        /*original*/ method)), 
+        /*location*/ method), 
+        /*original*/ method);
+    }
+    function createExpressionForObjectLiteralElementLike(factory, node, property, receiver) {
+        if (property.name && ts.isPrivateIdentifier(property.name)) {
+            ts.Debug.failBadSyntaxKind(property.name, "Private identifiers are not allowed in object literals.");
         }
-        function parseInterfaceDeclaration(node) {
-            node.kind = 246 /* InterfaceDeclaration */;
-            parseExpected(114 /* InterfaceKeyword */);
-            node.name = parseIdentifier();
-            node.typeParameters = parseTypeParameters();
-            node.heritageClauses = parseHeritageClauses();
-            node.members = parseObjectTypeMembers();
-            return finishNode(node);
-        }
-        function parseTypeAliasDeclaration(node) {
-            node.kind = 247 /* TypeAliasDeclaration */;
-            parseExpected(145 /* TypeKeyword */);
-            node.name = parseIdentifier();
-            node.typeParameters = parseTypeParameters();
-            parseExpected(62 /* EqualsToken */);
-            node.type = parseType();
-            parseSemicolon();
-            return finishNode(node);
+        switch (property.kind) {
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+                return createExpressionForAccessorDeclaration(factory, node.properties, property, receiver, !!node.multiLine);
+            case 288 /* PropertyAssignment */:
+                return createExpressionForPropertyAssignment(factory, property, receiver);
+            case 289 /* ShorthandPropertyAssignment */:
+                return createExpressionForShorthandPropertyAssignment(factory, property, receiver);
+            case 165 /* MethodDeclaration */:
+                return createExpressionForMethodDeclaration(factory, property, receiver);
         }
-        // In an ambient declaration, the grammar only allows integer literals as initializers.
-        // In a non-ambient declaration, the grammar allows uninitialized members only in a
-        // ConstantEnumMemberSection, which starts at the beginning of an enum declaration
-        // or any time an integer literal initializer is encountered.
-        function parseEnumMember() {
-            var node = createNodeWithJSDoc(284 /* EnumMember */);
-            node.name = parsePropertyName();
-            node.initializer = allowInAnd(parseInitializer);
-            return finishNode(node);
-        }
-        function parseEnumDeclaration(node) {
-            node.kind = 248 /* EnumDeclaration */;
-            parseExpected(88 /* EnumKeyword */);
-            node.name = parseIdentifier();
-            if (parseExpected(18 /* OpenBraceToken */)) {
-                node.members = doOutsideOfYieldAndAwaitContext(function () { return parseDelimitedList(6 /* EnumMembers */, parseEnumMember); });
-                parseExpected(19 /* CloseBraceToken */);
+    }
+    ts.createExpressionForObjectLiteralElementLike = createExpressionForObjectLiteralElementLike;
+    /**
+     * Gets whether an identifier should only be referred to by its internal name.
+     */
+    function isInternalName(node) {
+        return (ts.getEmitFlags(node) & 32768 /* InternalName */) !== 0;
+    }
+    ts.isInternalName = isInternalName;
+    /**
+     * Gets whether an identifier should only be referred to by its local name.
+     */
+    function isLocalName(node) {
+        return (ts.getEmitFlags(node) & 16384 /* LocalName */) !== 0;
+    }
+    ts.isLocalName = isLocalName;
+    /**
+     * Gets whether an identifier should only be referred to by its export representation if the
+     * name points to an exported symbol.
+     */
+    function isExportName(node) {
+        return (ts.getEmitFlags(node) & 8192 /* ExportName */) !== 0;
+    }
+    ts.isExportName = isExportName;
+    function isUseStrictPrologue(node) {
+        return ts.isStringLiteral(node.expression) && node.expression.text === "use strict";
+    }
+    function findUseStrictPrologue(statements) {
+        for (var _i = 0, statements_1 = statements; _i < statements_1.length; _i++) {
+            var statement = statements_1[_i];
+            if (ts.isPrologueDirective(statement)) {
+                if (isUseStrictPrologue(statement)) {
+                    return statement;
+                }
             }
             else {
-                node.members = createMissingList();
+                break;
             }
-            return finishNode(node);
         }
-        function parseModuleBlock() {
-            var node = createNode(250 /* ModuleBlock */);
-            if (parseExpected(18 /* OpenBraceToken */)) {
-                node.statements = parseList(1 /* BlockStatements */, parseStatement);
-                parseExpected(19 /* CloseBraceToken */);
-            }
-            else {
-                node.statements = createMissingList();
-            }
-            return finishNode(node);
+        return undefined;
+    }
+    ts.findUseStrictPrologue = findUseStrictPrologue;
+    function startsWithUseStrict(statements) {
+        var firstStatement = ts.firstOrUndefined(statements);
+        return firstStatement !== undefined
+            && ts.isPrologueDirective(firstStatement)
+            && isUseStrictPrologue(firstStatement);
+    }
+    ts.startsWithUseStrict = startsWithUseStrict;
+    function isCommaSequence(node) {
+        return node.kind === 216 /* BinaryExpression */ && node.operatorToken.kind === 27 /* CommaToken */ ||
+            node.kind === 337 /* CommaListExpression */;
+    }
+    ts.isCommaSequence = isCommaSequence;
+    function isOuterExpression(node, kinds) {
+        if (kinds === void 0) { kinds = 15 /* All */; }
+        switch (node.kind) {
+            case 207 /* ParenthesizedExpression */:
+                return (kinds & 1 /* Parentheses */) !== 0;
+            case 206 /* TypeAssertionExpression */:
+            case 224 /* AsExpression */:
+                return (kinds & 2 /* TypeAssertions */) !== 0;
+            case 225 /* NonNullExpression */:
+                return (kinds & 4 /* NonNullAssertions */) !== 0;
+            case 336 /* PartiallyEmittedExpression */:
+                return (kinds & 8 /* PartiallyEmittedExpressions */) !== 0;
         }
-        function parseModuleOrNamespaceDeclaration(node, flags) {
-            node.kind = 249 /* ModuleDeclaration */;
-            // If we are parsing a dotted namespace name, we want to
-            // propagate the 'Namespace' flag across the names if set.
-            var namespaceFlag = flags & 16 /* Namespace */;
-            node.flags |= flags;
-            node.name = parseIdentifier();
-            node.body = parseOptional(24 /* DotToken */)
-                ? parseModuleOrNamespaceDeclaration(createNode(0 /* Unknown */), 4 /* NestedNamespace */ | namespaceFlag)
-                : parseModuleBlock();
-            return finishNode(node);
+        return false;
+    }
+    ts.isOuterExpression = isOuterExpression;
+    function skipOuterExpressions(node, kinds) {
+        if (kinds === void 0) { kinds = 15 /* All */; }
+        while (isOuterExpression(node, kinds)) {
+            node = node.expression;
         }
-        function parseAmbientExternalModuleDeclaration(node) {
-            node.kind = 249 /* ModuleDeclaration */;
-            if (token() === 150 /* GlobalKeyword */) {
-                // parse 'global' as name of global scope augmentation
-                node.name = parseIdentifier();
-                node.flags |= 1024 /* GlobalAugmentation */;
+        return node;
+    }
+    ts.skipOuterExpressions = skipOuterExpressions;
+    function skipAssertions(node) {
+        return skipOuterExpressions(node, 6 /* Assertions */);
+    }
+    ts.skipAssertions = skipAssertions;
+    function startOnNewLine(node) {
+        return ts.setStartsOnNewLine(node, /*newLine*/ true);
+    }
+    ts.startOnNewLine = startOnNewLine;
+    function getExternalHelpersModuleName(node) {
+        var parseNode = ts.getOriginalNode(node, ts.isSourceFile);
+        var emitNode = parseNode && parseNode.emitNode;
+        return emitNode && emitNode.externalHelpersModuleName;
+    }
+    ts.getExternalHelpersModuleName = getExternalHelpersModuleName;
+    function hasRecordedExternalHelpers(sourceFile) {
+        var parseNode = ts.getOriginalNode(sourceFile, ts.isSourceFile);
+        var emitNode = parseNode && parseNode.emitNode;
+        return !!emitNode && (!!emitNode.externalHelpersModuleName || !!emitNode.externalHelpers);
+    }
+    ts.hasRecordedExternalHelpers = hasRecordedExternalHelpers;
+    function createExternalHelpersImportDeclarationIfNeeded(nodeFactory, helperFactory, sourceFile, compilerOptions, hasExportStarsToExportValues, hasImportStar, hasImportDefault) {
+        if (compilerOptions.importHelpers && ts.isEffectiveExternalModule(sourceFile, compilerOptions)) {
+            var namedBindings = void 0;
+            var moduleKind = ts.getEmitModuleKind(compilerOptions);
+            if (moduleKind >= ts.ModuleKind.ES2015 && moduleKind <= ts.ModuleKind.ESNext) {
+                // use named imports
+                var helpers = ts.getEmitHelpers(sourceFile);
+                if (helpers) {
+                    var helperNames = [];
+                    for (var _i = 0, helpers_3 = helpers; _i < helpers_3.length; _i++) {
+                        var helper = helpers_3[_i];
+                        if (!helper.scoped) {
+                            var importName = helper.importName;
+                            if (importName) {
+                                ts.pushIfUnique(helperNames, importName);
+                            }
+                        }
+                    }
+                    if (ts.some(helperNames)) {
+                        helperNames.sort(ts.compareStringsCaseSensitive);
+                        // Alias the imports if the names are used somewhere in the file.
+                        // NOTE: We don't need to care about global import collisions as this is a module.
+                        namedBindings = nodeFactory.createNamedImports(ts.map(helperNames, function (name) { return ts.isFileLevelUniqueName(sourceFile, name)
+                            ? nodeFactory.createImportSpecifier(/*propertyName*/ undefined, nodeFactory.createIdentifier(name))
+                            : nodeFactory.createImportSpecifier(nodeFactory.createIdentifier(name), helperFactory.getUnscopedHelperName(name)); }));
+                        var parseNode = ts.getOriginalNode(sourceFile, ts.isSourceFile);
+                        var emitNode = ts.getOrCreateEmitNode(parseNode);
+                        emitNode.externalHelpers = true;
+                    }
+                }
             }
             else {
-                node.name = parseLiteralNode();
-                node.name.text = internIdentifier(node.name.text);
-            }
-            if (token() === 18 /* OpenBraceToken */) {
-                node.body = parseModuleBlock();
+                // use a namespace import
+                var externalHelpersModuleName = getOrCreateExternalHelpersModuleNameIfNeeded(nodeFactory, sourceFile, compilerOptions, hasExportStarsToExportValues, hasImportStar || hasImportDefault);
+                if (externalHelpersModuleName) {
+                    namedBindings = nodeFactory.createNamespaceImport(externalHelpersModuleName);
+                }
             }
-            else {
-                parseSemicolon();
+            if (namedBindings) {
+                var externalHelpersImportDeclaration = nodeFactory.createImportDeclaration(
+                /*decorators*/ undefined, 
+                /*modifiers*/ undefined, nodeFactory.createImportClause(/*isTypeOnly*/ false, /*name*/ undefined, namedBindings), nodeFactory.createStringLiteral(ts.externalHelpersModuleNameText));
+                ts.addEmitFlags(externalHelpersImportDeclaration, 67108864 /* NeverApplyImportHelper */);
+                return externalHelpersImportDeclaration;
             }
-            return finishNode(node);
         }
-        function parseModuleDeclaration(node) {
-            var flags = 0;
-            if (token() === 150 /* GlobalKeyword */) {
-                // global augmentation
-                return parseAmbientExternalModuleDeclaration(node);
-            }
-            else if (parseOptional(136 /* NamespaceKeyword */)) {
-                flags |= 16 /* Namespace */;
+    }
+    ts.createExternalHelpersImportDeclarationIfNeeded = createExternalHelpersImportDeclarationIfNeeded;
+    function getOrCreateExternalHelpersModuleNameIfNeeded(factory, node, compilerOptions, hasExportStarsToExportValues, hasImportStarOrImportDefault) {
+        if (compilerOptions.importHelpers && ts.isEffectiveExternalModule(node, compilerOptions)) {
+            var externalHelpersModuleName = getExternalHelpersModuleName(node);
+            if (externalHelpersModuleName) {
+                return externalHelpersModuleName;
             }
-            else {
-                parseExpected(135 /* ModuleKeyword */);
-                if (token() === 10 /* StringLiteral */) {
-                    return parseAmbientExternalModuleDeclaration(node);
+            var moduleKind = ts.getEmitModuleKind(compilerOptions);
+            var create = (hasExportStarsToExportValues || (compilerOptions.esModuleInterop && hasImportStarOrImportDefault))
+                && moduleKind !== ts.ModuleKind.System
+                && moduleKind < ts.ModuleKind.ES2015;
+            if (!create) {
+                var helpers = ts.getEmitHelpers(node);
+                if (helpers) {
+                    for (var _i = 0, helpers_4 = helpers; _i < helpers_4.length; _i++) {
+                        var helper = helpers_4[_i];
+                        if (!helper.scoped) {
+                            create = true;
+                            break;
+                        }
+                    }
                 }
             }
-            return parseModuleOrNamespaceDeclaration(node, flags);
+            if (create) {
+                var parseNode = ts.getOriginalNode(node, ts.isSourceFile);
+                var emitNode = ts.getOrCreateEmitNode(parseNode);
+                return emitNode.externalHelpersModuleName || (emitNode.externalHelpersModuleName = factory.createUniqueName(ts.externalHelpersModuleNameText));
+            }
         }
-        function isExternalModuleReference() {
-            return token() === 139 /* RequireKeyword */ &&
-                lookAhead(nextTokenIsOpenParen);
+    }
+    ts.getOrCreateExternalHelpersModuleNameIfNeeded = getOrCreateExternalHelpersModuleNameIfNeeded;
+    /**
+     * Get the name of that target module from an import or export declaration
+     */
+    function getLocalNameForExternalImport(factory, node, sourceFile) {
+        var namespaceDeclaration = ts.getNamespaceDeclarationNode(node);
+        if (namespaceDeclaration && !ts.isDefaultImport(node) && !ts.isExportNamespaceAsDefaultDeclaration(node)) {
+            var name = namespaceDeclaration.name;
+            return ts.isGeneratedIdentifier(name) ? name : factory.createIdentifier(ts.getSourceTextOfNodeFromSourceFile(sourceFile, name) || ts.idText(name));
         }
-        function nextTokenIsOpenParen() {
-            return nextToken() === 20 /* OpenParenToken */;
+        if (node.kind === 261 /* ImportDeclaration */ && node.importClause) {
+            return factory.getGeneratedNameForNode(node);
         }
-        function nextTokenIsSlash() {
-            return nextToken() === 43 /* SlashToken */;
+        if (node.kind === 267 /* ExportDeclaration */ && node.moduleSpecifier) {
+            return factory.getGeneratedNameForNode(node);
         }
-        function parseNamespaceExportDeclaration(node) {
-            node.kind = 252 /* NamespaceExportDeclaration */;
-            parseExpected(123 /* AsKeyword */);
-            parseExpected(136 /* NamespaceKeyword */);
-            node.name = parseIdentifier();
-            parseSemicolon();
-            return finishNode(node);
+        return undefined;
+    }
+    ts.getLocalNameForExternalImport = getLocalNameForExternalImport;
+    /**
+     * Get the name of a target module from an import/export declaration as should be written in the emitted output.
+     * The emitted output name can be different from the input if:
+     *  1. The module has a /// <amd-module name="<new name>" />
+     *  2. --out or --outFile is used, making the name relative to the rootDir
+     *  3- The containing SourceFile has an entry in renamedDependencies for the import as requested by some module loaders (e.g. System).
+     * Otherwise, a new StringLiteral node representing the module name will be returned.
+     */
+    function getExternalModuleNameLiteral(factory, importNode, sourceFile, host, resolver, compilerOptions) {
+        var moduleName = ts.getExternalModuleName(importNode); // TODO: GH#18217
+        if (moduleName.kind === 10 /* StringLiteral */) {
+            return tryGetModuleNameFromDeclaration(importNode, host, factory, resolver, compilerOptions)
+                || tryRenameExternalModule(factory, moduleName, sourceFile)
+                || factory.cloneNode(moduleName);
         }
-        function parseImportDeclarationOrImportEqualsDeclaration(node) {
-            parseExpected(96 /* ImportKeyword */);
-            var afterImportPos = scanner.getStartPos();
-            var identifier;
-            if (isIdentifier()) {
-                identifier = parseIdentifier();
-            }
-            var isTypeOnly = false;
-            if (token() !== 149 /* FromKeyword */ &&
-                (identifier === null || identifier === void 0 ? void 0 : identifier.escapedText) === "type" &&
-                (isIdentifier() || tokenAfterImportDefinitelyProducesImportDeclaration())) {
-                isTypeOnly = true;
-                identifier = isIdentifier() ? parseIdentifier() : undefined;
-            }
-            if (identifier && !tokenAfterImportedIdentifierDefinitelyProducesImportDeclaration()) {
-                return parseImportEqualsDeclaration(node, identifier, isTypeOnly);
-            }
-            // Import statement
-            node.kind = 254 /* ImportDeclaration */;
-            // ImportDeclaration:
-            //  import ImportClause from ModuleSpecifier ;
-            //  import ModuleSpecifier;
-            if (identifier || // import id
-                token() === 41 /* AsteriskToken */ || // import *
-                token() === 18 /* OpenBraceToken */ // import {
-            ) {
-                node.importClause = parseImportClause(identifier, afterImportPos, isTypeOnly);
-                parseExpected(149 /* FromKeyword */);
-            }
-            node.moduleSpecifier = parseModuleSpecifier();
-            parseSemicolon();
-            return finishNode(node);
+        return undefined;
+    }
+    ts.getExternalModuleNameLiteral = getExternalModuleNameLiteral;
+    /**
+     * Some bundlers (SystemJS builder) sometimes want to rename dependencies.
+     * Here we check if alternative name was provided for a given moduleName and return it if possible.
+     */
+    function tryRenameExternalModule(factory, moduleName, sourceFile) {
+        var rename = sourceFile.renamedDependencies && sourceFile.renamedDependencies.get(moduleName.text);
+        return rename && factory.createStringLiteral(rename);
+    }
+    /**
+     * Get the name of a module as should be written in the emitted output.
+     * The emitted output name can be different from the input if:
+     *  1. The module has a /// <amd-module name="<new name>" />
+     *  2. --out or --outFile is used, making the name relative to the rootDir
+     * Otherwise, a new StringLiteral node representing the module name will be returned.
+     */
+    function tryGetModuleNameFromFile(factory, file, host, options) {
+        if (!file) {
+            return undefined;
         }
-        function tokenAfterImportDefinitelyProducesImportDeclaration() {
-            return token() === 41 /* AsteriskToken */ || token() === 18 /* OpenBraceToken */;
+        if (file.moduleName) {
+            return factory.createStringLiteral(file.moduleName);
         }
-        function tokenAfterImportedIdentifierDefinitelyProducesImportDeclaration() {
-            // In `import id ___`, the current token decides whether to produce
-            // an ImportDeclaration or ImportEqualsDeclaration.
-            return token() === 27 /* CommaToken */ || token() === 149 /* FromKeyword */;
+        if (!file.isDeclarationFile && ts.outFile(options)) {
+            return factory.createStringLiteral(ts.getExternalModuleNameFromPath(host, file.fileName));
         }
-        function parseImportEqualsDeclaration(node, identifier, isTypeOnly) {
-            node.kind = 253 /* ImportEqualsDeclaration */;
-            node.name = identifier;
-            parseExpected(62 /* EqualsToken */);
-            node.moduleReference = parseModuleReference();
-            parseSemicolon();
-            var finished = finishNode(node);
-            if (isTypeOnly) {
-                parseErrorAtRange(finished, ts.Diagnostics.Only_ECMAScript_imports_may_use_import_type);
-            }
-            return finished;
+        return undefined;
+    }
+    ts.tryGetModuleNameFromFile = tryGetModuleNameFromFile;
+    function tryGetModuleNameFromDeclaration(declaration, host, factory, resolver, compilerOptions) {
+        return tryGetModuleNameFromFile(factory, resolver.getExternalModuleFileFromDeclaration(declaration), host, compilerOptions);
+    }
+    /**
+     * Gets the initializer of an BindingOrAssignmentElement.
+     */
+    function getInitializerOfBindingOrAssignmentElement(bindingElement) {
+        if (ts.isDeclarationBindingElement(bindingElement)) {
+            // `1` in `let { a = 1 } = ...`
+            // `1` in `let { a: b = 1 } = ...`
+            // `1` in `let { a: {b} = 1 } = ...`
+            // `1` in `let { a: [b] = 1 } = ...`
+            // `1` in `let [a = 1] = ...`
+            // `1` in `let [{a} = 1] = ...`
+            // `1` in `let [[a] = 1] = ...`
+            return bindingElement.initializer;
         }
-        function parseImportClause(identifier, fullStart, isTypeOnly) {
-            // ImportClause:
-            //  ImportedDefaultBinding
-            //  NameSpaceImport
-            //  NamedImports
-            //  ImportedDefaultBinding, NameSpaceImport
-            //  ImportedDefaultBinding, NamedImports
-            var importClause = createNode(255 /* ImportClause */, fullStart);
-            importClause.isTypeOnly = isTypeOnly;
-            if (identifier) {
-                // ImportedDefaultBinding:
-                //  ImportedBinding
-                importClause.name = identifier;
-            }
-            // If there was no default import or if there is comma token after default import
-            // parse namespace or named imports
-            if (!importClause.name ||
-                parseOptional(27 /* CommaToken */)) {
-                importClause.namedBindings = token() === 41 /* AsteriskToken */ ? parseNamespaceImport() : parseNamedImportsOrExports(257 /* NamedImports */);
-            }
-            return finishNode(importClause);
+        if (ts.isPropertyAssignment(bindingElement)) {
+            // `1` in `({ a: b = 1 } = ...)`
+            // `1` in `({ a: {b} = 1 } = ...)`
+            // `1` in `({ a: [b] = 1 } = ...)`
+            var initializer = bindingElement.initializer;
+            return ts.isAssignmentExpression(initializer, /*excludeCompoundAssignment*/ true)
+                ? initializer.right
+                : undefined;
         }
-        function parseModuleReference() {
-            return isExternalModuleReference()
-                ? parseExternalModuleReference()
-                : parseEntityName(/*allowReservedWords*/ false);
+        if (ts.isShorthandPropertyAssignment(bindingElement)) {
+            // `1` in `({ a = 1 } = ...)`
+            return bindingElement.objectAssignmentInitializer;
         }
-        function parseExternalModuleReference() {
-            var node = createNode(265 /* ExternalModuleReference */);
-            parseExpected(139 /* RequireKeyword */);
-            parseExpected(20 /* OpenParenToken */);
-            node.expression = parseModuleSpecifier();
-            parseExpected(21 /* CloseParenToken */);
-            return finishNode(node);
+        if (ts.isAssignmentExpression(bindingElement, /*excludeCompoundAssignment*/ true)) {
+            // `1` in `[a = 1] = ...`
+            // `1` in `[{a} = 1] = ...`
+            // `1` in `[[a] = 1] = ...`
+            return bindingElement.right;
         }
-        function parseModuleSpecifier() {
-            if (token() === 10 /* StringLiteral */) {
-                var result = parseLiteralNode();
-                result.text = internIdentifier(result.text);
-                return result;
-            }
-            else {
-                // We allow arbitrary expressions here, even though the grammar only allows string
-                // literals.  We check to ensure that it is only a string literal later in the grammar
-                // check pass.
-                return parseExpression();
-            }
+        if (ts.isSpreadElement(bindingElement)) {
+            // Recovery consistent with existing emit.
+            return getInitializerOfBindingOrAssignmentElement(bindingElement.expression);
         }
-        function parseNamespaceImport() {
-            // NameSpaceImport:
-            //  * as ImportedBinding
-            var namespaceImport = createNode(256 /* NamespaceImport */);
-            parseExpected(41 /* AsteriskToken */);
-            parseExpected(123 /* AsKeyword */);
-            namespaceImport.name = parseIdentifier();
-            return finishNode(namespaceImport);
+    }
+    ts.getInitializerOfBindingOrAssignmentElement = getInitializerOfBindingOrAssignmentElement;
+    /**
+     * Gets the name of an BindingOrAssignmentElement.
+     */
+    function getTargetOfBindingOrAssignmentElement(bindingElement) {
+        if (ts.isDeclarationBindingElement(bindingElement)) {
+            // `a` in `let { a } = ...`
+            // `a` in `let { a = 1 } = ...`
+            // `b` in `let { a: b } = ...`
+            // `b` in `let { a: b = 1 } = ...`
+            // `a` in `let { ...a } = ...`
+            // `{b}` in `let { a: {b} } = ...`
+            // `{b}` in `let { a: {b} = 1 } = ...`
+            // `[b]` in `let { a: [b] } = ...`
+            // `[b]` in `let { a: [b] = 1 } = ...`
+            // `a` in `let [a] = ...`
+            // `a` in `let [a = 1] = ...`
+            // `a` in `let [...a] = ...`
+            // `{a}` in `let [{a}] = ...`
+            // `{a}` in `let [{a} = 1] = ...`
+            // `[a]` in `let [[a]] = ...`
+            // `[a]` in `let [[a] = 1] = ...`
+            return bindingElement.name;
         }
-        function parseNamedImportsOrExports(kind) {
-            var node = createNode(kind);
-            // NamedImports:
-            //  { }
-            //  { ImportsList }
-            //  { ImportsList, }
-            // ImportsList:
-            //  ImportSpecifier
-            //  ImportsList, ImportSpecifier
-            node.elements = parseBracketedList(23 /* ImportOrExportSpecifiers */, kind === 257 /* NamedImports */ ? parseImportSpecifier : parseExportSpecifier, 18 /* OpenBraceToken */, 19 /* CloseBraceToken */);
-            return finishNode(node);
+        if (ts.isObjectLiteralElementLike(bindingElement)) {
+            switch (bindingElement.kind) {
+                case 288 /* PropertyAssignment */:
+                    // `b` in `({ a: b } = ...)`
+                    // `b` in `({ a: b = 1 } = ...)`
+                    // `{b}` in `({ a: {b} } = ...)`
+                    // `{b}` in `({ a: {b} = 1 } = ...)`
+                    // `[b]` in `({ a: [b] } = ...)`
+                    // `[b]` in `({ a: [b] = 1 } = ...)`
+                    // `b.c` in `({ a: b.c } = ...)`
+                    // `b.c` in `({ a: b.c = 1 } = ...)`
+                    // `b[0]` in `({ a: b[0] } = ...)`
+                    // `b[0]` in `({ a: b[0] = 1 } = ...)`
+                    return getTargetOfBindingOrAssignmentElement(bindingElement.initializer);
+                case 289 /* ShorthandPropertyAssignment */:
+                    // `a` in `({ a } = ...)`
+                    // `a` in `({ a = 1 } = ...)`
+                    return bindingElement.name;
+                case 290 /* SpreadAssignment */:
+                    // `a` in `({ ...a } = ...)`
+                    return getTargetOfBindingOrAssignmentElement(bindingElement.expression);
+            }
+            // no target
+            return undefined;
         }
-        function parseExportSpecifier() {
-            return parseImportOrExportSpecifier(263 /* ExportSpecifier */);
+        if (ts.isAssignmentExpression(bindingElement, /*excludeCompoundAssignment*/ true)) {
+            // `a` in `[a = 1] = ...`
+            // `{a}` in `[{a} = 1] = ...`
+            // `[a]` in `[[a] = 1] = ...`
+            // `a.b` in `[a.b = 1] = ...`
+            // `a[0]` in `[a[0] = 1] = ...`
+            return getTargetOfBindingOrAssignmentElement(bindingElement.left);
         }
-        function parseImportSpecifier() {
-            return parseImportOrExportSpecifier(258 /* ImportSpecifier */);
+        if (ts.isSpreadElement(bindingElement)) {
+            // `a` in `[...a] = ...`
+            return getTargetOfBindingOrAssignmentElement(bindingElement.expression);
         }
-        function parseImportOrExportSpecifier(kind) {
-            var node = createNode(kind);
-            // ImportSpecifier:
-            //   BindingIdentifier
-            //   IdentifierName as BindingIdentifier
-            // ExportSpecifier:
-            //   IdentifierName
-            //   IdentifierName as IdentifierName
-            var checkIdentifierIsKeyword = ts.isKeyword(token()) && !isIdentifier();
-            var checkIdentifierStart = scanner.getTokenPos();
-            var checkIdentifierEnd = scanner.getTextPos();
-            var identifierName = parseIdentifierName();
-            if (token() === 123 /* AsKeyword */) {
-                node.propertyName = identifierName;
-                parseExpected(123 /* AsKeyword */);
-                checkIdentifierIsKeyword = ts.isKeyword(token()) && !isIdentifier();
-                checkIdentifierStart = scanner.getTokenPos();
-                checkIdentifierEnd = scanner.getTextPos();
-                node.name = parseIdentifierName();
-            }
-            else {
-                node.name = identifierName;
-            }
-            if (kind === 258 /* ImportSpecifier */ && checkIdentifierIsKeyword) {
-                parseErrorAt(checkIdentifierStart, checkIdentifierEnd, ts.Diagnostics.Identifier_expected);
-            }
-            return finishNode(node);
+        // `a` in `[a] = ...`
+        // `{a}` in `[{a}] = ...`
+        // `[a]` in `[[a]] = ...`
+        // `a.b` in `[a.b] = ...`
+        // `a[0]` in `[a[0]] = ...`
+        return bindingElement;
+    }
+    ts.getTargetOfBindingOrAssignmentElement = getTargetOfBindingOrAssignmentElement;
+    /**
+     * Determines whether an BindingOrAssignmentElement is a rest element.
+     */
+    function getRestIndicatorOfBindingOrAssignmentElement(bindingElement) {
+        switch (bindingElement.kind) {
+            case 160 /* Parameter */:
+            case 198 /* BindingElement */:
+                // `...` in `let [...a] = ...`
+                return bindingElement.dotDotDotToken;
+            case 220 /* SpreadElement */:
+            case 290 /* SpreadAssignment */:
+                // `...` in `[...a] = ...`
+                return bindingElement;
         }
-        function parseNamespaceExport(pos) {
-            var node = createNode(262 /* NamespaceExport */, pos);
-            node.name = parseIdentifier();
-            return finishNode(node);
-        }
-        function parseExportDeclaration(node) {
-            node.kind = 260 /* ExportDeclaration */;
-            node.isTypeOnly = parseOptional(145 /* TypeKeyword */);
-            var namespaceExportPos = scanner.getStartPos();
-            if (parseOptional(41 /* AsteriskToken */)) {
-                if (parseOptional(123 /* AsKeyword */)) {
-                    node.exportClause = parseNamespaceExport(namespaceExportPos);
+        return undefined;
+    }
+    ts.getRestIndicatorOfBindingOrAssignmentElement = getRestIndicatorOfBindingOrAssignmentElement;
+    /**
+     * Gets the property name of a BindingOrAssignmentElement
+     */
+    function getPropertyNameOfBindingOrAssignmentElement(bindingElement) {
+        var propertyName = tryGetPropertyNameOfBindingOrAssignmentElement(bindingElement);
+        ts.Debug.assert(!!propertyName || ts.isSpreadAssignment(bindingElement), "Invalid property name for binding element.");
+        return propertyName;
+    }
+    ts.getPropertyNameOfBindingOrAssignmentElement = getPropertyNameOfBindingOrAssignmentElement;
+    function tryGetPropertyNameOfBindingOrAssignmentElement(bindingElement) {
+        switch (bindingElement.kind) {
+            case 198 /* BindingElement */:
+                // `a` in `let { a: b } = ...`
+                // `[a]` in `let { [a]: b } = ...`
+                // `"a"` in `let { "a": b } = ...`
+                // `1` in `let { 1: b } = ...`
+                if (bindingElement.propertyName) {
+                    var propertyName = bindingElement.propertyName;
+                    if (ts.isPrivateIdentifier(propertyName)) {
+                        return ts.Debug.failBadSyntaxKind(propertyName);
+                    }
+                    return ts.isComputedPropertyName(propertyName) && isStringOrNumericLiteral(propertyName.expression)
+                        ? propertyName.expression
+                        : propertyName;
                 }
-                parseExpected(149 /* FromKeyword */);
-                node.moduleSpecifier = parseModuleSpecifier();
-            }
-            else {
-                node.exportClause = parseNamedImportsOrExports(261 /* NamedExports */);
-                // It is not uncommon to accidentally omit the 'from' keyword. Additionally, in editing scenarios,
-                // the 'from' keyword can be parsed as a named export when the export clause is unterminated (i.e. `export { from "moduleName";`)
-                // If we don't have a 'from' keyword, see if we have a string literal such that ASI won't take effect.
-                if (token() === 149 /* FromKeyword */ || (token() === 10 /* StringLiteral */ && !scanner.hasPrecedingLineBreak())) {
-                    parseExpected(149 /* FromKeyword */);
-                    node.moduleSpecifier = parseModuleSpecifier();
+                break;
+            case 288 /* PropertyAssignment */:
+                // `a` in `({ a: b } = ...)`
+                // `[a]` in `({ [a]: b } = ...)`
+                // `"a"` in `({ "a": b } = ...)`
+                // `1` in `({ 1: b } = ...)`
+                if (bindingElement.name) {
+                    var propertyName = bindingElement.name;
+                    if (ts.isPrivateIdentifier(propertyName)) {
+                        return ts.Debug.failBadSyntaxKind(propertyName);
+                    }
+                    return ts.isComputedPropertyName(propertyName) && isStringOrNumericLiteral(propertyName.expression)
+                        ? propertyName.expression
+                        : propertyName;
+                }
+                break;
+            case 290 /* SpreadAssignment */:
+                // `a` in `({ ...a } = ...)`
+                if (bindingElement.name && ts.isPrivateIdentifier(bindingElement.name)) {
+                    return ts.Debug.failBadSyntaxKind(bindingElement.name);
+                }
+                return bindingElement.name;
+        }
+        var target = getTargetOfBindingOrAssignmentElement(bindingElement);
+        if (target && ts.isPropertyName(target)) {
+            return target;
+        }
+    }
+    ts.tryGetPropertyNameOfBindingOrAssignmentElement = tryGetPropertyNameOfBindingOrAssignmentElement;
+    function isStringOrNumericLiteral(node) {
+        var kind = node.kind;
+        return kind === 10 /* StringLiteral */
+            || kind === 8 /* NumericLiteral */;
+    }
+    /**
+     * Gets the elements of a BindingOrAssignmentPattern
+     */
+    function getElementsOfBindingOrAssignmentPattern(name) {
+        switch (name.kind) {
+            case 196 /* ObjectBindingPattern */:
+            case 197 /* ArrayBindingPattern */:
+            case 199 /* ArrayLiteralExpression */:
+                // `a` in `{a}`
+                // `a` in `[a]`
+                return name.elements;
+            case 200 /* ObjectLiteralExpression */:
+                // `a` in `{a}`
+                return name.properties;
+        }
+    }
+    ts.getElementsOfBindingOrAssignmentPattern = getElementsOfBindingOrAssignmentPattern;
+    /* @internal */
+    function getJSDocTypeAliasName(fullName) {
+        if (fullName) {
+            var rightNode = fullName;
+            while (true) {
+                if (ts.isIdentifier(rightNode) || !rightNode.body) {
+                    return ts.isIdentifier(rightNode) ? rightNode : rightNode.name;
                 }
+                rightNode = rightNode.body;
             }
-            parseSemicolon();
-            return finishNode(node);
         }
-        function parseExportAssignment(node) {
-            node.kind = 259 /* ExportAssignment */;
-            if (parseOptional(62 /* EqualsToken */)) {
-                node.isExportEquals = true;
+    }
+    ts.getJSDocTypeAliasName = getJSDocTypeAliasName;
+    function canHaveModifiers(node) {
+        var kind = node.kind;
+        return kind === 160 /* Parameter */
+            || kind === 162 /* PropertySignature */
+            || kind === 163 /* PropertyDeclaration */
+            || kind === 164 /* MethodSignature */
+            || kind === 165 /* MethodDeclaration */
+            || kind === 166 /* Constructor */
+            || kind === 167 /* GetAccessor */
+            || kind === 168 /* SetAccessor */
+            || kind === 171 /* IndexSignature */
+            || kind === 208 /* FunctionExpression */
+            || kind === 209 /* ArrowFunction */
+            || kind === 221 /* ClassExpression */
+            || kind === 232 /* VariableStatement */
+            || kind === 251 /* FunctionDeclaration */
+            || kind === 252 /* ClassDeclaration */
+            || kind === 253 /* InterfaceDeclaration */
+            || kind === 254 /* TypeAliasDeclaration */
+            || kind === 255 /* EnumDeclaration */
+            || kind === 256 /* ModuleDeclaration */
+            || kind === 260 /* ImportEqualsDeclaration */
+            || kind === 261 /* ImportDeclaration */
+            || kind === 266 /* ExportAssignment */
+            || kind === 267 /* ExportDeclaration */;
+    }
+    ts.canHaveModifiers = canHaveModifiers;
+    /* @internal */
+    function isExportModifier(node) {
+        return node.kind === 92 /* ExportKeyword */;
+    }
+    ts.isExportModifier = isExportModifier;
+    /* @internal */
+    function isAsyncModifier(node) {
+        return node.kind === 129 /* AsyncKeyword */;
+    }
+    ts.isAsyncModifier = isAsyncModifier;
+    /* @internal */
+    function isStaticModifier(node) {
+        return node.kind === 123 /* StaticKeyword */;
+    }
+    ts.isStaticModifier = isStaticModifier;
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    function setTextRange(range, location) {
+        return location ? ts.setTextRangePosEnd(range, location.pos, location.end) : range;
+    }
+    ts.setTextRange = setTextRange;
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    var SignatureFlags;
+    (function (SignatureFlags) {
+        SignatureFlags[SignatureFlags["None"] = 0] = "None";
+        SignatureFlags[SignatureFlags["Yield"] = 1] = "Yield";
+        SignatureFlags[SignatureFlags["Await"] = 2] = "Await";
+        SignatureFlags[SignatureFlags["Type"] = 4] = "Type";
+        SignatureFlags[SignatureFlags["IgnoreMissingOpenBrace"] = 16] = "IgnoreMissingOpenBrace";
+        SignatureFlags[SignatureFlags["JSDoc"] = 32] = "JSDoc";
+    })(SignatureFlags || (SignatureFlags = {}));
+    var SpeculationKind;
+    (function (SpeculationKind) {
+        SpeculationKind[SpeculationKind["TryParse"] = 0] = "TryParse";
+        SpeculationKind[SpeculationKind["Lookahead"] = 1] = "Lookahead";
+        SpeculationKind[SpeculationKind["Reparse"] = 2] = "Reparse";
+    })(SpeculationKind || (SpeculationKind = {}));
+    var NodeConstructor;
+    var TokenConstructor;
+    var IdentifierConstructor;
+    var PrivateIdentifierConstructor;
+    var SourceFileConstructor;
+    /**
+     * NOTE: You should not use this, it is only exported to support `createNode` in `~/src/compat/deprecations.ts`.
+     */
+    /* @internal */
+    ts.parseBaseNodeFactory = {
+        createBaseSourceFileNode: function (kind) { return new (SourceFileConstructor || (SourceFileConstructor = ts.objectAllocator.getSourceFileConstructor()))(kind, -1, -1); },
+        createBaseIdentifierNode: function (kind) { return new (IdentifierConstructor || (IdentifierConstructor = ts.objectAllocator.getIdentifierConstructor()))(kind, -1, -1); },
+        createBasePrivateIdentifierNode: function (kind) { return new (PrivateIdentifierConstructor || (PrivateIdentifierConstructor = ts.objectAllocator.getPrivateIdentifierConstructor()))(kind, -1, -1); },
+        createBaseTokenNode: function (kind) { return new (TokenConstructor || (TokenConstructor = ts.objectAllocator.getTokenConstructor()))(kind, -1, -1); },
+        createBaseNode: function (kind) { return new (NodeConstructor || (NodeConstructor = ts.objectAllocator.getNodeConstructor()))(kind, -1, -1); },
+    };
+    /* @internal */
+    ts.parseNodeFactory = ts.createNodeFactory(1 /* NoParenthesizerRules */, ts.parseBaseNodeFactory);
+    function visitNode(cbNode, node) {
+        return node && cbNode(node);
+    }
+    function visitNodes(cbNode, cbNodes, nodes) {
+        if (nodes) {
+            if (cbNodes) {
+                return cbNodes(nodes);
             }
-            else {
-                parseExpected(84 /* DefaultKeyword */);
+            for (var _i = 0, nodes_1 = nodes; _i < nodes_1.length; _i++) {
+                var node = nodes_1[_i];
+                var result = cbNode(node);
+                if (result) {
+                    return result;
+                }
             }
-            node.expression = parseAssignmentExpressionOrHigher();
-            parseSemicolon();
-            return finishNode(node);
-        }
-        function setExternalModuleIndicator(sourceFile) {
-            // Try to use the first top-level import/export when available, then
-            // fall back to looking for an 'import.meta' somewhere in the tree if necessary.
-            sourceFile.externalModuleIndicator =
-                ts.forEach(sourceFile.statements, isAnExternalModuleIndicatorNode) ||
-                    getImportMetaIfNecessary(sourceFile);
         }
-        function isAnExternalModuleIndicatorNode(node) {
-            return hasModifierOfKind(node, 89 /* ExportKeyword */)
-                || node.kind === 253 /* ImportEqualsDeclaration */ && node.moduleReference.kind === 265 /* ExternalModuleReference */
-                || node.kind === 254 /* ImportDeclaration */
-                || node.kind === 259 /* ExportAssignment */
-                || node.kind === 260 /* ExportDeclaration */ ? node : undefined;
+    }
+    /*@internal*/
+    function isJSDocLikeText(text, start) {
+        return text.charCodeAt(start + 1) === 42 /* asterisk */ &&
+            text.charCodeAt(start + 2) === 42 /* asterisk */ &&
+            text.charCodeAt(start + 3) !== 47 /* slash */;
+    }
+    ts.isJSDocLikeText = isJSDocLikeText;
+    /**
+     * Invokes a callback for each child of the given node. The 'cbNode' callback is invoked for all child nodes
+     * stored in properties. If a 'cbNodes' callback is specified, it is invoked for embedded arrays; otherwise,
+     * embedded arrays are flattened and the 'cbNode' callback is invoked for each element. If a callback returns
+     * a truthy value, iteration stops and that value is returned. Otherwise, undefined is returned.
+     *
+     * @param node a given node to visit its children
+     * @param cbNode a callback to be invoked for all child nodes
+     * @param cbNodes a callback to be invoked for embedded array
+     *
+     * @remarks `forEachChild` must visit the children of a node in the order
+     * that they appear in the source code. The language service depends on this property to locate nodes by position.
+     */
+    function forEachChild(node, cbNode, cbNodes) {
+        if (!node || node.kind <= 156 /* LastToken */) {
+            return;
         }
-        function getImportMetaIfNecessary(sourceFile) {
-            return sourceFile.flags & 2097152 /* PossiblyContainsImportMeta */ ?
-                walkTreeForExternalModuleIndicators(sourceFile) :
-                undefined;
+        switch (node.kind) {
+            case 157 /* QualifiedName */:
+                return visitNode(cbNode, node.left) ||
+                    visitNode(cbNode, node.right);
+            case 159 /* TypeParameter */:
+                return visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.constraint) ||
+                    visitNode(cbNode, node.default) ||
+                    visitNode(cbNode, node.expression);
+            case 289 /* ShorthandPropertyAssignment */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.exclamationToken) ||
+                    visitNode(cbNode, node.equalsToken) ||
+                    visitNode(cbNode, node.objectAssignmentInitializer);
+            case 290 /* SpreadAssignment */:
+                return visitNode(cbNode, node.expression);
+            case 160 /* Parameter */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.dotDotDotToken) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.type) ||
+                    visitNode(cbNode, node.initializer);
+            case 163 /* PropertyDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.exclamationToken) ||
+                    visitNode(cbNode, node.type) ||
+                    visitNode(cbNode, node.initializer);
+            case 162 /* PropertySignature */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.type) ||
+                    visitNode(cbNode, node.initializer);
+            case 288 /* PropertyAssignment */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.initializer);
+            case 249 /* VariableDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.exclamationToken) ||
+                    visitNode(cbNode, node.type) ||
+                    visitNode(cbNode, node.initializer);
+            case 198 /* BindingElement */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.dotDotDotToken) ||
+                    visitNode(cbNode, node.propertyName) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.initializer);
+            case 174 /* FunctionType */:
+            case 175 /* ConstructorType */:
+            case 169 /* CallSignature */:
+            case 170 /* ConstructSignature */:
+            case 171 /* IndexSignature */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
+                    visitNodes(cbNode, cbNodes, node.parameters) ||
+                    visitNode(cbNode, node.type);
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
+            case 166 /* Constructor */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 208 /* FunctionExpression */:
+            case 251 /* FunctionDeclaration */:
+            case 209 /* ArrowFunction */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.asteriskToken) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.exclamationToken) ||
+                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
+                    visitNodes(cbNode, cbNodes, node.parameters) ||
+                    visitNode(cbNode, node.type) ||
+                    visitNode(cbNode, node.equalsGreaterThanToken) ||
+                    visitNode(cbNode, node.body);
+            case 173 /* TypeReference */:
+                return visitNode(cbNode, node.typeName) ||
+                    visitNodes(cbNode, cbNodes, node.typeArguments);
+            case 172 /* TypePredicate */:
+                return visitNode(cbNode, node.assertsModifier) ||
+                    visitNode(cbNode, node.parameterName) ||
+                    visitNode(cbNode, node.type);
+            case 176 /* TypeQuery */:
+                return visitNode(cbNode, node.exprName);
+            case 177 /* TypeLiteral */:
+                return visitNodes(cbNode, cbNodes, node.members);
+            case 178 /* ArrayType */:
+                return visitNode(cbNode, node.elementType);
+            case 179 /* TupleType */:
+                return visitNodes(cbNode, cbNodes, node.elements);
+            case 182 /* UnionType */:
+            case 183 /* IntersectionType */:
+                return visitNodes(cbNode, cbNodes, node.types);
+            case 184 /* ConditionalType */:
+                return visitNode(cbNode, node.checkType) ||
+                    visitNode(cbNode, node.extendsType) ||
+                    visitNode(cbNode, node.trueType) ||
+                    visitNode(cbNode, node.falseType);
+            case 185 /* InferType */:
+                return visitNode(cbNode, node.typeParameter);
+            case 195 /* ImportType */:
+                return visitNode(cbNode, node.argument) ||
+                    visitNode(cbNode, node.qualifier) ||
+                    visitNodes(cbNode, cbNodes, node.typeArguments);
+            case 186 /* ParenthesizedType */:
+            case 188 /* TypeOperator */:
+                return visitNode(cbNode, node.type);
+            case 189 /* IndexedAccessType */:
+                return visitNode(cbNode, node.objectType) ||
+                    visitNode(cbNode, node.indexType);
+            case 190 /* MappedType */:
+                return visitNode(cbNode, node.readonlyToken) ||
+                    visitNode(cbNode, node.typeParameter) ||
+                    visitNode(cbNode, node.nameType) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.type);
+            case 191 /* LiteralType */:
+                return visitNode(cbNode, node.literal);
+            case 192 /* NamedTupleMember */:
+                return visitNode(cbNode, node.dotDotDotToken) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.type);
+            case 196 /* ObjectBindingPattern */:
+            case 197 /* ArrayBindingPattern */:
+                return visitNodes(cbNode, cbNodes, node.elements);
+            case 199 /* ArrayLiteralExpression */:
+                return visitNodes(cbNode, cbNodes, node.elements);
+            case 200 /* ObjectLiteralExpression */:
+                return visitNodes(cbNode, cbNodes, node.properties);
+            case 201 /* PropertyAccessExpression */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.questionDotToken) ||
+                    visitNode(cbNode, node.name);
+            case 202 /* ElementAccessExpression */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.questionDotToken) ||
+                    visitNode(cbNode, node.argumentExpression);
+            case 203 /* CallExpression */:
+            case 204 /* NewExpression */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.questionDotToken) ||
+                    visitNodes(cbNode, cbNodes, node.typeArguments) ||
+                    visitNodes(cbNode, cbNodes, node.arguments);
+            case 205 /* TaggedTemplateExpression */:
+                return visitNode(cbNode, node.tag) ||
+                    visitNode(cbNode, node.questionDotToken) ||
+                    visitNodes(cbNode, cbNodes, node.typeArguments) ||
+                    visitNode(cbNode, node.template);
+            case 206 /* TypeAssertionExpression */:
+                return visitNode(cbNode, node.type) ||
+                    visitNode(cbNode, node.expression);
+            case 207 /* ParenthesizedExpression */:
+                return visitNode(cbNode, node.expression);
+            case 210 /* DeleteExpression */:
+                return visitNode(cbNode, node.expression);
+            case 211 /* TypeOfExpression */:
+                return visitNode(cbNode, node.expression);
+            case 212 /* VoidExpression */:
+                return visitNode(cbNode, node.expression);
+            case 214 /* PrefixUnaryExpression */:
+                return visitNode(cbNode, node.operand);
+            case 219 /* YieldExpression */:
+                return visitNode(cbNode, node.asteriskToken) ||
+                    visitNode(cbNode, node.expression);
+            case 213 /* AwaitExpression */:
+                return visitNode(cbNode, node.expression);
+            case 215 /* PostfixUnaryExpression */:
+                return visitNode(cbNode, node.operand);
+            case 216 /* BinaryExpression */:
+                return visitNode(cbNode, node.left) ||
+                    visitNode(cbNode, node.operatorToken) ||
+                    visitNode(cbNode, node.right);
+            case 224 /* AsExpression */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.type);
+            case 225 /* NonNullExpression */:
+                return visitNode(cbNode, node.expression);
+            case 226 /* MetaProperty */:
+                return visitNode(cbNode, node.name);
+            case 217 /* ConditionalExpression */:
+                return visitNode(cbNode, node.condition) ||
+                    visitNode(cbNode, node.questionToken) ||
+                    visitNode(cbNode, node.whenTrue) ||
+                    visitNode(cbNode, node.colonToken) ||
+                    visitNode(cbNode, node.whenFalse);
+            case 220 /* SpreadElement */:
+                return visitNode(cbNode, node.expression);
+            case 230 /* Block */:
+            case 257 /* ModuleBlock */:
+                return visitNodes(cbNode, cbNodes, node.statements);
+            case 297 /* SourceFile */:
+                return visitNodes(cbNode, cbNodes, node.statements) ||
+                    visitNode(cbNode, node.endOfFileToken);
+            case 232 /* VariableStatement */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.declarationList);
+            case 250 /* VariableDeclarationList */:
+                return visitNodes(cbNode, cbNodes, node.declarations);
+            case 233 /* ExpressionStatement */:
+                return visitNode(cbNode, node.expression);
+            case 234 /* IfStatement */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.thenStatement) ||
+                    visitNode(cbNode, node.elseStatement);
+            case 235 /* DoStatement */:
+                return visitNode(cbNode, node.statement) ||
+                    visitNode(cbNode, node.expression);
+            case 236 /* WhileStatement */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.statement);
+            case 237 /* ForStatement */:
+                return visitNode(cbNode, node.initializer) ||
+                    visitNode(cbNode, node.condition) ||
+                    visitNode(cbNode, node.incrementor) ||
+                    visitNode(cbNode, node.statement);
+            case 238 /* ForInStatement */:
+                return visitNode(cbNode, node.initializer) ||
+                    visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.statement);
+            case 239 /* ForOfStatement */:
+                return visitNode(cbNode, node.awaitModifier) ||
+                    visitNode(cbNode, node.initializer) ||
+                    visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.statement);
+            case 240 /* ContinueStatement */:
+            case 241 /* BreakStatement */:
+                return visitNode(cbNode, node.label);
+            case 242 /* ReturnStatement */:
+                return visitNode(cbNode, node.expression);
+            case 243 /* WithStatement */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.statement);
+            case 244 /* SwitchStatement */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNode(cbNode, node.caseBlock);
+            case 258 /* CaseBlock */:
+                return visitNodes(cbNode, cbNodes, node.clauses);
+            case 284 /* CaseClause */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNodes(cbNode, cbNodes, node.statements);
+            case 285 /* DefaultClause */:
+                return visitNodes(cbNode, cbNodes, node.statements);
+            case 245 /* LabeledStatement */:
+                return visitNode(cbNode, node.label) ||
+                    visitNode(cbNode, node.statement);
+            case 246 /* ThrowStatement */:
+                return visitNode(cbNode, node.expression);
+            case 247 /* TryStatement */:
+                return visitNode(cbNode, node.tryBlock) ||
+                    visitNode(cbNode, node.catchClause) ||
+                    visitNode(cbNode, node.finallyBlock);
+            case 287 /* CatchClause */:
+                return visitNode(cbNode, node.variableDeclaration) ||
+                    visitNode(cbNode, node.block);
+            case 161 /* Decorator */:
+                return visitNode(cbNode, node.expression);
+            case 252 /* ClassDeclaration */:
+            case 221 /* ClassExpression */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
+                    visitNodes(cbNode, cbNodes, node.heritageClauses) ||
+                    visitNodes(cbNode, cbNodes, node.members);
+            case 253 /* InterfaceDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
+                    visitNodes(cbNode, cbNodes, node.heritageClauses) ||
+                    visitNodes(cbNode, cbNodes, node.members);
+            case 254 /* TypeAliasDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNodes(cbNode, cbNodes, node.typeParameters) ||
+                    visitNode(cbNode, node.type);
+            case 255 /* EnumDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNodes(cbNode, cbNodes, node.members);
+            case 291 /* EnumMember */:
+                return visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.initializer);
+            case 256 /* ModuleDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.body);
+            case 260 /* ImportEqualsDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.moduleReference);
+            case 261 /* ImportDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.importClause) ||
+                    visitNode(cbNode, node.moduleSpecifier);
+            case 262 /* ImportClause */:
+                return visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.namedBindings);
+            case 259 /* NamespaceExportDeclaration */:
+                return visitNode(cbNode, node.name);
+            case 263 /* NamespaceImport */:
+                return visitNode(cbNode, node.name);
+            case 269 /* NamespaceExport */:
+                return visitNode(cbNode, node.name);
+            case 264 /* NamedImports */:
+            case 268 /* NamedExports */:
+                return visitNodes(cbNode, cbNodes, node.elements);
+            case 267 /* ExportDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.exportClause) ||
+                    visitNode(cbNode, node.moduleSpecifier);
+            case 265 /* ImportSpecifier */:
+            case 270 /* ExportSpecifier */:
+                return visitNode(cbNode, node.propertyName) ||
+                    visitNode(cbNode, node.name);
+            case 266 /* ExportAssignment */:
+                return visitNodes(cbNode, cbNodes, node.decorators) ||
+                    visitNodes(cbNode, cbNodes, node.modifiers) ||
+                    visitNode(cbNode, node.expression);
+            case 218 /* TemplateExpression */:
+                return visitNode(cbNode, node.head) || visitNodes(cbNode, cbNodes, node.templateSpans);
+            case 228 /* TemplateSpan */:
+                return visitNode(cbNode, node.expression) || visitNode(cbNode, node.literal);
+            case 193 /* TemplateLiteralType */:
+                return visitNode(cbNode, node.head) || visitNodes(cbNode, cbNodes, node.templateSpans);
+            case 194 /* TemplateLiteralTypeSpan */:
+                return visitNode(cbNode, node.type) || visitNode(cbNode, node.literal);
+            case 158 /* ComputedPropertyName */:
+                return visitNode(cbNode, node.expression);
+            case 286 /* HeritageClause */:
+                return visitNodes(cbNode, cbNodes, node.types);
+            case 223 /* ExpressionWithTypeArguments */:
+                return visitNode(cbNode, node.expression) ||
+                    visitNodes(cbNode, cbNodes, node.typeArguments);
+            case 272 /* ExternalModuleReference */:
+                return visitNode(cbNode, node.expression);
+            case 271 /* MissingDeclaration */:
+                return visitNodes(cbNode, cbNodes, node.decorators);
+            case 337 /* CommaListExpression */:
+                return visitNodes(cbNode, cbNodes, node.elements);
+            case 273 /* JsxElement */:
+                return visitNode(cbNode, node.openingElement) ||
+                    visitNodes(cbNode, cbNodes, node.children) ||
+                    visitNode(cbNode, node.closingElement);
+            case 277 /* JsxFragment */:
+                return visitNode(cbNode, node.openingFragment) ||
+                    visitNodes(cbNode, cbNodes, node.children) ||
+                    visitNode(cbNode, node.closingFragment);
+            case 274 /* JsxSelfClosingElement */:
+            case 275 /* JsxOpeningElement */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNodes(cbNode, cbNodes, node.typeArguments) ||
+                    visitNode(cbNode, node.attributes);
+            case 281 /* JsxAttributes */:
+                return visitNodes(cbNode, cbNodes, node.properties);
+            case 280 /* JsxAttribute */:
+                return visitNode(cbNode, node.name) ||
+                    visitNode(cbNode, node.initializer);
+            case 282 /* JsxSpreadAttribute */:
+                return visitNode(cbNode, node.expression);
+            case 283 /* JsxExpression */:
+                return visitNode(cbNode, node.dotDotDotToken) ||
+                    visitNode(cbNode, node.expression);
+            case 276 /* JsxClosingElement */:
+                return visitNode(cbNode, node.tagName);
+            case 180 /* OptionalType */:
+            case 181 /* RestType */:
+            case 301 /* JSDocTypeExpression */:
+            case 306 /* JSDocNonNullableType */:
+            case 305 /* JSDocNullableType */:
+            case 307 /* JSDocOptionalType */:
+            case 309 /* JSDocVariadicType */:
+                return visitNode(cbNode, node.type);
+            case 308 /* JSDocFunctionType */:
+                return visitNodes(cbNode, cbNodes, node.parameters) ||
+                    visitNode(cbNode, node.type);
+            case 311 /* JSDocComment */:
+                return visitNodes(cbNode, cbNodes, node.tags);
+            case 332 /* JSDocSeeTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNode(cbNode, node.name);
+            case 302 /* JSDocNameReference */:
+                return visitNode(cbNode, node.name);
+            case 326 /* JSDocParameterTag */:
+            case 333 /* JSDocPropertyTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    (node.isNameFirst
+                        ? visitNode(cbNode, node.name) ||
+                            visitNode(cbNode, node.typeExpression)
+                        : visitNode(cbNode, node.typeExpression) ||
+                            visitNode(cbNode, node.name));
+            case 317 /* JSDocAuthorTag */:
+                return visitNode(cbNode, node.tagName);
+            case 316 /* JSDocImplementsTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNode(cbNode, node.class);
+            case 315 /* JSDocAugmentsTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNode(cbNode, node.class);
+            case 330 /* JSDocTemplateTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNode(cbNode, node.constraint) ||
+                    visitNodes(cbNode, cbNodes, node.typeParameters);
+            case 331 /* JSDocTypedefTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    (node.typeExpression &&
+                        node.typeExpression.kind === 301 /* JSDocTypeExpression */
+                        ? visitNode(cbNode, node.typeExpression) ||
+                            visitNode(cbNode, node.fullName)
+                        : visitNode(cbNode, node.fullName) ||
+                            visitNode(cbNode, node.typeExpression));
+            case 324 /* JSDocCallbackTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNode(cbNode, node.fullName) ||
+                    visitNode(cbNode, node.typeExpression);
+            case 327 /* JSDocReturnTag */:
+            case 329 /* JSDocTypeTag */:
+            case 328 /* JSDocThisTag */:
+            case 325 /* JSDocEnumTag */:
+                return visitNode(cbNode, node.tagName) ||
+                    visitNode(cbNode, node.typeExpression);
+            case 313 /* JSDocSignature */:
+                return ts.forEach(node.typeParameters, cbNode) ||
+                    ts.forEach(node.parameters, cbNode) ||
+                    visitNode(cbNode, node.type);
+            case 312 /* JSDocTypeLiteral */:
+                return ts.forEach(node.jsDocPropertyTags, cbNode);
+            case 314 /* JSDocTag */:
+            case 319 /* JSDocClassTag */:
+            case 320 /* JSDocPublicTag */:
+            case 321 /* JSDocPrivateTag */:
+            case 322 /* JSDocProtectedTag */:
+            case 323 /* JSDocReadonlyTag */:
+                return visitNode(cbNode, node.tagName);
+            case 336 /* PartiallyEmittedExpression */:
+                return visitNode(cbNode, node.expression);
         }
-        function walkTreeForExternalModuleIndicators(node) {
-            return isImportMeta(node) ? node : forEachChild(node, walkTreeForExternalModuleIndicators);
+    }
+    ts.forEachChild = forEachChild;
+    /** @internal */
+    /**
+     * Invokes a callback for each child of the given node. The 'cbNode' callback is invoked for all child nodes
+     * stored in properties. If a 'cbNodes' callback is specified, it is invoked for embedded arrays; additionally,
+     * unlike `forEachChild`, embedded arrays are flattened and the 'cbNode' callback is invoked for each element.
+     *  If a callback returns a truthy value, iteration stops and that value is returned. Otherwise, undefined is returned.
+     *
+     * @param node a given node to visit its children
+     * @param cbNode a callback to be invoked for all child nodes
+     * @param cbNodes a callback to be invoked for embedded array
+     *
+     * @remarks Unlike `forEachChild`, `forEachChildRecursively` handles recursively invoking the traversal on each child node found,
+     * and while doing so, handles traversing the structure without relying on the callstack to encode the tree structure.
+     */
+    function forEachChildRecursively(rootNode, cbNode, cbNodes) {
+        var stack = [rootNode];
+        while (stack.length) {
+            var parent = stack.pop();
+            var res = visitAllPossibleChildren(parent, gatherPossibleChildren(parent));
+            if (res) {
+                return res;
+            }
         }
-        /** Do not use hasModifier inside the parser; it relies on parent pointers. Use this instead. */
-        function hasModifierOfKind(node, kind) {
-            return ts.some(node.modifiers, function (m) { return m.kind === kind; });
+        return;
+        function gatherPossibleChildren(node) {
+            var children = [];
+            forEachChild(node, addWorkItem, addWorkItem); // By using a stack above and `unshift` here, we emulate a depth-first preorder traversal
+            return children;
+            function addWorkItem(n) {
+                children.unshift(n);
+            }
         }
-        function isImportMeta(node) {
-            return ts.isMetaProperty(node) && node.keywordToken === 96 /* ImportKeyword */ && node.name.escapedText === "meta";
+        function visitAllPossibleChildren(parent, children) {
+            for (var _i = 0, children_5 = children; _i < children_5.length; _i++) {
+                var child = children_5[_i];
+                if (ts.isArray(child)) {
+                    if (cbNodes) {
+                        var res = cbNodes(child, parent);
+                        if (res) {
+                            if (res === "skip")
+                                continue;
+                            return res;
+                        }
+                    }
+                    for (var i = child.length - 1; i >= 0; i--) {
+                        var realChild = child[i];
+                        var res = cbNode(realChild, parent);
+                        if (res) {
+                            if (res === "skip")
+                                continue;
+                            return res;
+                        }
+                        stack.push(realChild);
+                    }
+                }
+                else {
+                    stack.push(child);
+                    var res = cbNode(child, parent);
+                    if (res) {
+                        if (res === "skip")
+                            continue;
+                        return res;
+                    }
+                }
+            }
         }
-        var ParsingContext;
-        (function (ParsingContext) {
-            ParsingContext[ParsingContext["SourceElements"] = 0] = "SourceElements";
-            ParsingContext[ParsingContext["BlockStatements"] = 1] = "BlockStatements";
-            ParsingContext[ParsingContext["SwitchClauses"] = 2] = "SwitchClauses";
-            ParsingContext[ParsingContext["SwitchClauseStatements"] = 3] = "SwitchClauseStatements";
-            ParsingContext[ParsingContext["TypeMembers"] = 4] = "TypeMembers";
-            ParsingContext[ParsingContext["ClassMembers"] = 5] = "ClassMembers";
-            ParsingContext[ParsingContext["EnumMembers"] = 6] = "EnumMembers";
-            ParsingContext[ParsingContext["HeritageClauseElement"] = 7] = "HeritageClauseElement";
-            ParsingContext[ParsingContext["VariableDeclarations"] = 8] = "VariableDeclarations";
-            ParsingContext[ParsingContext["ObjectBindingElements"] = 9] = "ObjectBindingElements";
-            ParsingContext[ParsingContext["ArrayBindingElements"] = 10] = "ArrayBindingElements";
-            ParsingContext[ParsingContext["ArgumentExpressions"] = 11] = "ArgumentExpressions";
-            ParsingContext[ParsingContext["ObjectLiteralMembers"] = 12] = "ObjectLiteralMembers";
-            ParsingContext[ParsingContext["JsxAttributes"] = 13] = "JsxAttributes";
-            ParsingContext[ParsingContext["JsxChildren"] = 14] = "JsxChildren";
-            ParsingContext[ParsingContext["ArrayLiteralMembers"] = 15] = "ArrayLiteralMembers";
-            ParsingContext[ParsingContext["Parameters"] = 16] = "Parameters";
-            ParsingContext[ParsingContext["JSDocParameters"] = 17] = "JSDocParameters";
-            ParsingContext[ParsingContext["RestProperties"] = 18] = "RestProperties";
-            ParsingContext[ParsingContext["TypeParameters"] = 19] = "TypeParameters";
-            ParsingContext[ParsingContext["TypeArguments"] = 20] = "TypeArguments";
-            ParsingContext[ParsingContext["TupleElementTypes"] = 21] = "TupleElementTypes";
-            ParsingContext[ParsingContext["HeritageClauses"] = 22] = "HeritageClauses";
-            ParsingContext[ParsingContext["ImportOrExportSpecifiers"] = 23] = "ImportOrExportSpecifiers";
-            ParsingContext[ParsingContext["Count"] = 24] = "Count"; // Number of parsing contexts
-        })(ParsingContext || (ParsingContext = {}));
-        var Tristate;
-        (function (Tristate) {
-            Tristate[Tristate["False"] = 0] = "False";
-            Tristate[Tristate["True"] = 1] = "True";
-            Tristate[Tristate["Unknown"] = 2] = "Unknown";
-        })(Tristate || (Tristate = {}));
-        var JSDocParser;
-        (function (JSDocParser) {
-            function parseJSDocTypeExpressionForTests(content, start, length) {
-                initializeState(content, 99 /* Latest */, /*_syntaxCursor:*/ undefined, 1 /* JS */);
-                sourceFile = createSourceFile("file.js", 99 /* Latest */, 1 /* JS */, /*isDeclarationFile*/ false);
-                scanner.setText(content, start, length);
-                currentToken = scanner.scan();
-                var jsDocTypeExpression = parseJSDocTypeExpression();
-                var diagnostics = parseDiagnostics;
-                clearState();
-                return jsDocTypeExpression ? { jsDocTypeExpression: jsDocTypeExpression, diagnostics: diagnostics } : undefined;
-            }
-            JSDocParser.parseJSDocTypeExpressionForTests = parseJSDocTypeExpressionForTests;
-            // Parses out a JSDoc type expression.
-            function parseJSDocTypeExpression(mayOmitBraces) {
-                var result = createNode(294 /* JSDocTypeExpression */);
-                var hasBrace = (mayOmitBraces ? parseOptional : parseExpected)(18 /* OpenBraceToken */);
-                result.type = doInsideOfContext(4194304 /* JSDoc */, parseJSDocType);
-                if (!mayOmitBraces || hasBrace) {
-                    parseExpectedJSDoc(19 /* CloseBraceToken */);
-                }
-                fixupParentReferences(result);
-                return finishNode(result);
-            }
-            JSDocParser.parseJSDocTypeExpression = parseJSDocTypeExpression;
-            function parseIsolatedJSDocComment(content, start, length) {
-                initializeState(content, 99 /* Latest */, /*_syntaxCursor:*/ undefined, 1 /* JS */);
-                sourceFile = { languageVariant: 0 /* Standard */, text: content };
-                var jsDoc = doInsideOfContext(4194304 /* JSDoc */, function () { return parseJSDocCommentWorker(start, length); });
-                var diagnostics = parseDiagnostics;
-                clearState();
-                return jsDoc ? { jsDoc: jsDoc, diagnostics: diagnostics } : undefined;
-            }
-            JSDocParser.parseIsolatedJSDocComment = parseIsolatedJSDocComment;
-            function parseJSDocComment(parent, start, length) {
-                var _a;
-                var saveToken = currentToken;
-                var saveParseDiagnosticsLength = parseDiagnostics.length;
-                var saveParseErrorBeforeNextFinishedNode = parseErrorBeforeNextFinishedNode;
-                var comment = doInsideOfContext(4194304 /* JSDoc */, function () { return parseJSDocCommentWorker(start, length); });
-                if (comment) {
-                    comment.parent = parent;
-                }
-                if (contextFlags & 131072 /* JavaScriptFile */) {
-                    if (!sourceFile.jsDocDiagnostics) {
-                        sourceFile.jsDocDiagnostics = [];
-                    }
-                    (_a = sourceFile.jsDocDiagnostics).push.apply(_a, parseDiagnostics);
-                }
-                currentToken = saveToken;
-                parseDiagnostics.length = saveParseDiagnosticsLength;
-                parseErrorBeforeNextFinishedNode = saveParseErrorBeforeNextFinishedNode;
-                return comment;
+    }
+    ts.forEachChildRecursively = forEachChildRecursively;
+    function createSourceFile(fileName, sourceText, languageVersion, setParentNodes, scriptKind) {
+        if (setParentNodes === void 0) { setParentNodes = false; }
+        var tracingData = ["parse" /* Parse */, "createSourceFile", { path: fileName }];
+        ts.tracing.begin.apply(ts.tracing, tracingData);
+        ts.performance.mark("beforeParse");
+        var result;
+        ts.perfLogger.logStartParseSourceFile(fileName);
+        if (languageVersion === 100 /* JSON */) {
+            result = Parser.parseSourceFile(fileName, sourceText, languageVersion, /*syntaxCursor*/ undefined, setParentNodes, 6 /* JSON */);
+        }
+        else {
+            result = Parser.parseSourceFile(fileName, sourceText, languageVersion, /*syntaxCursor*/ undefined, setParentNodes, scriptKind);
+        }
+        ts.perfLogger.logStopParseSourceFile();
+        ts.performance.mark("afterParse");
+        ts.performance.measure("Parse", "beforeParse", "afterParse");
+        ts.tracing.end.apply(ts.tracing, tracingData);
+        return result;
+    }
+    ts.createSourceFile = createSourceFile;
+    function parseIsolatedEntityName(text, languageVersion) {
+        return Parser.parseIsolatedEntityName(text, languageVersion);
+    }
+    ts.parseIsolatedEntityName = parseIsolatedEntityName;
+    /**
+     * Parse json text into SyntaxTree and return node and parse errors if any
+     * @param fileName
+     * @param sourceText
+     */
+    function parseJsonText(fileName, sourceText) {
+        return Parser.parseJsonText(fileName, sourceText);
+    }
+    ts.parseJsonText = parseJsonText;
+    // See also `isExternalOrCommonJsModule` in utilities.ts
+    function isExternalModule(file) {
+        return file.externalModuleIndicator !== undefined;
+    }
+    ts.isExternalModule = isExternalModule;
+    // Produces a new SourceFile for the 'newText' provided. The 'textChangeRange' parameter
+    // indicates what changed between the 'text' that this SourceFile has and the 'newText'.
+    // The SourceFile will be created with the compiler attempting to reuse as many nodes from
+    // this file as possible.
+    //
+    // Note: this function mutates nodes from this SourceFile. That means any existing nodes
+    // from this SourceFile that are being held onto may change as a result (including
+    // becoming detached from any SourceFile).  It is recommended that this SourceFile not
+    // be used once 'update' is called on it.
+    function updateSourceFile(sourceFile, newText, textChangeRange, aggressiveChecks) {
+        if (aggressiveChecks === void 0) { aggressiveChecks = false; }
+        var newSourceFile = IncrementalParser.updateSourceFile(sourceFile, newText, textChangeRange, aggressiveChecks);
+        // Because new source file node is created, it may not have the flag PossiblyContainDynamicImport. This is the case if there is no new edit to add dynamic import.
+        // We will manually port the flag to the new source file.
+        newSourceFile.flags |= (sourceFile.flags & 3145728 /* PermanentlySetIncrementalFlags */);
+        return newSourceFile;
+    }
+    ts.updateSourceFile = updateSourceFile;
+    /* @internal */
+    function parseIsolatedJSDocComment(content, start, length) {
+        var result = Parser.JSDocParser.parseIsolatedJSDocComment(content, start, length);
+        if (result && result.jsDoc) {
+            // because the jsDocComment was parsed out of the source file, it might
+            // not be covered by the fixupParentReferences.
+            Parser.fixupParentReferences(result.jsDoc);
+        }
+        return result;
+    }
+    ts.parseIsolatedJSDocComment = parseIsolatedJSDocComment;
+    /* @internal */
+    // Exposed only for testing.
+    function parseJSDocTypeExpressionForTests(content, start, length) {
+        return Parser.JSDocParser.parseJSDocTypeExpressionForTests(content, start, length);
+    }
+    ts.parseJSDocTypeExpressionForTests = parseJSDocTypeExpressionForTests;
+    // Implement the parser as a singleton module.  We do this for perf reasons because creating
+    // parser instances can actually be expensive enough to impact us on projects with many source
+    // files.
+    var Parser;
+    (function (Parser) {
+        // Share a single scanner across all calls to parse a source file.  This helps speed things
+        // up by avoiding the cost of creating/compiling scanners over and over again.
+        var scanner = ts.createScanner(99 /* Latest */, /*skipTrivia*/ true);
+        var disallowInAndDecoratorContext = 4096 /* DisallowInContext */ | 16384 /* DecoratorContext */;
+        // capture constructors in 'initializeState' to avoid null checks
+        // tslint:disable variable-name
+        var NodeConstructor;
+        var TokenConstructor;
+        var IdentifierConstructor;
+        var PrivateIdentifierConstructor;
+        var SourceFileConstructor;
+        // tslint:enable variable-name
+        function countNode(node) {
+            nodeCount++;
+            return node;
+        }
+        // Rather than using `createBaseNodeFactory` here, we establish a `BaseNodeFactory` that closes over the
+        // constructors above, which are reset each time `initializeState` is called.
+        var baseNodeFactory = {
+            createBaseSourceFileNode: function (kind) { return countNode(new SourceFileConstructor(kind, /*pos*/ 0, /*end*/ 0)); },
+            createBaseIdentifierNode: function (kind) { return countNode(new IdentifierConstructor(kind, /*pos*/ 0, /*end*/ 0)); },
+            createBasePrivateIdentifierNode: function (kind) { return countNode(new PrivateIdentifierConstructor(kind, /*pos*/ 0, /*end*/ 0)); },
+            createBaseTokenNode: function (kind) { return countNode(new TokenConstructor(kind, /*pos*/ 0, /*end*/ 0)); },
+            createBaseNode: function (kind) { return countNode(new NodeConstructor(kind, /*pos*/ 0, /*end*/ 0)); }
+        };
+        var factory = ts.createNodeFactory(1 /* NoParenthesizerRules */ | 2 /* NoNodeConverters */ | 8 /* NoOriginalNode */, baseNodeFactory);
+        var fileName;
+        var sourceFlags;
+        var sourceText;
+        var languageVersion;
+        var scriptKind;
+        var languageVariant;
+        var parseDiagnostics;
+        var jsDocDiagnostics;
+        var syntaxCursor;
+        var currentToken;
+        var nodeCount;
+        var identifiers;
+        var privateIdentifiers;
+        var identifierCount;
+        var parsingContext;
+        var notParenthesizedArrow;
+        // Flags that dictate what parsing context we're in.  For example:
+        // Whether or not we are in strict parsing mode.  All that changes in strict parsing mode is
+        // that some tokens that would be considered identifiers may be considered keywords.
+        //
+        // When adding more parser context flags, consider which is the more common case that the
+        // flag will be in.  This should be the 'false' state for that flag.  The reason for this is
+        // that we don't store data in our nodes unless the value is in the *non-default* state.  So,
+        // for example, more often than code 'allows-in' (or doesn't 'disallow-in').  We opt for
+        // 'disallow-in' set to 'false'.  Otherwise, if we had 'allowsIn' set to 'true', then almost
+        // all nodes would need extra state on them to store this info.
+        //
+        // Note: 'allowIn' and 'allowYield' track 1:1 with the [in] and [yield] concepts in the ES6
+        // grammar specification.
+        //
+        // An important thing about these context concepts.  By default they are effectively inherited
+        // while parsing through every grammar production.  i.e. if you don't change them, then when
+        // you parse a sub-production, it will have the same context values as the parent production.
+        // This is great most of the time.  After all, consider all the 'expression' grammar productions
+        // and how nearly all of them pass along the 'in' and 'yield' context values:
+        //
+        // EqualityExpression[In, Yield] :
+        //      RelationalExpression[?In, ?Yield]
+        //      EqualityExpression[?In, ?Yield] == RelationalExpression[?In, ?Yield]
+        //      EqualityExpression[?In, ?Yield] != RelationalExpression[?In, ?Yield]
+        //      EqualityExpression[?In, ?Yield] === RelationalExpression[?In, ?Yield]
+        //      EqualityExpression[?In, ?Yield] !== RelationalExpression[?In, ?Yield]
+        //
+        // Where you have to be careful is then understanding what the points are in the grammar
+        // where the values are *not* passed along.  For example:
+        //
+        // SingleNameBinding[Yield,GeneratorParameter]
+        //      [+GeneratorParameter]BindingIdentifier[Yield] Initializer[In]opt
+        //      [~GeneratorParameter]BindingIdentifier[?Yield]Initializer[In, ?Yield]opt
+        //
+        // Here this is saying that if the GeneratorParameter context flag is set, that we should
+        // explicitly set the 'yield' context flag to false before calling into the BindingIdentifier
+        // and we should explicitly unset the 'yield' context flag before calling into the Initializer.
+        // production.  Conversely, if the GeneratorParameter context flag is not set, then we
+        // should leave the 'yield' context flag alone.
+        //
+        // Getting this all correct is tricky and requires careful reading of the grammar to
+        // understand when these values should be changed versus when they should be inherited.
+        //
+        // Note: it should not be necessary to save/restore these flags during speculative/lookahead
+        // parsing.  These context flags are naturally stored and restored through normal recursive
+        // descent parsing and unwinding.
+        var contextFlags;
+        // Indicates whether we are currently parsing top-level statements.
+        var topLevel = true;
+        // Whether or not we've had a parse error since creating the last AST node.  If we have
+        // encountered an error, it will be stored on the next AST node we create.  Parse errors
+        // can be broken down into three categories:
+        //
+        // 1) An error that occurred during scanning.  For example, an unterminated literal, or a
+        //    character that was completely not understood.
+        //
+        // 2) A token was expected, but was not present.  This type of error is commonly produced
+        //    by the 'parseExpected' function.
+        //
+        // 3) A token was present that no parsing function was able to consume.  This type of error
+        //    only occurs in the 'abortParsingListOrMoveToNextToken' function when the parser
+        //    decides to skip the token.
+        //
+        // In all of these cases, we want to mark the next node as having had an error before it.
+        // With this mark, we can know in incremental settings if this node can be reused, or if
+        // we have to reparse it.  If we don't keep this information around, we may just reuse the
+        // node.  in that event we would then not produce the same errors as we did before, causing
+        // significant confusion problems.
+        //
+        // Note: it is necessary that this value be saved/restored during speculative/lookahead
+        // parsing.  During lookahead parsing, we will often create a node.  That node will have
+        // this value attached, and then this value will be set back to 'false'.  If we decide to
+        // rewind, we must get back to the same value we had prior to the lookahead.
+        //
+        // Note: any errors at the end of the file that do not precede a regular node, should get
+        // attached to the EOF token.
+        var parseErrorBeforeNextFinishedNode = false;
+        function parseSourceFile(fileName, sourceText, languageVersion, syntaxCursor, setParentNodes, scriptKind) {
+            if (setParentNodes === void 0) { setParentNodes = false; }
+            scriptKind = ts.ensureScriptKind(fileName, scriptKind);
+            if (scriptKind === 6 /* JSON */) {
+                var result_3 = parseJsonText(fileName, sourceText, languageVersion, syntaxCursor, setParentNodes);
+                ts.convertToObjectWorker(result_3, result_3.parseDiagnostics, /*returnValue*/ false, /*knownRootOptions*/ undefined, /*jsonConversionNotifier*/ undefined);
+                result_3.referencedFiles = ts.emptyArray;
+                result_3.typeReferenceDirectives = ts.emptyArray;
+                result_3.libReferenceDirectives = ts.emptyArray;
+                result_3.amdDependencies = ts.emptyArray;
+                result_3.hasNoDefaultLib = false;
+                result_3.pragmas = ts.emptyMap;
+                return result_3;
+            }
+            initializeState(fileName, sourceText, languageVersion, syntaxCursor, scriptKind);
+            var result = parseSourceFileWorker(languageVersion, setParentNodes, scriptKind);
+            clearState();
+            return result;
+        }
+        Parser.parseSourceFile = parseSourceFile;
+        function parseIsolatedEntityName(content, languageVersion) {
+            // Choice of `isDeclarationFile` should be arbitrary
+            initializeState("", content, languageVersion, /*syntaxCursor*/ undefined, 1 /* JS */);
+            // Prime the scanner.
+            nextToken();
+            var entityName = parseEntityName(/*allowReservedWords*/ true);
+            var isInvalid = token() === 1 /* EndOfFileToken */ && !parseDiagnostics.length;
+            clearState();
+            return isInvalid ? entityName : undefined;
+        }
+        Parser.parseIsolatedEntityName = parseIsolatedEntityName;
+        function parseJsonText(fileName, sourceText, languageVersion, syntaxCursor, setParentNodes) {
+            if (languageVersion === void 0) { languageVersion = 2 /* ES2015 */; }
+            if (setParentNodes === void 0) { setParentNodes = false; }
+            initializeState(fileName, sourceText, languageVersion, syntaxCursor, 6 /* JSON */);
+            sourceFlags = contextFlags;
+            // Prime the scanner.
+            nextToken();
+            var pos = getNodePos();
+            var statements, endOfFileToken;
+            if (token() === 1 /* EndOfFileToken */) {
+                statements = createNodeArray([], pos, pos);
+                endOfFileToken = parseTokenNode();
             }
-            JSDocParser.parseJSDocComment = parseJSDocComment;
-            var JSDocState;
-            (function (JSDocState) {
-                JSDocState[JSDocState["BeginningOfLine"] = 0] = "BeginningOfLine";
-                JSDocState[JSDocState["SawAsterisk"] = 1] = "SawAsterisk";
-                JSDocState[JSDocState["SavingComments"] = 2] = "SavingComments";
-                JSDocState[JSDocState["SavingBackticks"] = 3] = "SavingBackticks";
-            })(JSDocState || (JSDocState = {}));
-            var PropertyLikeParse;
-            (function (PropertyLikeParse) {
-                PropertyLikeParse[PropertyLikeParse["Property"] = 1] = "Property";
-                PropertyLikeParse[PropertyLikeParse["Parameter"] = 2] = "Parameter";
-                PropertyLikeParse[PropertyLikeParse["CallbackParameter"] = 4] = "CallbackParameter";
-            })(PropertyLikeParse || (PropertyLikeParse = {}));
-            function parseJSDocCommentWorker(start, length) {
-                if (start === void 0) { start = 0; }
-                var content = sourceText;
-                var end = length === undefined ? content.length : start + length;
-                length = end - start;
-                ts.Debug.assert(start >= 0);
-                ts.Debug.assert(start <= end);
-                ts.Debug.assert(end <= content.length);
-                // Check for /** (JSDoc opening part)
-                if (!isJSDocLikeText(content, start)) {
-                    return undefined;
-                }
-                var tags;
-                var tagsPos;
-                var tagsEnd;
-                var comments = [];
-                // + 3 for leading /**, - 5 in total for /** */
-                return scanner.scanRange(start + 3, length - 5, function () {
-                    // Initially we can parse out a tag.  We also have seen a starting asterisk.
-                    // This is so that /** * @type */ doesn't parse.
-                    var state = 1 /* SawAsterisk */;
-                    var margin;
-                    // + 4 for leading '/** '
-                    var indent = start - Math.max(content.lastIndexOf("\n", start), 0) + 4;
-                    function pushComment(text) {
-                        if (!margin) {
-                            margin = indent;
-                        }
-                        comments.push(text);
-                        indent += text.length;
-                    }
-                    nextTokenJSDoc();
-                    while (parseOptionalJsdoc(5 /* WhitespaceTrivia */))
-                        ;
-                    if (parseOptionalJsdoc(4 /* NewLineTrivia */)) {
-                        state = 0 /* BeginningOfLine */;
-                        indent = 0;
-                    }
-                    loop: while (true) {
-                        switch (token()) {
-                            case 59 /* AtToken */:
-                                if (state === 0 /* BeginningOfLine */ || state === 1 /* SawAsterisk */) {
-                                    removeTrailingWhitespace(comments);
-                                    addTag(parseTag(indent));
-                                    // NOTE: According to usejsdoc.org, a tag goes to end of line, except the last tag.
-                                    // Real-world comments may break this rule, so "BeginningOfLine" will not be a real line beginning
-                                    // for malformed examples like `/** @param {string} x @returns {number} the length */`
-                                    state = 0 /* BeginningOfLine */;
-                                    margin = undefined;
-                                }
-                                else {
-                                    pushComment(scanner.getTokenText());
-                                }
-                                break;
-                            case 4 /* NewLineTrivia */:
-                                comments.push(scanner.getTokenText());
-                                state = 0 /* BeginningOfLine */;
-                                indent = 0;
-                                break;
-                            case 41 /* AsteriskToken */:
-                                var asterisk = scanner.getTokenText();
-                                if (state === 1 /* SawAsterisk */ || state === 2 /* SavingComments */) {
-                                    // If we've already seen an asterisk, then we can no longer parse a tag on this line
-                                    state = 2 /* SavingComments */;
-                                    pushComment(asterisk);
-                                }
-                                else {
-                                    // Ignore the first asterisk on a line
-                                    state = 1 /* SawAsterisk */;
-                                    indent += asterisk.length;
-                                }
-                                break;
-                            case 5 /* WhitespaceTrivia */:
-                                // only collect whitespace if we're already saving comments or have just crossed the comment indent margin
-                                var whitespace = scanner.getTokenText();
-                                if (state === 2 /* SavingComments */) {
-                                    comments.push(whitespace);
-                                }
-                                else if (margin !== undefined && indent + whitespace.length > margin) {
-                                    comments.push(whitespace.slice(margin - indent - 1));
-                                }
-                                indent += whitespace.length;
-                                break;
-                            case 1 /* EndOfFileToken */:
-                                break loop;
-                            default:
-                                // Anything else is doc comment text. We just save it. Because it
-                                // wasn't a tag, we can no longer parse a tag on this line until we hit the next
-                                // line break.
-                                state = 2 /* SavingComments */;
-                                pushComment(scanner.getTokenText());
-                                break;
-                        }
-                        nextTokenJSDoc();
-                    }
-                    removeLeadingNewlines(comments);
-                    removeTrailingWhitespace(comments);
-                    return createJSDocComment();
-                });
-                function removeLeadingNewlines(comments) {
-                    while (comments.length && (comments[0] === "\n" || comments[0] === "\r")) {
-                        comments.shift();
-                    }
-                }
-                function removeTrailingWhitespace(comments) {
-                    while (comments.length && comments[comments.length - 1].trim() === "") {
-                        comments.pop();
-                    }
-                }
-                function createJSDocComment() {
-                    var result = createNode(303 /* JSDocComment */, start);
-                    result.tags = tags && createNodeArray(tags, tagsPos, tagsEnd);
-                    result.comment = comments.length ? comments.join("") : undefined;
-                    return finishNode(result, end);
-                }
-                function isNextNonwhitespaceTokenEndOfFile() {
-                    // We must use infinite lookahead, as there could be any number of newlines :(
-                    while (true) {
-                        nextTokenJSDoc();
-                        if (token() === 1 /* EndOfFileToken */) {
-                            return true;
-                        }
-                        if (!(token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */)) {
-                            return false;
-                        }
-                    }
-                }
-                function skipWhitespace() {
-                    if (token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
-                        if (lookAhead(isNextNonwhitespaceTokenEndOfFile)) {
-                            return; // Don't skip whitespace prior to EoF (or end of comment) - that shouldn't be included in any node's range
-                        }
-                    }
-                    while (token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
-                        nextTokenJSDoc();
-                    }
-                }
-                function skipWhitespaceOrAsterisk() {
-                    if (token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
-                        if (lookAhead(isNextNonwhitespaceTokenEndOfFile)) {
-                            return ""; // Don't skip whitespace prior to EoF (or end of comment) - that shouldn't be included in any node's range
-                        }
-                    }
-                    var precedingLineBreak = scanner.hasPrecedingLineBreak();
-                    var seenLineBreak = false;
-                    var indentText = "";
-                    while ((precedingLineBreak && token() === 41 /* AsteriskToken */) || token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
-                        indentText += scanner.getTokenText();
-                        if (token() === 4 /* NewLineTrivia */) {
-                            precedingLineBreak = true;
-                            seenLineBreak = true;
-                            indentText = "";
+            else {
+                var expression = void 0;
+                switch (token()) {
+                    case 22 /* OpenBracketToken */:
+                        expression = parseArrayLiteralExpression();
+                        break;
+                    case 109 /* TrueKeyword */:
+                    case 94 /* FalseKeyword */:
+                    case 103 /* NullKeyword */:
+                        expression = parseTokenNode();
+                        break;
+                    case 40 /* MinusToken */:
+                        if (lookAhead(function () { return nextToken() === 8 /* NumericLiteral */ && nextToken() !== 58 /* ColonToken */; })) {
+                            expression = parsePrefixUnaryExpression();
                         }
-                        else if (token() === 41 /* AsteriskToken */) {
-                            precedingLineBreak = false;
+                        else {
+                            expression = parseObjectLiteralExpression();
                         }
-                        nextTokenJSDoc();
-                    }
-                    return seenLineBreak ? indentText : "";
-                }
-                function parseTag(margin) {
-                    ts.Debug.assert(token() === 59 /* AtToken */);
-                    var start = scanner.getTokenPos();
-                    nextTokenJSDoc();
-                    var tagName = parseJSDocIdentifierName(/*message*/ undefined);
-                    var indentText = skipWhitespaceOrAsterisk();
-                    var tag;
-                    switch (tagName.escapedText) {
-                        case "author":
-                            tag = parseAuthorTag(start, tagName, margin);
-                            break;
-                        case "implements":
-                            tag = parseImplementsTag(start, tagName);
-                            break;
-                        case "augments":
-                        case "extends":
-                            tag = parseAugmentsTag(start, tagName);
-                            break;
-                        case "class":
-                        case "constructor":
-                            tag = parseSimpleTag(start, 310 /* JSDocClassTag */, tagName);
-                            break;
-                        case "public":
-                            tag = parseSimpleTag(start, 311 /* JSDocPublicTag */, tagName);
-                            break;
-                        case "private":
-                            tag = parseSimpleTag(start, 312 /* JSDocPrivateTag */, tagName);
-                            break;
-                        case "protected":
-                            tag = parseSimpleTag(start, 313 /* JSDocProtectedTag */, tagName);
-                            break;
-                        case "readonly":
-                            tag = parseSimpleTag(start, 314 /* JSDocReadonlyTag */, tagName);
-                            break;
-                        case "this":
-                            tag = parseThisTag(start, tagName);
-                            break;
-                        case "enum":
-                            tag = parseEnumTag(start, tagName);
-                            break;
-                        case "arg":
-                        case "argument":
-                        case "param":
-                            return parseParameterOrPropertyTag(start, tagName, 2 /* Parameter */, margin);
-                        case "return":
-                        case "returns":
-                            tag = parseReturnTag(start, tagName);
-                            break;
-                        case "template":
-                            tag = parseTemplateTag(start, tagName);
-                            break;
-                        case "type":
-                            tag = parseTypeTag(start, tagName);
-                            break;
-                        case "typedef":
-                            tag = parseTypedefTag(start, tagName, margin);
-                            break;
-                        case "callback":
-                            tag = parseCallbackTag(start, tagName, margin);
-                            break;
-                        default:
-                            tag = parseUnknownTag(start, tagName);
+                        break;
+                    case 8 /* NumericLiteral */:
+                    case 10 /* StringLiteral */:
+                        if (lookAhead(function () { return nextToken() !== 58 /* ColonToken */; })) {
+                            expression = parseLiteralNode();
                             break;
-                    }
-                    if (!tag.comment) {
-                        // some tags, like typedef and callback, have already parsed their comments earlier
-                        if (!indentText) {
-                            margin += tag.end - tag.pos;
-                        }
-                        tag.comment = parseTagComments(margin, indentText.slice(margin));
-                    }
-                    return tag;
-                }
-                function parseTagComments(indent, initialMargin) {
-                    var comments = [];
-                    var state = 0 /* BeginningOfLine */;
-                    var margin;
-                    function pushComment(text) {
-                        if (!margin) {
-                            margin = indent;
-                        }
-                        comments.push(text);
-                        indent += text.length;
-                    }
-                    if (initialMargin !== undefined) {
-                        // jump straight to saving comments if there is some initial indentation
-                        if (initialMargin !== "") {
-                            pushComment(initialMargin);
-                        }
-                        state = 1 /* SawAsterisk */;
-                    }
-                    var tok = token();
-                    loop: while (true) {
-                        switch (tok) {
-                            case 4 /* NewLineTrivia */:
-                                if (state >= 1 /* SawAsterisk */) {
-                                    state = 0 /* BeginningOfLine */;
-                                    // don't use pushComment here because we want to keep the margin unchanged
-                                    comments.push(scanner.getTokenText());
-                                }
-                                indent = 0;
-                                break;
-                            case 59 /* AtToken */:
-                                if (state === 3 /* SavingBackticks */) {
-                                    comments.push(scanner.getTokenText());
-                                    break;
-                                }
-                                scanner.setTextPos(scanner.getTextPos() - 1);
-                            // falls through
-                            case 1 /* EndOfFileToken */:
-                                // Done
-                                break loop;
-                            case 5 /* WhitespaceTrivia */:
-                                if (state === 2 /* SavingComments */ || state === 3 /* SavingBackticks */) {
-                                    pushComment(scanner.getTokenText());
-                                }
-                                else {
-                                    var whitespace = scanner.getTokenText();
-                                    // if the whitespace crosses the margin, take only the whitespace that passes the margin
-                                    if (margin !== undefined && indent + whitespace.length > margin) {
-                                        comments.push(whitespace.slice(margin - indent));
-                                    }
-                                    indent += whitespace.length;
-                                }
-                                break;
-                            case 18 /* OpenBraceToken */:
-                                state = 2 /* SavingComments */;
-                                if (lookAhead(function () { return nextTokenJSDoc() === 59 /* AtToken */ && ts.tokenIsIdentifierOrKeyword(nextTokenJSDoc()) && scanner.getTokenText() === "link"; })) {
-                                    pushComment(scanner.getTokenText());
-                                    nextTokenJSDoc();
-                                    pushComment(scanner.getTokenText());
-                                    nextTokenJSDoc();
-                                }
-                                pushComment(scanner.getTokenText());
-                                break;
-                            case 61 /* BacktickToken */:
-                                if (state === 3 /* SavingBackticks */) {
-                                    state = 2 /* SavingComments */;
-                                }
-                                else {
-                                    state = 3 /* SavingBackticks */;
-                                }
-                                pushComment(scanner.getTokenText());
-                                break;
-                            case 41 /* AsteriskToken */:
-                                if (state === 0 /* BeginningOfLine */) {
-                                    // leading asterisks start recording on the *next* (non-whitespace) token
-                                    state = 1 /* SawAsterisk */;
-                                    indent += 1;
-                                    break;
-                                }
-                            // record the * as a comment
-                            // falls through
-                            default:
-                                if (state !== 3 /* SavingBackticks */) {
-                                    state = 2 /* SavingComments */; // leading identifiers start recording as well
-                                }
-                                pushComment(scanner.getTokenText());
-                                break;
-                        }
-                        tok = nextTokenJSDoc();
-                    }
-                    removeLeadingNewlines(comments);
-                    removeTrailingWhitespace(comments);
-                    return comments.length === 0 ? undefined : comments.join("");
-                }
-                function parseUnknownTag(start, tagName) {
-                    var result = createNode(306 /* JSDocTag */, start);
-                    result.tagName = tagName;
-                    return finishNode(result);
-                }
-                function addTag(tag) {
-                    if (!tag) {
-                        return;
-                    }
-                    if (!tags) {
-                        tags = [tag];
-                        tagsPos = tag.pos;
-                    }
-                    else {
-                        tags.push(tag);
-                    }
-                    tagsEnd = tag.end;
-                }
-                function tryParseTypeExpression() {
-                    skipWhitespaceOrAsterisk();
-                    return token() === 18 /* OpenBraceToken */ ? parseJSDocTypeExpression() : undefined;
-                }
-                function parseBracketNameInPropertyAndParamTag() {
-                    // Looking for something like '[foo]', 'foo', '[foo.bar]' or 'foo.bar'
-                    var isBracketed = parseOptionalJsdoc(22 /* OpenBracketToken */);
-                    if (isBracketed) {
-                        skipWhitespace();
-                    }
-                    // a markdown-quoted name: `arg` is not legal jsdoc, but occurs in the wild
-                    var isBackquoted = parseOptionalJsdoc(61 /* BacktickToken */);
-                    var name = parseJSDocEntityName();
-                    if (isBackquoted) {
-                        parseExpectedTokenJSDoc(61 /* BacktickToken */);
-                    }
-                    if (isBracketed) {
-                        skipWhitespace();
-                        // May have an optional default, e.g. '[foo = 42]'
-                        if (parseOptionalToken(62 /* EqualsToken */)) {
-                            parseExpression();
-                        }
-                        parseExpected(23 /* CloseBracketToken */);
-                    }
-                    return { name: name, isBracketed: isBracketed };
-                }
-                function isObjectOrObjectArrayTypeReference(node) {
-                    switch (node.kind) {
-                        case 141 /* ObjectKeyword */:
-                            return true;
-                        case 174 /* ArrayType */:
-                            return isObjectOrObjectArrayTypeReference(node.elementType);
-                        default:
-                            return ts.isTypeReferenceNode(node) && ts.isIdentifier(node.typeName) && node.typeName.escapedText === "Object" && !node.typeArguments;
-                    }
-                }
-                function parseParameterOrPropertyTag(start, tagName, target, indent) {
-                    var typeExpression = tryParseTypeExpression();
-                    var isNameFirst = !typeExpression;
-                    skipWhitespaceOrAsterisk();
-                    var _a = parseBracketNameInPropertyAndParamTag(), name = _a.name, isBracketed = _a.isBracketed;
-                    skipWhitespace();
-                    if (isNameFirst) {
-                        typeExpression = tryParseTypeExpression();
-                    }
-                    var result = target === 1 /* Property */ ?
-                        createNode(323 /* JSDocPropertyTag */, start) :
-                        createNode(317 /* JSDocParameterTag */, start);
-                    var comment = parseTagComments(indent + scanner.getStartPos() - start);
-                    var nestedTypeLiteral = target !== 4 /* CallbackParameter */ && parseNestedTypeLiteral(typeExpression, name, target, indent);
-                    if (nestedTypeLiteral) {
-                        typeExpression = nestedTypeLiteral;
-                        isNameFirst = true;
-                    }
-                    result.tagName = tagName;
-                    result.typeExpression = typeExpression;
-                    result.name = name;
-                    result.isNameFirst = isNameFirst;
-                    result.isBracketed = isBracketed;
-                    result.comment = comment;
-                    return finishNode(result);
-                }
-                function parseNestedTypeLiteral(typeExpression, name, target, indent) {
-                    if (typeExpression && isObjectOrObjectArrayTypeReference(typeExpression.type)) {
-                        var typeLiteralExpression = createNode(294 /* JSDocTypeExpression */, scanner.getTokenPos());
-                        var child = void 0;
-                        var jsdocTypeLiteral = void 0;
-                        var start_3 = scanner.getStartPos();
-                        var children = void 0;
-                        while (child = tryParse(function () { return parseChildParameterOrPropertyTag(target, indent, name); })) {
-                            if (child.kind === 317 /* JSDocParameterTag */ || child.kind === 323 /* JSDocPropertyTag */) {
-                                children = ts.append(children, child);
-                            }
-                        }
-                        if (children) {
-                            jsdocTypeLiteral = createNode(304 /* JSDocTypeLiteral */, start_3);
-                            jsdocTypeLiteral.jsDocPropertyTags = children;
-                            if (typeExpression.type.kind === 174 /* ArrayType */) {
-                                jsdocTypeLiteral.isArrayType = true;
-                            }
-                            typeLiteralExpression.type = finishNode(jsdocTypeLiteral);
-                            return finishNode(typeLiteralExpression);
                         }
-                    }
-                }
-                function parseReturnTag(start, tagName) {
-                    if (ts.some(tags, ts.isJSDocReturnTag)) {
-                        parseErrorAt(tagName.pos, scanner.getTokenPos(), ts.Diagnostics._0_tag_already_specified, tagName.escapedText);
-                    }
-                    var result = createNode(318 /* JSDocReturnTag */, start);
-                    result.tagName = tagName;
-                    result.typeExpression = tryParseTypeExpression();
-                    return finishNode(result);
-                }
-                function parseTypeTag(start, tagName) {
-                    if (ts.some(tags, ts.isJSDocTypeTag)) {
-                        parseErrorAt(tagName.pos, scanner.getTokenPos(), ts.Diagnostics._0_tag_already_specified, tagName.escapedText);
-                    }
-                    var result = createNode(320 /* JSDocTypeTag */, start);
-                    result.tagName = tagName;
-                    result.typeExpression = parseJSDocTypeExpression(/*mayOmitBraces*/ true);
-                    return finishNode(result);
+                    // falls through
+                    default:
+                        expression = parseObjectLiteralExpression();
+                        break;
                 }
-                function parseAuthorTag(start, tagName, indent) {
-                    var result = createNode(309 /* JSDocAuthorTag */, start);
-                    result.tagName = tagName;
-                    var authorInfoWithEmail = tryParse(function () { return tryParseAuthorNameAndEmail(); });
-                    if (!authorInfoWithEmail) {
-                        return finishNode(result);
-                    }
-                    result.comment = authorInfoWithEmail;
-                    if (lookAhead(function () { return nextToken() !== 4 /* NewLineTrivia */; })) {
-                        var comment = parseTagComments(indent);
-                        if (comment) {
-                            result.comment += comment;
+                var statement = factory.createExpressionStatement(expression);
+                finishNode(statement, pos);
+                statements = createNodeArray([statement], pos);
+                endOfFileToken = parseExpectedToken(1 /* EndOfFileToken */, ts.Diagnostics.Unexpected_token);
+            }
+            // Set source file so that errors will be reported with this file name
+            var sourceFile = createSourceFile(fileName, 2 /* ES2015 */, 6 /* JSON */, /*isDeclaration*/ false, statements, endOfFileToken, sourceFlags);
+            if (setParentNodes) {
+                fixupParentReferences(sourceFile);
+            }
+            sourceFile.nodeCount = nodeCount;
+            sourceFile.identifierCount = identifierCount;
+            sourceFile.identifiers = identifiers;
+            sourceFile.parseDiagnostics = ts.attachFileToDiagnostics(parseDiagnostics, sourceFile);
+            if (jsDocDiagnostics) {
+                sourceFile.jsDocDiagnostics = ts.attachFileToDiagnostics(jsDocDiagnostics, sourceFile);
+            }
+            var result = sourceFile;
+            clearState();
+            return result;
+        }
+        Parser.parseJsonText = parseJsonText;
+        function initializeState(_fileName, _sourceText, _languageVersion, _syntaxCursor, _scriptKind) {
+            NodeConstructor = ts.objectAllocator.getNodeConstructor();
+            TokenConstructor = ts.objectAllocator.getTokenConstructor();
+            IdentifierConstructor = ts.objectAllocator.getIdentifierConstructor();
+            PrivateIdentifierConstructor = ts.objectAllocator.getPrivateIdentifierConstructor();
+            SourceFileConstructor = ts.objectAllocator.getSourceFileConstructor();
+            fileName = ts.normalizePath(_fileName);
+            sourceText = _sourceText;
+            languageVersion = _languageVersion;
+            syntaxCursor = _syntaxCursor;
+            scriptKind = _scriptKind;
+            languageVariant = ts.getLanguageVariant(_scriptKind);
+            parseDiagnostics = [];
+            parsingContext = 0;
+            identifiers = new ts.Map();
+            privateIdentifiers = new ts.Map();
+            identifierCount = 0;
+            nodeCount = 0;
+            sourceFlags = 0;
+            topLevel = true;
+            switch (scriptKind) {
+                case 1 /* JS */:
+                case 2 /* JSX */:
+                    contextFlags = 131072 /* JavaScriptFile */;
+                    break;
+                case 6 /* JSON */:
+                    contextFlags = 131072 /* JavaScriptFile */ | 33554432 /* JsonFile */;
+                    break;
+                default:
+                    contextFlags = 0 /* None */;
+                    break;
+            }
+            parseErrorBeforeNextFinishedNode = false;
+            // Initialize and prime the scanner before parsing the source elements.
+            scanner.setText(sourceText);
+            scanner.setOnError(scanError);
+            scanner.setScriptTarget(languageVersion);
+            scanner.setLanguageVariant(languageVariant);
+        }
+        function clearState() {
+            // Clear out the text the scanner is pointing at, so it doesn't keep anything alive unnecessarily.
+            scanner.clearCommentDirectives();
+            scanner.setText("");
+            scanner.setOnError(undefined);
+            // Clear any data.  We don't want to accidentally hold onto it for too long.
+            sourceText = undefined;
+            languageVersion = undefined;
+            syntaxCursor = undefined;
+            scriptKind = undefined;
+            languageVariant = undefined;
+            sourceFlags = 0;
+            parseDiagnostics = undefined;
+            jsDocDiagnostics = undefined;
+            parsingContext = 0;
+            identifiers = undefined;
+            notParenthesizedArrow = undefined;
+            topLevel = true;
+        }
+        function parseSourceFileWorker(languageVersion, setParentNodes, scriptKind) {
+            var isDeclarationFile = isDeclarationFileName(fileName);
+            if (isDeclarationFile) {
+                contextFlags |= 8388608 /* Ambient */;
+            }
+            sourceFlags = contextFlags;
+            // Prime the scanner.
+            nextToken();
+            var statements = parseList(0 /* SourceElements */, parseStatement);
+            ts.Debug.assert(token() === 1 /* EndOfFileToken */);
+            var endOfFileToken = addJSDocComment(parseTokenNode());
+            var sourceFile = createSourceFile(fileName, languageVersion, scriptKind, isDeclarationFile, statements, endOfFileToken, sourceFlags);
+            // A member of ReadonlyArray<T> isn't assignable to a member of T[] (and prevents a direct cast) - but this is where we set up those members so they can be readonly in the future
+            processCommentPragmas(sourceFile, sourceText);
+            processPragmasIntoFields(sourceFile, reportPragmaDiagnostic);
+            sourceFile.commentDirectives = scanner.getCommentDirectives();
+            sourceFile.nodeCount = nodeCount;
+            sourceFile.identifierCount = identifierCount;
+            sourceFile.identifiers = identifiers;
+            sourceFile.parseDiagnostics = ts.attachFileToDiagnostics(parseDiagnostics, sourceFile);
+            if (jsDocDiagnostics) {
+                sourceFile.jsDocDiagnostics = ts.attachFileToDiagnostics(jsDocDiagnostics, sourceFile);
+            }
+            if (setParentNodes) {
+                fixupParentReferences(sourceFile);
+            }
+            return sourceFile;
+            function reportPragmaDiagnostic(pos, end, diagnostic) {
+                parseDiagnostics.push(ts.createDetachedDiagnostic(fileName, pos, end, diagnostic));
+            }
+        }
+        function withJSDoc(node, hasJSDoc) {
+            return hasJSDoc ? addJSDocComment(node) : node;
+        }
+        var hasDeprecatedTag = false;
+        function addJSDocComment(node) {
+            ts.Debug.assert(!node.jsDoc); // Should only be called once per node
+            var jsDoc = ts.mapDefined(ts.getJSDocCommentRanges(node, sourceText), function (comment) { return JSDocParser.parseJSDocComment(node, comment.pos, comment.end - comment.pos); });
+            if (jsDoc.length)
+                node.jsDoc = jsDoc;
+            if (hasDeprecatedTag) {
+                hasDeprecatedTag = false;
+                node.flags |= 134217728 /* Deprecated */;
+            }
+            return node;
+        }
+        function reparseTopLevelAwait(sourceFile) {
+            var savedSyntaxCursor = syntaxCursor;
+            var baseSyntaxCursor = IncrementalParser.createSyntaxCursor(sourceFile);
+            syntaxCursor = { currentNode: currentNode };
+            var statements = [];
+            var savedParseDiagnostics = parseDiagnostics;
+            parseDiagnostics = [];
+            var pos = 0;
+            var start = findNextStatementWithAwait(sourceFile.statements, 0);
+            var _loop_3 = function () {
+                // append all statements between pos and start
+                var prevStatement = sourceFile.statements[pos];
+                var nextStatement = sourceFile.statements[start];
+                ts.addRange(statements, sourceFile.statements, pos, start);
+                pos = findNextStatementWithoutAwait(sourceFile.statements, start);
+                // append all diagnostics associated with the copied range
+                var diagnosticStart = ts.findIndex(savedParseDiagnostics, function (diagnostic) { return diagnostic.start >= prevStatement.pos; });
+                var diagnosticEnd = diagnosticStart >= 0 ? ts.findIndex(savedParseDiagnostics, function (diagnostic) { return diagnostic.start >= nextStatement.pos; }, diagnosticStart) : -1;
+                if (diagnosticStart >= 0) {
+                    ts.addRange(parseDiagnostics, savedParseDiagnostics, diagnosticStart, diagnosticEnd >= 0 ? diagnosticEnd : undefined);
+                }
+                // reparse all statements between start and pos. We skip existing diagnostics for the same range and allow the parser to generate new ones.
+                speculationHelper(function () {
+                    var savedContextFlags = contextFlags;
+                    contextFlags |= 32768 /* AwaitContext */;
+                    scanner.setTextPos(nextStatement.pos);
+                    nextToken();
+                    while (token() !== 1 /* EndOfFileToken */) {
+                        var startPos = scanner.getStartPos();
+                        var statement = parseListElement(0 /* SourceElements */, parseStatement);
+                        statements.push(statement);
+                        if (startPos === scanner.getStartPos()) {
+                            nextToken();
                         }
-                    }
-                    return finishNode(result);
-                }
-                function tryParseAuthorNameAndEmail() {
-                    var comments = [];
-                    var seenLessThan = false;
-                    var seenGreaterThan = false;
-                    var token = scanner.getToken();
-                    loop: while (true) {
-                        switch (token) {
-                            case 75 /* Identifier */:
-                            case 5 /* WhitespaceTrivia */:
-                            case 24 /* DotToken */:
-                            case 59 /* AtToken */:
-                                comments.push(scanner.getTokenText());
-                                break;
-                            case 29 /* LessThanToken */:
-                                if (seenLessThan || seenGreaterThan) {
-                                    return;
-                                }
-                                seenLessThan = true;
-                                comments.push(scanner.getTokenText());
+                        if (pos >= 0) {
+                            var nonAwaitStatement = sourceFile.statements[pos];
+                            if (statement.end === nonAwaitStatement.pos) {
+                                // done reparsing this section
                                 break;
-                            case 31 /* GreaterThanToken */:
-                                if (!seenLessThan || seenGreaterThan) {
-                                    return;
-                                }
-                                seenGreaterThan = true;
-                                comments.push(scanner.getTokenText());
-                                scanner.setTextPos(scanner.getTokenPos() + 1);
-                                break loop;
-                            case 4 /* NewLineTrivia */:
-                            case 1 /* EndOfFileToken */:
-                                break loop;
-                        }
-                        token = nextTokenJSDoc();
-                    }
-                    if (seenLessThan && seenGreaterThan) {
-                        return comments.length === 0 ? undefined : comments.join("");
-                    }
-                }
-                function parseImplementsTag(start, tagName) {
-                    var result = createNode(308 /* JSDocImplementsTag */, start);
-                    result.tagName = tagName;
-                    result.class = parseExpressionWithTypeArgumentsForAugments();
-                    return finishNode(result);
-                }
-                function parseAugmentsTag(start, tagName) {
-                    var result = createNode(307 /* JSDocAugmentsTag */, start);
-                    result.tagName = tagName;
-                    result.class = parseExpressionWithTypeArgumentsForAugments();
-                    return finishNode(result);
-                }
-                function parseExpressionWithTypeArgumentsForAugments() {
-                    var usedBrace = parseOptional(18 /* OpenBraceToken */);
-                    var node = createNode(216 /* ExpressionWithTypeArguments */);
-                    node.expression = parsePropertyAccessEntityNameExpression();
-                    node.typeArguments = tryParseTypeArguments();
-                    var res = finishNode(node);
-                    if (usedBrace) {
-                        parseExpected(19 /* CloseBraceToken */);
-                    }
-                    return res;
-                }
-                function parsePropertyAccessEntityNameExpression() {
-                    var node = parseJSDocIdentifierName();
-                    while (parseOptional(24 /* DotToken */)) {
-                        var prop = createNode(194 /* PropertyAccessExpression */, node.pos);
-                        prop.expression = node;
-                        prop.name = parseJSDocIdentifierName();
-                        node = finishNode(prop);
-                    }
-                    return node;
-                }
-                function parseSimpleTag(start, kind, tagName) {
-                    var tag = createNode(kind, start);
-                    tag.tagName = tagName;
-                    return finishNode(tag);
-                }
-                function parseThisTag(start, tagName) {
-                    var tag = createNode(319 /* JSDocThisTag */, start);
-                    tag.tagName = tagName;
-                    tag.typeExpression = parseJSDocTypeExpression(/*mayOmitBraces*/ true);
-                    skipWhitespace();
-                    return finishNode(tag);
-                }
-                function parseEnumTag(start, tagName) {
-                    var tag = createNode(316 /* JSDocEnumTag */, start);
-                    tag.tagName = tagName;
-                    tag.typeExpression = parseJSDocTypeExpression(/*mayOmitBraces*/ true);
-                    skipWhitespace();
-                    return finishNode(tag);
-                }
-                function parseTypedefTag(start, tagName, indent) {
-                    var typeExpression = tryParseTypeExpression();
-                    skipWhitespaceOrAsterisk();
-                    var typedefTag = createNode(322 /* JSDocTypedefTag */, start);
-                    typedefTag.tagName = tagName;
-                    typedefTag.fullName = parseJSDocTypeNameWithNamespace();
-                    typedefTag.name = getJSDocTypeAliasName(typedefTag.fullName);
-                    skipWhitespace();
-                    typedefTag.comment = parseTagComments(indent);
-                    typedefTag.typeExpression = typeExpression;
-                    var end;
-                    if (!typeExpression || isObjectOrObjectArrayTypeReference(typeExpression.type)) {
-                        var child = void 0;
-                        var jsdocTypeLiteral = void 0;
-                        var childTypeTag = void 0;
-                        while (child = tryParse(function () { return parseChildPropertyTag(indent); })) {
-                            if (!jsdocTypeLiteral) {
-                                jsdocTypeLiteral = createNode(304 /* JSDocTypeLiteral */, start);
-                            }
-                            if (child.kind === 320 /* JSDocTypeTag */) {
-                                if (childTypeTag) {
-                                    break;
-                                }
-                                else {
-                                    childTypeTag = child;
-                                }
-                            }
-                            else {
-                                jsdocTypeLiteral.jsDocPropertyTags = ts.append(jsdocTypeLiteral.jsDocPropertyTags, child);
-                            }
-                        }
-                        if (jsdocTypeLiteral) {
-                            if (typeExpression && typeExpression.type.kind === 174 /* ArrayType */) {
-                                jsdocTypeLiteral.isArrayType = true;
-                            }
-                            typedefTag.typeExpression = childTypeTag && childTypeTag.typeExpression && !isObjectOrObjectArrayTypeReference(childTypeTag.typeExpression.type) ?
-                                childTypeTag.typeExpression :
-                                finishNode(jsdocTypeLiteral);
-                            end = typedefTag.typeExpression.end;
-                        }
-                    }
-                    // Only include the characters between the name end and the next token if a comment was actually parsed out - otherwise it's just whitespace
-                    return finishNode(typedefTag, end || typedefTag.comment !== undefined ? scanner.getStartPos() : (typedefTag.fullName || typedefTag.typeExpression || typedefTag.tagName).end);
-                }
-                function parseJSDocTypeNameWithNamespace(nested) {
-                    var pos = scanner.getTokenPos();
-                    if (!ts.tokenIsIdentifierOrKeyword(token())) {
-                        return undefined;
-                    }
-                    var typeNameOrNamespaceName = parseJSDocIdentifierName();
-                    if (parseOptional(24 /* DotToken */)) {
-                        var jsDocNamespaceNode = createNode(249 /* ModuleDeclaration */, pos);
-                        if (nested) {
-                            jsDocNamespaceNode.flags |= 4 /* NestedNamespace */;
-                        }
-                        jsDocNamespaceNode.name = typeNameOrNamespaceName;
-                        jsDocNamespaceNode.body = parseJSDocTypeNameWithNamespace(/*nested*/ true);
-                        return finishNode(jsDocNamespaceNode);
-                    }
-                    if (nested) {
-                        typeNameOrNamespaceName.isInJSDocNamespace = true;
-                    }
-                    return typeNameOrNamespaceName;
-                }
-                function parseCallbackTag(start, tagName, indent) {
-                    var callbackTag = createNode(315 /* JSDocCallbackTag */, start);
-                    callbackTag.tagName = tagName;
-                    callbackTag.fullName = parseJSDocTypeNameWithNamespace();
-                    callbackTag.name = getJSDocTypeAliasName(callbackTag.fullName);
-                    skipWhitespace();
-                    callbackTag.comment = parseTagComments(indent);
-                    var child;
-                    var jsdocSignature = createNode(305 /* JSDocSignature */, start);
-                    jsdocSignature.parameters = [];
-                    while (child = tryParse(function () { return parseChildParameterOrPropertyTag(4 /* CallbackParameter */, indent); })) {
-                        jsdocSignature.parameters = ts.append(jsdocSignature.parameters, child);
-                    }
-                    var returnTag = tryParse(function () {
-                        if (parseOptionalJsdoc(59 /* AtToken */)) {
-                            var tag = parseTag(indent);
-                            if (tag && tag.kind === 318 /* JSDocReturnTag */) {
-                                return tag;
                             }
-                        }
-                    });
-                    if (returnTag) {
-                        jsdocSignature.type = returnTag;
-                    }
-                    callbackTag.typeExpression = finishNode(jsdocSignature);
-                    return finishNode(callbackTag);
-                }
-                function getJSDocTypeAliasName(fullName) {
-                    if (fullName) {
-                        var rightNode = fullName;
-                        while (true) {
-                            if (ts.isIdentifier(rightNode) || !rightNode.body) {
-                                return ts.isIdentifier(rightNode) ? rightNode : rightNode.name;
+                            if (statement.end > nonAwaitStatement.pos) {
+                                // we ate into the next statement, so we must reparse it.
+                                pos = findNextStatementWithoutAwait(sourceFile.statements, pos + 1);
                             }
-                            rightNode = rightNode.body;
-                        }
-                    }
-                }
-                function escapedTextsEqual(a, b) {
-                    while (!ts.isIdentifier(a) || !ts.isIdentifier(b)) {
-                        if (!ts.isIdentifier(a) && !ts.isIdentifier(b) && a.right.escapedText === b.right.escapedText) {
-                            a = a.left;
-                            b = b.left;
-                        }
-                        else {
-                            return false;
-                        }
-                    }
-                    return a.escapedText === b.escapedText;
-                }
-                function parseChildPropertyTag(indent) {
-                    return parseChildParameterOrPropertyTag(1 /* Property */, indent);
-                }
-                function parseChildParameterOrPropertyTag(target, indent, name) {
-                    var canParseTag = true;
-                    var seenAsterisk = false;
-                    while (true) {
-                        switch (nextTokenJSDoc()) {
-                            case 59 /* AtToken */:
-                                if (canParseTag) {
-                                    var child = tryParseChildTag(target, indent);
-                                    if (child && (child.kind === 317 /* JSDocParameterTag */ || child.kind === 323 /* JSDocPropertyTag */) &&
-                                        target !== 4 /* CallbackParameter */ &&
-                                        name && (ts.isIdentifier(child.name) || !escapedTextsEqual(name, child.name.left))) {
-                                        return false;
-                                    }
-                                    return child;
-                                }
-                                seenAsterisk = false;
-                                break;
-                            case 4 /* NewLineTrivia */:
-                                canParseTag = true;
-                                seenAsterisk = false;
-                                break;
-                            case 41 /* AsteriskToken */:
-                                if (seenAsterisk) {
-                                    canParseTag = false;
-                                }
-                                seenAsterisk = true;
-                                break;
-                            case 75 /* Identifier */:
-                                canParseTag = false;
-                                break;
-                            case 1 /* EndOfFileToken */:
-                                return false;
                         }
                     }
+                    contextFlags = savedContextFlags;
+                }, 2 /* Reparse */);
+                // find the next statement containing an `await`
+                start = pos >= 0 ? findNextStatementWithAwait(sourceFile.statements, pos) : -1;
+            };
+            while (start !== -1) {
+                _loop_3();
+            }
+            // append all statements between pos and the end of the list
+            if (pos >= 0) {
+                var prevStatement_1 = sourceFile.statements[pos];
+                ts.addRange(statements, sourceFile.statements, pos);
+                // append all diagnostics associated with the copied range
+                var diagnosticStart = ts.findIndex(savedParseDiagnostics, function (diagnostic) { return diagnostic.start >= prevStatement_1.pos; });
+                if (diagnosticStart >= 0) {
+                    ts.addRange(parseDiagnostics, savedParseDiagnostics, diagnosticStart);
                 }
-                function tryParseChildTag(target, indent) {
-                    ts.Debug.assert(token() === 59 /* AtToken */);
-                    var start = scanner.getStartPos();
-                    nextTokenJSDoc();
-                    var tagName = parseJSDocIdentifierName();
-                    skipWhitespace();
-                    var t;
-                    switch (tagName.escapedText) {
-                        case "type":
-                            return target === 1 /* Property */ && parseTypeTag(start, tagName);
-                        case "prop":
-                        case "property":
-                            t = 1 /* Property */;
-                            break;
-                        case "arg":
-                        case "argument":
-                        case "param":
-                            t = 2 /* Parameter */ | 4 /* CallbackParameter */;
-                            break;
-                        default:
-                            return false;
-                    }
-                    if (!(target & t)) {
-                        return false;
-                    }
-                    return parseParameterOrPropertyTag(start, tagName, target, indent);
-                }
-                function parseTemplateTag(start, tagName) {
-                    // the template tag looks like '@template {Constraint} T,U,V'
-                    var constraint;
-                    if (token() === 18 /* OpenBraceToken */) {
-                        constraint = parseJSDocTypeExpression();
-                    }
-                    var typeParameters = [];
-                    var typeParametersPos = getNodePos();
-                    do {
-                        skipWhitespace();
-                        var typeParameter = createNode(155 /* TypeParameter */);
-                        typeParameter.name = parseJSDocIdentifierName(ts.Diagnostics.Unexpected_token_A_type_parameter_name_was_expected_without_curly_braces);
-                        finishNode(typeParameter);
-                        skipWhitespaceOrAsterisk();
-                        typeParameters.push(typeParameter);
-                    } while (parseOptionalJsdoc(27 /* CommaToken */));
-                    var result = createNode(321 /* JSDocTemplateTag */, start);
-                    result.tagName = tagName;
-                    result.constraint = constraint;
-                    result.typeParameters = createNodeArray(typeParameters, typeParametersPos);
-                    finishNode(result);
-                    return result;
-                }
-                function parseOptionalJsdoc(t) {
-                    if (token() === t) {
-                        nextTokenJSDoc();
-                        return true;
-                    }
-                    return false;
-                }
-                function parseJSDocEntityName() {
-                    var entity = parseJSDocIdentifierName();
-                    if (parseOptional(22 /* OpenBracketToken */)) {
-                        parseExpected(23 /* CloseBracketToken */);
-                        // Note that y[] is accepted as an entity name, but the postfix brackets are not saved for checking.
-                        // Technically usejsdoc.org requires them for specifying a property of a type equivalent to Array<{ x: ...}>
-                        // but it's not worth it to enforce that restriction.
-                    }
-                    while (parseOptional(24 /* DotToken */)) {
-                        var name = parseJSDocIdentifierName();
-                        if (parseOptional(22 /* OpenBracketToken */)) {
-                            parseExpected(23 /* CloseBracketToken */);
-                        }
-                        entity = createQualifiedName(entity, name);
+            }
+            syntaxCursor = savedSyntaxCursor;
+            return factory.updateSourceFile(sourceFile, ts.setTextRange(factory.createNodeArray(statements), sourceFile.statements));
+            function containsPossibleTopLevelAwait(node) {
+                return !(node.flags & 32768 /* AwaitContext */)
+                    && !!(node.transformFlags & 8388608 /* ContainsPossibleTopLevelAwait */);
+            }
+            function findNextStatementWithAwait(statements, start) {
+                for (var i = start; i < statements.length; i++) {
+                    if (containsPossibleTopLevelAwait(statements[i])) {
+                        return i;
                     }
-                    return entity;
                 }
-                function parseJSDocIdentifierName(message) {
-                    if (!ts.tokenIsIdentifierOrKeyword(token())) {
-                        return createMissingNode(75 /* Identifier */, /*reportAtCurrentPosition*/ !message, message || ts.Diagnostics.Identifier_expected);
-                    }
-                    identifierCount++;
-                    var pos = scanner.getTokenPos();
-                    var end = scanner.getTextPos();
-                    var result = createNode(75 /* Identifier */, pos);
-                    if (token() !== 75 /* Identifier */) {
-                        result.originalKeywordKind = token();
+                return -1;
+            }
+            function findNextStatementWithoutAwait(statements, start) {
+                for (var i = start; i < statements.length; i++) {
+                    if (!containsPossibleTopLevelAwait(statements[i])) {
+                        return i;
                     }
-                    result.escapedText = ts.escapeLeadingUnderscores(internIdentifier(scanner.getTokenValue()));
-                    finishNode(result, end);
-                    nextTokenJSDoc();
-                    return result;
                 }
+                return -1;
             }
-        })(JSDocParser = Parser.JSDocParser || (Parser.JSDocParser = {}));
-    })(Parser || (Parser = {}));
-    var IncrementalParser;
-    (function (IncrementalParser) {
-        function updateSourceFile(sourceFile, newText, textChangeRange, aggressiveChecks) {
-            aggressiveChecks = aggressiveChecks || ts.Debug.shouldAssert(2 /* Aggressive */);
-            checkChangeRange(sourceFile, newText, textChangeRange, aggressiveChecks);
-            if (ts.textChangeRangeIsUnchanged(textChangeRange)) {
-                // if the text didn't change, then we can just return our current source file as-is.
-                return sourceFile;
+            function currentNode(position) {
+                var node = baseSyntaxCursor.currentNode(position);
+                if (topLevel && node && containsPossibleTopLevelAwait(node)) {
+                    node.intersectsChange = true;
+                }
+                return node;
             }
-            if (sourceFile.statements.length === 0) {
-                // If we don't have any statements in the current source file, then there's no real
-                // way to incrementally parse.  So just do a full parse instead.
-                return Parser.parseSourceFile(sourceFile.fileName, newText, sourceFile.languageVersion, /*syntaxCursor*/ undefined, /*setParentNodes*/ true, sourceFile.scriptKind);
+        }
+        function fixupParentReferences(rootNode) {
+            // normally parent references are set during binding. However, for clients that only need
+            // a syntax tree, and no semantic features, then the binding process is an unnecessary
+            // overhead.  This functions allows us to set all the parents, without all the expense of
+            // binding.
+            ts.setParentRecursive(rootNode, /*incremental*/ true);
+        }
+        Parser.fixupParentReferences = fixupParentReferences;
+        function createSourceFile(fileName, languageVersion, scriptKind, isDeclarationFile, statements, endOfFileToken, flags) {
+            // code from createNode is inlined here so createNode won't have to deal with special case of creating source files
+            // this is quite rare comparing to other nodes and createNode should be as fast as possible
+            var sourceFile = factory.createSourceFile(statements, endOfFileToken, flags);
+            ts.setTextRangePosWidth(sourceFile, 0, sourceText.length);
+            setExternalModuleIndicator(sourceFile);
+            // If we parsed this as an external module, it may contain top-level await
+            if (!isDeclarationFile && isExternalModule(sourceFile) && sourceFile.transformFlags & 8388608 /* ContainsPossibleTopLevelAwait */) {
+                sourceFile = reparseTopLevelAwait(sourceFile);
             }
-            // Make sure we're not trying to incrementally update a source file more than once.  Once
-            // we do an update the original source file is considered unusable from that point onwards.
-            //
-            // This is because we do incremental parsing in-place.  i.e. we take nodes from the old
-            // tree and give them new positions and parents.  From that point on, trusting the old
-            // tree at all is not possible as far too much of it may violate invariants.
-            var incrementalSourceFile = sourceFile;
-            ts.Debug.assert(!incrementalSourceFile.hasBeenIncrementallyParsed);
-            incrementalSourceFile.hasBeenIncrementallyParsed = true;
-            var oldText = sourceFile.text;
-            var syntaxCursor = createSyntaxCursor(sourceFile);
-            // Make the actual change larger so that we know to reparse anything whose lookahead
-            // might have intersected the change.
-            var changeRange = extendToAffectedRange(sourceFile, textChangeRange);
-            checkChangeRange(sourceFile, newText, changeRange, aggressiveChecks);
-            // Ensure that extending the affected range only moved the start of the change range
-            // earlier in the file.
-            ts.Debug.assert(changeRange.span.start <= textChangeRange.span.start);
-            ts.Debug.assert(ts.textSpanEnd(changeRange.span) === ts.textSpanEnd(textChangeRange.span));
-            ts.Debug.assert(ts.textSpanEnd(ts.textChangeRangeNewSpan(changeRange)) === ts.textSpanEnd(ts.textChangeRangeNewSpan(textChangeRange)));
-            // The is the amount the nodes after the edit range need to be adjusted.  It can be
-            // positive (if the edit added characters), negative (if the edit deleted characters)
-            // or zero (if this was a pure overwrite with nothing added/removed).
-            var delta = ts.textChangeRangeNewSpan(changeRange).length - changeRange.span.length;
-            // If we added or removed characters during the edit, then we need to go and adjust all
-            // the nodes after the edit.  Those nodes may move forward (if we inserted chars) or they
-            // may move backward (if we deleted chars).
-            //
-            // Doing this helps us out in two ways.  First, it means that any nodes/tokens we want
-            // to reuse are already at the appropriate position in the new text.  That way when we
-            // reuse them, we don't have to figure out if they need to be adjusted.  Second, it makes
-            // it very easy to determine if we can reuse a node.  If the node's position is at where
-            // we are in the text, then we can reuse it.  Otherwise we can't.  If the node's position
-            // is ahead of us, then we'll need to rescan tokens.  If the node's position is behind
-            // us, then we'll need to skip it or crumble it as appropriate
-            //
-            // We will also adjust the positions of nodes that intersect the change range as well.
-            // By doing this, we ensure that all the positions in the old tree are consistent, not
-            // just the positions of nodes entirely before/after the change range.  By being
-            // consistent, we can then easily map from positions to nodes in the old tree easily.
-            //
-            // Also, mark any syntax elements that intersect the changed span.  We know, up front,
-            // that we cannot reuse these elements.
-            updateTokenPositionsAndMarkElements(incrementalSourceFile, changeRange.span.start, ts.textSpanEnd(changeRange.span), ts.textSpanEnd(ts.textChangeRangeNewSpan(changeRange)), delta, oldText, newText, aggressiveChecks);
-            // Now that we've set up our internal incremental state just proceed and parse the
-            // source file in the normal fashion.  When possible the parser will retrieve and
-            // reuse nodes from the old tree.
-            //
-            // Note: passing in 'true' for setNodeParents is very important.  When incrementally
-            // parsing, we will be reusing nodes from the old tree, and placing it into new
-            // parents.  If we don't set the parents now, we'll end up with an observably
-            // inconsistent tree.  Setting the parents on the new tree should be very fast.  We
-            // will immediately bail out of walking any subtrees when we can see that their parents
-            // are already correct.
-            var result = Parser.parseSourceFile(sourceFile.fileName, newText, sourceFile.languageVersion, syntaxCursor, /*setParentNodes*/ true, sourceFile.scriptKind);
-            result.commentDirectives = getNewCommentDirectives(sourceFile.commentDirectives, result.commentDirectives, changeRange.span.start, ts.textSpanEnd(changeRange.span), delta, oldText, newText, aggressiveChecks);
-            return result;
+            sourceFile.text = sourceText;
+            sourceFile.bindDiagnostics = [];
+            sourceFile.bindSuggestionDiagnostics = undefined;
+            sourceFile.languageVersion = languageVersion;
+            sourceFile.fileName = fileName;
+            sourceFile.languageVariant = ts.getLanguageVariant(scriptKind);
+            sourceFile.isDeclarationFile = isDeclarationFile;
+            sourceFile.scriptKind = scriptKind;
+            return sourceFile;
         }
-        IncrementalParser.updateSourceFile = updateSourceFile;
-        function getNewCommentDirectives(oldDirectives, newDirectives, changeStart, changeRangeOldEnd, delta, oldText, newText, aggressiveChecks) {
-            if (!oldDirectives)
-                return newDirectives;
-            var commentDirectives;
-            var addedNewlyScannedDirectives = false;
-            for (var _i = 0, oldDirectives_1 = oldDirectives; _i < oldDirectives_1.length; _i++) {
-                var directive = oldDirectives_1[_i];
-                var range = directive.range, type = directive.type;
-                // Range before the change
-                if (range.end < changeStart) {
-                    commentDirectives = ts.append(commentDirectives, directive);
-                }
-                else if (range.pos > changeRangeOldEnd) {
-                    addNewlyScannedDirectives();
-                    // Node is entirely past the change range.  We need to move both its pos and
-                    // end, forward or backward appropriately.
-                    var updatedDirective = {
-                        range: { pos: range.pos + delta, end: range.end + delta },
-                        type: type
-                    };
-                    commentDirectives = ts.append(commentDirectives, updatedDirective);
-                    if (aggressiveChecks) {
-                        ts.Debug.assert(oldText.substring(range.pos, range.end) === newText.substring(updatedDirective.range.pos, updatedDirective.range.end));
-                    }
-                }
-                // Ignore ranges that fall in change range
+        function setContextFlag(val, flag) {
+            if (val) {
+                contextFlags |= flag;
             }
-            addNewlyScannedDirectives();
-            return commentDirectives;
-            function addNewlyScannedDirectives() {
-                if (addedNewlyScannedDirectives)
-                    return;
-                addedNewlyScannedDirectives = true;
-                if (!commentDirectives) {
-                    commentDirectives = newDirectives;
-                }
-                else if (newDirectives) {
-                    commentDirectives.push.apply(commentDirectives, newDirectives);
-                }
+            else {
+                contextFlags &= ~flag;
             }
         }
-        function moveElementEntirelyPastChangeRange(element, isArray, delta, oldText, newText, aggressiveChecks) {
-            if (isArray) {
-                visitArray(element);
+        function setDisallowInContext(val) {
+            setContextFlag(val, 4096 /* DisallowInContext */);
+        }
+        function setYieldContext(val) {
+            setContextFlag(val, 8192 /* YieldContext */);
+        }
+        function setDecoratorContext(val) {
+            setContextFlag(val, 16384 /* DecoratorContext */);
+        }
+        function setAwaitContext(val) {
+            setContextFlag(val, 32768 /* AwaitContext */);
+        }
+        function doOutsideOfContext(context, func) {
+            // contextFlagsToClear will contain only the context flags that are
+            // currently set that we need to temporarily clear
+            // We don't just blindly reset to the previous flags to ensure
+            // that we do not mutate cached flags for the incremental
+            // parser (ThisNodeHasError, ThisNodeOrAnySubNodesHasError, and
+            // HasAggregatedChildData).
+            var contextFlagsToClear = context & contextFlags;
+            if (contextFlagsToClear) {
+                // clear the requested context flags
+                setContextFlag(/*val*/ false, contextFlagsToClear);
+                var result = func();
+                // restore the context flags we just cleared
+                setContextFlag(/*val*/ true, contextFlagsToClear);
+                return result;
             }
-            else {
-                visitNode(element);
+            // no need to do anything special as we are not in any of the requested contexts
+            return func();
+        }
+        function doInsideOfContext(context, func) {
+            // contextFlagsToSet will contain only the context flags that
+            // are not currently set that we need to temporarily enable.
+            // We don't just blindly reset to the previous flags to ensure
+            // that we do not mutate cached flags for the incremental
+            // parser (ThisNodeHasError, ThisNodeOrAnySubNodesHasError, and
+            // HasAggregatedChildData).
+            var contextFlagsToSet = context & ~contextFlags;
+            if (contextFlagsToSet) {
+                // set the requested context flags
+                setContextFlag(/*val*/ true, contextFlagsToSet);
+                var result = func();
+                // reset the context flags we just set
+                setContextFlag(/*val*/ false, contextFlagsToSet);
+                return result;
             }
-            return;
-            function visitNode(node) {
-                var text = "";
-                if (aggressiveChecks && shouldCheckNode(node)) {
-                    text = oldText.substring(node.pos, node.end);
-                }
-                // Ditch any existing LS children we may have created.  This way we can avoid
-                // moving them forward.
-                if (node._children) {
-                    node._children = undefined;
-                }
-                node.pos += delta;
-                node.end += delta;
-                if (aggressiveChecks && shouldCheckNode(node)) {
-                    ts.Debug.assert(text === newText.substring(node.pos, node.end));
-                }
-                forEachChild(node, visitNode, visitArray);
-                if (ts.hasJSDocNodes(node)) {
-                    for (var _i = 0, _a = node.jsDoc; _i < _a.length; _i++) {
-                        var jsDocComment = _a[_i];
-                        visitNode(jsDocComment);
-                    }
-                }
-                checkNodePositions(node, aggressiveChecks);
+            // no need to do anything special as we are already in all of the requested contexts
+            return func();
+        }
+        function allowInAnd(func) {
+            return doOutsideOfContext(4096 /* DisallowInContext */, func);
+        }
+        function disallowInAnd(func) {
+            return doInsideOfContext(4096 /* DisallowInContext */, func);
+        }
+        function doInYieldContext(func) {
+            return doInsideOfContext(8192 /* YieldContext */, func);
+        }
+        function doInDecoratorContext(func) {
+            return doInsideOfContext(16384 /* DecoratorContext */, func);
+        }
+        function doInAwaitContext(func) {
+            return doInsideOfContext(32768 /* AwaitContext */, func);
+        }
+        function doOutsideOfAwaitContext(func) {
+            return doOutsideOfContext(32768 /* AwaitContext */, func);
+        }
+        function doInYieldAndAwaitContext(func) {
+            return doInsideOfContext(8192 /* YieldContext */ | 32768 /* AwaitContext */, func);
+        }
+        function doOutsideOfYieldAndAwaitContext(func) {
+            return doOutsideOfContext(8192 /* YieldContext */ | 32768 /* AwaitContext */, func);
+        }
+        function inContext(flags) {
+            return (contextFlags & flags) !== 0;
+        }
+        function inYieldContext() {
+            return inContext(8192 /* YieldContext */);
+        }
+        function inDisallowInContext() {
+            return inContext(4096 /* DisallowInContext */);
+        }
+        function inDecoratorContext() {
+            return inContext(16384 /* DecoratorContext */);
+        }
+        function inAwaitContext() {
+            return inContext(32768 /* AwaitContext */);
+        }
+        function parseErrorAtCurrentToken(message, arg0) {
+            parseErrorAt(scanner.getTokenPos(), scanner.getTextPos(), message, arg0);
+        }
+        function parseErrorAtPosition(start, length, message, arg0) {
+            // Don't report another error if it would just be at the same position as the last error.
+            var lastError = ts.lastOrUndefined(parseDiagnostics);
+            if (!lastError || start !== lastError.start) {
+                parseDiagnostics.push(ts.createDetachedDiagnostic(fileName, start, length, message, arg0));
             }
-            function visitArray(array) {
-                array._children = undefined;
-                array.pos += delta;
-                array.end += delta;
-                for (var _i = 0, array_8 = array; _i < array_8.length; _i++) {
-                    var node = array_8[_i];
-                    visitNode(node);
+            // Mark that we've encountered an error.  We'll set an appropriate bit on the next
+            // node we finish so that it can't be reused incrementally.
+            parseErrorBeforeNextFinishedNode = true;
+        }
+        function parseErrorAt(start, end, message, arg0) {
+            parseErrorAtPosition(start, end - start, message, arg0);
+        }
+        function parseErrorAtRange(range, message, arg0) {
+            parseErrorAt(range.pos, range.end, message, arg0);
+        }
+        function scanError(message, length) {
+            parseErrorAtPosition(scanner.getTextPos(), length, message);
+        }
+        function getNodePos() {
+            return scanner.getStartPos();
+        }
+        function hasPrecedingJSDocComment() {
+            return scanner.hasPrecedingJSDocComment();
+        }
+        // Use this function to access the current token instead of reading the currentToken
+        // variable. Since function results aren't narrowed in control flow analysis, this ensures
+        // that the type checker doesn't make wrong assumptions about the type of the current
+        // token (e.g. a call to nextToken() changes the current token but the checker doesn't
+        // reason about this side effect).  Mainstream VMs inline simple functions like this, so
+        // there is no performance penalty.
+        function token() {
+            return currentToken;
+        }
+        function nextTokenWithoutCheck() {
+            return currentToken = scanner.scan();
+        }
+        function nextTokenAnd(func) {
+            nextToken();
+            return func();
+        }
+        function nextToken() {
+            // if the keyword had an escape
+            if (ts.isKeyword(currentToken) && (scanner.hasUnicodeEscape() || scanner.hasExtendedUnicodeEscape())) {
+                // issue a parse error for the escape
+                parseErrorAt(scanner.getTokenPos(), scanner.getTextPos(), ts.Diagnostics.Keywords_cannot_contain_escape_characters);
+            }
+            return nextTokenWithoutCheck();
+        }
+        function nextTokenJSDoc() {
+            return currentToken = scanner.scanJsDocToken();
+        }
+        function reScanGreaterToken() {
+            return currentToken = scanner.reScanGreaterToken();
+        }
+        function reScanSlashToken() {
+            return currentToken = scanner.reScanSlashToken();
+        }
+        function reScanTemplateToken(isTaggedTemplate) {
+            return currentToken = scanner.reScanTemplateToken(isTaggedTemplate);
+        }
+        function reScanTemplateHeadOrNoSubstitutionTemplate() {
+            return currentToken = scanner.reScanTemplateHeadOrNoSubstitutionTemplate();
+        }
+        function reScanLessThanToken() {
+            return currentToken = scanner.reScanLessThanToken();
+        }
+        function scanJsxIdentifier() {
+            return currentToken = scanner.scanJsxIdentifier();
+        }
+        function scanJsxText() {
+            return currentToken = scanner.scanJsxToken();
+        }
+        function scanJsxAttributeValue() {
+            return currentToken = scanner.scanJsxAttributeValue();
+        }
+        function speculationHelper(callback, speculationKind) {
+            // Keep track of the state we'll need to rollback to if lookahead fails (or if the
+            // caller asked us to always reset our state).
+            var saveToken = currentToken;
+            var saveParseDiagnosticsLength = parseDiagnostics.length;
+            var saveParseErrorBeforeNextFinishedNode = parseErrorBeforeNextFinishedNode;
+            // Note: it is not actually necessary to save/restore the context flags here.  That's
+            // because the saving/restoring of these flags happens naturally through the recursive
+            // descent nature of our parser.  However, we still store this here just so we can
+            // assert that invariant holds.
+            var saveContextFlags = contextFlags;
+            // If we're only looking ahead, then tell the scanner to only lookahead as well.
+            // Otherwise, if we're actually speculatively parsing, then tell the scanner to do the
+            // same.
+            var result = speculationKind !== 0 /* TryParse */
+                ? scanner.lookAhead(callback)
+                : scanner.tryScan(callback);
+            ts.Debug.assert(saveContextFlags === contextFlags);
+            // If our callback returned something 'falsy' or we're just looking ahead,
+            // then unconditionally restore us to where we were.
+            if (!result || speculationKind !== 0 /* TryParse */) {
+                currentToken = saveToken;
+                if (speculationKind !== 2 /* Reparse */) {
+                    parseDiagnostics.length = saveParseDiagnosticsLength;
                 }
+                parseErrorBeforeNextFinishedNode = saveParseErrorBeforeNextFinishedNode;
             }
+            return result;
         }
-        function shouldCheckNode(node) {
-            switch (node.kind) {
-                case 10 /* StringLiteral */:
-                case 8 /* NumericLiteral */:
-                case 75 /* Identifier */:
-                    return true;
+        /** Invokes the provided callback then unconditionally restores the parser to the state it
+         * was in immediately prior to invoking the callback.  The result of invoking the callback
+         * is returned from this function.
+         */
+        function lookAhead(callback) {
+            return speculationHelper(callback, 1 /* Lookahead */);
+        }
+        /** Invokes the provided callback.  If the callback returns something falsy, then it restores
+         * the parser to the state it was in immediately prior to invoking the callback.  If the
+         * callback returns something truthy, then the parser state is not rolled back.  The result
+         * of invoking the callback is returned from this function.
+         */
+        function tryParse(callback) {
+            return speculationHelper(callback, 0 /* TryParse */);
+        }
+        function isBindingIdentifier() {
+            if (token() === 78 /* Identifier */) {
+                return true;
             }
-            return false;
+            return token() > 115 /* LastReservedWord */;
         }
-        function adjustIntersectingElement(element, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta) {
-            ts.Debug.assert(element.end >= changeStart, "Adjusting an element that was entirely before the change range");
-            ts.Debug.assert(element.pos <= changeRangeOldEnd, "Adjusting an element that was entirely after the change range");
-            ts.Debug.assert(element.pos <= element.end);
-            // We have an element that intersects the change range in some way.  It may have its
-            // start, or its end (or both) in the changed range.  We want to adjust any part
-            // that intersects such that the final tree is in a consistent state.  i.e. all
-            // children have spans within the span of their parent, and all siblings are ordered
-            // properly.
-            // We may need to update both the 'pos' and the 'end' of the element.
-            // If the 'pos' is before the start of the change, then we don't need to touch it.
-            // If it isn't, then the 'pos' must be inside the change.  How we update it will
-            // depend if delta is positive or negative. If delta is positive then we have
-            // something like:
-            //
-            //  -------------------AAA-----------------
-            //  -------------------BBBCCCCCCC-----------------
-            //
-            // In this case, we consider any node that started in the change range to still be
-            // starting at the same position.
-            //
-            // however, if the delta is negative, then we instead have something like this:
-            //
-            //  -------------------XXXYYYYYYY-----------------
-            //  -------------------ZZZ-----------------
-            //
-            // In this case, any element that started in the 'X' range will keep its position.
-            // However any element that started after that will have their pos adjusted to be
-            // at the end of the new range.  i.e. any node that started in the 'Y' range will
-            // be adjusted to have their start at the end of the 'Z' range.
-            //
-            // The element will keep its position if possible.  Or Move backward to the new-end
-            // if it's in the 'Y' range.
-            element.pos = Math.min(element.pos, changeRangeNewEnd);
-            // If the 'end' is after the change range, then we always adjust it by the delta
-            // amount.  However, if the end is in the change range, then how we adjust it
-            // will depend on if delta is positive or negative.  If delta is positive then we
-            // have something like:
-            //
-            //  -------------------AAA-----------------
-            //  -------------------BBBCCCCCCC-----------------
-            //
-            // In this case, we consider any node that ended inside the change range to keep its
-            // end position.
-            //
-            // however, if the delta is negative, then we instead have something like this:
-            //
-            //  -------------------XXXYYYYYYY-----------------
-            //  -------------------ZZZ-----------------
-            //
-            // In this case, any element that ended in the 'X' range will keep its position.
-            // However any element that ended after that will have their pos adjusted to be
-            // at the end of the new range.  i.e. any node that ended in the 'Y' range will
-            // be adjusted to have their end at the end of the 'Z' range.
-            if (element.end >= changeRangeOldEnd) {
-                // Element ends after the change range.  Always adjust the end pos.
-                element.end += delta;
+        // Ignore strict mode flag because we will report an error in type checker instead.
+        function isIdentifier() {
+            if (token() === 78 /* Identifier */) {
+                return true;
             }
-            else {
-                // Element ends in the change range.  The element will keep its position if
-                // possible. Or Move backward to the new-end if it's in the 'Y' range.
-                element.end = Math.min(element.end, changeRangeNewEnd);
+            // If we have a 'yield' keyword, and we're in the [yield] context, then 'yield' is
+            // considered a keyword and is not an identifier.
+            if (token() === 124 /* YieldKeyword */ && inYieldContext()) {
+                return false;
             }
-            ts.Debug.assert(element.pos <= element.end);
-            if (element.parent) {
-                ts.Debug.assert(element.pos >= element.parent.pos);
-                ts.Debug.assert(element.end <= element.parent.end);
+            // If we have a 'await' keyword, and we're in the [Await] context, then 'await' is
+            // considered a keyword and is not an identifier.
+            if (token() === 130 /* AwaitKeyword */ && inAwaitContext()) {
+                return false;
             }
+            return token() > 115 /* LastReservedWord */;
         }
-        function checkNodePositions(node, aggressiveChecks) {
-            if (aggressiveChecks) {
-                var pos_2 = node.pos;
-                var visitNode_1 = function (child) {
-                    ts.Debug.assert(child.pos >= pos_2);
-                    pos_2 = child.end;
-                };
-                if (ts.hasJSDocNodes(node)) {
-                    for (var _i = 0, _a = node.jsDoc; _i < _a.length; _i++) {
-                        var jsDocComment = _a[_i];
-                        visitNode_1(jsDocComment);
-                    }
+        function parseExpected(kind, diagnosticMessage, shouldAdvance) {
+            if (shouldAdvance === void 0) { shouldAdvance = true; }
+            if (token() === kind) {
+                if (shouldAdvance) {
+                    nextToken();
                 }
-                forEachChild(node, visitNode_1);
-                ts.Debug.assert(pos_2 <= node.end);
+                return true;
             }
-        }
-        function updateTokenPositionsAndMarkElements(sourceFile, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta, oldText, newText, aggressiveChecks) {
-            visitNode(sourceFile);
-            return;
-            function visitNode(child) {
-                ts.Debug.assert(child.pos <= child.end);
-                if (child.pos > changeRangeOldEnd) {
-                    // Node is entirely past the change range.  We need to move both its pos and
-                    // end, forward or backward appropriately.
-                    moveElementEntirelyPastChangeRange(child, /*isArray*/ false, delta, oldText, newText, aggressiveChecks);
-                    return;
-                }
-                // Check if the element intersects the change range.  If it does, then it is not
-                // reusable.  Also, we'll need to recurse to see what constituent portions we may
-                // be able to use.
-                var fullEnd = child.end;
-                if (fullEnd >= changeStart) {
-                    child.intersectsChange = true;
-                    child._children = undefined;
-                    // Adjust the pos or end (or both) of the intersecting element accordingly.
-                    adjustIntersectingElement(child, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta);
-                    forEachChild(child, visitNode, visitArray);
-                    if (ts.hasJSDocNodes(child)) {
-                        for (var _i = 0, _a = child.jsDoc; _i < _a.length; _i++) {
-                            var jsDocComment = _a[_i];
-                            visitNode(jsDocComment);
-                        }
-                    }
-                    checkNodePositions(child, aggressiveChecks);
-                    return;
-                }
-                // Otherwise, the node is entirely before the change range.  No need to do anything with it.
-                ts.Debug.assert(fullEnd < changeStart);
+            // Report specific message if provided with one.  Otherwise, report generic fallback message.
+            if (diagnosticMessage) {
+                parseErrorAtCurrentToken(diagnosticMessage);
             }
-            function visitArray(array) {
-                ts.Debug.assert(array.pos <= array.end);
-                if (array.pos > changeRangeOldEnd) {
-                    // Array is entirely after the change range.  We need to move it, and move any of
-                    // its children.
-                    moveElementEntirelyPastChangeRange(array, /*isArray*/ true, delta, oldText, newText, aggressiveChecks);
-                    return;
-                }
-                // Check if the element intersects the change range.  If it does, then it is not
-                // reusable.  Also, we'll need to recurse to see what constituent portions we may
-                // be able to use.
-                var fullEnd = array.end;
-                if (fullEnd >= changeStart) {
-                    array.intersectsChange = true;
-                    array._children = undefined;
-                    // Adjust the pos or end (or both) of the intersecting array accordingly.
-                    adjustIntersectingElement(array, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta);
-                    for (var _i = 0, array_9 = array; _i < array_9.length; _i++) {
-                        var node = array_9[_i];
-                        visitNode(node);
-                    }
-                    return;
-                }
-                // Otherwise, the array is entirely before the change range.  No need to do anything with it.
-                ts.Debug.assert(fullEnd < changeStart);
+            else {
+                parseErrorAtCurrentToken(ts.Diagnostics._0_expected, ts.tokenToString(kind));
             }
+            return false;
         }
-        function extendToAffectedRange(sourceFile, changeRange) {
-            // Consider the following code:
-            //      void foo() { /; }
-            //
-            // If the text changes with an insertion of / just before the semicolon then we end up with:
-            //      void foo() { //; }
-            //
-            // If we were to just use the changeRange a is, then we would not rescan the { token
-            // (as it does not intersect the actual original change range).  Because an edit may
-            // change the token touching it, we actually need to look back *at least* one token so
-            // that the prior token sees that change.
-            var maxLookahead = 1;
-            var start = changeRange.span.start;
-            // the first iteration aligns us with the change start. subsequent iteration move us to
-            // the left by maxLookahead tokens.  We only need to do this as long as we're not at the
-            // start of the tree.
-            for (var i = 0; start > 0 && i <= maxLookahead; i++) {
-                var nearestNode = findNearestNodeStartingBeforeOrAtPosition(sourceFile, start);
-                ts.Debug.assert(nearestNode.pos <= start);
-                var position = nearestNode.pos;
-                start = Math.max(0, position - 1);
+        function parseExpectedJSDoc(kind) {
+            if (token() === kind) {
+                nextTokenJSDoc();
+                return true;
             }
-            var finalSpan = ts.createTextSpanFromBounds(start, ts.textSpanEnd(changeRange.span));
-            var finalLength = changeRange.newLength + (changeRange.span.start - start);
-            return ts.createTextChangeRange(finalSpan, finalLength);
+            parseErrorAtCurrentToken(ts.Diagnostics._0_expected, ts.tokenToString(kind));
+            return false;
         }
-        function findNearestNodeStartingBeforeOrAtPosition(sourceFile, position) {
-            var bestResult = sourceFile;
-            var lastNodeEntirelyBeforePosition;
-            forEachChild(sourceFile, visit);
-            if (lastNodeEntirelyBeforePosition) {
-                var lastChildOfLastEntireNodeBeforePosition = getLastDescendant(lastNodeEntirelyBeforePosition);
-                if (lastChildOfLastEntireNodeBeforePosition.pos > bestResult.pos) {
-                    bestResult = lastChildOfLastEntireNodeBeforePosition;
-                }
+        function parseOptional(t) {
+            if (token() === t) {
+                nextToken();
+                return true;
             }
-            return bestResult;
-            function getLastDescendant(node) {
-                while (true) {
-                    var lastChild = ts.getLastChild(node);
-                    if (lastChild) {
-                        node = lastChild;
-                    }
-                    else {
-                        return node;
-                    }
-                }
+            return false;
+        }
+        function parseOptionalToken(t) {
+            if (token() === t) {
+                return parseTokenNode();
             }
-            function visit(child) {
-                if (ts.nodeIsMissing(child)) {
-                    // Missing nodes are effectively invisible to us.  We never even consider them
-                    // When trying to find the nearest node before us.
-                    return;
-                }
-                // If the child intersects this position, then this node is currently the nearest
-                // node that starts before the position.
-                if (child.pos <= position) {
-                    if (child.pos >= bestResult.pos) {
-                        // This node starts before the position, and is closer to the position than
-                        // the previous best node we found.  It is now the new best node.
-                        bestResult = child;
-                    }
-                    // Now, the node may overlap the position, or it may end entirely before the
-                    // position.  If it overlaps with the position, then either it, or one of its
-                    // children must be the nearest node before the position.  So we can just
-                    // recurse into this child to see if we can find something better.
-                    if (position < child.end) {
-                        // The nearest node is either this child, or one of the children inside
-                        // of it.  We've already marked this child as the best so far.  Recurse
-                        // in case one of the children is better.
-                        forEachChild(child, visit);
-                        // Once we look at the children of this node, then there's no need to
-                        // continue any further.
-                        return true;
-                    }
-                    else {
-                        ts.Debug.assert(child.end <= position);
-                        // The child ends entirely before this position.  Say you have the following
-                        // (where $ is the position)
-                        //
-                        //      <complex expr 1> ? <complex expr 2> $ : <...> <...>
-                        //
-                        // We would want to find the nearest preceding node in "complex expr 2".
-                        // To support that, we keep track of this node, and once we're done searching
-                        // for a best node, we recurse down this node to see if we can find a good
-                        // result in it.
-                        //
-                        // This approach allows us to quickly skip over nodes that are entirely
-                        // before the position, while still allowing us to find any nodes in the
-                        // last one that might be what we want.
-                        lastNodeEntirelyBeforePosition = child;
-                    }
-                }
-                else {
-                    ts.Debug.assert(child.pos > position);
-                    // We're now at a node that is entirely past the position we're searching for.
-                    // This node (and all following nodes) could never contribute to the result,
-                    // so just skip them by returning 'true' here.
-                    return true;
-                }
+            return undefined;
+        }
+        function parseOptionalTokenJSDoc(t) {
+            if (token() === t) {
+                return parseTokenNodeJSDoc();
             }
+            return undefined;
         }
-        function checkChangeRange(sourceFile, newText, textChangeRange, aggressiveChecks) {
-            var oldText = sourceFile.text;
-            if (textChangeRange) {
-                ts.Debug.assert((oldText.length - textChangeRange.span.length + textChangeRange.newLength) === newText.length);
-                if (aggressiveChecks || ts.Debug.shouldAssert(3 /* VeryAggressive */)) {
-                    var oldTextPrefix = oldText.substr(0, textChangeRange.span.start);
-                    var newTextPrefix = newText.substr(0, textChangeRange.span.start);
-                    ts.Debug.assert(oldTextPrefix === newTextPrefix);
-                    var oldTextSuffix = oldText.substring(ts.textSpanEnd(textChangeRange.span), oldText.length);
-                    var newTextSuffix = newText.substring(ts.textSpanEnd(ts.textChangeRangeNewSpan(textChangeRange)), newText.length);
-                    ts.Debug.assert(oldTextSuffix === newTextSuffix);
-                }
+        function parseExpectedToken(t, diagnosticMessage, arg0) {
+            return parseOptionalToken(t) ||
+                createMissingNode(t, /*reportAtCurrentPosition*/ false, diagnosticMessage || ts.Diagnostics._0_expected, arg0 || ts.tokenToString(t));
+        }
+        function parseExpectedTokenJSDoc(t) {
+            return parseOptionalTokenJSDoc(t) ||
+                createMissingNode(t, /*reportAtCurrentPosition*/ false, ts.Diagnostics._0_expected, ts.tokenToString(t));
+        }
+        function parseTokenNode() {
+            var pos = getNodePos();
+            var kind = token();
+            nextToken();
+            return finishNode(factory.createToken(kind), pos);
+        }
+        function parseTokenNodeJSDoc() {
+            var pos = getNodePos();
+            var kind = token();
+            nextTokenJSDoc();
+            return finishNode(factory.createToken(kind), pos);
+        }
+        function canParseSemicolon() {
+            // If there's a real semicolon, then we can always parse it out.
+            if (token() === 26 /* SemicolonToken */) {
+                return true;
             }
+            // We can parse out an optional semicolon in ASI cases in the following cases.
+            return token() === 19 /* CloseBraceToken */ || token() === 1 /* EndOfFileToken */ || scanner.hasPrecedingLineBreak();
         }
-        function createSyntaxCursor(sourceFile) {
-            var currentArray = sourceFile.statements;
-            var currentArrayIndex = 0;
-            ts.Debug.assert(currentArrayIndex < currentArray.length);
-            var current = currentArray[currentArrayIndex];
-            var lastQueriedPosition = -1 /* Value */;
-            return {
-                currentNode: function (position) {
-                    // Only compute the current node if the position is different than the last time
-                    // we were asked.  The parser commonly asks for the node at the same position
-                    // twice.  Once to know if can read an appropriate list element at a certain point,
-                    // and then to actually read and consume the node.
-                    if (position !== lastQueriedPosition) {
-                        // Much of the time the parser will need the very next node in the array that
-                        // we just returned a node from.So just simply check for that case and move
-                        // forward in the array instead of searching for the node again.
-                        if (current && current.end === position && currentArrayIndex < (currentArray.length - 1)) {
-                            currentArrayIndex++;
-                            current = currentArray[currentArrayIndex];
-                        }
-                        // If we don't have a node, or the node we have isn't in the right position,
-                        // then try to find a viable node at the position requested.
-                        if (!current || current.pos !== position) {
-                            findHighestListElementThatStartsAtPosition(position);
-                        }
-                    }
-                    // Cache this query so that we don't do any extra work if the parser calls back
-                    // into us.  Note: this is very common as the parser will make pairs of calls like
-                    // 'isListElement -> parseListElement'.  If we were unable to find a node when
-                    // called with 'isListElement', we don't want to redo the work when parseListElement
-                    // is called immediately after.
-                    lastQueriedPosition = position;
-                    // Either we don'd have a node, or we have a node at the position being asked for.
-                    ts.Debug.assert(!current || current.pos === position);
-                    return current;
-                }
-            };
-            // Finds the highest element in the tree we can find that starts at the provided position.
-            // The element must be a direct child of some node list in the tree.  This way after we
-            // return it, we can easily return its next sibling in the list.
-            function findHighestListElementThatStartsAtPosition(position) {
-                // Clear out any cached state about the last node we found.
-                currentArray = undefined;
-                currentArrayIndex = -1 /* Value */;
-                current = undefined;
-                // Recurse into the source file to find the highest node at this position.
-                forEachChild(sourceFile, visitNode, visitArray);
-                return;
-                function visitNode(node) {
-                    if (position >= node.pos && position < node.end) {
-                        // Position was within this node.  Keep searching deeper to find the node.
-                        forEachChild(node, visitNode, visitArray);
-                        // don't proceed any further in the search.
-                        return true;
-                    }
-                    // position wasn't in this node, have to keep searching.
-                    return false;
-                }
-                function visitArray(array) {
-                    if (position >= array.pos && position < array.end) {
-                        // position was in this array.  Search through this array to see if we find a
-                        // viable element.
-                        for (var i = 0; i < array.length; i++) {
-                            var child = array[i];
-                            if (child) {
-                                if (child.pos === position) {
-                                    // Found the right node.  We're done.
-                                    currentArray = array;
-                                    currentArrayIndex = i;
-                                    current = child;
-                                    return true;
-                                }
-                                else {
-                                    if (child.pos < position && position < child.end) {
-                                        // Position in somewhere within this child.  Search in it and
-                                        // stop searching in this array.
-                                        forEachChild(child, visitNode, visitArray);
-                                        return true;
-                                    }
-                                }
-                            }
-                        }
-                    }
-                    // position wasn't in this array, have to keep searching.
-                    return false;
+        function parseSemicolon() {
+            if (canParseSemicolon()) {
+                if (token() === 26 /* SemicolonToken */) {
+                    // consume the semicolon if it was explicitly provided.
+                    nextToken();
                 }
+                return true;
+            }
+            else {
+                return parseExpected(26 /* SemicolonToken */);
             }
         }
-        var InvalidPosition;
-        (function (InvalidPosition) {
-            InvalidPosition[InvalidPosition["Value"] = -1] = "Value";
-        })(InvalidPosition || (InvalidPosition = {}));
-    })(IncrementalParser || (IncrementalParser = {}));
-    /** @internal */
-    function isDeclarationFileName(fileName) {
-        return ts.fileExtensionIs(fileName, ".d.ts" /* Dts */);
-    }
-    ts.isDeclarationFileName = isDeclarationFileName;
-    /*@internal*/
-    function processCommentPragmas(context, sourceText) {
-        var pragmas = [];
-        for (var _i = 0, _a = ts.getLeadingCommentRanges(sourceText, 0) || ts.emptyArray; _i < _a.length; _i++) {
-            var range = _a[_i];
-            var comment = sourceText.substring(range.pos, range.end);
-            extractPragmas(pragmas, range, comment);
+        function createNodeArray(elements, pos, end, hasTrailingComma) {
+            var array = factory.createNodeArray(elements, hasTrailingComma);
+            ts.setTextRangePosEnd(array, pos, end !== null && end !== void 0 ? end : scanner.getStartPos());
+            return array;
         }
-        context.pragmas = ts.createMap();
-        for (var _b = 0, pragmas_1 = pragmas; _b < pragmas_1.length; _b++) {
-            var pragma = pragmas_1[_b];
-            if (context.pragmas.has(pragma.name)) {
-                var currentValue = context.pragmas.get(pragma.name);
-                if (currentValue instanceof Array) {
-                    currentValue.push(pragma.args);
-                }
-                else {
-                    context.pragmas.set(pragma.name, [currentValue, pragma.args]);
-                }
-                continue;
+        function finishNode(node, pos, end) {
+            ts.setTextRangePosEnd(node, pos, end !== null && end !== void 0 ? end : scanner.getStartPos());
+            if (contextFlags) {
+                node.flags |= contextFlags;
             }
-            context.pragmas.set(pragma.name, pragma.args);
+            // Keep track on the node if we encountered an error while parsing it.  If we did, then
+            // we cannot reuse the node incrementally.  Once we've marked this node, clear out the
+            // flag so that we don't mark any subsequent nodes.
+            if (parseErrorBeforeNextFinishedNode) {
+                parseErrorBeforeNextFinishedNode = false;
+                node.flags |= 65536 /* ThisNodeHasError */;
+            }
+            return node;
         }
-    }
-    ts.processCommentPragmas = processCommentPragmas;
-    /*@internal*/
-    function processPragmasIntoFields(context, reportDiagnostic) {
-        context.checkJsDirective = undefined;
-        context.referencedFiles = [];
-        context.typeReferenceDirectives = [];
-        context.libReferenceDirectives = [];
-        context.amdDependencies = [];
-        context.hasNoDefaultLib = false;
-        context.pragmas.forEach(function (entryOrList, key) {
-            // TODO: The below should be strongly type-guarded and not need casts/explicit annotations, since entryOrList is related to
-            // key and key is constrained to a union; but it's not (see GH#21483 for at least partial fix) :(
-            switch (key) {
-                case "reference": {
-                    var referencedFiles_1 = context.referencedFiles;
-                    var typeReferenceDirectives_1 = context.typeReferenceDirectives;
-                    var libReferenceDirectives_1 = context.libReferenceDirectives;
-                    ts.forEach(ts.toArray(entryOrList), function (arg) {
-                        var _a = arg.arguments, types = _a.types, lib = _a.lib, path = _a.path;
-                        if (arg.arguments["no-default-lib"]) {
-                            context.hasNoDefaultLib = true;
-                        }
-                        else if (types) {
-                            typeReferenceDirectives_1.push({ pos: types.pos, end: types.end, fileName: types.value });
-                        }
-                        else if (lib) {
-                            libReferenceDirectives_1.push({ pos: lib.pos, end: lib.end, fileName: lib.value });
-                        }
-                        else if (path) {
-                            referencedFiles_1.push({ pos: path.pos, end: path.end, fileName: path.value });
-                        }
-                        else {
-                            reportDiagnostic(arg.range.pos, arg.range.end - arg.range.pos, ts.Diagnostics.Invalid_reference_directive_syntax);
-                        }
-                    });
-                    break;
-                }
-                case "amd-dependency": {
-                    context.amdDependencies = ts.map(ts.toArray(entryOrList), function (x) { return ({ name: x.arguments.name, path: x.arguments.path }); });
-                    break;
-                }
-                case "amd-module": {
-                    if (entryOrList instanceof Array) {
-                        for (var _i = 0, entryOrList_1 = entryOrList; _i < entryOrList_1.length; _i++) {
-                            var entry = entryOrList_1[_i];
-                            if (context.moduleName) {
-                                // TODO: It's probably fine to issue this diagnostic on all instances of the pragma
-                                reportDiagnostic(entry.range.pos, entry.range.end - entry.range.pos, ts.Diagnostics.An_AMD_module_cannot_have_multiple_name_assignments);
-                            }
-                            context.moduleName = entry.arguments.name;
-                        }
-                    }
-                    else {
-                        context.moduleName = entryOrList.arguments.name;
-                    }
-                    break;
-                }
-                case "ts-nocheck":
-                case "ts-check": {
-                    // _last_ of either nocheck or check in a file is the "winner"
-                    ts.forEach(ts.toArray(entryOrList), function (entry) {
-                        if (!context.checkJsDirective || entry.range.pos > context.checkJsDirective.pos) {
-                            context.checkJsDirective = {
-                                enabled: key === "ts-check",
-                                end: entry.range.end,
-                                pos: entry.range.pos
-                            };
-                        }
-                    });
-                    break;
-                }
-                case "jsx": return; // Accessed directly
-                default: ts.Debug.fail("Unhandled pragma kind"); // Can this be made into an assertNever in the future?
+        function createMissingNode(kind, reportAtCurrentPosition, diagnosticMessage, arg0) {
+            if (reportAtCurrentPosition) {
+                parseErrorAtPosition(scanner.getStartPos(), 0, diagnosticMessage, arg0);
             }
-        });
-    }
-    ts.processPragmasIntoFields = processPragmasIntoFields;
-    var namedArgRegExCache = ts.createMap();
-    function getNamedArgRegEx(name) {
-        if (namedArgRegExCache.has(name)) {
-            return namedArgRegExCache.get(name);
+            else if (diagnosticMessage) {
+                parseErrorAtCurrentToken(diagnosticMessage, arg0);
+            }
+            var pos = getNodePos();
+            var result = kind === 78 /* Identifier */ ? factory.createIdentifier("", /*typeArguments*/ undefined, /*originalKeywordKind*/ undefined) :
+                ts.isTemplateLiteralKind(kind) ? factory.createTemplateLiteralLikeNode(kind, "", "", /*templateFlags*/ undefined) :
+                    kind === 8 /* NumericLiteral */ ? factory.createNumericLiteral("", /*numericLiteralFlags*/ undefined) :
+                        kind === 10 /* StringLiteral */ ? factory.createStringLiteral("", /*isSingleQuote*/ undefined) :
+                            kind === 271 /* MissingDeclaration */ ? factory.createMissingDeclaration() :
+                                factory.createToken(kind);
+            return finishNode(result, pos);
         }
-        var result = new RegExp("(\\s" + name + "\\s*=\\s*)('|\")(.+?)\\2", "im");
-        namedArgRegExCache.set(name, result);
-        return result;
-    }
-    var tripleSlashXMLCommentStartRegEx = /^\/\/\/\s*<(\S+)\s.*?\/>/im;
-    var singleLinePragmaRegEx = /^\/\/\/?\s*@(\S+)\s*(.*)\s*$/im;
-    function extractPragmas(pragmas, range, text) {
-        var tripleSlash = range.kind === 2 /* SingleLineCommentTrivia */ && tripleSlashXMLCommentStartRegEx.exec(text);
-        if (tripleSlash) {
-            var name = tripleSlash[1].toLowerCase(); // Technically unsafe cast, but we do it so the below check to make it safe typechecks
-            var pragma = ts.commentPragmas[name];
-            if (!pragma || !(pragma.kind & 1 /* TripleSlashXML */)) {
-                return;
+        function internIdentifier(text) {
+            var identifier = identifiers.get(text);
+            if (identifier === undefined) {
+                identifiers.set(text, identifier = text);
             }
-            if (pragma.args) {
-                var argument = {};
-                for (var _i = 0, _a = pragma.args; _i < _a.length; _i++) {
-                    var arg = _a[_i];
-                    var matcher = getNamedArgRegEx(arg.name);
-                    var matchResult = matcher.exec(text);
-                    if (!matchResult && !arg.optional) {
-                        return; // Missing required argument, don't parse
-                    }
-                    else if (matchResult) {
-                        if (arg.captureSpan) {
-                            var startPos = range.pos + matchResult.index + matchResult[1].length + matchResult[2].length;
-                            argument[arg.name] = {
-                                value: matchResult[3],
-                                pos: startPos,
-                                end: startPos + matchResult[3].length
-                            };
-                        }
-                        else {
-                            argument[arg.name] = matchResult[3];
-                        }
-                    }
-                }
-                pragmas.push({ name: name, args: { arguments: argument, range: range } });
+            return identifier;
+        }
+        // An identifier that starts with two underscores has an extra underscore character prepended to it to avoid issues
+        // with magic property names like '__proto__'. The 'identifiers' object is used to share a single string instance for
+        // each identifier in order to reduce memory consumption.
+        function createIdentifier(isIdentifier, diagnosticMessage, privateIdentifierDiagnosticMessage) {
+            identifierCount++;
+            if (isIdentifier) {
+                var pos = getNodePos();
+                // Store original token kind if it is not just an Identifier so we can report appropriate error later in type checker
+                var originalKeywordKind = token();
+                var text = internIdentifier(scanner.getTokenValue());
+                nextTokenWithoutCheck();
+                return finishNode(factory.createIdentifier(text, /*typeArguments*/ undefined, originalKeywordKind), pos);
             }
-            else {
-                pragmas.push({ name: name, args: { arguments: {}, range: range } });
+            if (token() === 79 /* PrivateIdentifier */) {
+                parseErrorAtCurrentToken(privateIdentifierDiagnosticMessage || ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
+                return createIdentifier(/*isIdentifier*/ true);
             }
-            return;
+            // Only for end of file because the error gets reported incorrectly on embedded script tags.
+            var reportAtCurrentPosition = token() === 1 /* EndOfFileToken */;
+            var isReservedWord = scanner.isReservedWord();
+            var msgArg = scanner.getTokenText();
+            var defaultMessage = isReservedWord ?
+                ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_that_cannot_be_used_here :
+                ts.Diagnostics.Identifier_expected;
+            return createMissingNode(78 /* Identifier */, reportAtCurrentPosition, diagnosticMessage || defaultMessage, msgArg);
         }
-        var singleLine = range.kind === 2 /* SingleLineCommentTrivia */ && singleLinePragmaRegEx.exec(text);
-        if (singleLine) {
-            return addPragmaForMatch(pragmas, range, 2 /* SingleLine */, singleLine);
+        function parseBindingIdentifier(privateIdentifierDiagnosticMessage) {
+            return createIdentifier(isBindingIdentifier(), /*diagnosticMessage*/ undefined, privateIdentifierDiagnosticMessage);
         }
-        if (range.kind === 3 /* MultiLineCommentTrivia */) {
-            var multiLinePragmaRegEx = /\s*@(\S+)\s*(.*)\s*$/gim; // Defined inline since it uses the "g" flag, which keeps a persistent index (for iterating)
-            var multiLineMatch = void 0;
-            while (multiLineMatch = multiLinePragmaRegEx.exec(text)) {
-                addPragmaForMatch(pragmas, range, 4 /* MultiLine */, multiLineMatch);
-            }
+        function parseIdentifier(diagnosticMessage, privateIdentifierDiagnosticMessage) {
+            return createIdentifier(isIdentifier(), diagnosticMessage, privateIdentifierDiagnosticMessage);
         }
-    }
-    function addPragmaForMatch(pragmas, range, kind, match) {
-        if (!match)
-            return;
-        var name = match[1].toLowerCase(); // Technically unsafe cast, but we do it so they below check to make it safe typechecks
-        var pragma = ts.commentPragmas[name];
-        if (!pragma || !(pragma.kind & kind)) {
-            return;
+        function parseIdentifierName(diagnosticMessage) {
+            return createIdentifier(ts.tokenIsIdentifierOrKeyword(token()), diagnosticMessage);
         }
-        var args = match[2]; // Split on spaces and match up positionally with definition
-        var argument = getNamedPragmaArguments(pragma, args);
-        if (argument === "fail")
-            return; // Missing required argument, fail to parse it
-        pragmas.push({ name: name, args: { arguments: argument, range: range } });
-        return;
-    }
-    function getNamedPragmaArguments(pragma, text) {
-        if (!text)
-            return {};
-        if (!pragma.args)
-            return {};
-        var args = text.split(/\s+/);
-        var argMap = {};
-        for (var i = 0; i < pragma.args.length; i++) {
-            var argument = pragma.args[i];
-            if (!args[i] && !argument.optional) {
-                return "fail";
+        function isLiteralPropertyName() {
+            return ts.tokenIsIdentifierOrKeyword(token()) ||
+                token() === 10 /* StringLiteral */ ||
+                token() === 8 /* NumericLiteral */;
+        }
+        function parsePropertyNameWorker(allowComputedPropertyNames) {
+            if (token() === 10 /* StringLiteral */ || token() === 8 /* NumericLiteral */) {
+                var node = parseLiteralNode();
+                node.text = internIdentifier(node.text);
+                return node;
             }
-            if (argument.captureSpan) {
-                return ts.Debug.fail("Capture spans not yet implemented for non-xml pragmas");
+            if (allowComputedPropertyNames && token() === 22 /* OpenBracketToken */) {
+                return parseComputedPropertyName();
             }
-            argMap[argument.name] = args[i];
+            if (token() === 79 /* PrivateIdentifier */) {
+                return parsePrivateIdentifier();
+            }
+            return parseIdentifierName();
         }
-        return argMap;
-    }
-    /** @internal */
-    function tagNamesAreEquivalent(lhs, rhs) {
-        if (lhs.kind !== rhs.kind) {
-            return false;
+        function parsePropertyName() {
+            return parsePropertyNameWorker(/*allowComputedPropertyNames*/ true);
         }
-        if (lhs.kind === 75 /* Identifier */) {
-            return lhs.escapedText === rhs.escapedText;
+        function parseComputedPropertyName() {
+            // PropertyName [Yield]:
+            //      LiteralPropertyName
+            //      ComputedPropertyName[?Yield]
+            var pos = getNodePos();
+            parseExpected(22 /* OpenBracketToken */);
+            // We parse any expression (including a comma expression). But the grammar
+            // says that only an assignment expression is allowed, so the grammar checker
+            // will error if it sees a comma expression.
+            var expression = allowInAnd(parseExpression);
+            parseExpected(23 /* CloseBracketToken */);
+            return finishNode(factory.createComputedPropertyName(expression), pos);
         }
-        if (lhs.kind === 104 /* ThisKeyword */) {
-            return true;
+        function internPrivateIdentifier(text) {
+            var privateIdentifier = privateIdentifiers.get(text);
+            if (privateIdentifier === undefined) {
+                privateIdentifiers.set(text, privateIdentifier = text);
+            }
+            return privateIdentifier;
         }
-        // If we are at this statement then we must have PropertyAccessExpression and because tag name in Jsx element can only
-        // take forms of JsxTagNameExpression which includes an identifier, "this" expression, or another propertyAccessExpression
-        // it is safe to case the expression property as such. See parseJsxElementName for how we parse tag name in Jsx element
-        return lhs.name.escapedText === rhs.name.escapedText &&
-            tagNamesAreEquivalent(lhs.expression, rhs.expression);
-    }
-    ts.tagNamesAreEquivalent = tagNamesAreEquivalent;
-})(ts || (ts = {}));
-var ts;
-(function (ts) {
-    /* @internal */
-    ts.compileOnSaveCommandLineOption = { name: "compileOnSave", type: "boolean" };
-    // NOTE: The order here is important to default lib ordering as entries will have the same
-    //       order in the generated program (see `getDefaultLibPriority` in program.ts). This
-    //       order also affects overload resolution when a type declared in one lib is
-    //       augmented in another lib.
-    var libEntries = [
-        // JavaScript only
-        ["es5", "lib.es5.d.ts"],
-        ["es6", "lib.es2015.d.ts"],
-        ["es2015", "lib.es2015.d.ts"],
-        ["es7", "lib.es2016.d.ts"],
-        ["es2016", "lib.es2016.d.ts"],
-        ["es2017", "lib.es2017.d.ts"],
-        ["es2018", "lib.es2018.d.ts"],
-        ["es2019", "lib.es2019.d.ts"],
-        ["es2020", "lib.es2020.d.ts"],
-        ["esnext", "lib.esnext.d.ts"],
-        // Host only
-        ["dom", "lib.dom.d.ts"],
-        ["dom.iterable", "lib.dom.iterable.d.ts"],
-        ["webworker", "lib.webworker.d.ts"],
-        ["webworker.importscripts", "lib.webworker.importscripts.d.ts"],
-        ["scripthost", "lib.scripthost.d.ts"],
-        // ES2015 Or ESNext By-feature options
-        ["es2015.core", "lib.es2015.core.d.ts"],
-        ["es2015.collection", "lib.es2015.collection.d.ts"],
-        ["es2015.generator", "lib.es2015.generator.d.ts"],
-        ["es2015.iterable", "lib.es2015.iterable.d.ts"],
-        ["es2015.promise", "lib.es2015.promise.d.ts"],
-        ["es2015.proxy", "lib.es2015.proxy.d.ts"],
-        ["es2015.reflect", "lib.es2015.reflect.d.ts"],
-        ["es2015.symbol", "lib.es2015.symbol.d.ts"],
-        ["es2015.symbol.wellknown", "lib.es2015.symbol.wellknown.d.ts"],
-        ["es2016.array.include", "lib.es2016.array.include.d.ts"],
-        ["es2017.object", "lib.es2017.object.d.ts"],
-        ["es2017.sharedmemory", "lib.es2017.sharedmemory.d.ts"],
-        ["es2017.string", "lib.es2017.string.d.ts"],
-        ["es2017.intl", "lib.es2017.intl.d.ts"],
-        ["es2017.typedarrays", "lib.es2017.typedarrays.d.ts"],
-        ["es2018.asyncgenerator", "lib.es2018.asyncgenerator.d.ts"],
-        ["es2018.asynciterable", "lib.es2018.asynciterable.d.ts"],
-        ["es2018.intl", "lib.es2018.intl.d.ts"],
-        ["es2018.promise", "lib.es2018.promise.d.ts"],
-        ["es2018.regexp", "lib.es2018.regexp.d.ts"],
-        ["es2019.array", "lib.es2019.array.d.ts"],
-        ["es2019.object", "lib.es2019.object.d.ts"],
-        ["es2019.string", "lib.es2019.string.d.ts"],
-        ["es2019.symbol", "lib.es2019.symbol.d.ts"],
-        ["es2020.bigint", "lib.es2020.bigint.d.ts"],
-        ["es2020.promise", "lib.es2020.promise.d.ts"],
-        ["es2020.string", "lib.es2020.string.d.ts"],
-        ["es2020.symbol.wellknown", "lib.es2020.symbol.wellknown.d.ts"],
-        ["esnext.array", "lib.es2019.array.d.ts"],
-        ["esnext.symbol", "lib.es2019.symbol.d.ts"],
-        ["esnext.asynciterable", "lib.es2018.asynciterable.d.ts"],
-        ["esnext.intl", "lib.esnext.intl.d.ts"],
-        ["esnext.bigint", "lib.es2020.bigint.d.ts"],
-        ["esnext.string", "lib.esnext.string.d.ts"],
-        ["esnext.promise", "lib.esnext.promise.d.ts"]
-    ];
-    /**
-     * An array of supported "lib" reference file names used to determine the order for inclusion
-     * when referenced, as well as for spelling suggestions. This ensures the correct ordering for
-     * overload resolution when a type declared in one lib is extended by another.
-     */
-    /* @internal */
-    ts.libs = libEntries.map(function (entry) { return entry[0]; });
-    /**
-     * A map of lib names to lib files. This map is used both for parsing the "lib" command line
-     * option as well as for resolving lib reference directives.
-     */
-    /* @internal */
-    ts.libMap = ts.createMapFromEntries(libEntries);
-    // Watch related options
-    /* @internal */
-    ts.optionsForWatch = [
-        {
-            name: "watchFile",
-            type: ts.createMapFromTemplate({
-                fixedpollinginterval: ts.WatchFileKind.FixedPollingInterval,
-                prioritypollinginterval: ts.WatchFileKind.PriorityPollingInterval,
-                dynamicprioritypolling: ts.WatchFileKind.DynamicPriorityPolling,
-                usefsevents: ts.WatchFileKind.UseFsEvents,
-                usefseventsonparentdirectory: ts.WatchFileKind.UseFsEventsOnParentDirectory,
-            }),
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Specify_strategy_for_watching_file_Colon_FixedPollingInterval_default_PriorityPollingInterval_DynamicPriorityPolling_UseFsEvents_UseFsEventsOnParentDirectory,
-        },
-        {
-            name: "watchDirectory",
-            type: ts.createMapFromTemplate({
-                usefsevents: ts.WatchDirectoryKind.UseFsEvents,
-                fixedpollinginterval: ts.WatchDirectoryKind.FixedPollingInterval,
-                dynamicprioritypolling: ts.WatchDirectoryKind.DynamicPriorityPolling,
-            }),
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Specify_strategy_for_watching_directory_on_platforms_that_don_t_support_recursive_watching_natively_Colon_UseFsEvents_default_FixedPollingInterval_DynamicPriorityPolling,
-        },
-        {
-            name: "fallbackPolling",
-            type: ts.createMapFromTemplate({
-                fixedinterval: ts.PollingWatchKind.FixedInterval,
-                priorityinterval: ts.PollingWatchKind.PriorityInterval,
-                dynamicpriority: ts.PollingWatchKind.DynamicPriority,
-            }),
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Specify_strategy_for_creating_a_polling_watch_when_it_fails_to_create_using_file_system_events_Colon_FixedInterval_default_PriorityInterval_DynamicPriority,
-        },
-        {
-            name: "synchronousWatchDirectory",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Synchronously_call_callbacks_and_update_the_state_of_directory_watchers_on_platforms_that_don_t_support_recursive_watching_natively,
-        },
-    ];
-    /* @internal */
-    ts.commonOptionsWithBuild = [
-        {
-            name: "help",
-            shortName: "h",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Print_this_message,
-        },
-        {
-            name: "help",
-            shortName: "?",
-            type: "boolean"
-        },
-        {
-            name: "watch",
-            shortName: "w",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Watch_input_files,
-        },
-        {
-            name: "preserveWatchOutput",
-            type: "boolean",
-            showInSimplifiedHelpView: false,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Whether_to_keep_outdated_console_output_in_watch_mode_instead_of_clearing_the_screen,
-        },
-        {
-            name: "listFiles",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Print_names_of_files_part_of_the_compilation
-        },
-        {
-            name: "listEmittedFiles",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Print_names_of_generated_files_part_of_the_compilation
-        },
-        {
-            name: "pretty",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Stylize_errors_and_messages_using_color_and_context_experimental
-        },
-        {
-            name: "traceResolution",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Enable_tracing_of_the_name_resolution_process
-        },
-        {
-            name: "diagnostics",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Show_diagnostic_information
-        },
-        {
-            name: "extendedDiagnostics",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Show_verbose_diagnostic_information
-        },
-        {
-            name: "generateCpuProfile",
-            type: "string",
-            isFilePath: true,
-            paramType: ts.Diagnostics.FILE_OR_DIRECTORY,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Generates_a_CPU_profile
-        },
-        {
-            name: "incremental",
-            shortName: "i",
-            type: "boolean",
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Enable_incremental_compilation,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "assumeChangesOnlyAffectDirectDependencies",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Have_recompiles_in_incremental_and_watch_assume_that_changes_within_a_file_will_only_affect_files_directly_depending_on_it
-        },
-        {
-            name: "locale",
-            type: "string",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.The_locale_used_when_displaying_messages_to_the_user_e_g_en_us
-        },
-    ];
-    /* @internal */
-    ts.optionDeclarations = __spreadArrays(ts.commonOptionsWithBuild, [
-        {
-            name: "all",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Show_all_compiler_options,
-        },
-        {
-            name: "version",
-            shortName: "v",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Print_the_compiler_s_version,
-        },
-        {
-            name: "init",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Initializes_a_TypeScript_project_and_creates_a_tsconfig_json_file,
-        },
-        {
-            name: "project",
-            shortName: "p",
-            type: "string",
-            isFilePath: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            paramType: ts.Diagnostics.FILE_OR_DIRECTORY,
-            description: ts.Diagnostics.Compile_the_project_given_the_path_to_its_configuration_file_or_to_a_folder_with_a_tsconfig_json,
-        },
-        {
-            name: "build",
-            type: "boolean",
-            shortName: "b",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Build_one_or_more_projects_and_their_dependencies_if_out_of_date
-        },
-        {
-            name: "showConfig",
-            type: "boolean",
-            category: ts.Diagnostics.Command_line_Options,
-            isCommandLineOnly: true,
-            description: ts.Diagnostics.Print_the_final_configuration_instead_of_building
-        },
-        {
-            name: "listFilesOnly",
-            type: "boolean",
-            category: ts.Diagnostics.Command_line_Options,
-            affectsSemanticDiagnostics: true,
-            affectsEmit: true,
-            isCommandLineOnly: true,
-            description: ts.Diagnostics.Print_names_of_files_that_are_part_of_the_compilation_and_then_stop_processing
-        },
-        // Basic
-        {
-            name: "target",
-            shortName: "t",
-            type: ts.createMapFromTemplate({
-                es3: 0 /* ES3 */,
-                es5: 1 /* ES5 */,
-                es6: 2 /* ES2015 */,
-                es2015: 2 /* ES2015 */,
-                es2016: 3 /* ES2016 */,
-                es2017: 4 /* ES2017 */,
-                es2018: 5 /* ES2018 */,
-                es2019: 6 /* ES2019 */,
-                es2020: 7 /* ES2020 */,
-                esnext: 99 /* ESNext */,
-            }),
-            affectsSourceFile: true,
-            affectsModuleResolution: true,
-            affectsEmit: true,
-            paramType: ts.Diagnostics.VERSION,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Specify_ECMAScript_target_version_Colon_ES3_default_ES5_ES2015_ES2016_ES2017_ES2018_ES2019_ES2020_or_ESNEXT,
-        },
-        {
-            name: "module",
-            shortName: "m",
-            type: ts.createMapFromTemplate({
-                none: ts.ModuleKind.None,
-                commonjs: ts.ModuleKind.CommonJS,
-                amd: ts.ModuleKind.AMD,
-                system: ts.ModuleKind.System,
-                umd: ts.ModuleKind.UMD,
-                es6: ts.ModuleKind.ES2015,
-                es2015: ts.ModuleKind.ES2015,
-                es2020: ts.ModuleKind.ES2020,
-                esnext: ts.ModuleKind.ESNext
-            }),
-            affectsModuleResolution: true,
-            affectsEmit: true,
-            paramType: ts.Diagnostics.KIND,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Specify_module_code_generation_Colon_none_commonjs_amd_system_umd_es2015_es2020_or_ESNext,
-        },
-        {
-            name: "lib",
-            type: "list",
-            element: {
-                name: "lib",
-                type: ts.libMap
-            },
-            affectsModuleResolution: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Specify_library_files_to_be_included_in_the_compilation,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "allowJs",
-            type: "boolean",
-            affectsModuleResolution: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Allow_javascript_files_to_be_compiled
-        },
-        {
-            name: "checkJs",
-            type: "boolean",
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Report_errors_in_js_files
-        },
-        {
-            name: "jsx",
-            type: ts.createMapFromTemplate({
-                "preserve": 1 /* Preserve */,
-                "react-native": 3 /* ReactNative */,
-                "react": 2 /* React */
-            }),
-            affectsSourceFile: true,
-            paramType: ts.Diagnostics.KIND,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Specify_JSX_code_generation_Colon_preserve_react_native_or_react,
-        },
-        {
-            name: "declaration",
-            shortName: "d",
-            type: "boolean",
-            affectsEmit: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Generates_corresponding_d_ts_file,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "declarationMap",
-            type: "boolean",
-            affectsEmit: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Generates_a_sourcemap_for_each_corresponding_d_ts_file,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "emitDeclarationOnly",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Only_emit_d_ts_declaration_files,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "sourceMap",
-            type: "boolean",
-            affectsEmit: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Generates_corresponding_map_file,
-        },
-        {
-            name: "outFile",
-            type: "string",
-            affectsEmit: true,
-            isFilePath: true,
-            paramType: ts.Diagnostics.FILE,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Concatenate_and_emit_output_to_single_file,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "outDir",
-            type: "string",
-            affectsEmit: true,
-            isFilePath: true,
-            paramType: ts.Diagnostics.DIRECTORY,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Redirect_output_structure_to_the_directory,
-        },
-        {
-            name: "rootDir",
-            type: "string",
-            affectsEmit: true,
-            isFilePath: true,
-            paramType: ts.Diagnostics.LOCATION,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Specify_the_root_directory_of_input_files_Use_to_control_the_output_directory_structure_with_outDir,
-        },
-        {
-            name: "composite",
-            type: "boolean",
-            affectsEmit: true,
-            isTSConfigOnly: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Enable_project_compilation,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "tsBuildInfoFile",
-            type: "string",
-            affectsEmit: true,
-            isFilePath: true,
-            paramType: ts.Diagnostics.FILE,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Specify_file_to_store_incremental_compilation_information,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "removeComments",
-            type: "boolean",
-            affectsEmit: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Do_not_emit_comments_to_output,
-        },
-        {
-            name: "noEmit",
-            type: "boolean",
-            affectsEmit: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Do_not_emit_outputs,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "importHelpers",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Import_emit_helpers_from_tslib
-        },
-        {
-            name: "importsNotUsedAsValues",
-            type: ts.createMapFromTemplate({
-                remove: 0 /* Remove */,
-                preserve: 1 /* Preserve */,
-                error: 2 /* Error */
-            }),
-            affectsEmit: true,
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Specify_emit_Slashchecking_behavior_for_imports_that_are_only_used_for_types
-        },
-        {
-            name: "downlevelIteration",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Provide_full_support_for_iterables_in_for_of_spread_and_destructuring_when_targeting_ES5_or_ES3
-        },
-        {
-            name: "isolatedModules",
-            type: "boolean",
-            category: ts.Diagnostics.Basic_Options,
-            description: ts.Diagnostics.Transpile_each_file_as_a_separate_module_similar_to_ts_transpileModule,
-            transpileOptionValue: true
-        },
-        // Strict Type Checks
-        {
-            name: "strict",
-            type: "boolean",
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Enable_all_strict_type_checking_options
-        },
-        {
-            name: "noImplicitAny",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Raise_error_on_expressions_and_declarations_with_an_implied_any_type
-        },
-        {
-            name: "strictNullChecks",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Enable_strict_null_checks
-        },
-        {
-            name: "strictFunctionTypes",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Enable_strict_checking_of_function_types
-        },
-        {
-            name: "strictBindCallApply",
-            type: "boolean",
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Enable_strict_bind_call_and_apply_methods_on_functions
-        },
-        {
-            name: "strictPropertyInitialization",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Enable_strict_checking_of_property_initialization_in_classes
-        },
-        {
-            name: "noImplicitThis",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Raise_error_on_this_expressions_with_an_implied_any_type,
-        },
-        {
-            name: "alwaysStrict",
-            type: "boolean",
-            affectsSourceFile: true,
-            strictFlag: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Strict_Type_Checking_Options,
-            description: ts.Diagnostics.Parse_in_strict_mode_and_emit_use_strict_for_each_source_file
-        },
-        // Additional Checks
-        {
-            name: "noUnusedLocals",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Additional_Checks,
-            description: ts.Diagnostics.Report_errors_on_unused_locals,
-        },
-        {
-            name: "noUnusedParameters",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Additional_Checks,
-            description: ts.Diagnostics.Report_errors_on_unused_parameters,
-        },
-        {
-            name: "noImplicitReturns",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Additional_Checks,
-            description: ts.Diagnostics.Report_error_when_not_all_code_paths_in_function_return_a_value
-        },
-        {
-            name: "noFallthroughCasesInSwitch",
-            type: "boolean",
-            affectsBindDiagnostics: true,
-            affectsSemanticDiagnostics: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Additional_Checks,
-            description: ts.Diagnostics.Report_errors_for_fallthrough_cases_in_switch_statement
-        },
-        // Module Resolution
-        {
-            name: "moduleResolution",
-            type: ts.createMapFromTemplate({
-                node: ts.ModuleResolutionKind.NodeJs,
-                classic: ts.ModuleResolutionKind.Classic,
-            }),
-            affectsModuleResolution: true,
-            paramType: ts.Diagnostics.STRATEGY,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Specify_module_resolution_strategy_Colon_node_Node_js_or_classic_TypeScript_pre_1_6,
-        },
-        {
-            name: "baseUrl",
-            type: "string",
-            affectsModuleResolution: true,
-            isFilePath: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Base_directory_to_resolve_non_absolute_module_names
-        },
-        {
-            // this option can only be specified in tsconfig.json
-            // use type = object to copy the value as-is
-            name: "paths",
-            type: "object",
-            affectsModuleResolution: true,
-            isTSConfigOnly: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.A_series_of_entries_which_re_map_imports_to_lookup_locations_relative_to_the_baseUrl,
-            transpileOptionValue: undefined
-        },
-        {
-            // this option can only be specified in tsconfig.json
-            // use type = object to copy the value as-is
-            name: "rootDirs",
-            type: "list",
-            isTSConfigOnly: true,
-            element: {
-                name: "rootDirs",
-                type: "string",
-                isFilePath: true
-            },
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.List_of_root_folders_whose_combined_content_represents_the_structure_of_the_project_at_runtime,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "typeRoots",
-            type: "list",
-            element: {
-                name: "typeRoots",
-                type: "string",
-                isFilePath: true
-            },
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.List_of_folders_to_include_type_definitions_from
-        },
-        {
-            name: "types",
-            type: "list",
-            element: {
-                name: "types",
-                type: "string"
-            },
-            affectsModuleResolution: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Type_declaration_files_to_be_included_in_compilation,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "allowSyntheticDefaultImports",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Allow_default_imports_from_modules_with_no_default_export_This_does_not_affect_code_emit_just_typechecking
-        },
-        {
-            name: "esModuleInterop",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            affectsEmit: true,
-            showInSimplifiedHelpView: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Enables_emit_interoperability_between_CommonJS_and_ES_Modules_via_creation_of_namespace_objects_for_all_imports_Implies_allowSyntheticDefaultImports
-        },
-        {
-            name: "preserveSymlinks",
-            type: "boolean",
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Do_not_resolve_the_real_path_of_symlinks,
-        },
-        {
-            name: "allowUmdGlobalAccess",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Module_Resolution_Options,
-            description: ts.Diagnostics.Allow_accessing_UMD_globals_from_modules,
-        },
-        // Source Maps
-        {
-            name: "sourceRoot",
-            type: "string",
-            affectsEmit: true,
-            paramType: ts.Diagnostics.LOCATION,
-            category: ts.Diagnostics.Source_Map_Options,
-            description: ts.Diagnostics.Specify_the_location_where_debugger_should_locate_TypeScript_files_instead_of_source_locations,
-        },
-        {
-            name: "mapRoot",
-            type: "string",
-            affectsEmit: true,
-            paramType: ts.Diagnostics.LOCATION,
-            category: ts.Diagnostics.Source_Map_Options,
-            description: ts.Diagnostics.Specify_the_location_where_debugger_should_locate_map_files_instead_of_generated_locations,
-        },
-        {
-            name: "inlineSourceMap",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Source_Map_Options,
-            description: ts.Diagnostics.Emit_a_single_file_with_source_maps_instead_of_having_a_separate_file
-        },
-        {
-            name: "inlineSources",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Source_Map_Options,
-            description: ts.Diagnostics.Emit_the_source_alongside_the_sourcemaps_within_a_single_file_requires_inlineSourceMap_or_sourceMap_to_be_set
-        },
-        // Experimental
-        {
-            name: "experimentalDecorators",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Experimental_Options,
-            description: ts.Diagnostics.Enables_experimental_support_for_ES7_decorators
-        },
-        {
-            name: "emitDecoratorMetadata",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            affectsEmit: true,
-            category: ts.Diagnostics.Experimental_Options,
-            description: ts.Diagnostics.Enables_experimental_support_for_emitting_type_metadata_for_decorators
-        },
-        // Advanced
-        {
-            name: "jsxFactory",
-            type: "string",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Specify_the_JSX_factory_function_to_use_when_targeting_react_JSX_emit_e_g_React_createElement_or_h
-        },
-        {
-            name: "resolveJsonModule",
-            type: "boolean",
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Include_modules_imported_with_json_extension
-        },
-        {
-            name: "out",
-            type: "string",
-            affectsEmit: true,
-            isFilePath: false,
-            // for correct behaviour, please use outFile
-            category: ts.Diagnostics.Advanced_Options,
-            paramType: ts.Diagnostics.FILE,
-            description: ts.Diagnostics.Deprecated_Use_outFile_instead_Concatenate_and_emit_output_to_single_file,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "reactNamespace",
-            type: "string",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Deprecated_Use_jsxFactory_instead_Specify_the_object_invoked_for_createElement_when_targeting_react_JSX_emit
-        },
-        {
-            name: "skipDefaultLibCheck",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Deprecated_Use_skipLibCheck_instead_Skip_type_checking_of_default_library_declaration_files
-        },
-        {
-            name: "charset",
-            type: "string",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.The_character_set_of_the_input_files
-        },
-        {
-            name: "emitBOM",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Emit_a_UTF_8_Byte_Order_Mark_BOM_in_the_beginning_of_output_files
-        },
-        {
-            name: "newLine",
-            type: ts.createMapFromTemplate({
-                crlf: 0 /* CarriageReturnLineFeed */,
-                lf: 1 /* LineFeed */
-            }),
-            affectsEmit: true,
-            paramType: ts.Diagnostics.NEWLINE,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Specify_the_end_of_line_sequence_to_be_used_when_emitting_files_Colon_CRLF_dos_or_LF_unix,
-        },
-        {
-            name: "noErrorTruncation",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_truncate_error_messages
-        },
-        {
-            name: "noLib",
-            type: "boolean",
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_include_the_default_library_file_lib_d_ts,
-            // We are not returning a sourceFile for lib file when asked by the program,
-            // so pass --noLib to avoid reporting a file not found error.
-            transpileOptionValue: true
-        },
-        {
-            name: "noResolve",
-            type: "boolean",
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_add_triple_slash_references_or_imported_modules_to_the_list_of_compiled_files,
-            // We are not doing a full typecheck, we are not resolving the whole context,
-            // so pass --noResolve to avoid reporting missing file errors.
-            transpileOptionValue: true
-        },
-        {
-            name: "stripInternal",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_emit_declarations_for_code_that_has_an_internal_annotation,
-        },
-        {
-            name: "disableSizeLimit",
-            type: "boolean",
-            affectsSourceFile: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Disable_size_limitations_on_JavaScript_projects
-        },
-        {
-            name: "disableSourceOfProjectReferenceRedirect",
-            type: "boolean",
-            isTSConfigOnly: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Disable_use_of_source_files_instead_of_declaration_files_from_referenced_projects
-        },
-        {
-            name: "disableSolutionSearching",
-            type: "boolean",
-            isTSConfigOnly: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Disable_solution_searching_for_this_project
-        },
-        {
-            name: "noImplicitUseStrict",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_emit_use_strict_directives_in_module_output
-        },
-        {
-            name: "noEmitHelpers",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_generate_custom_helper_functions_like_extends_in_compiled_output
-        },
-        {
-            name: "noEmitOnError",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_emit_outputs_if_any_errors_were_reported,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "preserveConstEnums",
-            type: "boolean",
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_erase_const_enum_declarations_in_generated_code
-        },
-        {
-            name: "declarationDir",
-            type: "string",
-            affectsEmit: true,
-            isFilePath: true,
-            paramType: ts.Diagnostics.DIRECTORY,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Output_directory_for_generated_declaration_files,
-            transpileOptionValue: undefined
-        },
-        {
-            name: "skipLibCheck",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Skip_type_checking_of_declaration_files,
-        },
-        {
-            name: "allowUnusedLabels",
-            type: "boolean",
-            affectsBindDiagnostics: true,
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_report_errors_on_unused_labels
-        },
-        {
-            name: "allowUnreachableCode",
-            type: "boolean",
-            affectsBindDiagnostics: true,
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Do_not_report_errors_on_unreachable_code
-        },
-        {
-            name: "suppressExcessPropertyErrors",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Suppress_excess_property_checks_for_object_literals,
-        },
-        {
-            name: "suppressImplicitAnyIndexErrors",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Suppress_noImplicitAny_errors_for_indexing_objects_lacking_index_signatures,
-        },
-        {
-            name: "forceConsistentCasingInFileNames",
-            type: "boolean",
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Disallow_inconsistently_cased_references_to_the_same_file
-        },
-        {
-            name: "maxNodeModuleJsDepth",
-            type: "number",
-            affectsModuleResolution: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.The_maximum_dependency_depth_to_search_under_node_modules_and_load_JavaScript_files
-        },
-        {
-            name: "noStrictGenericChecks",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Disable_strict_checking_of_generic_signatures_in_function_types,
-        },
-        {
-            name: "useDefineForClassFields",
-            type: "boolean",
-            affectsSemanticDiagnostics: true,
-            affectsEmit: true,
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Emit_class_fields_with_Define_instead_of_Set,
-        },
-        {
-            name: "keyofStringsOnly",
-            type: "boolean",
-            category: ts.Diagnostics.Advanced_Options,
-            description: ts.Diagnostics.Resolve_keyof_to_string_valued_property_names_only_no_numbers_or_symbols,
-        },
-        {
-            // A list of plugins to load in the language service
-            name: "plugins",
-            type: "list",
-            isTSConfigOnly: true,
-            element: {
-                name: "plugin",
-                type: "object"
-            },
-            description: ts.Diagnostics.List_of_language_service_plugins
-        },
-    ]);
-    /* @internal */
-    ts.semanticDiagnosticsOptionDeclarations = ts.optionDeclarations.filter(function (option) { return !!option.affectsSemanticDiagnostics; });
-    /* @internal */
-    ts.affectsEmitOptionDeclarations = ts.optionDeclarations.filter(function (option) { return !!option.affectsEmit; });
-    /* @internal */
-    ts.moduleResolutionOptionDeclarations = ts.optionDeclarations.filter(function (option) { return !!option.affectsModuleResolution; });
-    /* @internal */
-    ts.sourceFileAffectingCompilerOptions = ts.optionDeclarations.filter(function (option) {
-        return !!option.affectsSourceFile || !!option.affectsModuleResolution || !!option.affectsBindDiagnostics;
-    });
-    /* @internal */
-    ts.transpileOptionValueCompilerOptions = ts.optionDeclarations.filter(function (option) {
-        return ts.hasProperty(option, "transpileOptionValue");
-    });
-    /* @internal */
-    ts.buildOpts = __spreadArrays(ts.commonOptionsWithBuild, [
-        {
-            name: "verbose",
-            shortName: "v",
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Enable_verbose_logging,
-            type: "boolean"
-        },
-        {
-            name: "dry",
-            shortName: "d",
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Show_what_would_be_built_or_deleted_if_specified_with_clean,
-            type: "boolean"
-        },
-        {
-            name: "force",
-            shortName: "f",
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Build_all_projects_including_those_that_appear_to_be_up_to_date,
-            type: "boolean"
-        },
-        {
-            name: "clean",
-            category: ts.Diagnostics.Command_line_Options,
-            description: ts.Diagnostics.Delete_the_outputs_of_all_projects,
-            type: "boolean"
+        function parsePrivateIdentifier() {
+            var pos = getNodePos();
+            var node = factory.createPrivateIdentifier(internPrivateIdentifier(scanner.getTokenText()));
+            nextToken();
+            return finishNode(node, pos);
+        }
+        function parseContextualModifier(t) {
+            return token() === t && tryParse(nextTokenCanFollowModifier);
+        }
+        function nextTokenIsOnSameLineAndCanFollowModifier() {
+            nextToken();
+            if (scanner.hasPrecedingLineBreak()) {
+                return false;
+            }
+            return canFollowModifier();
+        }
+        function nextTokenCanFollowModifier() {
+            switch (token()) {
+                case 84 /* ConstKeyword */:
+                    // 'const' is only a modifier if followed by 'enum'.
+                    return nextToken() === 91 /* EnumKeyword */;
+                case 92 /* ExportKeyword */:
+                    nextToken();
+                    if (token() === 87 /* DefaultKeyword */) {
+                        return lookAhead(nextTokenCanFollowDefaultKeyword);
+                    }
+                    if (token() === 149 /* TypeKeyword */) {
+                        return lookAhead(nextTokenCanFollowExportModifier);
+                    }
+                    return canFollowExportModifier();
+                case 87 /* DefaultKeyword */:
+                    return nextTokenCanFollowDefaultKeyword();
+                case 123 /* StaticKeyword */:
+                case 134 /* GetKeyword */:
+                case 146 /* SetKeyword */:
+                    nextToken();
+                    return canFollowModifier();
+                default:
+                    return nextTokenIsOnSameLineAndCanFollowModifier();
+            }
+        }
+        function canFollowExportModifier() {
+            return token() !== 41 /* AsteriskToken */
+                && token() !== 126 /* AsKeyword */
+                && token() !== 18 /* OpenBraceToken */
+                && canFollowModifier();
+        }
+        function nextTokenCanFollowExportModifier() {
+            nextToken();
+            return canFollowExportModifier();
+        }
+        function parseAnyContextualModifier() {
+            return ts.isModifierKind(token()) && tryParse(nextTokenCanFollowModifier);
+        }
+        function canFollowModifier() {
+            return token() === 22 /* OpenBracketToken */
+                || token() === 18 /* OpenBraceToken */
+                || token() === 41 /* AsteriskToken */
+                || token() === 25 /* DotDotDotToken */
+                || isLiteralPropertyName();
+        }
+        function nextTokenCanFollowDefaultKeyword() {
+            nextToken();
+            return token() === 83 /* ClassKeyword */ || token() === 97 /* FunctionKeyword */ ||
+                token() === 117 /* InterfaceKeyword */ ||
+                (token() === 125 /* AbstractKeyword */ && lookAhead(nextTokenIsClassKeywordOnSameLine)) ||
+                (token() === 129 /* AsyncKeyword */ && lookAhead(nextTokenIsFunctionKeywordOnSameLine));
+        }
+        // True if positioned at the start of a list element
+        function isListElement(parsingContext, inErrorRecovery) {
+            var node = currentNode(parsingContext);
+            if (node) {
+                return true;
+            }
+            switch (parsingContext) {
+                case 0 /* SourceElements */:
+                case 1 /* BlockStatements */:
+                case 3 /* SwitchClauseStatements */:
+                    // If we're in error recovery, then we don't want to treat ';' as an empty statement.
+                    // The problem is that ';' can show up in far too many contexts, and if we see one
+                    // and assume it's a statement, then we may bail out inappropriately from whatever
+                    // we're parsing.  For example, if we have a semicolon in the middle of a class, then
+                    // we really don't want to assume the class is over and we're on a statement in the
+                    // outer module.  We just want to consume and move on.
+                    return !(token() === 26 /* SemicolonToken */ && inErrorRecovery) && isStartOfStatement();
+                case 2 /* SwitchClauses */:
+                    return token() === 81 /* CaseKeyword */ || token() === 87 /* DefaultKeyword */;
+                case 4 /* TypeMembers */:
+                    return lookAhead(isTypeMemberStart);
+                case 5 /* ClassMembers */:
+                    // We allow semicolons as class elements (as specified by ES6) as long as we're
+                    // not in error recovery.  If we're in error recovery, we don't want an errant
+                    // semicolon to be treated as a class member (since they're almost always used
+                    // for statements.
+                    return lookAhead(isClassMemberStart) || (token() === 26 /* SemicolonToken */ && !inErrorRecovery);
+                case 6 /* EnumMembers */:
+                    // Include open bracket computed properties. This technically also lets in indexers,
+                    // which would be a candidate for improved error reporting.
+                    return token() === 22 /* OpenBracketToken */ || isLiteralPropertyName();
+                case 12 /* ObjectLiteralMembers */:
+                    switch (token()) {
+                        case 22 /* OpenBracketToken */:
+                        case 41 /* AsteriskToken */:
+                        case 25 /* DotDotDotToken */:
+                        case 24 /* DotToken */: // Not an object literal member, but don't want to close the object (see `tests/cases/fourslash/completionsDotInObjectLiteral.ts`)
+                            return true;
+                        default:
+                            return isLiteralPropertyName();
+                    }
+                case 18 /* RestProperties */:
+                    return isLiteralPropertyName();
+                case 9 /* ObjectBindingElements */:
+                    return token() === 22 /* OpenBracketToken */ || token() === 25 /* DotDotDotToken */ || isLiteralPropertyName();
+                case 7 /* HeritageClauseElement */:
+                    // If we see `{ ... }` then only consume it as an expression if it is followed by `,` or `{`
+                    // That way we won't consume the body of a class in its heritage clause.
+                    if (token() === 18 /* OpenBraceToken */) {
+                        return lookAhead(isValidHeritageClauseObjectLiteral);
+                    }
+                    if (!inErrorRecovery) {
+                        return isStartOfLeftHandSideExpression() && !isHeritageClauseExtendsOrImplementsKeyword();
+                    }
+                    else {
+                        // If we're in error recovery we tighten up what we're willing to match.
+                        // That way we don't treat something like "this" as a valid heritage clause
+                        // element during recovery.
+                        return isIdentifier() && !isHeritageClauseExtendsOrImplementsKeyword();
+                    }
+                case 8 /* VariableDeclarations */:
+                    return isBindingIdentifierOrPrivateIdentifierOrPattern();
+                case 10 /* ArrayBindingElements */:
+                    return token() === 27 /* CommaToken */ || token() === 25 /* DotDotDotToken */ || isBindingIdentifierOrPrivateIdentifierOrPattern();
+                case 19 /* TypeParameters */:
+                    return isIdentifier();
+                case 15 /* ArrayLiteralMembers */:
+                    switch (token()) {
+                        case 27 /* CommaToken */:
+                        case 24 /* DotToken */: // Not an array literal member, but don't want to close the array (see `tests/cases/fourslash/completionsDotInArrayLiteralInObjectLiteral.ts`)
+                            return true;
+                    }
+                // falls through
+                case 11 /* ArgumentExpressions */:
+                    return token() === 25 /* DotDotDotToken */ || isStartOfExpression();
+                case 16 /* Parameters */:
+                    return isStartOfParameter(/*isJSDocParameter*/ false);
+                case 17 /* JSDocParameters */:
+                    return isStartOfParameter(/*isJSDocParameter*/ true);
+                case 20 /* TypeArguments */:
+                case 21 /* TupleElementTypes */:
+                    return token() === 27 /* CommaToken */ || isStartOfType();
+                case 22 /* HeritageClauses */:
+                    return isHeritageClause();
+                case 23 /* ImportOrExportSpecifiers */:
+                    return ts.tokenIsIdentifierOrKeyword(token());
+                case 13 /* JsxAttributes */:
+                    return ts.tokenIsIdentifierOrKeyword(token()) || token() === 18 /* OpenBraceToken */;
+                case 14 /* JsxChildren */:
+                    return true;
+            }
+            return ts.Debug.fail("Non-exhaustive case in 'isListElement'.");
+        }
+        function isValidHeritageClauseObjectLiteral() {
+            ts.Debug.assert(token() === 18 /* OpenBraceToken */);
+            if (nextToken() === 19 /* CloseBraceToken */) {
+                // if we see "extends {}" then only treat the {} as what we're extending (and not
+                // the class body) if we have:
+                //
+                //      extends {} {
+                //      extends {},
+                //      extends {} extends
+                //      extends {} implements
+                var next = nextToken();
+                return next === 27 /* CommaToken */ || next === 18 /* OpenBraceToken */ || next === 93 /* ExtendsKeyword */ || next === 116 /* ImplementsKeyword */;
+            }
+            return true;
+        }
+        function nextTokenIsIdentifier() {
+            nextToken();
+            return isIdentifier();
+        }
+        function nextTokenIsIdentifierOrKeyword() {
+            nextToken();
+            return ts.tokenIsIdentifierOrKeyword(token());
+        }
+        function nextTokenIsIdentifierOrKeywordOrGreaterThan() {
+            nextToken();
+            return ts.tokenIsIdentifierOrKeywordOrGreaterThan(token());
+        }
+        function isHeritageClauseExtendsOrImplementsKeyword() {
+            if (token() === 116 /* ImplementsKeyword */ ||
+                token() === 93 /* ExtendsKeyword */) {
+                return lookAhead(nextTokenIsStartOfExpression);
+            }
+            return false;
+        }
+        function nextTokenIsStartOfExpression() {
+            nextToken();
+            return isStartOfExpression();
+        }
+        function nextTokenIsStartOfType() {
+            nextToken();
+            return isStartOfType();
+        }
+        // True if positioned at a list terminator
+        function isListTerminator(kind) {
+            if (token() === 1 /* EndOfFileToken */) {
+                // Being at the end of the file ends all lists.
+                return true;
+            }
+            switch (kind) {
+                case 1 /* BlockStatements */:
+                case 2 /* SwitchClauses */:
+                case 4 /* TypeMembers */:
+                case 5 /* ClassMembers */:
+                case 6 /* EnumMembers */:
+                case 12 /* ObjectLiteralMembers */:
+                case 9 /* ObjectBindingElements */:
+                case 23 /* ImportOrExportSpecifiers */:
+                    return token() === 19 /* CloseBraceToken */;
+                case 3 /* SwitchClauseStatements */:
+                    return token() === 19 /* CloseBraceToken */ || token() === 81 /* CaseKeyword */ || token() === 87 /* DefaultKeyword */;
+                case 7 /* HeritageClauseElement */:
+                    return token() === 18 /* OpenBraceToken */ || token() === 93 /* ExtendsKeyword */ || token() === 116 /* ImplementsKeyword */;
+                case 8 /* VariableDeclarations */:
+                    return isVariableDeclaratorListTerminator();
+                case 19 /* TypeParameters */:
+                    // Tokens other than '>' are here for better error recovery
+                    return token() === 31 /* GreaterThanToken */ || token() === 20 /* OpenParenToken */ || token() === 18 /* OpenBraceToken */ || token() === 93 /* ExtendsKeyword */ || token() === 116 /* ImplementsKeyword */;
+                case 11 /* ArgumentExpressions */:
+                    // Tokens other than ')' are here for better error recovery
+                    return token() === 21 /* CloseParenToken */ || token() === 26 /* SemicolonToken */;
+                case 15 /* ArrayLiteralMembers */:
+                case 21 /* TupleElementTypes */:
+                case 10 /* ArrayBindingElements */:
+                    return token() === 23 /* CloseBracketToken */;
+                case 17 /* JSDocParameters */:
+                case 16 /* Parameters */:
+                case 18 /* RestProperties */:
+                    // Tokens other than ')' and ']' (the latter for index signatures) are here for better error recovery
+                    return token() === 21 /* CloseParenToken */ || token() === 23 /* CloseBracketToken */ /*|| token === SyntaxKind.OpenBraceToken*/;
+                case 20 /* TypeArguments */:
+                    // All other tokens should cause the type-argument to terminate except comma token
+                    return token() !== 27 /* CommaToken */;
+                case 22 /* HeritageClauses */:
+                    return token() === 18 /* OpenBraceToken */ || token() === 19 /* CloseBraceToken */;
+                case 13 /* JsxAttributes */:
+                    return token() === 31 /* GreaterThanToken */ || token() === 43 /* SlashToken */;
+                case 14 /* JsxChildren */:
+                    return token() === 29 /* LessThanToken */ && lookAhead(nextTokenIsSlash);
+                default:
+                    return false;
+            }
+        }
+        function isVariableDeclaratorListTerminator() {
+            // If we can consume a semicolon (either explicitly, or with ASI), then consider us done
+            // with parsing the list of variable declarators.
+            if (canParseSemicolon()) {
+                return true;
+            }
+            // in the case where we're parsing the variable declarator of a 'for-in' statement, we
+            // are done if we see an 'in' keyword in front of us. Same with for-of
+            if (isInOrOfKeyword(token())) {
+                return true;
+            }
+            // ERROR RECOVERY TWEAK:
+            // For better error recovery, if we see an '=>' then we just stop immediately.  We've got an
+            // arrow function here and it's going to be very unlikely that we'll resynchronize and get
+            // another variable declaration.
+            if (token() === 38 /* EqualsGreaterThanToken */) {
+                return true;
+            }
+            // Keep trying to parse out variable declarators.
+            return false;
+        }
+        // True if positioned at element or terminator of the current list or any enclosing list
+        function isInSomeParsingContext() {
+            for (var kind = 0; kind < 24 /* Count */; kind++) {
+                if (parsingContext & (1 << kind)) {
+                    if (isListElement(kind, /*inErrorRecovery*/ true) || isListTerminator(kind)) {
+                        return true;
+                    }
+                }
+            }
+            return false;
+        }
+        // Parses a list of elements
+        function parseList(kind, parseElement) {
+            var saveParsingContext = parsingContext;
+            parsingContext |= 1 << kind;
+            var list = [];
+            var listPos = getNodePos();
+            while (!isListTerminator(kind)) {
+                if (isListElement(kind, /*inErrorRecovery*/ false)) {
+                    var element = parseListElement(kind, parseElement);
+                    list.push(element);
+                    continue;
+                }
+                if (abortParsingListOrMoveToNextToken(kind)) {
+                    break;
+                }
+            }
+            parsingContext = saveParsingContext;
+            return createNodeArray(list, listPos);
+        }
+        function parseListElement(parsingContext, parseElement) {
+            var node = currentNode(parsingContext);
+            if (node) {
+                return consumeNode(node);
+            }
+            return parseElement();
+        }
+        function currentNode(parsingContext) {
+            // If we don't have a cursor or the parsing context isn't reusable, there's nothing to reuse.
+            //
+            // If there is an outstanding parse error that we've encountered, but not attached to
+            // some node, then we cannot get a node from the old source tree.  This is because we
+            // want to mark the next node we encounter as being unusable.
+            //
+            // Note: This may be too conservative.  Perhaps we could reuse the node and set the bit
+            // on it (or its leftmost child) as having the error.  For now though, being conservative
+            // is nice and likely won't ever affect perf.
+            if (!syntaxCursor || !isReusableParsingContext(parsingContext) || parseErrorBeforeNextFinishedNode) {
+                return undefined;
+            }
+            var node = syntaxCursor.currentNode(scanner.getStartPos());
+            // Can't reuse a missing node.
+            // Can't reuse a node that intersected the change range.
+            // Can't reuse a node that contains a parse error.  This is necessary so that we
+            // produce the same set of errors again.
+            if (ts.nodeIsMissing(node) || node.intersectsChange || ts.containsParseError(node)) {
+                return undefined;
+            }
+            // We can only reuse a node if it was parsed under the same strict mode that we're
+            // currently in.  i.e. if we originally parsed a node in non-strict mode, but then
+            // the user added 'using strict' at the top of the file, then we can't use that node
+            // again as the presence of strict mode may cause us to parse the tokens in the file
+            // differently.
+            //
+            // Note: we *can* reuse tokens when the strict mode changes.  That's because tokens
+            // are unaffected by strict mode.  It's just the parser will decide what to do with it
+            // differently depending on what mode it is in.
+            //
+            // This also applies to all our other context flags as well.
+            var nodeContextFlags = node.flags & 25358336 /* ContextFlags */;
+            if (nodeContextFlags !== contextFlags) {
+                return undefined;
+            }
+            // Ok, we have a node that looks like it could be reused.  Now verify that it is valid
+            // in the current list parsing context that we're currently at.
+            if (!canReuseNode(node, parsingContext)) {
+                return undefined;
+            }
+            if (node.jsDocCache) {
+                // jsDocCache may include tags from parent nodes, which might have been modified.
+                node.jsDocCache = undefined;
+            }
+            return node;
+        }
+        function consumeNode(node) {
+            // Move the scanner so it is after the node we just consumed.
+            scanner.setTextPos(node.end);
+            nextToken();
+            return node;
+        }
+        function isReusableParsingContext(parsingContext) {
+            switch (parsingContext) {
+                case 5 /* ClassMembers */:
+                case 2 /* SwitchClauses */:
+                case 0 /* SourceElements */:
+                case 1 /* BlockStatements */:
+                case 3 /* SwitchClauseStatements */:
+                case 6 /* EnumMembers */:
+                case 4 /* TypeMembers */:
+                case 8 /* VariableDeclarations */:
+                case 17 /* JSDocParameters */:
+                case 16 /* Parameters */:
+                    return true;
+            }
+            return false;
+        }
+        function canReuseNode(node, parsingContext) {
+            switch (parsingContext) {
+                case 5 /* ClassMembers */:
+                    return isReusableClassMember(node);
+                case 2 /* SwitchClauses */:
+                    return isReusableSwitchClause(node);
+                case 0 /* SourceElements */:
+                case 1 /* BlockStatements */:
+                case 3 /* SwitchClauseStatements */:
+                    return isReusableStatement(node);
+                case 6 /* EnumMembers */:
+                    return isReusableEnumMember(node);
+                case 4 /* TypeMembers */:
+                    return isReusableTypeMember(node);
+                case 8 /* VariableDeclarations */:
+                    return isReusableVariableDeclaration(node);
+                case 17 /* JSDocParameters */:
+                case 16 /* Parameters */:
+                    return isReusableParameter(node);
+                // Any other lists we do not care about reusing nodes in.  But feel free to add if
+                // you can do so safely.  Danger areas involve nodes that may involve speculative
+                // parsing.  If speculative parsing is involved with the node, then the range the
+                // parser reached while looking ahead might be in the edited range (see the example
+                // in canReuseVariableDeclaratorNode for a good case of this).
+                // case ParsingContext.HeritageClauses:
+                // This would probably be safe to reuse.  There is no speculative parsing with
+                // heritage clauses.
+                // case ParsingContext.TypeParameters:
+                // This would probably be safe to reuse.  There is no speculative parsing with
+                // type parameters.  Note that that's because type *parameters* only occur in
+                // unambiguous *type* contexts.  While type *arguments* occur in very ambiguous
+                // *expression* contexts.
+                // case ParsingContext.TupleElementTypes:
+                // This would probably be safe to reuse.  There is no speculative parsing with
+                // tuple types.
+                // Technically, type argument list types are probably safe to reuse.  While
+                // speculative parsing is involved with them (since type argument lists are only
+                // produced from speculative parsing a < as a type argument list), we only have
+                // the types because speculative parsing succeeded.  Thus, the lookahead never
+                // went past the end of the list and rewound.
+                // case ParsingContext.TypeArguments:
+                // Note: these are almost certainly not safe to ever reuse.  Expressions commonly
+                // need a large amount of lookahead, and we should not reuse them as they may
+                // have actually intersected the edit.
+                // case ParsingContext.ArgumentExpressions:
+                // This is not safe to reuse for the same reason as the 'AssignmentExpression'
+                // cases.  i.e. a property assignment may end with an expression, and thus might
+                // have lookahead far beyond it's old node.
+                // case ParsingContext.ObjectLiteralMembers:
+                // This is probably not safe to reuse.  There can be speculative parsing with
+                // type names in a heritage clause.  There can be generic names in the type
+                // name list, and there can be left hand side expressions (which can have type
+                // arguments.)
+                // case ParsingContext.HeritageClauseElement:
+                // Perhaps safe to reuse, but it's unlikely we'd see more than a dozen attributes
+                // on any given element. Same for children.
+                // case ParsingContext.JsxAttributes:
+                // case ParsingContext.JsxChildren:
+            }
+            return false;
+        }
+        function isReusableClassMember(node) {
+            if (node) {
+                switch (node.kind) {
+                    case 166 /* Constructor */:
+                    case 171 /* IndexSignature */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 163 /* PropertyDeclaration */:
+                    case 229 /* SemicolonClassElement */:
+                        return true;
+                    case 165 /* MethodDeclaration */:
+                        // Method declarations are not necessarily reusable.  An object-literal
+                        // may have a method calls "constructor(...)" and we must reparse that
+                        // into an actual .ConstructorDeclaration.
+                        var methodDeclaration = node;
+                        var nameIsConstructor = methodDeclaration.name.kind === 78 /* Identifier */ &&
+                            methodDeclaration.name.originalKeywordKind === 132 /* ConstructorKeyword */;
+                        return !nameIsConstructor;
+                }
+            }
+            return false;
+        }
+        function isReusableSwitchClause(node) {
+            if (node) {
+                switch (node.kind) {
+                    case 284 /* CaseClause */:
+                    case 285 /* DefaultClause */:
+                        return true;
+                }
+            }
+            return false;
+        }
+        function isReusableStatement(node) {
+            if (node) {
+                switch (node.kind) {
+                    case 251 /* FunctionDeclaration */:
+                    case 232 /* VariableStatement */:
+                    case 230 /* Block */:
+                    case 234 /* IfStatement */:
+                    case 233 /* ExpressionStatement */:
+                    case 246 /* ThrowStatement */:
+                    case 242 /* ReturnStatement */:
+                    case 244 /* SwitchStatement */:
+                    case 241 /* BreakStatement */:
+                    case 240 /* ContinueStatement */:
+                    case 238 /* ForInStatement */:
+                    case 239 /* ForOfStatement */:
+                    case 237 /* ForStatement */:
+                    case 236 /* WhileStatement */:
+                    case 243 /* WithStatement */:
+                    case 231 /* EmptyStatement */:
+                    case 247 /* TryStatement */:
+                    case 245 /* LabeledStatement */:
+                    case 235 /* DoStatement */:
+                    case 248 /* DebuggerStatement */:
+                    case 261 /* ImportDeclaration */:
+                    case 260 /* ImportEqualsDeclaration */:
+                    case 267 /* ExportDeclaration */:
+                    case 266 /* ExportAssignment */:
+                    case 256 /* ModuleDeclaration */:
+                    case 252 /* ClassDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 255 /* EnumDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                        return true;
+                }
+            }
+            return false;
+        }
+        function isReusableEnumMember(node) {
+            return node.kind === 291 /* EnumMember */;
+        }
+        function isReusableTypeMember(node) {
+            if (node) {
+                switch (node.kind) {
+                    case 170 /* ConstructSignature */:
+                    case 164 /* MethodSignature */:
+                    case 171 /* IndexSignature */:
+                    case 162 /* PropertySignature */:
+                    case 169 /* CallSignature */:
+                        return true;
+                }
+            }
+            return false;
+        }
+        function isReusableVariableDeclaration(node) {
+            if (node.kind !== 249 /* VariableDeclaration */) {
+                return false;
+            }
+            // Very subtle incremental parsing bug.  Consider the following code:
+            //
+            //      let v = new List < A, B
+            //
+            // This is actually legal code.  It's a list of variable declarators "v = new List<A"
+            // on one side and "B" on the other. If you then change that to:
+            //
+            //      let v = new List < A, B >()
+            //
+            // then we have a problem.  "v = new List<A" doesn't intersect the change range, so we
+            // start reparsing at "B" and we completely fail to handle this properly.
+            //
+            // In order to prevent this, we do not allow a variable declarator to be reused if it
+            // has an initializer.
+            var variableDeclarator = node;
+            return variableDeclarator.initializer === undefined;
+        }
+        function isReusableParameter(node) {
+            if (node.kind !== 160 /* Parameter */) {
+                return false;
+            }
+            // See the comment in isReusableVariableDeclaration for why we do this.
+            var parameter = node;
+            return parameter.initializer === undefined;
+        }
+        // Returns true if we should abort parsing.
+        function abortParsingListOrMoveToNextToken(kind) {
+            parsingContextErrors(kind);
+            if (isInSomeParsingContext()) {
+                return true;
+            }
+            nextToken();
+            return false;
+        }
+        function parsingContextErrors(context) {
+            switch (context) {
+                case 0 /* SourceElements */: return parseErrorAtCurrentToken(ts.Diagnostics.Declaration_or_statement_expected);
+                case 1 /* BlockStatements */: return parseErrorAtCurrentToken(ts.Diagnostics.Declaration_or_statement_expected);
+                case 2 /* SwitchClauses */: return parseErrorAtCurrentToken(ts.Diagnostics.case_or_default_expected);
+                case 3 /* SwitchClauseStatements */: return parseErrorAtCurrentToken(ts.Diagnostics.Statement_expected);
+                case 18 /* RestProperties */: // fallthrough
+                case 4 /* TypeMembers */: return parseErrorAtCurrentToken(ts.Diagnostics.Property_or_signature_expected);
+                case 5 /* ClassMembers */: return parseErrorAtCurrentToken(ts.Diagnostics.Unexpected_token_A_constructor_method_accessor_or_property_was_expected);
+                case 6 /* EnumMembers */: return parseErrorAtCurrentToken(ts.Diagnostics.Enum_member_expected);
+                case 7 /* HeritageClauseElement */: return parseErrorAtCurrentToken(ts.Diagnostics.Expression_expected);
+                case 8 /* VariableDeclarations */:
+                    return ts.isKeyword(token())
+                        ? parseErrorAtCurrentToken(ts.Diagnostics._0_is_not_allowed_as_a_variable_declaration_name, ts.tokenToString(token()))
+                        : parseErrorAtCurrentToken(ts.Diagnostics.Variable_declaration_expected);
+                case 9 /* ObjectBindingElements */: return parseErrorAtCurrentToken(ts.Diagnostics.Property_destructuring_pattern_expected);
+                case 10 /* ArrayBindingElements */: return parseErrorAtCurrentToken(ts.Diagnostics.Array_element_destructuring_pattern_expected);
+                case 11 /* ArgumentExpressions */: return parseErrorAtCurrentToken(ts.Diagnostics.Argument_expression_expected);
+                case 12 /* ObjectLiteralMembers */: return parseErrorAtCurrentToken(ts.Diagnostics.Property_assignment_expected);
+                case 15 /* ArrayLiteralMembers */: return parseErrorAtCurrentToken(ts.Diagnostics.Expression_or_comma_expected);
+                case 17 /* JSDocParameters */: return parseErrorAtCurrentToken(ts.Diagnostics.Parameter_declaration_expected);
+                case 16 /* Parameters */: return parseErrorAtCurrentToken(ts.Diagnostics.Parameter_declaration_expected);
+                case 19 /* TypeParameters */: return parseErrorAtCurrentToken(ts.Diagnostics.Type_parameter_declaration_expected);
+                case 20 /* TypeArguments */: return parseErrorAtCurrentToken(ts.Diagnostics.Type_argument_expected);
+                case 21 /* TupleElementTypes */: return parseErrorAtCurrentToken(ts.Diagnostics.Type_expected);
+                case 22 /* HeritageClauses */: return parseErrorAtCurrentToken(ts.Diagnostics.Unexpected_token_expected);
+                case 23 /* ImportOrExportSpecifiers */: return parseErrorAtCurrentToken(ts.Diagnostics.Identifier_expected);
+                case 13 /* JsxAttributes */: return parseErrorAtCurrentToken(ts.Diagnostics.Identifier_expected);
+                case 14 /* JsxChildren */: return parseErrorAtCurrentToken(ts.Diagnostics.Identifier_expected);
+                default: return [undefined]; // TODO: GH#18217 `default: Debug.assertNever(context);`
+            }
+        }
+        // Parses a comma-delimited list of elements
+        function parseDelimitedList(kind, parseElement, considerSemicolonAsDelimiter) {
+            var saveParsingContext = parsingContext;
+            parsingContext |= 1 << kind;
+            var list = [];
+            var listPos = getNodePos();
+            var commaStart = -1; // Meaning the previous token was not a comma
+            while (true) {
+                if (isListElement(kind, /*inErrorRecovery*/ false)) {
+                    var startPos = scanner.getStartPos();
+                    list.push(parseListElement(kind, parseElement));
+                    commaStart = scanner.getTokenPos();
+                    if (parseOptional(27 /* CommaToken */)) {
+                        // No need to check for a zero length node since we know we parsed a comma
+                        continue;
+                    }
+                    commaStart = -1; // Back to the state where the last token was not a comma
+                    if (isListTerminator(kind)) {
+                        break;
+                    }
+                    // We didn't get a comma, and the list wasn't terminated, explicitly parse
+                    // out a comma so we give a good error message.
+                    parseExpected(27 /* CommaToken */, getExpectedCommaDiagnostic(kind));
+                    // If the token was a semicolon, and the caller allows that, then skip it and
+                    // continue.  This ensures we get back on track and don't result in tons of
+                    // parse errors.  For example, this can happen when people do things like use
+                    // a semicolon to delimit object literal members.   Note: we'll have already
+                    // reported an error when we called parseExpected above.
+                    if (considerSemicolonAsDelimiter && token() === 26 /* SemicolonToken */ && !scanner.hasPrecedingLineBreak()) {
+                        nextToken();
+                    }
+                    if (startPos === scanner.getStartPos()) {
+                        // What we're parsing isn't actually remotely recognizable as a element and we've consumed no tokens whatsoever
+                        // Consume a token to advance the parser in some way and avoid an infinite loop
+                        // This can happen when we're speculatively parsing parenthesized expressions which we think may be arrow functions,
+                        // or when a modifier keyword which is disallowed as a parameter name (ie, `static` in strict mode) is supplied
+                        nextToken();
+                    }
+                    continue;
+                }
+                if (isListTerminator(kind)) {
+                    break;
+                }
+                if (abortParsingListOrMoveToNextToken(kind)) {
+                    break;
+                }
+            }
+            parsingContext = saveParsingContext;
+            // Recording the trailing comma is deliberately done after the previous
+            // loop, and not just if we see a list terminator. This is because the list
+            // may have ended incorrectly, but it is still important to know if there
+            // was a trailing comma.
+            // Check if the last token was a comma.
+            // Always preserve a trailing comma by marking it on the NodeArray
+            return createNodeArray(list, listPos, /*end*/ undefined, commaStart >= 0);
+        }
+        function getExpectedCommaDiagnostic(kind) {
+            return kind === 6 /* EnumMembers */ ? ts.Diagnostics.An_enum_member_name_must_be_followed_by_a_or : undefined;
+        }
+        function createMissingList() {
+            var list = createNodeArray([], getNodePos());
+            list.isMissingList = true;
+            return list;
+        }
+        function isMissingList(arr) {
+            return !!arr.isMissingList;
+        }
+        function parseBracketedList(kind, parseElement, open, close) {
+            if (parseExpected(open)) {
+                var result = parseDelimitedList(kind, parseElement);
+                parseExpected(close);
+                return result;
+            }
+            return createMissingList();
+        }
+        function parseEntityName(allowReservedWords, diagnosticMessage) {
+            var pos = getNodePos();
+            var entity = allowReservedWords ? parseIdentifierName(diagnosticMessage) : parseIdentifier(diagnosticMessage);
+            var dotPos = getNodePos();
+            while (parseOptional(24 /* DotToken */)) {
+                if (token() === 29 /* LessThanToken */) {
+                    // the entity is part of a JSDoc-style generic, so record the trailing dot for later error reporting
+                    entity.jsdocDotPos = dotPos;
+                    break;
+                }
+                dotPos = getNodePos();
+                entity = finishNode(factory.createQualifiedName(entity, parseRightSideOfDot(allowReservedWords, /* allowPrivateIdentifiers */ false)), pos);
+            }
+            return entity;
         }
-    ]);
-    /* @internal */
-    ts.typeAcquisitionDeclarations = [
-        {
-            /* @deprecated typingOptions.enableAutoDiscovery
-             * Use typeAcquisition.enable instead.
-             */
-            name: "enableAutoDiscovery",
-            type: "boolean",
-        },
-        {
-            name: "enable",
-            type: "boolean",
-        },
-        {
-            name: "include",
-            type: "list",
-            element: {
-                name: "include",
-                type: "string"
+        function createQualifiedName(entity, name) {
+            return finishNode(factory.createQualifiedName(entity, name), entity.pos);
+        }
+        function parseRightSideOfDot(allowIdentifierNames, allowPrivateIdentifiers) {
+            // Technically a keyword is valid here as all identifiers and keywords are identifier names.
+            // However, often we'll encounter this in error situations when the identifier or keyword
+            // is actually starting another valid construct.
+            //
+            // So, we check for the following specific case:
+            //
+            //      name.
+            //      identifierOrKeyword identifierNameOrKeyword
+            //
+            // Note: the newlines are important here.  For example, if that above code
+            // were rewritten into:
+            //
+            //      name.identifierOrKeyword
+            //      identifierNameOrKeyword
+            //
+            // Then we would consider it valid.  That's because ASI would take effect and
+            // the code would be implicitly: "name.identifierOrKeyword; identifierNameOrKeyword".
+            // In the first case though, ASI will not take effect because there is not a
+            // line terminator after the identifier or keyword.
+            if (scanner.hasPrecedingLineBreak() && ts.tokenIsIdentifierOrKeyword(token())) {
+                var matchesPattern = lookAhead(nextTokenIsIdentifierOrKeywordOnSameLine);
+                if (matchesPattern) {
+                    // Report that we need an identifier.  However, report it right after the dot,
+                    // and not on the next token.  This is because the next token might actually
+                    // be an identifier and the error would be quite confusing.
+                    return createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Identifier_expected);
+                }
+            }
+            if (token() === 79 /* PrivateIdentifier */) {
+                var node = parsePrivateIdentifier();
+                return allowPrivateIdentifiers ? node : createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Identifier_expected);
+            }
+            return allowIdentifierNames ? parseIdentifierName() : parseIdentifier();
+        }
+        function parseTemplateSpans(isTaggedTemplate) {
+            var pos = getNodePos();
+            var list = [];
+            var node;
+            do {
+                node = parseTemplateSpan(isTaggedTemplate);
+                list.push(node);
+            } while (node.literal.kind === 16 /* TemplateMiddle */);
+            return createNodeArray(list, pos);
+        }
+        function parseTemplateExpression(isTaggedTemplate) {
+            var pos = getNodePos();
+            return finishNode(factory.createTemplateExpression(parseTemplateHead(isTaggedTemplate), parseTemplateSpans(isTaggedTemplate)), pos);
+        }
+        function parseTemplateType() {
+            var pos = getNodePos();
+            return finishNode(factory.createTemplateLiteralType(parseTemplateHead(/*isTaggedTemplate*/ false), parseTemplateTypeSpans()), pos);
+        }
+        function parseTemplateTypeSpans() {
+            var pos = getNodePos();
+            var list = [];
+            var node;
+            do {
+                node = parseTemplateTypeSpan();
+                list.push(node);
+            } while (node.literal.kind === 16 /* TemplateMiddle */);
+            return createNodeArray(list, pos);
+        }
+        function parseTemplateTypeSpan() {
+            var pos = getNodePos();
+            return finishNode(factory.createTemplateLiteralTypeSpan(parseType(), parseLiteralOfTemplateSpan(/*isTaggedTemplate*/ false)), pos);
+        }
+        function parseLiteralOfTemplateSpan(isTaggedTemplate) {
+            if (token() === 19 /* CloseBraceToken */) {
+                reScanTemplateToken(isTaggedTemplate);
+                return parseTemplateMiddleOrTemplateTail();
+            }
+            else {
+                // TODO(rbuckton): Do we need to call `parseExpectedToken` or can we just call `createMissingNode` directly?
+                return parseExpectedToken(17 /* TemplateTail */, ts.Diagnostics._0_expected, ts.tokenToString(19 /* CloseBraceToken */));
+            }
+        }
+        function parseTemplateSpan(isTaggedTemplate) {
+            var pos = getNodePos();
+            return finishNode(factory.createTemplateSpan(allowInAnd(parseExpression), parseLiteralOfTemplateSpan(isTaggedTemplate)), pos);
+        }
+        function parseLiteralNode() {
+            return parseLiteralLikeNode(token());
+        }
+        function parseTemplateHead(isTaggedTemplate) {
+            if (isTaggedTemplate) {
+                reScanTemplateHeadOrNoSubstitutionTemplate();
+            }
+            var fragment = parseLiteralLikeNode(token());
+            ts.Debug.assert(fragment.kind === 15 /* TemplateHead */, "Template head has wrong token kind");
+            return fragment;
+        }
+        function parseTemplateMiddleOrTemplateTail() {
+            var fragment = parseLiteralLikeNode(token());
+            ts.Debug.assert(fragment.kind === 16 /* TemplateMiddle */ || fragment.kind === 17 /* TemplateTail */, "Template fragment has wrong token kind");
+            return fragment;
+        }
+        function getTemplateLiteralRawText(kind) {
+            var isLast = kind === 14 /* NoSubstitutionTemplateLiteral */ || kind === 17 /* TemplateTail */;
+            var tokenText = scanner.getTokenText();
+            return tokenText.substring(1, tokenText.length - (scanner.isUnterminated() ? 0 : isLast ? 1 : 2));
+        }
+        function parseLiteralLikeNode(kind) {
+            var pos = getNodePos();
+            var node = ts.isTemplateLiteralKind(kind) ? factory.createTemplateLiteralLikeNode(kind, scanner.getTokenValue(), getTemplateLiteralRawText(kind), scanner.getTokenFlags() & 2048 /* TemplateLiteralLikeFlags */) :
+                // Octal literals are not allowed in strict mode or ES5
+                // Note that theoretically the following condition would hold true literals like 009,
+                // which is not octal. But because of how the scanner separates the tokens, we would
+                // never get a token like this. Instead, we would get 00 and 9 as two separate tokens.
+                // We also do not need to check for negatives because any prefix operator would be part of a
+                // parent unary expression.
+                kind === 8 /* NumericLiteral */ ? factory.createNumericLiteral(scanner.getTokenValue(), scanner.getNumericLiteralFlags()) :
+                    kind === 10 /* StringLiteral */ ? factory.createStringLiteral(scanner.getTokenValue(), /*isSingleQuote*/ undefined, scanner.hasExtendedUnicodeEscape()) :
+                        ts.isLiteralKind(kind) ? factory.createLiteralLikeNode(kind, scanner.getTokenValue()) :
+                            ts.Debug.fail();
+            if (scanner.hasExtendedUnicodeEscape()) {
+                node.hasExtendedUnicodeEscape = true;
+            }
+            if (scanner.isUnterminated()) {
+                node.isUnterminated = true;
+            }
+            nextToken();
+            return finishNode(node, pos);
+        }
+        // TYPES
+        function parseEntityNameOfTypeReference() {
+            return parseEntityName(/*allowReservedWords*/ true, ts.Diagnostics.Type_expected);
+        }
+        function parseTypeArgumentsOfTypeReference() {
+            if (!scanner.hasPrecedingLineBreak() && reScanLessThanToken() === 29 /* LessThanToken */) {
+                return parseBracketedList(20 /* TypeArguments */, parseType, 29 /* LessThanToken */, 31 /* GreaterThanToken */);
+            }
+        }
+        function parseTypeReference() {
+            var pos = getNodePos();
+            return finishNode(factory.createTypeReferenceNode(parseEntityNameOfTypeReference(), parseTypeArgumentsOfTypeReference()), pos);
+        }
+        // If true, we should abort parsing an error function.
+        function typeHasArrowFunctionBlockingParseError(node) {
+            switch (node.kind) {
+                case 173 /* TypeReference */:
+                    return ts.nodeIsMissing(node.typeName);
+                case 174 /* FunctionType */:
+                case 175 /* ConstructorType */: {
+                    var _a = node, parameters = _a.parameters, type = _a.type;
+                    return isMissingList(parameters) || typeHasArrowFunctionBlockingParseError(type);
+                }
+                case 186 /* ParenthesizedType */:
+                    return typeHasArrowFunctionBlockingParseError(node.type);
+                default:
+                    return false;
+            }
+        }
+        function parseThisTypePredicate(lhs) {
+            nextToken();
+            return finishNode(factory.createTypePredicateNode(/*assertsModifier*/ undefined, lhs, parseType()), lhs.pos);
+        }
+        function parseThisTypeNode() {
+            var pos = getNodePos();
+            nextToken();
+            return finishNode(factory.createThisTypeNode(), pos);
+        }
+        function parseJSDocAllType() {
+            var pos = getNodePos();
+            nextToken();
+            return finishNode(factory.createJSDocAllType(), pos);
+        }
+        function parseJSDocNonNullableType() {
+            var pos = getNodePos();
+            nextToken();
+            return finishNode(factory.createJSDocNonNullableType(parseNonArrayType()), pos);
+        }
+        function parseJSDocUnknownOrNullableType() {
+            var pos = getNodePos();
+            // skip the ?
+            nextToken();
+            // Need to lookahead to decide if this is a nullable or unknown type.
+            // Here are cases where we'll pick the unknown type:
+            //
+            //      Foo(?,
+            //      { a: ? }
+            //      Foo(?)
+            //      Foo<?>
+            //      Foo(?=
+            //      (?|
+            if (token() === 27 /* CommaToken */ ||
+                token() === 19 /* CloseBraceToken */ ||
+                token() === 21 /* CloseParenToken */ ||
+                token() === 31 /* GreaterThanToken */ ||
+                token() === 62 /* EqualsToken */ ||
+                token() === 51 /* BarToken */) {
+                return finishNode(factory.createJSDocUnknownType(), pos);
+            }
+            else {
+                return finishNode(factory.createJSDocNullableType(parseType()), pos);
+            }
+        }
+        function parseJSDocFunctionType() {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            if (lookAhead(nextTokenIsOpenParen)) {
+                nextToken();
+                var parameters = parseParameters(4 /* Type */ | 32 /* JSDoc */);
+                var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+                return withJSDoc(finishNode(factory.createJSDocFunctionType(parameters, type), pos), hasJSDoc);
+            }
+            return finishNode(factory.createTypeReferenceNode(parseIdentifierName(), /*typeArguments*/ undefined), pos);
+        }
+        function parseJSDocParameter() {
+            var pos = getNodePos();
+            var name;
+            if (token() === 107 /* ThisKeyword */ || token() === 102 /* NewKeyword */) {
+                name = parseIdentifierName();
+                parseExpected(58 /* ColonToken */);
+            }
+            return finishNode(factory.createParameterDeclaration(
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*dotDotDotToken*/ undefined, 
+            // TODO(rbuckton): JSDoc parameters don't have names (except `this`/`new`), should we manufacture an empty identifier?
+            name, 
+            /*questionToken*/ undefined, parseJSDocType(), 
+            /*initializer*/ undefined), pos);
+        }
+        function parseJSDocType() {
+            scanner.setInJSDocType(true);
+            var pos = getNodePos();
+            if (parseOptional(139 /* ModuleKeyword */)) {
+                // TODO(rbuckton): We never set the type for a JSDocNamepathType. What should we put here?
+                var moduleTag = factory.createJSDocNamepathType(/*type*/ undefined);
+                terminate: while (true) {
+                    switch (token()) {
+                        case 19 /* CloseBraceToken */:
+                        case 1 /* EndOfFileToken */:
+                        case 27 /* CommaToken */:
+                        case 5 /* WhitespaceTrivia */:
+                            break terminate;
+                        default:
+                            nextTokenJSDoc();
+                    }
+                }
+                scanner.setInJSDocType(false);
+                return finishNode(moduleTag, pos);
+            }
+            var hasDotDotDot = parseOptional(25 /* DotDotDotToken */);
+            var type = parseTypeOrTypePredicate();
+            scanner.setInJSDocType(false);
+            if (hasDotDotDot) {
+                type = finishNode(factory.createJSDocVariadicType(type), pos);
+            }
+            if (token() === 62 /* EqualsToken */) {
+                nextToken();
+                return finishNode(factory.createJSDocOptionalType(type), pos);
+            }
+            return type;
+        }
+        function parseTypeQuery() {
+            var pos = getNodePos();
+            parseExpected(111 /* TypeOfKeyword */);
+            return finishNode(factory.createTypeQueryNode(parseEntityName(/*allowReservedWords*/ true)), pos);
+        }
+        function parseTypeParameter() {
+            var pos = getNodePos();
+            var name = parseIdentifier();
+            var constraint;
+            var expression;
+            if (parseOptional(93 /* ExtendsKeyword */)) {
+                // It's not uncommon for people to write improper constraints to a generic.  If the
+                // user writes a constraint that is an expression and not an actual type, then parse
+                // it out as an expression (so we can recover well), but report that a type is needed
+                // instead.
+                if (isStartOfType() || !isStartOfExpression()) {
+                    constraint = parseType();
+                }
+                else {
+                    // It was not a type, and it looked like an expression.  Parse out an expression
+                    // here so we recover well.  Note: it is important that we call parseUnaryExpression
+                    // and not parseExpression here.  If the user has:
+                    //
+                    //      <T extends "">
+                    //
+                    // We do *not* want to consume the `>` as we're consuming the expression for "".
+                    expression = parseUnaryExpressionOrHigher();
+                }
+            }
+            var defaultType = parseOptional(62 /* EqualsToken */) ? parseType() : undefined;
+            var node = factory.createTypeParameterDeclaration(name, constraint, defaultType);
+            node.expression = expression;
+            return finishNode(node, pos);
+        }
+        function parseTypeParameters() {
+            if (token() === 29 /* LessThanToken */) {
+                return parseBracketedList(19 /* TypeParameters */, parseTypeParameter, 29 /* LessThanToken */, 31 /* GreaterThanToken */);
+            }
+        }
+        function isStartOfParameter(isJSDocParameter) {
+            return token() === 25 /* DotDotDotToken */ ||
+                isBindingIdentifierOrPrivateIdentifierOrPattern() ||
+                ts.isModifierKind(token()) ||
+                token() === 59 /* AtToken */ ||
+                isStartOfType(/*inStartOfParameter*/ !isJSDocParameter);
+        }
+        function parseNameOfParameter(modifiers) {
+            // FormalParameter [Yield,Await]:
+            //      BindingElement[?Yield,?Await]
+            var name = parseIdentifierOrPattern(ts.Diagnostics.Private_identifiers_cannot_be_used_as_parameters);
+            if (ts.getFullWidth(name) === 0 && !ts.some(modifiers) && ts.isModifierKind(token())) {
+                // in cases like
+                // 'use strict'
+                // function foo(static)
+                // isParameter('static') === true, because of isModifier('static')
+                // however 'static' is not a legal identifier in a strict mode.
+                // so result of this function will be ParameterDeclaration (flags = 0, name = missing, type = undefined, initializer = undefined)
+                // and current token will not change => parsing of the enclosing parameter list will last till the end of time (or OOM)
+                // to avoid this we'll advance cursor to the next token.
+                nextToken();
+            }
+            return name;
+        }
+        function parseParameterInOuterAwaitContext() {
+            return parseParameterWorker(/*inOuterAwaitContext*/ true);
+        }
+        function parseParameter() {
+            return parseParameterWorker(/*inOuterAwaitContext*/ false);
+        }
+        function parseParameterWorker(inOuterAwaitContext) {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            if (token() === 107 /* ThisKeyword */) {
+                var node_1 = factory.createParameterDeclaration(
+                /*decorators*/ undefined, 
+                /*modifiers*/ undefined, 
+                /*dotDotDotToken*/ undefined, createIdentifier(/*isIdentifier*/ true), 
+                /*questionToken*/ undefined, parseTypeAnnotation(), 
+                /*initializer*/ undefined);
+                return withJSDoc(finishNode(node_1, pos), hasJSDoc);
+            }
+            // FormalParameter [Yield,Await]:
+            //      BindingElement[?Yield,?Await]
+            // Decorators are parsed in the outer [Await] context, the rest of the parameter is parsed in the function's [Await] context.
+            var decorators = inOuterAwaitContext ? doInAwaitContext(parseDecorators) : parseDecorators();
+            var savedTopLevel = topLevel;
+            topLevel = false;
+            var modifiers = parseModifiers();
+            var node = withJSDoc(finishNode(factory.createParameterDeclaration(decorators, modifiers, parseOptionalToken(25 /* DotDotDotToken */), parseNameOfParameter(modifiers), parseOptionalToken(57 /* QuestionToken */), parseTypeAnnotation(), parseInitializer()), pos), hasJSDoc);
+            topLevel = savedTopLevel;
+            return node;
+        }
+        function parseReturnType(returnToken, isType) {
+            if (shouldParseReturnType(returnToken, isType)) {
+                return parseTypeOrTypePredicate();
+            }
+        }
+        function shouldParseReturnType(returnToken, isType) {
+            if (returnToken === 38 /* EqualsGreaterThanToken */) {
+                parseExpected(returnToken);
+                return true;
+            }
+            else if (parseOptional(58 /* ColonToken */)) {
+                return true;
+            }
+            else if (isType && token() === 38 /* EqualsGreaterThanToken */) {
+                // This is easy to get backward, especially in type contexts, so parse the type anyway
+                parseErrorAtCurrentToken(ts.Diagnostics._0_expected, ts.tokenToString(58 /* ColonToken */));
+                nextToken();
+                return true;
+            }
+            return false;
+        }
+        function parseParametersWorker(flags) {
+            // FormalParameters [Yield,Await]: (modified)
+            //      [empty]
+            //      FormalParameterList[?Yield,Await]
+            //
+            // FormalParameter[Yield,Await]: (modified)
+            //      BindingElement[?Yield,Await]
+            //
+            // BindingElement [Yield,Await]: (modified)
+            //      SingleNameBinding[?Yield,?Await]
+            //      BindingPattern[?Yield,?Await]Initializer [In, ?Yield,?Await] opt
+            //
+            // SingleNameBinding [Yield,Await]:
+            //      BindingIdentifier[?Yield,?Await]Initializer [In, ?Yield,?Await] opt
+            var savedYieldContext = inYieldContext();
+            var savedAwaitContext = inAwaitContext();
+            setYieldContext(!!(flags & 1 /* Yield */));
+            setAwaitContext(!!(flags & 2 /* Await */));
+            var parameters = flags & 32 /* JSDoc */ ?
+                parseDelimitedList(17 /* JSDocParameters */, parseJSDocParameter) :
+                parseDelimitedList(16 /* Parameters */, savedAwaitContext ? parseParameterInOuterAwaitContext : parseParameter);
+            setYieldContext(savedYieldContext);
+            setAwaitContext(savedAwaitContext);
+            return parameters;
+        }
+        function parseParameters(flags) {
+            // FormalParameters [Yield,Await]: (modified)
+            //      [empty]
+            //      FormalParameterList[?Yield,Await]
+            //
+            // FormalParameter[Yield,Await]: (modified)
+            //      BindingElement[?Yield,Await]
+            //
+            // BindingElement [Yield,Await]: (modified)
+            //      SingleNameBinding[?Yield,?Await]
+            //      BindingPattern[?Yield,?Await]Initializer [In, ?Yield,?Await] opt
+            //
+            // SingleNameBinding [Yield,Await]:
+            //      BindingIdentifier[?Yield,?Await]Initializer [In, ?Yield,?Await] opt
+            if (!parseExpected(20 /* OpenParenToken */)) {
+                return createMissingList();
+            }
+            var parameters = parseParametersWorker(flags);
+            parseExpected(21 /* CloseParenToken */);
+            return parameters;
+        }
+        function parseTypeMemberSemicolon() {
+            // We allow type members to be separated by commas or (possibly ASI) semicolons.
+            // First check if it was a comma.  If so, we're done with the member.
+            if (parseOptional(27 /* CommaToken */)) {
+                return;
+            }
+            // Didn't have a comma.  We must have a (possible ASI) semicolon.
+            parseSemicolon();
+        }
+        function parseSignatureMember(kind) {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            if (kind === 170 /* ConstructSignature */) {
+                parseExpected(102 /* NewKeyword */);
+            }
+            var typeParameters = parseTypeParameters();
+            var parameters = parseParameters(4 /* Type */);
+            var type = parseReturnType(58 /* ColonToken */, /*isType*/ true);
+            parseTypeMemberSemicolon();
+            var node = kind === 169 /* CallSignature */
+                ? factory.createCallSignature(typeParameters, parameters, type)
+                : factory.createConstructSignature(typeParameters, parameters, type);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function isIndexSignature() {
+            return token() === 22 /* OpenBracketToken */ && lookAhead(isUnambiguouslyIndexSignature);
+        }
+        function isUnambiguouslyIndexSignature() {
+            // The only allowed sequence is:
+            //
+            //   [id:
+            //
+            // However, for error recovery, we also check the following cases:
+            //
+            //   [...
+            //   [id,
+            //   [id?,
+            //   [id?:
+            //   [id?]
+            //   [public id
+            //   [private id
+            //   [protected id
+            //   []
+            //
+            nextToken();
+            if (token() === 25 /* DotDotDotToken */ || token() === 23 /* CloseBracketToken */) {
+                return true;
+            }
+            if (ts.isModifierKind(token())) {
+                nextToken();
+                if (isIdentifier()) {
+                    return true;
+                }
+            }
+            else if (!isIdentifier()) {
+                return false;
+            }
+            else {
+                // Skip the identifier
+                nextToken();
+            }
+            // A colon signifies a well formed indexer
+            // A comma should be a badly formed indexer because comma expressions are not allowed
+            // in computed properties.
+            if (token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */) {
+                return true;
+            }
+            // Question mark could be an indexer with an optional property,
+            // or it could be a conditional expression in a computed property.
+            if (token() !== 57 /* QuestionToken */) {
+                return false;
+            }
+            // If any of the following tokens are after the question mark, it cannot
+            // be a conditional expression, so treat it as an indexer.
+            nextToken();
+            return token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */ || token() === 23 /* CloseBracketToken */;
+        }
+        function parseIndexSignatureDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            var parameters = parseBracketedList(16 /* Parameters */, parseParameter, 22 /* OpenBracketToken */, 23 /* CloseBracketToken */);
+            var type = parseTypeAnnotation();
+            parseTypeMemberSemicolon();
+            var node = factory.createIndexSignature(decorators, modifiers, parameters, type);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parsePropertyOrMethodSignature(pos, hasJSDoc, modifiers) {
+            var name = parsePropertyName();
+            var questionToken = parseOptionalToken(57 /* QuestionToken */);
+            var node;
+            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
+                // Method signatures don't exist in expression contexts.  So they have neither
+                // [Yield] nor [Await]
+                var typeParameters = parseTypeParameters();
+                var parameters = parseParameters(4 /* Type */);
+                var type = parseReturnType(58 /* ColonToken */, /*isType*/ true);
+                node = factory.createMethodSignature(modifiers, name, questionToken, typeParameters, parameters, type);
+            }
+            else {
+                var type = parseTypeAnnotation();
+                node = factory.createPropertySignature(modifiers, name, questionToken, type);
+                // Although type literal properties cannot not have initializers, we attempt
+                // to parse an initializer so we can report in the checker that an interface
+                // property or type literal property cannot have an initializer.
+                if (token() === 62 /* EqualsToken */)
+                    node.initializer = parseInitializer();
+            }
+            parseTypeMemberSemicolon();
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function isTypeMemberStart() {
+            // Return true if we have the start of a signature member
+            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
+                return true;
+            }
+            var idToken = false;
+            // Eat up all modifiers, but hold on to the last one in case it is actually an identifier
+            while (ts.isModifierKind(token())) {
+                idToken = true;
+                nextToken();
+            }
+            // Index signatures and computed property names are type members
+            if (token() === 22 /* OpenBracketToken */) {
+                return true;
+            }
+            // Try to get the first property-like token following all modifiers
+            if (isLiteralPropertyName()) {
+                idToken = true;
+                nextToken();
+            }
+            // If we were able to get any potential identifier, check that it is
+            // the start of a member declaration
+            if (idToken) {
+                return token() === 20 /* OpenParenToken */ ||
+                    token() === 29 /* LessThanToken */ ||
+                    token() === 57 /* QuestionToken */ ||
+                    token() === 58 /* ColonToken */ ||
+                    token() === 27 /* CommaToken */ ||
+                    canParseSemicolon();
+            }
+            return false;
+        }
+        function parseTypeMember() {
+            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
+                return parseSignatureMember(169 /* CallSignature */);
+            }
+            if (token() === 102 /* NewKeyword */ && lookAhead(nextTokenIsOpenParenOrLessThan)) {
+                return parseSignatureMember(170 /* ConstructSignature */);
+            }
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var modifiers = parseModifiers();
+            if (isIndexSignature()) {
+                return parseIndexSignatureDeclaration(pos, hasJSDoc, /*decorators*/ undefined, modifiers);
+            }
+            return parsePropertyOrMethodSignature(pos, hasJSDoc, modifiers);
+        }
+        function nextTokenIsOpenParenOrLessThan() {
+            nextToken();
+            return token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */;
+        }
+        function nextTokenIsDot() {
+            return nextToken() === 24 /* DotToken */;
+        }
+        function nextTokenIsOpenParenOrLessThanOrDot() {
+            switch (nextToken()) {
+                case 20 /* OpenParenToken */:
+                case 29 /* LessThanToken */:
+                case 24 /* DotToken */:
+                    return true;
+            }
+            return false;
+        }
+        function parseTypeLiteral() {
+            var pos = getNodePos();
+            return finishNode(factory.createTypeLiteralNode(parseObjectTypeMembers()), pos);
+        }
+        function parseObjectTypeMembers() {
+            var members;
+            if (parseExpected(18 /* OpenBraceToken */)) {
+                members = parseList(4 /* TypeMembers */, parseTypeMember);
+                parseExpected(19 /* CloseBraceToken */);
+            }
+            else {
+                members = createMissingList();
+            }
+            return members;
+        }
+        function isStartOfMappedType() {
+            nextToken();
+            if (token() === 39 /* PlusToken */ || token() === 40 /* MinusToken */) {
+                return nextToken() === 142 /* ReadonlyKeyword */;
+            }
+            if (token() === 142 /* ReadonlyKeyword */) {
+                nextToken();
+            }
+            return token() === 22 /* OpenBracketToken */ && nextTokenIsIdentifier() && nextToken() === 100 /* InKeyword */;
+        }
+        function parseMappedTypeParameter() {
+            var pos = getNodePos();
+            var name = parseIdentifierName();
+            parseExpected(100 /* InKeyword */);
+            var type = parseType();
+            return finishNode(factory.createTypeParameterDeclaration(name, type, /*defaultType*/ undefined), pos);
+        }
+        function parseMappedType() {
+            var pos = getNodePos();
+            parseExpected(18 /* OpenBraceToken */);
+            var readonlyToken;
+            if (token() === 142 /* ReadonlyKeyword */ || token() === 39 /* PlusToken */ || token() === 40 /* MinusToken */) {
+                readonlyToken = parseTokenNode();
+                if (readonlyToken.kind !== 142 /* ReadonlyKeyword */) {
+                    parseExpected(142 /* ReadonlyKeyword */);
+                }
+            }
+            parseExpected(22 /* OpenBracketToken */);
+            var typeParameter = parseMappedTypeParameter();
+            var nameType = parseOptional(126 /* AsKeyword */) ? parseType() : undefined;
+            parseExpected(23 /* CloseBracketToken */);
+            var questionToken;
+            if (token() === 57 /* QuestionToken */ || token() === 39 /* PlusToken */ || token() === 40 /* MinusToken */) {
+                questionToken = parseTokenNode();
+                if (questionToken.kind !== 57 /* QuestionToken */) {
+                    parseExpected(57 /* QuestionToken */);
+                }
+            }
+            var type = parseTypeAnnotation();
+            parseSemicolon();
+            parseExpected(19 /* CloseBraceToken */);
+            return finishNode(factory.createMappedTypeNode(readonlyToken, typeParameter, nameType, questionToken, type), pos);
+        }
+        function parseTupleElementType() {
+            var pos = getNodePos();
+            if (parseOptional(25 /* DotDotDotToken */)) {
+                return finishNode(factory.createRestTypeNode(parseType()), pos);
+            }
+            var type = parseType();
+            if (ts.isJSDocNullableType(type) && type.pos === type.type.pos) {
+                var node = factory.createOptionalTypeNode(type.type);
+                ts.setTextRange(node, type);
+                node.flags = type.flags;
+                return node;
+            }
+            return type;
+        }
+        function isNextTokenColonOrQuestionColon() {
+            return nextToken() === 58 /* ColonToken */ || (token() === 57 /* QuestionToken */ && nextToken() === 58 /* ColonToken */);
+        }
+        function isTupleElementName() {
+            if (token() === 25 /* DotDotDotToken */) {
+                return ts.tokenIsIdentifierOrKeyword(nextToken()) && isNextTokenColonOrQuestionColon();
+            }
+            return ts.tokenIsIdentifierOrKeyword(token()) && isNextTokenColonOrQuestionColon();
+        }
+        function parseTupleElementNameOrTupleElementType() {
+            if (lookAhead(isTupleElementName)) {
+                var pos = getNodePos();
+                var hasJSDoc = hasPrecedingJSDocComment();
+                var dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
+                var name = parseIdentifierName();
+                var questionToken = parseOptionalToken(57 /* QuestionToken */);
+                parseExpected(58 /* ColonToken */);
+                var type = parseTupleElementType();
+                var node = factory.createNamedTupleMember(dotDotDotToken, name, questionToken, type);
+                return withJSDoc(finishNode(node, pos), hasJSDoc);
+            }
+            return parseTupleElementType();
+        }
+        function parseTupleType() {
+            var pos = getNodePos();
+            return finishNode(factory.createTupleTypeNode(parseBracketedList(21 /* TupleElementTypes */, parseTupleElementNameOrTupleElementType, 22 /* OpenBracketToken */, 23 /* CloseBracketToken */)), pos);
+        }
+        function parseParenthesizedType() {
+            var pos = getNodePos();
+            parseExpected(20 /* OpenParenToken */);
+            var type = parseType();
+            parseExpected(21 /* CloseParenToken */);
+            return finishNode(factory.createParenthesizedType(type), pos);
+        }
+        function parseFunctionOrConstructorType() {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var isConstructorType = parseOptional(102 /* NewKeyword */);
+            var typeParameters = parseTypeParameters();
+            var parameters = parseParameters(4 /* Type */);
+            var type = parseReturnType(38 /* EqualsGreaterThanToken */, /*isType*/ false);
+            var node = isConstructorType
+                ? factory.createConstructorTypeNode(typeParameters, parameters, type)
+                : factory.createFunctionTypeNode(typeParameters, parameters, type);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parseKeywordAndNoDot() {
+            var node = parseTokenNode();
+            return token() === 24 /* DotToken */ ? undefined : node;
+        }
+        function parseLiteralTypeNode(negative) {
+            var pos = getNodePos();
+            if (negative) {
+                nextToken();
+            }
+            var expression = token() === 109 /* TrueKeyword */ || token() === 94 /* FalseKeyword */ || token() === 103 /* NullKeyword */ ?
+                parseTokenNode() :
+                parseLiteralLikeNode(token());
+            if (negative) {
+                expression = finishNode(factory.createPrefixUnaryExpression(40 /* MinusToken */, expression), pos);
+            }
+            return finishNode(factory.createLiteralTypeNode(expression), pos);
+        }
+        function isStartOfTypeOfImportType() {
+            nextToken();
+            return token() === 99 /* ImportKeyword */;
+        }
+        function parseImportType() {
+            sourceFlags |= 1048576 /* PossiblyContainsDynamicImport */;
+            var pos = getNodePos();
+            var isTypeOf = parseOptional(111 /* TypeOfKeyword */);
+            parseExpected(99 /* ImportKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var type = parseType();
+            parseExpected(21 /* CloseParenToken */);
+            var qualifier = parseOptional(24 /* DotToken */) ? parseEntityNameOfTypeReference() : undefined;
+            var typeArguments = parseTypeArgumentsOfTypeReference();
+            return finishNode(factory.createImportTypeNode(type, qualifier, typeArguments, isTypeOf), pos);
+        }
+        function nextTokenIsNumericOrBigIntLiteral() {
+            nextToken();
+            return token() === 8 /* NumericLiteral */ || token() === 9 /* BigIntLiteral */;
+        }
+        function parseNonArrayType() {
+            switch (token()) {
+                case 128 /* AnyKeyword */:
+                case 152 /* UnknownKeyword */:
+                case 147 /* StringKeyword */:
+                case 144 /* NumberKeyword */:
+                case 155 /* BigIntKeyword */:
+                case 148 /* SymbolKeyword */:
+                case 131 /* BooleanKeyword */:
+                case 150 /* UndefinedKeyword */:
+                case 141 /* NeverKeyword */:
+                case 145 /* ObjectKeyword */:
+                    // If these are followed by a dot, then parse these out as a dotted type reference instead.
+                    return tryParse(parseKeywordAndNoDot) || parseTypeReference();
+                case 65 /* AsteriskEqualsToken */:
+                    // If there is '*=', treat it as * followed by postfix =
+                    scanner.reScanAsteriskEqualsToken();
+                // falls through
+                case 41 /* AsteriskToken */:
+                    return parseJSDocAllType();
+                case 60 /* QuestionQuestionToken */:
+                    // If there is '??', treat it as prefix-'?' in JSDoc type.
+                    scanner.reScanQuestionToken();
+                // falls through
+                case 57 /* QuestionToken */:
+                    return parseJSDocUnknownOrNullableType();
+                case 97 /* FunctionKeyword */:
+                    return parseJSDocFunctionType();
+                case 53 /* ExclamationToken */:
+                    return parseJSDocNonNullableType();
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 10 /* StringLiteral */:
+                case 8 /* NumericLiteral */:
+                case 9 /* BigIntLiteral */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                case 103 /* NullKeyword */:
+                    return parseLiteralTypeNode();
+                case 40 /* MinusToken */:
+                    return lookAhead(nextTokenIsNumericOrBigIntLiteral) ? parseLiteralTypeNode(/*negative*/ true) : parseTypeReference();
+                case 113 /* VoidKeyword */:
+                    return parseTokenNode();
+                case 107 /* ThisKeyword */: {
+                    var thisKeyword = parseThisTypeNode();
+                    if (token() === 137 /* IsKeyword */ && !scanner.hasPrecedingLineBreak()) {
+                        return parseThisTypePredicate(thisKeyword);
+                    }
+                    else {
+                        return thisKeyword;
+                    }
+                }
+                case 111 /* TypeOfKeyword */:
+                    return lookAhead(isStartOfTypeOfImportType) ? parseImportType() : parseTypeQuery();
+                case 18 /* OpenBraceToken */:
+                    return lookAhead(isStartOfMappedType) ? parseMappedType() : parseTypeLiteral();
+                case 22 /* OpenBracketToken */:
+                    return parseTupleType();
+                case 20 /* OpenParenToken */:
+                    return parseParenthesizedType();
+                case 99 /* ImportKeyword */:
+                    return parseImportType();
+                case 127 /* AssertsKeyword */:
+                    return lookAhead(nextTokenIsIdentifierOrKeywordOnSameLine) ? parseAssertsTypePredicate() : parseTypeReference();
+                case 15 /* TemplateHead */:
+                    return parseTemplateType();
+                default:
+                    return parseTypeReference();
+            }
+        }
+        function isStartOfType(inStartOfParameter) {
+            switch (token()) {
+                case 128 /* AnyKeyword */:
+                case 152 /* UnknownKeyword */:
+                case 147 /* StringKeyword */:
+                case 144 /* NumberKeyword */:
+                case 155 /* BigIntKeyword */:
+                case 131 /* BooleanKeyword */:
+                case 142 /* ReadonlyKeyword */:
+                case 148 /* SymbolKeyword */:
+                case 151 /* UniqueKeyword */:
+                case 113 /* VoidKeyword */:
+                case 150 /* UndefinedKeyword */:
+                case 103 /* NullKeyword */:
+                case 107 /* ThisKeyword */:
+                case 111 /* TypeOfKeyword */:
+                case 141 /* NeverKeyword */:
+                case 18 /* OpenBraceToken */:
+                case 22 /* OpenBracketToken */:
+                case 29 /* LessThanToken */:
+                case 51 /* BarToken */:
+                case 50 /* AmpersandToken */:
+                case 102 /* NewKeyword */:
+                case 10 /* StringLiteral */:
+                case 8 /* NumericLiteral */:
+                case 9 /* BigIntLiteral */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                case 145 /* ObjectKeyword */:
+                case 41 /* AsteriskToken */:
+                case 57 /* QuestionToken */:
+                case 53 /* ExclamationToken */:
+                case 25 /* DotDotDotToken */:
+                case 135 /* InferKeyword */:
+                case 99 /* ImportKeyword */:
+                case 127 /* AssertsKeyword */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 15 /* TemplateHead */:
+                    return true;
+                case 97 /* FunctionKeyword */:
+                    return !inStartOfParameter;
+                case 40 /* MinusToken */:
+                    return !inStartOfParameter && lookAhead(nextTokenIsNumericOrBigIntLiteral);
+                case 20 /* OpenParenToken */:
+                    // Only consider '(' the start of a type if followed by ')', '...', an identifier, a modifier,
+                    // or something that starts a type. We don't want to consider things like '(1)' a type.
+                    return !inStartOfParameter && lookAhead(isStartOfParenthesizedOrFunctionType);
+                default:
+                    return isIdentifier();
+            }
+        }
+        function isStartOfParenthesizedOrFunctionType() {
+            nextToken();
+            return token() === 21 /* CloseParenToken */ || isStartOfParameter(/*isJSDocParameter*/ false) || isStartOfType();
+        }
+        function parsePostfixTypeOrHigher() {
+            var pos = getNodePos();
+            var type = parseNonArrayType();
+            while (!scanner.hasPrecedingLineBreak()) {
+                switch (token()) {
+                    case 53 /* ExclamationToken */:
+                        nextToken();
+                        type = finishNode(factory.createJSDocNonNullableType(type), pos);
+                        break;
+                    case 57 /* QuestionToken */:
+                        // If next token is start of a type we have a conditional type
+                        if (lookAhead(nextTokenIsStartOfType)) {
+                            return type;
+                        }
+                        nextToken();
+                        type = finishNode(factory.createJSDocNullableType(type), pos);
+                        break;
+                    case 22 /* OpenBracketToken */:
+                        parseExpected(22 /* OpenBracketToken */);
+                        if (isStartOfType()) {
+                            var indexType = parseType();
+                            parseExpected(23 /* CloseBracketToken */);
+                            type = finishNode(factory.createIndexedAccessTypeNode(type, indexType), pos);
+                        }
+                        else {
+                            parseExpected(23 /* CloseBracketToken */);
+                            type = finishNode(factory.createArrayTypeNode(type), pos);
+                        }
+                        break;
+                    default:
+                        return type;
+                }
+            }
+            return type;
+        }
+        function parseTypeOperator(operator) {
+            var pos = getNodePos();
+            parseExpected(operator);
+            return finishNode(factory.createTypeOperatorNode(operator, parseTypeOperatorOrHigher()), pos);
+        }
+        function parseTypeParameterOfInferType() {
+            var pos = getNodePos();
+            return finishNode(factory.createTypeParameterDeclaration(parseIdentifier(), 
+            /*constraint*/ undefined, 
+            /*defaultType*/ undefined), pos);
+        }
+        function parseInferType() {
+            var pos = getNodePos();
+            parseExpected(135 /* InferKeyword */);
+            return finishNode(factory.createInferTypeNode(parseTypeParameterOfInferType()), pos);
+        }
+        function parseTypeOperatorOrHigher() {
+            var operator = token();
+            switch (operator) {
+                case 138 /* KeyOfKeyword */:
+                case 151 /* UniqueKeyword */:
+                case 142 /* ReadonlyKeyword */:
+                    return parseTypeOperator(operator);
+                case 135 /* InferKeyword */:
+                    return parseInferType();
+            }
+            return parsePostfixTypeOrHigher();
+        }
+        function parseFunctionOrConstructorTypeToError(isInUnionType) {
+            // the function type and constructor type shorthand notation
+            // are not allowed directly in unions and intersections, but we'll
+            // try to parse them gracefully and issue a helpful message.
+            if (isStartOfFunctionTypeOrConstructorType()) {
+                var type = parseFunctionOrConstructorType();
+                var diagnostic = void 0;
+                if (ts.isFunctionTypeNode(type)) {
+                    diagnostic = isInUnionType
+                        ? ts.Diagnostics.Function_type_notation_must_be_parenthesized_when_used_in_a_union_type
+                        : ts.Diagnostics.Function_type_notation_must_be_parenthesized_when_used_in_an_intersection_type;
+                }
+                else {
+                    diagnostic = isInUnionType
+                        ? ts.Diagnostics.Constructor_type_notation_must_be_parenthesized_when_used_in_a_union_type
+                        : ts.Diagnostics.Constructor_type_notation_must_be_parenthesized_when_used_in_an_intersection_type;
+                }
+                parseErrorAtRange(type, diagnostic);
+                return type;
+            }
+            return undefined;
+        }
+        function parseUnionOrIntersectionType(operator, parseConstituentType, createTypeNode) {
+            var pos = getNodePos();
+            var isUnionType = operator === 51 /* BarToken */;
+            var hasLeadingOperator = parseOptional(operator);
+            var type = hasLeadingOperator && parseFunctionOrConstructorTypeToError(isUnionType)
+                || parseConstituentType();
+            if (token() === operator || hasLeadingOperator) {
+                var types = [type];
+                while (parseOptional(operator)) {
+                    types.push(parseFunctionOrConstructorTypeToError(isUnionType) || parseConstituentType());
+                }
+                type = finishNode(createTypeNode(createNodeArray(types, pos)), pos);
+            }
+            return type;
+        }
+        function parseIntersectionTypeOrHigher() {
+            return parseUnionOrIntersectionType(50 /* AmpersandToken */, parseTypeOperatorOrHigher, factory.createIntersectionTypeNode);
+        }
+        function parseUnionTypeOrHigher() {
+            return parseUnionOrIntersectionType(51 /* BarToken */, parseIntersectionTypeOrHigher, factory.createUnionTypeNode);
+        }
+        function isStartOfFunctionTypeOrConstructorType() {
+            if (token() === 29 /* LessThanToken */) {
+                return true;
+            }
+            if (token() === 20 /* OpenParenToken */ && lookAhead(isUnambiguouslyStartOfFunctionType)) {
+                return true;
+            }
+            return token() === 102 /* NewKeyword */;
+        }
+        function skipParameterStart() {
+            if (ts.isModifierKind(token())) {
+                // Skip modifiers
+                parseModifiers();
+            }
+            if (isIdentifier() || token() === 107 /* ThisKeyword */) {
+                nextToken();
+                return true;
+            }
+            if (token() === 22 /* OpenBracketToken */ || token() === 18 /* OpenBraceToken */) {
+                // Return true if we can parse an array or object binding pattern with no errors
+                var previousErrorCount = parseDiagnostics.length;
+                parseIdentifierOrPattern();
+                return previousErrorCount === parseDiagnostics.length;
+            }
+            return false;
+        }
+        function isUnambiguouslyStartOfFunctionType() {
+            nextToken();
+            if (token() === 21 /* CloseParenToken */ || token() === 25 /* DotDotDotToken */) {
+                // ( )
+                // ( ...
+                return true;
+            }
+            if (skipParameterStart()) {
+                // We successfully skipped modifiers (if any) and an identifier or binding pattern,
+                // now see if we have something that indicates a parameter declaration
+                if (token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */ ||
+                    token() === 57 /* QuestionToken */ || token() === 62 /* EqualsToken */) {
+                    // ( xxx :
+                    // ( xxx ,
+                    // ( xxx ?
+                    // ( xxx =
+                    return true;
+                }
+                if (token() === 21 /* CloseParenToken */) {
+                    nextToken();
+                    if (token() === 38 /* EqualsGreaterThanToken */) {
+                        // ( xxx ) =>
+                        return true;
+                    }
+                }
+            }
+            return false;
+        }
+        function parseTypeOrTypePredicate() {
+            var pos = getNodePos();
+            var typePredicateVariable = isIdentifier() && tryParse(parseTypePredicatePrefix);
+            var type = parseType();
+            if (typePredicateVariable) {
+                return finishNode(factory.createTypePredicateNode(/*assertsModifier*/ undefined, typePredicateVariable, type), pos);
+            }
+            else {
+                return type;
+            }
+        }
+        function parseTypePredicatePrefix() {
+            var id = parseIdentifier();
+            if (token() === 137 /* IsKeyword */ && !scanner.hasPrecedingLineBreak()) {
+                nextToken();
+                return id;
+            }
+        }
+        function parseAssertsTypePredicate() {
+            var pos = getNodePos();
+            var assertsModifier = parseExpectedToken(127 /* AssertsKeyword */);
+            var parameterName = token() === 107 /* ThisKeyword */ ? parseThisTypeNode() : parseIdentifier();
+            var type = parseOptional(137 /* IsKeyword */) ? parseType() : undefined;
+            return finishNode(factory.createTypePredicateNode(assertsModifier, parameterName, type), pos);
+        }
+        function parseType() {
+            // The rules about 'yield' only apply to actual code/expression contexts.  They don't
+            // apply to 'type' contexts.  So we disable these parameters here before moving on.
+            return doOutsideOfContext(40960 /* TypeExcludesFlags */, parseTypeWorker);
+        }
+        function parseTypeWorker(noConditionalTypes) {
+            if (isStartOfFunctionTypeOrConstructorType()) {
+                return parseFunctionOrConstructorType();
+            }
+            var pos = getNodePos();
+            var type = parseUnionTypeOrHigher();
+            if (!noConditionalTypes && !scanner.hasPrecedingLineBreak() && parseOptional(93 /* ExtendsKeyword */)) {
+                // The type following 'extends' is not permitted to be another conditional type
+                var extendsType = parseTypeWorker(/*noConditionalTypes*/ true);
+                parseExpected(57 /* QuestionToken */);
+                var trueType = parseTypeWorker();
+                parseExpected(58 /* ColonToken */);
+                var falseType = parseTypeWorker();
+                return finishNode(factory.createConditionalTypeNode(type, extendsType, trueType, falseType), pos);
+            }
+            return type;
+        }
+        function parseTypeAnnotation() {
+            return parseOptional(58 /* ColonToken */) ? parseType() : undefined;
+        }
+        // EXPRESSIONS
+        function isStartOfLeftHandSideExpression() {
+            switch (token()) {
+                case 107 /* ThisKeyword */:
+                case 105 /* SuperKeyword */:
+                case 103 /* NullKeyword */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                case 8 /* NumericLiteral */:
+                case 9 /* BigIntLiteral */:
+                case 10 /* StringLiteral */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 15 /* TemplateHead */:
+                case 20 /* OpenParenToken */:
+                case 22 /* OpenBracketToken */:
+                case 18 /* OpenBraceToken */:
+                case 97 /* FunctionKeyword */:
+                case 83 /* ClassKeyword */:
+                case 102 /* NewKeyword */:
+                case 43 /* SlashToken */:
+                case 67 /* SlashEqualsToken */:
+                case 78 /* Identifier */:
+                    return true;
+                case 99 /* ImportKeyword */:
+                    return lookAhead(nextTokenIsOpenParenOrLessThanOrDot);
+                default:
+                    return isIdentifier();
+            }
+        }
+        function isStartOfExpression() {
+            if (isStartOfLeftHandSideExpression()) {
+                return true;
+            }
+            switch (token()) {
+                case 39 /* PlusToken */:
+                case 40 /* MinusToken */:
+                case 54 /* TildeToken */:
+                case 53 /* ExclamationToken */:
+                case 88 /* DeleteKeyword */:
+                case 111 /* TypeOfKeyword */:
+                case 113 /* VoidKeyword */:
+                case 45 /* PlusPlusToken */:
+                case 46 /* MinusMinusToken */:
+                case 29 /* LessThanToken */:
+                case 130 /* AwaitKeyword */:
+                case 124 /* YieldKeyword */:
+                case 79 /* PrivateIdentifier */:
+                    // Yield/await always starts an expression.  Either it is an identifier (in which case
+                    // it is definitely an expression).  Or it's a keyword (either because we're in
+                    // a generator or async function, or in strict mode (or both)) and it started a yield or await expression.
+                    return true;
+                default:
+                    // Error tolerance.  If we see the start of some binary operator, we consider
+                    // that the start of an expression.  That way we'll parse out a missing identifier,
+                    // give a good message about an identifier being missing, and then consume the
+                    // rest of the binary expression.
+                    if (isBinaryOperator()) {
+                        return true;
+                    }
+                    return isIdentifier();
+            }
+        }
+        function isStartOfExpressionStatement() {
+            // As per the grammar, none of '{' or 'function' or 'class' can start an expression statement.
+            return token() !== 18 /* OpenBraceToken */ &&
+                token() !== 97 /* FunctionKeyword */ &&
+                token() !== 83 /* ClassKeyword */ &&
+                token() !== 59 /* AtToken */ &&
+                isStartOfExpression();
+        }
+        function parseExpression() {
+            // Expression[in]:
+            //      AssignmentExpression[in]
+            //      Expression[in] , AssignmentExpression[in]
+            // clear the decorator context when parsing Expression, as it should be unambiguous when parsing a decorator
+            var saveDecoratorContext = inDecoratorContext();
+            if (saveDecoratorContext) {
+                setDecoratorContext(/*val*/ false);
+            }
+            var pos = getNodePos();
+            var expr = parseAssignmentExpressionOrHigher();
+            var operatorToken;
+            while ((operatorToken = parseOptionalToken(27 /* CommaToken */))) {
+                expr = makeBinaryExpression(expr, operatorToken, parseAssignmentExpressionOrHigher(), pos);
+            }
+            if (saveDecoratorContext) {
+                setDecoratorContext(/*val*/ true);
+            }
+            return expr;
+        }
+        function parseInitializer() {
+            return parseOptional(62 /* EqualsToken */) ? parseAssignmentExpressionOrHigher() : undefined;
+        }
+        function parseAssignmentExpressionOrHigher() {
+            //  AssignmentExpression[in,yield]:
+            //      1) ConditionalExpression[?in,?yield]
+            //      2) LeftHandSideExpression = AssignmentExpression[?in,?yield]
+            //      3) LeftHandSideExpression AssignmentOperator AssignmentExpression[?in,?yield]
+            //      4) ArrowFunctionExpression[?in,?yield]
+            //      5) AsyncArrowFunctionExpression[in,yield,await]
+            //      6) [+Yield] YieldExpression[?In]
+            //
+            // Note: for ease of implementation we treat productions '2' and '3' as the same thing.
+            // (i.e. they're both BinaryExpressions with an assignment operator in it).
+            // First, do the simple check if we have a YieldExpression (production '6').
+            if (isYieldExpression()) {
+                return parseYieldExpression();
+            }
+            // Then, check if we have an arrow function (production '4' and '5') that starts with a parenthesized
+            // parameter list or is an async arrow function.
+            // AsyncArrowFunctionExpression:
+            //      1) async[no LineTerminator here]AsyncArrowBindingIdentifier[?Yield][no LineTerminator here]=>AsyncConciseBody[?In]
+            //      2) CoverCallExpressionAndAsyncArrowHead[?Yield, ?Await][no LineTerminator here]=>AsyncConciseBody[?In]
+            // Production (1) of AsyncArrowFunctionExpression is parsed in "tryParseAsyncSimpleArrowFunctionExpression".
+            // And production (2) is parsed in "tryParseParenthesizedArrowFunctionExpression".
+            //
+            // If we do successfully parse arrow-function, we must *not* recurse for productions 1, 2 or 3. An ArrowFunction is
+            // not a LeftHandSideExpression, nor does it start a ConditionalExpression.  So we are done
+            // with AssignmentExpression if we see one.
+            var arrowExpression = tryParseParenthesizedArrowFunctionExpression() || tryParseAsyncSimpleArrowFunctionExpression();
+            if (arrowExpression) {
+                return arrowExpression;
+            }
+            // Now try to see if we're in production '1', '2' or '3'.  A conditional expression can
+            // start with a LogicalOrExpression, while the assignment productions can only start with
+            // LeftHandSideExpressions.
+            //
+            // So, first, we try to just parse out a BinaryExpression.  If we get something that is a
+            // LeftHandSide or higher, then we can try to parse out the assignment expression part.
+            // Otherwise, we try to parse out the conditional expression bit.  We want to allow any
+            // binary expression here, so we pass in the 'lowest' precedence here so that it matches
+            // and consumes anything.
+            var pos = getNodePos();
+            var expr = parseBinaryExpressionOrHigher(0 /* Lowest */);
+            // To avoid a look-ahead, we did not handle the case of an arrow function with a single un-parenthesized
+            // parameter ('x => ...') above. We handle it here by checking if the parsed expression was a single
+            // identifier and the current token is an arrow.
+            if (expr.kind === 78 /* Identifier */ && token() === 38 /* EqualsGreaterThanToken */) {
+                return parseSimpleArrowFunctionExpression(pos, expr, /*asyncModifier*/ undefined);
+            }
+            // Now see if we might be in cases '2' or '3'.
+            // If the expression was a LHS expression, and we have an assignment operator, then
+            // we're in '2' or '3'. Consume the assignment and return.
+            //
+            // Note: we call reScanGreaterToken so that we get an appropriately merged token
+            // for cases like `> > =` becoming `>>=`
+            if (ts.isLeftHandSideExpression(expr) && ts.isAssignmentOperator(reScanGreaterToken())) {
+                return makeBinaryExpression(expr, parseTokenNode(), parseAssignmentExpressionOrHigher(), pos);
+            }
+            // It wasn't an assignment or a lambda.  This is a conditional expression:
+            return parseConditionalExpressionRest(expr, pos);
+        }
+        function isYieldExpression() {
+            if (token() === 124 /* YieldKeyword */) {
+                // If we have a 'yield' keyword, and this is a context where yield expressions are
+                // allowed, then definitely parse out a yield expression.
+                if (inYieldContext()) {
+                    return true;
+                }
+                // We're in a context where 'yield expr' is not allowed.  However, if we can
+                // definitely tell that the user was trying to parse a 'yield expr' and not
+                // just a normal expr that start with a 'yield' identifier, then parse out
+                // a 'yield expr'.  We can then report an error later that they are only
+                // allowed in generator expressions.
+                //
+                // for example, if we see 'yield(foo)', then we'll have to treat that as an
+                // invocation expression of something called 'yield'.  However, if we have
+                // 'yield foo' then that is not legal as a normal expression, so we can
+                // definitely recognize this as a yield expression.
+                //
+                // for now we just check if the next token is an identifier.  More heuristics
+                // can be added here later as necessary.  We just need to make sure that we
+                // don't accidentally consume something legal.
+                return lookAhead(nextTokenIsIdentifierOrKeywordOrLiteralOnSameLine);
+            }
+            return false;
+        }
+        function nextTokenIsIdentifierOnSameLine() {
+            nextToken();
+            return !scanner.hasPrecedingLineBreak() && isIdentifier();
+        }
+        function parseYieldExpression() {
+            var pos = getNodePos();
+            // YieldExpression[In] :
+            //      yield
+            //      yield [no LineTerminator here] [Lexical goal InputElementRegExp]AssignmentExpression[?In, Yield]
+            //      yield [no LineTerminator here] * [Lexical goal InputElementRegExp]AssignmentExpression[?In, Yield]
+            nextToken();
+            if (!scanner.hasPrecedingLineBreak() &&
+                (token() === 41 /* AsteriskToken */ || isStartOfExpression())) {
+                return finishNode(factory.createYieldExpression(parseOptionalToken(41 /* AsteriskToken */), parseAssignmentExpressionOrHigher()), pos);
+            }
+            else {
+                // if the next token is not on the same line as yield.  or we don't have an '*' or
+                // the start of an expression, then this is just a simple "yield" expression.
+                return finishNode(factory.createYieldExpression(/*asteriskToken*/ undefined, /*expression*/ undefined), pos);
+            }
+        }
+        function parseSimpleArrowFunctionExpression(pos, identifier, asyncModifier) {
+            ts.Debug.assert(token() === 38 /* EqualsGreaterThanToken */, "parseSimpleArrowFunctionExpression should only have been called if we had a =>");
+            var parameter = factory.createParameterDeclaration(
+            /*decorators*/ undefined, 
+            /*modifiers*/ undefined, 
+            /*dotDotDotToken*/ undefined, identifier, 
+            /*questionToken*/ undefined, 
+            /*type*/ undefined, 
+            /*initializer*/ undefined);
+            finishNode(parameter, identifier.pos);
+            var parameters = createNodeArray([parameter], parameter.pos, parameter.end);
+            var equalsGreaterThanToken = parseExpectedToken(38 /* EqualsGreaterThanToken */);
+            var body = parseArrowFunctionExpressionBody(/*isAsync*/ !!asyncModifier);
+            var node = factory.createArrowFunction(asyncModifier, /*typeParameters*/ undefined, parameters, /*type*/ undefined, equalsGreaterThanToken, body);
+            return addJSDocComment(finishNode(node, pos));
+        }
+        function tryParseParenthesizedArrowFunctionExpression() {
+            var triState = isParenthesizedArrowFunctionExpression();
+            if (triState === 0 /* False */) {
+                // It's definitely not a parenthesized arrow function expression.
+                return undefined;
+            }
+            // If we definitely have an arrow function, then we can just parse one, not requiring a
+            // following => or { token. Otherwise, we *might* have an arrow function.  Try to parse
+            // it out, but don't allow any ambiguity, and return 'undefined' if this could be an
+            // expression instead.
+            return triState === 1 /* True */ ?
+                parseParenthesizedArrowFunctionExpression(/*allowAmbiguity*/ true) :
+                tryParse(parsePossibleParenthesizedArrowFunctionExpression);
+        }
+        //  True        -> We definitely expect a parenthesized arrow function here.
+        //  False       -> There *cannot* be a parenthesized arrow function here.
+        //  Unknown     -> There *might* be a parenthesized arrow function here.
+        //                 Speculatively look ahead to be sure, and rollback if not.
+        function isParenthesizedArrowFunctionExpression() {
+            if (token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */ || token() === 129 /* AsyncKeyword */) {
+                return lookAhead(isParenthesizedArrowFunctionExpressionWorker);
+            }
+            if (token() === 38 /* EqualsGreaterThanToken */) {
+                // ERROR RECOVERY TWEAK:
+                // If we see a standalone => try to parse it as an arrow function expression as that's
+                // likely what the user intended to write.
+                return 1 /* True */;
+            }
+            // Definitely not a parenthesized arrow function.
+            return 0 /* False */;
+        }
+        function isParenthesizedArrowFunctionExpressionWorker() {
+            if (token() === 129 /* AsyncKeyword */) {
+                nextToken();
+                if (scanner.hasPrecedingLineBreak()) {
+                    return 0 /* False */;
+                }
+                if (token() !== 20 /* OpenParenToken */ && token() !== 29 /* LessThanToken */) {
+                    return 0 /* False */;
+                }
+            }
+            var first = token();
+            var second = nextToken();
+            if (first === 20 /* OpenParenToken */) {
+                if (second === 21 /* CloseParenToken */) {
+                    // Simple cases: "() =>", "(): ", and "() {".
+                    // This is an arrow function with no parameters.
+                    // The last one is not actually an arrow function,
+                    // but this is probably what the user intended.
+                    var third = nextToken();
+                    switch (third) {
+                        case 38 /* EqualsGreaterThanToken */:
+                        case 58 /* ColonToken */:
+                        case 18 /* OpenBraceToken */:
+                            return 1 /* True */;
+                        default:
+                            return 0 /* False */;
+                    }
+                }
+                // If encounter "([" or "({", this could be the start of a binding pattern.
+                // Examples:
+                //      ([ x ]) => { }
+                //      ({ x }) => { }
+                //      ([ x ])
+                //      ({ x })
+                if (second === 22 /* OpenBracketToken */ || second === 18 /* OpenBraceToken */) {
+                    return 2 /* Unknown */;
+                }
+                // Simple case: "(..."
+                // This is an arrow function with a rest parameter.
+                if (second === 25 /* DotDotDotToken */) {
+                    return 1 /* True */;
+                }
+                // Check for "(xxx yyy", where xxx is a modifier and yyy is an identifier. This
+                // isn't actually allowed, but we want to treat it as a lambda so we can provide
+                // a good error message.
+                if (ts.isModifierKind(second) && second !== 129 /* AsyncKeyword */ && lookAhead(nextTokenIsIdentifier)) {
+                    return 1 /* True */;
+                }
+                // If we had "(" followed by something that's not an identifier,
+                // then this definitely doesn't look like a lambda.  "this" is not
+                // valid, but we want to parse it and then give a semantic error.
+                if (!isIdentifier() && second !== 107 /* ThisKeyword */) {
+                    return 0 /* False */;
+                }
+                switch (nextToken()) {
+                    case 58 /* ColonToken */:
+                        // If we have something like "(a:", then we must have a
+                        // type-annotated parameter in an arrow function expression.
+                        return 1 /* True */;
+                    case 57 /* QuestionToken */:
+                        nextToken();
+                        // If we have "(a?:" or "(a?," or "(a?=" or "(a?)" then it is definitely a lambda.
+                        if (token() === 58 /* ColonToken */ || token() === 27 /* CommaToken */ || token() === 62 /* EqualsToken */ || token() === 21 /* CloseParenToken */) {
+                            return 1 /* True */;
+                        }
+                        // Otherwise it is definitely not a lambda.
+                        return 0 /* False */;
+                    case 27 /* CommaToken */:
+                    case 62 /* EqualsToken */:
+                    case 21 /* CloseParenToken */:
+                        // If we have "(a," or "(a=" or "(a)" this *could* be an arrow function
+                        return 2 /* Unknown */;
+                }
+                // It is definitely not an arrow function
+                return 0 /* False */;
             }
-        },
-        {
-            name: "exclude",
-            type: "list",
-            element: {
-                name: "exclude",
-                type: "string"
+            else {
+                ts.Debug.assert(first === 29 /* LessThanToken */);
+                // If we have "<" not followed by an identifier,
+                // then this definitely is not an arrow function.
+                if (!isIdentifier()) {
+                    return 0 /* False */;
+                }
+                // JSX overrides
+                if (languageVariant === 1 /* JSX */) {
+                    var isArrowFunctionInJsx = lookAhead(function () {
+                        var third = nextToken();
+                        if (third === 93 /* ExtendsKeyword */) {
+                            var fourth = nextToken();
+                            switch (fourth) {
+                                case 62 /* EqualsToken */:
+                                case 31 /* GreaterThanToken */:
+                                    return false;
+                                default:
+                                    return true;
+                            }
+                        }
+                        else if (third === 27 /* CommaToken */) {
+                            return true;
+                        }
+                        return false;
+                    });
+                    if (isArrowFunctionInJsx) {
+                        return 1 /* True */;
+                    }
+                    return 0 /* False */;
+                }
+                // This *could* be a parenthesized arrow function.
+                return 2 /* Unknown */;
             }
         }
-    ];
-    /*@internal*/
-    function createOptionNameMap(optionDeclarations) {
-        var optionsNameMap = ts.createMap();
-        var shortOptionNames = ts.createMap();
-        ts.forEach(optionDeclarations, function (option) {
-            optionsNameMap.set(option.name.toLowerCase(), option);
-            if (option.shortName) {
-                shortOptionNames.set(option.shortName, option.name);
+        function parsePossibleParenthesizedArrowFunctionExpression() {
+            var tokenPos = scanner.getTokenPos();
+            if (notParenthesizedArrow === null || notParenthesizedArrow === void 0 ? void 0 : notParenthesizedArrow.has(tokenPos)) {
+                return undefined;
             }
-        });
-        return { optionsNameMap: optionsNameMap, shortOptionNames: shortOptionNames };
-    }
-    ts.createOptionNameMap = createOptionNameMap;
-    var optionsNameMapCache;
-    /* @internal */
-    function getOptionsNameMap() {
-        return optionsNameMapCache || (optionsNameMapCache = createOptionNameMap(ts.optionDeclarations));
-    }
-    ts.getOptionsNameMap = getOptionsNameMap;
-    /* @internal */
-    ts.defaultInitCompilerOptions = {
-        module: ts.ModuleKind.CommonJS,
-        target: 1 /* ES5 */,
-        strict: true,
-        esModuleInterop: true,
-        forceConsistentCasingInFileNames: true,
-        skipLibCheck: true
-    };
-    /* @internal */
-    function convertEnableAutoDiscoveryToEnable(typeAcquisition) {
-        // Convert deprecated typingOptions.enableAutoDiscovery to typeAcquisition.enable
-        if (typeAcquisition && typeAcquisition.enableAutoDiscovery !== undefined && typeAcquisition.enable === undefined) {
-            return {
-                enable: typeAcquisition.enableAutoDiscovery,
-                include: typeAcquisition.include || [],
-                exclude: typeAcquisition.exclude || []
-            };
+            var result = parseParenthesizedArrowFunctionExpression(/*allowAmbiguity*/ false);
+            if (!result) {
+                (notParenthesizedArrow || (notParenthesizedArrow = new ts.Set())).add(tokenPos);
+            }
+            return result;
         }
-        return typeAcquisition;
-    }
-    ts.convertEnableAutoDiscoveryToEnable = convertEnableAutoDiscoveryToEnable;
-    /* @internal */
-    function createCompilerDiagnosticForInvalidCustomType(opt) {
-        return createDiagnosticForInvalidCustomType(opt, ts.createCompilerDiagnostic);
-    }
-    ts.createCompilerDiagnosticForInvalidCustomType = createCompilerDiagnosticForInvalidCustomType;
-    function createDiagnosticForInvalidCustomType(opt, createDiagnostic) {
-        var namesOfType = ts.arrayFrom(opt.type.keys()).map(function (key) { return "'" + key + "'"; }).join(", ");
-        return createDiagnostic(ts.Diagnostics.Argument_for_0_option_must_be_Colon_1, "--" + opt.name, namesOfType);
-    }
-    /* @internal */
-    function parseCustomTypeOption(opt, value, errors) {
-        return convertJsonOptionOfCustomType(opt, trimString(value || ""), errors);
-    }
-    ts.parseCustomTypeOption = parseCustomTypeOption;
-    /* @internal */
-    function parseListTypeOption(opt, value, errors) {
-        if (value === void 0) { value = ""; }
-        value = trimString(value);
-        if (ts.startsWith(value, "-")) {
+        function tryParseAsyncSimpleArrowFunctionExpression() {
+            // We do a check here so that we won't be doing unnecessarily call to "lookAhead"
+            if (token() === 129 /* AsyncKeyword */) {
+                if (lookAhead(isUnParenthesizedAsyncArrowFunctionWorker) === 1 /* True */) {
+                    var pos = getNodePos();
+                    var asyncModifier = parseModifiersForArrowFunction();
+                    var expr = parseBinaryExpressionOrHigher(0 /* Lowest */);
+                    return parseSimpleArrowFunctionExpression(pos, expr, asyncModifier);
+                }
+            }
             return undefined;
         }
-        if (value === "") {
-            return [];
-        }
-        var values = value.split(",");
-        switch (opt.element.type) {
-            case "number":
-                return ts.map(values, parseInt);
-            case "string":
-                return ts.map(values, function (v) { return v || ""; });
-            default:
-                return ts.mapDefined(values, function (v) { return parseCustomTypeOption(opt.element, v, errors); });
-        }
-    }
-    ts.parseListTypeOption = parseListTypeOption;
-    function getOptionName(option) {
-        return option.name;
-    }
-    function createUnknownOptionError(unknownOption, diagnostics, createDiagnostics, unknownOptionErrorText) {
-        var possibleOption = ts.getSpellingSuggestion(unknownOption, diagnostics.optionDeclarations, getOptionName);
-        return possibleOption ?
-            createDiagnostics(diagnostics.unknownDidYouMeanDiagnostic, unknownOptionErrorText || unknownOption, possibleOption.name) :
-            createDiagnostics(diagnostics.unknownOptionDiagnostic, unknownOptionErrorText || unknownOption);
-    }
-    /*@internal*/
-    function parseCommandLineWorker(diagnostics, commandLine, readFile) {
-        var options = {};
-        var watchOptions;
-        var fileNames = [];
-        var errors = [];
-        parseStrings(commandLine);
-        return {
-            options: options,
-            watchOptions: watchOptions,
-            fileNames: fileNames,
-            errors: errors
-        };
-        function parseStrings(args) {
-            var i = 0;
-            while (i < args.length) {
-                var s = args[i];
-                i++;
-                if (s.charCodeAt(0) === 64 /* at */) {
-                    parseResponseFile(s.slice(1));
+        function isUnParenthesizedAsyncArrowFunctionWorker() {
+            // AsyncArrowFunctionExpression:
+            //      1) async[no LineTerminator here]AsyncArrowBindingIdentifier[?Yield][no LineTerminator here]=>AsyncConciseBody[?In]
+            //      2) CoverCallExpressionAndAsyncArrowHead[?Yield, ?Await][no LineTerminator here]=>AsyncConciseBody[?In]
+            if (token() === 129 /* AsyncKeyword */) {
+                nextToken();
+                // If the "async" is followed by "=>" token then it is not a beginning of an async arrow-function
+                // but instead a simple arrow-function which will be parsed inside "parseAssignmentExpressionOrHigher"
+                if (scanner.hasPrecedingLineBreak() || token() === 38 /* EqualsGreaterThanToken */) {
+                    return 0 /* False */;
                 }
-                else if (s.charCodeAt(0) === 45 /* minus */) {
-                    var inputOptionName = s.slice(s.charCodeAt(1) === 45 /* minus */ ? 2 : 1);
-                    var opt = getOptionDeclarationFromName(diagnostics.getOptionsNameMap, inputOptionName, /*allowShort*/ true);
-                    if (opt) {
-                        i = parseOptionValue(args, i, diagnostics, opt, options, errors);
-                    }
-                    else {
-                        var watchOpt = getOptionDeclarationFromName(watchOptionsDidYouMeanDiagnostics.getOptionsNameMap, inputOptionName, /*allowShort*/ true);
-                        if (watchOpt) {
-                            i = parseOptionValue(args, i, watchOptionsDidYouMeanDiagnostics, watchOpt, watchOptions || (watchOptions = {}), errors);
-                        }
-                        else {
-                            errors.push(createUnknownOptionError(inputOptionName, diagnostics, ts.createCompilerDiagnostic, s));
-                        }
-                    }
+                // Check for un-parenthesized AsyncArrowFunction
+                var expr = parseBinaryExpressionOrHigher(0 /* Lowest */);
+                if (!scanner.hasPrecedingLineBreak() && expr.kind === 78 /* Identifier */ && token() === 38 /* EqualsGreaterThanToken */) {
+                    return 1 /* True */;
                 }
-                else {
-                    fileNames.push(s);
+            }
+            return 0 /* False */;
+        }
+        function parseParenthesizedArrowFunctionExpression(allowAmbiguity) {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var modifiers = parseModifiersForArrowFunction();
+            var isAsync = ts.some(modifiers, ts.isAsyncModifier) ? 2 /* Await */ : 0 /* None */;
+            // Arrow functions are never generators.
+            //
+            // If we're speculatively parsing a signature for a parenthesized arrow function, then
+            // we have to have a complete parameter list.  Otherwise we might see something like
+            // a => (b => c)
+            // And think that "(b =>" was actually a parenthesized arrow function with a missing
+            // close paren.
+            var typeParameters = parseTypeParameters();
+            var parameters;
+            if (!parseExpected(20 /* OpenParenToken */)) {
+                if (!allowAmbiguity) {
+                    return undefined;
+                }
+                parameters = createMissingList();
+            }
+            else {
+                parameters = parseParametersWorker(isAsync);
+                if (!parseExpected(21 /* CloseParenToken */) && !allowAmbiguity) {
+                    return undefined;
                 }
             }
+            var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+            if (type && !allowAmbiguity && typeHasArrowFunctionBlockingParseError(type)) {
+                return undefined;
+            }
+            // Parsing a signature isn't enough.
+            // Parenthesized arrow signatures often look like other valid expressions.
+            // For instance:
+            //  - "(x = 10)" is an assignment expression parsed as a signature with a default parameter value.
+            //  - "(x,y)" is a comma expression parsed as a signature with two parameters.
+            //  - "a ? (b): c" will have "(b):" parsed as a signature with a return type annotation.
+            //  - "a ? (b): function() {}" will too, since function() is a valid JSDoc function type.
+            //
+            // So we need just a bit of lookahead to ensure that it can only be a signature.
+            var hasJSDocFunctionType = type && ts.isJSDocFunctionType(type);
+            if (!allowAmbiguity && token() !== 38 /* EqualsGreaterThanToken */ && (hasJSDocFunctionType || token() !== 18 /* OpenBraceToken */)) {
+                // Returning undefined here will cause our caller to rewind to where we started from.
+                return undefined;
+            }
+            // If we have an arrow, then try to parse the body. Even if not, try to parse if we
+            // have an opening brace, just in case we're in an error state.
+            var lastToken = token();
+            var equalsGreaterThanToken = parseExpectedToken(38 /* EqualsGreaterThanToken */);
+            var body = (lastToken === 38 /* EqualsGreaterThanToken */ || lastToken === 18 /* OpenBraceToken */)
+                ? parseArrowFunctionExpressionBody(ts.some(modifiers, ts.isAsyncModifier))
+                : parseIdentifier();
+            var node = factory.createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        function parseResponseFile(fileName) {
-            var text = tryReadFile(fileName, readFile || (function (fileName) { return ts.sys.readFile(fileName); }));
-            if (!ts.isString(text)) {
-                errors.push(text);
-                return;
+        function parseArrowFunctionExpressionBody(isAsync) {
+            if (token() === 18 /* OpenBraceToken */) {
+                return parseFunctionBlock(isAsync ? 2 /* Await */ : 0 /* None */);
             }
-            var args = [];
-            var pos = 0;
+            if (token() !== 26 /* SemicolonToken */ &&
+                token() !== 97 /* FunctionKeyword */ &&
+                token() !== 83 /* ClassKeyword */ &&
+                isStartOfStatement() &&
+                !isStartOfExpressionStatement()) {
+                // Check if we got a plain statement (i.e. no expression-statements, no function/class expressions/declarations)
+                //
+                // Here we try to recover from a potential error situation in the case where the
+                // user meant to supply a block. For example, if the user wrote:
+                //
+                //  a =>
+                //      let v = 0;
+                //  }
+                //
+                // they may be missing an open brace.  Check to see if that's the case so we can
+                // try to recover better.  If we don't do this, then the next close curly we see may end
+                // up preemptively closing the containing construct.
+                //
+                // Note: even when 'IgnoreMissingOpenBrace' is passed, parseBody will still error.
+                return parseFunctionBlock(16 /* IgnoreMissingOpenBrace */ | (isAsync ? 2 /* Await */ : 0 /* None */));
+            }
+            var savedTopLevel = topLevel;
+            topLevel = false;
+            var node = isAsync
+                ? doInAwaitContext(parseAssignmentExpressionOrHigher)
+                : doOutsideOfAwaitContext(parseAssignmentExpressionOrHigher);
+            topLevel = savedTopLevel;
+            return node;
+        }
+        function parseConditionalExpressionRest(leftOperand, pos) {
+            // Note: we are passed in an expression which was produced from parseBinaryExpressionOrHigher.
+            var questionToken = parseOptionalToken(57 /* QuestionToken */);
+            if (!questionToken) {
+                return leftOperand;
+            }
+            // Note: we explicitly 'allowIn' in the whenTrue part of the condition expression, and
+            // we do not that for the 'whenFalse' part.
+            var colonToken;
+            return finishNode(factory.createConditionalExpression(leftOperand, questionToken, doOutsideOfContext(disallowInAndDecoratorContext, parseAssignmentExpressionOrHigher), colonToken = parseExpectedToken(58 /* ColonToken */), ts.nodeIsPresent(colonToken)
+                ? parseAssignmentExpressionOrHigher()
+                : createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ false, ts.Diagnostics._0_expected, ts.tokenToString(58 /* ColonToken */))), pos);
+        }
+        function parseBinaryExpressionOrHigher(precedence) {
+            var pos = getNodePos();
+            var leftOperand = parseUnaryExpressionOrHigher();
+            return parseBinaryExpressionRest(precedence, leftOperand, pos);
+        }
+        function isInOrOfKeyword(t) {
+            return t === 100 /* InKeyword */ || t === 156 /* OfKeyword */;
+        }
+        function parseBinaryExpressionRest(precedence, leftOperand, pos) {
             while (true) {
-                while (pos < text.length && text.charCodeAt(pos) <= 32 /* space */)
-                    pos++;
-                if (pos >= text.length)
+                // We either have a binary operator here, or we're finished.  We call
+                // reScanGreaterToken so that we merge token sequences like > and = into >=
+                reScanGreaterToken();
+                var newPrecedence = ts.getBinaryOperatorPrecedence(token());
+                // Check the precedence to see if we should "take" this operator
+                // - For left associative operator (all operator but **), consume the operator,
+                //   recursively call the function below, and parse binaryExpression as a rightOperand
+                //   of the caller if the new precedence of the operator is greater then or equal to the current precedence.
+                //   For example:
+                //      a - b - c;
+                //            ^token; leftOperand = b. Return b to the caller as a rightOperand
+                //      a * b - c
+                //            ^token; leftOperand = b. Return b to the caller as a rightOperand
+                //      a - b * c;
+                //            ^token; leftOperand = b. Return b * c to the caller as a rightOperand
+                // - For right associative operator (**), consume the operator, recursively call the function
+                //   and parse binaryExpression as a rightOperand of the caller if the new precedence of
+                //   the operator is strictly grater than the current precedence
+                //   For example:
+                //      a ** b ** c;
+                //             ^^token; leftOperand = b. Return b ** c to the caller as a rightOperand
+                //      a - b ** c;
+                //            ^^token; leftOperand = b. Return b ** c to the caller as a rightOperand
+                //      a ** b - c
+                //             ^token; leftOperand = b. Return b to the caller as a rightOperand
+                var consumeCurrentOperator = token() === 42 /* AsteriskAsteriskToken */ ?
+                    newPrecedence >= precedence :
+                    newPrecedence > precedence;
+                if (!consumeCurrentOperator) {
                     break;
-                var start = pos;
-                if (text.charCodeAt(start) === 34 /* doubleQuote */) {
-                    pos++;
-                    while (pos < text.length && text.charCodeAt(pos) !== 34 /* doubleQuote */)
-                        pos++;
-                    if (pos < text.length) {
-                        args.push(text.substring(start + 1, pos));
-                        pos++;
+                }
+                if (token() === 100 /* InKeyword */ && inDisallowInContext()) {
+                    break;
+                }
+                if (token() === 126 /* AsKeyword */) {
+                    // Make sure we *do* perform ASI for constructs like this:
+                    //    var x = foo
+                    //    as (Bar)
+                    // This should be parsed as an initialized variable, followed
+                    // by a function call to 'as' with the argument 'Bar'
+                    if (scanner.hasPrecedingLineBreak()) {
+                        break;
                     }
                     else {
-                        errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Unterminated_quoted_string_in_response_file_0, fileName));
+                        nextToken();
+                        leftOperand = makeAsExpression(leftOperand, parseType());
                     }
                 }
                 else {
-                    while (text.charCodeAt(pos) > 32 /* space */)
-                        pos++;
-                    args.push(text.substring(start, pos));
+                    leftOperand = makeBinaryExpression(leftOperand, parseTokenNode(), parseBinaryExpressionOrHigher(newPrecedence), pos);
                 }
             }
-            parseStrings(args);
+            return leftOperand;
         }
-    }
-    ts.parseCommandLineWorker = parseCommandLineWorker;
-    function parseOptionValue(args, i, diagnostics, opt, options, errors) {
-        if (opt.isTSConfigOnly) {
-            var optValue = args[i];
-            if (optValue === "null") {
-                options[opt.name] = undefined;
-                i++;
+        function isBinaryOperator() {
+            if (inDisallowInContext() && token() === 100 /* InKeyword */) {
+                return false;
             }
-            else if (opt.type === "boolean") {
-                if (optValue === "false") {
-                    options[opt.name] = false;
-                    i++;
+            return ts.getBinaryOperatorPrecedence(token()) > 0;
+        }
+        function makeBinaryExpression(left, operatorToken, right, pos) {
+            return finishNode(factory.createBinaryExpression(left, operatorToken, right), pos);
+        }
+        function makeAsExpression(left, right) {
+            return finishNode(factory.createAsExpression(left, right), left.pos);
+        }
+        function parsePrefixUnaryExpression() {
+            var pos = getNodePos();
+            return finishNode(factory.createPrefixUnaryExpression(token(), nextTokenAnd(parseSimpleUnaryExpression)), pos);
+        }
+        function parseDeleteExpression() {
+            var pos = getNodePos();
+            return finishNode(factory.createDeleteExpression(nextTokenAnd(parseSimpleUnaryExpression)), pos);
+        }
+        function parseTypeOfExpression() {
+            var pos = getNodePos();
+            return finishNode(factory.createTypeOfExpression(nextTokenAnd(parseSimpleUnaryExpression)), pos);
+        }
+        function parseVoidExpression() {
+            var pos = getNodePos();
+            return finishNode(factory.createVoidExpression(nextTokenAnd(parseSimpleUnaryExpression)), pos);
+        }
+        function isAwaitExpression() {
+            if (token() === 130 /* AwaitKeyword */) {
+                if (inAwaitContext()) {
+                    return true;
+                }
+                // here we are using similar heuristics as 'isYieldExpression'
+                return lookAhead(nextTokenIsIdentifierOrKeywordOrLiteralOnSameLine);
+            }
+            return false;
+        }
+        function parseAwaitExpression() {
+            var pos = getNodePos();
+            return finishNode(factory.createAwaitExpression(nextTokenAnd(parseSimpleUnaryExpression)), pos);
+        }
+        /**
+         * Parse ES7 exponential expression and await expression
+         *
+         * ES7 ExponentiationExpression:
+         *      1) UnaryExpression[?Yield]
+         *      2) UpdateExpression[?Yield] ** ExponentiationExpression[?Yield]
+         *
+         */
+        function parseUnaryExpressionOrHigher() {
+            /**
+             * ES7 UpdateExpression:
+             *      1) LeftHandSideExpression[?Yield]
+             *      2) LeftHandSideExpression[?Yield][no LineTerminator here]++
+             *      3) LeftHandSideExpression[?Yield][no LineTerminator here]--
+             *      4) ++UnaryExpression[?Yield]
+             *      5) --UnaryExpression[?Yield]
+             */
+            if (isUpdateExpression()) {
+                var pos = getNodePos();
+                var updateExpression = parseUpdateExpression();
+                return token() === 42 /* AsteriskAsteriskToken */ ?
+                    parseBinaryExpressionRest(ts.getBinaryOperatorPrecedence(token()), updateExpression, pos) :
+                    updateExpression;
+            }
+            /**
+             * ES7 UnaryExpression:
+             *      1) UpdateExpression[?yield]
+             *      2) delete UpdateExpression[?yield]
+             *      3) void UpdateExpression[?yield]
+             *      4) typeof UpdateExpression[?yield]
+             *      5) + UpdateExpression[?yield]
+             *      6) - UpdateExpression[?yield]
+             *      7) ~ UpdateExpression[?yield]
+             *      8) ! UpdateExpression[?yield]
+             */
+            var unaryOperator = token();
+            var simpleUnaryExpression = parseSimpleUnaryExpression();
+            if (token() === 42 /* AsteriskAsteriskToken */) {
+                var pos = ts.skipTrivia(sourceText, simpleUnaryExpression.pos);
+                var end = simpleUnaryExpression.end;
+                if (simpleUnaryExpression.kind === 206 /* TypeAssertionExpression */) {
+                    parseErrorAt(pos, end, ts.Diagnostics.A_type_assertion_expression_is_not_allowed_in_the_left_hand_side_of_an_exponentiation_expression_Consider_enclosing_the_expression_in_parentheses);
                 }
                 else {
-                    if (optValue === "true")
-                        i++;
-                    errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Option_0_can_only_be_specified_in_tsconfig_json_file_or_set_to_false_or_null_on_command_line, opt.name));
+                    parseErrorAt(pos, end, ts.Diagnostics.An_unary_expression_with_the_0_operator_is_not_allowed_in_the_left_hand_side_of_an_exponentiation_expression_Consider_enclosing_the_expression_in_parentheses, ts.tokenToString(unaryOperator));
                 }
             }
-            else {
-                errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Option_0_can_only_be_specified_in_tsconfig_json_file_or_set_to_null_on_command_line, opt.name));
-                if (optValue && !ts.startsWith(optValue, "-"))
-                    i++;
+            return simpleUnaryExpression;
+        }
+        /**
+         * Parse ES7 simple-unary expression or higher:
+         *
+         * ES7 UnaryExpression:
+         *      1) UpdateExpression[?yield]
+         *      2) delete UnaryExpression[?yield]
+         *      3) void UnaryExpression[?yield]
+         *      4) typeof UnaryExpression[?yield]
+         *      5) + UnaryExpression[?yield]
+         *      6) - UnaryExpression[?yield]
+         *      7) ~ UnaryExpression[?yield]
+         *      8) ! UnaryExpression[?yield]
+         *      9) [+Await] await UnaryExpression[?yield]
+         */
+        function parseSimpleUnaryExpression() {
+            switch (token()) {
+                case 39 /* PlusToken */:
+                case 40 /* MinusToken */:
+                case 54 /* TildeToken */:
+                case 53 /* ExclamationToken */:
+                    return parsePrefixUnaryExpression();
+                case 88 /* DeleteKeyword */:
+                    return parseDeleteExpression();
+                case 111 /* TypeOfKeyword */:
+                    return parseTypeOfExpression();
+                case 113 /* VoidKeyword */:
+                    return parseVoidExpression();
+                case 29 /* LessThanToken */:
+                    // This is modified UnaryExpression grammar in TypeScript
+                    //  UnaryExpression (modified):
+                    //      < type > UnaryExpression
+                    return parseTypeAssertion();
+                case 130 /* AwaitKeyword */:
+                    if (isAwaitExpression()) {
+                        return parseAwaitExpression();
+                    }
+                // falls through
+                default:
+                    return parseUpdateExpression();
             }
         }
-        else {
-            // Check to see if no argument was provided (e.g. "--locale" is the last command-line argument).
-            if (!args[i] && opt.type !== "boolean") {
-                errors.push(ts.createCompilerDiagnostic(diagnostics.optionTypeMismatchDiagnostic, opt.name, getCompilerOptionValueTypeString(opt)));
+        /**
+         * Check if the current token can possibly be an ES7 increment expression.
+         *
+         * ES7 UpdateExpression:
+         *      LeftHandSideExpression[?Yield]
+         *      LeftHandSideExpression[?Yield][no LineTerminator here]++
+         *      LeftHandSideExpression[?Yield][no LineTerminator here]--
+         *      ++LeftHandSideExpression[?Yield]
+         *      --LeftHandSideExpression[?Yield]
+         */
+        function isUpdateExpression() {
+            // This function is called inside parseUnaryExpression to decide
+            // whether to call parseSimpleUnaryExpression or call parseUpdateExpression directly
+            switch (token()) {
+                case 39 /* PlusToken */:
+                case 40 /* MinusToken */:
+                case 54 /* TildeToken */:
+                case 53 /* ExclamationToken */:
+                case 88 /* DeleteKeyword */:
+                case 111 /* TypeOfKeyword */:
+                case 113 /* VoidKeyword */:
+                case 130 /* AwaitKeyword */:
+                    return false;
+                case 29 /* LessThanToken */:
+                    // If we are not in JSX context, we are parsing TypeAssertion which is an UnaryExpression
+                    if (languageVariant !== 1 /* JSX */) {
+                        return false;
+                    }
+                // We are in JSX context and the token is part of JSXElement.
+                // falls through
+                default:
+                    return true;
             }
-            if (args[i] !== "null") {
-                switch (opt.type) {
-                    case "number":
-                        options[opt.name] = parseInt(args[i]);
-                        i++;
-                        break;
-                    case "boolean":
-                        // boolean flag has optional value true, false, others
-                        var optValue = args[i];
-                        options[opt.name] = optValue !== "false";
-                        // consume next argument as boolean flag value
-                        if (optValue === "false" || optValue === "true") {
-                            i++;
-                        }
-                        break;
-                    case "string":
-                        options[opt.name] = args[i] || "";
-                        i++;
-                        break;
-                    case "list":
-                        var result = parseListTypeOption(opt, args[i], errors);
-                        options[opt.name] = result || [];
-                        if (result) {
-                            i++;
-                        }
-                        break;
-                    // If not a primitive, the possible types are specified in what is effectively a map of options.
-                    default:
-                        options[opt.name] = parseCustomTypeOption(opt, args[i], errors);
-                        i++;
-                        break;
-                }
+        }
+        /**
+         * Parse ES7 UpdateExpression. UpdateExpression is used instead of ES6's PostFixExpression.
+         *
+         * ES7 UpdateExpression[yield]:
+         *      1) LeftHandSideExpression[?yield]
+         *      2) LeftHandSideExpression[?yield] [[no LineTerminator here]]++
+         *      3) LeftHandSideExpression[?yield] [[no LineTerminator here]]--
+         *      4) ++LeftHandSideExpression[?yield]
+         *      5) --LeftHandSideExpression[?yield]
+         * In TypeScript (2), (3) are parsed as PostfixUnaryExpression. (4), (5) are parsed as PrefixUnaryExpression
+         */
+        function parseUpdateExpression() {
+            if (token() === 45 /* PlusPlusToken */ || token() === 46 /* MinusMinusToken */) {
+                var pos = getNodePos();
+                return finishNode(factory.createPrefixUnaryExpression(token(), nextTokenAnd(parseLeftHandSideExpressionOrHigher)), pos);
             }
-            else {
-                options[opt.name] = undefined;
-                i++;
+            else if (languageVariant === 1 /* JSX */ && token() === 29 /* LessThanToken */ && lookAhead(nextTokenIsIdentifierOrKeywordOrGreaterThan)) {
+                // JSXElement is part of primaryExpression
+                return parseJsxElementOrSelfClosingElementOrFragment(/*inExpressionContext*/ true);
+            }
+            var expression = parseLeftHandSideExpressionOrHigher();
+            ts.Debug.assert(ts.isLeftHandSideExpression(expression));
+            if ((token() === 45 /* PlusPlusToken */ || token() === 46 /* MinusMinusToken */) && !scanner.hasPrecedingLineBreak()) {
+                var operator = token();
+                nextToken();
+                return finishNode(factory.createPostfixUnaryExpression(expression, operator), expression.pos);
             }
+            return expression;
         }
-        return i;
-    }
-    /*@internal*/
-    ts.compilerOptionsDidYouMeanDiagnostics = {
-        getOptionsNameMap: getOptionsNameMap,
-        optionDeclarations: ts.optionDeclarations,
-        unknownOptionDiagnostic: ts.Diagnostics.Unknown_compiler_option_0,
-        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_compiler_option_0_Did_you_mean_1,
-        optionTypeMismatchDiagnostic: ts.Diagnostics.Compiler_option_0_expects_an_argument
-    };
-    function parseCommandLine(commandLine, readFile) {
-        return parseCommandLineWorker(ts.compilerOptionsDidYouMeanDiagnostics, commandLine, readFile);
-    }
-    ts.parseCommandLine = parseCommandLine;
-    /** @internal */
-    function getOptionFromName(optionName, allowShort) {
-        return getOptionDeclarationFromName(getOptionsNameMap, optionName, allowShort);
-    }
-    ts.getOptionFromName = getOptionFromName;
-    function getOptionDeclarationFromName(getOptionNameMap, optionName, allowShort) {
-        if (allowShort === void 0) { allowShort = false; }
-        optionName = optionName.toLowerCase();
-        var _a = getOptionNameMap(), optionsNameMap = _a.optionsNameMap, shortOptionNames = _a.shortOptionNames;
-        // Try to translate short option names to their full equivalents.
-        if (allowShort) {
-            var short = shortOptionNames.get(optionName);
-            if (short !== undefined) {
-                optionName = short;
+        function parseLeftHandSideExpressionOrHigher() {
+            // Original Ecma:
+            // LeftHandSideExpression: See 11.2
+            //      NewExpression
+            //      CallExpression
+            //
+            // Our simplification:
+            //
+            // LeftHandSideExpression: See 11.2
+            //      MemberExpression
+            //      CallExpression
+            //
+            // See comment in parseMemberExpressionOrHigher on how we replaced NewExpression with
+            // MemberExpression to make our lives easier.
+            //
+            // to best understand the below code, it's important to see how CallExpression expands
+            // out into its own productions:
+            //
+            // CallExpression:
+            //      MemberExpression Arguments
+            //      CallExpression Arguments
+            //      CallExpression[Expression]
+            //      CallExpression.IdentifierName
+            //      import (AssignmentExpression)
+            //      super Arguments
+            //      super.IdentifierName
+            //
+            // Because of the recursion in these calls, we need to bottom out first. There are three
+            // bottom out states we can run into: 1) We see 'super' which must start either of
+            // the last two CallExpression productions. 2) We see 'import' which must start import call.
+            // 3)we have a MemberExpression which either completes the LeftHandSideExpression,
+            // or starts the beginning of the first four CallExpression productions.
+            var pos = getNodePos();
+            var expression;
+            if (token() === 99 /* ImportKeyword */) {
+                if (lookAhead(nextTokenIsOpenParenOrLessThan)) {
+                    // We don't want to eagerly consume all import keyword as import call expression so we look ahead to find "("
+                    // For example:
+                    //      var foo3 = require("subfolder
+                    //      import * as foo1 from "module-from-node
+                    // We want this import to be a statement rather than import call expression
+                    sourceFlags |= 1048576 /* PossiblyContainsDynamicImport */;
+                    expression = parseTokenNode();
+                }
+                else if (lookAhead(nextTokenIsDot)) {
+                    // This is an 'import.*' metaproperty (i.e. 'import.meta')
+                    nextToken(); // advance past the 'import'
+                    nextToken(); // advance past the dot
+                    expression = finishNode(factory.createMetaProperty(99 /* ImportKeyword */, parseIdentifierName()), pos);
+                    sourceFlags |= 2097152 /* PossiblyContainsImportMeta */;
+                }
+                else {
+                    expression = parseMemberExpressionOrHigher();
+                }
+            }
+            else {
+                expression = token() === 105 /* SuperKeyword */ ? parseSuperExpression() : parseMemberExpressionOrHigher();
             }
+            // Now, we *may* be complete.  However, we might have consumed the start of a
+            // CallExpression or OptionalExpression.  As such, we need to consume the rest
+            // of it here to be complete.
+            return parseCallExpressionRest(pos, expression);
         }
-        return optionsNameMap.get(optionName);
-    }
-    var buildOptionsNameMapCache;
-    function getBuildOptionsNameMap() {
-        return buildOptionsNameMapCache || (buildOptionsNameMapCache = createOptionNameMap(ts.buildOpts));
-    }
-    var buildOptionsDidYouMeanDiagnostics = {
-        getOptionsNameMap: getBuildOptionsNameMap,
-        optionDeclarations: ts.buildOpts,
-        unknownOptionDiagnostic: ts.Diagnostics.Unknown_build_option_0,
-        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_build_option_0_Did_you_mean_1,
-        optionTypeMismatchDiagnostic: ts.Diagnostics.Build_option_0_requires_a_value_of_type_1
-    };
-    /*@internal*/
-    function parseBuildCommand(args) {
-        var _a = parseCommandLineWorker(buildOptionsDidYouMeanDiagnostics, args), options = _a.options, watchOptions = _a.watchOptions, projects = _a.fileNames, errors = _a.errors;
-        var buildOptions = options;
-        if (projects.length === 0) {
-            // tsc -b invoked with no extra arguments; act as if invoked with "tsc -b ."
-            projects.push(".");
+        function parseMemberExpressionOrHigher() {
+            // Note: to make our lives simpler, we decompose the NewExpression productions and
+            // place ObjectCreationExpression and FunctionExpression into PrimaryExpression.
+            // like so:
+            //
+            //   PrimaryExpression : See 11.1
+            //      this
+            //      Identifier
+            //      Literal
+            //      ArrayLiteral
+            //      ObjectLiteral
+            //      (Expression)
+            //      FunctionExpression
+            //      new MemberExpression Arguments?
+            //
+            //   MemberExpression : See 11.2
+            //      PrimaryExpression
+            //      MemberExpression[Expression]
+            //      MemberExpression.IdentifierName
+            //
+            //   CallExpression : See 11.2
+            //      MemberExpression
+            //      CallExpression Arguments
+            //      CallExpression[Expression]
+            //      CallExpression.IdentifierName
+            //
+            // Technically this is ambiguous.  i.e. CallExpression defines:
+            //
+            //   CallExpression:
+            //      CallExpression Arguments
+            //
+            // If you see: "new Foo()"
+            //
+            // Then that could be treated as a single ObjectCreationExpression, or it could be
+            // treated as the invocation of "new Foo".  We disambiguate that in code (to match
+            // the original grammar) by making sure that if we see an ObjectCreationExpression
+            // we always consume arguments if they are there. So we treat "new Foo()" as an
+            // object creation only, and not at all as an invocation.  Another way to think
+            // about this is that for every "new" that we see, we will consume an argument list if
+            // it is there as part of the *associated* object creation node.  Any additional
+            // argument lists we see, will become invocation expressions.
+            //
+            // Because there are no other places in the grammar now that refer to FunctionExpression
+            // or ObjectCreationExpression, it is safe to push down into the PrimaryExpression
+            // production.
+            //
+            // Because CallExpression and MemberExpression are left recursive, we need to bottom out
+            // of the recursion immediately.  So we parse out a primary expression to start with.
+            var pos = getNodePos();
+            var expression = parsePrimaryExpression();
+            return parseMemberExpressionRest(pos, expression, /*allowOptionalChain*/ true);
         }
-        // Nonsensical combinations
-        if (buildOptions.clean && buildOptions.force) {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "clean", "force"));
+        function parseSuperExpression() {
+            var pos = getNodePos();
+            var expression = parseTokenNode();
+            if (token() === 29 /* LessThanToken */) {
+                var startPos = getNodePos();
+                var typeArguments = tryParse(parseTypeArgumentsInExpression);
+                if (typeArguments !== undefined) {
+                    parseErrorAt(startPos, getNodePos(), ts.Diagnostics.super_may_not_use_type_arguments);
+                }
+            }
+            if (token() === 20 /* OpenParenToken */ || token() === 24 /* DotToken */ || token() === 22 /* OpenBracketToken */) {
+                return expression;
+            }
+            // If we have seen "super" it must be followed by '(' or '.'.
+            // If it wasn't then just try to parse out a '.' and report an error.
+            parseExpectedToken(24 /* DotToken */, ts.Diagnostics.super_must_be_followed_by_an_argument_list_or_member_access);
+            // private names will never work with `super` (`super.#foo`), but that's a semantic error, not syntactic
+            return finishNode(factory.createPropertyAccessExpression(expression, parseRightSideOfDot(/*allowIdentifierNames*/ true, /*allowPrivateIdentifiers*/ true)), pos);
         }
-        if (buildOptions.clean && buildOptions.verbose) {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "clean", "verbose"));
+        function parseJsxElementOrSelfClosingElementOrFragment(inExpressionContext, topInvalidNodePosition) {
+            var pos = getNodePos();
+            var opening = parseJsxOpeningOrSelfClosingElementOrOpeningFragment(inExpressionContext);
+            var result;
+            if (opening.kind === 275 /* JsxOpeningElement */) {
+                var children = parseJsxChildren(opening);
+                var closingElement = parseJsxClosingElement(inExpressionContext);
+                if (!tagNamesAreEquivalent(opening.tagName, closingElement.tagName)) {
+                    parseErrorAtRange(closingElement, ts.Diagnostics.Expected_corresponding_JSX_closing_tag_for_0, ts.getTextOfNodeFromSourceText(sourceText, opening.tagName));
+                }
+                result = finishNode(factory.createJsxElement(opening, children, closingElement), pos);
+            }
+            else if (opening.kind === 278 /* JsxOpeningFragment */) {
+                result = finishNode(factory.createJsxFragment(opening, parseJsxChildren(opening), parseJsxClosingFragment(inExpressionContext)), pos);
+            }
+            else {
+                ts.Debug.assert(opening.kind === 274 /* JsxSelfClosingElement */);
+                // Nothing else to do for self-closing elements
+                result = opening;
+            }
+            // If the user writes the invalid code '<div></div><div></div>' in an expression context (i.e. not wrapped in
+            // an enclosing tag), we'll naively try to parse   ^ this as a 'less than' operator and the remainder of the tag
+            // as garbage, which will cause the formatter to badly mangle the JSX. Perform a speculative parse of a JSX
+            // element if we see a < token so that we can wrap it in a synthetic binary expression so the formatter
+            // does less damage and we can report a better error.
+            // Since JSX elements are invalid < operands anyway, this lookahead parse will only occur in error scenarios
+            // of one sort or another.
+            if (inExpressionContext && token() === 29 /* LessThanToken */) {
+                var topBadPos_1 = typeof topInvalidNodePosition === "undefined" ? result.pos : topInvalidNodePosition;
+                var invalidElement = tryParse(function () { return parseJsxElementOrSelfClosingElementOrFragment(/*inExpressionContext*/ true, topBadPos_1); });
+                if (invalidElement) {
+                    var operatorToken = createMissingNode(27 /* CommaToken */, /*reportAtCurrentPosition*/ false);
+                    ts.setTextRangePosWidth(operatorToken, invalidElement.pos, 0);
+                    parseErrorAt(ts.skipTrivia(sourceText, topBadPos_1), invalidElement.end, ts.Diagnostics.JSX_expressions_must_have_one_parent_element);
+                    return finishNode(factory.createBinaryExpression(result, operatorToken, invalidElement), pos);
+                }
+            }
+            return result;
         }
-        if (buildOptions.clean && buildOptions.watch) {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "clean", "watch"));
+        function parseJsxText() {
+            var pos = getNodePos();
+            var node = factory.createJsxText(scanner.getTokenValue(), currentToken === 12 /* JsxTextAllWhiteSpaces */);
+            currentToken = scanner.scanJsxToken();
+            return finishNode(node, pos);
         }
-        if (buildOptions.watch && buildOptions.dry) {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "watch", "dry"));
+        function parseJsxChild(openingTag, token) {
+            switch (token) {
+                case 1 /* EndOfFileToken */:
+                    // If we hit EOF, issue the error at the tag that lacks the closing element
+                    // rather than at the end of the file (which is useless)
+                    if (ts.isJsxOpeningFragment(openingTag)) {
+                        parseErrorAtRange(openingTag, ts.Diagnostics.JSX_fragment_has_no_corresponding_closing_tag);
+                    }
+                    else {
+                        // We want the error span to cover only 'Foo.Bar' in < Foo.Bar >
+                        // or to cover only 'Foo' in < Foo >
+                        var tag = openingTag.tagName;
+                        var start = ts.skipTrivia(sourceText, tag.pos);
+                        parseErrorAt(start, tag.end, ts.Diagnostics.JSX_element_0_has_no_corresponding_closing_tag, ts.getTextOfNodeFromSourceText(sourceText, openingTag.tagName));
+                    }
+                    return undefined;
+                case 30 /* LessThanSlashToken */:
+                case 7 /* ConflictMarkerTrivia */:
+                    return undefined;
+                case 11 /* JsxText */:
+                case 12 /* JsxTextAllWhiteSpaces */:
+                    return parseJsxText();
+                case 18 /* OpenBraceToken */:
+                    return parseJsxExpression(/*inExpressionContext*/ false);
+                case 29 /* LessThanToken */:
+                    return parseJsxElementOrSelfClosingElementOrFragment(/*inExpressionContext*/ false);
+                default:
+                    return ts.Debug.assertNever(token);
+            }
         }
-        return { buildOptions: buildOptions, watchOptions: watchOptions, projects: projects, errors: errors };
-    }
-    ts.parseBuildCommand = parseBuildCommand;
-    /* @internal */
-    function getDiagnosticText(_message) {
-        var _args = [];
-        for (var _i = 1; _i < arguments.length; _i++) {
-            _args[_i - 1] = arguments[_i];
+        function parseJsxChildren(openingTag) {
+            var list = [];
+            var listPos = getNodePos();
+            var saveParsingContext = parsingContext;
+            parsingContext |= 1 << 14 /* JsxChildren */;
+            while (true) {
+                var child = parseJsxChild(openingTag, currentToken = scanner.reScanJsxToken());
+                if (!child)
+                    break;
+                list.push(child);
+            }
+            parsingContext = saveParsingContext;
+            return createNodeArray(list, listPos);
         }
-        var diagnostic = ts.createCompilerDiagnostic.apply(undefined, arguments);
-        return diagnostic.messageText;
-    }
-    ts.getDiagnosticText = getDiagnosticText;
-    /**
-     * Reads the config file, reports errors if any and exits if the config file cannot be found
-     */
-    function getParsedCommandLineOfConfigFile(configFileName, optionsToExtend, host, extendedConfigCache, watchOptionsToExtend, extraFileExtensions) {
-        var configFileText = tryReadFile(configFileName, function (fileName) { return host.readFile(fileName); });
-        if (!ts.isString(configFileText)) {
-            host.onUnRecoverableConfigFileDiagnostic(configFileText);
-            return undefined;
+        function parseJsxAttributes() {
+            var pos = getNodePos();
+            return finishNode(factory.createJsxAttributes(parseList(13 /* JsxAttributes */, parseJsxAttribute)), pos);
         }
-        var result = ts.parseJsonText(configFileName, configFileText);
-        var cwd = host.getCurrentDirectory();
-        result.path = ts.toPath(configFileName, cwd, ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames));
-        result.resolvedPath = result.path;
-        result.originalFileName = result.fileName;
-        return parseJsonSourceFileConfigFileContent(result, host, ts.getNormalizedAbsolutePath(ts.getDirectoryPath(configFileName), cwd), optionsToExtend, ts.getNormalizedAbsolutePath(configFileName, cwd), 
-        /*resolutionStack*/ undefined, extraFileExtensions, extendedConfigCache, watchOptionsToExtend);
-    }
-    ts.getParsedCommandLineOfConfigFile = getParsedCommandLineOfConfigFile;
-    /**
-     * Read tsconfig.json file
-     * @param fileName The path to the config file
-     */
-    function readConfigFile(fileName, readFile) {
-        var textOrDiagnostic = tryReadFile(fileName, readFile);
-        return ts.isString(textOrDiagnostic) ? parseConfigFileTextToJson(fileName, textOrDiagnostic) : { config: {}, error: textOrDiagnostic };
-    }
-    ts.readConfigFile = readConfigFile;
-    /**
-     * Parse the text of the tsconfig.json file
-     * @param fileName The path to the config file
-     * @param jsonText The text of the config file
-     */
-    function parseConfigFileTextToJson(fileName, jsonText) {
-        var jsonSourceFile = ts.parseJsonText(fileName, jsonText);
-        return {
-            config: convertToObject(jsonSourceFile, jsonSourceFile.parseDiagnostics),
-            error: jsonSourceFile.parseDiagnostics.length ? jsonSourceFile.parseDiagnostics[0] : undefined
-        };
-    }
-    ts.parseConfigFileTextToJson = parseConfigFileTextToJson;
-    /**
-     * Read tsconfig.json file
-     * @param fileName The path to the config file
-     */
-    function readJsonConfigFile(fileName, readFile) {
-        var textOrDiagnostic = tryReadFile(fileName, readFile);
-        return ts.isString(textOrDiagnostic) ? ts.parseJsonText(fileName, textOrDiagnostic) : { parseDiagnostics: [textOrDiagnostic] };
-    }
-    ts.readJsonConfigFile = readJsonConfigFile;
-    /*@internal*/
-    function tryReadFile(fileName, readFile) {
-        var text;
-        try {
-            text = readFile(fileName);
+        function parseJsxOpeningOrSelfClosingElementOrOpeningFragment(inExpressionContext) {
+            var pos = getNodePos();
+            parseExpected(29 /* LessThanToken */);
+            if (token() === 31 /* GreaterThanToken */) {
+                // See below for explanation of scanJsxText
+                scanJsxText();
+                return finishNode(factory.createJsxOpeningFragment(), pos);
+            }
+            var tagName = parseJsxElementName();
+            var typeArguments = tryParseTypeArguments();
+            var attributes = parseJsxAttributes();
+            var node;
+            if (token() === 31 /* GreaterThanToken */) {
+                // Closing tag, so scan the immediately-following text with the JSX scanning instead
+                // of regular scanning to avoid treating illegal characters (e.g. '#') as immediate
+                // scanning errors
+                scanJsxText();
+                node = factory.createJsxOpeningElement(tagName, typeArguments, attributes);
+            }
+            else {
+                parseExpected(43 /* SlashToken */);
+                if (inExpressionContext) {
+                    parseExpected(31 /* GreaterThanToken */);
+                }
+                else {
+                    parseExpected(31 /* GreaterThanToken */, /*diagnostic*/ undefined, /*shouldAdvance*/ false);
+                    scanJsxText();
+                }
+                node = factory.createJsxSelfClosingElement(tagName, typeArguments, attributes);
+            }
+            return finishNode(node, pos);
         }
-        catch (e) {
-            return ts.createCompilerDiagnostic(ts.Diagnostics.Cannot_read_file_0_Colon_1, fileName, e.message);
+        function parseJsxElementName() {
+            var pos = getNodePos();
+            scanJsxIdentifier();
+            // JsxElement can have name in the form of
+            //      propertyAccessExpression
+            //      primaryExpression in the form of an identifier and "this" keyword
+            // We can't just simply use parseLeftHandSideExpressionOrHigher because then we will start consider class,function etc as a keyword
+            // We only want to consider "this" as a primaryExpression
+            var expression = token() === 107 /* ThisKeyword */ ?
+                parseTokenNode() : parseIdentifierName();
+            while (parseOptional(24 /* DotToken */)) {
+                expression = finishNode(factory.createPropertyAccessExpression(expression, parseRightSideOfDot(/*allowIdentifierNames*/ true, /*allowPrivateIdentifiers*/ false)), pos);
+            }
+            return expression;
         }
-        return text === undefined ? ts.createCompilerDiagnostic(ts.Diagnostics.Cannot_read_file_0, fileName) : text;
-    }
-    ts.tryReadFile = tryReadFile;
-    function commandLineOptionsToMap(options) {
-        return ts.arrayToMap(options, getOptionName);
-    }
-    var typeAcquisitionDidYouMeanDiagnostics = {
-        optionDeclarations: ts.typeAcquisitionDeclarations,
-        unknownOptionDiagnostic: ts.Diagnostics.Unknown_type_acquisition_option_0,
-        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_type_acquisition_option_0_Did_you_mean_1,
-    };
-    var watchOptionsNameMapCache;
-    function getWatchOptionsNameMap() {
-        return watchOptionsNameMapCache || (watchOptionsNameMapCache = createOptionNameMap(ts.optionsForWatch));
-    }
-    var watchOptionsDidYouMeanDiagnostics = {
-        getOptionsNameMap: getWatchOptionsNameMap,
-        optionDeclarations: ts.optionsForWatch,
-        unknownOptionDiagnostic: ts.Diagnostics.Unknown_watch_option_0,
-        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_watch_option_0_Did_you_mean_1,
-        optionTypeMismatchDiagnostic: ts.Diagnostics.Watch_option_0_requires_a_value_of_type_1
-    };
-    var commandLineCompilerOptionsMapCache;
-    function getCommandLineCompilerOptionsMap() {
-        return commandLineCompilerOptionsMapCache || (commandLineCompilerOptionsMapCache = commandLineOptionsToMap(ts.optionDeclarations));
-    }
-    var commandLineWatchOptionsMapCache;
-    function getCommandLineWatchOptionsMap() {
-        return commandLineWatchOptionsMapCache || (commandLineWatchOptionsMapCache = commandLineOptionsToMap(ts.optionsForWatch));
-    }
-    var commandLineTypeAcquisitionMapCache;
-    function getCommandLineTypeAcquisitionMap() {
-        return commandLineTypeAcquisitionMapCache || (commandLineTypeAcquisitionMapCache = commandLineOptionsToMap(ts.typeAcquisitionDeclarations));
-    }
-    var _tsconfigRootOptions;
-    function getTsconfigRootOptionsMap() {
-        if (_tsconfigRootOptions === undefined) {
-            _tsconfigRootOptions = {
-                name: undefined,
-                type: "object",
-                elementOptions: commandLineOptionsToMap([
-                    {
-                        name: "compilerOptions",
-                        type: "object",
-                        elementOptions: getCommandLineCompilerOptionsMap(),
-                        extraKeyDiagnostics: ts.compilerOptionsDidYouMeanDiagnostics,
-                    },
-                    {
-                        name: "watchOptions",
-                        type: "object",
-                        elementOptions: getCommandLineWatchOptionsMap(),
-                        extraKeyDiagnostics: watchOptionsDidYouMeanDiagnostics,
-                    },
-                    {
-                        name: "typingOptions",
-                        type: "object",
-                        elementOptions: getCommandLineTypeAcquisitionMap(),
-                        extraKeyDiagnostics: typeAcquisitionDidYouMeanDiagnostics,
-                    },
-                    {
-                        name: "typeAcquisition",
-                        type: "object",
-                        elementOptions: getCommandLineTypeAcquisitionMap(),
-                        extraKeyDiagnostics: typeAcquisitionDidYouMeanDiagnostics
-                    },
-                    {
-                        name: "extends",
-                        type: "string"
-                    },
-                    {
-                        name: "references",
-                        type: "list",
-                        element: {
-                            name: "references",
-                            type: "object"
-                        }
-                    },
-                    {
-                        name: "files",
-                        type: "list",
-                        element: {
-                            name: "files",
-                            type: "string"
-                        }
-                    },
-                    {
-                        name: "include",
-                        type: "list",
-                        element: {
-                            name: "include",
-                            type: "string"
-                        }
-                    },
-                    {
-                        name: "exclude",
-                        type: "list",
-                        element: {
-                            name: "exclude",
-                            type: "string"
-                        }
-                    },
-                    ts.compileOnSaveCommandLineOption
-                ])
-            };
+        function parseJsxExpression(inExpressionContext) {
+            var pos = getNodePos();
+            if (!parseExpected(18 /* OpenBraceToken */)) {
+                return undefined;
+            }
+            var dotDotDotToken;
+            var expression;
+            if (token() !== 19 /* CloseBraceToken */) {
+                dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
+                // Only an AssignmentExpression is valid here per the JSX spec,
+                // but we can unambiguously parse a comma sequence and provide
+                // a better error message in grammar checking.
+                expression = parseExpression();
+            }
+            if (inExpressionContext) {
+                parseExpected(19 /* CloseBraceToken */);
+            }
+            else {
+                if (parseExpected(19 /* CloseBraceToken */, /*message*/ undefined, /*shouldAdvance*/ false)) {
+                    scanJsxText();
+                }
+            }
+            return finishNode(factory.createJsxExpression(dotDotDotToken, expression), pos);
         }
-        return _tsconfigRootOptions;
-    }
-    /**
-     * Convert the json syntax tree into the json value
-     */
-    function convertToObject(sourceFile, errors) {
-        return convertToObjectWorker(sourceFile, errors, /*returnValue*/ true, /*knownRootOptions*/ undefined, /*jsonConversionNotifier*/ undefined);
-    }
-    ts.convertToObject = convertToObject;
-    /**
-     * Convert the json syntax tree into the json value and report errors
-     * This returns the json value (apart from checking errors) only if returnValue provided is true.
-     * Otherwise it just checks the errors and returns undefined
-     */
-    /*@internal*/
-    function convertToObjectWorker(sourceFile, errors, returnValue, knownRootOptions, jsonConversionNotifier) {
-        if (!sourceFile.statements.length) {
-            return returnValue ? {} : undefined;
+        function parseJsxAttribute() {
+            if (token() === 18 /* OpenBraceToken */) {
+                return parseJsxSpreadAttribute();
+            }
+            scanJsxIdentifier();
+            var pos = getNodePos();
+            return finishNode(factory.createJsxAttribute(parseIdentifierName(), token() !== 62 /* EqualsToken */ ? undefined :
+                scanJsxAttributeValue() === 10 /* StringLiteral */ ? parseLiteralNode() :
+                    parseJsxExpression(/*inExpressionContext*/ true)), pos);
         }
-        return convertPropertyValueToJson(sourceFile.statements[0].expression, knownRootOptions);
-        function isRootOptionMap(knownOptions) {
-            return knownRootOptions && knownRootOptions.elementOptions === knownOptions;
+        function parseJsxSpreadAttribute() {
+            var pos = getNodePos();
+            parseExpected(18 /* OpenBraceToken */);
+            parseExpected(25 /* DotDotDotToken */);
+            var expression = parseExpression();
+            parseExpected(19 /* CloseBraceToken */);
+            return finishNode(factory.createJsxSpreadAttribute(expression), pos);
         }
-        function convertObjectLiteralExpressionToJson(node, knownOptions, extraKeyDiagnostics, parentOption) {
-            var result = returnValue ? {} : undefined;
-            var _loop_3 = function (element) {
-                if (element.kind !== 281 /* PropertyAssignment */) {
-                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element, ts.Diagnostics.Property_assignment_expected));
-                    return "continue";
+        function parseJsxClosingElement(inExpressionContext) {
+            var pos = getNodePos();
+            parseExpected(30 /* LessThanSlashToken */);
+            var tagName = parseJsxElementName();
+            if (inExpressionContext) {
+                parseExpected(31 /* GreaterThanToken */);
+            }
+            else {
+                parseExpected(31 /* GreaterThanToken */, /*diagnostic*/ undefined, /*shouldAdvance*/ false);
+                scanJsxText();
+            }
+            return finishNode(factory.createJsxClosingElement(tagName), pos);
+        }
+        function parseJsxClosingFragment(inExpressionContext) {
+            var pos = getNodePos();
+            parseExpected(30 /* LessThanSlashToken */);
+            if (ts.tokenIsIdentifierOrKeyword(token())) {
+                parseErrorAtRange(parseJsxElementName(), ts.Diagnostics.Expected_corresponding_closing_tag_for_JSX_fragment);
+            }
+            if (inExpressionContext) {
+                parseExpected(31 /* GreaterThanToken */);
+            }
+            else {
+                parseExpected(31 /* GreaterThanToken */, /*diagnostic*/ undefined, /*shouldAdvance*/ false);
+                scanJsxText();
+            }
+            return finishNode(factory.createJsxJsxClosingFragment(), pos);
+        }
+        function parseTypeAssertion() {
+            var pos = getNodePos();
+            parseExpected(29 /* LessThanToken */);
+            var type = parseType();
+            parseExpected(31 /* GreaterThanToken */);
+            var expression = parseSimpleUnaryExpression();
+            return finishNode(factory.createTypeAssertion(type, expression), pos);
+        }
+        function nextTokenIsIdentifierOrKeywordOrOpenBracketOrTemplate() {
+            nextToken();
+            return ts.tokenIsIdentifierOrKeyword(token())
+                || token() === 22 /* OpenBracketToken */
+                || isTemplateStartOfTaggedTemplate();
+        }
+        function isStartOfOptionalPropertyOrElementAccessChain() {
+            return token() === 28 /* QuestionDotToken */
+                && lookAhead(nextTokenIsIdentifierOrKeywordOrOpenBracketOrTemplate);
+        }
+        function tryReparseOptionalChain(node) {
+            if (node.flags & 32 /* OptionalChain */) {
+                return true;
+            }
+            // check for an optional chain in a non-null expression
+            if (ts.isNonNullExpression(node)) {
+                var expr = node.expression;
+                while (ts.isNonNullExpression(expr) && !(expr.flags & 32 /* OptionalChain */)) {
+                    expr = expr.expression;
                 }
-                if (element.questionToken) {
-                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element.questionToken, ts.Diagnostics.The_0_modifier_can_only_be_used_in_TypeScript_files, "?"));
+                if (expr.flags & 32 /* OptionalChain */) {
+                    // this is part of an optional chain. Walk down from `node` to `expression` and set the flag.
+                    while (ts.isNonNullExpression(node)) {
+                        node.flags |= 32 /* OptionalChain */;
+                        node = node.expression;
+                    }
+                    return true;
                 }
-                if (!isDoubleQuotedString(element.name)) {
-                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element.name, ts.Diagnostics.String_literal_with_double_quotes_expected));
+            }
+            return false;
+        }
+        function parsePropertyAccessExpressionRest(pos, expression, questionDotToken) {
+            var name = parseRightSideOfDot(/*allowIdentifierNames*/ true, /*allowPrivateIdentifiers*/ true);
+            var isOptionalChain = questionDotToken || tryReparseOptionalChain(expression);
+            var propertyAccess = isOptionalChain ?
+                factory.createPropertyAccessChain(expression, questionDotToken, name) :
+                factory.createPropertyAccessExpression(expression, name);
+            if (isOptionalChain && ts.isPrivateIdentifier(propertyAccess.name)) {
+                parseErrorAtRange(propertyAccess.name, ts.Diagnostics.An_optional_chain_cannot_contain_private_identifiers);
+            }
+            return finishNode(propertyAccess, pos);
+        }
+        function parseElementAccessExpressionRest(pos, expression, questionDotToken) {
+            var argumentExpression;
+            if (token() === 23 /* CloseBracketToken */) {
+                argumentExpression = createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.An_element_access_expression_should_take_an_argument);
+            }
+            else {
+                var argument = allowInAnd(parseExpression);
+                if (ts.isStringOrNumericLiteralLike(argument)) {
+                    argument.text = internIdentifier(argument.text);
                 }
-                var textOfKey = ts.isComputedNonLiteralName(element.name) ? undefined : ts.getTextOfPropertyName(element.name);
-                var keyText = textOfKey && ts.unescapeLeadingUnderscores(textOfKey);
-                var option = keyText && knownOptions ? knownOptions.get(keyText) : undefined;
-                if (keyText && extraKeyDiagnostics && !option) {
-                    if (knownOptions) {
-                        errors.push(createUnknownOptionError(keyText, extraKeyDiagnostics, function (message, arg0, arg1) { return ts.createDiagnosticForNodeInSourceFile(sourceFile, element.name, message, arg0, arg1); }));
-                    }
-                    else {
-                        errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element.name, extraKeyDiagnostics.unknownOptionDiagnostic, keyText));
-                    }
+                argumentExpression = argument;
+            }
+            parseExpected(23 /* CloseBracketToken */);
+            var indexedAccess = questionDotToken || tryReparseOptionalChain(expression) ?
+                factory.createElementAccessChain(expression, questionDotToken, argumentExpression) :
+                factory.createElementAccessExpression(expression, argumentExpression);
+            return finishNode(indexedAccess, pos);
+        }
+        function parseMemberExpressionRest(pos, expression, allowOptionalChain) {
+            while (true) {
+                var questionDotToken = void 0;
+                var isPropertyAccess = false;
+                if (allowOptionalChain && isStartOfOptionalPropertyOrElementAccessChain()) {
+                    questionDotToken = parseExpectedToken(28 /* QuestionDotToken */);
+                    isPropertyAccess = ts.tokenIsIdentifierOrKeyword(token());
                 }
-                var value = convertPropertyValueToJson(element.initializer, option);
-                if (typeof keyText !== "undefined") {
-                    if (returnValue) {
-                        result[keyText] = value;
-                    }
-                    // Notify key value set, if user asked for it
-                    if (jsonConversionNotifier &&
-                        // Current callbacks are only on known parent option or if we are setting values in the root
-                        (parentOption || isRootOptionMap(knownOptions))) {
-                        var isValidOptionValue = isCompilerOptionsValue(option, value);
-                        if (parentOption) {
-                            if (isValidOptionValue) {
-                                // Notify option set in the parent if its a valid option value
-                                jsonConversionNotifier.onSetValidOptionKeyValueInParent(parentOption, option, value);
-                            }
-                        }
-                        else if (isRootOptionMap(knownOptions)) {
-                            if (isValidOptionValue) {
-                                // Notify about the valid root key value being set
-                                jsonConversionNotifier.onSetValidOptionKeyValueInRoot(keyText, element.name, value, element.initializer);
-                            }
-                            else if (!option) {
-                                // Notify about the unknown root key value being set
-                                jsonConversionNotifier.onSetUnknownOptionKeyValueInRoot(keyText, element.name, value, element.initializer);
-                            }
+                else {
+                    isPropertyAccess = parseOptional(24 /* DotToken */);
+                }
+                if (isPropertyAccess) {
+                    expression = parsePropertyAccessExpressionRest(pos, expression, questionDotToken);
+                    continue;
+                }
+                if (!questionDotToken && token() === 53 /* ExclamationToken */ && !scanner.hasPrecedingLineBreak()) {
+                    nextToken();
+                    expression = finishNode(factory.createNonNullExpression(expression), pos);
+                    continue;
+                }
+                // when in the [Decorator] context, we do not parse ElementAccess as it could be part of a ComputedPropertyName
+                if ((questionDotToken || !inDecoratorContext()) && parseOptional(22 /* OpenBracketToken */)) {
+                    expression = parseElementAccessExpressionRest(pos, expression, questionDotToken);
+                    continue;
+                }
+                if (isTemplateStartOfTaggedTemplate()) {
+                    expression = parseTaggedTemplateRest(pos, expression, questionDotToken, /*typeArguments*/ undefined);
+                    continue;
+                }
+                return expression;
+            }
+        }
+        function isTemplateStartOfTaggedTemplate() {
+            return token() === 14 /* NoSubstitutionTemplateLiteral */ || token() === 15 /* TemplateHead */;
+        }
+        function parseTaggedTemplateRest(pos, tag, questionDotToken, typeArguments) {
+            var tagExpression = factory.createTaggedTemplateExpression(tag, typeArguments, token() === 14 /* NoSubstitutionTemplateLiteral */ ?
+                (reScanTemplateHeadOrNoSubstitutionTemplate(), parseLiteralNode()) :
+                parseTemplateExpression(/*isTaggedTemplate*/ true));
+            if (questionDotToken || tag.flags & 32 /* OptionalChain */) {
+                tagExpression.flags |= 32 /* OptionalChain */;
+            }
+            tagExpression.questionDotToken = questionDotToken;
+            return finishNode(tagExpression, pos);
+        }
+        function parseCallExpressionRest(pos, expression) {
+            while (true) {
+                expression = parseMemberExpressionRest(pos, expression, /*allowOptionalChain*/ true);
+                var questionDotToken = parseOptionalToken(28 /* QuestionDotToken */);
+                // handle 'foo<<T>()'
+                if (token() === 29 /* LessThanToken */ || token() === 47 /* LessThanLessThanToken */) {
+                    // See if this is the start of a generic invocation.  If so, consume it and
+                    // keep checking for postfix expressions.  Otherwise, it's just a '<' that's
+                    // part of an arithmetic expression.  Break out so we consume it higher in the
+                    // stack.
+                    var typeArguments = tryParse(parseTypeArgumentsInExpression);
+                    if (typeArguments) {
+                        if (isTemplateStartOfTaggedTemplate()) {
+                            expression = parseTaggedTemplateRest(pos, expression, questionDotToken, typeArguments);
+                            continue;
                         }
+                        var argumentList = parseArgumentList();
+                        var callExpr = questionDotToken || tryReparseOptionalChain(expression) ?
+                            factory.createCallChain(expression, questionDotToken, typeArguments, argumentList) :
+                            factory.createCallExpression(expression, typeArguments, argumentList);
+                        expression = finishNode(callExpr, pos);
+                        continue;
                     }
                 }
-            };
-            for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
-                var element = _a[_i];
-                _loop_3(element);
+                else if (token() === 20 /* OpenParenToken */) {
+                    var argumentList = parseArgumentList();
+                    var callExpr = questionDotToken || tryReparseOptionalChain(expression) ?
+                        factory.createCallChain(expression, questionDotToken, /*typeArguments*/ undefined, argumentList) :
+                        factory.createCallExpression(expression, /*typeArguments*/ undefined, argumentList);
+                    expression = finishNode(callExpr, pos);
+                    continue;
+                }
+                if (questionDotToken) {
+                    // We failed to parse anything, so report a missing identifier here.
+                    var name = createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ false, ts.Diagnostics.Identifier_expected);
+                    expression = finishNode(factory.createPropertyAccessChain(expression, questionDotToken, name), pos);
+                }
+                break;
             }
+            return expression;
+        }
+        function parseArgumentList() {
+            parseExpected(20 /* OpenParenToken */);
+            var result = parseDelimitedList(11 /* ArgumentExpressions */, parseArgumentExpression);
+            parseExpected(21 /* CloseParenToken */);
             return result;
         }
-        function convertArrayLiteralExpressionToJson(elements, elementOption) {
-            if (!returnValue) {
-                return elements.forEach(function (element) { return convertPropertyValueToJson(element, elementOption); });
+        function parseTypeArgumentsInExpression() {
+            if (reScanLessThanToken() !== 29 /* LessThanToken */) {
+                return undefined;
             }
-            // Filter out invalid values
-            return ts.filter(elements.map(function (element) { return convertPropertyValueToJson(element, elementOption); }), function (v) { return v !== undefined; });
+            nextToken();
+            var typeArguments = parseDelimitedList(20 /* TypeArguments */, parseType);
+            if (!parseExpected(31 /* GreaterThanToken */)) {
+                // If it doesn't have the closing `>` then it's definitely not an type argument list.
+                return undefined;
+            }
+            // If we have a '<', then only parse this as a argument list if the type arguments
+            // are complete and we have an open paren.  if we don't, rewind and return nothing.
+            return typeArguments && canFollowTypeArgumentsInExpression()
+                ? typeArguments
+                : undefined;
         }
-        function convertPropertyValueToJson(valueExpression, option) {
-            switch (valueExpression.kind) {
-                case 106 /* TrueKeyword */:
-                    reportInvalidOptionValue(option && option.type !== "boolean");
+        function canFollowTypeArgumentsInExpression() {
+            switch (token()) {
+                case 20 /* OpenParenToken */: // foo<x>(
+                case 14 /* NoSubstitutionTemplateLiteral */: // foo<T> `...`
+                case 15 /* TemplateHead */: // foo<T> `...${100}...`
+                // these are the only tokens can legally follow a type argument
+                // list. So we definitely want to treat them as type arg lists.
+                // falls through
+                case 24 /* DotToken */: // foo<x>.
+                case 21 /* CloseParenToken */: // foo<x>)
+                case 23 /* CloseBracketToken */: // foo<x>]
+                case 58 /* ColonToken */: // foo<x>:
+                case 26 /* SemicolonToken */: // foo<x>;
+                case 57 /* QuestionToken */: // foo<x>?
+                case 34 /* EqualsEqualsToken */: // foo<x> ==
+                case 36 /* EqualsEqualsEqualsToken */: // foo<x> ===
+                case 35 /* ExclamationEqualsToken */: // foo<x> !=
+                case 37 /* ExclamationEqualsEqualsToken */: // foo<x> !==
+                case 55 /* AmpersandAmpersandToken */: // foo<x> &&
+                case 56 /* BarBarToken */: // foo<x> ||
+                case 60 /* QuestionQuestionToken */: // foo<x> ??
+                case 52 /* CaretToken */: // foo<x> ^
+                case 50 /* AmpersandToken */: // foo<x> &
+                case 51 /* BarToken */: // foo<x> |
+                case 19 /* CloseBraceToken */: // foo<x> }
+                case 1 /* EndOfFileToken */: // foo<x>
+                    // these cases can't legally follow a type arg list.  However, they're not legal
+                    // expressions either.  The user is probably in the middle of a generic type. So
+                    // treat it as such.
                     return true;
-                case 91 /* FalseKeyword */:
-                    reportInvalidOptionValue(option && option.type !== "boolean");
+                case 27 /* CommaToken */: // foo<x>,
+                case 18 /* OpenBraceToken */: // foo<x> {
+                // We don't want to treat these as type arguments.  Otherwise we'll parse this
+                // as an invocation expression.  Instead, we want to parse out the expression
+                // in isolation from the type arguments.
+                // falls through
+                default:
+                    // Anything else treat as an expression.
                     return false;
-                case 100 /* NullKeyword */:
-                    reportInvalidOptionValue(option && option.name === "extends"); // "extends" is the only option we don't allow null/undefined for
-                    return null; // eslint-disable-line no-null/no-null
-                case 10 /* StringLiteral */:
-                    if (!isDoubleQuotedString(valueExpression)) {
-                        errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, ts.Diagnostics.String_literal_with_double_quotes_expected));
-                    }
-                    reportInvalidOptionValue(option && (ts.isString(option.type) && option.type !== "string"));
-                    var text = valueExpression.text;
-                    if (option && !ts.isString(option.type)) {
-                        var customOption = option;
-                        // Validate custom option type
-                        if (!customOption.type.has(text.toLowerCase())) {
-                            errors.push(createDiagnosticForInvalidCustomType(customOption, function (message, arg0, arg1) { return ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, message, arg0, arg1); }));
-                        }
-                    }
-                    return text;
+            }
+        }
+        function parsePrimaryExpression() {
+            switch (token()) {
                 case 8 /* NumericLiteral */:
-                    reportInvalidOptionValue(option && option.type !== "number");
-                    return Number(valueExpression.text);
-                case 207 /* PrefixUnaryExpression */:
-                    if (valueExpression.operator !== 40 /* MinusToken */ || valueExpression.operand.kind !== 8 /* NumericLiteral */) {
-                        break; // not valid JSON syntax
-                    }
-                    reportInvalidOptionValue(option && option.type !== "number");
-                    return -Number(valueExpression.operand.text);
-                case 193 /* ObjectLiteralExpression */:
-                    reportInvalidOptionValue(option && option.type !== "object");
-                    var objectLiteralExpression = valueExpression;
-                    // Currently having element option declaration in the tsconfig with type "object"
-                    // determines if it needs onSetValidOptionKeyValueInParent callback or not
-                    // At moment there are only "compilerOptions", "typeAcquisition" and "typingOptions"
-                    // that satifies it and need it to modify options set in them (for normalizing file paths)
-                    // vs what we set in the json
-                    // If need arises, we can modify this interface and callbacks as needed
-                    if (option) {
-                        var _a = option, elementOptions = _a.elementOptions, extraKeyDiagnostics = _a.extraKeyDiagnostics, optionName = _a.name;
-                        return convertObjectLiteralExpressionToJson(objectLiteralExpression, elementOptions, extraKeyDiagnostics, optionName);
+                case 9 /* BigIntLiteral */:
+                case 10 /* StringLiteral */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                    return parseLiteralNode();
+                case 107 /* ThisKeyword */:
+                case 105 /* SuperKeyword */:
+                case 103 /* NullKeyword */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                    return parseTokenNode();
+                case 20 /* OpenParenToken */:
+                    return parseParenthesizedExpression();
+                case 22 /* OpenBracketToken */:
+                    return parseArrayLiteralExpression();
+                case 18 /* OpenBraceToken */:
+                    return parseObjectLiteralExpression();
+                case 129 /* AsyncKeyword */:
+                    // Async arrow functions are parsed earlier in parseAssignmentExpressionOrHigher.
+                    // If we encounter `async [no LineTerminator here] function` then this is an async
+                    // function; otherwise, its an identifier.
+                    if (!lookAhead(nextTokenIsFunctionKeywordOnSameLine)) {
+                        break;
                     }
-                    else {
-                        return convertObjectLiteralExpressionToJson(objectLiteralExpression, /* knownOptions*/ undefined, 
-                        /*extraKeyDiagnosticMessage */ undefined, /*parentOption*/ undefined);
+                    return parseFunctionExpression();
+                case 83 /* ClassKeyword */:
+                    return parseClassExpression();
+                case 97 /* FunctionKeyword */:
+                    return parseFunctionExpression();
+                case 102 /* NewKeyword */:
+                    return parseNewExpressionOrNewDotTarget();
+                case 43 /* SlashToken */:
+                case 67 /* SlashEqualsToken */:
+                    if (reScanSlashToken() === 13 /* RegularExpressionLiteral */) {
+                        return parseLiteralNode();
                     }
-                case 192 /* ArrayLiteralExpression */:
-                    reportInvalidOptionValue(option && option.type !== "list");
-                    return convertArrayLiteralExpressionToJson(valueExpression.elements, option && option.element);
+                    break;
+                case 15 /* TemplateHead */:
+                    return parseTemplateExpression(/* isTaggedTemplate */ false);
             }
-            // Not in expected format
-            if (option) {
-                reportInvalidOptionValue(/*isError*/ true);
+            return parseIdentifier(ts.Diagnostics.Expression_expected);
+        }
+        function parseParenthesizedExpression() {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            parseExpected(20 /* OpenParenToken */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(21 /* CloseParenToken */);
+            return withJSDoc(finishNode(factory.createParenthesizedExpression(expression), pos), hasJSDoc);
+        }
+        function parseSpreadElement() {
+            var pos = getNodePos();
+            parseExpected(25 /* DotDotDotToken */);
+            var expression = parseAssignmentExpressionOrHigher();
+            return finishNode(factory.createSpreadElement(expression), pos);
+        }
+        function parseArgumentOrArrayLiteralElement() {
+            return token() === 25 /* DotDotDotToken */ ? parseSpreadElement() :
+                token() === 27 /* CommaToken */ ? finishNode(factory.createOmittedExpression(), getNodePos()) :
+                    parseAssignmentExpressionOrHigher();
+        }
+        function parseArgumentExpression() {
+            return doOutsideOfContext(disallowInAndDecoratorContext, parseArgumentOrArrayLiteralElement);
+        }
+        function parseArrayLiteralExpression() {
+            var pos = getNodePos();
+            parseExpected(22 /* OpenBracketToken */);
+            var multiLine = scanner.hasPrecedingLineBreak();
+            var elements = parseDelimitedList(15 /* ArrayLiteralMembers */, parseArgumentOrArrayLiteralElement);
+            parseExpected(23 /* CloseBracketToken */);
+            return finishNode(factory.createArrayLiteralExpression(elements, multiLine), pos);
+        }
+        function parseObjectLiteralElement() {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            if (parseOptionalToken(25 /* DotDotDotToken */)) {
+                var expression = parseAssignmentExpressionOrHigher();
+                return withJSDoc(finishNode(factory.createSpreadAssignment(expression), pos), hasJSDoc);
+            }
+            var decorators = parseDecorators();
+            var modifiers = parseModifiers();
+            if (parseContextualModifier(134 /* GetKeyword */)) {
+                return parseAccessorDeclaration(pos, hasJSDoc, decorators, modifiers, 167 /* GetAccessor */);
+            }
+            if (parseContextualModifier(146 /* SetKeyword */)) {
+                return parseAccessorDeclaration(pos, hasJSDoc, decorators, modifiers, 168 /* SetAccessor */);
+            }
+            var asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
+            var tokenIsIdentifier = isIdentifier();
+            var name = parsePropertyName();
+            // Disallowing of optional property assignments and definite assignment assertion happens in the grammar checker.
+            var questionToken = parseOptionalToken(57 /* QuestionToken */);
+            var exclamationToken = parseOptionalToken(53 /* ExclamationToken */);
+            if (asteriskToken || token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
+                return parseMethodDeclaration(pos, hasJSDoc, decorators, modifiers, asteriskToken, name, questionToken, exclamationToken);
+            }
+            // check if it is short-hand property assignment or normal property assignment
+            // NOTE: if token is EqualsToken it is interpreted as CoverInitializedName production
+            // CoverInitializedName[Yield] :
+            //     IdentifierReference[?Yield] Initializer[In, ?Yield]
+            // this is necessary because ObjectLiteral productions are also used to cover grammar for ObjectAssignmentPattern
+            var node;
+            var isShorthandPropertyAssignment = tokenIsIdentifier && (token() !== 58 /* ColonToken */);
+            if (isShorthandPropertyAssignment) {
+                var equalsToken = parseOptionalToken(62 /* EqualsToken */);
+                var objectAssignmentInitializer = equalsToken ? allowInAnd(parseAssignmentExpressionOrHigher) : undefined;
+                node = factory.createShorthandPropertyAssignment(name, objectAssignmentInitializer);
+                // Save equals token for error reporting.
+                // TODO(rbuckton): Consider manufacturing this when we need to report an error as it is otherwise not useful.
+                node.equalsToken = equalsToken;
             }
             else {
-                errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, ts.Diagnostics.Property_value_can_only_be_string_literal_numeric_literal_true_false_null_object_literal_or_array_literal));
+                parseExpected(58 /* ColonToken */);
+                var initializer = allowInAnd(parseAssignmentExpressionOrHigher);
+                node = factory.createPropertyAssignment(name, initializer);
             }
-            return undefined;
-            function reportInvalidOptionValue(isError) {
-                if (isError) {
-                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, option.name, getCompilerOptionValueTypeString(option)));
+            // Decorators, Modifiers, questionToken, and exclamationToken are not supported by property assignments and are reported in the grammar checker
+            node.decorators = decorators;
+            node.modifiers = modifiers;
+            node.questionToken = questionToken;
+            node.exclamationToken = exclamationToken;
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parseObjectLiteralExpression() {
+            var pos = getNodePos();
+            var openBracePosition = scanner.getTokenPos();
+            parseExpected(18 /* OpenBraceToken */);
+            var multiLine = scanner.hasPrecedingLineBreak();
+            var properties = parseDelimitedList(12 /* ObjectLiteralMembers */, parseObjectLiteralElement, /*considerSemicolonAsDelimiter*/ true);
+            if (!parseExpected(19 /* CloseBraceToken */)) {
+                var lastError = ts.lastOrUndefined(parseDiagnostics);
+                if (lastError && lastError.code === ts.Diagnostics._0_expected.code) {
+                    ts.addRelatedInfo(lastError, ts.createDetachedDiagnostic(fileName, openBracePosition, 1, ts.Diagnostics.The_parser_expected_to_find_a_to_match_the_token_here));
                 }
             }
+            return finishNode(factory.createObjectLiteralExpression(properties, multiLine), pos);
         }
-        function isDoubleQuotedString(node) {
-            return ts.isStringLiteral(node) && ts.isStringDoubleQuoted(node, sourceFile);
-        }
-    }
-    ts.convertToObjectWorker = convertToObjectWorker;
-    function getCompilerOptionValueTypeString(option) {
-        return option.type === "list" ?
-            "Array" :
-            ts.isString(option.type) ? option.type : "string";
-    }
-    function isCompilerOptionsValue(option, value) {
-        if (option) {
-            if (isNullOrUndefined(value))
-                return true; // All options are undefinable/nullable
-            if (option.type === "list") {
-                return ts.isArray(value);
+        function parseFunctionExpression() {
+            // GeneratorExpression:
+            //      function* BindingIdentifier [Yield][opt](FormalParameters[Yield]){ GeneratorBody }
+            //
+            // FunctionExpression:
+            //      function BindingIdentifier[opt](FormalParameters){ FunctionBody }
+            var saveDecoratorContext = inDecoratorContext();
+            if (saveDecoratorContext) {
+                setDecoratorContext(/*val*/ false);
             }
-            var expectedType = ts.isString(option.type) ? option.type : "string";
-            return typeof value === expectedType;
-        }
-        return false;
-    }
-    /**
-     * Generate an uncommented, complete tsconfig for use with "--showConfig"
-     * @param configParseResult options to be generated into tsconfig.json
-     * @param configFileName name of the parsed config file - output paths will be generated relative to this
-     * @param host provides current directory and case sensitivity services
-     */
-    /** @internal */
-    function convertToTSConfig(configParseResult, configFileName, host) {
-        var getCanonicalFileName = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames);
-        var files = ts.map(ts.filter(configParseResult.fileNames, (!configParseResult.configFileSpecs || !configParseResult.configFileSpecs.validatedIncludeSpecs) ? function (_) { return true; } : matchesSpecs(configFileName, configParseResult.configFileSpecs.validatedIncludeSpecs, configParseResult.configFileSpecs.validatedExcludeSpecs, host)), function (f) { return ts.getRelativePathFromFile(ts.getNormalizedAbsolutePath(configFileName, host.getCurrentDirectory()), ts.getNormalizedAbsolutePath(f, host.getCurrentDirectory()), getCanonicalFileName); });
-        var optionMap = serializeCompilerOptions(configParseResult.options, { configFilePath: ts.getNormalizedAbsolutePath(configFileName, host.getCurrentDirectory()), useCaseSensitiveFileNames: host.useCaseSensitiveFileNames });
-        var watchOptionMap = configParseResult.watchOptions && serializeWatchOptions(configParseResult.watchOptions);
-        var config = __assign(__assign({ compilerOptions: __assign(__assign({}, optionMapToObject(optionMap)), { showConfig: undefined, configFile: undefined, configFilePath: undefined, help: undefined, init: undefined, listFiles: undefined, listEmittedFiles: undefined, project: undefined, build: undefined, version: undefined }), watchOptions: watchOptionMap && optionMapToObject(watchOptionMap), references: ts.map(configParseResult.projectReferences, function (r) { return (__assign(__assign({}, r), { path: r.originalPath ? r.originalPath : "", originalPath: undefined })); }), files: ts.length(files) ? files : undefined }, (configParseResult.configFileSpecs ? {
-            include: filterSameAsDefaultInclude(configParseResult.configFileSpecs.validatedIncludeSpecs),
-            exclude: configParseResult.configFileSpecs.validatedExcludeSpecs
-        } : {})), { compileOnSave: !!configParseResult.compileOnSave ? true : undefined });
-        return config;
-    }
-    ts.convertToTSConfig = convertToTSConfig;
-    function optionMapToObject(optionMap) {
-        return __assign({}, ts.arrayFrom(optionMap.entries()).reduce(function (prev, cur) {
-            var _a;
-            return (__assign(__assign({}, prev), (_a = {}, _a[cur[0]] = cur[1], _a)));
-        }, {}));
-    }
-    function filterSameAsDefaultInclude(specs) {
-        if (!ts.length(specs))
-            return undefined;
-        if (ts.length(specs) !== 1)
-            return specs;
-        if (specs[0] === "**/*")
-            return undefined;
-        return specs;
-    }
-    function matchesSpecs(path, includeSpecs, excludeSpecs, host) {
-        if (!includeSpecs)
-            return function (_) { return true; };
-        var patterns = ts.getFileMatcherPatterns(path, excludeSpecs, includeSpecs, host.useCaseSensitiveFileNames, host.getCurrentDirectory());
-        var excludeRe = patterns.excludePattern && ts.getRegexFromPattern(patterns.excludePattern, host.useCaseSensitiveFileNames);
-        var includeRe = patterns.includeFilePattern && ts.getRegexFromPattern(patterns.includeFilePattern, host.useCaseSensitiveFileNames);
-        if (includeRe) {
-            if (excludeRe) {
-                return function (path) { return !(includeRe.test(path) && !excludeRe.test(path)); };
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var modifiers = parseModifiers();
+            parseExpected(97 /* FunctionKeyword */);
+            var asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
+            var isGenerator = asteriskToken ? 1 /* Yield */ : 0 /* None */;
+            var isAsync = ts.some(modifiers, ts.isAsyncModifier) ? 2 /* Await */ : 0 /* None */;
+            var name = isGenerator && isAsync ? doInYieldAndAwaitContext(parseOptionalBindingIdentifier) :
+                isGenerator ? doInYieldContext(parseOptionalBindingIdentifier) :
+                    isAsync ? doInAwaitContext(parseOptionalBindingIdentifier) :
+                        parseOptionalBindingIdentifier();
+            var typeParameters = parseTypeParameters();
+            var parameters = parseParameters(isGenerator | isAsync);
+            var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+            var body = parseFunctionBlock(isGenerator | isAsync);
+            if (saveDecoratorContext) {
+                setDecoratorContext(/*val*/ true);
             }
-            return function (path) { return !includeRe.test(path); };
-        }
-        if (excludeRe) {
-            return function (path) { return excludeRe.test(path); };
-        }
-        return function (_) { return true; };
-    }
-    function getCustomTypeMapOfCommandLineOption(optionDefinition) {
-        if (optionDefinition.type === "string" || optionDefinition.type === "number" || optionDefinition.type === "boolean" || optionDefinition.type === "object") {
-            // this is of a type CommandLineOptionOfPrimitiveType
-            return undefined;
-        }
-        else if (optionDefinition.type === "list") {
-            return getCustomTypeMapOfCommandLineOption(optionDefinition.element);
+            var node = factory.createFunctionExpression(modifiers, asteriskToken, name, typeParameters, parameters, type, body);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        else {
-            return optionDefinition.type;
+        function parseOptionalBindingIdentifier() {
+            return isBindingIdentifier() ? parseBindingIdentifier() : undefined;
         }
-    }
-    function getNameOfCompilerOptionValue(value, customTypeMap) {
-        // There is a typeMap associated with this command-line option so use it to map value back to its name
-        return ts.forEachEntry(customTypeMap, function (mapValue, key) {
-            if (mapValue === value) {
-                return key;
+        function parseNewExpressionOrNewDotTarget() {
+            var pos = getNodePos();
+            parseExpected(102 /* NewKeyword */);
+            if (parseOptional(24 /* DotToken */)) {
+                var name = parseIdentifierName();
+                return finishNode(factory.createMetaProperty(102 /* NewKeyword */, name), pos);
             }
-        });
-    }
-    function serializeCompilerOptions(options, pathOptions) {
-        return serializeOptionBaseObject(options, getOptionsNameMap(), pathOptions);
-    }
-    function serializeWatchOptions(options) {
-        return serializeOptionBaseObject(options, getWatchOptionsNameMap());
-    }
-    function serializeOptionBaseObject(options, _a, pathOptions) {
-        var optionsNameMap = _a.optionsNameMap;
-        var result = ts.createMap();
-        var getCanonicalFileName = pathOptions && ts.createGetCanonicalFileName(pathOptions.useCaseSensitiveFileNames);
-        var _loop_4 = function (name) {
-            if (ts.hasProperty(options, name)) {
-                // tsconfig only options cannot be specified via command line,
-                // so we can assume that only types that can appear here string | number | boolean
-                if (optionsNameMap.has(name) && optionsNameMap.get(name).category === ts.Diagnostics.Command_line_Options) {
-                    return "continue";
+            var expressionPos = getNodePos();
+            var expression = parsePrimaryExpression();
+            var typeArguments;
+            while (true) {
+                expression = parseMemberExpressionRest(expressionPos, expression, /*allowOptionalChain*/ false);
+                typeArguments = tryParse(parseTypeArgumentsInExpression);
+                if (isTemplateStartOfTaggedTemplate()) {
+                    ts.Debug.assert(!!typeArguments, "Expected a type argument list; all plain tagged template starts should be consumed in 'parseMemberExpressionRest'");
+                    expression = parseTaggedTemplateRest(expressionPos, expression, /*optionalChain*/ undefined, typeArguments);
+                    typeArguments = undefined;
                 }
-                var value = options[name];
-                var optionDefinition = optionsNameMap.get(name.toLowerCase());
-                if (optionDefinition) {
-                    var customTypeMap_1 = getCustomTypeMapOfCommandLineOption(optionDefinition);
-                    if (!customTypeMap_1) {
-                        // There is no map associated with this compiler option then use the value as-is
-                        // This is the case if the value is expect to be string, number, boolean or list of string
-                        if (pathOptions && optionDefinition.isFilePath) {
-                            result.set(name, ts.getRelativePathFromFile(pathOptions.configFilePath, ts.getNormalizedAbsolutePath(value, ts.getDirectoryPath(pathOptions.configFilePath)), getCanonicalFileName));
-                        }
-                        else {
-                            result.set(name, value);
-                        }
-                    }
-                    else {
-                        if (optionDefinition.type === "list") {
-                            result.set(name, value.map(function (element) { return getNameOfCompilerOptionValue(element, customTypeMap_1); })); // TODO: GH#18217
-                        }
-                        else {
-                            // There is a typeMap associated with this command-line option so use it to map value back to its name
-                            result.set(name, getNameOfCompilerOptionValue(value, customTypeMap_1));
-                        }
+                break;
+            }
+            var argumentsArray;
+            if (token() === 20 /* OpenParenToken */) {
+                argumentsArray = parseArgumentList();
+            }
+            else if (typeArguments) {
+                parseErrorAt(pos, scanner.getStartPos(), ts.Diagnostics.A_new_expression_with_type_arguments_must_always_be_followed_by_a_parenthesized_argument_list);
+            }
+            return finishNode(factory.createNewExpression(expression, typeArguments, argumentsArray), pos);
+        }
+        // STATEMENTS
+        function parseBlock(ignoreMissingOpenBrace, diagnosticMessage) {
+            var pos = getNodePos();
+            var openBracePosition = scanner.getTokenPos();
+            if (parseExpected(18 /* OpenBraceToken */, diagnosticMessage) || ignoreMissingOpenBrace) {
+                var multiLine = scanner.hasPrecedingLineBreak();
+                var statements = parseList(1 /* BlockStatements */, parseStatement);
+                if (!parseExpected(19 /* CloseBraceToken */)) {
+                    var lastError = ts.lastOrUndefined(parseDiagnostics);
+                    if (lastError && lastError.code === ts.Diagnostics._0_expected.code) {
+                        ts.addRelatedInfo(lastError, ts.createDetachedDiagnostic(fileName, openBracePosition, 1, ts.Diagnostics.The_parser_expected_to_find_a_to_match_the_token_here));
                     }
                 }
+                return finishNode(factory.createBlock(statements, multiLine), pos);
+            }
+            else {
+                var statements = createMissingList();
+                return finishNode(factory.createBlock(statements, /*multiLine*/ undefined), pos);
             }
-        };
-        for (var name in options) {
-            _loop_4(name);
         }
-        return result;
-    }
-    /**
-     * Generate tsconfig configuration when running command line "--init"
-     * @param options commandlineOptions to be generated into tsconfig.json
-     * @param fileNames array of filenames to be generated into tsconfig.json
-     */
-    /* @internal */
-    function generateTSConfig(options, fileNames, newLine) {
-        var compilerOptions = ts.extend(options, ts.defaultInitCompilerOptions);
-        var compilerOptionsMap = serializeCompilerOptions(compilerOptions);
-        return writeConfigurations();
-        function getDefaultValueForOption(option) {
-            switch (option.type) {
-                case "number":
-                    return 1;
-                case "boolean":
-                    return true;
-                case "string":
-                    return option.isFilePath ? "./" : "";
-                case "list":
-                    return [];
-                case "object":
-                    return {};
-                default:
-                    var iterResult = option.type.keys().next();
-                    if (!iterResult.done)
-                        return iterResult.value;
-                    return ts.Debug.fail("Expected 'option.type' to have entries.");
+        function parseFunctionBlock(flags, diagnosticMessage) {
+            var savedYieldContext = inYieldContext();
+            setYieldContext(!!(flags & 1 /* Yield */));
+            var savedAwaitContext = inAwaitContext();
+            setAwaitContext(!!(flags & 2 /* Await */));
+            var savedTopLevel = topLevel;
+            topLevel = false;
+            // We may be in a [Decorator] context when parsing a function expression or
+            // arrow function. The body of the function is not in [Decorator] context.
+            var saveDecoratorContext = inDecoratorContext();
+            if (saveDecoratorContext) {
+                setDecoratorContext(/*val*/ false);
+            }
+            var block = parseBlock(!!(flags & 16 /* IgnoreMissingOpenBrace */), diagnosticMessage);
+            if (saveDecoratorContext) {
+                setDecoratorContext(/*val*/ true);
             }
+            topLevel = savedTopLevel;
+            setYieldContext(savedYieldContext);
+            setAwaitContext(savedAwaitContext);
+            return block;
         }
-        function makePadding(paddingLength) {
-            return Array(paddingLength + 1).join(" ");
+        function parseEmptyStatement() {
+            var pos = getNodePos();
+            parseExpected(26 /* SemicolonToken */);
+            return finishNode(factory.createEmptyStatement(), pos);
         }
-        function isAllowedOption(_a) {
-            var category = _a.category, name = _a.name;
-            // Skip options which do not have a category or have category `Command_line_Options`
-            // Exclude all possible `Advanced_Options` in tsconfig.json which were NOT defined in command line
-            return category !== undefined
-                && category !== ts.Diagnostics.Command_line_Options
-                && (category !== ts.Diagnostics.Advanced_Options || compilerOptionsMap.has(name));
+        function parseIfStatement() {
+            var pos = getNodePos();
+            parseExpected(98 /* IfKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(21 /* CloseParenToken */);
+            var thenStatement = parseStatement();
+            var elseStatement = parseOptional(90 /* ElseKeyword */) ? parseStatement() : undefined;
+            return finishNode(factory.createIfStatement(expression, thenStatement, elseStatement), pos);
         }
-        function writeConfigurations() {
-            // Filter applicable options to place in the file
-            var categorizedOptions = ts.createMultiMap();
-            for (var _i = 0, optionDeclarations_1 = ts.optionDeclarations; _i < optionDeclarations_1.length; _i++) {
-                var option = optionDeclarations_1[_i];
-                var category = option.category;
-                if (isAllowedOption(option)) {
-                    categorizedOptions.add(ts.getLocaleSpecificMessage(category), option);
-                }
-            }
-            // Serialize all options and their descriptions
-            var marginLength = 0;
-            var seenKnownKeys = 0;
-            var entries = [];
-            categorizedOptions.forEach(function (options, category) {
-                if (entries.length !== 0) {
-                    entries.push({ value: "" });
+        function parseDoStatement() {
+            var pos = getNodePos();
+            parseExpected(89 /* DoKeyword */);
+            var statement = parseStatement();
+            parseExpected(114 /* WhileKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(21 /* CloseParenToken */);
+            // From: https://mail.mozilla.org/pipermail/es-discuss/2011-August/016188.html
+            // 157 min --- All allen at wirfs-brock.com CONF --- "do{;}while(false)false" prohibited in
+            // spec but allowed in consensus reality. Approved -- this is the de-facto standard whereby
+            //  do;while(0)x will have a semicolon inserted before x.
+            parseOptional(26 /* SemicolonToken */);
+            return finishNode(factory.createDoStatement(statement, expression), pos);
+        }
+        function parseWhileStatement() {
+            var pos = getNodePos();
+            parseExpected(114 /* WhileKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(21 /* CloseParenToken */);
+            var statement = parseStatement();
+            return finishNode(factory.createWhileStatement(expression, statement), pos);
+        }
+        function parseForOrForInOrForOfStatement() {
+            var pos = getNodePos();
+            parseExpected(96 /* ForKeyword */);
+            var awaitToken = parseOptionalToken(130 /* AwaitKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var initializer;
+            if (token() !== 26 /* SemicolonToken */) {
+                if (token() === 112 /* VarKeyword */ || token() === 118 /* LetKeyword */ || token() === 84 /* ConstKeyword */) {
+                    initializer = parseVariableDeclarationList(/*inForStatementInitializer*/ true);
                 }
-                entries.push({ value: "/* " + category + " */" });
-                for (var _i = 0, options_1 = options; _i < options_1.length; _i++) {
-                    var option = options_1[_i];
-                    var optionName = void 0;
-                    if (compilerOptionsMap.has(option.name)) {
-                        optionName = "\"" + option.name + "\": " + JSON.stringify(compilerOptionsMap.get(option.name)) + ((seenKnownKeys += 1) === compilerOptionsMap.size ? "" : ",");
-                    }
-                    else {
-                        optionName = "// \"" + option.name + "\": " + JSON.stringify(getDefaultValueForOption(option)) + ",";
-                    }
-                    entries.push({
-                        value: optionName,
-                        description: "/* " + (option.description && ts.getLocaleSpecificMessage(option.description) || option.name) + " */"
-                    });
-                    marginLength = Math.max(optionName.length, marginLength);
+                else {
+                    initializer = disallowInAnd(parseExpression);
                 }
-            });
-            // Write the output
-            var tab = makePadding(2);
-            var result = [];
-            result.push("{");
-            result.push(tab + "\"compilerOptions\": {");
-            result.push("" + tab + tab + "/* " + ts.getLocaleSpecificMessage(ts.Diagnostics.Visit_https_Colon_Slash_Slashaka_ms_Slashtsconfig_json_to_read_more_about_this_file) + " */");
-            result.push("");
-            // Print out each row, aligning all the descriptions on the same column.
-            for (var _a = 0, entries_3 = entries; _a < entries_3.length; _a++) {
-                var entry = entries_3[_a];
-                var value = entry.value, _b = entry.description, description = _b === void 0 ? "" : _b;
-                result.push(value && "" + tab + tab + value + (description && (makePadding(marginLength - value.length + 2) + description)));
             }
-            if (fileNames.length) {
-                result.push(tab + "},");
-                result.push(tab + "\"files\": [");
-                for (var i = 0; i < fileNames.length; i++) {
-                    result.push("" + tab + tab + JSON.stringify(fileNames[i]) + (i === fileNames.length - 1 ? "" : ","));
-                }
-                result.push(tab + "]");
+            var node;
+            if (awaitToken ? parseExpected(156 /* OfKeyword */) : parseOptional(156 /* OfKeyword */)) {
+                var expression = allowInAnd(parseAssignmentExpressionOrHigher);
+                parseExpected(21 /* CloseParenToken */);
+                node = factory.createForOfStatement(awaitToken, initializer, expression, parseStatement());
+            }
+            else if (parseOptional(100 /* InKeyword */)) {
+                var expression = allowInAnd(parseExpression);
+                parseExpected(21 /* CloseParenToken */);
+                node = factory.createForInStatement(initializer, expression, parseStatement());
             }
             else {
-                result.push(tab + "}");
+                parseExpected(26 /* SemicolonToken */);
+                var condition = token() !== 26 /* SemicolonToken */ && token() !== 21 /* CloseParenToken */
+                    ? allowInAnd(parseExpression)
+                    : undefined;
+                parseExpected(26 /* SemicolonToken */);
+                var incrementor = token() !== 21 /* CloseParenToken */
+                    ? allowInAnd(parseExpression)
+                    : undefined;
+                parseExpected(21 /* CloseParenToken */);
+                node = factory.createForStatement(initializer, condition, incrementor, parseStatement());
             }
-            result.push("}");
-            return result.join(newLine) + newLine;
+            return finishNode(node, pos);
         }
-    }
-    ts.generateTSConfig = generateTSConfig;
-    /* @internal */
-    function convertToOptionsWithAbsolutePaths(options, toAbsolutePath) {
-        var result = {};
-        var optionsNameMap = getOptionsNameMap().optionsNameMap;
-        for (var name in options) {
-            if (ts.hasProperty(options, name)) {
-                result[name] = convertToOptionValueWithAbsolutePaths(optionsNameMap.get(name.toLowerCase()), options[name], toAbsolutePath);
+        function parseBreakOrContinueStatement(kind) {
+            var pos = getNodePos();
+            parseExpected(kind === 241 /* BreakStatement */ ? 80 /* BreakKeyword */ : 85 /* ContinueKeyword */);
+            var label = canParseSemicolon() ? undefined : parseIdentifier();
+            parseSemicolon();
+            var node = kind === 241 /* BreakStatement */
+                ? factory.createBreakStatement(label)
+                : factory.createContinueStatement(label);
+            return finishNode(node, pos);
+        }
+        function parseReturnStatement() {
+            var pos = getNodePos();
+            parseExpected(104 /* ReturnKeyword */);
+            var expression = canParseSemicolon() ? undefined : allowInAnd(parseExpression);
+            parseSemicolon();
+            return finishNode(factory.createReturnStatement(expression), pos);
+        }
+        function parseWithStatement() {
+            var pos = getNodePos();
+            parseExpected(115 /* WithKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(21 /* CloseParenToken */);
+            var statement = doInsideOfContext(16777216 /* InWithStatement */, parseStatement);
+            return finishNode(factory.createWithStatement(expression, statement), pos);
+        }
+        function parseCaseClause() {
+            var pos = getNodePos();
+            parseExpected(81 /* CaseKeyword */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(58 /* ColonToken */);
+            var statements = parseList(3 /* SwitchClauseStatements */, parseStatement);
+            return finishNode(factory.createCaseClause(expression, statements), pos);
+        }
+        function parseDefaultClause() {
+            var pos = getNodePos();
+            parseExpected(87 /* DefaultKeyword */);
+            parseExpected(58 /* ColonToken */);
+            var statements = parseList(3 /* SwitchClauseStatements */, parseStatement);
+            return finishNode(factory.createDefaultClause(statements), pos);
+        }
+        function parseCaseOrDefaultClause() {
+            return token() === 81 /* CaseKeyword */ ? parseCaseClause() : parseDefaultClause();
+        }
+        function parseCaseBlock() {
+            var pos = getNodePos();
+            parseExpected(18 /* OpenBraceToken */);
+            var clauses = parseList(2 /* SwitchClauses */, parseCaseOrDefaultClause);
+            parseExpected(19 /* CloseBraceToken */);
+            return finishNode(factory.createCaseBlock(clauses), pos);
+        }
+        function parseSwitchStatement() {
+            var pos = getNodePos();
+            parseExpected(106 /* SwitchKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var expression = allowInAnd(parseExpression);
+            parseExpected(21 /* CloseParenToken */);
+            var caseBlock = parseCaseBlock();
+            return finishNode(factory.createSwitchStatement(expression, caseBlock), pos);
+        }
+        function parseThrowStatement() {
+            // ThrowStatement[Yield] :
+            //      throw [no LineTerminator here]Expression[In, ?Yield];
+            var pos = getNodePos();
+            parseExpected(108 /* ThrowKeyword */);
+            // Because of automatic semicolon insertion, we need to report error if this
+            // throw could be terminated with a semicolon.  Note: we can't call 'parseExpression'
+            // directly as that might consume an expression on the following line.
+            // Instead, we create a "missing" identifier, but don't report an error. The actual error
+            // will be reported in the grammar walker.
+            var expression = scanner.hasPrecedingLineBreak() ? undefined : allowInAnd(parseExpression);
+            if (expression === undefined) {
+                identifierCount++;
+                expression = finishNode(factory.createIdentifier(""), getNodePos());
             }
+            parseSemicolon();
+            return finishNode(factory.createThrowStatement(expression), pos);
         }
-        if (result.configFilePath) {
-            result.configFilePath = toAbsolutePath(result.configFilePath);
+        // TODO: Review for error recovery
+        function parseTryStatement() {
+            var pos = getNodePos();
+            parseExpected(110 /* TryKeyword */);
+            var tryBlock = parseBlock(/*ignoreMissingOpenBrace*/ false);
+            var catchClause = token() === 82 /* CatchKeyword */ ? parseCatchClause() : undefined;
+            // If we don't have a catch clause, then we must have a finally clause.  Try to parse
+            // one out no matter what.
+            var finallyBlock;
+            if (!catchClause || token() === 95 /* FinallyKeyword */) {
+                parseExpected(95 /* FinallyKeyword */);
+                finallyBlock = parseBlock(/*ignoreMissingOpenBrace*/ false);
+            }
+            return finishNode(factory.createTryStatement(tryBlock, catchClause, finallyBlock), pos);
         }
-        return result;
-    }
-    ts.convertToOptionsWithAbsolutePaths = convertToOptionsWithAbsolutePaths;
-    function convertToOptionValueWithAbsolutePaths(option, value, toAbsolutePath) {
-        if (option && !isNullOrUndefined(value)) {
-            if (option.type === "list") {
-                var values = value;
-                if (option.element.isFilePath && values.length) {
-                    return values.map(toAbsolutePath);
-                }
+        function parseCatchClause() {
+            var pos = getNodePos();
+            parseExpected(82 /* CatchKeyword */);
+            var variableDeclaration;
+            if (parseOptional(20 /* OpenParenToken */)) {
+                variableDeclaration = parseVariableDeclaration();
+                parseExpected(21 /* CloseParenToken */);
             }
-            else if (option.isFilePath) {
-                return toAbsolutePath(value);
+            else {
+                // Keep shape of node to avoid degrading performance.
+                variableDeclaration = undefined;
             }
+            var block = parseBlock(/*ignoreMissingOpenBrace*/ false);
+            return finishNode(factory.createCatchClause(variableDeclaration, block), pos);
         }
-        return value;
-    }
-    /**
-     * Parse the contents of a config file (tsconfig.json).
-     * @param json The contents of the config file to parse
-     * @param host Instance of ParseConfigHost used to enumerate files in folder.
-     * @param basePath A root directory to resolve relative path entries in the config
-     *    file to. e.g. outDir
-     */
-    function parseJsonConfigFileContent(json, host, basePath, existingOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache, existingWatchOptions) {
-        return parseJsonConfigFileContentWorker(json, /*sourceFile*/ undefined, host, basePath, existingOptions, existingWatchOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache);
-    }
-    ts.parseJsonConfigFileContent = parseJsonConfigFileContent;
-    /**
-     * Parse the contents of a config file (tsconfig.json).
-     * @param jsonNode The contents of the config file to parse
-     * @param host Instance of ParseConfigHost used to enumerate files in folder.
-     * @param basePath A root directory to resolve relative path entries in the config
-     *    file to. e.g. outDir
-     */
-    function parseJsonSourceFileConfigFileContent(sourceFile, host, basePath, existingOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache, existingWatchOptions) {
-        return parseJsonConfigFileContentWorker(/*json*/ undefined, sourceFile, host, basePath, existingOptions, existingWatchOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache);
-    }
-    ts.parseJsonSourceFileConfigFileContent = parseJsonSourceFileConfigFileContent;
-    /*@internal*/
-    function setConfigFileInOptions(options, configFile) {
-        if (configFile) {
-            Object.defineProperty(options, "configFile", { enumerable: false, writable: false, value: configFile });
+        function parseDebuggerStatement() {
+            var pos = getNodePos();
+            parseExpected(86 /* DebuggerKeyword */);
+            parseSemicolon();
+            return finishNode(factory.createDebuggerStatement(), pos);
         }
-    }
-    ts.setConfigFileInOptions = setConfigFileInOptions;
-    function isNullOrUndefined(x) {
-        return x === undefined || x === null; // eslint-disable-line no-null/no-null
-    }
-    function directoryOfCombinedPath(fileName, basePath) {
-        // Use the `getNormalizedAbsolutePath` function to avoid canonicalizing the path, as it must remain noncanonical
-        // until consistent casing errors are reported
-        return ts.getDirectoryPath(ts.getNormalizedAbsolutePath(fileName, basePath));
-    }
-    /**
-     * Parse the contents of a config file from json or json source file (tsconfig.json).
-     * @param json The contents of the config file to parse
-     * @param sourceFile sourceFile corresponding to the Json
-     * @param host Instance of ParseConfigHost used to enumerate files in folder.
-     * @param basePath A root directory to resolve relative path entries in the config
-     *    file to. e.g. outDir
-     * @param resolutionStack Only present for backwards-compatibility. Should be empty.
-     */
-    function parseJsonConfigFileContentWorker(json, sourceFile, host, basePath, existingOptions, existingWatchOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache) {
-        if (existingOptions === void 0) { existingOptions = {}; }
-        if (resolutionStack === void 0) { resolutionStack = []; }
-        if (extraFileExtensions === void 0) { extraFileExtensions = []; }
-        ts.Debug.assert((json === undefined && sourceFile !== undefined) || (json !== undefined && sourceFile === undefined));
-        var errors = [];
-        var parsedConfig = parseConfig(json, sourceFile, host, basePath, configFileName, resolutionStack, errors, extendedConfigCache);
-        var raw = parsedConfig.raw;
-        var options = ts.extend(existingOptions, parsedConfig.options || {});
-        var watchOptions = existingWatchOptions && parsedConfig.watchOptions ?
-            ts.extend(existingWatchOptions, parsedConfig.watchOptions) :
-            parsedConfig.watchOptions || existingWatchOptions;
-        options.configFilePath = configFileName && ts.normalizeSlashes(configFileName);
-        setConfigFileInOptions(options, sourceFile);
-        var projectReferences;
-        var _a = getFileNames(), fileNames = _a.fileNames, wildcardDirectories = _a.wildcardDirectories, spec = _a.spec;
-        return {
-            options: options,
-            watchOptions: watchOptions,
-            fileNames: fileNames,
-            projectReferences: projectReferences,
-            typeAcquisition: parsedConfig.typeAcquisition || getDefaultTypeAcquisition(),
-            raw: raw,
-            errors: errors,
-            wildcardDirectories: wildcardDirectories,
-            compileOnSave: !!raw.compileOnSave,
-            configFileSpecs: spec
-        };
-        function getFileNames() {
-            var filesSpecs;
-            if (ts.hasProperty(raw, "files") && !isNullOrUndefined(raw.files)) {
-                if (ts.isArray(raw.files)) {
-                    filesSpecs = raw.files;
-                    var hasReferences = ts.hasProperty(raw, "references") && !isNullOrUndefined(raw.references);
-                    var hasZeroOrNoReferences = !hasReferences || raw.references.length === 0;
-                    var hasExtends = ts.hasProperty(raw, "extends");
-                    if (filesSpecs.length === 0 && hasZeroOrNoReferences && !hasExtends) {
-                        if (sourceFile) {
-                            var fileName = configFileName || "tsconfig.json";
-                            var diagnosticMessage = ts.Diagnostics.The_files_list_in_config_file_0_is_empty;
-                            var nodeValue = ts.firstDefined(ts.getTsConfigPropArray(sourceFile, "files"), function (property) { return property.initializer; });
-                            var error = nodeValue
-                                ? ts.createDiagnosticForNodeInSourceFile(sourceFile, nodeValue, diagnosticMessage, fileName)
-                                : ts.createCompilerDiagnostic(diagnosticMessage, fileName);
-                            errors.push(error);
+        function parseExpressionOrLabeledStatement() {
+            // Avoiding having to do the lookahead for a labeled statement by just trying to parse
+            // out an expression, seeing if it is identifier and then seeing if it is followed by
+            // a colon.
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var node;
+            var hasParen = token() === 20 /* OpenParenToken */;
+            var expression = allowInAnd(parseExpression);
+            if (ts.isIdentifier(expression) && parseOptional(58 /* ColonToken */)) {
+                node = factory.createLabeledStatement(expression, parseStatement());
+            }
+            else {
+                parseSemicolon();
+                node = factory.createExpressionStatement(expression);
+                if (hasParen) {
+                    // do not parse the same jsdoc twice
+                    hasJSDoc = false;
+                }
+            }
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function nextTokenIsIdentifierOrKeywordOnSameLine() {
+            nextToken();
+            return ts.tokenIsIdentifierOrKeyword(token()) && !scanner.hasPrecedingLineBreak();
+        }
+        function nextTokenIsClassKeywordOnSameLine() {
+            nextToken();
+            return token() === 83 /* ClassKeyword */ && !scanner.hasPrecedingLineBreak();
+        }
+        function nextTokenIsFunctionKeywordOnSameLine() {
+            nextToken();
+            return token() === 97 /* FunctionKeyword */ && !scanner.hasPrecedingLineBreak();
+        }
+        function nextTokenIsIdentifierOrKeywordOrLiteralOnSameLine() {
+            nextToken();
+            return (ts.tokenIsIdentifierOrKeyword(token()) || token() === 8 /* NumericLiteral */ || token() === 9 /* BigIntLiteral */ || token() === 10 /* StringLiteral */) && !scanner.hasPrecedingLineBreak();
+        }
+        function isDeclaration() {
+            while (true) {
+                switch (token()) {
+                    case 112 /* VarKeyword */:
+                    case 118 /* LetKeyword */:
+                    case 84 /* ConstKeyword */:
+                    case 97 /* FunctionKeyword */:
+                    case 83 /* ClassKeyword */:
+                    case 91 /* EnumKeyword */:
+                        return true;
+                    // 'declare', 'module', 'namespace', 'interface'* and 'type' are all legal JavaScript identifiers;
+                    // however, an identifier cannot be followed by another identifier on the same line. This is what we
+                    // count on to parse out the respective declarations. For instance, we exploit this to say that
+                    //
+                    //    namespace n
+                    //
+                    // can be none other than the beginning of a namespace declaration, but need to respect that JavaScript sees
+                    //
+                    //    namespace
+                    //    n
+                    //
+                    // as the identifier 'namespace' on one line followed by the identifier 'n' on another.
+                    // We need to look one token ahead to see if it permissible to try parsing a declaration.
+                    //
+                    // *Note*: 'interface' is actually a strict mode reserved word. So while
+                    //
+                    //   "use strict"
+                    //   interface
+                    //   I {}
+                    //
+                    // could be legal, it would add complexity for very little gain.
+                    case 117 /* InterfaceKeyword */:
+                    case 149 /* TypeKeyword */:
+                        return nextTokenIsIdentifierOnSameLine();
+                    case 139 /* ModuleKeyword */:
+                    case 140 /* NamespaceKeyword */:
+                        return nextTokenIsIdentifierOrStringLiteralOnSameLine();
+                    case 125 /* AbstractKeyword */:
+                    case 129 /* AsyncKeyword */:
+                    case 133 /* DeclareKeyword */:
+                    case 120 /* PrivateKeyword */:
+                    case 121 /* ProtectedKeyword */:
+                    case 122 /* PublicKeyword */:
+                    case 142 /* ReadonlyKeyword */:
+                        nextToken();
+                        // ASI takes effect for this modifier.
+                        if (scanner.hasPrecedingLineBreak()) {
+                            return false;
                         }
-                        else {
-                            createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.The_files_list_in_config_file_0_is_empty, configFileName || "tsconfig.json");
+                        continue;
+                    case 154 /* GlobalKeyword */:
+                        nextToken();
+                        return token() === 18 /* OpenBraceToken */ || token() === 78 /* Identifier */ || token() === 92 /* ExportKeyword */;
+                    case 99 /* ImportKeyword */:
+                        nextToken();
+                        return token() === 10 /* StringLiteral */ || token() === 41 /* AsteriskToken */ ||
+                            token() === 18 /* OpenBraceToken */ || ts.tokenIsIdentifierOrKeyword(token());
+                    case 92 /* ExportKeyword */:
+                        var currentToken_1 = nextToken();
+                        if (currentToken_1 === 149 /* TypeKeyword */) {
+                            currentToken_1 = lookAhead(nextToken);
+                        }
+                        if (currentToken_1 === 62 /* EqualsToken */ || currentToken_1 === 41 /* AsteriskToken */ ||
+                            currentToken_1 === 18 /* OpenBraceToken */ || currentToken_1 === 87 /* DefaultKeyword */ ||
+                            currentToken_1 === 126 /* AsKeyword */) {
+                            return true;
                         }
+                        continue;
+                    case 123 /* StaticKeyword */:
+                        nextToken();
+                        continue;
+                    default:
+                        return false;
+                }
+            }
+        }
+        function isStartOfDeclaration() {
+            return lookAhead(isDeclaration);
+        }
+        function isStartOfStatement() {
+            switch (token()) {
+                case 59 /* AtToken */:
+                case 26 /* SemicolonToken */:
+                case 18 /* OpenBraceToken */:
+                case 112 /* VarKeyword */:
+                case 118 /* LetKeyword */:
+                case 97 /* FunctionKeyword */:
+                case 83 /* ClassKeyword */:
+                case 91 /* EnumKeyword */:
+                case 98 /* IfKeyword */:
+                case 89 /* DoKeyword */:
+                case 114 /* WhileKeyword */:
+                case 96 /* ForKeyword */:
+                case 85 /* ContinueKeyword */:
+                case 80 /* BreakKeyword */:
+                case 104 /* ReturnKeyword */:
+                case 115 /* WithKeyword */:
+                case 106 /* SwitchKeyword */:
+                case 108 /* ThrowKeyword */:
+                case 110 /* TryKeyword */:
+                case 86 /* DebuggerKeyword */:
+                // 'catch' and 'finally' do not actually indicate that the code is part of a statement,
+                // however, we say they are here so that we may gracefully parse them and error later.
+                // falls through
+                case 82 /* CatchKeyword */:
+                case 95 /* FinallyKeyword */:
+                    return true;
+                case 99 /* ImportKeyword */:
+                    return isStartOfDeclaration() || lookAhead(nextTokenIsOpenParenOrLessThanOrDot);
+                case 84 /* ConstKeyword */:
+                case 92 /* ExportKeyword */:
+                    return isStartOfDeclaration();
+                case 129 /* AsyncKeyword */:
+                case 133 /* DeclareKeyword */:
+                case 117 /* InterfaceKeyword */:
+                case 139 /* ModuleKeyword */:
+                case 140 /* NamespaceKeyword */:
+                case 149 /* TypeKeyword */:
+                case 154 /* GlobalKeyword */:
+                    // When these don't start a declaration, they're an identifier in an expression statement
+                    return true;
+                case 122 /* PublicKeyword */:
+                case 120 /* PrivateKeyword */:
+                case 121 /* ProtectedKeyword */:
+                case 123 /* StaticKeyword */:
+                case 142 /* ReadonlyKeyword */:
+                    // When these don't start a declaration, they may be the start of a class member if an identifier
+                    // immediately follows. Otherwise they're an identifier in an expression statement.
+                    return isStartOfDeclaration() || !lookAhead(nextTokenIsIdentifierOrKeywordOnSameLine);
+                default:
+                    return isStartOfExpression();
+            }
+        }
+        function nextTokenIsIdentifierOrStartOfDestructuring() {
+            nextToken();
+            return isIdentifier() || token() === 18 /* OpenBraceToken */ || token() === 22 /* OpenBracketToken */;
+        }
+        function isLetDeclaration() {
+            // In ES6 'let' always starts a lexical declaration if followed by an identifier or {
+            // or [.
+            return lookAhead(nextTokenIsIdentifierOrStartOfDestructuring);
+        }
+        function parseStatement() {
+            switch (token()) {
+                case 26 /* SemicolonToken */:
+                    return parseEmptyStatement();
+                case 18 /* OpenBraceToken */:
+                    return parseBlock(/*ignoreMissingOpenBrace*/ false);
+                case 112 /* VarKeyword */:
+                    return parseVariableStatement(getNodePos(), hasPrecedingJSDocComment(), /*decorators*/ undefined, /*modifiers*/ undefined);
+                case 118 /* LetKeyword */:
+                    if (isLetDeclaration()) {
+                        return parseVariableStatement(getNodePos(), hasPrecedingJSDocComment(), /*decorators*/ undefined, /*modifiers*/ undefined);
+                    }
+                    break;
+                case 97 /* FunctionKeyword */:
+                    return parseFunctionDeclaration(getNodePos(), hasPrecedingJSDocComment(), /*decorators*/ undefined, /*modifiers*/ undefined);
+                case 83 /* ClassKeyword */:
+                    return parseClassDeclaration(getNodePos(), hasPrecedingJSDocComment(), /*decorators*/ undefined, /*modifiers*/ undefined);
+                case 98 /* IfKeyword */:
+                    return parseIfStatement();
+                case 89 /* DoKeyword */:
+                    return parseDoStatement();
+                case 114 /* WhileKeyword */:
+                    return parseWhileStatement();
+                case 96 /* ForKeyword */:
+                    return parseForOrForInOrForOfStatement();
+                case 85 /* ContinueKeyword */:
+                    return parseBreakOrContinueStatement(240 /* ContinueStatement */);
+                case 80 /* BreakKeyword */:
+                    return parseBreakOrContinueStatement(241 /* BreakStatement */);
+                case 104 /* ReturnKeyword */:
+                    return parseReturnStatement();
+                case 115 /* WithKeyword */:
+                    return parseWithStatement();
+                case 106 /* SwitchKeyword */:
+                    return parseSwitchStatement();
+                case 108 /* ThrowKeyword */:
+                    return parseThrowStatement();
+                case 110 /* TryKeyword */:
+                // Include 'catch' and 'finally' for error recovery.
+                // falls through
+                case 82 /* CatchKeyword */:
+                case 95 /* FinallyKeyword */:
+                    return parseTryStatement();
+                case 86 /* DebuggerKeyword */:
+                    return parseDebuggerStatement();
+                case 59 /* AtToken */:
+                    return parseDeclaration();
+                case 129 /* AsyncKeyword */:
+                case 117 /* InterfaceKeyword */:
+                case 149 /* TypeKeyword */:
+                case 139 /* ModuleKeyword */:
+                case 140 /* NamespaceKeyword */:
+                case 133 /* DeclareKeyword */:
+                case 84 /* ConstKeyword */:
+                case 91 /* EnumKeyword */:
+                case 92 /* ExportKeyword */:
+                case 99 /* ImportKeyword */:
+                case 120 /* PrivateKeyword */:
+                case 121 /* ProtectedKeyword */:
+                case 122 /* PublicKeyword */:
+                case 125 /* AbstractKeyword */:
+                case 123 /* StaticKeyword */:
+                case 142 /* ReadonlyKeyword */:
+                case 154 /* GlobalKeyword */:
+                    if (isStartOfDeclaration()) {
+                        return parseDeclaration();
                     }
-                }
-                else {
-                    createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "files", "Array");
-                }
+                    break;
             }
-            var includeSpecs;
-            if (ts.hasProperty(raw, "include") && !isNullOrUndefined(raw.include)) {
-                if (ts.isArray(raw.include)) {
-                    includeSpecs = raw.include;
-                }
-                else {
-                    createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "include", "Array");
+            return parseExpressionOrLabeledStatement();
+        }
+        function isDeclareModifier(modifier) {
+            return modifier.kind === 133 /* DeclareKeyword */;
+        }
+        function parseDeclaration() {
+            // TODO: Can we hold onto the parsed decorators/modifiers and advance the scanner
+            //       if we can't reuse the declaration, so that we don't do this work twice?
+            //
+            // `parseListElement` attempted to get the reused node at this position,
+            // but the ambient context flag was not yet set, so the node appeared
+            // not reusable in that context.
+            var isAmbient = ts.some(lookAhead(function () { return (parseDecorators(), parseModifiers()); }), isDeclareModifier);
+            if (isAmbient) {
+                var node = tryReuseAmbientDeclaration();
+                if (node) {
+                    return node;
                 }
             }
-            var excludeSpecs;
-            if (ts.hasProperty(raw, "exclude") && !isNullOrUndefined(raw.exclude)) {
-                if (ts.isArray(raw.exclude)) {
-                    excludeSpecs = raw.exclude;
-                }
-                else {
-                    createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "exclude", "Array");
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var decorators = parseDecorators();
+            var modifiers = parseModifiers();
+            if (isAmbient) {
+                for (var _i = 0, _a = modifiers; _i < _a.length; _i++) {
+                    var m = _a[_i];
+                    m.flags |= 8388608 /* Ambient */;
                 }
+                return doInsideOfContext(8388608 /* Ambient */, function () { return parseDeclarationWorker(pos, hasJSDoc, decorators, modifiers); });
             }
-            else if (raw.compilerOptions) {
-                var outDir = raw.compilerOptions.outDir;
-                var declarationDir = raw.compilerOptions.declarationDir;
-                if (outDir || declarationDir) {
-                    excludeSpecs = [outDir, declarationDir].filter(function (d) { return !!d; });
-                }
+            else {
+                return parseDeclarationWorker(pos, hasJSDoc, decorators, modifiers);
             }
-            if (filesSpecs === undefined && includeSpecs === undefined) {
-                includeSpecs = ["**/*"];
+        }
+        function tryReuseAmbientDeclaration() {
+            return doInsideOfContext(8388608 /* Ambient */, function () {
+                var node = currentNode(parsingContext);
+                if (node) {
+                    return consumeNode(node);
+                }
+            });
+        }
+        function parseDeclarationWorker(pos, hasJSDoc, decorators, modifiers) {
+            switch (token()) {
+                case 112 /* VarKeyword */:
+                case 118 /* LetKeyword */:
+                case 84 /* ConstKeyword */:
+                    return parseVariableStatement(pos, hasJSDoc, decorators, modifiers);
+                case 97 /* FunctionKeyword */:
+                    return parseFunctionDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 83 /* ClassKeyword */:
+                    return parseClassDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 117 /* InterfaceKeyword */:
+                    return parseInterfaceDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 149 /* TypeKeyword */:
+                    return parseTypeAliasDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 91 /* EnumKeyword */:
+                    return parseEnumDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 154 /* GlobalKeyword */:
+                case 139 /* ModuleKeyword */:
+                case 140 /* NamespaceKeyword */:
+                    return parseModuleDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 99 /* ImportKeyword */:
+                    return parseImportDeclarationOrImportEqualsDeclaration(pos, hasJSDoc, decorators, modifiers);
+                case 92 /* ExportKeyword */:
+                    nextToken();
+                    switch (token()) {
+                        case 87 /* DefaultKeyword */:
+                        case 62 /* EqualsToken */:
+                            return parseExportAssignment(pos, hasJSDoc, decorators, modifiers);
+                        case 126 /* AsKeyword */:
+                            return parseNamespaceExportDeclaration(pos, hasJSDoc, decorators, modifiers);
+                        default:
+                            return parseExportDeclaration(pos, hasJSDoc, decorators, modifiers);
+                    }
+                default:
+                    if (decorators || modifiers) {
+                        // We reached this point because we encountered decorators and/or modifiers and assumed a declaration
+                        // would follow. For recovery and error reporting purposes, return an incomplete declaration.
+                        var missing = createMissingNode(271 /* MissingDeclaration */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Declaration_expected);
+                        ts.setTextRangePos(missing, pos);
+                        missing.decorators = decorators;
+                        missing.modifiers = modifiers;
+                        return missing;
+                    }
+                    return undefined; // TODO: GH#18217
             }
-            var result = matchFileNames(filesSpecs, includeSpecs, excludeSpecs, configFileName ? directoryOfCombinedPath(configFileName, basePath) : basePath, options, host, errors, extraFileExtensions, sourceFile);
-            if (shouldReportNoInputFiles(result, canJsonReportNoInutFiles(raw), resolutionStack)) {
-                errors.push(getErrorForNoInputFiles(result.spec, configFileName));
+        }
+        function nextTokenIsIdentifierOrStringLiteralOnSameLine() {
+            nextToken();
+            return !scanner.hasPrecedingLineBreak() && (isIdentifier() || token() === 10 /* StringLiteral */);
+        }
+        function parseFunctionBlockOrSemicolon(flags, diagnosticMessage) {
+            if (token() !== 18 /* OpenBraceToken */ && canParseSemicolon()) {
+                parseSemicolon();
+                return;
             }
-            if (ts.hasProperty(raw, "references") && !isNullOrUndefined(raw.references)) {
-                if (ts.isArray(raw.references)) {
-                    for (var _i = 0, _a = raw.references; _i < _a.length; _i++) {
-                        var ref = _a[_i];
-                        if (typeof ref.path !== "string") {
-                            createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "reference.path", "string");
-                        }
-                        else {
-                            (projectReferences || (projectReferences = [])).push({
-                                path: ts.getNormalizedAbsolutePath(ref.path, basePath),
-                                originalPath: ref.path,
-                                prepend: ref.prepend,
-                                circular: ref.circular
-                            });
-                        }
-                    }
-                }
-                else {
-                    createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "references", "Array");
-                }
+            return parseFunctionBlock(flags, diagnosticMessage);
+        }
+        // DECLARATIONS
+        function parseArrayBindingElement() {
+            var pos = getNodePos();
+            if (token() === 27 /* CommaToken */) {
+                return finishNode(factory.createOmittedExpression(), pos);
             }
-            return result;
+            var dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
+            var name = parseIdentifierOrPattern();
+            var initializer = parseInitializer();
+            return finishNode(factory.createBindingElement(dotDotDotToken, /*propertyName*/ undefined, name, initializer), pos);
         }
-        function createCompilerDiagnosticOnlyIfJson(message, arg0, arg1) {
-            if (!sourceFile) {
-                errors.push(ts.createCompilerDiagnostic(message, arg0, arg1));
+        function parseObjectBindingElement() {
+            var pos = getNodePos();
+            var dotDotDotToken = parseOptionalToken(25 /* DotDotDotToken */);
+            var tokenIsIdentifier = isBindingIdentifier();
+            var propertyName = parsePropertyName();
+            var name;
+            if (tokenIsIdentifier && token() !== 58 /* ColonToken */) {
+                name = propertyName;
+                propertyName = undefined;
             }
+            else {
+                parseExpected(58 /* ColonToken */);
+                name = parseIdentifierOrPattern();
+            }
+            var initializer = parseInitializer();
+            return finishNode(factory.createBindingElement(dotDotDotToken, propertyName, name, initializer), pos);
         }
-    }
-    function isErrorNoInputFiles(error) {
-        return error.code === ts.Diagnostics.No_inputs_were_found_in_config_file_0_Specified_include_paths_were_1_and_exclude_paths_were_2.code;
-    }
-    function getErrorForNoInputFiles(_a, configFileName) {
-        var includeSpecs = _a.includeSpecs, excludeSpecs = _a.excludeSpecs;
-        return ts.createCompilerDiagnostic(ts.Diagnostics.No_inputs_were_found_in_config_file_0_Specified_include_paths_were_1_and_exclude_paths_were_2, configFileName || "tsconfig.json", JSON.stringify(includeSpecs || []), JSON.stringify(excludeSpecs || []));
-    }
-    function shouldReportNoInputFiles(result, canJsonReportNoInutFiles, resolutionStack) {
-        return result.fileNames.length === 0 && canJsonReportNoInutFiles && (!resolutionStack || resolutionStack.length === 0);
-    }
-    /*@internal*/
-    function canJsonReportNoInutFiles(raw) {
-        return !ts.hasProperty(raw, "files") && !ts.hasProperty(raw, "references");
-    }
-    ts.canJsonReportNoInutFiles = canJsonReportNoInutFiles;
-    /*@internal*/
-    function updateErrorForNoInputFiles(result, configFileName, configFileSpecs, configParseDiagnostics, canJsonReportNoInutFiles) {
-        var existingErrors = configParseDiagnostics.length;
-        if (shouldReportNoInputFiles(result, canJsonReportNoInutFiles)) {
-            configParseDiagnostics.push(getErrorForNoInputFiles(configFileSpecs, configFileName));
+        function parseObjectBindingPattern() {
+            var pos = getNodePos();
+            parseExpected(18 /* OpenBraceToken */);
+            var elements = parseDelimitedList(9 /* ObjectBindingElements */, parseObjectBindingElement);
+            parseExpected(19 /* CloseBraceToken */);
+            return finishNode(factory.createObjectBindingPattern(elements), pos);
         }
-        else {
-            ts.filterMutate(configParseDiagnostics, function (error) { return !isErrorNoInputFiles(error); });
+        function parseArrayBindingPattern() {
+            var pos = getNodePos();
+            parseExpected(22 /* OpenBracketToken */);
+            var elements = parseDelimitedList(10 /* ArrayBindingElements */, parseArrayBindingElement);
+            parseExpected(23 /* CloseBracketToken */);
+            return finishNode(factory.createArrayBindingPattern(elements), pos);
         }
-        return existingErrors !== configParseDiagnostics.length;
-    }
-    ts.updateErrorForNoInputFiles = updateErrorForNoInputFiles;
-    function isSuccessfulParsedTsconfig(value) {
-        return !!value.options;
-    }
-    /**
-     * This *just* extracts options/include/exclude/files out of a config file.
-     * It does *not* resolve the included files.
-     */
-    function parseConfig(json, sourceFile, host, basePath, configFileName, resolutionStack, errors, extendedConfigCache) {
-        basePath = ts.normalizeSlashes(basePath);
-        var resolvedPath = ts.getNormalizedAbsolutePath(configFileName || "", basePath);
-        if (resolutionStack.indexOf(resolvedPath) >= 0) {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Circularity_detected_while_resolving_configuration_Colon_0, __spreadArrays(resolutionStack, [resolvedPath]).join(" -> ")));
-            return { raw: json || convertToObject(sourceFile, errors) };
+        function isBindingIdentifierOrPrivateIdentifierOrPattern() {
+            return token() === 18 /* OpenBraceToken */
+                || token() === 22 /* OpenBracketToken */
+                || token() === 79 /* PrivateIdentifier */
+                || isBindingIdentifier();
         }
-        var ownConfig = json ?
-            parseOwnConfigOfJson(json, host, basePath, configFileName, errors) :
-            parseOwnConfigOfJsonSourceFile(sourceFile, host, basePath, configFileName, errors);
-        if (ownConfig.extendedConfigPath) {
-            // copy the resolution stack so it is never reused between branches in potential diamond-problem scenarios.
-            resolutionStack = resolutionStack.concat([resolvedPath]);
-            var extendedConfig = getExtendedConfig(sourceFile, ownConfig.extendedConfigPath, host, basePath, resolutionStack, errors, extendedConfigCache);
-            if (extendedConfig && isSuccessfulParsedTsconfig(extendedConfig)) {
-                var baseRaw_1 = extendedConfig.raw;
-                var raw_1 = ownConfig.raw;
-                var setPropertyInRawIfNotUndefined = function (propertyName) {
-                    var value = raw_1[propertyName] || baseRaw_1[propertyName];
-                    if (value) {
-                        raw_1[propertyName] = value;
-                    }
-                };
-                setPropertyInRawIfNotUndefined("include");
-                setPropertyInRawIfNotUndefined("exclude");
-                setPropertyInRawIfNotUndefined("files");
-                if (raw_1.compileOnSave === undefined) {
-                    raw_1.compileOnSave = baseRaw_1.compileOnSave;
-                }
-                ownConfig.options = ts.assign({}, extendedConfig.options, ownConfig.options);
-                ownConfig.watchOptions = ownConfig.watchOptions && extendedConfig.watchOptions ?
-                    ts.assign({}, extendedConfig.watchOptions, ownConfig.watchOptions) :
-                    ownConfig.watchOptions || extendedConfig.watchOptions;
-                // TODO extend type typeAcquisition
+        function parseIdentifierOrPattern(privateIdentifierDiagnosticMessage) {
+            if (token() === 22 /* OpenBracketToken */) {
+                return parseArrayBindingPattern();
             }
+            if (token() === 18 /* OpenBraceToken */) {
+                return parseObjectBindingPattern();
+            }
+            return parseBindingIdentifier(privateIdentifierDiagnosticMessage);
         }
-        return ownConfig;
-    }
-    function parseOwnConfigOfJson(json, host, basePath, configFileName, errors) {
-        if (ts.hasProperty(json, "excludes")) {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Unknown_option_excludes_Did_you_mean_exclude));
+        function parseVariableDeclarationAllowExclamation() {
+            return parseVariableDeclaration(/*allowExclamation*/ true);
         }
-        var options = convertCompilerOptionsFromJsonWorker(json.compilerOptions, basePath, errors, configFileName);
-        // typingOptions has been deprecated and is only supported for backward compatibility purposes.
-        // It should be removed in future releases - use typeAcquisition instead.
-        var typeAcquisition = convertTypeAcquisitionFromJsonWorker(json.typeAcquisition || json.typingOptions, basePath, errors, configFileName);
-        var watchOptions = convertWatchOptionsFromJsonWorker(json.watchOptions, basePath, errors);
-        json.compileOnSave = convertCompileOnSaveOptionFromJson(json, basePath, errors);
-        var extendedConfigPath;
-        if (json.extends) {
-            if (!ts.isString(json.extends)) {
-                errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "extends", "string"));
+        function parseVariableDeclaration(allowExclamation) {
+            var pos = getNodePos();
+            var name = parseIdentifierOrPattern(ts.Diagnostics.Private_identifiers_are_not_allowed_in_variable_declarations);
+            var exclamationToken;
+            if (allowExclamation && name.kind === 78 /* Identifier */ &&
+                token() === 53 /* ExclamationToken */ && !scanner.hasPrecedingLineBreak()) {
+                exclamationToken = parseTokenNode();
+            }
+            var type = parseTypeAnnotation();
+            var initializer = isInOrOfKeyword(token()) ? undefined : parseInitializer();
+            var node = factory.createVariableDeclaration(name, exclamationToken, type, initializer);
+            return finishNode(node, pos);
+        }
+        function parseVariableDeclarationList(inForStatementInitializer) {
+            var pos = getNodePos();
+            var flags = 0;
+            switch (token()) {
+                case 112 /* VarKeyword */:
+                    break;
+                case 118 /* LetKeyword */:
+                    flags |= 1 /* Let */;
+                    break;
+                case 84 /* ConstKeyword */:
+                    flags |= 2 /* Const */;
+                    break;
+                default:
+                    ts.Debug.fail();
+            }
+            nextToken();
+            // The user may have written the following:
+            //
+            //    for (let of X) { }
+            //
+            // In this case, we want to parse an empty declaration list, and then parse 'of'
+            // as a keyword. The reason this is not automatic is that 'of' is a valid identifier.
+            // So we need to look ahead to determine if 'of' should be treated as a keyword in
+            // this context.
+            // The checker will then give an error that there is an empty declaration list.
+            var declarations;
+            if (token() === 156 /* OfKeyword */ && lookAhead(canFollowContextualOfKeyword)) {
+                declarations = createMissingList();
             }
             else {
-                var newBase = configFileName ? directoryOfCombinedPath(configFileName, basePath) : basePath;
-                extendedConfigPath = getExtendsConfigPath(json.extends, host, newBase, errors, ts.createCompilerDiagnostic);
+                var savedDisallowIn = inDisallowInContext();
+                setDisallowInContext(inForStatementInitializer);
+                declarations = parseDelimitedList(8 /* VariableDeclarations */, inForStatementInitializer ? parseVariableDeclaration : parseVariableDeclarationAllowExclamation);
+                setDisallowInContext(savedDisallowIn);
             }
+            return finishNode(factory.createVariableDeclarationList(declarations, flags), pos);
         }
-        return { raw: json, options: options, watchOptions: watchOptions, typeAcquisition: typeAcquisition, extendedConfigPath: extendedConfigPath };
-    }
-    function parseOwnConfigOfJsonSourceFile(sourceFile, host, basePath, configFileName, errors) {
-        var options = getDefaultCompilerOptions(configFileName);
-        var typeAcquisition, typingOptionstypeAcquisition;
-        var watchOptions;
-        var extendedConfigPath;
-        var optionsIterator = {
-            onSetValidOptionKeyValueInParent: function (parentOption, option, value) {
-                var currentOption;
-                switch (parentOption) {
-                    case "compilerOptions":
-                        currentOption = options;
-                        break;
-                    case "watchOptions":
-                        currentOption = (watchOptions || (watchOptions = {}));
-                        break;
-                    case "typeAcquisition":
-                        currentOption = (typeAcquisition || (typeAcquisition = getDefaultTypeAcquisition(configFileName)));
-                        break;
-                    case "typingOptions":
-                        currentOption = (typingOptionstypeAcquisition || (typingOptionstypeAcquisition = getDefaultTypeAcquisition(configFileName)));
-                        break;
-                    default:
-                        ts.Debug.fail("Unknown option");
-                }
-                currentOption[option.name] = normalizeOptionValue(option, basePath, value);
-            },
-            onSetValidOptionKeyValueInRoot: function (key, _keyNode, value, valueNode) {
-                switch (key) {
-                    case "extends":
-                        var newBase = configFileName ? directoryOfCombinedPath(configFileName, basePath) : basePath;
-                        extendedConfigPath = getExtendsConfigPath(value, host, newBase, errors, function (message, arg0) {
-                            return ts.createDiagnosticForNodeInSourceFile(sourceFile, valueNode, message, arg0);
-                        });
-                        return;
+        function canFollowContextualOfKeyword() {
+            return nextTokenIsIdentifier() && nextToken() === 21 /* CloseParenToken */;
+        }
+        function parseVariableStatement(pos, hasJSDoc, decorators, modifiers) {
+            var declarationList = parseVariableDeclarationList(/*inForStatementInitializer*/ false);
+            parseSemicolon();
+            var node = factory.createVariableStatement(modifiers, declarationList);
+            // Decorators are not allowed on a variable statement, so we keep track of them to report them in the grammar checker.
+            node.decorators = decorators;
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parseFunctionDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            var savedAwaitContext = inAwaitContext();
+            var modifierFlags = ts.modifiersToFlags(modifiers);
+            parseExpected(97 /* FunctionKeyword */);
+            var asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
+            // We don't parse the name here in await context, instead we will report a grammar error in the checker.
+            var name = modifierFlags & 512 /* Default */ ? parseOptionalBindingIdentifier() : parseBindingIdentifier();
+            var isGenerator = asteriskToken ? 1 /* Yield */ : 0 /* None */;
+            var isAsync = modifierFlags & 256 /* Async */ ? 2 /* Await */ : 0 /* None */;
+            var typeParameters = parseTypeParameters();
+            if (modifierFlags & 1 /* Export */)
+                setAwaitContext(/*value*/ true);
+            var parameters = parseParameters(isGenerator | isAsync);
+            var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+            var body = parseFunctionBlockOrSemicolon(isGenerator | isAsync, ts.Diagnostics.or_expected);
+            setAwaitContext(savedAwaitContext);
+            var node = factory.createFunctionDeclaration(decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parseConstructorName() {
+            if (token() === 132 /* ConstructorKeyword */) {
+                return parseExpected(132 /* ConstructorKeyword */);
+            }
+            if (token() === 10 /* StringLiteral */ && lookAhead(nextToken) === 20 /* OpenParenToken */) {
+                return tryParse(function () {
+                    var literalNode = parseLiteralNode();
+                    return literalNode.text === "constructor" ? literalNode : undefined;
+                });
+            }
+        }
+        function tryParseConstructorDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            return tryParse(function () {
+                if (parseConstructorName()) {
+                    var typeParameters = parseTypeParameters();
+                    var parameters = parseParameters(0 /* None */);
+                    var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+                    var body = parseFunctionBlockOrSemicolon(0 /* None */, ts.Diagnostics.or_expected);
+                    var node = factory.createConstructorDeclaration(decorators, modifiers, parameters, body);
+                    // Attach `typeParameters` and `type` if they exist so that we can report them in the grammar checker.
+                    node.typeParameters = typeParameters;
+                    node.type = type;
+                    return withJSDoc(finishNode(node, pos), hasJSDoc);
                 }
-            },
-            onSetUnknownOptionKeyValueInRoot: function (key, keyNode, _value, _valueNode) {
-                if (key === "excludes") {
-                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, keyNode, ts.Diagnostics.Unknown_option_excludes_Did_you_mean_exclude));
+            });
+        }
+        function parseMethodDeclaration(pos, hasJSDoc, decorators, modifiers, asteriskToken, name, questionToken, exclamationToken, diagnosticMessage) {
+            var isGenerator = asteriskToken ? 1 /* Yield */ : 0 /* None */;
+            var isAsync = ts.some(modifiers, ts.isAsyncModifier) ? 2 /* Await */ : 0 /* None */;
+            var typeParameters = parseTypeParameters();
+            var parameters = parseParameters(isGenerator | isAsync);
+            var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+            var body = parseFunctionBlockOrSemicolon(isGenerator | isAsync, diagnosticMessage);
+            var node = factory.createMethodDeclaration(decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body);
+            // An exclamation token on a method is invalid syntax and will be handled by the grammar checker
+            node.exclamationToken = exclamationToken;
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parsePropertyDeclaration(pos, hasJSDoc, decorators, modifiers, name, questionToken) {
+            var exclamationToken = !questionToken && !scanner.hasPrecedingLineBreak() ? parseOptionalToken(53 /* ExclamationToken */) : undefined;
+            var type = parseTypeAnnotation();
+            var initializer = doOutsideOfContext(8192 /* YieldContext */ | 32768 /* AwaitContext */ | 4096 /* DisallowInContext */, parseInitializer);
+            parseSemicolon();
+            var node = factory.createPropertyDeclaration(decorators, modifiers, name, questionToken || exclamationToken, type, initializer);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parsePropertyOrMethodDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            var asteriskToken = parseOptionalToken(41 /* AsteriskToken */);
+            var name = parsePropertyName();
+            // Note: this is not legal as per the grammar.  But we allow it in the parser and
+            // report an error in the grammar checker.
+            var questionToken = parseOptionalToken(57 /* QuestionToken */);
+            if (asteriskToken || token() === 20 /* OpenParenToken */ || token() === 29 /* LessThanToken */) {
+                return parseMethodDeclaration(pos, hasJSDoc, decorators, modifiers, asteriskToken, name, questionToken, /*exclamationToken*/ undefined, ts.Diagnostics.or_expected);
+            }
+            return parsePropertyDeclaration(pos, hasJSDoc, decorators, modifiers, name, questionToken);
+        }
+        function parseAccessorDeclaration(pos, hasJSDoc, decorators, modifiers, kind) {
+            var name = parsePropertyName();
+            var typeParameters = parseTypeParameters();
+            var parameters = parseParameters(0 /* None */);
+            var type = parseReturnType(58 /* ColonToken */, /*isType*/ false);
+            var body = parseFunctionBlockOrSemicolon(0 /* None */);
+            var node = kind === 167 /* GetAccessor */
+                ? factory.createGetAccessorDeclaration(decorators, modifiers, name, parameters, type, body)
+                : factory.createSetAccessorDeclaration(decorators, modifiers, name, parameters, body);
+            // Keep track of `typeParameters` (for both) and `type` (for setters) if they were parsed those indicate grammar errors
+            node.typeParameters = typeParameters;
+            if (type && node.kind === 168 /* SetAccessor */)
+                node.type = type;
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function isClassMemberStart() {
+            var idToken;
+            if (token() === 59 /* AtToken */) {
+                return true;
+            }
+            // Eat up all modifiers, but hold on to the last one in case it is actually an identifier.
+            while (ts.isModifierKind(token())) {
+                idToken = token();
+                // If the idToken is a class modifier (protected, private, public, and static), it is
+                // certain that we are starting to parse class member. This allows better error recovery
+                // Example:
+                //      public foo() ...     // true
+                //      public @dec blah ... // true; we will then report an error later
+                //      export public ...    // true; we will then report an error later
+                if (ts.isClassMemberModifier(idToken)) {
+                    return true;
                 }
+                nextToken();
             }
-        };
-        var json = convertToObjectWorker(sourceFile, errors, /*returnValue*/ true, getTsconfigRootOptionsMap(), optionsIterator);
-        if (!typeAcquisition) {
-            if (typingOptionstypeAcquisition) {
-                typeAcquisition = (typingOptionstypeAcquisition.enableAutoDiscovery !== undefined) ?
-                    {
-                        enable: typingOptionstypeAcquisition.enableAutoDiscovery,
-                        include: typingOptionstypeAcquisition.include,
-                        exclude: typingOptionstypeAcquisition.exclude
-                    } :
-                    typingOptionstypeAcquisition;
+            if (token() === 41 /* AsteriskToken */) {
+                return true;
             }
-            else {
-                typeAcquisition = getDefaultTypeAcquisition(configFileName);
+            // Try to get the first property-like token following all modifiers.
+            // This can either be an identifier or the 'get' or 'set' keywords.
+            if (isLiteralPropertyName()) {
+                idToken = token();
+                nextToken();
             }
-        }
-        return { raw: json, options: options, watchOptions: watchOptions, typeAcquisition: typeAcquisition, extendedConfigPath: extendedConfigPath };
-    }
-    function getExtendsConfigPath(extendedConfig, host, basePath, errors, createDiagnostic) {
-        extendedConfig = ts.normalizeSlashes(extendedConfig);
-        if (ts.isRootedDiskPath(extendedConfig) || ts.startsWith(extendedConfig, "./") || ts.startsWith(extendedConfig, "../")) {
-            var extendedConfigPath = ts.getNormalizedAbsolutePath(extendedConfig, basePath);
-            if (!host.fileExists(extendedConfigPath) && !ts.endsWith(extendedConfigPath, ".json" /* Json */)) {
-                extendedConfigPath = extendedConfigPath + ".json";
-                if (!host.fileExists(extendedConfigPath)) {
-                    errors.push(createDiagnostic(ts.Diagnostics.File_0_not_found, extendedConfig));
-                    return undefined;
+            // Index signatures and computed properties are class members; we can parse.
+            if (token() === 22 /* OpenBracketToken */) {
+                return true;
+            }
+            // If we were able to get any potential identifier...
+            if (idToken !== undefined) {
+                // If we have a non-keyword identifier, or if we have an accessor, then it's safe to parse.
+                if (!ts.isKeyword(idToken) || idToken === 146 /* SetKeyword */ || idToken === 134 /* GetKeyword */) {
+                    return true;
+                }
+                // If it *is* a keyword, but not an accessor, check a little farther along
+                // to see if it should actually be parsed as a class member.
+                switch (token()) {
+                    case 20 /* OpenParenToken */: // Method declaration
+                    case 29 /* LessThanToken */: // Generic Method declaration
+                    case 53 /* ExclamationToken */: // Non-null assertion on property name
+                    case 58 /* ColonToken */: // Type Annotation for declaration
+                    case 62 /* EqualsToken */: // Initializer for declaration
+                    case 57 /* QuestionToken */: // Not valid, but permitted so that it gets caught later on.
+                        return true;
+                    default:
+                        // Covers
+                        //  - Semicolons     (declaration termination)
+                        //  - Closing braces (end-of-class, must be declaration)
+                        //  - End-of-files   (not valid, but permitted so that it gets caught later on)
+                        //  - Line-breaks    (enabling *automatic semicolon insertion*)
+                        return canParseSemicolon();
                 }
             }
-            return extendedConfigPath;
+            return false;
         }
-        // If the path isn't a rooted or relative path, resolve like a module
-        var resolved = ts.nodeModuleNameResolver(extendedConfig, ts.combinePaths(basePath, "tsconfig.json"), { moduleResolution: ts.ModuleResolutionKind.NodeJs }, host, /*cache*/ undefined, /*projectRefs*/ undefined, /*lookupConfig*/ true);
-        if (resolved.resolvedModule) {
-            return resolved.resolvedModule.resolvedFileName;
+        function parseDecoratorExpression() {
+            if (inAwaitContext() && token() === 130 /* AwaitKeyword */) {
+                // `@await` is is disallowed in an [Await] context, but can cause parsing to go off the rails
+                // This simply parses the missing identifier and moves on.
+                var pos = getNodePos();
+                var awaitExpression = parseIdentifier(ts.Diagnostics.Expression_expected);
+                nextToken();
+                var memberExpression = parseMemberExpressionRest(pos, awaitExpression, /*allowOptionalChain*/ true);
+                return parseCallExpressionRest(pos, memberExpression);
+            }
+            return parseLeftHandSideExpressionOrHigher();
         }
-        errors.push(createDiagnostic(ts.Diagnostics.File_0_not_found, extendedConfig));
-        return undefined;
-    }
-    function getExtendedConfig(sourceFile, extendedConfigPath, host, basePath, resolutionStack, errors, extendedConfigCache) {
-        var _a;
-        var path = host.useCaseSensitiveFileNames ? extendedConfigPath : ts.toFileNameLowerCase(extendedConfigPath);
-        var value;
-        var extendedResult;
-        var extendedConfig;
-        if (extendedConfigCache && (value = extendedConfigCache.get(path))) {
-            (extendedResult = value.extendedResult, extendedConfig = value.extendedConfig);
+        function tryParseDecorator() {
+            var pos = getNodePos();
+            if (!parseOptional(59 /* AtToken */)) {
+                return undefined;
+            }
+            var expression = doInDecoratorContext(parseDecoratorExpression);
+            return finishNode(factory.createDecorator(expression), pos);
         }
-        else {
-            extendedResult = readJsonConfigFile(extendedConfigPath, function (path) { return host.readFile(path); });
-            if (!extendedResult.parseDiagnostics.length) {
-                var extendedDirname = ts.getDirectoryPath(extendedConfigPath);
-                extendedConfig = parseConfig(/*json*/ undefined, extendedResult, host, extendedDirname, ts.getBaseFileName(extendedConfigPath), resolutionStack, errors, extendedConfigCache);
-                if (isSuccessfulParsedTsconfig(extendedConfig)) {
-                    // Update the paths to reflect base path
-                    var relativeDifference_1 = ts.convertToRelativePath(extendedDirname, basePath, ts.identity);
-                    var updatePath_1 = function (path) { return ts.isRootedDiskPath(path) ? path : ts.combinePaths(relativeDifference_1, path); };
-                    var mapPropertiesInRawIfNotUndefined = function (propertyName) {
-                        if (raw_2[propertyName]) {
-                            raw_2[propertyName] = ts.map(raw_2[propertyName], updatePath_1);
-                        }
-                    };
-                    var raw_2 = extendedConfig.raw;
-                    mapPropertiesInRawIfNotUndefined("include");
-                    mapPropertiesInRawIfNotUndefined("exclude");
-                    mapPropertiesInRawIfNotUndefined("files");
+        function parseDecorators() {
+            var pos = getNodePos();
+            var list, decorator;
+            while (decorator = tryParseDecorator()) {
+                list = ts.append(list, decorator);
+            }
+            return list && createNodeArray(list, pos);
+        }
+        function tryParseModifier(permitInvalidConstAsModifier) {
+            var pos = getNodePos();
+            var kind = token();
+            if (token() === 84 /* ConstKeyword */ && permitInvalidConstAsModifier) {
+                // We need to ensure that any subsequent modifiers appear on the same line
+                // so that when 'const' is a standalone declaration, we don't issue an error.
+                if (!tryParse(nextTokenIsOnSameLineAndCanFollowModifier)) {
+                    return undefined;
                 }
             }
-            if (extendedConfigCache) {
-                extendedConfigCache.set(path, { extendedResult: extendedResult, extendedConfig: extendedConfig });
+            else {
+                if (!parseAnyContextualModifier()) {
+                    return undefined;
+                }
             }
+            return finishNode(factory.createToken(kind), pos);
         }
-        if (sourceFile) {
-            sourceFile.extendedSourceFiles = [extendedResult.fileName];
-            if (extendedResult.extendedSourceFiles) {
-                (_a = sourceFile.extendedSourceFiles).push.apply(_a, extendedResult.extendedSourceFiles);
+        /*
+         * There are situations in which a modifier like 'const' will appear unexpectedly, such as on a class member.
+         * In those situations, if we are entirely sure that 'const' is not valid on its own (such as when ASI takes effect
+         * and turns it into a standalone declaration), then it is better to parse it and report an error later.
+         *
+         * In such situations, 'permitInvalidConstAsModifier' should be set to true.
+         */
+        function parseModifiers(permitInvalidConstAsModifier) {
+            var pos = getNodePos();
+            var list, modifier;
+            while (modifier = tryParseModifier(permitInvalidConstAsModifier)) {
+                list = ts.append(list, modifier);
             }
+            return list && createNodeArray(list, pos);
         }
-        if (extendedResult.parseDiagnostics.length) {
-            errors.push.apply(errors, extendedResult.parseDiagnostics);
-            return undefined;
+        function parseModifiersForArrowFunction() {
+            var modifiers;
+            if (token() === 129 /* AsyncKeyword */) {
+                var pos = getNodePos();
+                nextToken();
+                var modifier = finishNode(factory.createToken(129 /* AsyncKeyword */), pos);
+                modifiers = createNodeArray([modifier], pos);
+            }
+            return modifiers;
         }
-        return extendedConfig;
-    }
-    function convertCompileOnSaveOptionFromJson(jsonOption, basePath, errors) {
-        if (!ts.hasProperty(jsonOption, ts.compileOnSaveCommandLineOption.name)) {
-            return false;
+        function parseClassElement() {
+            var pos = getNodePos();
+            if (token() === 26 /* SemicolonToken */) {
+                nextToken();
+                return finishNode(factory.createSemicolonClassElement(), pos);
+            }
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var decorators = parseDecorators();
+            var modifiers = parseModifiers(/*permitInvalidConstAsModifier*/ true);
+            if (parseContextualModifier(134 /* GetKeyword */)) {
+                return parseAccessorDeclaration(pos, hasJSDoc, decorators, modifiers, 167 /* GetAccessor */);
+            }
+            if (parseContextualModifier(146 /* SetKeyword */)) {
+                return parseAccessorDeclaration(pos, hasJSDoc, decorators, modifiers, 168 /* SetAccessor */);
+            }
+            if (token() === 132 /* ConstructorKeyword */ || token() === 10 /* StringLiteral */) {
+                var constructorDeclaration = tryParseConstructorDeclaration(pos, hasJSDoc, decorators, modifiers);
+                if (constructorDeclaration) {
+                    return constructorDeclaration;
+                }
+            }
+            if (isIndexSignature()) {
+                return parseIndexSignatureDeclaration(pos, hasJSDoc, decorators, modifiers);
+            }
+            // It is very important that we check this *after* checking indexers because
+            // the [ token can start an index signature or a computed property name
+            if (ts.tokenIsIdentifierOrKeyword(token()) ||
+                token() === 10 /* StringLiteral */ ||
+                token() === 8 /* NumericLiteral */ ||
+                token() === 41 /* AsteriskToken */ ||
+                token() === 22 /* OpenBracketToken */) {
+                var isAmbient = ts.some(modifiers, isDeclareModifier);
+                if (isAmbient) {
+                    for (var _i = 0, _a = modifiers; _i < _a.length; _i++) {
+                        var m = _a[_i];
+                        m.flags |= 8388608 /* Ambient */;
+                    }
+                    return doInsideOfContext(8388608 /* Ambient */, function () { return parsePropertyOrMethodDeclaration(pos, hasJSDoc, decorators, modifiers); });
+                }
+                else {
+                    return parsePropertyOrMethodDeclaration(pos, hasJSDoc, decorators, modifiers);
+                }
+            }
+            if (decorators || modifiers) {
+                // treat this as a property declaration with a missing name.
+                var name = createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ true, ts.Diagnostics.Declaration_expected);
+                return parsePropertyDeclaration(pos, hasJSDoc, decorators, modifiers, name, /*questionToken*/ undefined);
+            }
+            // 'isClassMemberStart' should have hinted not to attempt parsing.
+            return ts.Debug.fail("Should not have attempted to parse class member declaration.");
         }
-        var result = convertJsonOption(ts.compileOnSaveCommandLineOption, jsonOption.compileOnSave, basePath, errors);
-        return typeof result === "boolean" && result;
-    }
-    function convertCompilerOptionsFromJson(jsonOptions, basePath, configFileName) {
-        var errors = [];
-        var options = convertCompilerOptionsFromJsonWorker(jsonOptions, basePath, errors, configFileName);
-        return { options: options, errors: errors };
-    }
-    ts.convertCompilerOptionsFromJson = convertCompilerOptionsFromJson;
-    function convertTypeAcquisitionFromJson(jsonOptions, basePath, configFileName) {
-        var errors = [];
-        var options = convertTypeAcquisitionFromJsonWorker(jsonOptions, basePath, errors, configFileName);
-        return { options: options, errors: errors };
-    }
-    ts.convertTypeAcquisitionFromJson = convertTypeAcquisitionFromJson;
-    function getDefaultCompilerOptions(configFileName) {
-        var options = configFileName && ts.getBaseFileName(configFileName) === "jsconfig.json"
-            ? { allowJs: true, maxNodeModuleJsDepth: 2, allowSyntheticDefaultImports: true, skipLibCheck: true, noEmit: true }
-            : {};
-        return options;
-    }
-    function convertCompilerOptionsFromJsonWorker(jsonOptions, basePath, errors, configFileName) {
-        var options = getDefaultCompilerOptions(configFileName);
-        convertOptionsFromJson(getCommandLineCompilerOptionsMap(), jsonOptions, basePath, options, ts.compilerOptionsDidYouMeanDiagnostics, errors);
-        if (configFileName) {
-            options.configFilePath = ts.normalizeSlashes(configFileName);
+        function parseClassExpression() {
+            return parseClassDeclarationOrExpression(getNodePos(), hasPrecedingJSDocComment(), /*decorators*/ undefined, /*modifiers*/ undefined, 221 /* ClassExpression */);
         }
-        return options;
-    }
-    function getDefaultTypeAcquisition(configFileName) {
-        return { enable: !!configFileName && ts.getBaseFileName(configFileName) === "jsconfig.json", include: [], exclude: [] };
-    }
-    function convertTypeAcquisitionFromJsonWorker(jsonOptions, basePath, errors, configFileName) {
-        var options = getDefaultTypeAcquisition(configFileName);
-        var typeAcquisition = convertEnableAutoDiscoveryToEnable(jsonOptions);
-        convertOptionsFromJson(getCommandLineTypeAcquisitionMap(), typeAcquisition, basePath, options, typeAcquisitionDidYouMeanDiagnostics, errors);
-        return options;
-    }
-    function convertWatchOptionsFromJsonWorker(jsonOptions, basePath, errors) {
-        return convertOptionsFromJson(getCommandLineWatchOptionsMap(), jsonOptions, basePath, /*defaultOptions*/ undefined, watchOptionsDidYouMeanDiagnostics, errors);
-    }
-    function convertOptionsFromJson(optionsNameMap, jsonOptions, basePath, defaultOptions, diagnostics, errors) {
-        if (!jsonOptions) {
-            return;
+        function parseClassDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            return parseClassDeclarationOrExpression(pos, hasJSDoc, decorators, modifiers, 252 /* ClassDeclaration */);
         }
-        for (var id in jsonOptions) {
-            var opt = optionsNameMap.get(id);
-            if (opt) {
-                (defaultOptions || (defaultOptions = {}))[opt.name] = convertJsonOption(opt, jsonOptions[id], basePath, errors);
+        function parseClassDeclarationOrExpression(pos, hasJSDoc, decorators, modifiers, kind) {
+            var savedAwaitContext = inAwaitContext();
+            parseExpected(83 /* ClassKeyword */);
+            // We don't parse the name here in await context, instead we will report a grammar error in the checker.
+            var name = parseNameOfClassDeclarationOrExpression();
+            var typeParameters = parseTypeParameters();
+            if (ts.some(modifiers, ts.isExportModifier))
+                setAwaitContext(/*value*/ true);
+            var heritageClauses = parseHeritageClauses();
+            var members;
+            if (parseExpected(18 /* OpenBraceToken */)) {
+                // ClassTail[Yield,Await] : (Modified) See 14.5
+                //      ClassHeritage[?Yield,?Await]opt { ClassBody[?Yield,?Await]opt }
+                members = parseClassMembers();
+                parseExpected(19 /* CloseBraceToken */);
             }
             else {
-                errors.push(createUnknownOptionError(id, diagnostics, ts.createCompilerDiagnostic));
+                members = createMissingList();
             }
+            setAwaitContext(savedAwaitContext);
+            var node = kind === 252 /* ClassDeclaration */
+                ? factory.createClassDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members)
+                : factory.createClassExpression(decorators, modifiers, name, typeParameters, heritageClauses, members);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        return defaultOptions;
-    }
-    function convertJsonOption(opt, value, basePath, errors) {
-        if (isCompilerOptionsValue(opt, value)) {
-            var optType = opt.type;
-            if (optType === "list" && ts.isArray(value)) {
-                return convertJsonOptionOfListType(opt, value, basePath, errors);
-            }
-            else if (!ts.isString(optType)) {
-                return convertJsonOptionOfCustomType(opt, value, errors);
-            }
-            return normalizeNonListOptionValue(opt, basePath, value);
+        function parseNameOfClassDeclarationOrExpression() {
+            // implements is a future reserved word so
+            // 'class implements' might mean either
+            // - class expression with omitted name, 'implements' starts heritage clause
+            // - class with name 'implements'
+            // 'isImplementsClause' helps to disambiguate between these two cases
+            return isBindingIdentifier() && !isImplementsClause()
+                ? createIdentifier(isBindingIdentifier())
+                : undefined;
         }
-        else {
-            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, opt.name, getCompilerOptionValueTypeString(opt)));
+        function isImplementsClause() {
+            return token() === 116 /* ImplementsKeyword */ && lookAhead(nextTokenIsIdentifierOrKeyword);
         }
-    }
-    function normalizeOptionValue(option, basePath, value) {
-        if (isNullOrUndefined(value))
-            return undefined;
-        if (option.type === "list") {
-            var listOption_1 = option;
-            if (listOption_1.element.isFilePath || !ts.isString(listOption_1.element.type)) {
-                return ts.filter(ts.map(value, function (v) { return normalizeOptionValue(listOption_1.element, basePath, v); }), function (v) { return !!v; });
+        function parseHeritageClauses() {
+            // ClassTail[Yield,Await] : (Modified) See 14.5
+            //      ClassHeritage[?Yield,?Await]opt { ClassBody[?Yield,?Await]opt }
+            if (isHeritageClause()) {
+                return parseList(22 /* HeritageClauses */, parseHeritageClause);
             }
-            return value;
+            return undefined;
         }
-        else if (!ts.isString(option.type)) {
-            return option.type.get(ts.isString(value) ? value.toLowerCase() : value);
+        function parseHeritageClause() {
+            var pos = getNodePos();
+            var tok = token();
+            ts.Debug.assert(tok === 93 /* ExtendsKeyword */ || tok === 116 /* ImplementsKeyword */); // isListElement() should ensure this.
+            nextToken();
+            var types = parseDelimitedList(7 /* HeritageClauseElement */, parseExpressionWithTypeArguments);
+            return finishNode(factory.createHeritageClause(tok, types), pos);
         }
-        return normalizeNonListOptionValue(option, basePath, value);
-    }
-    function normalizeNonListOptionValue(option, basePath, value) {
-        if (option.isFilePath) {
-            value = ts.getNormalizedAbsolutePath(value, basePath);
-            if (value === "") {
-                value = ".";
-            }
+        function parseExpressionWithTypeArguments() {
+            var pos = getNodePos();
+            var expression = parseLeftHandSideExpressionOrHigher();
+            var typeArguments = tryParseTypeArguments();
+            return finishNode(factory.createExpressionWithTypeArguments(expression, typeArguments), pos);
         }
-        return value;
-    }
-    function convertJsonOptionOfCustomType(opt, value, errors) {
-        if (isNullOrUndefined(value))
-            return undefined;
-        var key = value.toLowerCase();
-        var val = opt.type.get(key);
-        if (val !== undefined) {
-            return val;
+        function tryParseTypeArguments() {
+            return token() === 29 /* LessThanToken */ ?
+                parseBracketedList(20 /* TypeArguments */, parseType, 29 /* LessThanToken */, 31 /* GreaterThanToken */) : undefined;
         }
-        else {
-            errors.push(createCompilerDiagnosticForInvalidCustomType(opt));
+        function isHeritageClause() {
+            return token() === 93 /* ExtendsKeyword */ || token() === 116 /* ImplementsKeyword */;
         }
-    }
-    function convertJsonOptionOfListType(option, values, basePath, errors) {
-        return ts.filter(ts.map(values, function (v) { return convertJsonOption(option.element, v, basePath, errors); }), function (v) { return !!v; });
-    }
-    function trimString(s) {
-        return typeof s.trim === "function" ? s.trim() : s.replace(/^[\s]+|[\s]+$/g, "");
-    }
-    /**
-     * Tests for a path that ends in a recursive directory wildcard.
-     * Matches **, \**, **\, and \**\, but not a**b.
-     *
-     * NOTE: used \ in place of / above to avoid issues with multiline comments.
-     *
-     * Breakdown:
-     *  (^|\/)      # matches either the beginning of the string or a directory separator.
-     *  \*\*        # matches the recursive directory wildcard "**".
-     *  \/?$        # matches an optional trailing directory separator at the end of the string.
-     */
-    var invalidTrailingRecursionPattern = /(^|\/)\*\*\/?$/;
-    /**
-     * Tests for a path where .. appears after a recursive directory wildcard.
-     * Matches **\..\*, **\a\..\*, and **\.., but not ..\**\*
-     *
-     * NOTE: used \ in place of / above to avoid issues with multiline comments.
-     *
-     * Breakdown:
-     *  (^|\/)      # matches either the beginning of the string or a directory separator.
-     *  \*\*\/      # matches a recursive directory wildcard "**" followed by a directory separator.
-     *  (.*\/)?     # optionally matches any number of characters followed by a directory separator.
-     *  \.\.        # matches a parent directory path component ".."
-     *  ($|\/)      # matches either the end of the string or a directory separator.
-     */
-    var invalidDotDotAfterRecursiveWildcardPattern = /(^|\/)\*\*\/(.*\/)?\.\.($|\/)/;
-    /**
-     * Tests for a path containing a wildcard character in a directory component of the path.
-     * Matches \*\, \?\, and \a*b\, but not \a\ or \a\*.
-     *
-     * NOTE: used \ in place of / above to avoid issues with multiline comments.
-     *
-     * Breakdown:
-     *  \/          # matches a directory separator.
-     *  [^/]*?      # matches any number of characters excluding directory separators (non-greedy).
-     *  [*?]        # matches either a wildcard character (* or ?)
-     *  [^/]*       # matches any number of characters excluding directory separators (greedy).
-     *  \/          # matches a directory separator.
-     */
-    var watchRecursivePattern = /\/[^/]*?[*?][^/]*\//;
-    /**
-     * Matches the portion of a wildcard path that does not contain wildcards.
-     * Matches \a of \a\*, or \a\b\c of \a\b\c\?\d.
-     *
-     * NOTE: used \ in place of / above to avoid issues with multiline comments.
-     *
-     * Breakdown:
-     *  ^                   # matches the beginning of the string
-     *  [^*?]*              # matches any number of non-wildcard characters
-     *  (?=\/[^/]*[*?])     # lookahead that matches a directory separator followed by
-     *                      # a path component that contains at least one wildcard character (* or ?).
-     */
-    var wildcardDirectoryPattern = /^[^*?]*(?=\/[^/]*[*?])/;
-    /**
-     * Expands an array of file specifications.
-     *
-     * @param filesSpecs The literal file names to include.
-     * @param includeSpecs The wildcard file specifications to include.
-     * @param excludeSpecs The wildcard file specifications to exclude.
-     * @param basePath The base path for any relative file specifications.
-     * @param options Compiler options.
-     * @param host The host used to resolve files and directories.
-     * @param errors An array for diagnostic reporting.
-     */
-    function matchFileNames(filesSpecs, includeSpecs, excludeSpecs, basePath, options, host, errors, extraFileExtensions, jsonSourceFile) {
-        basePath = ts.normalizePath(basePath);
-        var validatedIncludeSpecs, validatedExcludeSpecs;
-        // The exclude spec list is converted into a regular expression, which allows us to quickly
-        // test whether a file or directory should be excluded before recursively traversing the
-        // file system.
-        if (includeSpecs) {
-            validatedIncludeSpecs = validateSpecs(includeSpecs, errors, /*allowTrailingRecursion*/ false, jsonSourceFile, "include");
+        function parseClassMembers() {
+            return parseList(5 /* ClassMembers */, parseClassElement);
         }
-        if (excludeSpecs) {
-            validatedExcludeSpecs = validateSpecs(excludeSpecs, errors, /*allowTrailingRecursion*/ true, jsonSourceFile, "exclude");
+        function parseInterfaceDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            parseExpected(117 /* InterfaceKeyword */);
+            var name = parseIdentifier();
+            var typeParameters = parseTypeParameters();
+            var heritageClauses = parseHeritageClauses();
+            var members = parseObjectTypeMembers();
+            var node = factory.createInterfaceDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parseTypeAliasDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            parseExpected(149 /* TypeKeyword */);
+            var name = parseIdentifier();
+            var typeParameters = parseTypeParameters();
+            parseExpected(62 /* EqualsToken */);
+            var type = token() === 136 /* IntrinsicKeyword */ && tryParse(parseKeywordAndNoDot) || parseType();
+            parseSemicolon();
+            var node = factory.createTypeAliasDeclaration(decorators, modifiers, name, typeParameters, type);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        // Wildcard directories (provided as part of a wildcard path) are stored in a
-        // file map that marks whether it was a regular wildcard match (with a `*` or `?` token),
-        // or a recursive directory. This information is used by filesystem watchers to monitor for
-        // new entries in these paths.
-        var wildcardDirectories = getWildcardDirectories(validatedIncludeSpecs, validatedExcludeSpecs, basePath, host.useCaseSensitiveFileNames);
-        var spec = { filesSpecs: filesSpecs, includeSpecs: includeSpecs, excludeSpecs: excludeSpecs, validatedIncludeSpecs: validatedIncludeSpecs, validatedExcludeSpecs: validatedExcludeSpecs, wildcardDirectories: wildcardDirectories };
-        return getFileNamesFromConfigSpecs(spec, basePath, options, host, extraFileExtensions);
-    }
-    /**
-     * Gets the file names from the provided config file specs that contain, files, include, exclude and
-     * other properties needed to resolve the file names
-     * @param spec The config file specs extracted with file names to include, wildcards to include/exclude and other details
-     * @param basePath The base path for any relative file specifications.
-     * @param options Compiler options.
-     * @param host The host used to resolve files and directories.
-     * @param extraFileExtensions optionaly file extra file extension information from host
-     */
-    /* @internal */
-    function getFileNamesFromConfigSpecs(spec, basePath, options, host, extraFileExtensions) {
-        if (extraFileExtensions === void 0) { extraFileExtensions = []; }
-        basePath = ts.normalizePath(basePath);
-        var keyMapper = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames);
-        // Literal file names (provided via the "files" array in tsconfig.json) are stored in a
-        // file map with a possibly case insensitive key. We use this map later when when including
-        // wildcard paths.
-        var literalFileMap = ts.createMap();
-        // Wildcard paths (provided via the "includes" array in tsconfig.json) are stored in a
-        // file map with a possibly case insensitive key. We use this map to store paths matched
-        // via wildcard, and to handle extension priority.
-        var wildcardFileMap = ts.createMap();
-        // Wildcard paths of json files (provided via the "includes" array in tsconfig.json) are stored in a
-        // file map with a possibly case insensitive key. We use this map to store paths matched
-        // via wildcard of *.json kind
-        var wildCardJsonFileMap = ts.createMap();
-        var filesSpecs = spec.filesSpecs, validatedIncludeSpecs = spec.validatedIncludeSpecs, validatedExcludeSpecs = spec.validatedExcludeSpecs, wildcardDirectories = spec.wildcardDirectories;
-        // Rather than requery this for each file and filespec, we query the supported extensions
-        // once and store it on the expansion context.
-        var supportedExtensions = ts.getSupportedExtensions(options, extraFileExtensions);
-        var supportedExtensionsWithJsonIfResolveJsonModule = ts.getSuppoertedExtensionsWithJsonIfResolveJsonModule(options, supportedExtensions);
-        // Literal files are always included verbatim. An "include" or "exclude" specification cannot
-        // remove a literal file.
-        if (filesSpecs) {
-            for (var _i = 0, filesSpecs_1 = filesSpecs; _i < filesSpecs_1.length; _i++) {
-                var fileName = filesSpecs_1[_i];
-                var file = ts.getNormalizedAbsolutePath(fileName, basePath);
-                literalFileMap.set(keyMapper(file), file);
+        // In an ambient declaration, the grammar only allows integer literals as initializers.
+        // In a non-ambient declaration, the grammar allows uninitialized members only in a
+        // ConstantEnumMemberSection, which starts at the beginning of an enum declaration
+        // or any time an integer literal initializer is encountered.
+        function parseEnumMember() {
+            var pos = getNodePos();
+            var hasJSDoc = hasPrecedingJSDocComment();
+            var name = parsePropertyName();
+            var initializer = allowInAnd(parseInitializer);
+            return withJSDoc(finishNode(factory.createEnumMember(name, initializer), pos), hasJSDoc);
+        }
+        function parseEnumDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            parseExpected(91 /* EnumKeyword */);
+            var name = parseIdentifier();
+            var members;
+            if (parseExpected(18 /* OpenBraceToken */)) {
+                members = doOutsideOfYieldAndAwaitContext(function () { return parseDelimitedList(6 /* EnumMembers */, parseEnumMember); });
+                parseExpected(19 /* CloseBraceToken */);
             }
-        }
-        var jsonOnlyIncludeRegexes;
-        if (validatedIncludeSpecs && validatedIncludeSpecs.length > 0) {
-            var _loop_5 = function (file) {
-                if (ts.fileExtensionIs(file, ".json" /* Json */)) {
-                    // Valid only if *.json specified
-                    if (!jsonOnlyIncludeRegexes) {
-                        var includes = validatedIncludeSpecs.filter(function (s) { return ts.endsWith(s, ".json" /* Json */); });
-                        var includeFilePatterns = ts.map(ts.getRegularExpressionsForWildcards(includes, basePath, "files"), function (pattern) { return "^" + pattern + "$"; });
-                        jsonOnlyIncludeRegexes = includeFilePatterns ? includeFilePatterns.map(function (pattern) { return ts.getRegexFromPattern(pattern, host.useCaseSensitiveFileNames); }) : ts.emptyArray;
-                    }
-                    var includeIndex = ts.findIndex(jsonOnlyIncludeRegexes, function (re) { return re.test(file); });
-                    if (includeIndex !== -1) {
-                        var key_1 = keyMapper(file);
-                        if (!literalFileMap.has(key_1) && !wildCardJsonFileMap.has(key_1)) {
-                            wildCardJsonFileMap.set(key_1, file);
-                        }
-                    }
-                    return "continue";
-                }
-                // If we have already included a literal or wildcard path with a
-                // higher priority extension, we should skip this file.
-                //
-                // This handles cases where we may encounter both <file>.ts and
-                // <file>.d.ts (or <file>.js if "allowJs" is enabled) in the same
-                // directory when they are compilation outputs.
-                if (hasFileWithHigherPriorityExtension(file, literalFileMap, wildcardFileMap, supportedExtensions, keyMapper)) {
-                    return "continue";
-                }
-                // We may have included a wildcard path with a lower priority
-                // extension due to the user-defined order of entries in the
-                // "include" array. If there is a lower priority extension in the
-                // same directory, we should remove it.
-                removeWildcardFilesWithLowerPriorityExtension(file, wildcardFileMap, supportedExtensions, keyMapper);
-                var key = keyMapper(file);
-                if (!literalFileMap.has(key) && !wildcardFileMap.has(key)) {
-                    wildcardFileMap.set(key, file);
-                }
-            };
-            for (var _a = 0, _b = host.readDirectory(basePath, supportedExtensionsWithJsonIfResolveJsonModule, validatedExcludeSpecs, validatedIncludeSpecs, /*depth*/ undefined); _a < _b.length; _a++) {
-                var file = _b[_a];
-                _loop_5(file);
+            else {
+                members = createMissingList();
             }
+            var node = factory.createEnumDeclaration(decorators, modifiers, name, members);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        var literalFiles = ts.arrayFrom(literalFileMap.values());
-        var wildcardFiles = ts.arrayFrom(wildcardFileMap.values());
-        return {
-            fileNames: literalFiles.concat(wildcardFiles, ts.arrayFrom(wildCardJsonFileMap.values())),
-            wildcardDirectories: wildcardDirectories,
-            spec: spec
-        };
-    }
-    ts.getFileNamesFromConfigSpecs = getFileNamesFromConfigSpecs;
-    function validateSpecs(specs, errors, allowTrailingRecursion, jsonSourceFile, specKey) {
-        return specs.filter(function (spec) {
-            var diag = specToDiagnostic(spec, allowTrailingRecursion);
-            if (diag !== undefined) {
-                errors.push(createDiagnostic(diag, spec));
+        function parseModuleBlock() {
+            var pos = getNodePos();
+            var statements;
+            if (parseExpected(18 /* OpenBraceToken */)) {
+                statements = parseList(1 /* BlockStatements */, parseStatement);
+                parseExpected(19 /* CloseBraceToken */);
             }
-            return diag === undefined;
-        });
-        function createDiagnostic(message, spec) {
-            var element = ts.getTsConfigPropArrayElementValue(jsonSourceFile, specKey, spec);
-            return element ?
-                ts.createDiagnosticForNodeInSourceFile(jsonSourceFile, element, message, spec) :
-                ts.createCompilerDiagnostic(message, spec);
+            else {
+                statements = createMissingList();
+            }
+            return finishNode(factory.createModuleBlock(statements), pos);
         }
-    }
-    function specToDiagnostic(spec, allowTrailingRecursion) {
-        if (!allowTrailingRecursion && invalidTrailingRecursionPattern.test(spec)) {
-            return ts.Diagnostics.File_specification_cannot_end_in_a_recursive_directory_wildcard_Asterisk_Asterisk_Colon_0;
+        function parseModuleOrNamespaceDeclaration(pos, hasJSDoc, decorators, modifiers, flags) {
+            // If we are parsing a dotted namespace name, we want to
+            // propagate the 'Namespace' flag across the names if set.
+            var namespaceFlag = flags & 16 /* Namespace */;
+            var name = parseIdentifier();
+            var body = parseOptional(24 /* DotToken */)
+                ? parseModuleOrNamespaceDeclaration(getNodePos(), /*hasJSDoc*/ false, /*decorators*/ undefined, /*modifiers*/ undefined, 4 /* NestedNamespace */ | namespaceFlag)
+                : parseModuleBlock();
+            var node = factory.createModuleDeclaration(decorators, modifiers, name, body, flags);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        else if (invalidDotDotAfterRecursiveWildcardPattern.test(spec)) {
-            return ts.Diagnostics.File_specification_cannot_contain_a_parent_directory_that_appears_after_a_recursive_directory_wildcard_Asterisk_Asterisk_Colon_0;
+        function parseAmbientExternalModuleDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            var flags = 0;
+            var name;
+            if (token() === 154 /* GlobalKeyword */) {
+                // parse 'global' as name of global scope augmentation
+                name = parseIdentifier();
+                flags |= 1024 /* GlobalAugmentation */;
+            }
+            else {
+                name = parseLiteralNode();
+                name.text = internIdentifier(name.text);
+            }
+            var body;
+            if (token() === 18 /* OpenBraceToken */) {
+                body = parseModuleBlock();
+            }
+            else {
+                parseSemicolon();
+            }
+            var node = factory.createModuleDeclaration(decorators, modifiers, name, body, flags);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-    }
-    /**
-     * Gets directories in a set of include patterns that should be watched for changes.
-     */
-    function getWildcardDirectories(include, exclude, path, useCaseSensitiveFileNames) {
-        // We watch a directory recursively if it contains a wildcard anywhere in a directory segment
-        // of the pattern:
-        //
-        //  /a/b/**/d   - Watch /a/b recursively to catch changes to any d in any subfolder recursively
-        //  /a/b/*/d    - Watch /a/b recursively to catch any d in any immediate subfolder, even if a new subfolder is added
-        //  /a/b        - Watch /a/b recursively to catch changes to anything in any recursive subfoler
-        //
-        // We watch a directory without recursion if it contains a wildcard in the file segment of
-        // the pattern:
-        //
-        //  /a/b/*      - Watch /a/b directly to catch any new file
-        //  /a/b/a?z    - Watch /a/b directly to catch any new file matching a?z
-        var rawExcludeRegex = ts.getRegularExpressionForWildcard(exclude, path, "exclude");
-        var excludeRegex = rawExcludeRegex && new RegExp(rawExcludeRegex, useCaseSensitiveFileNames ? "" : "i");
-        var wildcardDirectories = {};
-        if (include !== undefined) {
-            var recursiveKeys = [];
-            for (var _i = 0, include_1 = include; _i < include_1.length; _i++) {
-                var file = include_1[_i];
-                var spec = ts.normalizePath(ts.combinePaths(path, file));
-                if (excludeRegex && excludeRegex.test(spec)) {
-                    continue;
-                }
-                var match = getWildcardDirectoryFromSpec(spec, useCaseSensitiveFileNames);
-                if (match) {
-                    var key = match.key, flags = match.flags;
-                    var existingFlags = wildcardDirectories[key];
-                    if (existingFlags === undefined || existingFlags < flags) {
-                        wildcardDirectories[key] = flags;
-                        if (flags === 1 /* Recursive */) {
-                            recursiveKeys.push(key);
-                        }
-                    }
-                }
+        function parseModuleDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            var flags = 0;
+            if (token() === 154 /* GlobalKeyword */) {
+                // global augmentation
+                return parseAmbientExternalModuleDeclaration(pos, hasJSDoc, decorators, modifiers);
             }
-            // Remove any subpaths under an existing recursively watched directory.
-            for (var key in wildcardDirectories) {
-                if (ts.hasProperty(wildcardDirectories, key)) {
-                    for (var _a = 0, recursiveKeys_1 = recursiveKeys; _a < recursiveKeys_1.length; _a++) {
-                        var recursiveKey = recursiveKeys_1[_a];
-                        if (key !== recursiveKey && ts.containsPath(recursiveKey, key, path, !useCaseSensitiveFileNames)) {
-                            delete wildcardDirectories[key];
-                        }
-                    }
+            else if (parseOptional(140 /* NamespaceKeyword */)) {
+                flags |= 16 /* Namespace */;
+            }
+            else {
+                parseExpected(139 /* ModuleKeyword */);
+                if (token() === 10 /* StringLiteral */) {
+                    return parseAmbientExternalModuleDeclaration(pos, hasJSDoc, decorators, modifiers);
                 }
             }
+            return parseModuleOrNamespaceDeclaration(pos, hasJSDoc, decorators, modifiers, flags);
         }
-        return wildcardDirectories;
-    }
-    function getWildcardDirectoryFromSpec(spec, useCaseSensitiveFileNames) {
-        var match = wildcardDirectoryPattern.exec(spec);
-        if (match) {
-            return {
-                key: useCaseSensitiveFileNames ? match[0] : ts.toFileNameLowerCase(match[0]),
-                flags: watchRecursivePattern.test(spec) ? 1 /* Recursive */ : 0 /* None */
-            };
-        }
-        if (ts.isImplicitGlob(spec)) {
-            return { key: spec, flags: 1 /* Recursive */ };
+        function isExternalModuleReference() {
+            return token() === 143 /* RequireKeyword */ &&
+                lookAhead(nextTokenIsOpenParen);
         }
-        return undefined;
-    }
-    /**
-     * Determines whether a literal or wildcard file has already been included that has a higher
-     * extension priority.
-     *
-     * @param file The path to the file.
-     * @param extensionPriority The priority of the extension.
-     * @param context The expansion context.
-     */
-    function hasFileWithHigherPriorityExtension(file, literalFiles, wildcardFiles, extensions, keyMapper) {
-        var extensionPriority = ts.getExtensionPriority(file, extensions);
-        var adjustedExtensionPriority = ts.adjustExtensionPriority(extensionPriority, extensions);
-        for (var i = 0 /* Highest */; i < adjustedExtensionPriority; i++) {
-            var higherPriorityExtension = extensions[i];
-            var higherPriorityPath = keyMapper(ts.changeExtension(file, higherPriorityExtension));
-            if (literalFiles.has(higherPriorityPath) || wildcardFiles.has(higherPriorityPath)) {
-                return true;
-            }
+        function nextTokenIsOpenParen() {
+            return nextToken() === 20 /* OpenParenToken */;
         }
-        return false;
-    }
-    /**
-     * Removes files included via wildcard expansion with a lower extension priority that have
-     * already been included.
-     *
-     * @param file The path to the file.
-     * @param extensionPriority The priority of the extension.
-     * @param context The expansion context.
-     */
-    function removeWildcardFilesWithLowerPriorityExtension(file, wildcardFiles, extensions, keyMapper) {
-        var extensionPriority = ts.getExtensionPriority(file, extensions);
-        var nextExtensionPriority = ts.getNextLowestExtensionPriority(extensionPriority, extensions);
-        for (var i = nextExtensionPriority; i < extensions.length; i++) {
-            var lowerPriorityExtension = extensions[i];
-            var lowerPriorityPath = keyMapper(ts.changeExtension(file, lowerPriorityExtension));
-            wildcardFiles.delete(lowerPriorityPath);
+        function nextTokenIsSlash() {
+            return nextToken() === 43 /* SlashToken */;
         }
-    }
-    /**
-     * Produces a cleaned version of compiler options with personally identifying info (aka, paths) removed.
-     * Also converts enum values back to strings.
-     */
-    /* @internal */
-    function convertCompilerOptionsForTelemetry(opts) {
-        var out = {};
-        for (var key in opts) {
-            if (opts.hasOwnProperty(key)) {
-                var type = getOptionFromName(key);
-                if (type !== undefined) { // Ignore unknown options
-                    out[key] = getOptionValueWithEmptyStrings(opts[key], type);
-                }
+        function parseNamespaceExportDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            parseExpected(126 /* AsKeyword */);
+            parseExpected(140 /* NamespaceKeyword */);
+            var name = parseIdentifier();
+            parseSemicolon();
+            var node = factory.createNamespaceExportDeclaration(name);
+            // NamespaceExportDeclaration nodes cannot have decorators or modifiers, so we attach them here so we can report them in the grammar checker
+            node.decorators = decorators;
+            node.modifiers = modifiers;
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
+        }
+        function parseImportDeclarationOrImportEqualsDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            parseExpected(99 /* ImportKeyword */);
+            var afterImportPos = scanner.getStartPos();
+            // We don't parse the identifier here in await context, instead we will report a grammar error in the checker.
+            var identifier;
+            if (isIdentifier()) {
+                identifier = parseIdentifier();
             }
-        }
-        return out;
-    }
-    ts.convertCompilerOptionsForTelemetry = convertCompilerOptionsForTelemetry;
-    function getOptionValueWithEmptyStrings(value, option) {
-        switch (option.type) {
-            case "object": // "paths". Can't get any useful information from the value since we blank out strings, so just return "".
-                return "";
-            case "string": // Could be any arbitrary string -- use empty string instead.
-                return "";
-            case "number": // Allow numbers, but be sure to check it's actually a number.
-                return typeof value === "number" ? value : "";
-            case "boolean":
-                return typeof value === "boolean" ? value : "";
-            case "list":
-                var elementType_1 = option.element;
-                return ts.isArray(value) ? value.map(function (v) { return getOptionValueWithEmptyStrings(v, elementType_1); }) : "";
-            default:
-                return ts.forEachEntry(option.type, function (optionEnumValue, optionStringValue) {
-                    if (optionEnumValue === value) {
-                        return optionStringValue;
-                    }
-                }); // TODO: GH#18217
-        }
-    }
-})(ts || (ts = {}));
-var ts;
-(function (ts) {
-    function trace(host) {
-        host.trace(ts.formatMessage.apply(undefined, arguments));
-    }
-    ts.trace = trace;
-    /* @internal */
-    function isTraceEnabled(compilerOptions, host) {
-        return !!compilerOptions.traceResolution && host.trace !== undefined;
-    }
-    ts.isTraceEnabled = isTraceEnabled;
-    function withPackageId(packageInfo, r) {
-        var packageId;
-        if (r && packageInfo) {
-            var packageJsonContent = packageInfo.packageJsonContent;
-            if (typeof packageJsonContent.name === "string" && typeof packageJsonContent.version === "string") {
-                packageId = {
-                    name: packageJsonContent.name,
-                    subModuleName: r.path.slice(packageInfo.packageDirectory.length + ts.directorySeparator.length),
-                    version: packageJsonContent.version
-                };
+            var isTypeOnly = false;
+            if (token() !== 153 /* FromKeyword */ &&
+                (identifier === null || identifier === void 0 ? void 0 : identifier.escapedText) === "type" &&
+                (isIdentifier() || tokenAfterImportDefinitelyProducesImportDeclaration())) {
+                isTypeOnly = true;
+                identifier = isIdentifier() ? parseIdentifier() : undefined;
             }
+            if (identifier && !tokenAfterImportedIdentifierDefinitelyProducesImportDeclaration()) {
+                return parseImportEqualsDeclaration(pos, hasJSDoc, decorators, modifiers, identifier, isTypeOnly);
+            }
+            // ImportDeclaration:
+            //  import ImportClause from ModuleSpecifier ;
+            //  import ModuleSpecifier;
+            var importClause;
+            if (identifier || // import id
+                token() === 41 /* AsteriskToken */ || // import *
+                token() === 18 /* OpenBraceToken */ // import {
+            ) {
+                importClause = parseImportClause(identifier, afterImportPos, isTypeOnly);
+                parseExpected(153 /* FromKeyword */);
+            }
+            var moduleSpecifier = parseModuleSpecifier();
+            parseSemicolon();
+            var node = factory.createImportDeclaration(decorators, modifiers, importClause, moduleSpecifier);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        return r && { path: r.path, extension: r.ext, packageId: packageId };
-    }
-    function noPackageId(r) {
-        return withPackageId(/*packageInfo*/ undefined, r);
-    }
-    function removeIgnoredPackageId(r) {
-        if (r) {
-            ts.Debug.assert(r.packageId === undefined);
-            return { path: r.path, ext: r.extension };
-        }
-    }
-    /**
-     * Kinds of file that we are currently looking for.
-     * Typically there is one pass with Extensions.TypeScript, then a second pass with Extensions.JavaScript.
-     */
-    var Extensions;
-    (function (Extensions) {
-        Extensions[Extensions["TypeScript"] = 0] = "TypeScript";
-        Extensions[Extensions["JavaScript"] = 1] = "JavaScript";
-        Extensions[Extensions["Json"] = 2] = "Json";
-        Extensions[Extensions["TSConfig"] = 3] = "TSConfig";
-        Extensions[Extensions["DtsOnly"] = 4] = "DtsOnly"; /** Only '.d.ts' */
-    })(Extensions || (Extensions = {}));
-    /** Used with `Extensions.DtsOnly` to extract the path from TypeScript results. */
-    function resolvedTypeScriptOnly(resolved) {
-        if (!resolved) {
-            return undefined;
+        function tokenAfterImportDefinitelyProducesImportDeclaration() {
+            return token() === 41 /* AsteriskToken */ || token() === 18 /* OpenBraceToken */;
         }
-        ts.Debug.assert(ts.extensionIsTS(resolved.extension));
-        return { fileName: resolved.path, packageId: resolved.packageId };
-    }
-    function createResolvedModuleWithFailedLookupLocations(resolved, isExternalLibraryImport, failedLookupLocations, resultFromCache) {
-        var _a;
-        if (resultFromCache) {
-            (_a = resultFromCache.failedLookupLocations).push.apply(_a, failedLookupLocations);
-            return resultFromCache;
+        function tokenAfterImportedIdentifierDefinitelyProducesImportDeclaration() {
+            // In `import id ___`, the current token decides whether to produce
+            // an ImportDeclaration or ImportEqualsDeclaration.
+            return token() === 27 /* CommaToken */ || token() === 153 /* FromKeyword */;
         }
-        return {
-            resolvedModule: resolved && { resolvedFileName: resolved.path, originalPath: resolved.originalPath === true ? undefined : resolved.originalPath, extension: resolved.extension, isExternalLibraryImport: isExternalLibraryImport, packageId: resolved.packageId },
-            failedLookupLocations: failedLookupLocations
-        };
-    }
-    function readPackageJsonField(jsonContent, fieldName, typeOfTag, state) {
-        if (!ts.hasProperty(jsonContent, fieldName)) {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.package_json_does_not_have_a_0_field, fieldName);
+        function parseImportEqualsDeclaration(pos, hasJSDoc, decorators, modifiers, identifier, isTypeOnly) {
+            parseExpected(62 /* EqualsToken */);
+            var moduleReference = parseModuleReference();
+            parseSemicolon();
+            var node = factory.createImportEqualsDeclaration(decorators, modifiers, identifier, moduleReference);
+            var finished = withJSDoc(finishNode(node, pos), hasJSDoc);
+            if (isTypeOnly) {
+                parseErrorAtRange(finished, ts.Diagnostics.Only_ECMAScript_imports_may_use_import_type);
             }
-            return;
+            return finished;
         }
-        var value = jsonContent[fieldName];
-        if (typeof value !== typeOfTag || value === null) { // eslint-disable-line no-null/no-null
-            if (state.traceEnabled) {
-                // eslint-disable-next-line no-null/no-null
-                trace(state.host, ts.Diagnostics.Expected_type_of_0_field_in_package_json_to_be_1_got_2, fieldName, typeOfTag, value === null ? "null" : typeof value);
+        function parseImportClause(identifier, pos, isTypeOnly) {
+            // ImportClause:
+            //  ImportedDefaultBinding
+            //  NameSpaceImport
+            //  NamedImports
+            //  ImportedDefaultBinding, NameSpaceImport
+            //  ImportedDefaultBinding, NamedImports
+            // If there was no default import or if there is comma token after default import
+            // parse namespace or named imports
+            var namedBindings;
+            if (!identifier ||
+                parseOptional(27 /* CommaToken */)) {
+                namedBindings = token() === 41 /* AsteriskToken */ ? parseNamespaceImport() : parseNamedImportsOrExports(264 /* NamedImports */);
             }
-            return;
+            return finishNode(factory.createImportClause(isTypeOnly, identifier, namedBindings), pos);
         }
-        return value;
-    }
-    function readPackageJsonPathField(jsonContent, fieldName, baseDirectory, state) {
-        var fileName = readPackageJsonField(jsonContent, fieldName, "string", state);
-        if (fileName === undefined) {
-            return;
+        function parseModuleReference() {
+            return isExternalModuleReference()
+                ? parseExternalModuleReference()
+                : parseEntityName(/*allowReservedWords*/ false);
         }
-        if (!fileName) {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.package_json_had_a_falsy_0_field, fieldName);
+        function parseExternalModuleReference() {
+            var pos = getNodePos();
+            parseExpected(143 /* RequireKeyword */);
+            parseExpected(20 /* OpenParenToken */);
+            var expression = parseModuleSpecifier();
+            parseExpected(21 /* CloseParenToken */);
+            return finishNode(factory.createExternalModuleReference(expression), pos);
+        }
+        function parseModuleSpecifier() {
+            if (token() === 10 /* StringLiteral */) {
+                var result = parseLiteralNode();
+                result.text = internIdentifier(result.text);
+                return result;
+            }
+            else {
+                // We allow arbitrary expressions here, even though the grammar only allows string
+                // literals.  We check to ensure that it is only a string literal later in the grammar
+                // check pass.
+                return parseExpression();
             }
-            return;
         }
-        var path = ts.normalizePath(ts.combinePaths(baseDirectory, fileName));
-        if (state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.package_json_has_0_field_1_that_references_2, fieldName, fileName, path);
+        function parseNamespaceImport() {
+            // NameSpaceImport:
+            //  * as ImportedBinding
+            var pos = getNodePos();
+            parseExpected(41 /* AsteriskToken */);
+            parseExpected(126 /* AsKeyword */);
+            var name = parseIdentifier();
+            return finishNode(factory.createNamespaceImport(name), pos);
         }
-        return path;
-    }
-    function readPackageJsonTypesFields(jsonContent, baseDirectory, state) {
-        return readPackageJsonPathField(jsonContent, "typings", baseDirectory, state)
-            || readPackageJsonPathField(jsonContent, "types", baseDirectory, state);
-    }
-    function readPackageJsonTSConfigField(jsonContent, baseDirectory, state) {
-        return readPackageJsonPathField(jsonContent, "tsconfig", baseDirectory, state);
-    }
-    function readPackageJsonMainField(jsonContent, baseDirectory, state) {
-        return readPackageJsonPathField(jsonContent, "main", baseDirectory, state);
-    }
-    function readPackageJsonTypesVersionsField(jsonContent, state) {
-        var typesVersions = readPackageJsonField(jsonContent, "typesVersions", "object", state);
-        if (typesVersions === undefined)
-            return;
-        if (state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_field_with_version_specific_path_mappings);
+        function parseNamedImportsOrExports(kind) {
+            var pos = getNodePos();
+            // NamedImports:
+            //  { }
+            //  { ImportsList }
+            //  { ImportsList, }
+            // ImportsList:
+            //  ImportSpecifier
+            //  ImportsList, ImportSpecifier
+            var node = kind === 264 /* NamedImports */
+                ? factory.createNamedImports(parseBracketedList(23 /* ImportOrExportSpecifiers */, parseImportSpecifier, 18 /* OpenBraceToken */, 19 /* CloseBraceToken */))
+                : factory.createNamedExports(parseBracketedList(23 /* ImportOrExportSpecifiers */, parseExportSpecifier, 18 /* OpenBraceToken */, 19 /* CloseBraceToken */));
+            return finishNode(node, pos);
         }
-        return typesVersions;
-    }
-    function readPackageJsonTypesVersionPaths(jsonContent, state) {
-        var typesVersions = readPackageJsonTypesVersionsField(jsonContent, state);
-        if (typesVersions === undefined)
-            return;
-        if (state.traceEnabled) {
-            for (var key in typesVersions) {
-                if (ts.hasProperty(typesVersions, key) && !ts.VersionRange.tryParse(key)) {
-                    trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_entry_0_that_is_not_a_valid_semver_range, key);
-                }
-            }
+        function parseExportSpecifier() {
+            return parseImportOrExportSpecifier(270 /* ExportSpecifier */);
         }
-        var result = getPackageJsonTypesVersionsPaths(typesVersions);
-        if (!result) {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.package_json_does_not_have_a_typesVersions_entry_that_matches_version_0, ts.versionMajorMinor);
+        function parseImportSpecifier() {
+            return parseImportOrExportSpecifier(265 /* ImportSpecifier */);
+        }
+        function parseImportOrExportSpecifier(kind) {
+            var pos = getNodePos();
+            // ImportSpecifier:
+            //   BindingIdentifier
+            //   IdentifierName as BindingIdentifier
+            // ExportSpecifier:
+            //   IdentifierName
+            //   IdentifierName as IdentifierName
+            var checkIdentifierIsKeyword = ts.isKeyword(token()) && !isIdentifier();
+            var checkIdentifierStart = scanner.getTokenPos();
+            var checkIdentifierEnd = scanner.getTextPos();
+            var identifierName = parseIdentifierName();
+            var propertyName;
+            var name;
+            if (token() === 126 /* AsKeyword */) {
+                propertyName = identifierName;
+                parseExpected(126 /* AsKeyword */);
+                checkIdentifierIsKeyword = ts.isKeyword(token()) && !isIdentifier();
+                checkIdentifierStart = scanner.getTokenPos();
+                checkIdentifierEnd = scanner.getTextPos();
+                name = parseIdentifierName();
             }
-            return;
+            else {
+                name = identifierName;
+            }
+            if (kind === 265 /* ImportSpecifier */ && checkIdentifierIsKeyword) {
+                parseErrorAt(checkIdentifierStart, checkIdentifierEnd, ts.Diagnostics.Identifier_expected);
+            }
+            var node = kind === 265 /* ImportSpecifier */
+                ? factory.createImportSpecifier(propertyName, name)
+                : factory.createExportSpecifier(propertyName, name);
+            return finishNode(node, pos);
         }
-        var bestVersionKey = result.version, bestVersionPaths = result.paths;
-        if (typeof bestVersionPaths !== "object") {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Expected_type_of_0_field_in_package_json_to_be_1_got_2, "typesVersions['" + bestVersionKey + "']", "object", typeof bestVersionPaths);
+        function parseNamespaceExport(pos) {
+            return finishNode(factory.createNamespaceExport(parseIdentifierName()), pos);
+        }
+        function parseExportDeclaration(pos, hasJSDoc, decorators, modifiers) {
+            var savedAwaitContext = inAwaitContext();
+            setAwaitContext(/*value*/ true);
+            var exportClause;
+            var moduleSpecifier;
+            var isTypeOnly = parseOptional(149 /* TypeKeyword */);
+            var namespaceExportPos = getNodePos();
+            if (parseOptional(41 /* AsteriskToken */)) {
+                if (parseOptional(126 /* AsKeyword */)) {
+                    exportClause = parseNamespaceExport(namespaceExportPos);
+                }
+                parseExpected(153 /* FromKeyword */);
+                moduleSpecifier = parseModuleSpecifier();
             }
-            return;
+            else {
+                exportClause = parseNamedImportsOrExports(268 /* NamedExports */);
+                // It is not uncommon to accidentally omit the 'from' keyword. Additionally, in editing scenarios,
+                // the 'from' keyword can be parsed as a named export when the export clause is unterminated (i.e. `export { from "moduleName";`)
+                // If we don't have a 'from' keyword, see if we have a string literal such that ASI won't take effect.
+                if (token() === 153 /* FromKeyword */ || (token() === 10 /* StringLiteral */ && !scanner.hasPrecedingLineBreak())) {
+                    parseExpected(153 /* FromKeyword */);
+                    moduleSpecifier = parseModuleSpecifier();
+                }
+            }
+            parseSemicolon();
+            setAwaitContext(savedAwaitContext);
+            var node = factory.createExportDeclaration(decorators, modifiers, isTypeOnly, exportClause, moduleSpecifier);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-        return result;
-    }
-    var typeScriptVersion;
-    /* @internal */
-    function getPackageJsonTypesVersionsPaths(typesVersions) {
-        if (!typeScriptVersion)
-            typeScriptVersion = new ts.Version(ts.version);
-        for (var key in typesVersions) {
-            if (!ts.hasProperty(typesVersions, key))
-                continue;
-            var keyRange = ts.VersionRange.tryParse(key);
-            if (keyRange === undefined) {
-                continue;
+        function parseExportAssignment(pos, hasJSDoc, decorators, modifiers) {
+            var savedAwaitContext = inAwaitContext();
+            setAwaitContext(/*value*/ true);
+            var isExportEquals;
+            if (parseOptional(62 /* EqualsToken */)) {
+                isExportEquals = true;
             }
-            // return the first entry whose range matches the current compiler version.
-            if (keyRange.test(typeScriptVersion)) {
-                return { version: key, paths: typesVersions[key] };
+            else {
+                parseExpected(87 /* DefaultKeyword */);
             }
+            var expression = parseAssignmentExpressionOrHigher();
+            parseSemicolon();
+            setAwaitContext(savedAwaitContext);
+            var node = factory.createExportAssignment(decorators, modifiers, isExportEquals, expression);
+            return withJSDoc(finishNode(node, pos), hasJSDoc);
         }
-    }
-    ts.getPackageJsonTypesVersionsPaths = getPackageJsonTypesVersionsPaths;
-    function getEffectiveTypeRoots(options, host) {
-        if (options.typeRoots) {
-            return options.typeRoots;
+        function setExternalModuleIndicator(sourceFile) {
+            // Try to use the first top-level import/export when available, then
+            // fall back to looking for an 'import.meta' somewhere in the tree if necessary.
+            sourceFile.externalModuleIndicator =
+                ts.forEach(sourceFile.statements, isAnExternalModuleIndicatorNode) ||
+                    getImportMetaIfNecessary(sourceFile);
         }
-        var currentDirectory;
-        if (options.configFilePath) {
-            currentDirectory = ts.getDirectoryPath(options.configFilePath);
+        function isAnExternalModuleIndicatorNode(node) {
+            return hasModifierOfKind(node, 92 /* ExportKeyword */)
+                || ts.isImportEqualsDeclaration(node) && ts.isExternalModuleReference(node.moduleReference)
+                || ts.isImportDeclaration(node)
+                || ts.isExportAssignment(node)
+                || ts.isExportDeclaration(node) ? node : undefined;
         }
-        else if (host.getCurrentDirectory) {
-            currentDirectory = host.getCurrentDirectory();
+        function getImportMetaIfNecessary(sourceFile) {
+            return sourceFile.flags & 2097152 /* PossiblyContainsImportMeta */ ?
+                walkTreeForExternalModuleIndicators(sourceFile) :
+                undefined;
         }
-        if (currentDirectory !== undefined) {
-            return getDefaultTypeRoots(currentDirectory, host);
+        function walkTreeForExternalModuleIndicators(node) {
+            return isImportMeta(node) ? node : forEachChild(node, walkTreeForExternalModuleIndicators);
         }
-    }
-    ts.getEffectiveTypeRoots = getEffectiveTypeRoots;
-    /**
-     * Returns the path to every node_modules/@types directory from some ancestor directory.
-     * Returns undefined if there are none.
-     */
-    function getDefaultTypeRoots(currentDirectory, host) {
-        if (!host.directoryExists) {
-            return [ts.combinePaths(currentDirectory, nodeModulesAtTypes)];
-            // And if it doesn't exist, tough.
+        /** Do not use hasModifier inside the parser; it relies on parent pointers. Use this instead. */
+        function hasModifierOfKind(node, kind) {
+            return ts.some(node.modifiers, function (m) { return m.kind === kind; });
         }
-        var typeRoots;
-        ts.forEachAncestorDirectory(ts.normalizePath(currentDirectory), function (directory) {
-            var atTypes = ts.combinePaths(directory, nodeModulesAtTypes);
-            if (host.directoryExists(atTypes)) {
-                (typeRoots || (typeRoots = [])).push(atTypes);
-            }
-            return undefined;
-        });
-        return typeRoots;
-    }
-    var nodeModulesAtTypes = ts.combinePaths("node_modules", "@types");
-    /**
-     * @param {string | undefined} containingFile - file that contains type reference directive, can be undefined if containing file is unknown.
-     * This is possible in case if resolution is performed for directives specified via 'types' parameter. In this case initial path for secondary lookups
-     * is assumed to be the same as root directory of the project.
-     */
-    function resolveTypeReferenceDirective(typeReferenceDirectiveName, containingFile, options, host, redirectedReference) {
-        var traceEnabled = isTraceEnabled(options, host);
-        if (redirectedReference) {
-            options = redirectedReference.commandLine.options;
+        function isImportMeta(node) {
+            return ts.isMetaProperty(node) && node.keywordToken === 99 /* ImportKeyword */ && node.name.escapedText === "meta";
         }
-        var failedLookupLocations = [];
-        var moduleResolutionState = { compilerOptions: options, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
-        var typeRoots = getEffectiveTypeRoots(options, host);
-        if (traceEnabled) {
-            if (containingFile === undefined) {
-                if (typeRoots === undefined) {
-                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_not_set_root_directory_not_set, typeReferenceDirectiveName);
-                }
-                else {
-                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_not_set_root_directory_1, typeReferenceDirectiveName, typeRoots);
+        var ParsingContext;
+        (function (ParsingContext) {
+            ParsingContext[ParsingContext["SourceElements"] = 0] = "SourceElements";
+            ParsingContext[ParsingContext["BlockStatements"] = 1] = "BlockStatements";
+            ParsingContext[ParsingContext["SwitchClauses"] = 2] = "SwitchClauses";
+            ParsingContext[ParsingContext["SwitchClauseStatements"] = 3] = "SwitchClauseStatements";
+            ParsingContext[ParsingContext["TypeMembers"] = 4] = "TypeMembers";
+            ParsingContext[ParsingContext["ClassMembers"] = 5] = "ClassMembers";
+            ParsingContext[ParsingContext["EnumMembers"] = 6] = "EnumMembers";
+            ParsingContext[ParsingContext["HeritageClauseElement"] = 7] = "HeritageClauseElement";
+            ParsingContext[ParsingContext["VariableDeclarations"] = 8] = "VariableDeclarations";
+            ParsingContext[ParsingContext["ObjectBindingElements"] = 9] = "ObjectBindingElements";
+            ParsingContext[ParsingContext["ArrayBindingElements"] = 10] = "ArrayBindingElements";
+            ParsingContext[ParsingContext["ArgumentExpressions"] = 11] = "ArgumentExpressions";
+            ParsingContext[ParsingContext["ObjectLiteralMembers"] = 12] = "ObjectLiteralMembers";
+            ParsingContext[ParsingContext["JsxAttributes"] = 13] = "JsxAttributes";
+            ParsingContext[ParsingContext["JsxChildren"] = 14] = "JsxChildren";
+            ParsingContext[ParsingContext["ArrayLiteralMembers"] = 15] = "ArrayLiteralMembers";
+            ParsingContext[ParsingContext["Parameters"] = 16] = "Parameters";
+            ParsingContext[ParsingContext["JSDocParameters"] = 17] = "JSDocParameters";
+            ParsingContext[ParsingContext["RestProperties"] = 18] = "RestProperties";
+            ParsingContext[ParsingContext["TypeParameters"] = 19] = "TypeParameters";
+            ParsingContext[ParsingContext["TypeArguments"] = 20] = "TypeArguments";
+            ParsingContext[ParsingContext["TupleElementTypes"] = 21] = "TupleElementTypes";
+            ParsingContext[ParsingContext["HeritageClauses"] = 22] = "HeritageClauses";
+            ParsingContext[ParsingContext["ImportOrExportSpecifiers"] = 23] = "ImportOrExportSpecifiers";
+            ParsingContext[ParsingContext["Count"] = 24] = "Count"; // Number of parsing contexts
+        })(ParsingContext || (ParsingContext = {}));
+        var Tristate;
+        (function (Tristate) {
+            Tristate[Tristate["False"] = 0] = "False";
+            Tristate[Tristate["True"] = 1] = "True";
+            Tristate[Tristate["Unknown"] = 2] = "Unknown";
+        })(Tristate || (Tristate = {}));
+        var JSDocParser;
+        (function (JSDocParser) {
+            function parseJSDocTypeExpressionForTests(content, start, length) {
+                initializeState("file.js", content, 99 /* Latest */, /*_syntaxCursor:*/ undefined, 1 /* JS */);
+                scanner.setText(content, start, length);
+                currentToken = scanner.scan();
+                var jsDocTypeExpression = parseJSDocTypeExpression();
+                var sourceFile = createSourceFile("file.js", 99 /* Latest */, 1 /* JS */, /*isDeclarationFile*/ false, [], factory.createToken(1 /* EndOfFileToken */), 0 /* None */);
+                var diagnostics = ts.attachFileToDiagnostics(parseDiagnostics, sourceFile);
+                if (jsDocDiagnostics) {
+                    sourceFile.jsDocDiagnostics = ts.attachFileToDiagnostics(jsDocDiagnostics, sourceFile);
                 }
+                clearState();
+                return jsDocTypeExpression ? { jsDocTypeExpression: jsDocTypeExpression, diagnostics: diagnostics } : undefined;
             }
-            else {
-                if (typeRoots === undefined) {
-                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_1_root_directory_not_set, typeReferenceDirectiveName, containingFile);
+            JSDocParser.parseJSDocTypeExpressionForTests = parseJSDocTypeExpressionForTests;
+            // Parses out a JSDoc type expression.
+            function parseJSDocTypeExpression(mayOmitBraces) {
+                var pos = getNodePos();
+                var hasBrace = (mayOmitBraces ? parseOptional : parseExpected)(18 /* OpenBraceToken */);
+                var type = doInsideOfContext(4194304 /* JSDoc */, parseJSDocType);
+                if (!mayOmitBraces || hasBrace) {
+                    parseExpectedJSDoc(19 /* CloseBraceToken */);
                 }
-                else {
-                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_1_root_directory_2, typeReferenceDirectiveName, containingFile, typeRoots);
+                var result = factory.createJSDocTypeExpression(type);
+                fixupParentReferences(result);
+                return finishNode(result, pos);
+            }
+            JSDocParser.parseJSDocTypeExpression = parseJSDocTypeExpression;
+            function parseJSDocNameReference() {
+                var pos = getNodePos();
+                var hasBrace = parseOptional(18 /* OpenBraceToken */);
+                var entityName = parseEntityName(/* allowReservedWords*/ false);
+                if (hasBrace) {
+                    parseExpectedJSDoc(19 /* CloseBraceToken */);
                 }
+                var result = factory.createJSDocNameReference(entityName);
+                fixupParentReferences(result);
+                return finishNode(result, pos);
             }
-            if (redirectedReference) {
-                trace(host, ts.Diagnostics.Using_compiler_options_of_project_reference_redirect_0, redirectedReference.sourceFile.fileName);
+            JSDocParser.parseJSDocNameReference = parseJSDocNameReference;
+            function parseIsolatedJSDocComment(content, start, length) {
+                initializeState("", content, 99 /* Latest */, /*_syntaxCursor:*/ undefined, 1 /* JS */);
+                var jsDoc = doInsideOfContext(4194304 /* JSDoc */, function () { return parseJSDocCommentWorker(start, length); });
+                var sourceFile = { languageVariant: 0 /* Standard */, text: content };
+                var diagnostics = ts.attachFileToDiagnostics(parseDiagnostics, sourceFile);
+                clearState();
+                return jsDoc ? { jsDoc: jsDoc, diagnostics: diagnostics } : undefined;
             }
-        }
-        var resolved = primaryLookup();
-        var primary = true;
-        if (!resolved) {
-            resolved = secondaryLookup();
-            primary = false;
-        }
-        var resolvedTypeReferenceDirective;
-        if (resolved) {
-            var fileName = resolved.fileName, packageId = resolved.packageId;
-            var resolvedFileName = options.preserveSymlinks ? fileName : realPath(fileName, host, traceEnabled);
-            if (traceEnabled) {
-                if (packageId) {
-                    trace(host, ts.Diagnostics.Type_reference_directive_0_was_successfully_resolved_to_1_with_Package_ID_2_primary_Colon_3, typeReferenceDirectiveName, resolvedFileName, ts.packageIdToString(packageId), primary);
-                }
-                else {
-                    trace(host, ts.Diagnostics.Type_reference_directive_0_was_successfully_resolved_to_1_primary_Colon_2, typeReferenceDirectiveName, resolvedFileName, primary);
+            JSDocParser.parseIsolatedJSDocComment = parseIsolatedJSDocComment;
+            function parseJSDocComment(parent, start, length) {
+                var saveToken = currentToken;
+                var saveParseDiagnosticsLength = parseDiagnostics.length;
+                var saveParseErrorBeforeNextFinishedNode = parseErrorBeforeNextFinishedNode;
+                var comment = doInsideOfContext(4194304 /* JSDoc */, function () { return parseJSDocCommentWorker(start, length); });
+                ts.setParent(comment, parent);
+                if (contextFlags & 131072 /* JavaScriptFile */) {
+                    if (!jsDocDiagnostics) {
+                        jsDocDiagnostics = [];
+                    }
+                    jsDocDiagnostics.push.apply(jsDocDiagnostics, parseDiagnostics);
                 }
+                currentToken = saveToken;
+                parseDiagnostics.length = saveParseDiagnosticsLength;
+                parseErrorBeforeNextFinishedNode = saveParseErrorBeforeNextFinishedNode;
+                return comment;
             }
-            resolvedTypeReferenceDirective = { primary: primary, resolvedFileName: resolvedFileName, packageId: packageId, isExternalLibraryImport: pathContainsNodeModules(fileName) };
-        }
-        return { resolvedTypeReferenceDirective: resolvedTypeReferenceDirective, failedLookupLocations: failedLookupLocations };
-        function primaryLookup() {
-            // Check primary library paths
-            if (typeRoots && typeRoots.length) {
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Resolving_with_primary_search_path_0, typeRoots.join(", "));
+            JSDocParser.parseJSDocComment = parseJSDocComment;
+            var JSDocState;
+            (function (JSDocState) {
+                JSDocState[JSDocState["BeginningOfLine"] = 0] = "BeginningOfLine";
+                JSDocState[JSDocState["SawAsterisk"] = 1] = "SawAsterisk";
+                JSDocState[JSDocState["SavingComments"] = 2] = "SavingComments";
+                JSDocState[JSDocState["SavingBackticks"] = 3] = "SavingBackticks";
+            })(JSDocState || (JSDocState = {}));
+            var PropertyLikeParse;
+            (function (PropertyLikeParse) {
+                PropertyLikeParse[PropertyLikeParse["Property"] = 1] = "Property";
+                PropertyLikeParse[PropertyLikeParse["Parameter"] = 2] = "Parameter";
+                PropertyLikeParse[PropertyLikeParse["CallbackParameter"] = 4] = "CallbackParameter";
+            })(PropertyLikeParse || (PropertyLikeParse = {}));
+            function parseJSDocCommentWorker(start, length) {
+                if (start === void 0) { start = 0; }
+                var content = sourceText;
+                var end = length === undefined ? content.length : start + length;
+                length = end - start;
+                ts.Debug.assert(start >= 0);
+                ts.Debug.assert(start <= end);
+                ts.Debug.assert(end <= content.length);
+                // Check for /** (JSDoc opening part)
+                if (!isJSDocLikeText(content, start)) {
+                    return undefined;
                 }
-                return ts.firstDefined(typeRoots, function (typeRoot) {
-                    var candidate = ts.combinePaths(typeRoot, typeReferenceDirectiveName);
-                    var candidateDirectory = ts.getDirectoryPath(candidate);
-                    var directoryExists = ts.directoryProbablyExists(candidateDirectory, host);
-                    if (!directoryExists && traceEnabled) {
-                        trace(host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, candidateDirectory);
+                var tags;
+                var tagsPos;
+                var tagsEnd;
+                var comments = [];
+                // + 3 for leading /**, - 5 in total for /** */
+                return scanner.scanRange(start + 3, length - 5, function () {
+                    // Initially we can parse out a tag.  We also have seen a starting asterisk.
+                    // This is so that /** * @type */ doesn't parse.
+                    var state = 1 /* SawAsterisk */;
+                    var margin;
+                    // + 4 for leading '/** '
+                    // + 1 because the last index of \n is always one index before the first character in the line and coincidentally, if there is no \n before start, it is -1, which is also one index before the first character
+                    var indent = start - (content.lastIndexOf("\n", start) + 1) + 4;
+                    function pushComment(text) {
+                        if (!margin) {
+                            margin = indent;
+                        }
+                        comments.push(text);
+                        indent += text.length;
                     }
-                    return resolvedTypeScriptOnly(loadNodeModuleFromDirectory(Extensions.DtsOnly, candidate, !directoryExists, moduleResolutionState));
+                    nextTokenJSDoc();
+                    while (parseOptionalJsdoc(5 /* WhitespaceTrivia */))
+                        ;
+                    if (parseOptionalJsdoc(4 /* NewLineTrivia */)) {
+                        state = 0 /* BeginningOfLine */;
+                        indent = 0;
+                    }
+                    loop: while (true) {
+                        switch (token()) {
+                            case 59 /* AtToken */:
+                                if (state === 0 /* BeginningOfLine */ || state === 1 /* SawAsterisk */) {
+                                    removeTrailingWhitespace(comments);
+                                    addTag(parseTag(indent));
+                                    // NOTE: According to usejsdoc.org, a tag goes to end of line, except the last tag.
+                                    // Real-world comments may break this rule, so "BeginningOfLine" will not be a real line beginning
+                                    // for malformed examples like `/** @param {string} x @returns {number} the length */`
+                                    state = 0 /* BeginningOfLine */;
+                                    margin = undefined;
+                                }
+                                else {
+                                    pushComment(scanner.getTokenText());
+                                }
+                                break;
+                            case 4 /* NewLineTrivia */:
+                                comments.push(scanner.getTokenText());
+                                state = 0 /* BeginningOfLine */;
+                                indent = 0;
+                                break;
+                            case 41 /* AsteriskToken */:
+                                var asterisk = scanner.getTokenText();
+                                if (state === 1 /* SawAsterisk */ || state === 2 /* SavingComments */) {
+                                    // If we've already seen an asterisk, then we can no longer parse a tag on this line
+                                    state = 2 /* SavingComments */;
+                                    pushComment(asterisk);
+                                }
+                                else {
+                                    // Ignore the first asterisk on a line
+                                    state = 1 /* SawAsterisk */;
+                                    indent += asterisk.length;
+                                }
+                                break;
+                            case 5 /* WhitespaceTrivia */:
+                                // only collect whitespace if we're already saving comments or have just crossed the comment indent margin
+                                var whitespace = scanner.getTokenText();
+                                if (state === 2 /* SavingComments */) {
+                                    comments.push(whitespace);
+                                }
+                                else if (margin !== undefined && indent + whitespace.length > margin) {
+                                    comments.push(whitespace.slice(margin - indent));
+                                }
+                                indent += whitespace.length;
+                                break;
+                            case 1 /* EndOfFileToken */:
+                                break loop;
+                            default:
+                                // Anything else is doc comment text. We just save it. Because it
+                                // wasn't a tag, we can no longer parse a tag on this line until we hit the next
+                                // line break.
+                                state = 2 /* SavingComments */;
+                                pushComment(scanner.getTokenText());
+                                break;
+                        }
+                        nextTokenJSDoc();
+                    }
+                    removeLeadingNewlines(comments);
+                    removeTrailingWhitespace(comments);
+                    return createJSDocComment();
                 });
-            }
-            else {
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Root_directory_cannot_be_determined_skipping_primary_search_paths);
+                function removeLeadingNewlines(comments) {
+                    while (comments.length && (comments[0] === "\n" || comments[0] === "\r")) {
+                        comments.shift();
+                    }
                 }
-            }
-        }
-        function secondaryLookup() {
-            var initialLocationForSecondaryLookup = containingFile && ts.getDirectoryPath(containingFile);
-            if (initialLocationForSecondaryLookup !== undefined) {
-                // check secondary locations
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Looking_up_in_node_modules_folder_initial_location_0, initialLocationForSecondaryLookup);
+                function removeTrailingWhitespace(comments) {
+                    while (comments.length && comments[comments.length - 1].trim() === "") {
+                        comments.pop();
+                    }
                 }
-                var result = void 0;
-                if (!ts.isExternalModuleNameRelative(typeReferenceDirectiveName)) {
-                    var searchResult = loadModuleFromNearestNodeModulesDirectory(Extensions.DtsOnly, typeReferenceDirectiveName, initialLocationForSecondaryLookup, moduleResolutionState, /*cache*/ undefined, /*redirectedReference*/ undefined);
-                    result = searchResult && searchResult.value;
+                function createJSDocComment() {
+                    var comment = comments.length ? comments.join("") : undefined;
+                    var tagsArray = tags && createNodeArray(tags, tagsPos, tagsEnd);
+                    return finishNode(factory.createJSDocComment(comment, tagsArray), start, end);
                 }
-                else {
-                    var candidate = ts.normalizePathAndParts(ts.combinePaths(initialLocationForSecondaryLookup, typeReferenceDirectiveName)).path;
-                    result = nodeLoadModuleByRelativeName(Extensions.DtsOnly, candidate, /*onlyRecordFailures*/ false, moduleResolutionState, /*considerPackageJson*/ true);
+                function isNextNonwhitespaceTokenEndOfFile() {
+                    // We must use infinite lookahead, as there could be any number of newlines :(
+                    while (true) {
+                        nextTokenJSDoc();
+                        if (token() === 1 /* EndOfFileToken */) {
+                            return true;
+                        }
+                        if (!(token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */)) {
+                            return false;
+                        }
+                    }
                 }
-                var resolvedFile = resolvedTypeScriptOnly(result);
-                if (!resolvedFile && traceEnabled) {
-                    trace(host, ts.Diagnostics.Type_reference_directive_0_was_not_resolved, typeReferenceDirectiveName);
+                function skipWhitespace() {
+                    if (token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
+                        if (lookAhead(isNextNonwhitespaceTokenEndOfFile)) {
+                            return; // Don't skip whitespace prior to EoF (or end of comment) - that shouldn't be included in any node's range
+                        }
+                    }
+                    while (token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
+                        nextTokenJSDoc();
+                    }
                 }
-                return resolvedFile;
-            }
-            else {
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Containing_file_is_not_specified_and_root_directory_cannot_be_determined_skipping_lookup_in_node_modules_folder);
+                function skipWhitespaceOrAsterisk() {
+                    if (token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
+                        if (lookAhead(isNextNonwhitespaceTokenEndOfFile)) {
+                            return ""; // Don't skip whitespace prior to EoF (or end of comment) - that shouldn't be included in any node's range
+                        }
+                    }
+                    var precedingLineBreak = scanner.hasPrecedingLineBreak();
+                    var seenLineBreak = false;
+                    var indentText = "";
+                    while ((precedingLineBreak && token() === 41 /* AsteriskToken */) || token() === 5 /* WhitespaceTrivia */ || token() === 4 /* NewLineTrivia */) {
+                        indentText += scanner.getTokenText();
+                        if (token() === 4 /* NewLineTrivia */) {
+                            precedingLineBreak = true;
+                            seenLineBreak = true;
+                            indentText = "";
+                        }
+                        else if (token() === 41 /* AsteriskToken */) {
+                            precedingLineBreak = false;
+                        }
+                        nextTokenJSDoc();
+                    }
+                    return seenLineBreak ? indentText : "";
                 }
-            }
-        }
-    }
-    ts.resolveTypeReferenceDirective = resolveTypeReferenceDirective;
-    /**
-     * Given a set of options, returns the set of type directive names
-     *   that should be included for this program automatically.
-     * This list could either come from the config file,
-     *   or from enumerating the types root + initial secondary types lookup location.
-     * More type directives might appear in the program later as a result of loading actual source files;
-     *   this list is only the set of defaults that are implicitly included.
-     */
-    function getAutomaticTypeDirectiveNames(options, host) {
-        // Use explicit type list from tsconfig.json
-        if (options.types) {
-            return options.types;
-        }
-        // Walk the primary type lookup locations
-        var result = [];
-        if (host.directoryExists && host.getDirectories) {
-            var typeRoots = getEffectiveTypeRoots(options, host);
-            if (typeRoots) {
-                for (var _i = 0, typeRoots_1 = typeRoots; _i < typeRoots_1.length; _i++) {
-                    var root = typeRoots_1[_i];
-                    if (host.directoryExists(root)) {
-                        for (var _a = 0, _b = host.getDirectories(root); _a < _b.length; _a++) {
-                            var typeDirectivePath = _b[_a];
-                            var normalized = ts.normalizePath(typeDirectivePath);
-                            var packageJsonPath = ts.combinePaths(root, normalized, "package.json");
-                            // `types-publisher` sometimes creates packages with `"typings": null` for packages that don't provide their own types.
-                            // See `createNotNeededPackageJSON` in the types-publisher` repo.
-                            // eslint-disable-next-line no-null/no-null
-                            var isNotNeededPackage = host.fileExists(packageJsonPath) && ts.readJson(packageJsonPath, host).typings === null;
-                            if (!isNotNeededPackage) {
-                                var baseFileName = ts.getBaseFileName(normalized);
-                                // At this stage, skip results with leading dot.
-                                if (baseFileName.charCodeAt(0) !== 46 /* dot */) {
-                                    // Return just the type directive names
-                                    result.push(baseFileName);
+                function parseTag(margin) {
+                    ts.Debug.assert(token() === 59 /* AtToken */);
+                    var start = scanner.getTokenPos();
+                    nextTokenJSDoc();
+                    var tagName = parseJSDocIdentifierName(/*message*/ undefined);
+                    var indentText = skipWhitespaceOrAsterisk();
+                    var tag;
+                    switch (tagName.escapedText) {
+                        case "author":
+                            tag = parseAuthorTag(start, tagName, margin, indentText);
+                            break;
+                        case "implements":
+                            tag = parseImplementsTag(start, tagName, margin, indentText);
+                            break;
+                        case "augments":
+                        case "extends":
+                            tag = parseAugmentsTag(start, tagName, margin, indentText);
+                            break;
+                        case "class":
+                        case "constructor":
+                            tag = parseSimpleTag(start, factory.createJSDocClassTag, tagName, margin, indentText);
+                            break;
+                        case "public":
+                            tag = parseSimpleTag(start, factory.createJSDocPublicTag, tagName, margin, indentText);
+                            break;
+                        case "private":
+                            tag = parseSimpleTag(start, factory.createJSDocPrivateTag, tagName, margin, indentText);
+                            break;
+                        case "protected":
+                            tag = parseSimpleTag(start, factory.createJSDocProtectedTag, tagName, margin, indentText);
+                            break;
+                        case "readonly":
+                            tag = parseSimpleTag(start, factory.createJSDocReadonlyTag, tagName, margin, indentText);
+                            break;
+                        case "deprecated":
+                            hasDeprecatedTag = true;
+                            tag = parseSimpleTag(start, factory.createJSDocDeprecatedTag, tagName, margin, indentText);
+                            break;
+                        case "this":
+                            tag = parseThisTag(start, tagName, margin, indentText);
+                            break;
+                        case "enum":
+                            tag = parseEnumTag(start, tagName, margin, indentText);
+                            break;
+                        case "arg":
+                        case "argument":
+                        case "param":
+                            return parseParameterOrPropertyTag(start, tagName, 2 /* Parameter */, margin);
+                        case "return":
+                        case "returns":
+                            tag = parseReturnTag(start, tagName, margin, indentText);
+                            break;
+                        case "template":
+                            tag = parseTemplateTag(start, tagName, margin, indentText);
+                            break;
+                        case "type":
+                            tag = parseTypeTag(start, tagName, margin, indentText);
+                            break;
+                        case "typedef":
+                            tag = parseTypedefTag(start, tagName, margin, indentText);
+                            break;
+                        case "callback":
+                            tag = parseCallbackTag(start, tagName, margin, indentText);
+                            break;
+                        case "see":
+                            tag = parseSeeTag(start, tagName, margin, indentText);
+                            break;
+                        default:
+                            tag = parseUnknownTag(start, tagName, margin, indentText);
+                            break;
+                    }
+                    return tag;
+                }
+                function parseTrailingTagComments(pos, end, margin, indentText) {
+                    // some tags, like typedef and callback, have already parsed their comments earlier
+                    if (!indentText) {
+                        margin += end - pos;
+                    }
+                    return parseTagComments(margin, indentText.slice(margin));
+                }
+                function parseTagComments(indent, initialMargin) {
+                    var comments = [];
+                    var state = 0 /* BeginningOfLine */;
+                    var margin;
+                    function pushComment(text) {
+                        if (!margin) {
+                            margin = indent;
+                        }
+                        comments.push(text);
+                        indent += text.length;
+                    }
+                    if (initialMargin !== undefined) {
+                        // jump straight to saving comments if there is some initial indentation
+                        if (initialMargin !== "") {
+                            pushComment(initialMargin);
+                        }
+                        state = 1 /* SawAsterisk */;
+                    }
+                    var tok = token();
+                    loop: while (true) {
+                        switch (tok) {
+                            case 4 /* NewLineTrivia */:
+                                state = 0 /* BeginningOfLine */;
+                                // don't use pushComment here because we want to keep the margin unchanged
+                                comments.push(scanner.getTokenText());
+                                indent = 0;
+                                break;
+                            case 59 /* AtToken */:
+                                if (state === 3 /* SavingBackticks */) {
+                                    comments.push(scanner.getTokenText());
+                                    break;
                                 }
-                            }
+                                scanner.setTextPos(scanner.getTextPos() - 1);
+                            // falls through
+                            case 1 /* EndOfFileToken */:
+                                // Done
+                                break loop;
+                            case 5 /* WhitespaceTrivia */:
+                                if (state === 2 /* SavingComments */ || state === 3 /* SavingBackticks */) {
+                                    pushComment(scanner.getTokenText());
+                                }
+                                else {
+                                    var whitespace = scanner.getTokenText();
+                                    // if the whitespace crosses the margin, take only the whitespace that passes the margin
+                                    if (margin !== undefined && indent + whitespace.length > margin) {
+                                        comments.push(whitespace.slice(margin - indent));
+                                    }
+                                    indent += whitespace.length;
+                                }
+                                break;
+                            case 18 /* OpenBraceToken */:
+                                state = 2 /* SavingComments */;
+                                if (lookAhead(function () { return nextTokenJSDoc() === 59 /* AtToken */ && ts.tokenIsIdentifierOrKeyword(nextTokenJSDoc()) && scanner.getTokenText() === "link"; })) {
+                                    pushComment(scanner.getTokenText());
+                                    nextTokenJSDoc();
+                                    pushComment(scanner.getTokenText());
+                                    nextTokenJSDoc();
+                                }
+                                pushComment(scanner.getTokenText());
+                                break;
+                            case 61 /* BacktickToken */:
+                                if (state === 3 /* SavingBackticks */) {
+                                    state = 2 /* SavingComments */;
+                                }
+                                else {
+                                    state = 3 /* SavingBackticks */;
+                                }
+                                pushComment(scanner.getTokenText());
+                                break;
+                            case 41 /* AsteriskToken */:
+                                if (state === 0 /* BeginningOfLine */) {
+                                    // leading asterisks start recording on the *next* (non-whitespace) token
+                                    state = 1 /* SawAsterisk */;
+                                    indent += 1;
+                                    break;
+                                }
+                            // record the * as a comment
+                            // falls through
+                            default:
+                                if (state !== 3 /* SavingBackticks */) {
+                                    state = 2 /* SavingComments */; // leading identifiers start recording as well
+                                }
+                                pushComment(scanner.getTokenText());
+                                break;
                         }
+                        tok = nextTokenJSDoc();
                     }
+                    removeLeadingNewlines(comments);
+                    removeTrailingWhitespace(comments);
+                    return comments.length === 0 ? undefined : comments.join("");
                 }
-            }
-        }
-        return result;
-    }
-    ts.getAutomaticTypeDirectiveNames = getAutomaticTypeDirectiveNames;
-    function createModuleResolutionCache(currentDirectory, getCanonicalFileName, options) {
-        return createModuleResolutionCacheWithMaps(createCacheWithRedirects(options), createCacheWithRedirects(options), currentDirectory, getCanonicalFileName);
-    }
-    ts.createModuleResolutionCache = createModuleResolutionCache;
-    /*@internal*/
-    function createCacheWithRedirects(options) {
-        var ownMap = ts.createMap();
-        var redirectsMap = ts.createMap();
-        return {
-            ownMap: ownMap,
-            redirectsMap: redirectsMap,
-            getOrCreateMapOfCacheRedirects: getOrCreateMapOfCacheRedirects,
-            clear: clear,
-            setOwnOptions: setOwnOptions,
-            setOwnMap: setOwnMap
-        };
-        function setOwnOptions(newOptions) {
-            options = newOptions;
-        }
-        function setOwnMap(newOwnMap) {
-            ownMap = newOwnMap;
-        }
-        function getOrCreateMapOfCacheRedirects(redirectedReference) {
-            if (!redirectedReference) {
-                return ownMap;
-            }
-            var path = redirectedReference.sourceFile.path;
-            var redirects = redirectsMap.get(path);
-            if (!redirects) {
-                // Reuse map if redirected reference map uses same resolution
-                redirects = !options || ts.optionsHaveModuleResolutionChanges(options, redirectedReference.commandLine.options) ? ts.createMap() : ownMap;
-                redirectsMap.set(path, redirects);
-            }
-            return redirects;
-        }
-        function clear() {
-            ownMap.clear();
-            redirectsMap.clear();
-        }
-    }
-    ts.createCacheWithRedirects = createCacheWithRedirects;
-    /*@internal*/
-    function createModuleResolutionCacheWithMaps(directoryToModuleNameMap, moduleNameToDirectoryMap, currentDirectory, getCanonicalFileName) {
-        return { getOrCreateCacheForDirectory: getOrCreateCacheForDirectory, getOrCreateCacheForModuleName: getOrCreateCacheForModuleName, directoryToModuleNameMap: directoryToModuleNameMap, moduleNameToDirectoryMap: moduleNameToDirectoryMap };
-        function getOrCreateCacheForDirectory(directoryName, redirectedReference) {
-            var path = ts.toPath(directoryName, currentDirectory, getCanonicalFileName);
-            return getOrCreateCache(directoryToModuleNameMap, redirectedReference, path, ts.createMap);
-        }
-        function getOrCreateCacheForModuleName(nonRelativeModuleName, redirectedReference) {
-            ts.Debug.assert(!ts.isExternalModuleNameRelative(nonRelativeModuleName));
-            return getOrCreateCache(moduleNameToDirectoryMap, redirectedReference, nonRelativeModuleName, createPerModuleNameCache);
-        }
-        function getOrCreateCache(cacheWithRedirects, redirectedReference, key, create) {
-            var cache = cacheWithRedirects.getOrCreateMapOfCacheRedirects(redirectedReference);
-            var result = cache.get(key);
-            if (!result) {
-                result = create();
-                cache.set(key, result);
-            }
-            return result;
-        }
-        function createPerModuleNameCache() {
-            var directoryPathMap = ts.createMap();
-            return { get: get, set: set };
-            function get(directory) {
-                return directoryPathMap.get(ts.toPath(directory, currentDirectory, getCanonicalFileName));
-            }
-            /**
-             * At first this function add entry directory -> module resolution result to the table.
-             * Then it computes the set of parent folders for 'directory' that should have the same module resolution result
-             * and for every parent folder in set it adds entry: parent -> module resolution. .
-             * Lets say we first directory name: /a/b/c/d/e and resolution result is: /a/b/bar.ts.
-             * Set of parent folders that should have the same result will be:
-             * [
-             *     /a/b/c/d, /a/b/c, /a/b
-             * ]
-             * this means that request for module resolution from file in any of these folder will be immediately found in cache.
-             */
-            function set(directory, result) {
-                var path = ts.toPath(directory, currentDirectory, getCanonicalFileName);
-                // if entry is already in cache do nothing
-                if (directoryPathMap.has(path)) {
-                    return;
+                function parseUnknownTag(start, tagName, indent, indentText) {
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocUnknownTag(tagName, parseTrailingTagComments(start, end, indent, indentText)), start, end);
                 }
-                directoryPathMap.set(path, result);
-                var resolvedFileName = result.resolvedModule &&
-                    (result.resolvedModule.originalPath || result.resolvedModule.resolvedFileName);
-                // find common prefix between directory and resolved file name
-                // this common prefix should be the shortest path that has the same resolution
-                // directory: /a/b/c/d/e
-                // resolvedFileName: /a/b/foo.d.ts
-                // commonPrefix: /a/b
-                // for failed lookups cache the result for every directory up to root
-                var commonPrefix = resolvedFileName && getCommonPrefix(path, resolvedFileName);
-                var current = path;
-                while (current !== commonPrefix) {
-                    var parent = ts.getDirectoryPath(current);
-                    if (parent === current || directoryPathMap.has(parent)) {
-                        break;
+                function addTag(tag) {
+                    if (!tag) {
+                        return;
                     }
-                    directoryPathMap.set(parent, result);
-                    current = parent;
+                    if (!tags) {
+                        tags = [tag];
+                        tagsPos = tag.pos;
+                    }
+                    else {
+                        tags.push(tag);
+                    }
+                    tagsEnd = tag.end;
                 }
-            }
-            function getCommonPrefix(directory, resolution) {
-                var resolutionDirectory = ts.toPath(ts.getDirectoryPath(resolution), currentDirectory, getCanonicalFileName);
-                // find first position where directory and resolution differs
-                var i = 0;
-                var limit = Math.min(directory.length, resolutionDirectory.length);
-                while (i < limit && directory.charCodeAt(i) === resolutionDirectory.charCodeAt(i)) {
-                    i++;
+                function tryParseTypeExpression() {
+                    skipWhitespaceOrAsterisk();
+                    return token() === 18 /* OpenBraceToken */ ? parseJSDocTypeExpression() : undefined;
                 }
-                if (i === directory.length && (resolutionDirectory.length === i || resolutionDirectory[i] === ts.directorySeparator)) {
-                    return directory;
+                function parseBracketNameInPropertyAndParamTag() {
+                    // Looking for something like '[foo]', 'foo', '[foo.bar]' or 'foo.bar'
+                    var isBracketed = parseOptionalJsdoc(22 /* OpenBracketToken */);
+                    if (isBracketed) {
+                        skipWhitespace();
+                    }
+                    // a markdown-quoted name: `arg` is not legal jsdoc, but occurs in the wild
+                    var isBackquoted = parseOptionalJsdoc(61 /* BacktickToken */);
+                    var name = parseJSDocEntityName();
+                    if (isBackquoted) {
+                        parseExpectedTokenJSDoc(61 /* BacktickToken */);
+                    }
+                    if (isBracketed) {
+                        skipWhitespace();
+                        // May have an optional default, e.g. '[foo = 42]'
+                        if (parseOptionalToken(62 /* EqualsToken */)) {
+                            parseExpression();
+                        }
+                        parseExpected(23 /* CloseBracketToken */);
+                    }
+                    return { name: name, isBracketed: isBracketed };
                 }
-                var rootLength = ts.getRootLength(directory);
-                if (i < rootLength) {
-                    return undefined;
+                function isObjectOrObjectArrayTypeReference(node) {
+                    switch (node.kind) {
+                        case 145 /* ObjectKeyword */:
+                            return true;
+                        case 178 /* ArrayType */:
+                            return isObjectOrObjectArrayTypeReference(node.elementType);
+                        default:
+                            return ts.isTypeReferenceNode(node) && ts.isIdentifier(node.typeName) && node.typeName.escapedText === "Object" && !node.typeArguments;
+                    }
                 }
-                var sep = directory.lastIndexOf(ts.directorySeparator, i - 1);
-                if (sep === -1) {
-                    return undefined;
+                function parseParameterOrPropertyTag(start, tagName, target, indent) {
+                    var typeExpression = tryParseTypeExpression();
+                    var isNameFirst = !typeExpression;
+                    skipWhitespaceOrAsterisk();
+                    var _a = parseBracketNameInPropertyAndParamTag(), name = _a.name, isBracketed = _a.isBracketed;
+                    skipWhitespace();
+                    if (isNameFirst) {
+                        typeExpression = tryParseTypeExpression();
+                    }
+                    var comment = parseTagComments(indent + scanner.getStartPos() - start);
+                    var nestedTypeLiteral = target !== 4 /* CallbackParameter */ && parseNestedTypeLiteral(typeExpression, name, target, indent);
+                    if (nestedTypeLiteral) {
+                        typeExpression = nestedTypeLiteral;
+                        isNameFirst = true;
+                    }
+                    var result = target === 1 /* Property */
+                        ? factory.createJSDocPropertyTag(tagName, name, isBracketed, typeExpression, isNameFirst, comment)
+                        : factory.createJSDocParameterTag(tagName, name, isBracketed, typeExpression, isNameFirst, comment);
+                    return finishNode(result, start);
                 }
-                return directory.substr(0, Math.max(sep, rootLength));
-            }
-        }
-    }
-    ts.createModuleResolutionCacheWithMaps = createModuleResolutionCacheWithMaps;
-    function resolveModuleNameFromCache(moduleName, containingFile, cache) {
-        var containingDirectory = ts.getDirectoryPath(containingFile);
-        var perFolderCache = cache && cache.getOrCreateCacheForDirectory(containingDirectory);
-        return perFolderCache && perFolderCache.get(moduleName);
-    }
-    ts.resolveModuleNameFromCache = resolveModuleNameFromCache;
-    function resolveModuleName(moduleName, containingFile, compilerOptions, host, cache, redirectedReference) {
-        var traceEnabled = isTraceEnabled(compilerOptions, host);
-        if (redirectedReference) {
-            compilerOptions = redirectedReference.commandLine.options;
-        }
-        if (traceEnabled) {
-            trace(host, ts.Diagnostics.Resolving_module_0_from_1, moduleName, containingFile);
-            if (redirectedReference) {
-                trace(host, ts.Diagnostics.Using_compiler_options_of_project_reference_redirect_0, redirectedReference.sourceFile.fileName);
-            }
-        }
-        var containingDirectory = ts.getDirectoryPath(containingFile);
-        var perFolderCache = cache && cache.getOrCreateCacheForDirectory(containingDirectory, redirectedReference);
-        var result = perFolderCache && perFolderCache.get(moduleName);
-        if (result) {
-            if (traceEnabled) {
-                trace(host, ts.Diagnostics.Resolution_for_module_0_was_found_in_cache_from_location_1, moduleName, containingDirectory);
-            }
-        }
-        else {
-            var moduleResolution = compilerOptions.moduleResolution;
-            if (moduleResolution === undefined) {
-                moduleResolution = ts.getEmitModuleKind(compilerOptions) === ts.ModuleKind.CommonJS ? ts.ModuleResolutionKind.NodeJs : ts.ModuleResolutionKind.Classic;
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Module_resolution_kind_is_not_specified_using_0, ts.ModuleResolutionKind[moduleResolution]);
+                function parseNestedTypeLiteral(typeExpression, name, target, indent) {
+                    if (typeExpression && isObjectOrObjectArrayTypeReference(typeExpression.type)) {
+                        var pos = getNodePos();
+                        var child = void 0;
+                        var children = void 0;
+                        while (child = tryParse(function () { return parseChildParameterOrPropertyTag(target, indent, name); })) {
+                            if (child.kind === 326 /* JSDocParameterTag */ || child.kind === 333 /* JSDocPropertyTag */) {
+                                children = ts.append(children, child);
+                            }
+                        }
+                        if (children) {
+                            var literal = finishNode(factory.createJSDocTypeLiteral(children, typeExpression.type.kind === 178 /* ArrayType */), pos);
+                            return finishNode(factory.createJSDocTypeExpression(literal), pos);
+                        }
+                    }
                 }
-            }
-            else {
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Explicitly_specified_module_resolution_kind_Colon_0, ts.ModuleResolutionKind[moduleResolution]);
+                function parseReturnTag(start, tagName, indent, indentText) {
+                    if (ts.some(tags, ts.isJSDocReturnTag)) {
+                        parseErrorAt(tagName.pos, scanner.getTokenPos(), ts.Diagnostics._0_tag_already_specified, tagName.escapedText);
+                    }
+                    var typeExpression = tryParseTypeExpression();
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocReturnTag(tagName, typeExpression, parseTrailingTagComments(start, end, indent, indentText)), start, end);
                 }
-            }
-            ts.perfLogger.logStartResolveModule(moduleName /* , containingFile, ModuleResolutionKind[moduleResolution]*/);
-            switch (moduleResolution) {
-                case ts.ModuleResolutionKind.NodeJs:
-                    result = nodeModuleNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference);
-                    break;
-                case ts.ModuleResolutionKind.Classic:
-                    result = classicNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference);
-                    break;
-                default:
-                    return ts.Debug.fail("Unexpected moduleResolution: " + moduleResolution);
-            }
-            if (result && result.resolvedModule)
-                ts.perfLogger.logInfoEvent("Module \"" + moduleName + "\" resolved to \"" + result.resolvedModule.resolvedFileName + "\"");
-            ts.perfLogger.logStopResolveModule((result && result.resolvedModule) ? "" + result.resolvedModule.resolvedFileName : "null");
-            if (perFolderCache) {
-                perFolderCache.set(moduleName, result);
-                if (!ts.isExternalModuleNameRelative(moduleName)) {
-                    // put result in per-module name cache
-                    cache.getOrCreateCacheForModuleName(moduleName, redirectedReference).set(containingDirectory, result);
+                function parseTypeTag(start, tagName, indent, indentText) {
+                    if (ts.some(tags, ts.isJSDocTypeTag)) {
+                        parseErrorAt(tagName.pos, scanner.getTokenPos(), ts.Diagnostics._0_tag_already_specified, tagName.escapedText);
+                    }
+                    var typeExpression = parseJSDocTypeExpression(/*mayOmitBraces*/ true);
+                    var end = getNodePos();
+                    var comments = indent !== undefined && indentText !== undefined ? parseTrailingTagComments(start, end, indent, indentText) : undefined;
+                    return finishNode(factory.createJSDocTypeTag(tagName, typeExpression, comments), start, end);
+                }
+                function parseSeeTag(start, tagName, indent, indentText) {
+                    var nameExpression = parseJSDocNameReference();
+                    var end = getNodePos();
+                    var comments = indent !== undefined && indentText !== undefined ? parseTrailingTagComments(start, end, indent, indentText) : undefined;
+                    return finishNode(factory.createJSDocSeeTag(tagName, nameExpression, comments), start, end);
+                }
+                function parseAuthorTag(start, tagName, indent, indentText) {
+                    var authorInfoWithEmail = tryParse(function () { return tryParseAuthorNameAndEmail(); });
+                    if (!authorInfoWithEmail) {
+                        var end_1 = getNodePos();
+                        return finishNode(factory.createJSDocAuthorTag(tagName, parseTrailingTagComments(start, end_1, indent, indentText)), start, end_1);
+                    }
+                    var comments = authorInfoWithEmail;
+                    if (lookAhead(function () { return nextToken() !== 4 /* NewLineTrivia */; })) {
+                        var comment = parseTagComments(indent);
+                        if (comment) {
+                            comments += comment;
+                        }
+                    }
+                    return finishNode(factory.createJSDocAuthorTag(tagName, comments), start);
+                }
+                function tryParseAuthorNameAndEmail() {
+                    var comments = [];
+                    var seenLessThan = false;
+                    var seenGreaterThan = false;
+                    var token = scanner.getToken();
+                    loop: while (true) {
+                        switch (token) {
+                            case 78 /* Identifier */:
+                            case 5 /* WhitespaceTrivia */:
+                            case 24 /* DotToken */:
+                            case 59 /* AtToken */:
+                                comments.push(scanner.getTokenText());
+                                break;
+                            case 29 /* LessThanToken */:
+                                if (seenLessThan || seenGreaterThan) {
+                                    return;
+                                }
+                                seenLessThan = true;
+                                comments.push(scanner.getTokenText());
+                                break;
+                            case 31 /* GreaterThanToken */:
+                                if (!seenLessThan || seenGreaterThan) {
+                                    return;
+                                }
+                                seenGreaterThan = true;
+                                comments.push(scanner.getTokenText());
+                                scanner.setTextPos(scanner.getTokenPos() + 1);
+                                break loop;
+                            case 4 /* NewLineTrivia */:
+                            case 1 /* EndOfFileToken */:
+                                break loop;
+                        }
+                        token = nextTokenJSDoc();
+                    }
+                    if (seenLessThan && seenGreaterThan) {
+                        return comments.length === 0 ? undefined : comments.join("");
+                    }
                 }
-            }
-        }
-        if (traceEnabled) {
-            if (result.resolvedModule) {
-                if (result.resolvedModule.packageId) {
-                    trace(host, ts.Diagnostics.Module_name_0_was_successfully_resolved_to_1_with_Package_ID_2, moduleName, result.resolvedModule.resolvedFileName, ts.packageIdToString(result.resolvedModule.packageId));
+                function parseImplementsTag(start, tagName, margin, indentText) {
+                    var className = parseExpressionWithTypeArgumentsForAugments();
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocImplementsTag(tagName, className, parseTrailingTagComments(start, end, margin, indentText)), start, end);
                 }
-                else {
-                    trace(host, ts.Diagnostics.Module_name_0_was_successfully_resolved_to_1, moduleName, result.resolvedModule.resolvedFileName);
+                function parseAugmentsTag(start, tagName, margin, indentText) {
+                    var className = parseExpressionWithTypeArgumentsForAugments();
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocAugmentsTag(tagName, className, parseTrailingTagComments(start, end, margin, indentText)), start, end);
                 }
-            }
-            else {
-                trace(host, ts.Diagnostics.Module_name_0_was_not_resolved, moduleName);
-            }
-        }
-        return result;
-    }
-    ts.resolveModuleName = resolveModuleName;
-    /**
-     * Any module resolution kind can be augmented with optional settings: 'baseUrl', 'paths' and 'rootDirs' - they are used to
-     * mitigate differences between design time structure of the project and its runtime counterpart so the same import name
-     * can be resolved successfully by TypeScript compiler and runtime module loader.
-     * If these settings are set then loading procedure will try to use them to resolve module name and it can of failure it will
-     * fallback to standard resolution routine.
-     *
-     * - baseUrl - this setting controls how non-relative module names are resolved. If this setting is specified then non-relative
-     * names will be resolved relative to baseUrl: i.e. if baseUrl is '/a/b' then candidate location to resolve module name 'c/d' will
-     * be '/a/b/c/d'
-     * - paths - this setting can only be used when baseUrl is specified. allows to tune how non-relative module names
-     * will be resolved based on the content of the module name.
-     * Structure of 'paths' compiler options
-     * 'paths': {
-     *    pattern-1: [...substitutions],
-     *    pattern-2: [...substitutions],
-     *    ...
-     *    pattern-n: [...substitutions]
-     * }
-     * Pattern here is a string that can contain zero or one '*' character. During module resolution module name will be matched against
-     * all patterns in the list. Matching for patterns that don't contain '*' means that module name must be equal to pattern respecting the case.
-     * If pattern contains '*' then to match pattern "<prefix>*<suffix>" module name must start with the <prefix> and end with <suffix>.
-     * <MatchedStar> denotes part of the module name between <prefix> and <suffix>.
-     * If module name can be matches with multiple patterns then pattern with the longest prefix will be picked.
-     * After selecting pattern we'll use list of substitutions to get candidate locations of the module and the try to load module
-     * from the candidate location.
-     * Substitution is a string that can contain zero or one '*'. To get candidate location from substitution we'll pick every
-     * substitution in the list and replace '*' with <MatchedStar> string. If candidate location is not rooted it
-     * will be converted to absolute using baseUrl.
-     * For example:
-     * baseUrl: /a/b/c
-     * "paths": {
-     *     // match all module names
-     *     "*": [
-     *         "*",        // use matched name as is,
-     *                     // <matched name> will be looked as /a/b/c/<matched name>
-     *
-     *         "folder1/*" // substitution will convert matched name to 'folder1/<matched name>',
-     *                     // since it is not rooted then final candidate location will be /a/b/c/folder1/<matched name>
-     *     ],
-     *     // match module names that start with 'components/'
-     *     "components/*": [ "/root/components/*" ] // substitution will convert /components/folder1/<matched name> to '/root/components/folder1/<matched name>',
-     *                                              // it is rooted so it will be final candidate location
-     * }
-     *
-     * 'rootDirs' allows the project to be spreaded across multiple locations and resolve modules with relative names as if
-     * they were in the same location. For example lets say there are two files
-     * '/local/src/content/file1.ts'
-     * '/shared/components/contracts/src/content/protocols/file2.ts'
-     * After bundling content of '/shared/components/contracts/src' will be merged with '/local/src' so
-     * if file1 has the following import 'import {x} from "./protocols/file2"' it will be resolved successfully in runtime.
-     * 'rootDirs' provides the way to tell compiler that in order to get the whole project it should behave as if content of all
-     * root dirs were merged together.
-     * I.e. for the example above 'rootDirs' will have two entries: [ '/local/src', '/shared/components/contracts/src' ].
-     * Compiler will first convert './protocols/file2' into absolute path relative to the location of containing file:
-     * '/local/src/content/protocols/file2' and try to load it - failure.
-     * Then it will search 'rootDirs' looking for a longest matching prefix of this absolute path and if such prefix is found - absolute path will
-     * be converted to a path relative to found rootDir entry './content/protocols/file2' (*). As a last step compiler will check all remaining
-     * entries in 'rootDirs', use them to build absolute path out of (*) and try to resolve module from this location.
-     */
-    function tryLoadModuleUsingOptionalResolutionSettings(extensions, moduleName, containingDirectory, loader, state) {
-        var resolved = tryLoadModuleUsingPathsIfEligible(extensions, moduleName, loader, state);
-        if (resolved)
-            return resolved.value;
-        if (!ts.isExternalModuleNameRelative(moduleName)) {
-            return tryLoadModuleUsingBaseUrl(extensions, moduleName, loader, state);
-        }
-        else {
-            return tryLoadModuleUsingRootDirs(extensions, moduleName, containingDirectory, loader, state);
-        }
-    }
-    function tryLoadModuleUsingPathsIfEligible(extensions, moduleName, loader, state) {
-        var _a = state.compilerOptions, baseUrl = _a.baseUrl, paths = _a.paths;
-        if (baseUrl && paths && !ts.pathIsRelative(moduleName)) {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.baseUrl_option_is_set_to_0_using_this_value_to_resolve_non_relative_module_name_1, baseUrl, moduleName);
-                trace(state.host, ts.Diagnostics.paths_option_is_specified_looking_for_a_pattern_to_match_module_name_0, moduleName);
-            }
-            return tryLoadModuleUsingPaths(extensions, moduleName, baseUrl, paths, loader, /*onlyRecordFailures*/ false, state);
-        }
-    }
-    function tryLoadModuleUsingRootDirs(extensions, moduleName, containingDirectory, loader, state) {
-        if (!state.compilerOptions.rootDirs) {
-            return undefined;
-        }
-        if (state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.rootDirs_option_is_set_using_it_to_resolve_relative_module_name_0, moduleName);
-        }
-        var candidate = ts.normalizePath(ts.combinePaths(containingDirectory, moduleName));
-        var matchedRootDir;
-        var matchedNormalizedPrefix;
-        for (var _i = 0, _a = state.compilerOptions.rootDirs; _i < _a.length; _i++) {
-            var rootDir = _a[_i];
-            // rootDirs are expected to be absolute
-            // in case of tsconfig.json this will happen automatically - compiler will expand relative names
-            // using location of tsconfig.json as base location
-            var normalizedRoot = ts.normalizePath(rootDir);
-            if (!ts.endsWith(normalizedRoot, ts.directorySeparator)) {
-                normalizedRoot += ts.directorySeparator;
-            }
-            var isLongestMatchingPrefix = ts.startsWith(candidate, normalizedRoot) &&
-                (matchedNormalizedPrefix === undefined || matchedNormalizedPrefix.length < normalizedRoot.length);
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Checking_if_0_is_the_longest_matching_prefix_for_1_2, normalizedRoot, candidate, isLongestMatchingPrefix);
-            }
-            if (isLongestMatchingPrefix) {
-                matchedNormalizedPrefix = normalizedRoot;
-                matchedRootDir = rootDir;
-            }
-        }
-        if (matchedNormalizedPrefix) {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Longest_matching_prefix_for_0_is_1, candidate, matchedNormalizedPrefix);
-            }
-            var suffix = candidate.substr(matchedNormalizedPrefix.length);
-            // first - try to load from a initial location
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Loading_0_from_the_root_dir_1_candidate_location_2, suffix, matchedNormalizedPrefix, candidate);
-            }
-            var resolvedFileName = loader(extensions, candidate, !ts.directoryProbablyExists(containingDirectory, state.host), state);
-            if (resolvedFileName) {
-                return resolvedFileName;
-            }
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Trying_other_entries_in_rootDirs);
-            }
-            // then try to resolve using remaining entries in rootDirs
-            for (var _b = 0, _c = state.compilerOptions.rootDirs; _b < _c.length; _b++) {
-                var rootDir = _c[_b];
-                if (rootDir === matchedRootDir) {
-                    // skip the initially matched entry
-                    continue;
+                function parseExpressionWithTypeArgumentsForAugments() {
+                    var usedBrace = parseOptional(18 /* OpenBraceToken */);
+                    var pos = getNodePos();
+                    var expression = parsePropertyAccessEntityNameExpression();
+                    var typeArguments = tryParseTypeArguments();
+                    var node = factory.createExpressionWithTypeArguments(expression, typeArguments);
+                    var res = finishNode(node, pos);
+                    if (usedBrace) {
+                        parseExpected(19 /* CloseBraceToken */);
+                    }
+                    return res;
                 }
-                var candidate_1 = ts.combinePaths(ts.normalizePath(rootDir), suffix);
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.Loading_0_from_the_root_dir_1_candidate_location_2, suffix, rootDir, candidate_1);
+                function parsePropertyAccessEntityNameExpression() {
+                    var pos = getNodePos();
+                    var node = parseJSDocIdentifierName();
+                    while (parseOptional(24 /* DotToken */)) {
+                        var name = parseJSDocIdentifierName();
+                        node = finishNode(factory.createPropertyAccessExpression(node, name), pos);
+                    }
+                    return node;
                 }
-                var baseDirectory = ts.getDirectoryPath(candidate_1);
-                var resolvedFileName_1 = loader(extensions, candidate_1, !ts.directoryProbablyExists(baseDirectory, state.host), state);
-                if (resolvedFileName_1) {
-                    return resolvedFileName_1;
+                function parseSimpleTag(start, createTag, tagName, margin, indentText) {
+                    var end = getNodePos();
+                    return finishNode(createTag(tagName, parseTrailingTagComments(start, end, margin, indentText)), start, end);
                 }
-            }
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Module_resolution_using_rootDirs_has_failed);
-            }
-        }
-        return undefined;
-    }
-    function tryLoadModuleUsingBaseUrl(extensions, moduleName, loader, state) {
-        var baseUrl = state.compilerOptions.baseUrl;
-        if (!baseUrl) {
-            return undefined;
-        }
-        if (state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.baseUrl_option_is_set_to_0_using_this_value_to_resolve_non_relative_module_name_1, baseUrl, moduleName);
-        }
-        var candidate = ts.normalizePath(ts.combinePaths(baseUrl, moduleName));
-        if (state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.Resolving_module_name_0_relative_to_base_url_1_2, moduleName, baseUrl, candidate);
-        }
-        return loader(extensions, candidate, !ts.directoryProbablyExists(ts.getDirectoryPath(candidate), state.host), state);
-    }
-    /**
-     * Expose resolution logic to allow us to use Node module resolution logic from arbitrary locations.
-     * No way to do this with `require()`: https://github.com/nodejs/node/issues/5963
-     * Throws an error if the module can't be resolved.
-     */
-    /* @internal */
-    function resolveJSModule(moduleName, initialDir, host) {
-        var _a = tryResolveJSModuleWorker(moduleName, initialDir, host), resolvedModule = _a.resolvedModule, failedLookupLocations = _a.failedLookupLocations;
-        if (!resolvedModule) {
-            throw new Error("Could not resolve JS module '" + moduleName + "' starting at '" + initialDir + "'. Looked in: " + failedLookupLocations.join(", "));
-        }
-        return resolvedModule.resolvedFileName;
-    }
-    ts.resolveJSModule = resolveJSModule;
-    /* @internal */
-    function tryResolveJSModule(moduleName, initialDir, host) {
-        var resolvedModule = tryResolveJSModuleWorker(moduleName, initialDir, host).resolvedModule;
-        return resolvedModule && resolvedModule.resolvedFileName;
-    }
-    ts.tryResolveJSModule = tryResolveJSModule;
-    var jsOnlyExtensions = [Extensions.JavaScript];
-    var tsExtensions = [Extensions.TypeScript, Extensions.JavaScript];
-    var tsPlusJsonExtensions = __spreadArrays(tsExtensions, [Extensions.Json]);
-    var tsconfigExtensions = [Extensions.TSConfig];
-    function tryResolveJSModuleWorker(moduleName, initialDir, host) {
-        return nodeModuleNameResolverWorker(moduleName, initialDir, { moduleResolution: ts.ModuleResolutionKind.NodeJs, allowJs: true }, host, /*cache*/ undefined, jsOnlyExtensions, /*redirectedReferences*/ undefined);
-    }
-    function nodeModuleNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference, lookupConfig) {
-        return nodeModuleNameResolverWorker(moduleName, ts.getDirectoryPath(containingFile), compilerOptions, host, cache, lookupConfig ? tsconfigExtensions : (compilerOptions.resolveJsonModule ? tsPlusJsonExtensions : tsExtensions), redirectedReference);
-    }
-    ts.nodeModuleNameResolver = nodeModuleNameResolver;
-    function nodeModuleNameResolverWorker(moduleName, containingDirectory, compilerOptions, host, cache, extensions, redirectedReference) {
-        var _a, _b;
-        var traceEnabled = isTraceEnabled(compilerOptions, host);
-        var failedLookupLocations = [];
-        var state = { compilerOptions: compilerOptions, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
-        var result = ts.forEach(extensions, function (ext) { return tryResolve(ext); });
-        return createResolvedModuleWithFailedLookupLocations((_a = result === null || result === void 0 ? void 0 : result.value) === null || _a === void 0 ? void 0 : _a.resolved, (_b = result === null || result === void 0 ? void 0 : result.value) === null || _b === void 0 ? void 0 : _b.isExternalLibraryImport, failedLookupLocations, state.resultFromCache);
-        function tryResolve(extensions) {
-            var loader = function (extensions, candidate, onlyRecordFailures, state) { return nodeLoadModuleByRelativeName(extensions, candidate, onlyRecordFailures, state, /*considerPackageJson*/ true); };
-            var resolved = tryLoadModuleUsingOptionalResolutionSettings(extensions, moduleName, containingDirectory, loader, state);
-            if (resolved) {
-                return toSearchResult({ resolved: resolved, isExternalLibraryImport: pathContainsNodeModules(resolved.path) });
-            }
-            if (!ts.isExternalModuleNameRelative(moduleName)) {
-                if (traceEnabled) {
-                    trace(host, ts.Diagnostics.Loading_module_0_from_node_modules_folder_target_file_type_1, moduleName, Extensions[extensions]);
+                function parseThisTag(start, tagName, margin, indentText) {
+                    var typeExpression = parseJSDocTypeExpression(/*mayOmitBraces*/ true);
+                    skipWhitespace();
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocThisTag(tagName, typeExpression, parseTrailingTagComments(start, end, margin, indentText)), start, end);
                 }
-                var resolved_1 = loadModuleFromNearestNodeModulesDirectory(extensions, moduleName, containingDirectory, state, cache, redirectedReference);
-                if (!resolved_1)
-                    return undefined;
-                var resolvedValue = resolved_1.value;
-                if (!compilerOptions.preserveSymlinks && resolvedValue && !resolvedValue.originalPath) {
-                    var path = realPath(resolvedValue.path, host, traceEnabled);
-                    var originalPath = path === resolvedValue.path ? undefined : resolvedValue.path;
-                    resolvedValue = __assign(__assign({}, resolvedValue), { path: path, originalPath: originalPath });
+                function parseEnumTag(start, tagName, margin, indentText) {
+                    var typeExpression = parseJSDocTypeExpression(/*mayOmitBraces*/ true);
+                    skipWhitespace();
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocEnumTag(tagName, typeExpression, parseTrailingTagComments(start, end, margin, indentText)), start, end);
                 }
-                // For node_modules lookups, get the real path so that multiple accesses to an `npm link`-ed module do not create duplicate files.
-                return { value: resolvedValue && { resolved: resolvedValue, isExternalLibraryImport: true } };
-            }
-            else {
-                var _a = ts.normalizePathAndParts(ts.combinePaths(containingDirectory, moduleName)), candidate = _a.path, parts = _a.parts;
-                var resolved_2 = nodeLoadModuleByRelativeName(extensions, candidate, /*onlyRecordFailures*/ false, state, /*considerPackageJson*/ true);
-                // Treat explicit "node_modules" import as an external library import.
-                return resolved_2 && toSearchResult({ resolved: resolved_2, isExternalLibraryImport: ts.contains(parts, "node_modules") });
-            }
-        }
-    }
-    function realPath(path, host, traceEnabled) {
-        if (!host.realpath) {
-            return path;
-        }
-        var real = ts.normalizePath(host.realpath(path));
-        if (traceEnabled) {
-            trace(host, ts.Diagnostics.Resolving_real_path_for_0_result_1, path, real);
-        }
-        ts.Debug.assert(host.fileExists(real), path + " linked to nonexistent file " + real);
-        return real;
-    }
-    function nodeLoadModuleByRelativeName(extensions, candidate, onlyRecordFailures, state, considerPackageJson) {
-        if (state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.Loading_module_as_file_Slash_folder_candidate_module_location_0_target_file_type_1, candidate, Extensions[extensions]);
-        }
-        if (!ts.hasTrailingDirectorySeparator(candidate)) {
-            if (!onlyRecordFailures) {
-                var parentOfCandidate = ts.getDirectoryPath(candidate);
-                if (!ts.directoryProbablyExists(parentOfCandidate, state.host)) {
-                    if (state.traceEnabled) {
-                        trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, parentOfCandidate);
+                function parseTypedefTag(start, tagName, indent, indentText) {
+                    var _a;
+                    var typeExpression = tryParseTypeExpression();
+                    skipWhitespaceOrAsterisk();
+                    var fullName = parseJSDocTypeNameWithNamespace();
+                    skipWhitespace();
+                    var comment = parseTagComments(indent);
+                    var end;
+                    if (!typeExpression || isObjectOrObjectArrayTypeReference(typeExpression.type)) {
+                        var child = void 0;
+                        var childTypeTag = void 0;
+                        var jsDocPropertyTags = void 0;
+                        var hasChildren = false;
+                        while (child = tryParse(function () { return parseChildPropertyTag(indent); })) {
+                            hasChildren = true;
+                            if (child.kind === 329 /* JSDocTypeTag */) {
+                                if (childTypeTag) {
+                                    parseErrorAtCurrentToken(ts.Diagnostics.A_JSDoc_typedef_comment_may_not_contain_multiple_type_tags);
+                                    var lastError = ts.lastOrUndefined(parseDiagnostics);
+                                    if (lastError) {
+                                        ts.addRelatedInfo(lastError, ts.createDetachedDiagnostic(fileName, 0, 0, ts.Diagnostics.The_tag_was_first_specified_here));
+                                    }
+                                    break;
+                                }
+                                else {
+                                    childTypeTag = child;
+                                }
+                            }
+                            else {
+                                jsDocPropertyTags = ts.append(jsDocPropertyTags, child);
+                            }
+                        }
+                        if (hasChildren) {
+                            var isArrayType = typeExpression && typeExpression.type.kind === 178 /* ArrayType */;
+                            var jsdocTypeLiteral = factory.createJSDocTypeLiteral(jsDocPropertyTags, isArrayType);
+                            typeExpression = childTypeTag && childTypeTag.typeExpression && !isObjectOrObjectArrayTypeReference(childTypeTag.typeExpression.type) ?
+                                childTypeTag.typeExpression :
+                                finishNode(jsdocTypeLiteral, start);
+                            end = typeExpression.end;
+                        }
                     }
-                    onlyRecordFailures = true;
+                    // Only include the characters between the name end and the next token if a comment was actually parsed out - otherwise it's just whitespace
+                    end = end || comment !== undefined ?
+                        getNodePos() :
+                        ((_a = fullName !== null && fullName !== void 0 ? fullName : typeExpression) !== null && _a !== void 0 ? _a : tagName).end;
+                    if (!comment) {
+                        comment = parseTrailingTagComments(start, end, indent, indentText);
+                    }
+                    var typedefTag = factory.createJSDocTypedefTag(tagName, typeExpression, fullName, comment);
+                    return finishNode(typedefTag, start, end);
                 }
-            }
-            var resolvedFromFile = loadModuleFromFile(extensions, candidate, onlyRecordFailures, state);
-            if (resolvedFromFile) {
-                var packageDirectory = considerPackageJson ? parseNodeModuleFromPath(resolvedFromFile) : undefined;
-                var packageInfo = packageDirectory ? getPackageJsonInfo(packageDirectory, /*onlyRecordFailures*/ false, state) : undefined;
-                return withPackageId(packageInfo, resolvedFromFile);
-            }
-        }
-        if (!onlyRecordFailures) {
-            var candidateExists = ts.directoryProbablyExists(candidate, state.host);
-            if (!candidateExists) {
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, candidate);
+                function parseJSDocTypeNameWithNamespace(nested) {
+                    var pos = scanner.getTokenPos();
+                    if (!ts.tokenIsIdentifierOrKeyword(token())) {
+                        return undefined;
+                    }
+                    var typeNameOrNamespaceName = parseJSDocIdentifierName();
+                    if (parseOptional(24 /* DotToken */)) {
+                        var body = parseJSDocTypeNameWithNamespace(/*nested*/ true);
+                        var jsDocNamespaceNode = factory.createModuleDeclaration(
+                        /*decorators*/ undefined, 
+                        /*modifiers*/ undefined, typeNameOrNamespaceName, body, nested ? 4 /* NestedNamespace */ : undefined);
+                        return finishNode(jsDocNamespaceNode, pos);
+                    }
+                    if (nested) {
+                        typeNameOrNamespaceName.isInJSDocNamespace = true;
+                    }
+                    return typeNameOrNamespaceName;
                 }
-                onlyRecordFailures = true;
-            }
-        }
-        return loadNodeModuleFromDirectory(extensions, candidate, onlyRecordFailures, state, considerPackageJson);
-    }
-    /*@internal*/
-    ts.nodeModulesPathPart = "/node_modules/";
-    /*@internal*/
-    function pathContainsNodeModules(path) {
-        return ts.stringContains(path, ts.nodeModulesPathPart);
-    }
-    ts.pathContainsNodeModules = pathContainsNodeModules;
-    /**
-     * This will be called on the successfully resolved path from `loadModuleFromFile`.
-     * (Not neeeded for `loadModuleFromNodeModules` as that looks up the `package.json` as part of resolution.)
-     *
-     * packageDirectory is the directory of the package itself.
-     *   For `blah/node_modules/foo/index.d.ts` this is packageDirectory: "foo"
-     *   For `/node_modules/foo/bar.d.ts` this is packageDirectory: "foo"
-     *   For `/node_modules/@types/foo/bar/index.d.ts` this is packageDirectory: "@types/foo"
-     *   For `/node_modules/foo/bar/index.d.ts` this is packageDirectory: "foo"
-     */
-    function parseNodeModuleFromPath(resolved) {
-        var path = ts.normalizePath(resolved.path);
-        var idx = path.lastIndexOf(ts.nodeModulesPathPart);
-        if (idx === -1) {
-            return undefined;
-        }
-        var indexAfterNodeModules = idx + ts.nodeModulesPathPart.length;
-        var indexAfterPackageName = moveToNextDirectorySeparatorIfAvailable(path, indexAfterNodeModules);
-        if (path.charCodeAt(indexAfterNodeModules) === 64 /* at */) {
-            indexAfterPackageName = moveToNextDirectorySeparatorIfAvailable(path, indexAfterPackageName);
-        }
-        return path.slice(0, indexAfterPackageName);
-    }
-    function moveToNextDirectorySeparatorIfAvailable(path, prevSeparatorIndex) {
-        var nextSeparatorIndex = path.indexOf(ts.directorySeparator, prevSeparatorIndex + 1);
-        return nextSeparatorIndex === -1 ? prevSeparatorIndex : nextSeparatorIndex;
-    }
-    function loadModuleFromFileNoPackageId(extensions, candidate, onlyRecordFailures, state) {
-        return noPackageId(loadModuleFromFile(extensions, candidate, onlyRecordFailures, state));
-    }
-    /**
-     * @param {boolean} onlyRecordFailures - if true then function won't try to actually load files but instead record all attempts as failures. This flag is necessary
-     * in cases when we know upfront that all load attempts will fail (because containing folder does not exists) however we still need to record all failed lookup locations.
-     */
-    function loadModuleFromFile(extensions, candidate, onlyRecordFailures, state) {
-        if (extensions === Extensions.Json || extensions === Extensions.TSConfig) {
-            var extensionLess = ts.tryRemoveExtension(candidate, ".json" /* Json */);
-            return (extensionLess === undefined && extensions === Extensions.Json) ? undefined : tryAddingExtensions(extensionLess || candidate, extensions, onlyRecordFailures, state);
-        }
-        // First, try adding an extension. An import of "foo" could be matched by a file "foo.ts", or "foo.js" by "foo.js.ts"
-        var resolvedByAddingExtension = tryAddingExtensions(candidate, extensions, onlyRecordFailures, state);
-        if (resolvedByAddingExtension) {
-            return resolvedByAddingExtension;
-        }
-        // If that didn't work, try stripping a ".js" or ".jsx" extension and replacing it with a TypeScript one;
-        // e.g. "./foo.js" can be matched by "./foo.ts" or "./foo.d.ts"
-        if (ts.hasJSFileExtension(candidate)) {
-            var extensionless = ts.removeFileExtension(candidate);
-            if (state.traceEnabled) {
-                var extension = candidate.substring(extensionless.length);
-                trace(state.host, ts.Diagnostics.File_name_0_has_a_1_extension_stripping_it, candidate, extension);
-            }
-            return tryAddingExtensions(extensionless, extensions, onlyRecordFailures, state);
-        }
-    }
-    /** Try to return an existing file that adds one of the `extensions` to `candidate`. */
-    function tryAddingExtensions(candidate, extensions, onlyRecordFailures, state) {
-        if (!onlyRecordFailures) {
-            // check if containing folder exists - if it doesn't then just record failures for all supported extensions without disk probing
-            var directory = ts.getDirectoryPath(candidate);
-            if (directory) {
-                onlyRecordFailures = !ts.directoryProbablyExists(directory, state.host);
-            }
-        }
-        switch (extensions) {
-            case Extensions.DtsOnly:
-                return tryExtension(".d.ts" /* Dts */);
-            case Extensions.TypeScript:
-                return tryExtension(".ts" /* Ts */) || tryExtension(".tsx" /* Tsx */) || tryExtension(".d.ts" /* Dts */);
-            case Extensions.JavaScript:
-                return tryExtension(".js" /* Js */) || tryExtension(".jsx" /* Jsx */);
-            case Extensions.TSConfig:
-            case Extensions.Json:
-                return tryExtension(".json" /* Json */);
-        }
-        function tryExtension(ext) {
-            var path = tryFile(candidate + ext, onlyRecordFailures, state);
-            return path === undefined ? undefined : { path: path, ext: ext };
-        }
-    }
-    /** Return the file if it exists. */
-    function tryFile(fileName, onlyRecordFailures, state) {
-        if (!onlyRecordFailures) {
-            if (state.host.fileExists(fileName)) {
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.File_0_exist_use_it_as_a_name_resolution_result, fileName);
+                function parseCallbackTagParameters(indent) {
+                    var pos = getNodePos();
+                    var child;
+                    var parameters;
+                    while (child = tryParse(function () { return parseChildParameterOrPropertyTag(4 /* CallbackParameter */, indent); })) {
+                        parameters = ts.append(parameters, child);
+                    }
+                    return createNodeArray(parameters || [], pos);
                 }
-                return fileName;
-            }
-            else {
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.File_0_does_not_exist, fileName);
+                function parseCallbackTag(start, tagName, indent, indentText) {
+                    var fullName = parseJSDocTypeNameWithNamespace();
+                    skipWhitespace();
+                    var comment = parseTagComments(indent);
+                    var parameters = parseCallbackTagParameters(indent);
+                    var returnTag = tryParse(function () {
+                        if (parseOptionalJsdoc(59 /* AtToken */)) {
+                            var tag = parseTag(indent);
+                            if (tag && tag.kind === 327 /* JSDocReturnTag */) {
+                                return tag;
+                            }
+                        }
+                    });
+                    var typeExpression = finishNode(factory.createJSDocSignature(/*typeParameters*/ undefined, parameters, returnTag), start);
+                    var end = getNodePos();
+                    if (!comment) {
+                        comment = parseTrailingTagComments(start, end, indent, indentText);
+                    }
+                    return finishNode(factory.createJSDocCallbackTag(tagName, typeExpression, fullName, comment), start, end);
                 }
-            }
-        }
-        state.failedLookupLocations.push(fileName);
-        return undefined;
-    }
-    function loadNodeModuleFromDirectory(extensions, candidate, onlyRecordFailures, state, considerPackageJson) {
-        if (considerPackageJson === void 0) { considerPackageJson = true; }
-        var packageInfo = considerPackageJson ? getPackageJsonInfo(candidate, onlyRecordFailures, state) : undefined;
-        var packageJsonContent = packageInfo && packageInfo.packageJsonContent;
-        var versionPaths = packageInfo && packageInfo.versionPaths;
-        return withPackageId(packageInfo, loadNodeModuleFromDirectoryWorker(extensions, candidate, onlyRecordFailures, state, packageJsonContent, versionPaths));
-    }
-    function getPackageJsonInfo(packageDirectory, onlyRecordFailures, state) {
-        var host = state.host, traceEnabled = state.traceEnabled;
-        var directoryExists = !onlyRecordFailures && ts.directoryProbablyExists(packageDirectory, host);
-        var packageJsonPath = ts.combinePaths(packageDirectory, "package.json");
-        if (directoryExists && host.fileExists(packageJsonPath)) {
-            var packageJsonContent = ts.readJson(packageJsonPath, host);
-            if (traceEnabled) {
-                trace(host, ts.Diagnostics.Found_package_json_at_0, packageJsonPath);
-            }
-            var versionPaths = readPackageJsonTypesVersionPaths(packageJsonContent, state);
-            return { packageDirectory: packageDirectory, packageJsonContent: packageJsonContent, versionPaths: versionPaths };
-        }
-        else {
-            if (directoryExists && traceEnabled) {
-                trace(host, ts.Diagnostics.File_0_does_not_exist, packageJsonPath);
-            }
-            // record package json as one of failed lookup locations - in the future if this file will appear it will invalidate resolution results
-            state.failedLookupLocations.push(packageJsonPath);
-        }
-    }
-    function loadNodeModuleFromDirectoryWorker(extensions, candidate, onlyRecordFailures, state, jsonContent, versionPaths) {
-        var packageFile;
-        if (jsonContent) {
-            switch (extensions) {
-                case Extensions.JavaScript:
-                case Extensions.Json:
-                    packageFile = readPackageJsonMainField(jsonContent, candidate, state);
-                    break;
-                case Extensions.TypeScript:
-                    // When resolving typescript modules, try resolving using main field as well
-                    packageFile = readPackageJsonTypesFields(jsonContent, candidate, state) || readPackageJsonMainField(jsonContent, candidate, state);
-                    break;
-                case Extensions.DtsOnly:
-                    packageFile = readPackageJsonTypesFields(jsonContent, candidate, state);
-                    break;
-                case Extensions.TSConfig:
-                    packageFile = readPackageJsonTSConfigField(jsonContent, candidate, state);
-                    break;
-                default:
-                    return ts.Debug.assertNever(extensions);
-            }
-        }
-        var loader = function (extensions, candidate, onlyRecordFailures, state) {
-            var fromFile = tryFile(candidate, onlyRecordFailures, state);
-            if (fromFile) {
-                var resolved = resolvedIfExtensionMatches(extensions, fromFile);
-                if (resolved) {
-                    return noPackageId(resolved);
+                function escapedTextsEqual(a, b) {
+                    while (!ts.isIdentifier(a) || !ts.isIdentifier(b)) {
+                        if (!ts.isIdentifier(a) && !ts.isIdentifier(b) && a.right.escapedText === b.right.escapedText) {
+                            a = a.left;
+                            b = b.left;
+                        }
+                        else {
+                            return false;
+                        }
+                    }
+                    return a.escapedText === b.escapedText;
                 }
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.File_0_has_an_unsupported_extension_so_skipping_it, fromFile);
+                function parseChildPropertyTag(indent) {
+                    return parseChildParameterOrPropertyTag(1 /* Property */, indent);
                 }
-            }
-            // Even if extensions is DtsOnly, we can still look up a .ts file as a result of package.json "types"
-            var nextExtensions = extensions === Extensions.DtsOnly ? Extensions.TypeScript : extensions;
-            // Don't do package.json lookup recursively, because Node.js' package lookup doesn't.
-            return nodeLoadModuleByRelativeName(nextExtensions, candidate, onlyRecordFailures, state, /*considerPackageJson*/ false);
-        };
-        var onlyRecordFailuresForPackageFile = packageFile ? !ts.directoryProbablyExists(ts.getDirectoryPath(packageFile), state.host) : undefined;
-        var onlyRecordFailuresForIndex = onlyRecordFailures || !ts.directoryProbablyExists(candidate, state.host);
-        var indexPath = ts.combinePaths(candidate, extensions === Extensions.TSConfig ? "tsconfig" : "index");
-        if (versionPaths && (!packageFile || ts.containsPath(candidate, packageFile))) {
-            var moduleName = ts.getRelativePathFromDirectory(candidate, packageFile || indexPath, /*ignoreCase*/ false);
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_entry_0_that_matches_compiler_version_1_looking_for_a_pattern_to_match_module_name_2, versionPaths.version, ts.version, moduleName);
-            }
-            var result = tryLoadModuleUsingPaths(extensions, moduleName, candidate, versionPaths.paths, loader, onlyRecordFailuresForPackageFile || onlyRecordFailuresForIndex, state);
-            if (result) {
-                return removeIgnoredPackageId(result.value);
-            }
-        }
-        // It won't have a `packageId` set, because we disabled `considerPackageJson`.
-        var packageFileResult = packageFile && removeIgnoredPackageId(loader(extensions, packageFile, onlyRecordFailuresForPackageFile, state));
-        if (packageFileResult)
-            return packageFileResult;
-        return loadModuleFromFile(extensions, indexPath, onlyRecordFailuresForIndex, state);
-    }
-    /** Resolve from an arbitrarily specified file. Return `undefined` if it has an unsupported extension. */
-    function resolvedIfExtensionMatches(extensions, path) {
-        var ext = ts.tryGetExtensionFromPath(path);
-        return ext !== undefined && extensionIsOk(extensions, ext) ? { path: path, ext: ext } : undefined;
-    }
-    /** True if `extension` is one of the supported `extensions`. */
-    function extensionIsOk(extensions, extension) {
-        switch (extensions) {
-            case Extensions.JavaScript:
-                return extension === ".js" /* Js */ || extension === ".jsx" /* Jsx */;
-            case Extensions.TSConfig:
-            case Extensions.Json:
-                return extension === ".json" /* Json */;
-            case Extensions.TypeScript:
-                return extension === ".ts" /* Ts */ || extension === ".tsx" /* Tsx */ || extension === ".d.ts" /* Dts */;
-            case Extensions.DtsOnly:
-                return extension === ".d.ts" /* Dts */;
-        }
-    }
-    /* @internal */
-    function parsePackageName(moduleName) {
-        var idx = moduleName.indexOf(ts.directorySeparator);
-        if (moduleName[0] === "@") {
-            idx = moduleName.indexOf(ts.directorySeparator, idx + 1);
-        }
-        return idx === -1 ? { packageName: moduleName, rest: "" } : { packageName: moduleName.slice(0, idx), rest: moduleName.slice(idx + 1) };
-    }
-    ts.parsePackageName = parsePackageName;
-    function loadModuleFromNearestNodeModulesDirectory(extensions, moduleName, directory, state, cache, redirectedReference) {
-        return loadModuleFromNearestNodeModulesDirectoryWorker(extensions, moduleName, directory, state, /*typesScopeOnly*/ false, cache, redirectedReference);
-    }
-    function loadModuleFromNearestNodeModulesDirectoryTypesScope(moduleName, directory, state) {
-        // Extensions parameter here doesn't actually matter, because typesOnly ensures we're just doing @types lookup, which is always DtsOnly.
-        return loadModuleFromNearestNodeModulesDirectoryWorker(Extensions.DtsOnly, moduleName, directory, state, /*typesScopeOnly*/ true, /*cache*/ undefined, /*redirectedReference*/ undefined);
-    }
-    function loadModuleFromNearestNodeModulesDirectoryWorker(extensions, moduleName, directory, state, typesScopeOnly, cache, redirectedReference) {
-        var perModuleNameCache = cache && cache.getOrCreateCacheForModuleName(moduleName, redirectedReference);
-        return ts.forEachAncestorDirectory(ts.normalizeSlashes(directory), function (ancestorDirectory) {
-            if (ts.getBaseFileName(ancestorDirectory) !== "node_modules") {
-                var resolutionFromCache = tryFindNonRelativeModuleNameInCache(perModuleNameCache, moduleName, ancestorDirectory, state);
-                if (resolutionFromCache) {
-                    return resolutionFromCache;
+                function parseChildParameterOrPropertyTag(target, indent, name) {
+                    var canParseTag = true;
+                    var seenAsterisk = false;
+                    while (true) {
+                        switch (nextTokenJSDoc()) {
+                            case 59 /* AtToken */:
+                                if (canParseTag) {
+                                    var child = tryParseChildTag(target, indent);
+                                    if (child && (child.kind === 326 /* JSDocParameterTag */ || child.kind === 333 /* JSDocPropertyTag */) &&
+                                        target !== 4 /* CallbackParameter */ &&
+                                        name && (ts.isIdentifier(child.name) || !escapedTextsEqual(name, child.name.left))) {
+                                        return false;
+                                    }
+                                    return child;
+                                }
+                                seenAsterisk = false;
+                                break;
+                            case 4 /* NewLineTrivia */:
+                                canParseTag = true;
+                                seenAsterisk = false;
+                                break;
+                            case 41 /* AsteriskToken */:
+                                if (seenAsterisk) {
+                                    canParseTag = false;
+                                }
+                                seenAsterisk = true;
+                                break;
+                            case 78 /* Identifier */:
+                                canParseTag = false;
+                                break;
+                            case 1 /* EndOfFileToken */:
+                                return false;
+                        }
+                    }
                 }
-                return toSearchResult(loadModuleFromImmediateNodeModulesDirectory(extensions, moduleName, ancestorDirectory, state, typesScopeOnly));
-            }
-        });
-    }
-    function loadModuleFromImmediateNodeModulesDirectory(extensions, moduleName, directory, state, typesScopeOnly) {
-        var nodeModulesFolder = ts.combinePaths(directory, "node_modules");
-        var nodeModulesFolderExists = ts.directoryProbablyExists(nodeModulesFolder, state.host);
-        if (!nodeModulesFolderExists && state.traceEnabled) {
-            trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, nodeModulesFolder);
-        }
-        var packageResult = typesScopeOnly ? undefined : loadModuleFromSpecificNodeModulesDirectory(extensions, moduleName, nodeModulesFolder, nodeModulesFolderExists, state);
-        if (packageResult) {
-            return packageResult;
-        }
-        if (extensions === Extensions.TypeScript || extensions === Extensions.DtsOnly) {
-            var nodeModulesAtTypes_1 = ts.combinePaths(nodeModulesFolder, "@types");
-            var nodeModulesAtTypesExists = nodeModulesFolderExists;
-            if (nodeModulesFolderExists && !ts.directoryProbablyExists(nodeModulesAtTypes_1, state.host)) {
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, nodeModulesAtTypes_1);
+                function tryParseChildTag(target, indent) {
+                    ts.Debug.assert(token() === 59 /* AtToken */);
+                    var start = scanner.getStartPos();
+                    nextTokenJSDoc();
+                    var tagName = parseJSDocIdentifierName();
+                    skipWhitespace();
+                    var t;
+                    switch (tagName.escapedText) {
+                        case "type":
+                            return target === 1 /* Property */ && parseTypeTag(start, tagName);
+                        case "prop":
+                        case "property":
+                            t = 1 /* Property */;
+                            break;
+                        case "arg":
+                        case "argument":
+                        case "param":
+                            t = 2 /* Parameter */ | 4 /* CallbackParameter */;
+                            break;
+                        default:
+                            return false;
+                    }
+                    if (!(target & t)) {
+                        return false;
+                    }
+                    return parseParameterOrPropertyTag(start, tagName, target, indent);
                 }
-                nodeModulesAtTypesExists = false;
-            }
-            return loadModuleFromSpecificNodeModulesDirectory(Extensions.DtsOnly, mangleScopedPackageNameWithTrace(moduleName, state), nodeModulesAtTypes_1, nodeModulesAtTypesExists, state);
-        }
-    }
-    function loadModuleFromSpecificNodeModulesDirectory(extensions, moduleName, nodeModulesDirectory, nodeModulesDirectoryExists, state) {
-        var candidate = ts.normalizePath(ts.combinePaths(nodeModulesDirectory, moduleName));
-        // First look for a nested package.json, as in `node_modules/foo/bar/package.json`.
-        var packageInfo = getPackageJsonInfo(candidate, !nodeModulesDirectoryExists, state);
-        if (packageInfo) {
-            var fromFile = loadModuleFromFile(extensions, candidate, !nodeModulesDirectoryExists, state);
-            if (fromFile) {
-                return noPackageId(fromFile);
-            }
-            var fromDirectory = loadNodeModuleFromDirectoryWorker(extensions, candidate, !nodeModulesDirectoryExists, state, packageInfo.packageJsonContent, packageInfo.versionPaths);
-            return withPackageId(packageInfo, fromDirectory);
-        }
-        var loader = function (extensions, candidate, onlyRecordFailures, state) {
-            var pathAndExtension = loadModuleFromFile(extensions, candidate, onlyRecordFailures, state) ||
-                loadNodeModuleFromDirectoryWorker(extensions, candidate, onlyRecordFailures, state, packageInfo && packageInfo.packageJsonContent, packageInfo && packageInfo.versionPaths);
-            return withPackageId(packageInfo, pathAndExtension);
-        };
-        var _a = parsePackageName(moduleName), packageName = _a.packageName, rest = _a.rest;
-        if (rest !== "") { // If "rest" is empty, we just did this search above.
-            var packageDirectory = ts.combinePaths(nodeModulesDirectory, packageName);
-            // Don't use a "types" or "main" from here because we're not loading the root, but a subdirectory -- just here for the packageId and path mappings.
-            packageInfo = getPackageJsonInfo(packageDirectory, !nodeModulesDirectoryExists, state);
-            if (packageInfo && packageInfo.versionPaths) {
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_entry_0_that_matches_compiler_version_1_looking_for_a_pattern_to_match_module_name_2, packageInfo.versionPaths.version, ts.version, rest);
+                function parseTemplateTagTypeParameter() {
+                    var typeParameterPos = getNodePos();
+                    var name = parseJSDocIdentifierName(ts.Diagnostics.Unexpected_token_A_type_parameter_name_was_expected_without_curly_braces);
+                    return finishNode(factory.createTypeParameterDeclaration(name, /*constraint*/ undefined, /*defaultType*/ undefined), typeParameterPos);
                 }
-                var packageDirectoryExists = nodeModulesDirectoryExists && ts.directoryProbablyExists(packageDirectory, state.host);
-                var fromPaths = tryLoadModuleUsingPaths(extensions, rest, packageDirectory, packageInfo.versionPaths.paths, loader, !packageDirectoryExists, state);
-                if (fromPaths) {
-                    return fromPaths.value;
+                function parseTemplateTagTypeParameters() {
+                    var pos = getNodePos();
+                    var typeParameters = [];
+                    do {
+                        skipWhitespace();
+                        typeParameters.push(parseTemplateTagTypeParameter());
+                        skipWhitespaceOrAsterisk();
+                    } while (parseOptionalJsdoc(27 /* CommaToken */));
+                    return createNodeArray(typeParameters, pos);
                 }
-            }
-        }
-        return loader(extensions, candidate, !nodeModulesDirectoryExists, state);
-    }
-    function tryLoadModuleUsingPaths(extensions, moduleName, baseDirectory, paths, loader, onlyRecordFailures, state) {
-        var matchedPattern = ts.matchPatternOrExact(ts.getOwnKeys(paths), moduleName);
-        if (matchedPattern) {
-            var matchedStar_1 = ts.isString(matchedPattern) ? undefined : ts.matchedText(matchedPattern, moduleName);
-            var matchedPatternText = ts.isString(matchedPattern) ? matchedPattern : ts.patternText(matchedPattern);
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Module_name_0_matched_pattern_1, moduleName, matchedPatternText);
-            }
-            var resolved = ts.forEach(paths[matchedPatternText], function (subst) {
-                var path = matchedStar_1 ? subst.replace("*", matchedStar_1) : subst;
-                var candidate = ts.normalizePath(ts.combinePaths(baseDirectory, path));
-                if (state.traceEnabled) {
-                    trace(state.host, ts.Diagnostics.Trying_substitution_0_candidate_module_location_Colon_1, subst, path);
+                function parseTemplateTag(start, tagName, indent, indentText) {
+                    // The template tag looks like one of the following:
+                    //   @template T,U,V
+                    //   @template {Constraint} T
+                    //
+                    // According to the [closure docs](https://github.com/google/closure-compiler/wiki/Generic-Types#multiple-bounded-template-types):
+                    //   > Multiple bounded generics cannot be declared on the same line. For the sake of clarity, if multiple templates share the same
+                    //   > type bound they must be declared on separate lines.
+                    //
+                    // TODO: Determine whether we should enforce this in the checker.
+                    // TODO: Consider moving the `constraint` to the first type parameter as we could then remove `getEffectiveConstraintOfTypeParameter`.
+                    // TODO: Consider only parsing a single type parameter if there is a constraint.
+                    var constraint = token() === 18 /* OpenBraceToken */ ? parseJSDocTypeExpression() : undefined;
+                    var typeParameters = parseTemplateTagTypeParameters();
+                    var end = getNodePos();
+                    return finishNode(factory.createJSDocTemplateTag(tagName, constraint, typeParameters, parseTrailingTagComments(start, end, indent, indentText)), start, end);
                 }
-                // A path mapping may have an extension, in contrast to an import, which should omit it.
-                var extension = ts.tryGetExtensionFromPath(candidate);
-                if (extension !== undefined) {
-                    var path_1 = tryFile(candidate, onlyRecordFailures, state);
-                    if (path_1 !== undefined) {
-                        return noPackageId({ path: path_1, ext: extension });
+                function parseOptionalJsdoc(t) {
+                    if (token() === t) {
+                        nextTokenJSDoc();
+                        return true;
                     }
+                    return false;
                 }
-                return loader(extensions, candidate, onlyRecordFailures || !ts.directoryProbablyExists(ts.getDirectoryPath(candidate), state.host), state);
-            });
-            return { value: resolved };
-        }
-    }
-    /** Double underscores are used in DefinitelyTyped to delimit scoped packages. */
-    var mangledScopedPackageSeparator = "__";
-    /** For a scoped package, we must look in `@types/foo__bar` instead of `@types/@foo/bar`. */
-    function mangleScopedPackageNameWithTrace(packageName, state) {
-        var mangled = mangleScopedPackageName(packageName);
-        if (state.traceEnabled && mangled !== packageName) {
-            trace(state.host, ts.Diagnostics.Scoped_package_detected_looking_in_0, mangled);
-        }
-        return mangled;
-    }
-    /* @internal */
-    function getTypesPackageName(packageName) {
-        return "@types/" + mangleScopedPackageName(packageName);
-    }
-    ts.getTypesPackageName = getTypesPackageName;
-    /* @internal */
-    function mangleScopedPackageName(packageName) {
-        if (ts.startsWith(packageName, "@")) {
-            var replaceSlash = packageName.replace(ts.directorySeparator, mangledScopedPackageSeparator);
-            if (replaceSlash !== packageName) {
-                return replaceSlash.slice(1); // Take off the "@"
-            }
-        }
-        return packageName;
-    }
-    ts.mangleScopedPackageName = mangleScopedPackageName;
-    /* @internal */
-    function getPackageNameFromTypesPackageName(mangledName) {
-        var withoutAtTypePrefix = ts.removePrefix(mangledName, "@types/");
-        if (withoutAtTypePrefix !== mangledName) {
-            return unmangleScopedPackageName(withoutAtTypePrefix);
-        }
-        return mangledName;
-    }
-    ts.getPackageNameFromTypesPackageName = getPackageNameFromTypesPackageName;
-    /* @internal */
-    function unmangleScopedPackageName(typesPackageName) {
-        return ts.stringContains(typesPackageName, mangledScopedPackageSeparator) ?
-            "@" + typesPackageName.replace(mangledScopedPackageSeparator, ts.directorySeparator) :
-            typesPackageName;
-    }
-    ts.unmangleScopedPackageName = unmangleScopedPackageName;
-    function tryFindNonRelativeModuleNameInCache(cache, moduleName, containingDirectory, state) {
-        var result = cache && cache.get(containingDirectory);
-        if (result) {
-            if (state.traceEnabled) {
-                trace(state.host, ts.Diagnostics.Resolution_for_module_0_was_found_in_cache_from_location_1, moduleName, containingDirectory);
-            }
-            state.resultFromCache = result;
-            return { value: result.resolvedModule && { path: result.resolvedModule.resolvedFileName, originalPath: result.resolvedModule.originalPath || true, extension: result.resolvedModule.extension, packageId: result.resolvedModule.packageId } };
-        }
-    }
-    function classicNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference) {
-        var traceEnabled = isTraceEnabled(compilerOptions, host);
-        var failedLookupLocations = [];
-        var state = { compilerOptions: compilerOptions, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
-        var containingDirectory = ts.getDirectoryPath(containingFile);
-        var resolved = tryResolve(Extensions.TypeScript) || tryResolve(Extensions.JavaScript);
-        // No originalPath because classic resolution doesn't resolve realPath
-        return createResolvedModuleWithFailedLookupLocations(resolved && resolved.value, /*isExternalLibraryImport*/ false, failedLookupLocations, state.resultFromCache);
-        function tryResolve(extensions) {
-            var resolvedUsingSettings = tryLoadModuleUsingOptionalResolutionSettings(extensions, moduleName, containingDirectory, loadModuleFromFileNoPackageId, state);
-            if (resolvedUsingSettings) {
-                return { value: resolvedUsingSettings };
-            }
-            if (!ts.isExternalModuleNameRelative(moduleName)) {
-                var perModuleNameCache_1 = cache && cache.getOrCreateCacheForModuleName(moduleName, redirectedReference);
-                // Climb up parent directories looking for a module.
-                var resolved_3 = ts.forEachAncestorDirectory(containingDirectory, function (directory) {
-                    var resolutionFromCache = tryFindNonRelativeModuleNameInCache(perModuleNameCache_1, moduleName, directory, state);
-                    if (resolutionFromCache) {
-                        return resolutionFromCache;
+                function parseJSDocEntityName() {
+                    var entity = parseJSDocIdentifierName();
+                    if (parseOptional(22 /* OpenBracketToken */)) {
+                        parseExpected(23 /* CloseBracketToken */);
+                        // Note that y[] is accepted as an entity name, but the postfix brackets are not saved for checking.
+                        // Technically usejsdoc.org requires them for specifying a property of a type equivalent to Array<{ x: ...}>
+                        // but it's not worth it to enforce that restriction.
                     }
-                    var searchName = ts.normalizePath(ts.combinePaths(directory, moduleName));
-                    return toSearchResult(loadModuleFromFileNoPackageId(extensions, searchName, /*onlyRecordFailures*/ false, state));
-                });
-                if (resolved_3) {
-                    return resolved_3;
+                    while (parseOptional(24 /* DotToken */)) {
+                        var name = parseJSDocIdentifierName();
+                        if (parseOptional(22 /* OpenBracketToken */)) {
+                            parseExpected(23 /* CloseBracketToken */);
+                        }
+                        entity = createQualifiedName(entity, name);
+                    }
+                    return entity;
                 }
-                if (extensions === Extensions.TypeScript) {
-                    // If we didn't find the file normally, look it up in @types.
-                    return loadModuleFromNearestNodeModulesDirectoryTypesScope(moduleName, containingDirectory, state);
+                function parseJSDocIdentifierName(message) {
+                    if (!ts.tokenIsIdentifierOrKeyword(token())) {
+                        return createMissingNode(78 /* Identifier */, /*reportAtCurrentPosition*/ !message, message || ts.Diagnostics.Identifier_expected);
+                    }
+                    identifierCount++;
+                    var pos = scanner.getTokenPos();
+                    var end = scanner.getTextPos();
+                    var originalKeywordKind = token();
+                    var text = internIdentifier(scanner.getTokenValue());
+                    var result = finishNode(factory.createIdentifier(text, /*typeArguments*/ undefined, originalKeywordKind), pos, end);
+                    nextTokenJSDoc();
+                    return result;
                 }
             }
-            else {
-                var candidate = ts.normalizePath(ts.combinePaths(containingDirectory, moduleName));
-                return toSearchResult(loadModuleFromFileNoPackageId(extensions, candidate, /*onlyRecordFailures*/ false, state));
+        })(JSDocParser = Parser.JSDocParser || (Parser.JSDocParser = {}));
+    })(Parser || (Parser = {}));
+    var IncrementalParser;
+    (function (IncrementalParser) {
+        function updateSourceFile(sourceFile, newText, textChangeRange, aggressiveChecks) {
+            aggressiveChecks = aggressiveChecks || ts.Debug.shouldAssert(2 /* Aggressive */);
+            checkChangeRange(sourceFile, newText, textChangeRange, aggressiveChecks);
+            if (ts.textChangeRangeIsUnchanged(textChangeRange)) {
+                // if the text didn't change, then we can just return our current source file as-is.
+                return sourceFile;
             }
+            if (sourceFile.statements.length === 0) {
+                // If we don't have any statements in the current source file, then there's no real
+                // way to incrementally parse.  So just do a full parse instead.
+                return Parser.parseSourceFile(sourceFile.fileName, newText, sourceFile.languageVersion, /*syntaxCursor*/ undefined, /*setParentNodes*/ true, sourceFile.scriptKind);
+            }
+            // Make sure we're not trying to incrementally update a source file more than once.  Once
+            // we do an update the original source file is considered unusable from that point onwards.
+            //
+            // This is because we do incremental parsing in-place.  i.e. we take nodes from the old
+            // tree and give them new positions and parents.  From that point on, trusting the old
+            // tree at all is not possible as far too much of it may violate invariants.
+            var incrementalSourceFile = sourceFile;
+            ts.Debug.assert(!incrementalSourceFile.hasBeenIncrementallyParsed);
+            incrementalSourceFile.hasBeenIncrementallyParsed = true;
+            Parser.fixupParentReferences(incrementalSourceFile);
+            var oldText = sourceFile.text;
+            var syntaxCursor = createSyntaxCursor(sourceFile);
+            // Make the actual change larger so that we know to reparse anything whose lookahead
+            // might have intersected the change.
+            var changeRange = extendToAffectedRange(sourceFile, textChangeRange);
+            checkChangeRange(sourceFile, newText, changeRange, aggressiveChecks);
+            // Ensure that extending the affected range only moved the start of the change range
+            // earlier in the file.
+            ts.Debug.assert(changeRange.span.start <= textChangeRange.span.start);
+            ts.Debug.assert(ts.textSpanEnd(changeRange.span) === ts.textSpanEnd(textChangeRange.span));
+            ts.Debug.assert(ts.textSpanEnd(ts.textChangeRangeNewSpan(changeRange)) === ts.textSpanEnd(ts.textChangeRangeNewSpan(textChangeRange)));
+            // The is the amount the nodes after the edit range need to be adjusted.  It can be
+            // positive (if the edit added characters), negative (if the edit deleted characters)
+            // or zero (if this was a pure overwrite with nothing added/removed).
+            var delta = ts.textChangeRangeNewSpan(changeRange).length - changeRange.span.length;
+            // If we added or removed characters during the edit, then we need to go and adjust all
+            // the nodes after the edit.  Those nodes may move forward (if we inserted chars) or they
+            // may move backward (if we deleted chars).
+            //
+            // Doing this helps us out in two ways.  First, it means that any nodes/tokens we want
+            // to reuse are already at the appropriate position in the new text.  That way when we
+            // reuse them, we don't have to figure out if they need to be adjusted.  Second, it makes
+            // it very easy to determine if we can reuse a node.  If the node's position is at where
+            // we are in the text, then we can reuse it.  Otherwise we can't.  If the node's position
+            // is ahead of us, then we'll need to rescan tokens.  If the node's position is behind
+            // us, then we'll need to skip it or crumble it as appropriate
+            //
+            // We will also adjust the positions of nodes that intersect the change range as well.
+            // By doing this, we ensure that all the positions in the old tree are consistent, not
+            // just the positions of nodes entirely before/after the change range.  By being
+            // consistent, we can then easily map from positions to nodes in the old tree easily.
+            //
+            // Also, mark any syntax elements that intersect the changed span.  We know, up front,
+            // that we cannot reuse these elements.
+            updateTokenPositionsAndMarkElements(incrementalSourceFile, changeRange.span.start, ts.textSpanEnd(changeRange.span), ts.textSpanEnd(ts.textChangeRangeNewSpan(changeRange)), delta, oldText, newText, aggressiveChecks);
+            // Now that we've set up our internal incremental state just proceed and parse the
+            // source file in the normal fashion.  When possible the parser will retrieve and
+            // reuse nodes from the old tree.
+            //
+            // Note: passing in 'true' for setNodeParents is very important.  When incrementally
+            // parsing, we will be reusing nodes from the old tree, and placing it into new
+            // parents.  If we don't set the parents now, we'll end up with an observably
+            // inconsistent tree.  Setting the parents on the new tree should be very fast.  We
+            // will immediately bail out of walking any subtrees when we can see that their parents
+            // are already correct.
+            var result = Parser.parseSourceFile(sourceFile.fileName, newText, sourceFile.languageVersion, syntaxCursor, /*setParentNodes*/ true, sourceFile.scriptKind);
+            result.commentDirectives = getNewCommentDirectives(sourceFile.commentDirectives, result.commentDirectives, changeRange.span.start, ts.textSpanEnd(changeRange.span), delta, oldText, newText, aggressiveChecks);
+            return result;
         }
-    }
-    ts.classicNameResolver = classicNameResolver;
-    /**
-     * A host may load a module from a global cache of typings.
-     * This is the minumum code needed to expose that functionality; the rest is in the host.
-     */
-    /* @internal */
-    function loadModuleFromGlobalCache(moduleName, projectName, compilerOptions, host, globalCache) {
-        var traceEnabled = isTraceEnabled(compilerOptions, host);
-        if (traceEnabled) {
-            trace(host, ts.Diagnostics.Auto_discovery_for_typings_is_enabled_in_project_0_Running_extra_resolution_pass_for_module_1_using_cache_location_2, projectName, moduleName, globalCache);
-        }
-        var failedLookupLocations = [];
-        var state = { compilerOptions: compilerOptions, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
-        var resolved = loadModuleFromImmediateNodeModulesDirectory(Extensions.DtsOnly, moduleName, globalCache, state, /*typesScopeOnly*/ false);
-        return createResolvedModuleWithFailedLookupLocations(resolved, /*isExternalLibraryImport*/ true, failedLookupLocations, state.resultFromCache);
-    }
-    ts.loadModuleFromGlobalCache = loadModuleFromGlobalCache;
-    /**
-     * Wraps value to SearchResult.
-     * @returns undefined if value is undefined or { value } otherwise
-     */
-    function toSearchResult(value) {
-        return value !== undefined ? { value: value } : undefined;
-    }
-})(ts || (ts = {}));
-/* @internal */
-var ts;
-(function (ts) {
-    var ModuleInstanceState;
-    (function (ModuleInstanceState) {
-        ModuleInstanceState[ModuleInstanceState["NonInstantiated"] = 0] = "NonInstantiated";
-        ModuleInstanceState[ModuleInstanceState["Instantiated"] = 1] = "Instantiated";
-        ModuleInstanceState[ModuleInstanceState["ConstEnumOnly"] = 2] = "ConstEnumOnly";
-    })(ModuleInstanceState = ts.ModuleInstanceState || (ts.ModuleInstanceState = {}));
-    function getModuleInstanceState(node, visited) {
-        if (node.body && !node.body.parent) {
-            // getModuleInstanceStateForAliasTarget needs to walk up the parent chain, so parent pointers must be set on this tree already
-            setParentPointers(node, node.body);
-        }
-        return node.body ? getModuleInstanceStateCached(node.body, visited) : 1 /* Instantiated */;
-    }
-    ts.getModuleInstanceState = getModuleInstanceState;
-    function getModuleInstanceStateCached(node, visited) {
-        if (visited === void 0) { visited = ts.createMap(); }
-        var nodeId = "" + ts.getNodeId(node);
-        if (visited.has(nodeId)) {
-            return visited.get(nodeId) || 0 /* NonInstantiated */;
-        }
-        visited.set(nodeId, undefined);
-        var result = getModuleInstanceStateWorker(node, visited);
-        visited.set(nodeId, result);
-        return result;
-    }
-    function getModuleInstanceStateWorker(node, visited) {
-        // A module is uninstantiated if it contains only
-        switch (node.kind) {
-            // 1. interface declarations, type alias declarations
-            case 246 /* InterfaceDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-                return 0 /* NonInstantiated */;
-            // 2. const enum declarations
-            case 248 /* EnumDeclaration */:
-                if (ts.isEnumConst(node)) {
-                    return 2 /* ConstEnumOnly */;
-                }
-                break;
-            // 3. non-exported import declarations
-            case 254 /* ImportDeclaration */:
-            case 253 /* ImportEqualsDeclaration */:
-                if (!(ts.hasModifier(node, 1 /* Export */))) {
-                    return 0 /* NonInstantiated */;
-                }
-                break;
-            // 4. Export alias declarations pointing at only uninstantiated modules or things uninstantiated modules contain
-            case 260 /* ExportDeclaration */:
-                var exportDeclaration = node;
-                if (!exportDeclaration.moduleSpecifier && exportDeclaration.exportClause && exportDeclaration.exportClause.kind === 261 /* NamedExports */) {
-                    var state = 0 /* NonInstantiated */;
-                    for (var _i = 0, _a = exportDeclaration.exportClause.elements; _i < _a.length; _i++) {
-                        var specifier = _a[_i];
-                        var specifierState = getModuleInstanceStateForAliasTarget(specifier, visited);
-                        if (specifierState > state) {
-                            state = specifierState;
-                        }
-                        if (state === 1 /* Instantiated */) {
-                            return state;
-                        }
-                    }
-                    return state;
+        IncrementalParser.updateSourceFile = updateSourceFile;
+        function getNewCommentDirectives(oldDirectives, newDirectives, changeStart, changeRangeOldEnd, delta, oldText, newText, aggressiveChecks) {
+            if (!oldDirectives)
+                return newDirectives;
+            var commentDirectives;
+            var addedNewlyScannedDirectives = false;
+            for (var _i = 0, oldDirectives_1 = oldDirectives; _i < oldDirectives_1.length; _i++) {
+                var directive = oldDirectives_1[_i];
+                var range = directive.range, type = directive.type;
+                // Range before the change
+                if (range.end < changeStart) {
+                    commentDirectives = ts.append(commentDirectives, directive);
                 }
-                break;
-            // 5. other uninstantiated module declarations.
-            case 250 /* ModuleBlock */: {
-                var state_1 = 0 /* NonInstantiated */;
-                ts.forEachChild(node, function (n) {
-                    var childState = getModuleInstanceStateCached(n, visited);
-                    switch (childState) {
-                        case 0 /* NonInstantiated */:
-                            // child is non-instantiated - continue searching
-                            return;
-                        case 2 /* ConstEnumOnly */:
-                            // child is const enum only - record state and continue searching
-                            state_1 = 2 /* ConstEnumOnly */;
-                            return;
-                        case 1 /* Instantiated */:
-                            // child is instantiated - record state and stop
-                            state_1 = 1 /* Instantiated */;
-                            return true;
-                        default:
-                            ts.Debug.assertNever(childState);
+                else if (range.pos > changeRangeOldEnd) {
+                    addNewlyScannedDirectives();
+                    // Node is entirely past the change range.  We need to move both its pos and
+                    // end, forward or backward appropriately.
+                    var updatedDirective = {
+                        range: { pos: range.pos + delta, end: range.end + delta },
+                        type: type
+                    };
+                    commentDirectives = ts.append(commentDirectives, updatedDirective);
+                    if (aggressiveChecks) {
+                        ts.Debug.assert(oldText.substring(range.pos, range.end) === newText.substring(updatedDirective.range.pos, updatedDirective.range.end));
                     }
-                });
-                return state_1;
-            }
-            case 249 /* ModuleDeclaration */:
-                return getModuleInstanceState(node, visited);
-            case 75 /* Identifier */:
-                // Only jsdoc typedef definition can exist in jsdoc namespace, and it should
-                // be considered the same as type alias
-                if (node.isInJSDocNamespace) {
-                    return 0 /* NonInstantiated */;
                 }
-        }
-        return 1 /* Instantiated */;
-    }
-    function getModuleInstanceStateForAliasTarget(specifier, visited) {
-        var name = specifier.propertyName || specifier.name;
-        var p = specifier.parent;
-        while (p) {
-            if (ts.isBlock(p) || ts.isModuleBlock(p) || ts.isSourceFile(p)) {
-                var statements = p.statements;
-                var found = void 0;
-                for (var _i = 0, statements_1 = statements; _i < statements_1.length; _i++) {
-                    var statement = statements_1[_i];
-                    if (ts.nodeHasName(statement, name)) {
-                        if (!statement.parent) {
-                            setParentPointers(p, statement);
-                        }
-                        var state = getModuleInstanceStateCached(statement, visited);
-                        if (found === undefined || state > found) {
-                            found = state;
-                        }
-                        if (found === 1 /* Instantiated */) {
-                            return found;
-                        }
-                    }
+                // Ignore ranges that fall in change range
+            }
+            addNewlyScannedDirectives();
+            return commentDirectives;
+            function addNewlyScannedDirectives() {
+                if (addedNewlyScannedDirectives)
+                    return;
+                addedNewlyScannedDirectives = true;
+                if (!commentDirectives) {
+                    commentDirectives = newDirectives;
                 }
-                if (found !== undefined) {
-                    return found;
+                else if (newDirectives) {
+                    commentDirectives.push.apply(commentDirectives, newDirectives);
                 }
             }
-            p = p.parent;
-        }
-        return 1 /* Instantiated */; // Couldn't locate, assume could refer to a value
-    }
-    var ContainerFlags;
-    (function (ContainerFlags) {
-        // The current node is not a container, and no container manipulation should happen before
-        // recursing into it.
-        ContainerFlags[ContainerFlags["None"] = 0] = "None";
-        // The current node is a container.  It should be set as the current container (and block-
-        // container) before recursing into it.  The current node does not have locals.  Examples:
-        //
-        //      Classes, ObjectLiterals, TypeLiterals, Interfaces...
-        ContainerFlags[ContainerFlags["IsContainer"] = 1] = "IsContainer";
-        // The current node is a block-scoped-container.  It should be set as the current block-
-        // container before recursing into it.  Examples:
-        //
-        //      Blocks (when not parented by functions), Catch clauses, For/For-in/For-of statements...
-        ContainerFlags[ContainerFlags["IsBlockScopedContainer"] = 2] = "IsBlockScopedContainer";
-        // The current node is the container of a control flow path. The current control flow should
-        // be saved and restored, and a new control flow initialized within the container.
-        ContainerFlags[ContainerFlags["IsControlFlowContainer"] = 4] = "IsControlFlowContainer";
-        ContainerFlags[ContainerFlags["IsFunctionLike"] = 8] = "IsFunctionLike";
-        ContainerFlags[ContainerFlags["IsFunctionExpression"] = 16] = "IsFunctionExpression";
-        ContainerFlags[ContainerFlags["HasLocals"] = 32] = "HasLocals";
-        ContainerFlags[ContainerFlags["IsInterface"] = 64] = "IsInterface";
-        ContainerFlags[ContainerFlags["IsObjectLiteralOrClassExpressionMethod"] = 128] = "IsObjectLiteralOrClassExpressionMethod";
-    })(ContainerFlags || (ContainerFlags = {}));
-    function initFlowNode(node) {
-        ts.Debug.attachFlowNodeDebugInfo(node);
-        return node;
-    }
-    var binder = createBinder();
-    function bindSourceFile(file, options) {
-        ts.performance.mark("beforeBind");
-        ts.perfLogger.logStartBindFile("" + file.fileName);
-        binder(file, options);
-        ts.perfLogger.logStopBindFile();
-        ts.performance.mark("afterBind");
-        ts.performance.measure("Bind", "beforeBind", "afterBind");
-    }
-    ts.bindSourceFile = bindSourceFile;
-    function createBinder() {
-        var file;
-        var options;
-        var languageVersion;
-        var parent;
-        var container;
-        var thisParentContainer; // Container one level up
-        var blockScopeContainer;
-        var lastContainer;
-        var delayedTypeAliases;
-        var seenThisKeyword;
-        // state used by control flow analysis
-        var currentFlow;
-        var currentBreakTarget;
-        var currentContinueTarget;
-        var currentReturnTarget;
-        var currentTrueTarget;
-        var currentFalseTarget;
-        var currentExceptionTarget;
-        var preSwitchCaseFlow;
-        var activeLabelList;
-        var hasExplicitReturn;
-        // state used for emit helpers
-        var emitFlags;
-        // If this file is an external module, then it is automatically in strict-mode according to
-        // ES6.  If it is not an external module, then we'll determine if it is in strict mode or
-        // not depending on if we see "use strict" in certain places or if we hit a class/namespace
-        // or if compiler options contain alwaysStrict.
-        var inStrictMode;
-        var symbolCount = 0;
-        var Symbol;
-        var classifiableNames;
-        var unreachableFlow = { flags: 1 /* Unreachable */ };
-        var reportedUnreachableFlow = { flags: 1 /* Unreachable */ };
-        // state used to aggregate transform flags during bind.
-        var subtreeTransformFlags = 0 /* None */;
-        var skipTransformFlagAggregation;
-        /**
-         * Inside the binder, we may create a diagnostic for an as-yet unbound node (with potentially no parent pointers, implying no accessible source file)
-         * If so, the node _must_ be in the current file (as that's the only way anything could have traversed to it to yield it as the error node)
-         * This version of `createDiagnosticForNode` uses the binder's context to account for this, and always yields correct diagnostics even in these situations.
-         */
-        function createDiagnosticForNode(node, message, arg0, arg1, arg2) {
-            return ts.createDiagnosticForNodeInSourceFile(ts.getSourceFileOfNode(node) || file, node, message, arg0, arg1, arg2);
-        }
-        function bindSourceFile(f, opts) {
-            file = f;
-            options = opts;
-            languageVersion = ts.getEmitScriptTarget(options);
-            inStrictMode = bindInStrictMode(file, opts);
-            classifiableNames = ts.createUnderscoreEscapedMap();
-            symbolCount = 0;
-            skipTransformFlagAggregation = file.isDeclarationFile;
-            Symbol = ts.objectAllocator.getSymbolConstructor();
-            // Attach debugging information if necessary
-            ts.Debug.attachFlowNodeDebugInfo(unreachableFlow);
-            ts.Debug.attachFlowNodeDebugInfo(reportedUnreachableFlow);
-            if (!file.locals) {
-                bind(file);
-                file.symbolCount = symbolCount;
-                file.classifiableNames = classifiableNames;
-                delayedBindJSDocTypedefTag();
-            }
-            file = undefined;
-            options = undefined;
-            languageVersion = undefined;
-            parent = undefined;
-            container = undefined;
-            thisParentContainer = undefined;
-            blockScopeContainer = undefined;
-            lastContainer = undefined;
-            delayedTypeAliases = undefined;
-            seenThisKeyword = false;
-            currentFlow = undefined;
-            currentBreakTarget = undefined;
-            currentContinueTarget = undefined;
-            currentReturnTarget = undefined;
-            currentTrueTarget = undefined;
-            currentFalseTarget = undefined;
-            currentExceptionTarget = undefined;
-            activeLabelList = undefined;
-            hasExplicitReturn = false;
-            emitFlags = 0 /* None */;
-            subtreeTransformFlags = 0 /* None */;
         }
-        return bindSourceFile;
-        function bindInStrictMode(file, opts) {
-            if (ts.getStrictOptionValue(opts, "alwaysStrict") && !file.isDeclarationFile) {
-                // bind in strict mode source files with alwaysStrict option
-                return true;
+        function moveElementEntirelyPastChangeRange(element, isArray, delta, oldText, newText, aggressiveChecks) {
+            if (isArray) {
+                visitArray(element);
             }
             else {
-                return !!file.externalModuleIndicator;
-            }
-        }
-        function createSymbol(flags, name) {
-            symbolCount++;
-            return new Symbol(flags, name);
-        }
-        function addDeclarationToSymbol(symbol, node, symbolFlags) {
-            symbol.flags |= symbolFlags;
-            node.symbol = symbol;
-            symbol.declarations = ts.appendIfUnique(symbol.declarations, node);
-            if (symbolFlags & (32 /* Class */ | 384 /* Enum */ | 1536 /* Module */ | 3 /* Variable */) && !symbol.exports) {
-                symbol.exports = ts.createSymbolTable();
+                visitNode(element);
             }
-            if (symbolFlags & (32 /* Class */ | 64 /* Interface */ | 2048 /* TypeLiteral */ | 4096 /* ObjectLiteral */) && !symbol.members) {
-                symbol.members = ts.createSymbolTable();
+            return;
+            function visitNode(node) {
+                var text = "";
+                if (aggressiveChecks && shouldCheckNode(node)) {
+                    text = oldText.substring(node.pos, node.end);
+                }
+                // Ditch any existing LS children we may have created.  This way we can avoid
+                // moving them forward.
+                if (node._children) {
+                    node._children = undefined;
+                }
+                ts.setTextRangePosEnd(node, node.pos + delta, node.end + delta);
+                if (aggressiveChecks && shouldCheckNode(node)) {
+                    ts.Debug.assert(text === newText.substring(node.pos, node.end));
+                }
+                forEachChild(node, visitNode, visitArray);
+                if (ts.hasJSDocNodes(node)) {
+                    for (var _i = 0, _a = node.jsDoc; _i < _a.length; _i++) {
+                        var jsDocComment = _a[_i];
+                        visitNode(jsDocComment);
+                    }
+                }
+                checkNodePositions(node, aggressiveChecks);
             }
-            // On merge of const enum module with class or function, reset const enum only flag (namespaces will already recalculate)
-            if (symbol.constEnumOnlyModule && (symbol.flags & (16 /* Function */ | 32 /* Class */ | 256 /* RegularEnum */))) {
-                symbol.constEnumOnlyModule = false;
+            function visitArray(array) {
+                array._children = undefined;
+                ts.setTextRangePosEnd(array, array.pos + delta, array.end + delta);
+                for (var _i = 0, array_8 = array; _i < array_8.length; _i++) {
+                    var node = array_8[_i];
+                    visitNode(node);
+                }
             }
-            if (symbolFlags & 111551 /* Value */) {
-                ts.setValueDeclaration(symbol, node);
+        }
+        function shouldCheckNode(node) {
+            switch (node.kind) {
+                case 10 /* StringLiteral */:
+                case 8 /* NumericLiteral */:
+                case 78 /* Identifier */:
+                    return true;
             }
+            return false;
         }
-        // Should not be called on a declaration with a computed property name,
-        // unless it is a well known Symbol.
-        function getDeclarationName(node) {
-            if (node.kind === 259 /* ExportAssignment */) {
-                return node.isExportEquals ? "export=" /* ExportEquals */ : "default" /* Default */;
+        function adjustIntersectingElement(element, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta) {
+            ts.Debug.assert(element.end >= changeStart, "Adjusting an element that was entirely before the change range");
+            ts.Debug.assert(element.pos <= changeRangeOldEnd, "Adjusting an element that was entirely after the change range");
+            ts.Debug.assert(element.pos <= element.end);
+            // We have an element that intersects the change range in some way.  It may have its
+            // start, or its end (or both) in the changed range.  We want to adjust any part
+            // that intersects such that the final tree is in a consistent state.  i.e. all
+            // children have spans within the span of their parent, and all siblings are ordered
+            // properly.
+            // We may need to update both the 'pos' and the 'end' of the element.
+            // If the 'pos' is before the start of the change, then we don't need to touch it.
+            // If it isn't, then the 'pos' must be inside the change.  How we update it will
+            // depend if delta is positive or negative. If delta is positive then we have
+            // something like:
+            //
+            //  -------------------AAA-----------------
+            //  -------------------BBBCCCCCCC-----------------
+            //
+            // In this case, we consider any node that started in the change range to still be
+            // starting at the same position.
+            //
+            // however, if the delta is negative, then we instead have something like this:
+            //
+            //  -------------------XXXYYYYYYY-----------------
+            //  -------------------ZZZ-----------------
+            //
+            // In this case, any element that started in the 'X' range will keep its position.
+            // However any element that started after that will have their pos adjusted to be
+            // at the end of the new range.  i.e. any node that started in the 'Y' range will
+            // be adjusted to have their start at the end of the 'Z' range.
+            //
+            // The element will keep its position if possible.  Or Move backward to the new-end
+            // if it's in the 'Y' range.
+            var pos = Math.min(element.pos, changeRangeNewEnd);
+            // If the 'end' is after the change range, then we always adjust it by the delta
+            // amount.  However, if the end is in the change range, then how we adjust it
+            // will depend on if delta is positive or negative.  If delta is positive then we
+            // have something like:
+            //
+            //  -------------------AAA-----------------
+            //  -------------------BBBCCCCCCC-----------------
+            //
+            // In this case, we consider any node that ended inside the change range to keep its
+            // end position.
+            //
+            // however, if the delta is negative, then we instead have something like this:
+            //
+            //  -------------------XXXYYYYYYY-----------------
+            //  -------------------ZZZ-----------------
+            //
+            // In this case, any element that ended in the 'X' range will keep its position.
+            // However any element that ended after that will have their pos adjusted to be
+            // at the end of the new range.  i.e. any node that ended in the 'Y' range will
+            // be adjusted to have their end at the end of the 'Z' range.
+            var end = element.end >= changeRangeOldEnd ?
+                // Element ends after the change range.  Always adjust the end pos.
+                element.end + delta :
+                // Element ends in the change range.  The element will keep its position if
+                // possible. Or Move backward to the new-end if it's in the 'Y' range.
+                Math.min(element.end, changeRangeNewEnd);
+            ts.Debug.assert(pos <= end);
+            if (element.parent) {
+                ts.Debug.assertGreaterThanOrEqual(pos, element.parent.pos);
+                ts.Debug.assertLessThanOrEqual(end, element.parent.end);
             }
-            var name = ts.getNameOfDeclaration(node);
-            if (name) {
-                if (ts.isAmbientModule(node)) {
-                    var moduleName = ts.getTextOfIdentifierOrLiteral(name);
-                    return (ts.isGlobalScopeAugmentation(node) ? "__global" : "\"" + moduleName + "\"");
-                }
-                if (name.kind === 154 /* ComputedPropertyName */) {
-                    var nameExpression = name.expression;
-                    // treat computed property names where expression is string/numeric literal as just string/numeric literal
-                    if (ts.isStringOrNumericLiteralLike(nameExpression)) {
-                        return ts.escapeLeadingUnderscores(nameExpression.text);
-                    }
-                    if (ts.isSignedNumericLiteral(nameExpression)) {
-                        return ts.tokenToString(nameExpression.operator) + nameExpression.operand.text;
+            ts.setTextRangePosEnd(element, pos, end);
+        }
+        function checkNodePositions(node, aggressiveChecks) {
+            if (aggressiveChecks) {
+                var pos_2 = node.pos;
+                var visitNode_1 = function (child) {
+                    ts.Debug.assert(child.pos >= pos_2);
+                    pos_2 = child.end;
+                };
+                if (ts.hasJSDocNodes(node)) {
+                    for (var _i = 0, _a = node.jsDoc; _i < _a.length; _i++) {
+                        var jsDocComment = _a[_i];
+                        visitNode_1(jsDocComment);
                     }
-                    ts.Debug.assert(ts.isWellKnownSymbolSyntactically(nameExpression));
-                    return ts.getPropertyNameForKnownSymbolName(ts.idText(nameExpression.name));
                 }
-                if (ts.isWellKnownSymbolSyntactically(name)) {
-                    return ts.getPropertyNameForKnownSymbolName(ts.idText(name.name));
+                forEachChild(node, visitNode_1);
+                ts.Debug.assert(pos_2 <= node.end);
+            }
+        }
+        function updateTokenPositionsAndMarkElements(sourceFile, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta, oldText, newText, aggressiveChecks) {
+            visitNode(sourceFile);
+            return;
+            function visitNode(child) {
+                ts.Debug.assert(child.pos <= child.end);
+                if (child.pos > changeRangeOldEnd) {
+                    // Node is entirely past the change range.  We need to move both its pos and
+                    // end, forward or backward appropriately.
+                    moveElementEntirelyPastChangeRange(child, /*isArray*/ false, delta, oldText, newText, aggressiveChecks);
+                    return;
                 }
-                if (ts.isPrivateIdentifier(name)) {
-                    // containingClass exists because private names only allowed inside classes
-                    var containingClass = ts.getContainingClass(node);
-                    if (!containingClass) {
-                        // we can get here in cases where there is already a parse error.
-                        return undefined;
+                // Check if the element intersects the change range.  If it does, then it is not
+                // reusable.  Also, we'll need to recurse to see what constituent portions we may
+                // be able to use.
+                var fullEnd = child.end;
+                if (fullEnd >= changeStart) {
+                    child.intersectsChange = true;
+                    child._children = undefined;
+                    // Adjust the pos or end (or both) of the intersecting element accordingly.
+                    adjustIntersectingElement(child, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta);
+                    forEachChild(child, visitNode, visitArray);
+                    if (ts.hasJSDocNodes(child)) {
+                        for (var _i = 0, _a = child.jsDoc; _i < _a.length; _i++) {
+                            var jsDocComment = _a[_i];
+                            visitNode(jsDocComment);
+                        }
                     }
-                    var containingClassSymbol = containingClass.symbol;
-                    return ts.getSymbolNameForPrivateIdentifier(containingClassSymbol, name.escapedText);
+                    checkNodePositions(child, aggressiveChecks);
+                    return;
                 }
-                return ts.isPropertyNameLiteral(name) ? ts.getEscapedTextOfIdentifierOrLiteral(name) : undefined;
+                // Otherwise, the node is entirely before the change range.  No need to do anything with it.
+                ts.Debug.assert(fullEnd < changeStart);
             }
-            switch (node.kind) {
-                case 162 /* Constructor */:
-                    return "__constructor" /* Constructor */;
-                case 170 /* FunctionType */:
-                case 165 /* CallSignature */:
-                case 305 /* JSDocSignature */:
-                    return "__call" /* Call */;
-                case 171 /* ConstructorType */:
-                case 166 /* ConstructSignature */:
-                    return "__new" /* New */;
-                case 167 /* IndexSignature */:
-                    return "__index" /* Index */;
-                case 260 /* ExportDeclaration */:
-                    return "__export" /* ExportStar */;
-                case 290 /* SourceFile */:
-                    // json file should behave as
-                    // module.exports = ...
-                    return "export=" /* ExportEquals */;
-                case 209 /* BinaryExpression */:
-                    if (ts.getAssignmentDeclarationKind(node) === 2 /* ModuleExports */) {
-                        // module.exports = ...
-                        return "export=" /* ExportEquals */;
+            function visitArray(array) {
+                ts.Debug.assert(array.pos <= array.end);
+                if (array.pos > changeRangeOldEnd) {
+                    // Array is entirely after the change range.  We need to move it, and move any of
+                    // its children.
+                    moveElementEntirelyPastChangeRange(array, /*isArray*/ true, delta, oldText, newText, aggressiveChecks);
+                    return;
+                }
+                // Check if the element intersects the change range.  If it does, then it is not
+                // reusable.  Also, we'll need to recurse to see what constituent portions we may
+                // be able to use.
+                var fullEnd = array.end;
+                if (fullEnd >= changeStart) {
+                    array.intersectsChange = true;
+                    array._children = undefined;
+                    // Adjust the pos or end (or both) of the intersecting array accordingly.
+                    adjustIntersectingElement(array, changeStart, changeRangeOldEnd, changeRangeNewEnd, delta);
+                    for (var _i = 0, array_9 = array; _i < array_9.length; _i++) {
+                        var node = array_9[_i];
+                        visitNode(node);
                     }
-                    ts.Debug.fail("Unknown binary declaration kind");
-                    break;
-                case 300 /* JSDocFunctionType */:
-                    return (ts.isJSDocConstructSignature(node) ? "__new" /* New */ : "__call" /* Call */);
-                case 156 /* Parameter */:
-                    // Parameters with names are handled at the top of this function.  Parameters
-                    // without names can only come from JSDocFunctionTypes.
-                    ts.Debug.assert(node.parent.kind === 300 /* JSDocFunctionType */, "Impossible parameter parent kind", function () { return "parent is: " + (ts.SyntaxKind ? ts.SyntaxKind[node.parent.kind] : node.parent.kind) + ", expected JSDocFunctionType"; });
-                    var functionType = node.parent;
-                    var index = functionType.parameters.indexOf(node);
-                    return "arg" + index;
+                    return;
+                }
+                // Otherwise, the array is entirely before the change range.  No need to do anything with it.
+                ts.Debug.assert(fullEnd < changeStart);
             }
         }
-        function getDisplayName(node) {
-            return ts.isNamedDeclaration(node) ? ts.declarationNameToString(node.name) : ts.unescapeLeadingUnderscores(ts.Debug.checkDefined(getDeclarationName(node)));
-        }
-        /**
-         * Declares a Symbol for the node and adds it to symbols. Reports errors for conflicting identifier names.
-         * @param symbolTable - The symbol table which node will be added to.
-         * @param parent - node's parent declaration.
-         * @param node - The declaration to be added to the symbol table
-         * @param includes - The SymbolFlags that node has in addition to its declaration type (eg: export, ambient, etc.)
-         * @param excludes - The flags which node cannot be declared alongside in a symbol table. Used to report forbidden declarations.
-         */
-        function declareSymbol(symbolTable, parent, node, includes, excludes, isReplaceableByMethod) {
-            ts.Debug.assert(!ts.hasDynamicName(node));
-            var isDefaultExport = ts.hasModifier(node, 512 /* Default */) || ts.isExportSpecifier(node) && node.name.escapedText === "default";
-            // The exported symbol for an export default function/class node is always named "default"
-            var name = isDefaultExport && parent ? "default" /* Default */ : getDeclarationName(node);
-            var symbol;
-            if (name === undefined) {
-                symbol = createSymbol(0 /* None */, "__missing" /* Missing */);
+        function extendToAffectedRange(sourceFile, changeRange) {
+            // Consider the following code:
+            //      void foo() { /; }
+            //
+            // If the text changes with an insertion of / just before the semicolon then we end up with:
+            //      void foo() { //; }
+            //
+            // If we were to just use the changeRange a is, then we would not rescan the { token
+            // (as it does not intersect the actual original change range).  Because an edit may
+            // change the token touching it, we actually need to look back *at least* one token so
+            // that the prior token sees that change.
+            var maxLookahead = 1;
+            var start = changeRange.span.start;
+            // the first iteration aligns us with the change start. subsequent iteration move us to
+            // the left by maxLookahead tokens.  We only need to do this as long as we're not at the
+            // start of the tree.
+            for (var i = 0; start > 0 && i <= maxLookahead; i++) {
+                var nearestNode = findNearestNodeStartingBeforeOrAtPosition(sourceFile, start);
+                ts.Debug.assert(nearestNode.pos <= start);
+                var position = nearestNode.pos;
+                start = Math.max(0, position - 1);
             }
-            else {
-                // Check and see if the symbol table already has a symbol with this name.  If not,
-                // create a new symbol with this name and add it to the table.  Note that we don't
-                // give the new symbol any flags *yet*.  This ensures that it will not conflict
-                // with the 'excludes' flags we pass in.
-                //
-                // If we do get an existing symbol, see if it conflicts with the new symbol we're
-                // creating.  For example, a 'var' symbol and a 'class' symbol will conflict within
-                // the same symbol table.  If we have a conflict, report the issue on each
-                // declaration we have for this symbol, and then create a new symbol for this
-                // declaration.
-                //
-                // Note that when properties declared in Javascript constructors
-                // (marked by isReplaceableByMethod) conflict with another symbol, the property loses.
-                // Always. This allows the common Javascript pattern of overwriting a prototype method
-                // with an bound instance method of the same type: `this.method = this.method.bind(this)`
-                //
-                // If we created a new symbol, either because we didn't have a symbol with this name
-                // in the symbol table, or we conflicted with an existing symbol, then just add this
-                // node as the sole declaration of the new symbol.
-                //
-                // Otherwise, we'll be merging into a compatible existing symbol (for example when
-                // you have multiple 'vars' with the same name in the same container).  In this case
-                // just add this node into the declarations list of the symbol.
-                symbol = symbolTable.get(name);
-                if (includes & 2885600 /* Classifiable */) {
-                    classifiableNames.set(name, true);
+            var finalSpan = ts.createTextSpanFromBounds(start, ts.textSpanEnd(changeRange.span));
+            var finalLength = changeRange.newLength + (changeRange.span.start - start);
+            return ts.createTextChangeRange(finalSpan, finalLength);
+        }
+        function findNearestNodeStartingBeforeOrAtPosition(sourceFile, position) {
+            var bestResult = sourceFile;
+            var lastNodeEntirelyBeforePosition;
+            forEachChild(sourceFile, visit);
+            if (lastNodeEntirelyBeforePosition) {
+                var lastChildOfLastEntireNodeBeforePosition = getLastDescendant(lastNodeEntirelyBeforePosition);
+                if (lastChildOfLastEntireNodeBeforePosition.pos > bestResult.pos) {
+                    bestResult = lastChildOfLastEntireNodeBeforePosition;
                 }
-                if (!symbol) {
-                    symbolTable.set(name, symbol = createSymbol(0 /* None */, name));
-                    if (isReplaceableByMethod)
-                        symbol.isReplaceableByMethod = true;
+            }
+            return bestResult;
+            function getLastDescendant(node) {
+                while (true) {
+                    var lastChild = ts.getLastChild(node);
+                    if (lastChild) {
+                        node = lastChild;
+                    }
+                    else {
+                        return node;
+                    }
                 }
-                else if (isReplaceableByMethod && !symbol.isReplaceableByMethod) {
-                    // A symbol already exists, so don't add this as a declaration.
-                    return symbol;
+            }
+            function visit(child) {
+                if (ts.nodeIsMissing(child)) {
+                    // Missing nodes are effectively invisible to us.  We never even consider them
+                    // When trying to find the nearest node before us.
+                    return;
                 }
-                else if (symbol.flags & excludes) {
-                    if (symbol.isReplaceableByMethod) {
-                        // Javascript constructor-declared symbols can be discarded in favor of
-                        // prototype symbols like methods.
-                        symbolTable.set(name, symbol = createSymbol(0 /* None */, name));
+                // If the child intersects this position, then this node is currently the nearest
+                // node that starts before the position.
+                if (child.pos <= position) {
+                    if (child.pos >= bestResult.pos) {
+                        // This node starts before the position, and is closer to the position than
+                        // the previous best node we found.  It is now the new best node.
+                        bestResult = child;
                     }
-                    else if (!(includes & 3 /* Variable */ && symbol.flags & 67108864 /* Assignment */)) {
-                        // Assignment declarations are allowed to merge with variables, no matter what other flags they have.
-                        if (ts.isNamedDeclaration(node)) {
-                            node.name.parent = node;
+                    // Now, the node may overlap the position, or it may end entirely before the
+                    // position.  If it overlaps with the position, then either it, or one of its
+                    // children must be the nearest node before the position.  So we can just
+                    // recurse into this child to see if we can find something better.
+                    if (position < child.end) {
+                        // The nearest node is either this child, or one of the children inside
+                        // of it.  We've already marked this child as the best so far.  Recurse
+                        // in case one of the children is better.
+                        forEachChild(child, visit);
+                        // Once we look at the children of this node, then there's no need to
+                        // continue any further.
+                        return true;
+                    }
+                    else {
+                        ts.Debug.assert(child.end <= position);
+                        // The child ends entirely before this position.  Say you have the following
+                        // (where $ is the position)
+                        //
+                        //      <complex expr 1> ? <complex expr 2> $ : <...> <...>
+                        //
+                        // We would want to find the nearest preceding node in "complex expr 2".
+                        // To support that, we keep track of this node, and once we're done searching
+                        // for a best node, we recurse down this node to see if we can find a good
+                        // result in it.
+                        //
+                        // This approach allows us to quickly skip over nodes that are entirely
+                        // before the position, while still allowing us to find any nodes in the
+                        // last one that might be what we want.
+                        lastNodeEntirelyBeforePosition = child;
+                    }
+                }
+                else {
+                    ts.Debug.assert(child.pos > position);
+                    // We're now at a node that is entirely past the position we're searching for.
+                    // This node (and all following nodes) could never contribute to the result,
+                    // so just skip them by returning 'true' here.
+                    return true;
+                }
+            }
+        }
+        function checkChangeRange(sourceFile, newText, textChangeRange, aggressiveChecks) {
+            var oldText = sourceFile.text;
+            if (textChangeRange) {
+                ts.Debug.assert((oldText.length - textChangeRange.span.length + textChangeRange.newLength) === newText.length);
+                if (aggressiveChecks || ts.Debug.shouldAssert(3 /* VeryAggressive */)) {
+                    var oldTextPrefix = oldText.substr(0, textChangeRange.span.start);
+                    var newTextPrefix = newText.substr(0, textChangeRange.span.start);
+                    ts.Debug.assert(oldTextPrefix === newTextPrefix);
+                    var oldTextSuffix = oldText.substring(ts.textSpanEnd(textChangeRange.span), oldText.length);
+                    var newTextSuffix = newText.substring(ts.textSpanEnd(ts.textChangeRangeNewSpan(textChangeRange)), newText.length);
+                    ts.Debug.assert(oldTextSuffix === newTextSuffix);
+                }
+            }
+        }
+        function createSyntaxCursor(sourceFile) {
+            var currentArray = sourceFile.statements;
+            var currentArrayIndex = 0;
+            ts.Debug.assert(currentArrayIndex < currentArray.length);
+            var current = currentArray[currentArrayIndex];
+            var lastQueriedPosition = -1 /* Value */;
+            return {
+                currentNode: function (position) {
+                    // Only compute the current node if the position is different than the last time
+                    // we were asked.  The parser commonly asks for the node at the same position
+                    // twice.  Once to know if can read an appropriate list element at a certain point,
+                    // and then to actually read and consume the node.
+                    if (position !== lastQueriedPosition) {
+                        // Much of the time the parser will need the very next node in the array that
+                        // we just returned a node from.So just simply check for that case and move
+                        // forward in the array instead of searching for the node again.
+                        if (current && current.end === position && currentArrayIndex < (currentArray.length - 1)) {
+                            currentArrayIndex++;
+                            current = currentArray[currentArrayIndex];
                         }
-                        // Report errors every position with duplicate declaration
-                        // Report errors on previous encountered declarations
-                        var message_1 = symbol.flags & 2 /* BlockScopedVariable */
-                            ? ts.Diagnostics.Cannot_redeclare_block_scoped_variable_0
-                            : ts.Diagnostics.Duplicate_identifier_0;
-                        var messageNeedsName_1 = true;
-                        if (symbol.flags & 384 /* Enum */ || includes & 384 /* Enum */) {
-                            message_1 = ts.Diagnostics.Enum_declarations_can_only_merge_with_namespace_or_other_enum_declarations;
-                            messageNeedsName_1 = false;
+                        // If we don't have a node, or the node we have isn't in the right position,
+                        // then try to find a viable node at the position requested.
+                        if (!current || current.pos !== position) {
+                            findHighestListElementThatStartsAtPosition(position);
                         }
-                        var multipleDefaultExports_1 = false;
-                        if (ts.length(symbol.declarations)) {
-                            // If the current node is a default export of some sort, then check if
-                            // there are any other default exports that we need to error on.
-                            // We'll know whether we have other default exports depending on if `symbol` already has a declaration list set.
-                            if (isDefaultExport) {
-                                message_1 = ts.Diagnostics.A_module_cannot_have_multiple_default_exports;
-                                messageNeedsName_1 = false;
-                                multipleDefaultExports_1 = true;
-                            }
-                            else {
-                                // This is to properly report an error in the case "export default { }" is after export default of class declaration or function declaration.
-                                // Error on multiple export default in the following case:
-                                // 1. multiple export default of class declaration or function declaration by checking NodeFlags.Default
-                                // 2. multiple export default of export assignment. This one doesn't have NodeFlags.Default on (as export default doesn't considered as modifiers)
-                                if (symbol.declarations && symbol.declarations.length &&
-                                    (node.kind === 259 /* ExportAssignment */ && !node.isExportEquals)) {
-                                    message_1 = ts.Diagnostics.A_module_cannot_have_multiple_default_exports;
-                                    messageNeedsName_1 = false;
-                                    multipleDefaultExports_1 = true;
+                    }
+                    // Cache this query so that we don't do any extra work if the parser calls back
+                    // into us.  Note: this is very common as the parser will make pairs of calls like
+                    // 'isListElement -> parseListElement'.  If we were unable to find a node when
+                    // called with 'isListElement', we don't want to redo the work when parseListElement
+                    // is called immediately after.
+                    lastQueriedPosition = position;
+                    // Either we don'd have a node, or we have a node at the position being asked for.
+                    ts.Debug.assert(!current || current.pos === position);
+                    return current;
+                }
+            };
+            // Finds the highest element in the tree we can find that starts at the provided position.
+            // The element must be a direct child of some node list in the tree.  This way after we
+            // return it, we can easily return its next sibling in the list.
+            function findHighestListElementThatStartsAtPosition(position) {
+                // Clear out any cached state about the last node we found.
+                currentArray = undefined;
+                currentArrayIndex = -1 /* Value */;
+                current = undefined;
+                // Recurse into the source file to find the highest node at this position.
+                forEachChild(sourceFile, visitNode, visitArray);
+                return;
+                function visitNode(node) {
+                    if (position >= node.pos && position < node.end) {
+                        // Position was within this node.  Keep searching deeper to find the node.
+                        forEachChild(node, visitNode, visitArray);
+                        // don't proceed any further in the search.
+                        return true;
+                    }
+                    // position wasn't in this node, have to keep searching.
+                    return false;
+                }
+                function visitArray(array) {
+                    if (position >= array.pos && position < array.end) {
+                        // position was in this array.  Search through this array to see if we find a
+                        // viable element.
+                        for (var i = 0; i < array.length; i++) {
+                            var child = array[i];
+                            if (child) {
+                                if (child.pos === position) {
+                                    // Found the right node.  We're done.
+                                    currentArray = array;
+                                    currentArrayIndex = i;
+                                    current = child;
+                                    return true;
+                                }
+                                else {
+                                    if (child.pos < position && position < child.end) {
+                                        // Position in somewhere within this child.  Search in it and
+                                        // stop searching in this array.
+                                        forEachChild(child, visitNode, visitArray);
+                                        return true;
+                                    }
                                 }
                             }
                         }
-                        var relatedInformation_1 = [];
-                        if (ts.isTypeAliasDeclaration(node) && ts.nodeIsMissing(node.type) && ts.hasModifier(node, 1 /* Export */) && symbol.flags & (2097152 /* Alias */ | 788968 /* Type */ | 1920 /* Namespace */)) {
-                            // export type T; - may have meant export type { T }?
-                            relatedInformation_1.push(createDiagnosticForNode(node, ts.Diagnostics.Did_you_mean_0, "export type { " + ts.unescapeLeadingUnderscores(node.name.escapedText) + " }"));
-                        }
-                        var declarationName_1 = ts.getNameOfDeclaration(node) || node;
-                        ts.forEach(symbol.declarations, function (declaration, index) {
-                            var decl = ts.getNameOfDeclaration(declaration) || declaration;
-                            var diag = createDiagnosticForNode(decl, message_1, messageNeedsName_1 ? getDisplayName(declaration) : undefined);
-                            file.bindDiagnostics.push(multipleDefaultExports_1 ? ts.addRelatedInfo(diag, createDiagnosticForNode(declarationName_1, index === 0 ? ts.Diagnostics.Another_export_default_is_here : ts.Diagnostics.and_here)) : diag);
-                            if (multipleDefaultExports_1) {
-                                relatedInformation_1.push(createDiagnosticForNode(decl, ts.Diagnostics.The_first_export_default_is_here));
-                            }
-                        });
-                        var diag = createDiagnosticForNode(declarationName_1, message_1, messageNeedsName_1 ? getDisplayName(node) : undefined);
-                        file.bindDiagnostics.push(ts.addRelatedInfo.apply(void 0, __spreadArrays([diag], relatedInformation_1)));
-                        symbol = createSymbol(0 /* None */, name);
                     }
+                    // position wasn't in this array, have to keep searching.
+                    return false;
                 }
             }
-            addDeclarationToSymbol(symbol, node, includes);
-            if (symbol.parent) {
-                ts.Debug.assert(symbol.parent === parent, "Existing symbol parent should match new one");
-            }
-            else {
-                symbol.parent = parent;
-            }
-            return symbol;
         }
-        function declareModuleMember(node, symbolFlags, symbolExcludes) {
-            var hasExportModifier = ts.getCombinedModifierFlags(node) & 1 /* Export */;
-            if (symbolFlags & 2097152 /* Alias */) {
-                if (node.kind === 263 /* ExportSpecifier */ || (node.kind === 253 /* ImportEqualsDeclaration */ && hasExportModifier)) {
-                    return declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes);
+        IncrementalParser.createSyntaxCursor = createSyntaxCursor;
+        var InvalidPosition;
+        (function (InvalidPosition) {
+            InvalidPosition[InvalidPosition["Value"] = -1] = "Value";
+        })(InvalidPosition || (InvalidPosition = {}));
+    })(IncrementalParser || (IncrementalParser = {}));
+    /** @internal */
+    function isDeclarationFileName(fileName) {
+        return ts.fileExtensionIs(fileName, ".d.ts" /* Dts */);
+    }
+    ts.isDeclarationFileName = isDeclarationFileName;
+    /*@internal*/
+    function processCommentPragmas(context, sourceText) {
+        var pragmas = [];
+        for (var _i = 0, _a = ts.getLeadingCommentRanges(sourceText, 0) || ts.emptyArray; _i < _a.length; _i++) {
+            var range = _a[_i];
+            var comment = sourceText.substring(range.pos, range.end);
+            extractPragmas(pragmas, range, comment);
+        }
+        context.pragmas = new ts.Map();
+        for (var _b = 0, pragmas_1 = pragmas; _b < pragmas_1.length; _b++) {
+            var pragma = pragmas_1[_b];
+            if (context.pragmas.has(pragma.name)) {
+                var currentValue = context.pragmas.get(pragma.name);
+                if (currentValue instanceof Array) {
+                    currentValue.push(pragma.args);
                 }
                 else {
-                    return declareSymbol(container.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
+                    context.pragmas.set(pragma.name, [currentValue, pragma.args]);
                 }
+                continue;
             }
-            else {
-                // Exported module members are given 2 symbols: A local symbol that is classified with an ExportValue flag,
-                // and an associated export symbol with all the correct flags set on it. There are 2 main reasons:
-                //
-                //   1. We treat locals and exports of the same name as mutually exclusive within a container.
-                //      That means the binder will issue a Duplicate Identifier error if you mix locals and exports
-                //      with the same name in the same container.
-                //      TODO: Make this a more specific error and decouple it from the exclusion logic.
-                //   2. When we checkIdentifier in the checker, we set its resolved symbol to the local symbol,
-                //      but return the export symbol (by calling getExportSymbolOfValueSymbolIfExported). That way
-                //      when the emitter comes back to it, it knows not to qualify the name if it was found in a containing scope.
-                // NOTE: Nested ambient modules always should go to to 'locals' table to prevent their automatic merge
-                //       during global merging in the checker. Why? The only case when ambient module is permitted inside another module is module augmentation
-                //       and this case is specially handled. Module augmentations should only be merged with original module definition
-                //       and should never be merged directly with other augmentation, and the latter case would be possible if automatic merge is allowed.
-                if (ts.isJSDocTypeAlias(node))
-                    ts.Debug.assert(ts.isInJSFile(node)); // We shouldn't add symbols for JSDoc nodes if not in a JS file.
-                if ((!ts.isAmbientModule(node) && (hasExportModifier || container.flags & 64 /* ExportContext */)) || ts.isJSDocTypeAlias(node)) {
-                    if (!container.locals || (ts.hasModifier(node, 512 /* Default */) && !getDeclarationName(node))) {
-                        return declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes); // No local symbol for an unnamed default!
-                    }
-                    var exportKind = symbolFlags & 111551 /* Value */ ? 1048576 /* ExportValue */ : 0;
-                    var local = declareSymbol(container.locals, /*parent*/ undefined, node, exportKind, symbolExcludes);
-                    local.exportSymbol = declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes);
-                    node.localSymbol = local;
-                    return local;
+            context.pragmas.set(pragma.name, pragma.args);
+        }
+    }
+    ts.processCommentPragmas = processCommentPragmas;
+    /*@internal*/
+    function processPragmasIntoFields(context, reportDiagnostic) {
+        context.checkJsDirective = undefined;
+        context.referencedFiles = [];
+        context.typeReferenceDirectives = [];
+        context.libReferenceDirectives = [];
+        context.amdDependencies = [];
+        context.hasNoDefaultLib = false;
+        context.pragmas.forEach(function (entryOrList, key) {
+            // TODO: The below should be strongly type-guarded and not need casts/explicit annotations, since entryOrList is related to
+            // key and key is constrained to a union; but it's not (see GH#21483 for at least partial fix) :(
+            switch (key) {
+                case "reference": {
+                    var referencedFiles_1 = context.referencedFiles;
+                    var typeReferenceDirectives_1 = context.typeReferenceDirectives;
+                    var libReferenceDirectives_1 = context.libReferenceDirectives;
+                    ts.forEach(ts.toArray(entryOrList), function (arg) {
+                        var _a = arg.arguments, types = _a.types, lib = _a.lib, path = _a.path;
+                        if (arg.arguments["no-default-lib"]) {
+                            context.hasNoDefaultLib = true;
+                        }
+                        else if (types) {
+                            typeReferenceDirectives_1.push({ pos: types.pos, end: types.end, fileName: types.value });
+                        }
+                        else if (lib) {
+                            libReferenceDirectives_1.push({ pos: lib.pos, end: lib.end, fileName: lib.value });
+                        }
+                        else if (path) {
+                            referencedFiles_1.push({ pos: path.pos, end: path.end, fileName: path.value });
+                        }
+                        else {
+                            reportDiagnostic(arg.range.pos, arg.range.end - arg.range.pos, ts.Diagnostics.Invalid_reference_directive_syntax);
+                        }
+                    });
+                    break;
                 }
-                else {
-                    return declareSymbol(container.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
+                case "amd-dependency": {
+                    context.amdDependencies = ts.map(ts.toArray(entryOrList), function (x) { return ({ name: x.arguments.name, path: x.arguments.path }); });
+                    break;
                 }
-            }
-        }
-        // All container nodes are kept on a linked list in declaration order. This list is used by
-        // the getLocalNameOfContainer function in the type checker to validate that the local name
-        // used for a container is unique.
-        function bindContainer(node, containerFlags) {
-            // Before we recurse into a node's children, we first save the existing parent, container
-            // and block-container.  Then after we pop out of processing the children, we restore
-            // these saved values.
-            var saveContainer = container;
-            var saveThisParentContainer = thisParentContainer;
-            var savedBlockScopeContainer = blockScopeContainer;
-            // Depending on what kind of node this is, we may have to adjust the current container
-            // and block-container.   If the current node is a container, then it is automatically
-            // considered the current block-container as well.  Also, for containers that we know
-            // may contain locals, we eagerly initialize the .locals field. We do this because
-            // it's highly likely that the .locals will be needed to place some child in (for example,
-            // a parameter, or variable declaration).
-            //
-            // However, we do not proactively create the .locals for block-containers because it's
-            // totally normal and common for block-containers to never actually have a block-scoped
-            // variable in them.  We don't want to end up allocating an object for every 'block' we
-            // run into when most of them won't be necessary.
-            //
-            // Finally, if this is a block-container, then we clear out any existing .locals object
-            // it may contain within it.  This happens in incremental scenarios.  Because we can be
-            // reusing a node from a previous compilation, that node may have had 'locals' created
-            // for it.  We must clear this so we don't accidentally move any stale data forward from
-            // a previous compilation.
-            if (containerFlags & 1 /* IsContainer */) {
-                if (node.kind !== 202 /* ArrowFunction */) {
-                    thisParentContainer = container;
+                case "amd-module": {
+                    if (entryOrList instanceof Array) {
+                        for (var _i = 0, entryOrList_1 = entryOrList; _i < entryOrList_1.length; _i++) {
+                            var entry = entryOrList_1[_i];
+                            if (context.moduleName) {
+                                // TODO: It's probably fine to issue this diagnostic on all instances of the pragma
+                                reportDiagnostic(entry.range.pos, entry.range.end - entry.range.pos, ts.Diagnostics.An_AMD_module_cannot_have_multiple_name_assignments);
+                            }
+                            context.moduleName = entry.arguments.name;
+                        }
+                    }
+                    else {
+                        context.moduleName = entryOrList.arguments.name;
+                    }
+                    break;
                 }
-                container = blockScopeContainer = node;
-                if (containerFlags & 32 /* HasLocals */) {
-                    container.locals = ts.createSymbolTable();
+                case "ts-nocheck":
+                case "ts-check": {
+                    // _last_ of either nocheck or check in a file is the "winner"
+                    ts.forEach(ts.toArray(entryOrList), function (entry) {
+                        if (!context.checkJsDirective || entry.range.pos > context.checkJsDirective.pos) {
+                            context.checkJsDirective = {
+                                enabled: key === "ts-check",
+                                end: entry.range.end,
+                                pos: entry.range.pos
+                            };
+                        }
+                    });
+                    break;
                 }
-                addToContainerChain(container);
+                case "jsx":
+                case "jsxfrag":
+                case "jsximportsource":
+                case "jsxruntime":
+                    return; // Accessed directly
+                default: ts.Debug.fail("Unhandled pragma kind"); // Can this be made into an assertNever in the future?
             }
-            else if (containerFlags & 2 /* IsBlockScopedContainer */) {
-                blockScopeContainer = node;
-                blockScopeContainer.locals = undefined;
+        });
+    }
+    ts.processPragmasIntoFields = processPragmasIntoFields;
+    var namedArgRegExCache = new ts.Map();
+    function getNamedArgRegEx(name) {
+        if (namedArgRegExCache.has(name)) {
+            return namedArgRegExCache.get(name);
+        }
+        var result = new RegExp("(\\s" + name + "\\s*=\\s*)('|\")(.+?)\\2", "im");
+        namedArgRegExCache.set(name, result);
+        return result;
+    }
+    var tripleSlashXMLCommentStartRegEx = /^\/\/\/\s*<(\S+)\s.*?\/>/im;
+    var singleLinePragmaRegEx = /^\/\/\/?\s*@(\S+)\s*(.*)\s*$/im;
+    function extractPragmas(pragmas, range, text) {
+        var tripleSlash = range.kind === 2 /* SingleLineCommentTrivia */ && tripleSlashXMLCommentStartRegEx.exec(text);
+        if (tripleSlash) {
+            var name = tripleSlash[1].toLowerCase(); // Technically unsafe cast, but we do it so the below check to make it safe typechecks
+            var pragma = ts.commentPragmas[name];
+            if (!pragma || !(pragma.kind & 1 /* TripleSlashXML */)) {
+                return;
             }
-            if (containerFlags & 4 /* IsControlFlowContainer */) {
-                var saveCurrentFlow = currentFlow;
-                var saveBreakTarget = currentBreakTarget;
-                var saveContinueTarget = currentContinueTarget;
-                var saveReturnTarget = currentReturnTarget;
-                var saveExceptionTarget = currentExceptionTarget;
-                var saveActiveLabelList = activeLabelList;
-                var saveHasExplicitReturn = hasExplicitReturn;
-                var isIIFE = containerFlags & 16 /* IsFunctionExpression */ && !ts.hasModifier(node, 256 /* Async */) &&
-                    !node.asteriskToken && !!ts.getImmediatelyInvokedFunctionExpression(node);
-                // A non-async, non-generator IIFE is considered part of the containing control flow. Return statements behave
-                // similarly to break statements that exit to a label just past the statement body.
-                if (!isIIFE) {
-                    currentFlow = initFlowNode({ flags: 2 /* Start */ });
-                    if (containerFlags & (16 /* IsFunctionExpression */ | 128 /* IsObjectLiteralOrClassExpressionMethod */)) {
-                        currentFlow.node = node;
+            if (pragma.args) {
+                var argument = {};
+                for (var _i = 0, _a = pragma.args; _i < _a.length; _i++) {
+                    var arg = _a[_i];
+                    var matcher = getNamedArgRegEx(arg.name);
+                    var matchResult = matcher.exec(text);
+                    if (!matchResult && !arg.optional) {
+                        return; // Missing required argument, don't parse
                     }
-                }
-                // We create a return control flow graph for IIFEs and constructors. For constructors
-                // we use the return control flow graph in strict property initialization checks.
-                currentReturnTarget = isIIFE || node.kind === 162 /* Constructor */ ? createBranchLabel() : undefined;
-                currentExceptionTarget = undefined;
-                currentBreakTarget = undefined;
-                currentContinueTarget = undefined;
-                activeLabelList = undefined;
-                hasExplicitReturn = false;
-                bindChildren(node);
-                // Reset all reachability check related flags on node (for incremental scenarios)
-                node.flags &= ~2816 /* ReachabilityAndEmitFlags */;
-                if (!(currentFlow.flags & 1 /* Unreachable */) && containerFlags & 8 /* IsFunctionLike */ && ts.nodeIsPresent(node.body)) {
-                    node.flags |= 256 /* HasImplicitReturn */;
-                    if (hasExplicitReturn)
-                        node.flags |= 512 /* HasExplicitReturn */;
-                    node.endFlowNode = currentFlow;
-                }
-                if (node.kind === 290 /* SourceFile */) {
-                    node.flags |= emitFlags;
-                }
-                if (currentReturnTarget) {
-                    addAntecedent(currentReturnTarget, currentFlow);
-                    currentFlow = finishFlowLabel(currentReturnTarget);
-                    if (node.kind === 162 /* Constructor */) {
-                        node.returnFlowNode = currentFlow;
+                    else if (matchResult) {
+                        if (arg.captureSpan) {
+                            var startPos = range.pos + matchResult.index + matchResult[1].length + matchResult[2].length;
+                            argument[arg.name] = {
+                                value: matchResult[3],
+                                pos: startPos,
+                                end: startPos + matchResult[3].length
+                            };
+                        }
+                        else {
+                            argument[arg.name] = matchResult[3];
+                        }
                     }
                 }
-                if (!isIIFE) {
-                    currentFlow = saveCurrentFlow;
-                }
-                currentBreakTarget = saveBreakTarget;
-                currentContinueTarget = saveContinueTarget;
-                currentReturnTarget = saveReturnTarget;
-                currentExceptionTarget = saveExceptionTarget;
-                activeLabelList = saveActiveLabelList;
-                hasExplicitReturn = saveHasExplicitReturn;
+                pragmas.push({ name: name, args: { arguments: argument, range: range } });
             }
-            else if (containerFlags & 64 /* IsInterface */) {
-                seenThisKeyword = false;
-                bindChildren(node);
-                node.flags = seenThisKeyword ? node.flags | 128 /* ContainsThis */ : node.flags & ~128 /* ContainsThis */;
+            else {
+                pragmas.push({ name: name, args: { arguments: {}, range: range } });
+            }
+            return;
+        }
+        var singleLine = range.kind === 2 /* SingleLineCommentTrivia */ && singleLinePragmaRegEx.exec(text);
+        if (singleLine) {
+            return addPragmaForMatch(pragmas, range, 2 /* SingleLine */, singleLine);
+        }
+        if (range.kind === 3 /* MultiLineCommentTrivia */) {
+            var multiLinePragmaRegEx = /\s*@(\S+)\s*(.*)\s*$/gim; // Defined inline since it uses the "g" flag, which keeps a persistent index (for iterating)
+            var multiLineMatch = void 0;
+            while (multiLineMatch = multiLinePragmaRegEx.exec(text)) {
+                addPragmaForMatch(pragmas, range, 4 /* MultiLine */, multiLineMatch);
+            }
+        }
+    }
+    function addPragmaForMatch(pragmas, range, kind, match) {
+        if (!match)
+            return;
+        var name = match[1].toLowerCase(); // Technically unsafe cast, but we do it so they below check to make it safe typechecks
+        var pragma = ts.commentPragmas[name];
+        if (!pragma || !(pragma.kind & kind)) {
+            return;
+        }
+        var args = match[2]; // Split on spaces and match up positionally with definition
+        var argument = getNamedPragmaArguments(pragma, args);
+        if (argument === "fail")
+            return; // Missing required argument, fail to parse it
+        pragmas.push({ name: name, args: { arguments: argument, range: range } });
+        return;
+    }
+    function getNamedPragmaArguments(pragma, text) {
+        if (!text)
+            return {};
+        if (!pragma.args)
+            return {};
+        var args = text.split(/\s+/);
+        var argMap = {};
+        for (var i = 0; i < pragma.args.length; i++) {
+            var argument = pragma.args[i];
+            if (!args[i] && !argument.optional) {
+                return "fail";
             }
-            else {
-                bindChildren(node);
+            if (argument.captureSpan) {
+                return ts.Debug.fail("Capture spans not yet implemented for non-xml pragmas");
             }
-            container = saveContainer;
-            thisParentContainer = saveThisParentContainer;
-            blockScopeContainer = savedBlockScopeContainer;
+            argMap[argument.name] = args[i];
+        }
+        return argMap;
+    }
+    /** @internal */
+    function tagNamesAreEquivalent(lhs, rhs) {
+        if (lhs.kind !== rhs.kind) {
+            return false;
+        }
+        if (lhs.kind === 78 /* Identifier */) {
+            return lhs.escapedText === rhs.escapedText;
+        }
+        if (lhs.kind === 107 /* ThisKeyword */) {
+            return true;
+        }
+        // If we are at this statement then we must have PropertyAccessExpression and because tag name in Jsx element can only
+        // take forms of JsxTagNameExpression which includes an identifier, "this" expression, or another propertyAccessExpression
+        // it is safe to case the expression property as such. See parseJsxElementName for how we parse tag name in Jsx element
+        return lhs.name.escapedText === rhs.name.escapedText &&
+            tagNamesAreEquivalent(lhs.expression, rhs.expression);
+    }
+    ts.tagNamesAreEquivalent = tagNamesAreEquivalent;
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    /* @internal */
+    ts.compileOnSaveCommandLineOption = { name: "compileOnSave", type: "boolean" };
+    var jsxOptionMap = new ts.Map(ts.getEntries({
+        "preserve": 1 /* Preserve */,
+        "react-native": 3 /* ReactNative */,
+        "react": 2 /* React */,
+        "react-jsx": 4 /* ReactJSX */,
+        "react-jsxdev": 5 /* ReactJSXDev */,
+    }));
+    /* @internal */
+    ts.inverseJsxOptionMap = new ts.Map(ts.arrayFrom(ts.mapIterator(jsxOptionMap.entries(), function (_a) {
+        var key = _a[0], value = _a[1];
+        return ["" + value, key];
+    })));
+    // NOTE: The order here is important to default lib ordering as entries will have the same
+    //       order in the generated program (see `getDefaultLibPriority` in program.ts). This
+    //       order also affects overload resolution when a type declared in one lib is
+    //       augmented in another lib.
+    var libEntries = [
+        // JavaScript only
+        ["es5", "lib.es5.d.ts"],
+        ["es6", "lib.es2015.d.ts"],
+        ["es2015", "lib.es2015.d.ts"],
+        ["es7", "lib.es2016.d.ts"],
+        ["es2016", "lib.es2016.d.ts"],
+        ["es2017", "lib.es2017.d.ts"],
+        ["es2018", "lib.es2018.d.ts"],
+        ["es2019", "lib.es2019.d.ts"],
+        ["es2020", "lib.es2020.d.ts"],
+        ["esnext", "lib.esnext.d.ts"],
+        // Host only
+        ["dom", "lib.dom.d.ts"],
+        ["dom.iterable", "lib.dom.iterable.d.ts"],
+        ["webworker", "lib.webworker.d.ts"],
+        ["webworker.importscripts", "lib.webworker.importscripts.d.ts"],
+        ["webworker.iterable", "lib.webworker.iterable.d.ts"],
+        ["scripthost", "lib.scripthost.d.ts"],
+        // ES2015 Or ESNext By-feature options
+        ["es2015.core", "lib.es2015.core.d.ts"],
+        ["es2015.collection", "lib.es2015.collection.d.ts"],
+        ["es2015.generator", "lib.es2015.generator.d.ts"],
+        ["es2015.iterable", "lib.es2015.iterable.d.ts"],
+        ["es2015.promise", "lib.es2015.promise.d.ts"],
+        ["es2015.proxy", "lib.es2015.proxy.d.ts"],
+        ["es2015.reflect", "lib.es2015.reflect.d.ts"],
+        ["es2015.symbol", "lib.es2015.symbol.d.ts"],
+        ["es2015.symbol.wellknown", "lib.es2015.symbol.wellknown.d.ts"],
+        ["es2016.array.include", "lib.es2016.array.include.d.ts"],
+        ["es2017.object", "lib.es2017.object.d.ts"],
+        ["es2017.sharedmemory", "lib.es2017.sharedmemory.d.ts"],
+        ["es2017.string", "lib.es2017.string.d.ts"],
+        ["es2017.intl", "lib.es2017.intl.d.ts"],
+        ["es2017.typedarrays", "lib.es2017.typedarrays.d.ts"],
+        ["es2018.asyncgenerator", "lib.es2018.asyncgenerator.d.ts"],
+        ["es2018.asynciterable", "lib.es2018.asynciterable.d.ts"],
+        ["es2018.intl", "lib.es2018.intl.d.ts"],
+        ["es2018.promise", "lib.es2018.promise.d.ts"],
+        ["es2018.regexp", "lib.es2018.regexp.d.ts"],
+        ["es2019.array", "lib.es2019.array.d.ts"],
+        ["es2019.object", "lib.es2019.object.d.ts"],
+        ["es2019.string", "lib.es2019.string.d.ts"],
+        ["es2019.symbol", "lib.es2019.symbol.d.ts"],
+        ["es2020.bigint", "lib.es2020.bigint.d.ts"],
+        ["es2020.promise", "lib.es2020.promise.d.ts"],
+        ["es2020.sharedmemory", "lib.es2020.sharedmemory.d.ts"],
+        ["es2020.string", "lib.es2020.string.d.ts"],
+        ["es2020.symbol.wellknown", "lib.es2020.symbol.wellknown.d.ts"],
+        ["es2020.intl", "lib.es2020.intl.d.ts"],
+        ["esnext.array", "lib.es2019.array.d.ts"],
+        ["esnext.symbol", "lib.es2019.symbol.d.ts"],
+        ["esnext.asynciterable", "lib.es2018.asynciterable.d.ts"],
+        ["esnext.intl", "lib.esnext.intl.d.ts"],
+        ["esnext.bigint", "lib.es2020.bigint.d.ts"],
+        ["esnext.string", "lib.esnext.string.d.ts"],
+        ["esnext.promise", "lib.esnext.promise.d.ts"],
+        ["esnext.weakref", "lib.esnext.weakref.d.ts"]
+    ];
+    /**
+     * An array of supported "lib" reference file names used to determine the order for inclusion
+     * when referenced, as well as for spelling suggestions. This ensures the correct ordering for
+     * overload resolution when a type declared in one lib is extended by another.
+     */
+    /* @internal */
+    ts.libs = libEntries.map(function (entry) { return entry[0]; });
+    /**
+     * A map of lib names to lib files. This map is used both for parsing the "lib" command line
+     * option as well as for resolving lib reference directives.
+     */
+    /* @internal */
+    ts.libMap = new ts.Map(libEntries);
+    // Watch related options
+    /* @internal */
+    ts.optionsForWatch = [
+        {
+            name: "watchFile",
+            type: new ts.Map(ts.getEntries({
+                fixedpollinginterval: ts.WatchFileKind.FixedPollingInterval,
+                prioritypollinginterval: ts.WatchFileKind.PriorityPollingInterval,
+                dynamicprioritypolling: ts.WatchFileKind.DynamicPriorityPolling,
+                usefsevents: ts.WatchFileKind.UseFsEvents,
+                usefseventsonparentdirectory: ts.WatchFileKind.UseFsEventsOnParentDirectory,
+            })),
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_strategy_for_watching_file_Colon_FixedPollingInterval_default_PriorityPollingInterval_DynamicPriorityPolling_UseFsEvents_UseFsEventsOnParentDirectory,
+        },
+        {
+            name: "watchDirectory",
+            type: new ts.Map(ts.getEntries({
+                usefsevents: ts.WatchDirectoryKind.UseFsEvents,
+                fixedpollinginterval: ts.WatchDirectoryKind.FixedPollingInterval,
+                dynamicprioritypolling: ts.WatchDirectoryKind.DynamicPriorityPolling,
+            })),
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_strategy_for_watching_directory_on_platforms_that_don_t_support_recursive_watching_natively_Colon_UseFsEvents_default_FixedPollingInterval_DynamicPriorityPolling,
+        },
+        {
+            name: "fallbackPolling",
+            type: new ts.Map(ts.getEntries({
+                fixedinterval: ts.PollingWatchKind.FixedInterval,
+                priorityinterval: ts.PollingWatchKind.PriorityInterval,
+                dynamicpriority: ts.PollingWatchKind.DynamicPriority,
+            })),
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_strategy_for_creating_a_polling_watch_when_it_fails_to_create_using_file_system_events_Colon_FixedInterval_default_PriorityInterval_DynamicPriority,
+        },
+        {
+            name: "synchronousWatchDirectory",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Synchronously_call_callbacks_and_update_the_state_of_directory_watchers_on_platforms_that_don_t_support_recursive_watching_natively,
+        },
+    ];
+    /* @internal */
+    ts.commonOptionsWithBuild = [
+        {
+            name: "help",
+            shortName: "h",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Print_this_message,
+        },
+        {
+            name: "help",
+            shortName: "?",
+            type: "boolean"
+        },
+        {
+            name: "watch",
+            shortName: "w",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Watch_input_files,
+        },
+        {
+            name: "preserveWatchOutput",
+            type: "boolean",
+            showInSimplifiedHelpView: false,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Whether_to_keep_outdated_console_output_in_watch_mode_instead_of_clearing_the_screen,
+        },
+        {
+            name: "listFiles",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Print_names_of_files_part_of_the_compilation
+        },
+        {
+            name: "listEmittedFiles",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Print_names_of_generated_files_part_of_the_compilation
+        },
+        {
+            name: "pretty",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Stylize_errors_and_messages_using_color_and_context_experimental
+        },
+        {
+            name: "traceResolution",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Enable_tracing_of_the_name_resolution_process
+        },
+        {
+            name: "diagnostics",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Show_diagnostic_information
+        },
+        {
+            name: "extendedDiagnostics",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Show_verbose_diagnostic_information
+        },
+        {
+            name: "generateCpuProfile",
+            type: "string",
+            isFilePath: true,
+            paramType: ts.Diagnostics.FILE_OR_DIRECTORY,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Generates_a_CPU_profile
+        },
+        {
+            name: "generateTrace",
+            type: "string",
+            isFilePath: true,
+            isCommandLineOnly: true,
+            paramType: ts.Diagnostics.DIRECTORY,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Generates_an_event_trace_and_a_list_of_types
+        },
+        {
+            name: "incremental",
+            shortName: "i",
+            type: "boolean",
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Enable_incremental_compilation,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "assumeChangesOnlyAffectDirectDependencies",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Have_recompiles_in_incremental_and_watch_assume_that_changes_within_a_file_will_only_affect_files_directly_depending_on_it
+        },
+        {
+            name: "locale",
+            type: "string",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.The_locale_used_when_displaying_messages_to_the_user_e_g_en_us
+        },
+    ];
+    /* @internal */
+    ts.optionDeclarations = __spreadArrays(ts.commonOptionsWithBuild, [
+        {
+            name: "all",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Show_all_compiler_options,
+        },
+        {
+            name: "version",
+            shortName: "v",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Print_the_compiler_s_version,
+        },
+        {
+            name: "init",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Initializes_a_TypeScript_project_and_creates_a_tsconfig_json_file,
+        },
+        {
+            name: "project",
+            shortName: "p",
+            type: "string",
+            isFilePath: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            paramType: ts.Diagnostics.FILE_OR_DIRECTORY,
+            description: ts.Diagnostics.Compile_the_project_given_the_path_to_its_configuration_file_or_to_a_folder_with_a_tsconfig_json,
+        },
+        {
+            name: "build",
+            type: "boolean",
+            shortName: "b",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Build_one_or_more_projects_and_their_dependencies_if_out_of_date
+        },
+        {
+            name: "showConfig",
+            type: "boolean",
+            category: ts.Diagnostics.Command_line_Options,
+            isCommandLineOnly: true,
+            description: ts.Diagnostics.Print_the_final_configuration_instead_of_building
+        },
+        {
+            name: "listFilesOnly",
+            type: "boolean",
+            category: ts.Diagnostics.Command_line_Options,
+            affectsSemanticDiagnostics: true,
+            affectsEmit: true,
+            isCommandLineOnly: true,
+            description: ts.Diagnostics.Print_names_of_files_that_are_part_of_the_compilation_and_then_stop_processing
+        },
+        // Basic
+        {
+            name: "target",
+            shortName: "t",
+            type: new ts.Map(ts.getEntries({
+                es3: 0 /* ES3 */,
+                es5: 1 /* ES5 */,
+                es6: 2 /* ES2015 */,
+                es2015: 2 /* ES2015 */,
+                es2016: 3 /* ES2016 */,
+                es2017: 4 /* ES2017 */,
+                es2018: 5 /* ES2018 */,
+                es2019: 6 /* ES2019 */,
+                es2020: 7 /* ES2020 */,
+                esnext: 99 /* ESNext */,
+            })),
+            affectsSourceFile: true,
+            affectsModuleResolution: true,
+            affectsEmit: true,
+            paramType: ts.Diagnostics.VERSION,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Specify_ECMAScript_target_version_Colon_ES3_default_ES5_ES2015_ES2016_ES2017_ES2018_ES2019_ES2020_or_ESNEXT,
+        },
+        {
+            name: "module",
+            shortName: "m",
+            type: new ts.Map(ts.getEntries({
+                none: ts.ModuleKind.None,
+                commonjs: ts.ModuleKind.CommonJS,
+                amd: ts.ModuleKind.AMD,
+                system: ts.ModuleKind.System,
+                umd: ts.ModuleKind.UMD,
+                es6: ts.ModuleKind.ES2015,
+                es2015: ts.ModuleKind.ES2015,
+                es2020: ts.ModuleKind.ES2020,
+                esnext: ts.ModuleKind.ESNext
+            })),
+            affectsModuleResolution: true,
+            affectsEmit: true,
+            paramType: ts.Diagnostics.KIND,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Specify_module_code_generation_Colon_none_commonjs_amd_system_umd_es2015_es2020_or_ESNext,
+        },
+        {
+            name: "lib",
+            type: "list",
+            element: {
+                name: "lib",
+                type: ts.libMap
+            },
+            affectsModuleResolution: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Specify_library_files_to_be_included_in_the_compilation,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "allowJs",
+            type: "boolean",
+            affectsModuleResolution: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Allow_javascript_files_to_be_compiled
+        },
+        {
+            name: "checkJs",
+            type: "boolean",
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Report_errors_in_js_files
+        },
+        {
+            name: "jsx",
+            type: jsxOptionMap,
+            affectsSourceFile: true,
+            affectsEmit: true,
+            affectsModuleResolution: true,
+            paramType: ts.Diagnostics.KIND,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Specify_JSX_code_generation_Colon_preserve_react_native_or_react,
+        },
+        {
+            name: "declaration",
+            shortName: "d",
+            type: "boolean",
+            affectsEmit: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Generates_corresponding_d_ts_file,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "declarationMap",
+            type: "boolean",
+            affectsEmit: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Generates_a_sourcemap_for_each_corresponding_d_ts_file,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "emitDeclarationOnly",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Only_emit_d_ts_declaration_files,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "sourceMap",
+            type: "boolean",
+            affectsEmit: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Generates_corresponding_map_file,
+        },
+        {
+            name: "outFile",
+            type: "string",
+            affectsEmit: true,
+            isFilePath: true,
+            paramType: ts.Diagnostics.FILE,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Concatenate_and_emit_output_to_single_file,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "outDir",
+            type: "string",
+            affectsEmit: true,
+            isFilePath: true,
+            paramType: ts.Diagnostics.DIRECTORY,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Redirect_output_structure_to_the_directory,
+        },
+        {
+            name: "rootDir",
+            type: "string",
+            affectsEmit: true,
+            isFilePath: true,
+            paramType: ts.Diagnostics.LOCATION,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Specify_the_root_directory_of_input_files_Use_to_control_the_output_directory_structure_with_outDir,
+        },
+        {
+            name: "composite",
+            type: "boolean",
+            affectsEmit: true,
+            isTSConfigOnly: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Enable_project_compilation,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "tsBuildInfoFile",
+            type: "string",
+            affectsEmit: true,
+            isFilePath: true,
+            paramType: ts.Diagnostics.FILE,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Specify_file_to_store_incremental_compilation_information,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "removeComments",
+            type: "boolean",
+            affectsEmit: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Do_not_emit_comments_to_output,
+        },
+        {
+            name: "noEmit",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Do_not_emit_outputs,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "importHelpers",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Import_emit_helpers_from_tslib
+        },
+        {
+            name: "importsNotUsedAsValues",
+            type: new ts.Map(ts.getEntries({
+                remove: 0 /* Remove */,
+                preserve: 1 /* Preserve */,
+                error: 2 /* Error */
+            })),
+            affectsEmit: true,
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_emit_Slashchecking_behavior_for_imports_that_are_only_used_for_types
+        },
+        {
+            name: "downlevelIteration",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Provide_full_support_for_iterables_in_for_of_spread_and_destructuring_when_targeting_ES5_or_ES3
+        },
+        {
+            name: "isolatedModules",
+            type: "boolean",
+            category: ts.Diagnostics.Basic_Options,
+            description: ts.Diagnostics.Transpile_each_file_as_a_separate_module_similar_to_ts_transpileModule,
+            transpileOptionValue: true
+        },
+        // Strict Type Checks
+        {
+            name: "strict",
+            type: "boolean",
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Enable_all_strict_type_checking_options
+        },
+        {
+            name: "noImplicitAny",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Raise_error_on_expressions_and_declarations_with_an_implied_any_type
+        },
+        {
+            name: "strictNullChecks",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Enable_strict_null_checks
+        },
+        {
+            name: "strictFunctionTypes",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Enable_strict_checking_of_function_types
+        },
+        {
+            name: "strictBindCallApply",
+            type: "boolean",
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Enable_strict_bind_call_and_apply_methods_on_functions
+        },
+        {
+            name: "strictPropertyInitialization",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Enable_strict_checking_of_property_initialization_in_classes
+        },
+        {
+            name: "noImplicitThis",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Raise_error_on_this_expressions_with_an_implied_any_type,
+        },
+        {
+            name: "alwaysStrict",
+            type: "boolean",
+            affectsSourceFile: true,
+            strictFlag: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Strict_Type_Checking_Options,
+            description: ts.Diagnostics.Parse_in_strict_mode_and_emit_use_strict_for_each_source_file
+        },
+        // Additional Checks
+        {
+            name: "noUnusedLocals",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Additional_Checks,
+            description: ts.Diagnostics.Report_errors_on_unused_locals,
+        },
+        {
+            name: "noUnusedParameters",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Additional_Checks,
+            description: ts.Diagnostics.Report_errors_on_unused_parameters,
+        },
+        {
+            name: "noImplicitReturns",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Additional_Checks,
+            description: ts.Diagnostics.Report_error_when_not_all_code_paths_in_function_return_a_value
+        },
+        {
+            name: "noFallthroughCasesInSwitch",
+            type: "boolean",
+            affectsBindDiagnostics: true,
+            affectsSemanticDiagnostics: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Additional_Checks,
+            description: ts.Diagnostics.Report_errors_for_fallthrough_cases_in_switch_statement
+        },
+        {
+            name: "noUncheckedIndexedAccess",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            showInSimplifiedHelpView: false,
+            category: ts.Diagnostics.Additional_Checks,
+            description: ts.Diagnostics.Include_undefined_in_index_signature_results
+        },
+        // Module Resolution
+        {
+            name: "moduleResolution",
+            type: new ts.Map(ts.getEntries({
+                node: ts.ModuleResolutionKind.NodeJs,
+                classic: ts.ModuleResolutionKind.Classic,
+            })),
+            affectsModuleResolution: true,
+            paramType: ts.Diagnostics.STRATEGY,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Specify_module_resolution_strategy_Colon_node_Node_js_or_classic_TypeScript_pre_1_6,
+        },
+        {
+            name: "baseUrl",
+            type: "string",
+            affectsModuleResolution: true,
+            isFilePath: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Base_directory_to_resolve_non_absolute_module_names
+        },
+        {
+            // this option can only be specified in tsconfig.json
+            // use type = object to copy the value as-is
+            name: "paths",
+            type: "object",
+            affectsModuleResolution: true,
+            isTSConfigOnly: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.A_series_of_entries_which_re_map_imports_to_lookup_locations_relative_to_the_baseUrl,
+            transpileOptionValue: undefined
+        },
+        {
+            // this option can only be specified in tsconfig.json
+            // use type = object to copy the value as-is
+            name: "rootDirs",
+            type: "list",
+            isTSConfigOnly: true,
+            element: {
+                name: "rootDirs",
+                type: "string",
+                isFilePath: true
+            },
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.List_of_root_folders_whose_combined_content_represents_the_structure_of_the_project_at_runtime,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "typeRoots",
+            type: "list",
+            element: {
+                name: "typeRoots",
+                type: "string",
+                isFilePath: true
+            },
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.List_of_folders_to_include_type_definitions_from
+        },
+        {
+            name: "types",
+            type: "list",
+            element: {
+                name: "types",
+                type: "string"
+            },
+            affectsModuleResolution: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Type_declaration_files_to_be_included_in_compilation,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "allowSyntheticDefaultImports",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Allow_default_imports_from_modules_with_no_default_export_This_does_not_affect_code_emit_just_typechecking
+        },
+        {
+            name: "esModuleInterop",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            affectsEmit: true,
+            showInSimplifiedHelpView: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Enables_emit_interoperability_between_CommonJS_and_ES_Modules_via_creation_of_namespace_objects_for_all_imports_Implies_allowSyntheticDefaultImports
+        },
+        {
+            name: "preserveSymlinks",
+            type: "boolean",
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Do_not_resolve_the_real_path_of_symlinks,
+        },
+        {
+            name: "allowUmdGlobalAccess",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Module_Resolution_Options,
+            description: ts.Diagnostics.Allow_accessing_UMD_globals_from_modules,
+        },
+        // Source Maps
+        {
+            name: "sourceRoot",
+            type: "string",
+            affectsEmit: true,
+            paramType: ts.Diagnostics.LOCATION,
+            category: ts.Diagnostics.Source_Map_Options,
+            description: ts.Diagnostics.Specify_the_location_where_debugger_should_locate_TypeScript_files_instead_of_source_locations,
+        },
+        {
+            name: "mapRoot",
+            type: "string",
+            affectsEmit: true,
+            paramType: ts.Diagnostics.LOCATION,
+            category: ts.Diagnostics.Source_Map_Options,
+            description: ts.Diagnostics.Specify_the_location_where_debugger_should_locate_map_files_instead_of_generated_locations,
+        },
+        {
+            name: "inlineSourceMap",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Source_Map_Options,
+            description: ts.Diagnostics.Emit_a_single_file_with_source_maps_instead_of_having_a_separate_file
+        },
+        {
+            name: "inlineSources",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Source_Map_Options,
+            description: ts.Diagnostics.Emit_the_source_alongside_the_sourcemaps_within_a_single_file_requires_inlineSourceMap_or_sourceMap_to_be_set
+        },
+        // Experimental
+        {
+            name: "experimentalDecorators",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Experimental_Options,
+            description: ts.Diagnostics.Enables_experimental_support_for_ES7_decorators
+        },
+        {
+            name: "emitDecoratorMetadata",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            affectsEmit: true,
+            category: ts.Diagnostics.Experimental_Options,
+            description: ts.Diagnostics.Enables_experimental_support_for_emitting_type_metadata_for_decorators
+        },
+        // Advanced
+        {
+            name: "jsxFactory",
+            type: "string",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_the_JSX_factory_function_to_use_when_targeting_react_JSX_emit_e_g_React_createElement_or_h
+        },
+        {
+            name: "jsxFragmentFactory",
+            type: "string",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_the_JSX_fragment_factory_function_to_use_when_targeting_react_JSX_emit_with_jsxFactory_compiler_option_is_specified_e_g_Fragment
+        },
+        {
+            name: "jsxImportSource",
+            type: "string",
+            affectsSemanticDiagnostics: true,
+            affectsEmit: true,
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_the_module_specifier_to_be_used_to_import_the_jsx_and_jsxs_factory_functions_from_eg_react
+        },
+        {
+            name: "resolveJsonModule",
+            type: "boolean",
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Include_modules_imported_with_json_extension
+        },
+        {
+            name: "out",
+            type: "string",
+            affectsEmit: true,
+            isFilePath: false,
+            // for correct behaviour, please use outFile
+            category: ts.Diagnostics.Advanced_Options,
+            paramType: ts.Diagnostics.FILE,
+            description: ts.Diagnostics.Deprecated_Use_outFile_instead_Concatenate_and_emit_output_to_single_file,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "reactNamespace",
+            type: "string",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Deprecated_Use_jsxFactory_instead_Specify_the_object_invoked_for_createElement_when_targeting_react_JSX_emit
+        },
+        {
+            name: "skipDefaultLibCheck",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Deprecated_Use_skipLibCheck_instead_Skip_type_checking_of_default_library_declaration_files
+        },
+        {
+            name: "charset",
+            type: "string",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.The_character_set_of_the_input_files
+        },
+        {
+            name: "emitBOM",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Emit_a_UTF_8_Byte_Order_Mark_BOM_in_the_beginning_of_output_files
+        },
+        {
+            name: "newLine",
+            type: new ts.Map(ts.getEntries({
+                crlf: 0 /* CarriageReturnLineFeed */,
+                lf: 1 /* LineFeed */
+            })),
+            affectsEmit: true,
+            paramType: ts.Diagnostics.NEWLINE,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Specify_the_end_of_line_sequence_to_be_used_when_emitting_files_Colon_CRLF_dos_or_LF_unix,
+        },
+        {
+            name: "noErrorTruncation",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_truncate_error_messages
+        },
+        {
+            name: "noLib",
+            type: "boolean",
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_include_the_default_library_file_lib_d_ts,
+            // We are not returning a sourceFile for lib file when asked by the program,
+            // so pass --noLib to avoid reporting a file not found error.
+            transpileOptionValue: true
+        },
+        {
+            name: "noResolve",
+            type: "boolean",
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_add_triple_slash_references_or_imported_modules_to_the_list_of_compiled_files,
+            // We are not doing a full typecheck, we are not resolving the whole context,
+            // so pass --noResolve to avoid reporting missing file errors.
+            transpileOptionValue: true
+        },
+        {
+            name: "stripInternal",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_emit_declarations_for_code_that_has_an_internal_annotation,
+        },
+        {
+            name: "disableSizeLimit",
+            type: "boolean",
+            affectsSourceFile: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Disable_size_limitations_on_JavaScript_projects
+        },
+        {
+            name: "disableSourceOfProjectReferenceRedirect",
+            type: "boolean",
+            isTSConfigOnly: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Disable_use_of_source_files_instead_of_declaration_files_from_referenced_projects
+        },
+        {
+            name: "disableSolutionSearching",
+            type: "boolean",
+            isTSConfigOnly: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Disable_solution_searching_for_this_project
+        },
+        {
+            name: "disableReferencedProjectLoad",
+            type: "boolean",
+            isTSConfigOnly: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Disable_loading_referenced_projects
+        },
+        {
+            name: "noImplicitUseStrict",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_emit_use_strict_directives_in_module_output
+        },
+        {
+            name: "noEmitHelpers",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_generate_custom_helper_functions_like_extends_in_compiled_output
+        },
+        {
+            name: "noEmitOnError",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_emit_outputs_if_any_errors_were_reported,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "preserveConstEnums",
+            type: "boolean",
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_erase_const_enum_declarations_in_generated_code
+        },
+        {
+            name: "declarationDir",
+            type: "string",
+            affectsEmit: true,
+            isFilePath: true,
+            paramType: ts.Diagnostics.DIRECTORY,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Output_directory_for_generated_declaration_files,
+            transpileOptionValue: undefined
+        },
+        {
+            name: "skipLibCheck",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Skip_type_checking_of_declaration_files,
+        },
+        {
+            name: "allowUnusedLabels",
+            type: "boolean",
+            affectsBindDiagnostics: true,
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_report_errors_on_unused_labels
+        },
+        {
+            name: "allowUnreachableCode",
+            type: "boolean",
+            affectsBindDiagnostics: true,
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Do_not_report_errors_on_unreachable_code
+        },
+        {
+            name: "suppressExcessPropertyErrors",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Suppress_excess_property_checks_for_object_literals,
+        },
+        {
+            name: "suppressImplicitAnyIndexErrors",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Suppress_noImplicitAny_errors_for_indexing_objects_lacking_index_signatures,
+        },
+        {
+            name: "forceConsistentCasingInFileNames",
+            type: "boolean",
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Disallow_inconsistently_cased_references_to_the_same_file
+        },
+        {
+            name: "maxNodeModuleJsDepth",
+            type: "number",
+            affectsModuleResolution: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.The_maximum_dependency_depth_to_search_under_node_modules_and_load_JavaScript_files
+        },
+        {
+            name: "noStrictGenericChecks",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Disable_strict_checking_of_generic_signatures_in_function_types,
+        },
+        {
+            name: "useDefineForClassFields",
+            type: "boolean",
+            affectsSemanticDiagnostics: true,
+            affectsEmit: true,
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Emit_class_fields_with_Define_instead_of_Set,
+        },
+        {
+            name: "keyofStringsOnly",
+            type: "boolean",
+            category: ts.Diagnostics.Advanced_Options,
+            description: ts.Diagnostics.Resolve_keyof_to_string_valued_property_names_only_no_numbers_or_symbols,
+        },
+        {
+            // A list of plugins to load in the language service
+            name: "plugins",
+            type: "list",
+            isTSConfigOnly: true,
+            element: {
+                name: "plugin",
+                type: "object"
+            },
+            description: ts.Diagnostics.List_of_language_service_plugins
+        },
+    ]);
+    /* @internal */
+    ts.semanticDiagnosticsOptionDeclarations = ts.optionDeclarations.filter(function (option) { return !!option.affectsSemanticDiagnostics; });
+    /* @internal */
+    ts.affectsEmitOptionDeclarations = ts.optionDeclarations.filter(function (option) { return !!option.affectsEmit; });
+    /* @internal */
+    ts.moduleResolutionOptionDeclarations = ts.optionDeclarations.filter(function (option) { return !!option.affectsModuleResolution; });
+    /* @internal */
+    ts.sourceFileAffectingCompilerOptions = ts.optionDeclarations.filter(function (option) {
+        return !!option.affectsSourceFile || !!option.affectsModuleResolution || !!option.affectsBindDiagnostics;
+    });
+    /* @internal */
+    ts.transpileOptionValueCompilerOptions = ts.optionDeclarations.filter(function (option) {
+        return ts.hasProperty(option, "transpileOptionValue");
+    });
+    /* @internal */
+    ts.buildOpts = __spreadArrays(ts.commonOptionsWithBuild, [
+        {
+            name: "verbose",
+            shortName: "v",
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Enable_verbose_logging,
+            type: "boolean"
+        },
+        {
+            name: "dry",
+            shortName: "d",
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Show_what_would_be_built_or_deleted_if_specified_with_clean,
+            type: "boolean"
+        },
+        {
+            name: "force",
+            shortName: "f",
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Build_all_projects_including_those_that_appear_to_be_up_to_date,
+            type: "boolean"
+        },
+        {
+            name: "clean",
+            category: ts.Diagnostics.Command_line_Options,
+            description: ts.Diagnostics.Delete_the_outputs_of_all_projects,
+            type: "boolean"
         }
-        function bindChildren(node) {
-            if (skipTransformFlagAggregation) {
-                bindChildrenWorker(node);
+    ]);
+    /* @internal */
+    ts.typeAcquisitionDeclarations = [
+        {
+            /* @deprecated typingOptions.enableAutoDiscovery
+             * Use typeAcquisition.enable instead.
+             */
+            name: "enableAutoDiscovery",
+            type: "boolean",
+        },
+        {
+            name: "enable",
+            type: "boolean",
+        },
+        {
+            name: "include",
+            type: "list",
+            element: {
+                name: "include",
+                type: "string"
             }
-            else if (node.transformFlags & 536870912 /* HasComputedFlags */) {
-                skipTransformFlagAggregation = true;
-                bindChildrenWorker(node);
-                skipTransformFlagAggregation = false;
-                subtreeTransformFlags |= node.transformFlags & ~getTransformFlagsSubtreeExclusions(node.kind);
+        },
+        {
+            name: "exclude",
+            type: "list",
+            element: {
+                name: "exclude",
+                type: "string"
             }
-            else {
-                var savedSubtreeTransformFlags = subtreeTransformFlags;
-                subtreeTransformFlags = 0;
-                bindChildrenWorker(node);
-                subtreeTransformFlags = savedSubtreeTransformFlags | computeTransformFlagsForNode(node, subtreeTransformFlags);
+        },
+        {
+            name: "disableFilenameBasedTypeAcquisition",
+            type: "boolean",
+        },
+    ];
+    /*@internal*/
+    function createOptionNameMap(optionDeclarations) {
+        var optionsNameMap = new ts.Map();
+        var shortOptionNames = new ts.Map();
+        ts.forEach(optionDeclarations, function (option) {
+            optionsNameMap.set(option.name.toLowerCase(), option);
+            if (option.shortName) {
+                shortOptionNames.set(option.shortName, option.name);
             }
+        });
+        return { optionsNameMap: optionsNameMap, shortOptionNames: shortOptionNames };
+    }
+    ts.createOptionNameMap = createOptionNameMap;
+    var optionsNameMapCache;
+    /* @internal */
+    function getOptionsNameMap() {
+        return optionsNameMapCache || (optionsNameMapCache = createOptionNameMap(ts.optionDeclarations));
+    }
+    ts.getOptionsNameMap = getOptionsNameMap;
+    /* @internal */
+    ts.defaultInitCompilerOptions = {
+        module: ts.ModuleKind.CommonJS,
+        target: 1 /* ES5 */,
+        strict: true,
+        esModuleInterop: true,
+        forceConsistentCasingInFileNames: true,
+        skipLibCheck: true
+    };
+    /* @internal */
+    function convertEnableAutoDiscoveryToEnable(typeAcquisition) {
+        // Convert deprecated typingOptions.enableAutoDiscovery to typeAcquisition.enable
+        if (typeAcquisition && typeAcquisition.enableAutoDiscovery !== undefined && typeAcquisition.enable === undefined) {
+            return {
+                enable: typeAcquisition.enableAutoDiscovery,
+                include: typeAcquisition.include || [],
+                exclude: typeAcquisition.exclude || []
+            };
         }
-        function bindEachFunctionsFirst(nodes) {
-            bindEach(nodes, function (n) { return n.kind === 244 /* FunctionDeclaration */ ? bind(n) : undefined; });
-            bindEach(nodes, function (n) { return n.kind !== 244 /* FunctionDeclaration */ ? bind(n) : undefined; });
+        return typeAcquisition;
+    }
+    ts.convertEnableAutoDiscoveryToEnable = convertEnableAutoDiscoveryToEnable;
+    /* @internal */
+    function createCompilerDiagnosticForInvalidCustomType(opt) {
+        return createDiagnosticForInvalidCustomType(opt, ts.createCompilerDiagnostic);
+    }
+    ts.createCompilerDiagnosticForInvalidCustomType = createCompilerDiagnosticForInvalidCustomType;
+    function createDiagnosticForInvalidCustomType(opt, createDiagnostic) {
+        var namesOfType = ts.arrayFrom(opt.type.keys()).map(function (key) { return "'" + key + "'"; }).join(", ");
+        return createDiagnostic(ts.Diagnostics.Argument_for_0_option_must_be_Colon_1, "--" + opt.name, namesOfType);
+    }
+    /* @internal */
+    function parseCustomTypeOption(opt, value, errors) {
+        return convertJsonOptionOfCustomType(opt, trimString(value || ""), errors);
+    }
+    ts.parseCustomTypeOption = parseCustomTypeOption;
+    /* @internal */
+    function parseListTypeOption(opt, value, errors) {
+        if (value === void 0) { value = ""; }
+        value = trimString(value);
+        if (ts.startsWith(value, "-")) {
+            return undefined;
         }
-        function bindEach(nodes, bindFunction) {
-            if (bindFunction === void 0) { bindFunction = bind; }
-            if (nodes === undefined) {
-                return;
-            }
-            if (skipTransformFlagAggregation) {
-                ts.forEach(nodes, bindFunction);
-            }
-            else {
-                var savedSubtreeTransformFlags = subtreeTransformFlags;
-                subtreeTransformFlags = 0 /* None */;
-                var nodeArrayFlags = 0 /* None */;
-                for (var _i = 0, nodes_2 = nodes; _i < nodes_2.length; _i++) {
-                    var node = nodes_2[_i];
-                    bindFunction(node);
-                    nodeArrayFlags |= node.transformFlags & ~536870912 /* HasComputedFlags */;
-                }
-                nodes.transformFlags = nodeArrayFlags | 536870912 /* HasComputedFlags */;
-                subtreeTransformFlags |= savedSubtreeTransformFlags;
-            }
+        if (value === "") {
+            return [];
         }
-        function bindEachChild(node) {
-            ts.forEachChild(node, bind, bindEach);
+        var values = value.split(",");
+        switch (opt.element.type) {
+            case "number":
+                return ts.map(values, parseInt);
+            case "string":
+                return ts.map(values, function (v) { return v || ""; });
+            default:
+                return ts.mapDefined(values, function (v) { return parseCustomTypeOption(opt.element, v, errors); });
         }
-        function bindChildrenWorker(node) {
-            if (checkUnreachable(node)) {
-                bindEachChild(node);
-                bindJSDoc(node);
-                return;
-            }
-            if (node.kind >= 225 /* FirstStatement */ && node.kind <= 241 /* LastStatement */ && !options.allowUnreachableCode) {
-                node.flowNode = currentFlow;
-            }
-            switch (node.kind) {
-                case 229 /* WhileStatement */:
-                    bindWhileStatement(node);
-                    break;
-                case 228 /* DoStatement */:
-                    bindDoStatement(node);
-                    break;
-                case 230 /* ForStatement */:
-                    bindForStatement(node);
-                    break;
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
-                    bindForInOrForOfStatement(node);
-                    break;
-                case 227 /* IfStatement */:
-                    bindIfStatement(node);
-                    break;
-                case 235 /* ReturnStatement */:
-                case 239 /* ThrowStatement */:
-                    bindReturnOrThrow(node);
-                    break;
-                case 234 /* BreakStatement */:
-                case 233 /* ContinueStatement */:
-                    bindBreakOrContinueStatement(node);
-                    break;
-                case 240 /* TryStatement */:
-                    bindTryStatement(node);
-                    break;
-                case 237 /* SwitchStatement */:
-                    bindSwitchStatement(node);
-                    break;
-                case 251 /* CaseBlock */:
-                    bindCaseBlock(node);
-                    break;
-                case 277 /* CaseClause */:
-                    bindCaseClause(node);
-                    break;
-                case 226 /* ExpressionStatement */:
-                    bindExpressionStatement(node);
-                    break;
-                case 238 /* LabeledStatement */:
-                    bindLabeledStatement(node);
-                    break;
-                case 207 /* PrefixUnaryExpression */:
-                    bindPrefixUnaryExpressionFlow(node);
-                    break;
-                case 208 /* PostfixUnaryExpression */:
-                    bindPostfixUnaryExpressionFlow(node);
-                    break;
-                case 209 /* BinaryExpression */:
-                    bindBinaryExpressionFlow(node);
-                    break;
-                case 203 /* DeleteExpression */:
-                    bindDeleteExpressionFlow(node);
-                    break;
-                case 210 /* ConditionalExpression */:
-                    bindConditionalExpressionFlow(node);
-                    break;
-                case 242 /* VariableDeclaration */:
-                    bindVariableDeclarationFlow(node);
-                    break;
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    bindAccessExpressionFlow(node);
-                    break;
-                case 196 /* CallExpression */:
-                    bindCallExpressionFlow(node);
-                    break;
-                case 218 /* NonNullExpression */:
-                    bindNonNullExpressionFlow(node);
-                    break;
-                case 322 /* JSDocTypedefTag */:
-                case 315 /* JSDocCallbackTag */:
-                case 316 /* JSDocEnumTag */:
-                    bindJSDocTypeAlias(node);
-                    break;
-                // In source files and blocks, bind functions first to match hoisting that occurs at runtime
-                case 290 /* SourceFile */: {
-                    bindEachFunctionsFirst(node.statements);
-                    bind(node.endOfFileToken);
-                    break;
+    }
+    ts.parseListTypeOption = parseListTypeOption;
+    function getOptionName(option) {
+        return option.name;
+    }
+    function createUnknownOptionError(unknownOption, diagnostics, createDiagnostics, unknownOptionErrorText) {
+        var possibleOption = ts.getSpellingSuggestion(unknownOption, diagnostics.optionDeclarations, getOptionName);
+        return possibleOption ?
+            createDiagnostics(diagnostics.unknownDidYouMeanDiagnostic, unknownOptionErrorText || unknownOption, possibleOption.name) :
+            createDiagnostics(diagnostics.unknownOptionDiagnostic, unknownOptionErrorText || unknownOption);
+    }
+    /*@internal*/
+    function parseCommandLineWorker(diagnostics, commandLine, readFile) {
+        var options = {};
+        var watchOptions;
+        var fileNames = [];
+        var errors = [];
+        parseStrings(commandLine);
+        return {
+            options: options,
+            watchOptions: watchOptions,
+            fileNames: fileNames,
+            errors: errors
+        };
+        function parseStrings(args) {
+            var i = 0;
+            while (i < args.length) {
+                var s = args[i];
+                i++;
+                if (s.charCodeAt(0) === 64 /* at */) {
+                    parseResponseFile(s.slice(1));
                 }
-                case 223 /* Block */:
-                case 250 /* ModuleBlock */:
-                    bindEachFunctionsFirst(node.statements);
-                    break;
-                default:
-                    bindEachChild(node);
-                    break;
-            }
-            bindJSDoc(node);
-        }
-        function isNarrowingExpression(expr) {
-            switch (expr.kind) {
-                case 75 /* Identifier */:
-                case 104 /* ThisKeyword */:
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    return containsNarrowableReference(expr);
-                case 196 /* CallExpression */:
-                    return hasNarrowableArgument(expr);
-                case 200 /* ParenthesizedExpression */:
-                    return isNarrowingExpression(expr.expression);
-                case 209 /* BinaryExpression */:
-                    return isNarrowingBinaryExpression(expr);
-                case 207 /* PrefixUnaryExpression */:
-                    return expr.operator === 53 /* ExclamationToken */ && isNarrowingExpression(expr.operand);
-                case 204 /* TypeOfExpression */:
-                    return isNarrowingExpression(expr.expression);
-            }
-            return false;
-        }
-        function isNarrowableReference(expr) {
-            return expr.kind === 75 /* Identifier */ || expr.kind === 104 /* ThisKeyword */ || expr.kind === 102 /* SuperKeyword */ ||
-                (ts.isPropertyAccessExpression(expr) || ts.isNonNullExpression(expr) || ts.isParenthesizedExpression(expr)) && isNarrowableReference(expr.expression) ||
-                ts.isElementAccessExpression(expr) && ts.isStringOrNumericLiteralLike(expr.argumentExpression) && isNarrowableReference(expr.expression);
-        }
-        function containsNarrowableReference(expr) {
-            return isNarrowableReference(expr) || ts.isOptionalChain(expr) && containsNarrowableReference(expr.expression);
-        }
-        function hasNarrowableArgument(expr) {
-            if (expr.arguments) {
-                for (var _i = 0, _a = expr.arguments; _i < _a.length; _i++) {
-                    var argument = _a[_i];
-                    if (containsNarrowableReference(argument)) {
-                        return true;
+                else if (s.charCodeAt(0) === 45 /* minus */) {
+                    var inputOptionName = s.slice(s.charCodeAt(1) === 45 /* minus */ ? 2 : 1);
+                    var opt = getOptionDeclarationFromName(diagnostics.getOptionsNameMap, inputOptionName, /*allowShort*/ true);
+                    if (opt) {
+                        i = parseOptionValue(args, i, diagnostics, opt, options, errors);
                     }
-                }
-            }
-            if (expr.expression.kind === 194 /* PropertyAccessExpression */ &&
-                containsNarrowableReference(expr.expression.expression)) {
-                return true;
-            }
-            return false;
-        }
-        function isNarrowingTypeofOperands(expr1, expr2) {
-            return ts.isTypeOfExpression(expr1) && isNarrowableOperand(expr1.expression) && ts.isStringLiteralLike(expr2);
-        }
-        function isNarrowableInOperands(left, right) {
-            return ts.isStringLiteralLike(left) && isNarrowingExpression(right);
-        }
-        function isNarrowingBinaryExpression(expr) {
-            switch (expr.operatorToken.kind) {
-                case 62 /* EqualsToken */:
-                    return containsNarrowableReference(expr.left);
-                case 34 /* EqualsEqualsToken */:
-                case 35 /* ExclamationEqualsToken */:
-                case 36 /* EqualsEqualsEqualsToken */:
-                case 37 /* ExclamationEqualsEqualsToken */:
-                    return isNarrowableOperand(expr.left) || isNarrowableOperand(expr.right) ||
-                        isNarrowingTypeofOperands(expr.right, expr.left) || isNarrowingTypeofOperands(expr.left, expr.right);
-                case 98 /* InstanceOfKeyword */:
-                    return isNarrowableOperand(expr.left);
-                case 97 /* InKeyword */:
-                    return isNarrowableInOperands(expr.left, expr.right);
-                case 27 /* CommaToken */:
-                    return isNarrowingExpression(expr.right);
-            }
-            return false;
-        }
-        function isNarrowableOperand(expr) {
-            switch (expr.kind) {
-                case 200 /* ParenthesizedExpression */:
-                    return isNarrowableOperand(expr.expression);
-                case 209 /* BinaryExpression */:
-                    switch (expr.operatorToken.kind) {
-                        case 62 /* EqualsToken */:
-                            return isNarrowableOperand(expr.left);
-                        case 27 /* CommaToken */:
-                            return isNarrowableOperand(expr.right);
+                    else {
+                        var watchOpt = getOptionDeclarationFromName(watchOptionsDidYouMeanDiagnostics.getOptionsNameMap, inputOptionName, /*allowShort*/ true);
+                        if (watchOpt) {
+                            i = parseOptionValue(args, i, watchOptionsDidYouMeanDiagnostics, watchOpt, watchOptions || (watchOptions = {}), errors);
+                        }
+                        else {
+                            errors.push(createUnknownOptionError(inputOptionName, diagnostics, ts.createCompilerDiagnostic, s));
+                        }
                     }
-            }
-            return containsNarrowableReference(expr);
-        }
-        function createBranchLabel() {
-            return initFlowNode({ flags: 4 /* BranchLabel */, antecedents: undefined });
-        }
-        function createLoopLabel() {
-            return initFlowNode({ flags: 8 /* LoopLabel */, antecedents: undefined });
-        }
-        function createReduceLabel(target, antecedents, antecedent) {
-            return initFlowNode({ flags: 1024 /* ReduceLabel */, target: target, antecedents: antecedents, antecedent: antecedent });
-        }
-        function setFlowNodeReferenced(flow) {
-            // On first reference we set the Referenced flag, thereafter we set the Shared flag
-            flow.flags |= flow.flags & 2048 /* Referenced */ ? 4096 /* Shared */ : 2048 /* Referenced */;
-        }
-        function addAntecedent(label, antecedent) {
-            if (!(antecedent.flags & 1 /* Unreachable */) && !ts.contains(label.antecedents, antecedent)) {
-                (label.antecedents || (label.antecedents = [])).push(antecedent);
-                setFlowNodeReferenced(antecedent);
-            }
-        }
-        function createFlowCondition(flags, antecedent, expression) {
-            if (antecedent.flags & 1 /* Unreachable */) {
-                return antecedent;
-            }
-            if (!expression) {
-                return flags & 32 /* TrueCondition */ ? antecedent : unreachableFlow;
-            }
-            if ((expression.kind === 106 /* TrueKeyword */ && flags & 64 /* FalseCondition */ ||
-                expression.kind === 91 /* FalseKeyword */ && flags & 32 /* TrueCondition */) &&
-                !ts.isExpressionOfOptionalChainRoot(expression) && !ts.isNullishCoalesce(expression.parent)) {
-                return unreachableFlow;
-            }
-            if (!isNarrowingExpression(expression)) {
-                return antecedent;
-            }
-            setFlowNodeReferenced(antecedent);
-            return initFlowNode({ flags: flags, antecedent: antecedent, node: expression });
-        }
-        function createFlowSwitchClause(antecedent, switchStatement, clauseStart, clauseEnd) {
-            setFlowNodeReferenced(antecedent);
-            return initFlowNode({ flags: 128 /* SwitchClause */, antecedent: antecedent, switchStatement: switchStatement, clauseStart: clauseStart, clauseEnd: clauseEnd });
-        }
-        function createFlowMutation(flags, antecedent, node) {
-            setFlowNodeReferenced(antecedent);
-            var result = initFlowNode({ flags: flags, antecedent: antecedent, node: node });
-            if (currentExceptionTarget) {
-                addAntecedent(currentExceptionTarget, result);
-            }
-            return result;
-        }
-        function createFlowCall(antecedent, node) {
-            setFlowNodeReferenced(antecedent);
-            return initFlowNode({ flags: 512 /* Call */, antecedent: antecedent, node: node });
-        }
-        function finishFlowLabel(flow) {
-            var antecedents = flow.antecedents;
-            if (!antecedents) {
-                return unreachableFlow;
-            }
-            if (antecedents.length === 1) {
-                return antecedents[0];
-            }
-            return flow;
-        }
-        function isStatementCondition(node) {
-            var parent = node.parent;
-            switch (parent.kind) {
-                case 227 /* IfStatement */:
-                case 229 /* WhileStatement */:
-                case 228 /* DoStatement */:
-                    return parent.expression === node;
-                case 230 /* ForStatement */:
-                case 210 /* ConditionalExpression */:
-                    return parent.condition === node;
-            }
-            return false;
-        }
-        function isLogicalExpression(node) {
-            while (true) {
-                if (node.kind === 200 /* ParenthesizedExpression */) {
-                    node = node.expression;
-                }
-                else if (node.kind === 207 /* PrefixUnaryExpression */ && node.operator === 53 /* ExclamationToken */) {
-                    node = node.operand;
                 }
                 else {
-                    return node.kind === 209 /* BinaryExpression */ && (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */ ||
-                        node.operatorToken.kind === 56 /* BarBarToken */ ||
-                        node.operatorToken.kind === 60 /* QuestionQuestionToken */);
+                    fileNames.push(s);
                 }
             }
         }
-        function isTopLevelLogicalExpression(node) {
-            while (ts.isParenthesizedExpression(node.parent) ||
-                ts.isPrefixUnaryExpression(node.parent) && node.parent.operator === 53 /* ExclamationToken */) {
-                node = node.parent;
-            }
-            return !isStatementCondition(node) &&
-                !isLogicalExpression(node.parent) &&
-                !(ts.isOptionalChain(node.parent) && node.parent.expression === node);
-        }
-        function doWithConditionalBranches(action, value, trueTarget, falseTarget) {
-            var savedTrueTarget = currentTrueTarget;
-            var savedFalseTarget = currentFalseTarget;
-            currentTrueTarget = trueTarget;
-            currentFalseTarget = falseTarget;
-            action(value);
-            currentTrueTarget = savedTrueTarget;
-            currentFalseTarget = savedFalseTarget;
-        }
-        function bindCondition(node, trueTarget, falseTarget) {
-            doWithConditionalBranches(bind, node, trueTarget, falseTarget);
-            if (!node || !isLogicalExpression(node) && !(ts.isOptionalChain(node) && ts.isOutermostOptionalChain(node))) {
-                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
-                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
-            }
-        }
-        function bindIterativeStatement(node, breakTarget, continueTarget) {
-            var saveBreakTarget = currentBreakTarget;
-            var saveContinueTarget = currentContinueTarget;
-            currentBreakTarget = breakTarget;
-            currentContinueTarget = continueTarget;
-            bind(node);
-            currentBreakTarget = saveBreakTarget;
-            currentContinueTarget = saveContinueTarget;
-        }
-        function setContinueTarget(node, target) {
-            var label = activeLabelList;
-            while (label && node.parent.kind === 238 /* LabeledStatement */) {
-                label.continueTarget = target;
-                label = label.next;
-                node = node.parent;
-            }
-            return target;
-        }
-        function bindWhileStatement(node) {
-            var preWhileLabel = setContinueTarget(node, createLoopLabel());
-            var preBodyLabel = createBranchLabel();
-            var postWhileLabel = createBranchLabel();
-            addAntecedent(preWhileLabel, currentFlow);
-            currentFlow = preWhileLabel;
-            bindCondition(node.expression, preBodyLabel, postWhileLabel);
-            currentFlow = finishFlowLabel(preBodyLabel);
-            bindIterativeStatement(node.statement, postWhileLabel, preWhileLabel);
-            addAntecedent(preWhileLabel, currentFlow);
-            currentFlow = finishFlowLabel(postWhileLabel);
-        }
-        function bindDoStatement(node) {
-            var preDoLabel = createLoopLabel();
-            var preConditionLabel = setContinueTarget(node, createBranchLabel());
-            var postDoLabel = createBranchLabel();
-            addAntecedent(preDoLabel, currentFlow);
-            currentFlow = preDoLabel;
-            bindIterativeStatement(node.statement, postDoLabel, preConditionLabel);
-            addAntecedent(preConditionLabel, currentFlow);
-            currentFlow = finishFlowLabel(preConditionLabel);
-            bindCondition(node.expression, preDoLabel, postDoLabel);
-            currentFlow = finishFlowLabel(postDoLabel);
-        }
-        function bindForStatement(node) {
-            var preLoopLabel = setContinueTarget(node, createLoopLabel());
-            var preBodyLabel = createBranchLabel();
-            var postLoopLabel = createBranchLabel();
-            bind(node.initializer);
-            addAntecedent(preLoopLabel, currentFlow);
-            currentFlow = preLoopLabel;
-            bindCondition(node.condition, preBodyLabel, postLoopLabel);
-            currentFlow = finishFlowLabel(preBodyLabel);
-            bindIterativeStatement(node.statement, postLoopLabel, preLoopLabel);
-            bind(node.incrementor);
-            addAntecedent(preLoopLabel, currentFlow);
-            currentFlow = finishFlowLabel(postLoopLabel);
-        }
-        function bindForInOrForOfStatement(node) {
-            var preLoopLabel = setContinueTarget(node, createLoopLabel());
-            var postLoopLabel = createBranchLabel();
-            bind(node.expression);
-            addAntecedent(preLoopLabel, currentFlow);
-            currentFlow = preLoopLabel;
-            if (node.kind === 232 /* ForOfStatement */) {
-                bind(node.awaitModifier);
-            }
-            addAntecedent(postLoopLabel, currentFlow);
-            bind(node.initializer);
-            if (node.initializer.kind !== 243 /* VariableDeclarationList */) {
-                bindAssignmentTargetFlow(node.initializer);
+        function parseResponseFile(fileName) {
+            var text = tryReadFile(fileName, readFile || (function (fileName) { return ts.sys.readFile(fileName); }));
+            if (!ts.isString(text)) {
+                errors.push(text);
+                return;
             }
-            bindIterativeStatement(node.statement, postLoopLabel, preLoopLabel);
-            addAntecedent(preLoopLabel, currentFlow);
-            currentFlow = finishFlowLabel(postLoopLabel);
-        }
-        function bindIfStatement(node) {
-            var thenLabel = createBranchLabel();
-            var elseLabel = createBranchLabel();
-            var postIfLabel = createBranchLabel();
-            bindCondition(node.expression, thenLabel, elseLabel);
-            currentFlow = finishFlowLabel(thenLabel);
-            bind(node.thenStatement);
-            addAntecedent(postIfLabel, currentFlow);
-            currentFlow = finishFlowLabel(elseLabel);
-            bind(node.elseStatement);
-            addAntecedent(postIfLabel, currentFlow);
-            currentFlow = finishFlowLabel(postIfLabel);
-        }
-        function bindReturnOrThrow(node) {
-            bind(node.expression);
-            if (node.kind === 235 /* ReturnStatement */) {
-                hasExplicitReturn = true;
-                if (currentReturnTarget) {
-                    addAntecedent(currentReturnTarget, currentFlow);
+            var args = [];
+            var pos = 0;
+            while (true) {
+                while (pos < text.length && text.charCodeAt(pos) <= 32 /* space */)
+                    pos++;
+                if (pos >= text.length)
+                    break;
+                var start = pos;
+                if (text.charCodeAt(start) === 34 /* doubleQuote */) {
+                    pos++;
+                    while (pos < text.length && text.charCodeAt(pos) !== 34 /* doubleQuote */)
+                        pos++;
+                    if (pos < text.length) {
+                        args.push(text.substring(start + 1, pos));
+                        pos++;
+                    }
+                    else {
+                        errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Unterminated_quoted_string_in_response_file_0, fileName));
+                    }
                 }
-            }
-            currentFlow = unreachableFlow;
-        }
-        function findActiveLabel(name) {
-            for (var label = activeLabelList; label; label = label.next) {
-                if (label.name === name) {
-                    return label;
+                else {
+                    while (text.charCodeAt(pos) > 32 /* space */)
+                        pos++;
+                    args.push(text.substring(start, pos));
                 }
             }
-            return undefined;
+            parseStrings(args);
         }
-        function bindBreakOrContinueFlow(node, breakTarget, continueTarget) {
-            var flowLabel = node.kind === 234 /* BreakStatement */ ? breakTarget : continueTarget;
-            if (flowLabel) {
-                addAntecedent(flowLabel, currentFlow);
-                currentFlow = unreachableFlow;
+    }
+    ts.parseCommandLineWorker = parseCommandLineWorker;
+    function parseOptionValue(args, i, diagnostics, opt, options, errors) {
+        if (opt.isTSConfigOnly) {
+            var optValue = args[i];
+            if (optValue === "null") {
+                options[opt.name] = undefined;
+                i++;
             }
-        }
-        function bindBreakOrContinueStatement(node) {
-            bind(node.label);
-            if (node.label) {
-                var activeLabel = findActiveLabel(node.label.escapedText);
-                if (activeLabel) {
-                    activeLabel.referenced = true;
-                    bindBreakOrContinueFlow(node, activeLabel.breakTarget, activeLabel.continueTarget);
+            else if (opt.type === "boolean") {
+                if (optValue === "false") {
+                    options[opt.name] = false;
+                    i++;
+                }
+                else {
+                    if (optValue === "true")
+                        i++;
+                    errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Option_0_can_only_be_specified_in_tsconfig_json_file_or_set_to_false_or_null_on_command_line, opt.name));
                 }
             }
             else {
-                bindBreakOrContinueFlow(node, currentBreakTarget, currentContinueTarget);
+                errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Option_0_can_only_be_specified_in_tsconfig_json_file_or_set_to_null_on_command_line, opt.name));
+                if (optValue && !ts.startsWith(optValue, "-"))
+                    i++;
             }
         }
-        function bindTryStatement(node) {
-            // We conservatively assume that *any* code in the try block can cause an exception, but we only need
-            // to track code that causes mutations (because only mutations widen the possible control flow type of
-            // a variable). The exceptionLabel is the target label for control flows that result from exceptions.
-            // We add all mutation flow nodes as antecedents of this label such that we can analyze them as possible
-            // antecedents of the start of catch or finally blocks. Furthermore, we add the current control flow to
-            // represent exceptions that occur before any mutations.
-            var saveReturnTarget = currentReturnTarget;
-            var saveExceptionTarget = currentExceptionTarget;
-            var normalExitLabel = createBranchLabel();
-            var returnLabel = createBranchLabel();
-            var exceptionLabel = createBranchLabel();
-            if (node.finallyBlock) {
-                currentReturnTarget = returnLabel;
-            }
-            addAntecedent(exceptionLabel, currentFlow);
-            currentExceptionTarget = exceptionLabel;
-            bind(node.tryBlock);
-            addAntecedent(normalExitLabel, currentFlow);
-            if (node.catchClause) {
-                // Start of catch clause is the target of exceptions from try block.
-                currentFlow = finishFlowLabel(exceptionLabel);
-                // The currentExceptionTarget now represents control flows from exceptions in the catch clause.
-                // Effectively, in a try-catch-finally, if an exception occurs in the try block, the catch block
-                // acts like a second try block.
-                exceptionLabel = createBranchLabel();
-                addAntecedent(exceptionLabel, currentFlow);
-                currentExceptionTarget = exceptionLabel;
-                bind(node.catchClause);
-                addAntecedent(normalExitLabel, currentFlow);
+        else {
+            // Check to see if no argument was provided (e.g. "--locale" is the last command-line argument).
+            if (!args[i] && opt.type !== "boolean") {
+                errors.push(ts.createCompilerDiagnostic(diagnostics.optionTypeMismatchDiagnostic, opt.name, getCompilerOptionValueTypeString(opt)));
             }
-            currentReturnTarget = saveReturnTarget;
-            currentExceptionTarget = saveExceptionTarget;
-            if (node.finallyBlock) {
-                // Possible ways control can reach the finally block:
-                // 1) Normal completion of try block of a try-finally or try-catch-finally
-                // 2) Normal completion of catch block (following exception in try block) of a try-catch-finally
-                // 3) Return in try or catch block of a try-finally or try-catch-finally
-                // 4) Exception in try block of a try-finally
-                // 5) Exception in catch block of a try-catch-finally
-                // When analyzing a control flow graph that starts inside a finally block we want to consider all
-                // five possibilities above. However, when analyzing a control flow graph that starts outside (past)
-                // the finally block, we only want to consider the first two (if we're past a finally block then it
-                // must have completed normally). Likewise, when analyzing a control flow graph from return statements
-                // in try or catch blocks in an IIFE, we only want to consider the third. To make this possible, we
-                // inject a ReduceLabel node into the control flow graph. This node contains an alternate reduced
-                // set of antecedents for the pre-finally label. As control flow analysis passes by a ReduceLabel
-                // node, the pre-finally label is temporarily switched to the reduced antecedent set.
-                var finallyLabel = createBranchLabel();
-                finallyLabel.antecedents = ts.concatenate(ts.concatenate(normalExitLabel.antecedents, exceptionLabel.antecedents), returnLabel.antecedents);
-                currentFlow = finallyLabel;
-                bind(node.finallyBlock);
-                if (currentFlow.flags & 1 /* Unreachable */) {
-                    // If the end of the finally block is unreachable, the end of the entire try statement is unreachable.
-                    currentFlow = unreachableFlow;
-                }
-                else {
-                    // If we have an IIFE return target and return statements in the try or catch blocks, add a control
-                    // flow that goes back through the finally block and back through only the return statements.
-                    if (currentReturnTarget && returnLabel.antecedents) {
-                        addAntecedent(currentReturnTarget, createReduceLabel(finallyLabel, returnLabel.antecedents, currentFlow));
-                    }
-                    // If the end of the finally block is reachable, but the end of the try and catch blocks are not,
-                    // convert the current flow to unreachable. For example, 'try { return 1; } finally { ... }' should
-                    // result in an unreachable current control flow.
-                    currentFlow = normalExitLabel.antecedents ? createReduceLabel(finallyLabel, normalExitLabel.antecedents, currentFlow) : unreachableFlow;
+            if (args[i] !== "null") {
+                switch (opt.type) {
+                    case "number":
+                        options[opt.name] = parseInt(args[i]);
+                        i++;
+                        break;
+                    case "boolean":
+                        // boolean flag has optional value true, false, others
+                        var optValue = args[i];
+                        options[opt.name] = optValue !== "false";
+                        // consume next argument as boolean flag value
+                        if (optValue === "false" || optValue === "true") {
+                            i++;
+                        }
+                        break;
+                    case "string":
+                        options[opt.name] = args[i] || "";
+                        i++;
+                        break;
+                    case "list":
+                        var result = parseListTypeOption(opt, args[i], errors);
+                        options[opt.name] = result || [];
+                        if (result) {
+                            i++;
+                        }
+                        break;
+                    // If not a primitive, the possible types are specified in what is effectively a map of options.
+                    default:
+                        options[opt.name] = parseCustomTypeOption(opt, args[i], errors);
+                        i++;
+                        break;
                 }
             }
             else {
-                currentFlow = finishFlowLabel(normalExitLabel);
+                options[opt.name] = undefined;
+                i++;
             }
         }
-        function bindSwitchStatement(node) {
-            var postSwitchLabel = createBranchLabel();
-            bind(node.expression);
-            var saveBreakTarget = currentBreakTarget;
-            var savePreSwitchCaseFlow = preSwitchCaseFlow;
-            currentBreakTarget = postSwitchLabel;
-            preSwitchCaseFlow = currentFlow;
-            bind(node.caseBlock);
-            addAntecedent(postSwitchLabel, currentFlow);
-            var hasDefault = ts.forEach(node.caseBlock.clauses, function (c) { return c.kind === 278 /* DefaultClause */; });
-            // We mark a switch statement as possibly exhaustive if it has no default clause and if all
-            // case clauses have unreachable end points (e.g. they all return). Note, we no longer need
-            // this property in control flow analysis, it's there only for backwards compatibility.
-            node.possiblyExhaustive = !hasDefault && !postSwitchLabel.antecedents;
-            if (!hasDefault) {
-                addAntecedent(postSwitchLabel, createFlowSwitchClause(preSwitchCaseFlow, node, 0, 0));
+        return i;
+    }
+    /*@internal*/
+    ts.compilerOptionsDidYouMeanDiagnostics = {
+        getOptionsNameMap: getOptionsNameMap,
+        optionDeclarations: ts.optionDeclarations,
+        unknownOptionDiagnostic: ts.Diagnostics.Unknown_compiler_option_0,
+        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_compiler_option_0_Did_you_mean_1,
+        optionTypeMismatchDiagnostic: ts.Diagnostics.Compiler_option_0_expects_an_argument
+    };
+    function parseCommandLine(commandLine, readFile) {
+        return parseCommandLineWorker(ts.compilerOptionsDidYouMeanDiagnostics, commandLine, readFile);
+    }
+    ts.parseCommandLine = parseCommandLine;
+    /** @internal */
+    function getOptionFromName(optionName, allowShort) {
+        return getOptionDeclarationFromName(getOptionsNameMap, optionName, allowShort);
+    }
+    ts.getOptionFromName = getOptionFromName;
+    function getOptionDeclarationFromName(getOptionNameMap, optionName, allowShort) {
+        if (allowShort === void 0) { allowShort = false; }
+        optionName = optionName.toLowerCase();
+        var _a = getOptionNameMap(), optionsNameMap = _a.optionsNameMap, shortOptionNames = _a.shortOptionNames;
+        // Try to translate short option names to their full equivalents.
+        if (allowShort) {
+            var short = shortOptionNames.get(optionName);
+            if (short !== undefined) {
+                optionName = short;
             }
-            currentBreakTarget = saveBreakTarget;
-            preSwitchCaseFlow = savePreSwitchCaseFlow;
-            currentFlow = finishFlowLabel(postSwitchLabel);
         }
-        function bindCaseBlock(node) {
-            var savedSubtreeTransformFlags = subtreeTransformFlags;
-            subtreeTransformFlags = 0;
-            var clauses = node.clauses;
-            var isNarrowingSwitch = isNarrowingExpression(node.parent.expression);
-            var fallthroughFlow = unreachableFlow;
-            for (var i = 0; i < clauses.length; i++) {
-                var clauseStart = i;
-                while (!clauses[i].statements.length && i + 1 < clauses.length) {
-                    bind(clauses[i]);
-                    i++;
-                }
-                var preCaseLabel = createBranchLabel();
-                addAntecedent(preCaseLabel, isNarrowingSwitch ? createFlowSwitchClause(preSwitchCaseFlow, node.parent, clauseStart, i + 1) : preSwitchCaseFlow);
-                addAntecedent(preCaseLabel, fallthroughFlow);
-                currentFlow = finishFlowLabel(preCaseLabel);
-                var clause = clauses[i];
-                bind(clause);
-                fallthroughFlow = currentFlow;
-                if (!(currentFlow.flags & 1 /* Unreachable */) && i !== clauses.length - 1 && options.noFallthroughCasesInSwitch) {
-                    clause.fallthroughFlowNode = currentFlow;
-                }
-            }
-            clauses.transformFlags = subtreeTransformFlags | 536870912 /* HasComputedFlags */;
-            subtreeTransformFlags |= savedSubtreeTransformFlags;
+        return optionsNameMap.get(optionName);
+    }
+    var buildOptionsNameMapCache;
+    function getBuildOptionsNameMap() {
+        return buildOptionsNameMapCache || (buildOptionsNameMapCache = createOptionNameMap(ts.buildOpts));
+    }
+    var buildOptionsDidYouMeanDiagnostics = {
+        getOptionsNameMap: getBuildOptionsNameMap,
+        optionDeclarations: ts.buildOpts,
+        unknownOptionDiagnostic: ts.Diagnostics.Unknown_build_option_0,
+        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_build_option_0_Did_you_mean_1,
+        optionTypeMismatchDiagnostic: ts.Diagnostics.Build_option_0_requires_a_value_of_type_1
+    };
+    /*@internal*/
+    function parseBuildCommand(args) {
+        var _a = parseCommandLineWorker(buildOptionsDidYouMeanDiagnostics, args), options = _a.options, watchOptions = _a.watchOptions, projects = _a.fileNames, errors = _a.errors;
+        var buildOptions = options;
+        if (projects.length === 0) {
+            // tsc -b invoked with no extra arguments; act as if invoked with "tsc -b ."
+            projects.push(".");
         }
-        function bindCaseClause(node) {
-            var saveCurrentFlow = currentFlow;
-            currentFlow = preSwitchCaseFlow;
-            bind(node.expression);
-            currentFlow = saveCurrentFlow;
-            bindEach(node.statements);
+        // Nonsensical combinations
+        if (buildOptions.clean && buildOptions.force) {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "clean", "force"));
         }
-        function bindExpressionStatement(node) {
-            bind(node.expression);
-            // A top level call expression with a dotted function name and at least one argument
-            // is potentially an assertion and is therefore included in the control flow.
-            if (node.expression.kind === 196 /* CallExpression */) {
-                var call = node.expression;
-                if (ts.isDottedName(call.expression)) {
-                    currentFlow = createFlowCall(currentFlow, call);
-                }
-            }
+        if (buildOptions.clean && buildOptions.verbose) {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "clean", "verbose"));
         }
-        function bindLabeledStatement(node) {
-            var postStatementLabel = createBranchLabel();
-            activeLabelList = {
-                next: activeLabelList,
-                name: node.label.escapedText,
-                breakTarget: postStatementLabel,
-                continueTarget: undefined,
-                referenced: false
+        if (buildOptions.clean && buildOptions.watch) {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "clean", "watch"));
+        }
+        if (buildOptions.watch && buildOptions.dry) {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Options_0_and_1_cannot_be_combined, "watch", "dry"));
+        }
+        return { buildOptions: buildOptions, watchOptions: watchOptions, projects: projects, errors: errors };
+    }
+    ts.parseBuildCommand = parseBuildCommand;
+    /* @internal */
+    function getDiagnosticText(_message) {
+        var _args = [];
+        for (var _i = 1; _i < arguments.length; _i++) {
+            _args[_i - 1] = arguments[_i];
+        }
+        var diagnostic = ts.createCompilerDiagnostic.apply(undefined, arguments);
+        return diagnostic.messageText;
+    }
+    ts.getDiagnosticText = getDiagnosticText;
+    /**
+     * Reads the config file, reports errors if any and exits if the config file cannot be found
+     */
+    function getParsedCommandLineOfConfigFile(configFileName, optionsToExtend, host, extendedConfigCache, watchOptionsToExtend, extraFileExtensions) {
+        var configFileText = tryReadFile(configFileName, function (fileName) { return host.readFile(fileName); });
+        if (!ts.isString(configFileText)) {
+            host.onUnRecoverableConfigFileDiagnostic(configFileText);
+            return undefined;
+        }
+        var result = ts.parseJsonText(configFileName, configFileText);
+        var cwd = host.getCurrentDirectory();
+        result.path = ts.toPath(configFileName, cwd, ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames));
+        result.resolvedPath = result.path;
+        result.originalFileName = result.fileName;
+        return parseJsonSourceFileConfigFileContent(result, host, ts.getNormalizedAbsolutePath(ts.getDirectoryPath(configFileName), cwd), optionsToExtend, ts.getNormalizedAbsolutePath(configFileName, cwd), 
+        /*resolutionStack*/ undefined, extraFileExtensions, extendedConfigCache, watchOptionsToExtend);
+    }
+    ts.getParsedCommandLineOfConfigFile = getParsedCommandLineOfConfigFile;
+    /**
+     * Read tsconfig.json file
+     * @param fileName The path to the config file
+     */
+    function readConfigFile(fileName, readFile) {
+        var textOrDiagnostic = tryReadFile(fileName, readFile);
+        return ts.isString(textOrDiagnostic) ? parseConfigFileTextToJson(fileName, textOrDiagnostic) : { config: {}, error: textOrDiagnostic };
+    }
+    ts.readConfigFile = readConfigFile;
+    /**
+     * Parse the text of the tsconfig.json file
+     * @param fileName The path to the config file
+     * @param jsonText The text of the config file
+     */
+    function parseConfigFileTextToJson(fileName, jsonText) {
+        var jsonSourceFile = ts.parseJsonText(fileName, jsonText);
+        return {
+            config: convertToObject(jsonSourceFile, jsonSourceFile.parseDiagnostics),
+            error: jsonSourceFile.parseDiagnostics.length ? jsonSourceFile.parseDiagnostics[0] : undefined
+        };
+    }
+    ts.parseConfigFileTextToJson = parseConfigFileTextToJson;
+    /**
+     * Read tsconfig.json file
+     * @param fileName The path to the config file
+     */
+    function readJsonConfigFile(fileName, readFile) {
+        var textOrDiagnostic = tryReadFile(fileName, readFile);
+        return ts.isString(textOrDiagnostic) ? ts.parseJsonText(fileName, textOrDiagnostic) : { fileName: fileName, parseDiagnostics: [textOrDiagnostic] };
+    }
+    ts.readJsonConfigFile = readJsonConfigFile;
+    /*@internal*/
+    function tryReadFile(fileName, readFile) {
+        var text;
+        try {
+            text = readFile(fileName);
+        }
+        catch (e) {
+            return ts.createCompilerDiagnostic(ts.Diagnostics.Cannot_read_file_0_Colon_1, fileName, e.message);
+        }
+        return text === undefined ? ts.createCompilerDiagnostic(ts.Diagnostics.Cannot_read_file_0, fileName) : text;
+    }
+    ts.tryReadFile = tryReadFile;
+    function commandLineOptionsToMap(options) {
+        return ts.arrayToMap(options, getOptionName);
+    }
+    var typeAcquisitionDidYouMeanDiagnostics = {
+        optionDeclarations: ts.typeAcquisitionDeclarations,
+        unknownOptionDiagnostic: ts.Diagnostics.Unknown_type_acquisition_option_0,
+        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_type_acquisition_option_0_Did_you_mean_1,
+    };
+    var watchOptionsNameMapCache;
+    function getWatchOptionsNameMap() {
+        return watchOptionsNameMapCache || (watchOptionsNameMapCache = createOptionNameMap(ts.optionsForWatch));
+    }
+    var watchOptionsDidYouMeanDiagnostics = {
+        getOptionsNameMap: getWatchOptionsNameMap,
+        optionDeclarations: ts.optionsForWatch,
+        unknownOptionDiagnostic: ts.Diagnostics.Unknown_watch_option_0,
+        unknownDidYouMeanDiagnostic: ts.Diagnostics.Unknown_watch_option_0_Did_you_mean_1,
+        optionTypeMismatchDiagnostic: ts.Diagnostics.Watch_option_0_requires_a_value_of_type_1
+    };
+    var commandLineCompilerOptionsMapCache;
+    function getCommandLineCompilerOptionsMap() {
+        return commandLineCompilerOptionsMapCache || (commandLineCompilerOptionsMapCache = commandLineOptionsToMap(ts.optionDeclarations));
+    }
+    var commandLineWatchOptionsMapCache;
+    function getCommandLineWatchOptionsMap() {
+        return commandLineWatchOptionsMapCache || (commandLineWatchOptionsMapCache = commandLineOptionsToMap(ts.optionsForWatch));
+    }
+    var commandLineTypeAcquisitionMapCache;
+    function getCommandLineTypeAcquisitionMap() {
+        return commandLineTypeAcquisitionMapCache || (commandLineTypeAcquisitionMapCache = commandLineOptionsToMap(ts.typeAcquisitionDeclarations));
+    }
+    var _tsconfigRootOptions;
+    function getTsconfigRootOptionsMap() {
+        if (_tsconfigRootOptions === undefined) {
+            _tsconfigRootOptions = {
+                name: undefined,
+                type: "object",
+                elementOptions: commandLineOptionsToMap([
+                    {
+                        name: "compilerOptions",
+                        type: "object",
+                        elementOptions: getCommandLineCompilerOptionsMap(),
+                        extraKeyDiagnostics: ts.compilerOptionsDidYouMeanDiagnostics,
+                    },
+                    {
+                        name: "watchOptions",
+                        type: "object",
+                        elementOptions: getCommandLineWatchOptionsMap(),
+                        extraKeyDiagnostics: watchOptionsDidYouMeanDiagnostics,
+                    },
+                    {
+                        name: "typingOptions",
+                        type: "object",
+                        elementOptions: getCommandLineTypeAcquisitionMap(),
+                        extraKeyDiagnostics: typeAcquisitionDidYouMeanDiagnostics,
+                    },
+                    {
+                        name: "typeAcquisition",
+                        type: "object",
+                        elementOptions: getCommandLineTypeAcquisitionMap(),
+                        extraKeyDiagnostics: typeAcquisitionDidYouMeanDiagnostics
+                    },
+                    {
+                        name: "extends",
+                        type: "string"
+                    },
+                    {
+                        name: "references",
+                        type: "list",
+                        element: {
+                            name: "references",
+                            type: "object"
+                        }
+                    },
+                    {
+                        name: "files",
+                        type: "list",
+                        element: {
+                            name: "files",
+                            type: "string"
+                        }
+                    },
+                    {
+                        name: "include",
+                        type: "list",
+                        element: {
+                            name: "include",
+                            type: "string"
+                        }
+                    },
+                    {
+                        name: "exclude",
+                        type: "list",
+                        element: {
+                            name: "exclude",
+                            type: "string"
+                        }
+                    },
+                    ts.compileOnSaveCommandLineOption
+                ])
             };
-            bind(node.label);
-            bind(node.statement);
-            if (!activeLabelList.referenced && !options.allowUnusedLabels) {
-                errorOrSuggestionOnNode(ts.unusedLabelIsError(options), node.label, ts.Diagnostics.Unused_label);
-            }
-            activeLabelList = activeLabelList.next;
-            addAntecedent(postStatementLabel, currentFlow);
-            currentFlow = finishFlowLabel(postStatementLabel);
         }
-        function bindDestructuringTargetFlow(node) {
-            if (node.kind === 209 /* BinaryExpression */ && node.operatorToken.kind === 62 /* EqualsToken */) {
-                bindAssignmentTargetFlow(node.left);
-            }
-            else {
-                bindAssignmentTargetFlow(node);
-            }
+        return _tsconfigRootOptions;
+    }
+    /**
+     * Convert the json syntax tree into the json value
+     */
+    function convertToObject(sourceFile, errors) {
+        return convertToObjectWorker(sourceFile, errors, /*returnValue*/ true, /*knownRootOptions*/ undefined, /*jsonConversionNotifier*/ undefined);
+    }
+    ts.convertToObject = convertToObject;
+    /**
+     * Convert the json syntax tree into the json value and report errors
+     * This returns the json value (apart from checking errors) only if returnValue provided is true.
+     * Otherwise it just checks the errors and returns undefined
+     */
+    /*@internal*/
+    function convertToObjectWorker(sourceFile, errors, returnValue, knownRootOptions, jsonConversionNotifier) {
+        if (!sourceFile.statements.length) {
+            return returnValue ? {} : undefined;
+        }
+        return convertPropertyValueToJson(sourceFile.statements[0].expression, knownRootOptions);
+        function isRootOptionMap(knownOptions) {
+            return knownRootOptions && knownRootOptions.elementOptions === knownOptions;
         }
-        function bindAssignmentTargetFlow(node) {
-            if (isNarrowableReference(node)) {
-                currentFlow = createFlowMutation(16 /* Assignment */, currentFlow, node);
-            }
-            else if (node.kind === 192 /* ArrayLiteralExpression */) {
-                for (var _i = 0, _a = node.elements; _i < _a.length; _i++) {
-                    var e = _a[_i];
-                    if (e.kind === 213 /* SpreadElement */) {
-                        bindAssignmentTargetFlow(e.expression);
+        function convertObjectLiteralExpressionToJson(node, knownOptions, extraKeyDiagnostics, parentOption) {
+            var result = returnValue ? {} : undefined;
+            var _loop_4 = function (element) {
+                if (element.kind !== 288 /* PropertyAssignment */) {
+                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element, ts.Diagnostics.Property_assignment_expected));
+                    return "continue";
+                }
+                if (element.questionToken) {
+                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element.questionToken, ts.Diagnostics.The_0_modifier_can_only_be_used_in_TypeScript_files, "?"));
+                }
+                if (!isDoubleQuotedString(element.name)) {
+                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element.name, ts.Diagnostics.String_literal_with_double_quotes_expected));
+                }
+                var textOfKey = ts.isComputedNonLiteralName(element.name) ? undefined : ts.getTextOfPropertyName(element.name);
+                var keyText = textOfKey && ts.unescapeLeadingUnderscores(textOfKey);
+                var option = keyText && knownOptions ? knownOptions.get(keyText) : undefined;
+                if (keyText && extraKeyDiagnostics && !option) {
+                    if (knownOptions) {
+                        errors.push(createUnknownOptionError(keyText, extraKeyDiagnostics, function (message, arg0, arg1) { return ts.createDiagnosticForNodeInSourceFile(sourceFile, element.name, message, arg0, arg1); }));
                     }
                     else {
-                        bindDestructuringTargetFlow(e);
+                        errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, element.name, extraKeyDiagnostics.unknownOptionDiagnostic, keyText));
                     }
                 }
-            }
-            else if (node.kind === 193 /* ObjectLiteralExpression */) {
-                for (var _b = 0, _c = node.properties; _b < _c.length; _b++) {
-                    var p = _c[_b];
-                    if (p.kind === 281 /* PropertyAssignment */) {
-                        bindDestructuringTargetFlow(p.initializer);
-                    }
-                    else if (p.kind === 282 /* ShorthandPropertyAssignment */) {
-                        bindAssignmentTargetFlow(p.name);
+                var value = convertPropertyValueToJson(element.initializer, option);
+                if (typeof keyText !== "undefined") {
+                    if (returnValue) {
+                        result[keyText] = value;
                     }
-                    else if (p.kind === 283 /* SpreadAssignment */) {
-                        bindAssignmentTargetFlow(p.expression);
+                    // Notify key value set, if user asked for it
+                    if (jsonConversionNotifier &&
+                        // Current callbacks are only on known parent option or if we are setting values in the root
+                        (parentOption || isRootOptionMap(knownOptions))) {
+                        var isValidOptionValue = isCompilerOptionsValue(option, value);
+                        if (parentOption) {
+                            if (isValidOptionValue) {
+                                // Notify option set in the parent if its a valid option value
+                                jsonConversionNotifier.onSetValidOptionKeyValueInParent(parentOption, option, value);
+                            }
+                        }
+                        else if (isRootOptionMap(knownOptions)) {
+                            if (isValidOptionValue) {
+                                // Notify about the valid root key value being set
+                                jsonConversionNotifier.onSetValidOptionKeyValueInRoot(keyText, element.name, value, element.initializer);
+                            }
+                            else if (!option) {
+                                // Notify about the unknown root key value being set
+                                jsonConversionNotifier.onSetUnknownOptionKeyValueInRoot(keyText, element.name, value, element.initializer);
+                            }
+                        }
                     }
                 }
+            };
+            for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
+                var element = _a[_i];
+                _loop_4(element);
             }
+            return result;
         }
-        function bindLogicalExpression(node, trueTarget, falseTarget) {
-            var preRightLabel = createBranchLabel();
-            if (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */) {
-                bindCondition(node.left, preRightLabel, falseTarget);
-            }
-            else {
-                bindCondition(node.left, trueTarget, preRightLabel);
-            }
-            currentFlow = finishFlowLabel(preRightLabel);
-            bind(node.operatorToken);
-            bindCondition(node.right, trueTarget, falseTarget);
-        }
-        function bindPrefixUnaryExpressionFlow(node) {
-            if (node.operator === 53 /* ExclamationToken */) {
-                var saveTrueTarget = currentTrueTarget;
-                currentTrueTarget = currentFalseTarget;
-                currentFalseTarget = saveTrueTarget;
-                bindEachChild(node);
-                currentFalseTarget = currentTrueTarget;
-                currentTrueTarget = saveTrueTarget;
-            }
-            else {
-                bindEachChild(node);
-                if (node.operator === 45 /* PlusPlusToken */ || node.operator === 46 /* MinusMinusToken */) {
-                    bindAssignmentTargetFlow(node.operand);
-                }
-            }
-        }
-        function bindPostfixUnaryExpressionFlow(node) {
-            bindEachChild(node);
-            if (node.operator === 45 /* PlusPlusToken */ || node.operator === 46 /* MinusMinusToken */) {
-                bindAssignmentTargetFlow(node.operand);
+        function convertArrayLiteralExpressionToJson(elements, elementOption) {
+            if (!returnValue) {
+                return elements.forEach(function (element) { return convertPropertyValueToJson(element, elementOption); });
             }
+            // Filter out invalid values
+            return ts.filter(elements.map(function (element) { return convertPropertyValueToJson(element, elementOption); }), function (v) { return v !== undefined; });
         }
-        var BindBinaryExpressionFlowState;
-        (function (BindBinaryExpressionFlowState) {
-            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["BindThenBindChildren"] = 0] = "BindThenBindChildren";
-            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["MaybeBindLeft"] = 1] = "MaybeBindLeft";
-            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["BindToken"] = 2] = "BindToken";
-            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["BindRight"] = 3] = "BindRight";
-            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["FinishBind"] = 4] = "FinishBind";
-        })(BindBinaryExpressionFlowState || (BindBinaryExpressionFlowState = {}));
-        function bindBinaryExpressionFlow(node) {
-            var workStacks = {
-                expr: [node],
-                state: [1 /* MaybeBindLeft */],
-                inStrictMode: [undefined],
-                parent: [undefined],
-                subtreeFlags: [undefined]
-            };
-            var stackIndex = 0;
-            while (stackIndex >= 0) {
-                node = workStacks.expr[stackIndex];
-                switch (workStacks.state[stackIndex]) {
-                    case 0 /* BindThenBindChildren */: {
-                        // This state is used only when recuring, to emulate the work that `bind` does before
-                        // reaching `bindChildren`. A normal call to `bindBinaryExpressionFlow` will already have done this work.
-                        node.parent = parent;
-                        var saveInStrictMode = inStrictMode;
-                        bindWorker(node);
-                        var saveParent = parent;
-                        parent = node;
-                        var subtreeFlagsState = void 0;
-                        // While this next part does the work of `bindChildren` before it descends into `bindChildrenWorker`
-                        // and uses `subtreeFlagsState` to queue up the work that needs to be done once the node is bound.
-                        if (skipTransformFlagAggregation) {
-                            // do nothing extra
-                        }
-                        else if (node.transformFlags & 536870912 /* HasComputedFlags */) {
-                            skipTransformFlagAggregation = true;
-                            subtreeFlagsState = -1;
-                        }
-                        else {
-                            var savedSubtreeTransformFlags = subtreeTransformFlags;
-                            subtreeTransformFlags = 0;
-                            subtreeFlagsState = savedSubtreeTransformFlags;
-                        }
-                        advanceState(1 /* MaybeBindLeft */, saveInStrictMode, saveParent, subtreeFlagsState);
-                        break;
+        function convertPropertyValueToJson(valueExpression, option) {
+            switch (valueExpression.kind) {
+                case 109 /* TrueKeyword */:
+                    reportInvalidOptionValue(option && option.type !== "boolean");
+                    return true;
+                case 94 /* FalseKeyword */:
+                    reportInvalidOptionValue(option && option.type !== "boolean");
+                    return false;
+                case 103 /* NullKeyword */:
+                    reportInvalidOptionValue(option && option.name === "extends"); // "extends" is the only option we don't allow null/undefined for
+                    return null; // eslint-disable-line no-null/no-null
+                case 10 /* StringLiteral */:
+                    if (!isDoubleQuotedString(valueExpression)) {
+                        errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, ts.Diagnostics.String_literal_with_double_quotes_expected));
                     }
-                    case 1 /* MaybeBindLeft */: {
-                        var operator = node.operatorToken.kind;
-                        // TODO: bindLogicalExpression is recursive - if we want to handle deeply nested `&&` expressions
-                        // we'll need to handle the `bindLogicalExpression` scenarios in this state machine, too
-                        // For now, though, since the common cases are chained `+`, leaving it recursive is fine
-                        if (operator === 55 /* AmpersandAmpersandToken */ || operator === 56 /* BarBarToken */ || operator === 60 /* QuestionQuestionToken */) {
-                            if (isTopLevelLogicalExpression(node)) {
-                                var postExpressionLabel = createBranchLabel();
-                                bindLogicalExpression(node, postExpressionLabel, postExpressionLabel);
-                                currentFlow = finishFlowLabel(postExpressionLabel);
-                            }
-                            else {
-                                bindLogicalExpression(node, currentTrueTarget, currentFalseTarget);
-                            }
-                            completeNode();
-                        }
-                        else {
-                            advanceState(2 /* BindToken */);
-                            maybeBind(node.left);
+                    reportInvalidOptionValue(option && (ts.isString(option.type) && option.type !== "string"));
+                    var text = valueExpression.text;
+                    if (option && !ts.isString(option.type)) {
+                        var customOption = option;
+                        // Validate custom option type
+                        if (!customOption.type.has(text.toLowerCase())) {
+                            errors.push(createDiagnosticForInvalidCustomType(customOption, function (message, arg0, arg1) { return ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, message, arg0, arg1); }));
                         }
-                        break;
-                    }
-                    case 2 /* BindToken */: {
-                        advanceState(3 /* BindRight */);
-                        maybeBind(node.operatorToken);
-                        break;
-                    }
-                    case 3 /* BindRight */: {
-                        advanceState(4 /* FinishBind */);
-                        maybeBind(node.right);
-                        break;
                     }
-                    case 4 /* FinishBind */: {
-                        var operator = node.operatorToken.kind;
-                        if (ts.isAssignmentOperator(operator) && !ts.isAssignmentTarget(node)) {
-                            bindAssignmentTargetFlow(node.left);
-                            if (operator === 62 /* EqualsToken */ && node.left.kind === 195 /* ElementAccessExpression */) {
-                                var elementAccess = node.left;
-                                if (isNarrowableOperand(elementAccess.expression)) {
-                                    currentFlow = createFlowMutation(256 /* ArrayMutation */, currentFlow, node);
-                                }
-                            }
-                        }
-                        completeNode();
-                        break;
+                    return text;
+                case 8 /* NumericLiteral */:
+                    reportInvalidOptionValue(option && option.type !== "number");
+                    return Number(valueExpression.text);
+                case 214 /* PrefixUnaryExpression */:
+                    if (valueExpression.operator !== 40 /* MinusToken */ || valueExpression.operand.kind !== 8 /* NumericLiteral */) {
+                        break; // not valid JSON syntax
                     }
-                    default: return ts.Debug.fail("Invalid state " + workStacks.state[stackIndex] + " for bindBinaryExpressionFlow");
-                }
-            }
-            /**
-             * Note that `advanceState` sets the _current_ head state, and that `maybeBind` potentially pushes on a new
-             * head state; so `advanceState` must be called before any `maybeBind` during a state's execution.
-             */
-            function advanceState(state, isInStrictMode, parent, subtreeFlags) {
-                workStacks.state[stackIndex] = state;
-                if (isInStrictMode !== undefined) {
-                    workStacks.inStrictMode[stackIndex] = isInStrictMode;
-                }
-                if (parent !== undefined) {
-                    workStacks.parent[stackIndex] = parent;
-                }
-                if (subtreeFlags !== undefined) {
-                    workStacks.subtreeFlags[stackIndex] = subtreeFlags;
-                }
-            }
-            function completeNode() {
-                if (workStacks.inStrictMode[stackIndex] !== undefined) {
-                    if (workStacks.subtreeFlags[stackIndex] === -1) {
-                        skipTransformFlagAggregation = false;
-                        subtreeTransformFlags |= node.transformFlags & ~getTransformFlagsSubtreeExclusions(node.kind);
+                    reportInvalidOptionValue(option && option.type !== "number");
+                    return -Number(valueExpression.operand.text);
+                case 200 /* ObjectLiteralExpression */:
+                    reportInvalidOptionValue(option && option.type !== "object");
+                    var objectLiteralExpression = valueExpression;
+                    // Currently having element option declaration in the tsconfig with type "object"
+                    // determines if it needs onSetValidOptionKeyValueInParent callback or not
+                    // At moment there are only "compilerOptions", "typeAcquisition" and "typingOptions"
+                    // that satifies it and need it to modify options set in them (for normalizing file paths)
+                    // vs what we set in the json
+                    // If need arises, we can modify this interface and callbacks as needed
+                    if (option) {
+                        var _a = option, elementOptions = _a.elementOptions, extraKeyDiagnostics = _a.extraKeyDiagnostics, optionName = _a.name;
+                        return convertObjectLiteralExpressionToJson(objectLiteralExpression, elementOptions, extraKeyDiagnostics, optionName);
                     }
-                    else if (workStacks.subtreeFlags[stackIndex] !== undefined) {
-                        subtreeTransformFlags = workStacks.subtreeFlags[stackIndex] | computeTransformFlagsForNode(node, subtreeTransformFlags);
+                    else {
+                        return convertObjectLiteralExpressionToJson(objectLiteralExpression, /* knownOptions*/ undefined, 
+                        /*extraKeyDiagnosticMessage */ undefined, /*parentOption*/ undefined);
                     }
-                    inStrictMode = workStacks.inStrictMode[stackIndex];
-                    parent = workStacks.parent[stackIndex];
-                }
-                stackIndex--;
-            }
-            /**
-             * If `node` is a BinaryExpression, adds it to the local work stack, otherwise recursively binds it
-             */
-            function maybeBind(node) {
-                if (node && ts.isBinaryExpression(node)) {
-                    stackIndex++;
-                    workStacks.expr[stackIndex] = node;
-                    workStacks.state[stackIndex] = 0 /* BindThenBindChildren */;
-                    workStacks.inStrictMode[stackIndex] = undefined;
-                    workStacks.parent[stackIndex] = undefined;
-                    workStacks.subtreeFlags[stackIndex] = undefined;
-                }
-                else {
-                    bind(node);
-                }
-            }
-        }
-        function bindDeleteExpressionFlow(node) {
-            bindEachChild(node);
-            if (node.expression.kind === 194 /* PropertyAccessExpression */) {
-                bindAssignmentTargetFlow(node.expression);
+                case 199 /* ArrayLiteralExpression */:
+                    reportInvalidOptionValue(option && option.type !== "list");
+                    return convertArrayLiteralExpressionToJson(valueExpression.elements, option && option.element);
             }
-        }
-        function bindConditionalExpressionFlow(node) {
-            var trueLabel = createBranchLabel();
-            var falseLabel = createBranchLabel();
-            var postExpressionLabel = createBranchLabel();
-            bindCondition(node.condition, trueLabel, falseLabel);
-            currentFlow = finishFlowLabel(trueLabel);
-            bind(node.questionToken);
-            bind(node.whenTrue);
-            addAntecedent(postExpressionLabel, currentFlow);
-            currentFlow = finishFlowLabel(falseLabel);
-            bind(node.colonToken);
-            bind(node.whenFalse);
-            addAntecedent(postExpressionLabel, currentFlow);
-            currentFlow = finishFlowLabel(postExpressionLabel);
-        }
-        function bindInitializedVariableFlow(node) {
-            var name = !ts.isOmittedExpression(node) ? node.name : undefined;
-            if (ts.isBindingPattern(name)) {
-                for (var _i = 0, _a = name.elements; _i < _a.length; _i++) {
-                    var child = _a[_i];
-                    bindInitializedVariableFlow(child);
-                }
+            // Not in expected format
+            if (option) {
+                reportInvalidOptionValue(/*isError*/ true);
             }
             else {
-                currentFlow = createFlowMutation(16 /* Assignment */, currentFlow, node);
-            }
-        }
-        function bindVariableDeclarationFlow(node) {
-            bindEachChild(node);
-            if (node.initializer || ts.isForInOrOfStatement(node.parent.parent)) {
-                bindInitializedVariableFlow(node);
+                errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, ts.Diagnostics.Property_value_can_only_be_string_literal_numeric_literal_true_false_null_object_literal_or_array_literal));
             }
-        }
-        function bindJSDocTypeAlias(node) {
-            node.tagName.parent = node;
-            if (node.kind !== 316 /* JSDocEnumTag */ && node.fullName) {
-                setParentPointers(node, node.fullName);
+            return undefined;
+            function reportInvalidOptionValue(isError) {
+                if (isError) {
+                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, valueExpression, ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, option.name, getCompilerOptionValueTypeString(option)));
+                }
             }
         }
-        function bindJSDocClassTag(node) {
-            bindEachChild(node);
-            var host = ts.getHostSignatureFromJSDoc(node);
-            if (host && host.kind !== 161 /* MethodDeclaration */) {
-                addDeclarationToSymbol(host.symbol, host, 32 /* Class */);
-            }
+        function isDoubleQuotedString(node) {
+            return ts.isStringLiteral(node) && ts.isStringDoubleQuoted(node, sourceFile);
         }
-        function bindOptionalExpression(node, trueTarget, falseTarget) {
-            doWithConditionalBranches(bind, node, trueTarget, falseTarget);
-            if (!ts.isOptionalChain(node) || ts.isOutermostOptionalChain(node)) {
-                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
-                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
+    }
+    ts.convertToObjectWorker = convertToObjectWorker;
+    function getCompilerOptionValueTypeString(option) {
+        return option.type === "list" ?
+            "Array" :
+            ts.isString(option.type) ? option.type : "string";
+    }
+    function isCompilerOptionsValue(option, value) {
+        if (option) {
+            if (isNullOrUndefined(value))
+                return true; // All options are undefinable/nullable
+            if (option.type === "list") {
+                return ts.isArray(value);
             }
+            var expectedType = ts.isString(option.type) ? option.type : "string";
+            return typeof value === expectedType;
         }
-        function bindOptionalChainRest(node) {
-            switch (node.kind) {
-                case 194 /* PropertyAccessExpression */:
-                    bind(node.questionDotToken);
-                    bind(node.name);
-                    break;
-                case 195 /* ElementAccessExpression */:
-                    bind(node.questionDotToken);
-                    bind(node.argumentExpression);
-                    break;
-                case 196 /* CallExpression */:
-                    bind(node.questionDotToken);
-                    bindEach(node.typeArguments);
-                    bindEach(node.arguments);
-                    break;
+        return false;
+    }
+    /**
+     * Generate an uncommented, complete tsconfig for use with "--showConfig"
+     * @param configParseResult options to be generated into tsconfig.json
+     * @param configFileName name of the parsed config file - output paths will be generated relative to this
+     * @param host provides current directory and case sensitivity services
+     */
+    /** @internal */
+    function convertToTSConfig(configParseResult, configFileName, host) {
+        var getCanonicalFileName = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames);
+        var files = ts.map(ts.filter(configParseResult.fileNames, (!configParseResult.configFileSpecs || !configParseResult.configFileSpecs.validatedIncludeSpecs) ? function (_) { return true; } : matchesSpecs(configFileName, configParseResult.configFileSpecs.validatedIncludeSpecs, configParseResult.configFileSpecs.validatedExcludeSpecs, host)), function (f) { return ts.getRelativePathFromFile(ts.getNormalizedAbsolutePath(configFileName, host.getCurrentDirectory()), ts.getNormalizedAbsolutePath(f, host.getCurrentDirectory()), getCanonicalFileName); });
+        var optionMap = serializeCompilerOptions(configParseResult.options, { configFilePath: ts.getNormalizedAbsolutePath(configFileName, host.getCurrentDirectory()), useCaseSensitiveFileNames: host.useCaseSensitiveFileNames });
+        var watchOptionMap = configParseResult.watchOptions && serializeWatchOptions(configParseResult.watchOptions);
+        var config = __assign(__assign({ compilerOptions: __assign(__assign({}, optionMapToObject(optionMap)), { showConfig: undefined, configFile: undefined, configFilePath: undefined, help: undefined, init: undefined, listFiles: undefined, listEmittedFiles: undefined, project: undefined, build: undefined, version: undefined }), watchOptions: watchOptionMap && optionMapToObject(watchOptionMap), references: ts.map(configParseResult.projectReferences, function (r) { return (__assign(__assign({}, r), { path: r.originalPath ? r.originalPath : "", originalPath: undefined })); }), files: ts.length(files) ? files : undefined }, (configParseResult.configFileSpecs ? {
+            include: filterSameAsDefaultInclude(configParseResult.configFileSpecs.validatedIncludeSpecs),
+            exclude: configParseResult.configFileSpecs.validatedExcludeSpecs
+        } : {})), { compileOnSave: !!configParseResult.compileOnSave ? true : undefined });
+        return config;
+    }
+    ts.convertToTSConfig = convertToTSConfig;
+    function optionMapToObject(optionMap) {
+        return __assign({}, ts.arrayFrom(optionMap.entries()).reduce(function (prev, cur) {
+            var _a;
+            return (__assign(__assign({}, prev), (_a = {}, _a[cur[0]] = cur[1], _a)));
+        }, {}));
+    }
+    function filterSameAsDefaultInclude(specs) {
+        if (!ts.length(specs))
+            return undefined;
+        if (ts.length(specs) !== 1)
+            return specs;
+        if (specs[0] === "**/*")
+            return undefined;
+        return specs;
+    }
+    function matchesSpecs(path, includeSpecs, excludeSpecs, host) {
+        if (!includeSpecs)
+            return function (_) { return true; };
+        var patterns = ts.getFileMatcherPatterns(path, excludeSpecs, includeSpecs, host.useCaseSensitiveFileNames, host.getCurrentDirectory());
+        var excludeRe = patterns.excludePattern && ts.getRegexFromPattern(patterns.excludePattern, host.useCaseSensitiveFileNames);
+        var includeRe = patterns.includeFilePattern && ts.getRegexFromPattern(patterns.includeFilePattern, host.useCaseSensitiveFileNames);
+        if (includeRe) {
+            if (excludeRe) {
+                return function (path) { return !(includeRe.test(path) && !excludeRe.test(path)); };
             }
+            return function (path) { return !includeRe.test(path); };
         }
-        function bindOptionalChain(node, trueTarget, falseTarget) {
-            // For an optional chain, we emulate the behavior of a logical expression:
-            //
-            // a?.b         -> a && a.b
-            // a?.b.c       -> a && a.b.c
-            // a?.b?.c      -> a && a.b && a.b.c
-            // a?.[x = 1]   -> a && a[x = 1]
-            //
-            // To do this we descend through the chain until we reach the root of a chain (the expression with a `?.`)
-            // and build it's CFA graph as if it were the first condition (`a && ...`). Then we bind the rest
-            // of the node as part of the "true" branch, and continue to do so as we ascend back up to the outermost
-            // chain node. We then treat the entire node as the right side of the expression.
-            var preChainLabel = ts.isOptionalChainRoot(node) ? createBranchLabel() : undefined;
-            bindOptionalExpression(node.expression, preChainLabel || trueTarget, falseTarget);
-            if (preChainLabel) {
-                currentFlow = finishFlowLabel(preChainLabel);
-            }
-            doWithConditionalBranches(bindOptionalChainRest, node, trueTarget, falseTarget);
-            if (ts.isOutermostOptionalChain(node)) {
-                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
-                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
-            }
+        if (excludeRe) {
+            return function (path) { return excludeRe.test(path); };
         }
-        function bindOptionalChainFlow(node) {
-            if (isTopLevelLogicalExpression(node)) {
-                var postExpressionLabel = createBranchLabel();
-                bindOptionalChain(node, postExpressionLabel, postExpressionLabel);
-                currentFlow = finishFlowLabel(postExpressionLabel);
-            }
-            else {
-                bindOptionalChain(node, currentTrueTarget, currentFalseTarget);
-            }
+        return function (_) { return true; };
+    }
+    function getCustomTypeMapOfCommandLineOption(optionDefinition) {
+        if (optionDefinition.type === "string" || optionDefinition.type === "number" || optionDefinition.type === "boolean" || optionDefinition.type === "object") {
+            // this is of a type CommandLineOptionOfPrimitiveType
+            return undefined;
         }
-        function bindNonNullExpressionFlow(node) {
-            if (ts.isOptionalChain(node)) {
-                bindOptionalChainFlow(node);
-            }
-            else {
-                bindEachChild(node);
-            }
+        else if (optionDefinition.type === "list") {
+            return getCustomTypeMapOfCommandLineOption(optionDefinition.element);
         }
-        function bindAccessExpressionFlow(node) {
-            if (ts.isOptionalChain(node)) {
-                bindOptionalChainFlow(node);
-            }
-            else {
-                bindEachChild(node);
-            }
+        else {
+            return optionDefinition.type;
         }
-        function bindCallExpressionFlow(node) {
-            if (ts.isOptionalChain(node)) {
-                bindOptionalChainFlow(node);
-            }
-            else {
-                // If the target of the call expression is a function expression or arrow function we have
-                // an immediately invoked function expression (IIFE). Initialize the flowNode property to
-                // the current control flow (which includes evaluation of the IIFE arguments).
-                var expr = ts.skipParentheses(node.expression);
-                if (expr.kind === 201 /* FunctionExpression */ || expr.kind === 202 /* ArrowFunction */) {
-                    bindEach(node.typeArguments);
-                    bindEach(node.arguments);
-                    bind(node.expression);
-                }
-                else {
-                    bindEachChild(node);
-                }
+    }
+    function getNameOfCompilerOptionValue(value, customTypeMap) {
+        // There is a typeMap associated with this command-line option so use it to map value back to its name
+        return ts.forEachEntry(customTypeMap, function (mapValue, key) {
+            if (mapValue === value) {
+                return key;
             }
-            if (node.expression.kind === 194 /* PropertyAccessExpression */) {
-                var propertyAccess = node.expression;
-                if (ts.isIdentifier(propertyAccess.name) && isNarrowableOperand(propertyAccess.expression) && ts.isPushOrUnshiftIdentifier(propertyAccess.name)) {
-                    currentFlow = createFlowMutation(256 /* ArrayMutation */, currentFlow, node);
+        });
+    }
+    function serializeCompilerOptions(options, pathOptions) {
+        return serializeOptionBaseObject(options, getOptionsNameMap(), pathOptions);
+    }
+    function serializeWatchOptions(options) {
+        return serializeOptionBaseObject(options, getWatchOptionsNameMap());
+    }
+    function serializeOptionBaseObject(options, _a, pathOptions) {
+        var optionsNameMap = _a.optionsNameMap;
+        var result = new ts.Map();
+        var getCanonicalFileName = pathOptions && ts.createGetCanonicalFileName(pathOptions.useCaseSensitiveFileNames);
+        var _loop_5 = function (name) {
+            if (ts.hasProperty(options, name)) {
+                // tsconfig only options cannot be specified via command line,
+                // so we can assume that only types that can appear here string | number | boolean
+                if (optionsNameMap.has(name) && optionsNameMap.get(name).category === ts.Diagnostics.Command_line_Options) {
+                    return "continue";
                 }
-            }
-        }
-        function getContainerFlags(node) {
-            switch (node.kind) {
-                case 214 /* ClassExpression */:
-                case 245 /* ClassDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 193 /* ObjectLiteralExpression */:
-                case 173 /* TypeLiteral */:
-                case 304 /* JSDocTypeLiteral */:
-                case 274 /* JsxAttributes */:
-                    return 1 /* IsContainer */;
-                case 246 /* InterfaceDeclaration */:
-                    return 1 /* IsContainer */ | 64 /* IsInterface */;
-                case 249 /* ModuleDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 186 /* MappedType */:
-                    return 1 /* IsContainer */ | 32 /* HasLocals */;
-                case 290 /* SourceFile */:
-                    return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */;
-                case 161 /* MethodDeclaration */:
-                    if (ts.isObjectLiteralOrClassExpressionMethod(node)) {
-                        return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */ | 8 /* IsFunctionLike */ | 128 /* IsObjectLiteralOrClassExpressionMethod */;
+                var value = options[name];
+                var optionDefinition = optionsNameMap.get(name.toLowerCase());
+                if (optionDefinition) {
+                    var customTypeMap_1 = getCustomTypeMapOfCommandLineOption(optionDefinition);
+                    if (!customTypeMap_1) {
+                        // There is no map associated with this compiler option then use the value as-is
+                        // This is the case if the value is expect to be string, number, boolean or list of string
+                        if (pathOptions && optionDefinition.isFilePath) {
+                            result.set(name, ts.getRelativePathFromFile(pathOptions.configFilePath, ts.getNormalizedAbsolutePath(value, ts.getDirectoryPath(pathOptions.configFilePath)), getCanonicalFileName));
+                        }
+                        else {
+                            result.set(name, value);
+                        }
                     }
-                // falls through
-                case 162 /* Constructor */:
-                case 244 /* FunctionDeclaration */:
-                case 160 /* MethodSignature */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 165 /* CallSignature */:
-                case 305 /* JSDocSignature */:
-                case 300 /* JSDocFunctionType */:
-                case 170 /* FunctionType */:
-                case 166 /* ConstructSignature */:
-                case 167 /* IndexSignature */:
-                case 171 /* ConstructorType */:
-                    return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */ | 8 /* IsFunctionLike */;
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                    return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */ | 8 /* IsFunctionLike */ | 16 /* IsFunctionExpression */;
-                case 250 /* ModuleBlock */:
-                    return 4 /* IsControlFlowContainer */;
-                case 159 /* PropertyDeclaration */:
-                    return node.initializer ? 4 /* IsControlFlowContainer */ : 0;
-                case 280 /* CatchClause */:
-                case 230 /* ForStatement */:
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
-                case 251 /* CaseBlock */:
-                    return 2 /* IsBlockScopedContainer */;
-                case 223 /* Block */:
-                    // do not treat blocks directly inside a function as a block-scoped-container.
-                    // Locals that reside in this block should go to the function locals. Otherwise 'x'
-                    // would not appear to be a redeclaration of a block scoped local in the following
-                    // example:
-                    //
-                    //      function foo() {
-                    //          var x;
-                    //          let x;
-                    //      }
-                    //
-                    // If we placed 'var x' into the function locals and 'let x' into the locals of
-                    // the block, then there would be no collision.
-                    //
-                    // By not creating a new block-scoped-container here, we ensure that both 'var x'
-                    // and 'let x' go into the Function-container's locals, and we do get a collision
-                    // conflict.
-                    return ts.isFunctionLike(node.parent) ? 0 /* None */ : 2 /* IsBlockScopedContainer */;
-            }
-            return 0 /* None */;
-        }
-        function addToContainerChain(next) {
-            if (lastContainer) {
-                lastContainer.nextContainer = next;
-            }
-            lastContainer = next;
-        }
-        function declareSymbolAndAddToSymbolTable(node, symbolFlags, symbolExcludes) {
-            switch (container.kind) {
-                // Modules, source files, and classes need specialized handling for how their
-                // members are declared (for example, a member of a class will go into a specific
-                // symbol table depending on if it is static or not). We defer to specialized
-                // handlers to take care of declaring these child members.
-                case 249 /* ModuleDeclaration */:
-                    return declareModuleMember(node, symbolFlags, symbolExcludes);
-                case 290 /* SourceFile */:
-                    return declareSourceFileMember(node, symbolFlags, symbolExcludes);
-                case 214 /* ClassExpression */:
-                case 245 /* ClassDeclaration */:
-                    return declareClassMember(node, symbolFlags, symbolExcludes);
-                case 248 /* EnumDeclaration */:
-                    return declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes);
-                case 173 /* TypeLiteral */:
-                case 304 /* JSDocTypeLiteral */:
-                case 193 /* ObjectLiteralExpression */:
-                case 246 /* InterfaceDeclaration */:
-                case 274 /* JsxAttributes */:
-                    // Interface/Object-types always have their children added to the 'members' of
-                    // their container. They are only accessible through an instance of their
-                    // container, and are never in scope otherwise (even inside the body of the
-                    // object / type / interface declaring them). An exception is type parameters,
-                    // which are in scope without qualification (similar to 'locals').
-                    return declareSymbol(container.symbol.members, container.symbol, node, symbolFlags, symbolExcludes);
-                case 170 /* FunctionType */:
-                case 171 /* ConstructorType */:
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 305 /* JSDocSignature */:
-                case 167 /* IndexSignature */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 162 /* Constructor */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 300 /* JSDocFunctionType */:
-                case 322 /* JSDocTypedefTag */:
-                case 315 /* JSDocCallbackTag */:
-                case 247 /* TypeAliasDeclaration */:
-                case 186 /* MappedType */:
-                    // All the children of these container types are never visible through another
-                    // symbol (i.e. through another symbol's 'exports' or 'members').  Instead,
-                    // they're only accessed 'lexically' (i.e. from code that exists underneath
-                    // their container in the tree). To accomplish this, we simply add their declared
-                    // symbol to the 'locals' of the container.  These symbols can then be found as
-                    // the type checker walks up the containers, checking them for matching names.
-                    return declareSymbol(container.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
-            }
-        }
-        function declareClassMember(node, symbolFlags, symbolExcludes) {
-            return ts.hasModifier(node, 32 /* Static */)
-                ? declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes)
-                : declareSymbol(container.symbol.members, container.symbol, node, symbolFlags, symbolExcludes);
-        }
-        function declareSourceFileMember(node, symbolFlags, symbolExcludes) {
-            return ts.isExternalModule(file)
-                ? declareModuleMember(node, symbolFlags, symbolExcludes)
-                : declareSymbol(file.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
-        }
-        function hasExportDeclarations(node) {
-            var body = ts.isSourceFile(node) ? node : ts.tryCast(node.body, ts.isModuleBlock);
-            return !!body && body.statements.some(function (s) { return ts.isExportDeclaration(s) || ts.isExportAssignment(s); });
-        }
-        function setExportContextFlag(node) {
-            // A declaration source file or ambient module declaration that contains no export declarations (but possibly regular
-            // declarations with export modifiers) is an export context in which declarations are implicitly exported.
-            if (node.flags & 8388608 /* Ambient */ && !hasExportDeclarations(node)) {
-                node.flags |= 64 /* ExportContext */;
-            }
-            else {
-                node.flags &= ~64 /* ExportContext */;
-            }
-        }
-        function bindModuleDeclaration(node) {
-            setExportContextFlag(node);
-            if (ts.isAmbientModule(node)) {
-                if (ts.hasModifier(node, 1 /* Export */)) {
-                    errorOnFirstToken(node, ts.Diagnostics.export_modifier_cannot_be_applied_to_ambient_modules_and_module_augmentations_since_they_are_always_visible);
-                }
-                if (ts.isModuleAugmentationExternal(node)) {
-                    declareModuleSymbol(node);
-                }
-                else {
-                    var pattern = void 0;
-                    if (node.name.kind === 10 /* StringLiteral */) {
-                        var text = node.name.text;
-                        if (ts.hasZeroOrOneAsteriskCharacter(text)) {
-                            pattern = ts.tryParsePattern(text);
+                    else {
+                        if (optionDefinition.type === "list") {
+                            result.set(name, value.map(function (element) { return getNameOfCompilerOptionValue(element, customTypeMap_1); })); // TODO: GH#18217
                         }
                         else {
-                            errorOnFirstToken(node.name, ts.Diagnostics.Pattern_0_can_have_at_most_one_Asterisk_character, text);
+                            // There is a typeMap associated with this command-line option so use it to map value back to its name
+                            result.set(name, getNameOfCompilerOptionValue(value, customTypeMap_1));
                         }
                     }
-                    var symbol = declareSymbolAndAddToSymbolTable(node, 512 /* ValueModule */, 110735 /* ValueModuleExcludes */);
-                    file.patternAmbientModules = ts.append(file.patternAmbientModules, pattern && { pattern: pattern, symbol: symbol });
-                }
-            }
-            else {
-                var state = declareModuleSymbol(node);
-                if (state !== 0 /* NonInstantiated */) {
-                    var symbol = node.symbol;
-                    // if module was already merged with some function, class or non-const enum, treat it as non-const-enum-only
-                    symbol.constEnumOnlyModule = (!(symbol.flags & (16 /* Function */ | 32 /* Class */ | 256 /* RegularEnum */)))
-                        // Current must be `const enum` only
-                        && state === 2 /* ConstEnumOnly */
-                        // Can't have been set to 'false' in a previous merged symbol. ('undefined' OK)
-                        && symbol.constEnumOnlyModule !== false;
                 }
             }
+        };
+        for (var name in options) {
+            _loop_5(name);
         }
-        function declareModuleSymbol(node) {
-            var state = getModuleInstanceState(node);
-            var instantiated = state !== 0 /* NonInstantiated */;
-            declareSymbolAndAddToSymbolTable(node, instantiated ? 512 /* ValueModule */ : 1024 /* NamespaceModule */, instantiated ? 110735 /* ValueModuleExcludes */ : 0 /* NamespaceModuleExcludes */);
-            return state;
-        }
-        function bindFunctionOrConstructorType(node) {
-            // For a given function symbol "<...>(...) => T" we want to generate a symbol identical
-            // to the one we would get for: { <...>(...): T }
-            //
-            // We do that by making an anonymous type literal symbol, and then setting the function
-            // symbol as its sole member. To the rest of the system, this symbol will be indistinguishable
-            // from an actual type literal symbol you would have gotten had you used the long form.
-            var symbol = createSymbol(131072 /* Signature */, getDeclarationName(node)); // TODO: GH#18217
-            addDeclarationToSymbol(symbol, node, 131072 /* Signature */);
-            var typeLiteralSymbol = createSymbol(2048 /* TypeLiteral */, "__type" /* Type */);
-            addDeclarationToSymbol(typeLiteralSymbol, node, 2048 /* TypeLiteral */);
-            typeLiteralSymbol.members = ts.createSymbolTable();
-            typeLiteralSymbol.members.set(symbol.escapedName, symbol);
-        }
-        function bindObjectLiteralExpression(node) {
-            var ElementKind;
-            (function (ElementKind) {
-                ElementKind[ElementKind["Property"] = 1] = "Property";
-                ElementKind[ElementKind["Accessor"] = 2] = "Accessor";
-            })(ElementKind || (ElementKind = {}));
-            if (inStrictMode && !ts.isAssignmentTarget(node)) {
-                var seen = ts.createUnderscoreEscapedMap();
-                for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
-                    var prop = _a[_i];
-                    if (prop.kind === 283 /* SpreadAssignment */ || prop.name.kind !== 75 /* Identifier */) {
-                        continue;
-                    }
-                    var identifier = prop.name;
-                    // ECMA-262 11.1.5 Object Initializer
-                    // If previous is not undefined then throw a SyntaxError exception if any of the following conditions are true
-                    // a.This production is contained in strict code and IsDataDescriptor(previous) is true and
-                    // IsDataDescriptor(propId.descriptor) is true.
-                    //    b.IsDataDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true.
-                    //    c.IsAccessorDescriptor(previous) is true and IsDataDescriptor(propId.descriptor) is true.
-                    //    d.IsAccessorDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true
-                    // and either both previous and propId.descriptor have[[Get]] fields or both previous and propId.descriptor have[[Set]] fields
-                    var currentKind = prop.kind === 281 /* PropertyAssignment */ || prop.kind === 282 /* ShorthandPropertyAssignment */ || prop.kind === 161 /* MethodDeclaration */
-                        ? 1 /* Property */
-                        : 2 /* Accessor */;
-                    var existingKind = seen.get(identifier.escapedText);
-                    if (!existingKind) {
-                        seen.set(identifier.escapedText, currentKind);
-                        continue;
-                    }
-                    if (currentKind === 1 /* Property */ && existingKind === 1 /* Property */) {
-                        var span = ts.getErrorSpanForNode(file, identifier);
-                        file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, ts.Diagnostics.An_object_literal_cannot_have_multiple_properties_with_the_same_name_in_strict_mode));
-                    }
-                }
+        return result;
+    }
+    /**
+     * Generate tsconfig configuration when running command line "--init"
+     * @param options commandlineOptions to be generated into tsconfig.json
+     * @param fileNames array of filenames to be generated into tsconfig.json
+     */
+    /* @internal */
+    function generateTSConfig(options, fileNames, newLine) {
+        var compilerOptions = ts.extend(options, ts.defaultInitCompilerOptions);
+        var compilerOptionsMap = serializeCompilerOptions(compilerOptions);
+        return writeConfigurations();
+        function getDefaultValueForOption(option) {
+            switch (option.type) {
+                case "number":
+                    return 1;
+                case "boolean":
+                    return true;
+                case "string":
+                    return option.isFilePath ? "./" : "";
+                case "list":
+                    return [];
+                case "object":
+                    return {};
+                default:
+                    var iterResult = option.type.keys().next();
+                    if (!iterResult.done)
+                        return iterResult.value;
+                    return ts.Debug.fail("Expected 'option.type' to have entries.");
             }
-            return bindAnonymousDeclaration(node, 4096 /* ObjectLiteral */, "__object" /* Object */);
         }
-        function bindJsxAttributes(node) {
-            return bindAnonymousDeclaration(node, 4096 /* ObjectLiteral */, "__jsxAttributes" /* JSXAttributes */);
+        function makePadding(paddingLength) {
+            return Array(paddingLength + 1).join(" ");
         }
-        function bindJsxAttribute(node, symbolFlags, symbolExcludes) {
-            return declareSymbolAndAddToSymbolTable(node, symbolFlags, symbolExcludes);
+        function isAllowedOption(_a) {
+            var category = _a.category, name = _a.name;
+            // Skip options which do not have a category or have category `Command_line_Options`
+            // Exclude all possible `Advanced_Options` in tsconfig.json which were NOT defined in command line
+            return category !== undefined
+                && category !== ts.Diagnostics.Command_line_Options
+                && (category !== ts.Diagnostics.Advanced_Options || compilerOptionsMap.has(name));
         }
-        function bindAnonymousDeclaration(node, symbolFlags, name) {
-            var symbol = createSymbol(symbolFlags, name);
-            if (symbolFlags & (8 /* EnumMember */ | 106500 /* ClassMember */)) {
-                symbol.parent = container.symbol;
+        function writeConfigurations() {
+            // Filter applicable options to place in the file
+            var categorizedOptions = ts.createMultiMap();
+            for (var _i = 0, optionDeclarations_1 = ts.optionDeclarations; _i < optionDeclarations_1.length; _i++) {
+                var option = optionDeclarations_1[_i];
+                var category = option.category;
+                if (isAllowedOption(option)) {
+                    categorizedOptions.add(ts.getLocaleSpecificMessage(category), option);
+                }
             }
-            addDeclarationToSymbol(symbol, node, symbolFlags);
-            return symbol;
-        }
-        function bindBlockScopedDeclaration(node, symbolFlags, symbolExcludes) {
-            switch (blockScopeContainer.kind) {
-                case 249 /* ModuleDeclaration */:
-                    declareModuleMember(node, symbolFlags, symbolExcludes);
-                    break;
-                case 290 /* SourceFile */:
-                    if (ts.isExternalOrCommonJsModule(container)) {
-                        declareModuleMember(node, symbolFlags, symbolExcludes);
-                        break;
-                    }
-                // falls through
-                default:
-                    if (!blockScopeContainer.locals) {
-                        blockScopeContainer.locals = ts.createSymbolTable();
-                        addToContainerChain(blockScopeContainer);
+            // Serialize all options and their descriptions
+            var marginLength = 0;
+            var seenKnownKeys = 0;
+            var entries = [];
+            categorizedOptions.forEach(function (options, category) {
+                if (entries.length !== 0) {
+                    entries.push({ value: "" });
+                }
+                entries.push({ value: "/* " + category + " */" });
+                for (var _i = 0, options_1 = options; _i < options_1.length; _i++) {
+                    var option = options_1[_i];
+                    var optionName = void 0;
+                    if (compilerOptionsMap.has(option.name)) {
+                        optionName = "\"" + option.name + "\": " + JSON.stringify(compilerOptionsMap.get(option.name)) + ((seenKnownKeys += 1) === compilerOptionsMap.size ? "" : ",");
                     }
-                    declareSymbol(blockScopeContainer.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
-            }
-        }
-        function delayedBindJSDocTypedefTag() {
-            if (!delayedTypeAliases) {
-                return;
-            }
-            var saveContainer = container;
-            var saveLastContainer = lastContainer;
-            var saveBlockScopeContainer = blockScopeContainer;
-            var saveParent = parent;
-            var saveCurrentFlow = currentFlow;
-            for (var _i = 0, delayedTypeAliases_1 = delayedTypeAliases; _i < delayedTypeAliases_1.length; _i++) {
-                var typeAlias = delayedTypeAliases_1[_i];
-                var host = ts.getJSDocHost(typeAlias);
-                container = ts.findAncestor(host.parent, function (n) { return !!(getContainerFlags(n) & 1 /* IsContainer */); }) || file;
-                blockScopeContainer = ts.getEnclosingBlockScopeContainer(host) || file;
-                currentFlow = initFlowNode({ flags: 2 /* Start */ });
-                parent = typeAlias;
-                bind(typeAlias.typeExpression);
-                var declName = ts.getNameOfDeclaration(typeAlias);
-                if ((ts.isJSDocEnumTag(typeAlias) || !typeAlias.fullName) && declName && ts.isPropertyAccessEntityNameExpression(declName.parent)) {
-                    // typedef anchored to an A.B.C assignment - we need to bind into B's namespace under name C
-                    var isTopLevel = isTopLevelNamespaceAssignment(declName.parent);
-                    if (isTopLevel) {
-                        bindPotentiallyMissingNamespaces(file.symbol, declName.parent, isTopLevel, !!ts.findAncestor(declName, function (d) { return ts.isPropertyAccessExpression(d) && d.name.escapedText === "prototype"; }), /*containerIsClass*/ false);
-                        var oldContainer = container;
-                        switch (ts.getAssignmentDeclarationPropertyAccessKind(declName.parent)) {
-                            case 1 /* ExportsProperty */:
-                            case 2 /* ModuleExports */:
-                                if (!ts.isExternalOrCommonJsModule(file)) {
-                                    container = undefined;
-                                }
-                                else {
-                                    container = file;
-                                }
-                                break;
-                            case 4 /* ThisProperty */:
-                                container = declName.parent.expression;
-                                break;
-                            case 3 /* PrototypeProperty */:
-                                container = declName.parent.expression.name;
-                                break;
-                            case 5 /* Property */:
-                                container = isExportsOrModuleExportsOrAlias(file, declName.parent.expression) ? file
-                                    : ts.isPropertyAccessExpression(declName.parent.expression) ? declName.parent.expression.name
-                                        : declName.parent.expression;
-                                break;
-                            case 0 /* None */:
-                                return ts.Debug.fail("Shouldn't have detected typedef or enum on non-assignment declaration");
-                        }
-                        if (container) {
-                            declareModuleMember(typeAlias, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
-                        }
-                        container = oldContainer;
+                    else {
+                        optionName = "// \"" + option.name + "\": " + JSON.stringify(getDefaultValueForOption(option)) + ",";
                     }
+                    entries.push({
+                        value: optionName,
+                        description: "/* " + (option.description && ts.getLocaleSpecificMessage(option.description) || option.name) + " */"
+                    });
+                    marginLength = Math.max(optionName.length, marginLength);
                 }
-                else if (ts.isJSDocEnumTag(typeAlias) || !typeAlias.fullName || typeAlias.fullName.kind === 75 /* Identifier */) {
-                    parent = typeAlias.parent;
-                    bindBlockScopedDeclaration(typeAlias, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
-                }
-                else {
-                    bind(typeAlias.fullName);
-                }
+            });
+            // Write the output
+            var tab = makePadding(2);
+            var result = [];
+            result.push("{");
+            result.push(tab + "\"compilerOptions\": {");
+            result.push("" + tab + tab + "/* " + ts.getLocaleSpecificMessage(ts.Diagnostics.Visit_https_Colon_Slash_Slashaka_ms_Slashtsconfig_json_to_read_more_about_this_file) + " */");
+            result.push("");
+            // Print out each row, aligning all the descriptions on the same column.
+            for (var _a = 0, entries_2 = entries; _a < entries_2.length; _a++) {
+                var entry = entries_2[_a];
+                var value = entry.value, _b = entry.description, description = _b === void 0 ? "" : _b;
+                result.push(value && "" + tab + tab + value + (description && (makePadding(marginLength - value.length + 2) + description)));
             }
-            container = saveContainer;
-            lastContainer = saveLastContainer;
-            blockScopeContainer = saveBlockScopeContainer;
-            parent = saveParent;
-            currentFlow = saveCurrentFlow;
-        }
-        // The binder visits every node in the syntax tree so it is a convenient place to perform a single localized
-        // check for reserved words used as identifiers in strict mode code.
-        function checkStrictModeIdentifier(node) {
-            if (inStrictMode &&
-                node.originalKeywordKind >= 113 /* FirstFutureReservedWord */ &&
-                node.originalKeywordKind <= 121 /* LastFutureReservedWord */ &&
-                !ts.isIdentifierName(node) &&
-                !(node.flags & 8388608 /* Ambient */) &&
-                !(node.flags & 4194304 /* JSDoc */)) {
-                // Report error only if there are no parse errors in file
-                if (!file.parseDiagnostics.length) {
-                    file.bindDiagnostics.push(createDiagnosticForNode(node, getStrictModeIdentifierMessage(node), ts.declarationNameToString(node)));
+            if (fileNames.length) {
+                result.push(tab + "},");
+                result.push(tab + "\"files\": [");
+                for (var i = 0; i < fileNames.length; i++) {
+                    result.push("" + tab + tab + JSON.stringify(fileNames[i]) + (i === fileNames.length - 1 ? "" : ","));
                 }
+                result.push(tab + "]");
             }
-        }
-        function getStrictModeIdentifierMessage(node) {
-            // Provide specialized messages to help the user understand why we think they're in
-            // strict mode.
-            if (ts.getContainingClass(node)) {
-                return ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode_Class_definitions_are_automatically_in_strict_mode;
-            }
-            if (file.externalModuleIndicator) {
-                return ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode_Modules_are_automatically_in_strict_mode;
+            else {
+                result.push(tab + "}");
             }
-            return ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode;
+            result.push("}");
+            return result.join(newLine) + newLine;
         }
-        // The binder visits every node, so this is a good place to check for
-        // the reserved private name (there is only one)
-        function checkPrivateIdentifier(node) {
-            if (node.escapedText === "#constructor") {
-                // Report error only if there are no parse errors in file
-                if (!file.parseDiagnostics.length) {
-                    file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.constructor_is_a_reserved_word, ts.declarationNameToString(node)));
-                }
+    }
+    ts.generateTSConfig = generateTSConfig;
+    /* @internal */
+    function convertToOptionsWithAbsolutePaths(options, toAbsolutePath) {
+        var result = {};
+        var optionsNameMap = getOptionsNameMap().optionsNameMap;
+        for (var name in options) {
+            if (ts.hasProperty(options, name)) {
+                result[name] = convertToOptionValueWithAbsolutePaths(optionsNameMap.get(name.toLowerCase()), options[name], toAbsolutePath);
             }
         }
-        function checkStrictModeBinaryExpression(node) {
-            if (inStrictMode && ts.isLeftHandSideExpression(node.left) && ts.isAssignmentOperator(node.operatorToken.kind)) {
-                // ECMA 262 (Annex C) The identifier eval or arguments may not appear as the LeftHandSideExpression of an
-                // Assignment operator(11.13) or of a PostfixExpression(11.3)
-                checkStrictModeEvalOrArguments(node, node.left);
-            }
+        if (result.configFilePath) {
+            result.configFilePath = toAbsolutePath(result.configFilePath);
         }
-        function checkStrictModeCatchClause(node) {
-            // It is a SyntaxError if a TryStatement with a Catch occurs within strict code and the Identifier of the
-            // Catch production is eval or arguments
-            if (inStrictMode && node.variableDeclaration) {
-                checkStrictModeEvalOrArguments(node, node.variableDeclaration.name);
+        return result;
+    }
+    ts.convertToOptionsWithAbsolutePaths = convertToOptionsWithAbsolutePaths;
+    function convertToOptionValueWithAbsolutePaths(option, value, toAbsolutePath) {
+        if (option && !isNullOrUndefined(value)) {
+            if (option.type === "list") {
+                var values = value;
+                if (option.element.isFilePath && values.length) {
+                    return values.map(toAbsolutePath);
+                }
             }
-        }
-        function checkStrictModeDeleteExpression(node) {
-            // Grammar checking
-            if (inStrictMode && node.expression.kind === 75 /* Identifier */) {
-                // When a delete operator occurs within strict mode code, a SyntaxError is thrown if its
-                // UnaryExpression is a direct reference to a variable, function argument, or function name
-                var span = ts.getErrorSpanForNode(file, node.expression);
-                file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, ts.Diagnostics.delete_cannot_be_called_on_an_identifier_in_strict_mode));
+            else if (option.isFilePath) {
+                return toAbsolutePath(value);
             }
         }
-        function isEvalOrArgumentsIdentifier(node) {
-            return ts.isIdentifier(node) && (node.escapedText === "eval" || node.escapedText === "arguments");
+        return value;
+    }
+    /**
+     * Parse the contents of a config file (tsconfig.json).
+     * @param json The contents of the config file to parse
+     * @param host Instance of ParseConfigHost used to enumerate files in folder.
+     * @param basePath A root directory to resolve relative path entries in the config
+     *    file to. e.g. outDir
+     */
+    function parseJsonConfigFileContent(json, host, basePath, existingOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache, existingWatchOptions) {
+        return parseJsonConfigFileContentWorker(json, /*sourceFile*/ undefined, host, basePath, existingOptions, existingWatchOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache);
+    }
+    ts.parseJsonConfigFileContent = parseJsonConfigFileContent;
+    /**
+     * Parse the contents of a config file (tsconfig.json).
+     * @param jsonNode The contents of the config file to parse
+     * @param host Instance of ParseConfigHost used to enumerate files in folder.
+     * @param basePath A root directory to resolve relative path entries in the config
+     *    file to. e.g. outDir
+     */
+    function parseJsonSourceFileConfigFileContent(sourceFile, host, basePath, existingOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache, existingWatchOptions) {
+        return parseJsonConfigFileContentWorker(/*json*/ undefined, sourceFile, host, basePath, existingOptions, existingWatchOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache);
+    }
+    ts.parseJsonSourceFileConfigFileContent = parseJsonSourceFileConfigFileContent;
+    /*@internal*/
+    function setConfigFileInOptions(options, configFile) {
+        if (configFile) {
+            Object.defineProperty(options, "configFile", { enumerable: false, writable: false, value: configFile });
         }
-        function checkStrictModeEvalOrArguments(contextNode, name) {
-            if (name && name.kind === 75 /* Identifier */) {
-                var identifier = name;
-                if (isEvalOrArgumentsIdentifier(identifier)) {
-                    // We check first if the name is inside class declaration or class expression; if so give explicit message
-                    // otherwise report generic error message.
-                    var span = ts.getErrorSpanForNode(file, name);
-                    file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, getStrictModeEvalOrArgumentsMessage(contextNode), ts.idText(identifier)));
+    }
+    ts.setConfigFileInOptions = setConfigFileInOptions;
+    function isNullOrUndefined(x) {
+        return x === undefined || x === null; // eslint-disable-line no-null/no-null
+    }
+    function directoryOfCombinedPath(fileName, basePath) {
+        // Use the `getNormalizedAbsolutePath` function to avoid canonicalizing the path, as it must remain noncanonical
+        // until consistent casing errors are reported
+        return ts.getDirectoryPath(ts.getNormalizedAbsolutePath(fileName, basePath));
+    }
+    /**
+     * Parse the contents of a config file from json or json source file (tsconfig.json).
+     * @param json The contents of the config file to parse
+     * @param sourceFile sourceFile corresponding to the Json
+     * @param host Instance of ParseConfigHost used to enumerate files in folder.
+     * @param basePath A root directory to resolve relative path entries in the config
+     *    file to. e.g. outDir
+     * @param resolutionStack Only present for backwards-compatibility. Should be empty.
+     */
+    function parseJsonConfigFileContentWorker(json, sourceFile, host, basePath, existingOptions, existingWatchOptions, configFileName, resolutionStack, extraFileExtensions, extendedConfigCache) {
+        if (existingOptions === void 0) { existingOptions = {}; }
+        if (resolutionStack === void 0) { resolutionStack = []; }
+        if (extraFileExtensions === void 0) { extraFileExtensions = []; }
+        ts.Debug.assert((json === undefined && sourceFile !== undefined) || (json !== undefined && sourceFile === undefined));
+        var errors = [];
+        var parsedConfig = parseConfig(json, sourceFile, host, basePath, configFileName, resolutionStack, errors, extendedConfigCache);
+        var raw = parsedConfig.raw;
+        var options = ts.extend(existingOptions, parsedConfig.options || {});
+        var watchOptions = existingWatchOptions && parsedConfig.watchOptions ?
+            ts.extend(existingWatchOptions, parsedConfig.watchOptions) :
+            parsedConfig.watchOptions || existingWatchOptions;
+        options.configFilePath = configFileName && ts.normalizeSlashes(configFileName);
+        setConfigFileInOptions(options, sourceFile);
+        var projectReferences;
+        var _a = getFileNames(), fileNames = _a.fileNames, wildcardDirectories = _a.wildcardDirectories, spec = _a.spec;
+        return {
+            options: options,
+            watchOptions: watchOptions,
+            fileNames: fileNames,
+            projectReferences: projectReferences,
+            typeAcquisition: parsedConfig.typeAcquisition || getDefaultTypeAcquisition(),
+            raw: raw,
+            errors: errors,
+            wildcardDirectories: wildcardDirectories,
+            compileOnSave: !!raw.compileOnSave,
+            configFileSpecs: spec
+        };
+        function getFileNames() {
+            var referencesOfRaw = getPropFromRaw("references", function (element) { return typeof element === "object"; }, "object");
+            if (ts.isArray(referencesOfRaw)) {
+                for (var _i = 0, referencesOfRaw_1 = referencesOfRaw; _i < referencesOfRaw_1.length; _i++) {
+                    var ref = referencesOfRaw_1[_i];
+                    if (typeof ref.path !== "string") {
+                        createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "reference.path", "string");
+                    }
+                    else {
+                        (projectReferences || (projectReferences = [])).push({
+                            path: ts.getNormalizedAbsolutePath(ref.path, basePath),
+                            originalPath: ref.path,
+                            prepend: ref.prepend,
+                            circular: ref.circular
+                        });
+                    }
                 }
             }
-        }
-        function getStrictModeEvalOrArgumentsMessage(node) {
-            // Provide specialized messages to help the user understand why we think they're in
-            // strict mode.
-            if (ts.getContainingClass(node)) {
-                return ts.Diagnostics.Invalid_use_of_0_Class_definitions_are_automatically_in_strict_mode;
-            }
-            if (file.externalModuleIndicator) {
-                return ts.Diagnostics.Invalid_use_of_0_Modules_are_automatically_in_strict_mode;
+            var filesSpecs = toPropValue(getSpecsFromRaw("files"));
+            if (filesSpecs) {
+                var hasZeroOrNoReferences = referencesOfRaw === "no-prop" || ts.isArray(referencesOfRaw) && referencesOfRaw.length === 0;
+                var hasExtends = ts.hasProperty(raw, "extends");
+                if (filesSpecs.length === 0 && hasZeroOrNoReferences && !hasExtends) {
+                    if (sourceFile) {
+                        var fileName = configFileName || "tsconfig.json";
+                        var diagnosticMessage = ts.Diagnostics.The_files_list_in_config_file_0_is_empty;
+                        var nodeValue = ts.firstDefined(ts.getTsConfigPropArray(sourceFile, "files"), function (property) { return property.initializer; });
+                        var error = nodeValue
+                            ? ts.createDiagnosticForNodeInSourceFile(sourceFile, nodeValue, diagnosticMessage, fileName)
+                            : ts.createCompilerDiagnostic(diagnosticMessage, fileName);
+                        errors.push(error);
+                    }
+                    else {
+                        createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.The_files_list_in_config_file_0_is_empty, configFileName || "tsconfig.json");
+                    }
+                }
             }
-            return ts.Diagnostics.Invalid_use_of_0_in_strict_mode;
-        }
-        function checkStrictModeFunctionName(node) {
-            if (inStrictMode) {
-                // It is a SyntaxError if the identifier eval or arguments appears within a FormalParameterList of a strict mode FunctionDeclaration or FunctionExpression (13.1))
-                checkStrictModeEvalOrArguments(node, node.name);
+            var includeSpecs = toPropValue(getSpecsFromRaw("include"));
+            var excludeOfRaw = getSpecsFromRaw("exclude");
+            var excludeSpecs = toPropValue(excludeOfRaw);
+            if (excludeOfRaw === "no-prop" && raw.compilerOptions) {
+                var outDir = raw.compilerOptions.outDir;
+                var declarationDir = raw.compilerOptions.declarationDir;
+                if (outDir || declarationDir) {
+                    excludeSpecs = [outDir, declarationDir].filter(function (d) { return !!d; });
+                }
             }
-        }
-        function getStrictModeBlockScopeFunctionDeclarationMessage(node) {
-            // Provide specialized messages to help the user understand why we think they're in
-            // strict mode.
-            if (ts.getContainingClass(node)) {
-                return ts.Diagnostics.Function_declarations_are_not_allowed_inside_blocks_in_strict_mode_when_targeting_ES3_or_ES5_Class_definitions_are_automatically_in_strict_mode;
+            if (filesSpecs === undefined && includeSpecs === undefined) {
+                includeSpecs = ["**/*"];
             }
-            if (file.externalModuleIndicator) {
-                return ts.Diagnostics.Function_declarations_are_not_allowed_inside_blocks_in_strict_mode_when_targeting_ES3_or_ES5_Modules_are_automatically_in_strict_mode;
+            var result = matchFileNames(filesSpecs, includeSpecs, excludeSpecs, configFileName ? directoryOfCombinedPath(configFileName, basePath) : basePath, options, host, errors, extraFileExtensions, sourceFile);
+            if (shouldReportNoInputFiles(result, canJsonReportNoInputFiles(raw), resolutionStack)) {
+                errors.push(getErrorForNoInputFiles(result.spec, configFileName));
             }
-            return ts.Diagnostics.Function_declarations_are_not_allowed_inside_blocks_in_strict_mode_when_targeting_ES3_or_ES5;
+            return result;
         }
-        function checkStrictModeFunctionDeclaration(node) {
-            if (languageVersion < 2 /* ES2015 */) {
-                // Report error if function is not top level function declaration
-                if (blockScopeContainer.kind !== 290 /* SourceFile */ &&
-                    blockScopeContainer.kind !== 249 /* ModuleDeclaration */ &&
-                    !ts.isFunctionLike(blockScopeContainer)) {
-                    // We check first if the name is inside class declaration or class expression; if so give explicit message
-                    // otherwise report generic error message.
-                    var errorSpan = ts.getErrorSpanForNode(file, node);
-                    file.bindDiagnostics.push(ts.createFileDiagnostic(file, errorSpan.start, errorSpan.length, getStrictModeBlockScopeFunctionDeclarationMessage(node)));
+        function toPropValue(specResult) {
+            return ts.isArray(specResult) ? specResult : undefined;
+        }
+        function getSpecsFromRaw(prop) {
+            return getPropFromRaw(prop, ts.isString, "string");
+        }
+        function getPropFromRaw(prop, validateElement, elementTypeName) {
+            if (ts.hasProperty(raw, prop) && !isNullOrUndefined(raw[prop])) {
+                if (ts.isArray(raw[prop])) {
+                    var result = raw[prop];
+                    if (!sourceFile && !ts.every(result, validateElement)) {
+                        errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, prop, elementTypeName));
+                    }
+                    return result;
+                }
+                else {
+                    createCompilerDiagnosticOnlyIfJson(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, prop, "Array");
+                    return "not-array";
                 }
             }
+            return "no-prop";
         }
-        function checkStrictModeNumericLiteral(node) {
-            if (inStrictMode && node.numericLiteralFlags & 32 /* Octal */) {
-                file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Octal_literals_are_not_allowed_in_strict_mode));
+        function createCompilerDiagnosticOnlyIfJson(message, arg0, arg1) {
+            if (!sourceFile) {
+                errors.push(ts.createCompilerDiagnostic(message, arg0, arg1));
             }
         }
-        function checkStrictModePostfixUnaryExpression(node) {
-            // Grammar checking
-            // The identifier eval or arguments may not appear as the LeftHandSideExpression of an
-            // Assignment operator(11.13) or of a PostfixExpression(11.3) or as the UnaryExpression
-            // operated upon by a Prefix Increment(11.4.4) or a Prefix Decrement(11.4.5) operator.
-            if (inStrictMode) {
-                checkStrictModeEvalOrArguments(node, node.operand);
-            }
+    }
+    function isErrorNoInputFiles(error) {
+        return error.code === ts.Diagnostics.No_inputs_were_found_in_config_file_0_Specified_include_paths_were_1_and_exclude_paths_were_2.code;
+    }
+    function getErrorForNoInputFiles(_a, configFileName) {
+        var includeSpecs = _a.includeSpecs, excludeSpecs = _a.excludeSpecs;
+        return ts.createCompilerDiagnostic(ts.Diagnostics.No_inputs_were_found_in_config_file_0_Specified_include_paths_were_1_and_exclude_paths_were_2, configFileName || "tsconfig.json", JSON.stringify(includeSpecs || []), JSON.stringify(excludeSpecs || []));
+    }
+    function shouldReportNoInputFiles(result, canJsonReportNoInutFiles, resolutionStack) {
+        return result.fileNames.length === 0 && canJsonReportNoInutFiles && (!resolutionStack || resolutionStack.length === 0);
+    }
+    /*@internal*/
+    function canJsonReportNoInputFiles(raw) {
+        return !ts.hasProperty(raw, "files") && !ts.hasProperty(raw, "references");
+    }
+    ts.canJsonReportNoInputFiles = canJsonReportNoInputFiles;
+    /*@internal*/
+    function updateErrorForNoInputFiles(result, configFileName, configFileSpecs, configParseDiagnostics, canJsonReportNoInutFiles) {
+        var existingErrors = configParseDiagnostics.length;
+        if (shouldReportNoInputFiles(result, canJsonReportNoInutFiles)) {
+            configParseDiagnostics.push(getErrorForNoInputFiles(configFileSpecs, configFileName));
         }
-        function checkStrictModePrefixUnaryExpression(node) {
-            // Grammar checking
-            if (inStrictMode) {
-                if (node.operator === 45 /* PlusPlusToken */ || node.operator === 46 /* MinusMinusToken */) {
-                    checkStrictModeEvalOrArguments(node, node.operand);
-                }
-            }
+        else {
+            ts.filterMutate(configParseDiagnostics, function (error) { return !isErrorNoInputFiles(error); });
         }
-        function checkStrictModeWithStatement(node) {
-            // Grammar checking for withStatement
-            if (inStrictMode) {
-                errorOnFirstToken(node, ts.Diagnostics.with_statements_are_not_allowed_in_strict_mode);
-            }
+        return existingErrors !== configParseDiagnostics.length;
+    }
+    ts.updateErrorForNoInputFiles = updateErrorForNoInputFiles;
+    function isSuccessfulParsedTsconfig(value) {
+        return !!value.options;
+    }
+    /**
+     * This *just* extracts options/include/exclude/files out of a config file.
+     * It does *not* resolve the included files.
+     */
+    function parseConfig(json, sourceFile, host, basePath, configFileName, resolutionStack, errors, extendedConfigCache) {
+        var _a;
+        basePath = ts.normalizeSlashes(basePath);
+        var resolvedPath = ts.getNormalizedAbsolutePath(configFileName || "", basePath);
+        if (resolutionStack.indexOf(resolvedPath) >= 0) {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Circularity_detected_while_resolving_configuration_Colon_0, __spreadArrays(resolutionStack, [resolvedPath]).join(" -> ")));
+            return { raw: json || convertToObject(sourceFile, errors) };
         }
-        function checkStrictModeLabeledStatement(node) {
-            // Grammar checking for labeledStatement
-            if (inStrictMode && options.target >= 2 /* ES2015 */) {
-                if (ts.isDeclarationStatement(node.statement) || ts.isVariableStatement(node.statement)) {
-                    errorOnFirstToken(node.label, ts.Diagnostics.A_label_is_not_allowed_here);
+        var ownConfig = json ?
+            parseOwnConfigOfJson(json, host, basePath, configFileName, errors) :
+            parseOwnConfigOfJsonSourceFile(sourceFile, host, basePath, configFileName, errors);
+        if ((_a = ownConfig.options) === null || _a === void 0 ? void 0 : _a.paths) {
+            // If we end up needing to resolve relative paths from 'paths' relative to
+            // the config file location, we'll need to know where that config file was.
+            // Since 'paths' can be inherited from an extended config in another directory,
+            // we wouldn't know which directory to use unless we store it here.
+            ownConfig.options.pathsBasePath = basePath;
+        }
+        if (ownConfig.extendedConfigPath) {
+            // copy the resolution stack so it is never reused between branches in potential diamond-problem scenarios.
+            resolutionStack = resolutionStack.concat([resolvedPath]);
+            var extendedConfig = getExtendedConfig(sourceFile, ownConfig.extendedConfigPath, host, basePath, resolutionStack, errors, extendedConfigCache);
+            if (extendedConfig && isSuccessfulParsedTsconfig(extendedConfig)) {
+                var baseRaw_1 = extendedConfig.raw;
+                var raw_1 = ownConfig.raw;
+                var setPropertyInRawIfNotUndefined = function (propertyName) {
+                    var value = raw_1[propertyName] || baseRaw_1[propertyName];
+                    if (value) {
+                        raw_1[propertyName] = value;
+                    }
+                };
+                setPropertyInRawIfNotUndefined("include");
+                setPropertyInRawIfNotUndefined("exclude");
+                setPropertyInRawIfNotUndefined("files");
+                if (raw_1.compileOnSave === undefined) {
+                    raw_1.compileOnSave = baseRaw_1.compileOnSave;
                 }
+                ownConfig.options = ts.assign({}, extendedConfig.options, ownConfig.options);
+                ownConfig.watchOptions = ownConfig.watchOptions && extendedConfig.watchOptions ?
+                    ts.assign({}, extendedConfig.watchOptions, ownConfig.watchOptions) :
+                    ownConfig.watchOptions || extendedConfig.watchOptions;
+                // TODO extend type typeAcquisition
             }
         }
-        function errorOnFirstToken(node, message, arg0, arg1, arg2) {
-            var span = ts.getSpanOfTokenAtPosition(file, node.pos);
-            file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, message, arg0, arg1, arg2));
-        }
-        function errorOrSuggestionOnNode(isError, node, message) {
-            errorOrSuggestionOnRange(isError, node, node, message);
-        }
-        function errorOrSuggestionOnRange(isError, startNode, endNode, message) {
-            addErrorOrSuggestionDiagnostic(isError, { pos: ts.getTokenPosOfNode(startNode, file), end: endNode.end }, message);
+        return ownConfig;
+    }
+    function parseOwnConfigOfJson(json, host, basePath, configFileName, errors) {
+        if (ts.hasProperty(json, "excludes")) {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Unknown_option_excludes_Did_you_mean_exclude));
         }
-        function addErrorOrSuggestionDiagnostic(isError, range, message) {
-            var diag = ts.createFileDiagnostic(file, range.pos, range.end - range.pos, message);
-            if (isError) {
-                file.bindDiagnostics.push(diag);
+        var options = convertCompilerOptionsFromJsonWorker(json.compilerOptions, basePath, errors, configFileName);
+        // typingOptions has been deprecated and is only supported for backward compatibility purposes.
+        // It should be removed in future releases - use typeAcquisition instead.
+        var typeAcquisition = convertTypeAcquisitionFromJsonWorker(json.typeAcquisition || json.typingOptions, basePath, errors, configFileName);
+        var watchOptions = convertWatchOptionsFromJsonWorker(json.watchOptions, basePath, errors);
+        json.compileOnSave = convertCompileOnSaveOptionFromJson(json, basePath, errors);
+        var extendedConfigPath;
+        if (json.extends) {
+            if (!ts.isString(json.extends)) {
+                errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, "extends", "string"));
             }
             else {
-                file.bindSuggestionDiagnostics = ts.append(file.bindSuggestionDiagnostics, __assign(__assign({}, diag), { category: ts.DiagnosticCategory.Suggestion }));
+                var newBase = configFileName ? directoryOfCombinedPath(configFileName, basePath) : basePath;
+                extendedConfigPath = getExtendsConfigPath(json.extends, host, newBase, errors, ts.createCompilerDiagnostic);
             }
         }
-        function bind(node) {
-            if (!node) {
-                return;
-            }
-            node.parent = parent;
-            var saveInStrictMode = inStrictMode;
-            // Even though in the AST the jsdoc @typedef node belongs to the current node,
-            // its symbol might be in the same scope with the current node's symbol. Consider:
-            //
-            //     /** @typedef {string | number} MyType */
-            //     function foo();
-            //
-            // Here the current node is "foo", which is a container, but the scope of "MyType" should
-            // not be inside "foo". Therefore we always bind @typedef before bind the parent node,
-            // and skip binding this tag later when binding all the other jsdoc tags.
-            // First we bind declaration nodes to a symbol if possible. We'll both create a symbol
-            // and then potentially add the symbol to an appropriate symbol table. Possible
-            // destination symbol tables are:
-            //
-            //  1) The 'exports' table of the current container's symbol.
-            //  2) The 'members' table of the current container's symbol.
-            //  3) The 'locals' table of the current container.
-            //
-            // However, not all symbols will end up in any of these tables. 'Anonymous' symbols
-            // (like TypeLiterals for example) will not be put in any table.
-            bindWorker(node);
-            // Then we recurse into the children of the node to bind them as well. For certain
-            // symbols we do specialized work when we recurse. For example, we'll keep track of
-            // the current 'container' node when it changes. This helps us know which symbol table
-            // a local should go into for example. Since terminal nodes are known not to have
-            // children, as an optimization we don't process those.
-            if (node.kind > 152 /* LastToken */) {
-                var saveParent = parent;
-                parent = node;
-                var containerFlags = getContainerFlags(node);
-                if (containerFlags === 0 /* None */) {
-                    bindChildren(node);
+        return { raw: json, options: options, watchOptions: watchOptions, typeAcquisition: typeAcquisition, extendedConfigPath: extendedConfigPath };
+    }
+    function parseOwnConfigOfJsonSourceFile(sourceFile, host, basePath, configFileName, errors) {
+        var options = getDefaultCompilerOptions(configFileName);
+        var typeAcquisition, typingOptionstypeAcquisition;
+        var watchOptions;
+        var extendedConfigPath;
+        var optionsIterator = {
+            onSetValidOptionKeyValueInParent: function (parentOption, option, value) {
+                var currentOption;
+                switch (parentOption) {
+                    case "compilerOptions":
+                        currentOption = options;
+                        break;
+                    case "watchOptions":
+                        currentOption = (watchOptions || (watchOptions = {}));
+                        break;
+                    case "typeAcquisition":
+                        currentOption = (typeAcquisition || (typeAcquisition = getDefaultTypeAcquisition(configFileName)));
+                        break;
+                    case "typingOptions":
+                        currentOption = (typingOptionstypeAcquisition || (typingOptionstypeAcquisition = getDefaultTypeAcquisition(configFileName)));
+                        break;
+                    default:
+                        ts.Debug.fail("Unknown option");
                 }
-                else {
-                    bindContainer(node, containerFlags);
+                currentOption[option.name] = normalizeOptionValue(option, basePath, value);
+            },
+            onSetValidOptionKeyValueInRoot: function (key, _keyNode, value, valueNode) {
+                switch (key) {
+                    case "extends":
+                        var newBase = configFileName ? directoryOfCombinedPath(configFileName, basePath) : basePath;
+                        extendedConfigPath = getExtendsConfigPath(value, host, newBase, errors, function (message, arg0) {
+                            return ts.createDiagnosticForNodeInSourceFile(sourceFile, valueNode, message, arg0);
+                        });
+                        return;
+                }
+            },
+            onSetUnknownOptionKeyValueInRoot: function (key, keyNode, _value, _valueNode) {
+                if (key === "excludes") {
+                    errors.push(ts.createDiagnosticForNodeInSourceFile(sourceFile, keyNode, ts.Diagnostics.Unknown_option_excludes_Did_you_mean_exclude));
                 }
-                parent = saveParent;
             }
-            else if (!skipTransformFlagAggregation && (node.transformFlags & 536870912 /* HasComputedFlags */) === 0) {
-                subtreeTransformFlags |= computeTransformFlagsForNode(node, 0);
-                var saveParent = parent;
-                if (node.kind === 1 /* EndOfFileToken */)
-                    parent = node;
-                bindJSDoc(node);
-                parent = saveParent;
+        };
+        var json = convertToObjectWorker(sourceFile, errors, /*returnValue*/ true, getTsconfigRootOptionsMap(), optionsIterator);
+        if (!typeAcquisition) {
+            if (typingOptionstypeAcquisition) {
+                typeAcquisition = (typingOptionstypeAcquisition.enableAutoDiscovery !== undefined) ?
+                    {
+                        enable: typingOptionstypeAcquisition.enableAutoDiscovery,
+                        include: typingOptionstypeAcquisition.include,
+                        exclude: typingOptionstypeAcquisition.exclude
+                    } :
+                    typingOptionstypeAcquisition;
             }
-            inStrictMode = saveInStrictMode;
-        }
-        function bindJSDoc(node) {
-            if (ts.hasJSDocNodes(node)) {
-                if (ts.isInJSFile(node)) {
-                    for (var _i = 0, _a = node.jsDoc; _i < _a.length; _i++) {
-                        var j = _a[_i];
-                        bind(j);
-                    }
-                }
-                else {
-                    for (var _b = 0, _c = node.jsDoc; _b < _c.length; _b++) {
-                        var j = _c[_b];
-                        setParentPointers(node, j);
-                    }
-                }
+            else {
+                typeAcquisition = getDefaultTypeAcquisition(configFileName);
             }
         }
-        function updateStrictModeStatementList(statements) {
-            if (!inStrictMode) {
-                for (var _i = 0, statements_2 = statements; _i < statements_2.length; _i++) {
-                    var statement = statements_2[_i];
-                    if (!ts.isPrologueDirective(statement)) {
-                        return;
-                    }
-                    if (isUseStrictPrologueDirective(statement)) {
-                        inStrictMode = true;
-                        return;
-                    }
+        return { raw: json, options: options, watchOptions: watchOptions, typeAcquisition: typeAcquisition, extendedConfigPath: extendedConfigPath };
+    }
+    function getExtendsConfigPath(extendedConfig, host, basePath, errors, createDiagnostic) {
+        extendedConfig = ts.normalizeSlashes(extendedConfig);
+        if (ts.isRootedDiskPath(extendedConfig) || ts.startsWith(extendedConfig, "./") || ts.startsWith(extendedConfig, "../")) {
+            var extendedConfigPath = ts.getNormalizedAbsolutePath(extendedConfig, basePath);
+            if (!host.fileExists(extendedConfigPath) && !ts.endsWith(extendedConfigPath, ".json" /* Json */)) {
+                extendedConfigPath = extendedConfigPath + ".json";
+                if (!host.fileExists(extendedConfigPath)) {
+                    errors.push(createDiagnostic(ts.Diagnostics.File_0_not_found, extendedConfig));
+                    return undefined;
                 }
             }
+            return extendedConfigPath;
         }
-        /// Should be called only on prologue directives (isPrologueDirective(node) should be true)
-        function isUseStrictPrologueDirective(node) {
-            var nodeText = ts.getSourceTextOfNodeFromSourceFile(file, node.expression);
-            // Note: the node text must be exactly "use strict" or 'use strict'.  It is not ok for the
-            // string to contain unicode escapes (as per ES5).
-            return nodeText === '"use strict"' || nodeText === "'use strict'";
+        // If the path isn't a rooted or relative path, resolve like a module
+        var resolved = ts.nodeModuleNameResolver(extendedConfig, ts.combinePaths(basePath, "tsconfig.json"), { moduleResolution: ts.ModuleResolutionKind.NodeJs }, host, /*cache*/ undefined, /*projectRefs*/ undefined, /*lookupConfig*/ true);
+        if (resolved.resolvedModule) {
+            return resolved.resolvedModule.resolvedFileName;
         }
-        function bindWorker(node) {
-            switch (node.kind) {
-                /* Strict mode checks */
-                case 75 /* Identifier */:
-                    // for typedef type names with namespaces, bind the new jsdoc type symbol here
-                    // because it requires all containing namespaces to be in effect, namely the
-                    // current "blockScopeContainer" needs to be set to its immediate namespace parent.
-                    if (node.isInJSDocNamespace) {
-                        var parentNode = node.parent;
-                        while (parentNode && !ts.isJSDocTypeAlias(parentNode)) {
-                            parentNode = parentNode.parent;
+        errors.push(createDiagnostic(ts.Diagnostics.File_0_not_found, extendedConfig));
+        return undefined;
+    }
+    function getExtendedConfig(sourceFile, extendedConfigPath, host, basePath, resolutionStack, errors, extendedConfigCache) {
+        var _a;
+        var path = host.useCaseSensitiveFileNames ? extendedConfigPath : ts.toFileNameLowerCase(extendedConfigPath);
+        var value;
+        var extendedResult;
+        var extendedConfig;
+        if (extendedConfigCache && (value = extendedConfigCache.get(path))) {
+            (extendedResult = value.extendedResult, extendedConfig = value.extendedConfig);
+        }
+        else {
+            extendedResult = readJsonConfigFile(extendedConfigPath, function (path) { return host.readFile(path); });
+            if (!extendedResult.parseDiagnostics.length) {
+                var extendedDirname = ts.getDirectoryPath(extendedConfigPath);
+                extendedConfig = parseConfig(/*json*/ undefined, extendedResult, host, extendedDirname, ts.getBaseFileName(extendedConfigPath), resolutionStack, errors, extendedConfigCache);
+                if (isSuccessfulParsedTsconfig(extendedConfig)) {
+                    // Update the paths to reflect base path
+                    var relativeDifference_1 = ts.convertToRelativePath(extendedDirname, basePath, ts.identity);
+                    var updatePath_1 = function (path) { return ts.isRootedDiskPath(path) ? path : ts.combinePaths(relativeDifference_1, path); };
+                    var mapPropertiesInRawIfNotUndefined = function (propertyName) {
+                        if (raw_2[propertyName]) {
+                            raw_2[propertyName] = ts.map(raw_2[propertyName], updatePath_1);
                         }
-                        bindBlockScopedDeclaration(parentNode, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
-                        break;
-                    }
-                // falls through
-                case 104 /* ThisKeyword */:
-                    if (currentFlow && (ts.isExpression(node) || parent.kind === 282 /* ShorthandPropertyAssignment */)) {
-                        node.flowNode = currentFlow;
-                    }
-                    return checkStrictModeIdentifier(node);
-                case 76 /* PrivateIdentifier */:
-                    return checkPrivateIdentifier(node);
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    var expr = node;
-                    if (currentFlow && isNarrowableReference(expr)) {
-                        expr.flowNode = currentFlow;
-                    }
-                    if (ts.isSpecialPropertyDeclaration(expr)) {
-                        bindSpecialPropertyDeclaration(expr);
-                    }
-                    if (ts.isInJSFile(expr) &&
-                        file.commonJsModuleIndicator &&
-                        ts.isModuleExportsAccessExpression(expr) &&
-                        !lookupSymbolForNameWorker(blockScopeContainer, "module")) {
-                        declareSymbol(file.locals, /*parent*/ undefined, expr.expression, 1 /* FunctionScopedVariable */ | 134217728 /* ModuleExports */, 111550 /* FunctionScopedVariableExcludes */);
-                    }
-                    break;
-                case 209 /* BinaryExpression */:
-                    var specialKind = ts.getAssignmentDeclarationKind(node);
-                    switch (specialKind) {
-                        case 1 /* ExportsProperty */:
-                            bindExportsPropertyAssignment(node);
-                            break;
-                        case 2 /* ModuleExports */:
-                            bindModuleExportsAssignment(node);
-                            break;
-                        case 3 /* PrototypeProperty */:
-                            bindPrototypePropertyAssignment(node.left, node);
-                            break;
-                        case 6 /* Prototype */:
-                            bindPrototypeAssignment(node);
-                            break;
-                        case 4 /* ThisProperty */:
-                            bindThisPropertyAssignment(node);
-                            break;
-                        case 5 /* Property */:
-                            bindSpecialPropertyAssignment(node);
-                            break;
-                        case 0 /* None */:
-                            // Nothing to do
-                            break;
-                        default:
-                            ts.Debug.fail("Unknown binary expression special property assignment kind");
-                    }
-                    return checkStrictModeBinaryExpression(node);
-                case 280 /* CatchClause */:
-                    return checkStrictModeCatchClause(node);
-                case 203 /* DeleteExpression */:
-                    return checkStrictModeDeleteExpression(node);
-                case 8 /* NumericLiteral */:
-                    return checkStrictModeNumericLiteral(node);
-                case 208 /* PostfixUnaryExpression */:
-                    return checkStrictModePostfixUnaryExpression(node);
-                case 207 /* PrefixUnaryExpression */:
-                    return checkStrictModePrefixUnaryExpression(node);
-                case 236 /* WithStatement */:
-                    return checkStrictModeWithStatement(node);
-                case 238 /* LabeledStatement */:
-                    return checkStrictModeLabeledStatement(node);
-                case 183 /* ThisType */:
-                    seenThisKeyword = true;
-                    return;
-                case 168 /* TypePredicate */:
-                    break; // Binding the children will handle everything
-                case 155 /* TypeParameter */:
-                    return bindTypeParameter(node);
-                case 156 /* Parameter */:
-                    return bindParameter(node);
-                case 242 /* VariableDeclaration */:
-                    return bindVariableDeclarationOrBindingElement(node);
-                case 191 /* BindingElement */:
-                    node.flowNode = currentFlow;
-                    return bindVariableDeclarationOrBindingElement(node);
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                    return bindPropertyWorker(node);
-                case 281 /* PropertyAssignment */:
-                case 282 /* ShorthandPropertyAssignment */:
-                    return bindPropertyOrMethodOrAccessor(node, 4 /* Property */, 0 /* PropertyExcludes */);
-                case 284 /* EnumMember */:
-                    return bindPropertyOrMethodOrAccessor(node, 8 /* EnumMember */, 900095 /* EnumMemberExcludes */);
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 167 /* IndexSignature */:
-                    return declareSymbolAndAddToSymbolTable(node, 131072 /* Signature */, 0 /* None */);
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    // If this is an ObjectLiteralExpression method, then it sits in the same space
-                    // as other properties in the object literal.  So we use SymbolFlags.PropertyExcludes
-                    // so that it will conflict with any other object literal members with the same
-                    // name.
-                    return bindPropertyOrMethodOrAccessor(node, 8192 /* Method */ | (node.questionToken ? 16777216 /* Optional */ : 0 /* None */), ts.isObjectLiteralMethod(node) ? 0 /* PropertyExcludes */ : 103359 /* MethodExcludes */);
-                case 244 /* FunctionDeclaration */:
-                    return bindFunctionDeclaration(node);
-                case 162 /* Constructor */:
-                    return declareSymbolAndAddToSymbolTable(node, 16384 /* Constructor */, /*symbolExcludes:*/ 0 /* None */);
-                case 163 /* GetAccessor */:
-                    return bindPropertyOrMethodOrAccessor(node, 32768 /* GetAccessor */, 46015 /* GetAccessorExcludes */);
-                case 164 /* SetAccessor */:
-                    return bindPropertyOrMethodOrAccessor(node, 65536 /* SetAccessor */, 78783 /* SetAccessorExcludes */);
-                case 170 /* FunctionType */:
-                case 300 /* JSDocFunctionType */:
-                case 305 /* JSDocSignature */:
-                case 171 /* ConstructorType */:
-                    return bindFunctionOrConstructorType(node);
-                case 173 /* TypeLiteral */:
-                case 304 /* JSDocTypeLiteral */:
-                case 186 /* MappedType */:
-                    return bindAnonymousTypeWorker(node);
-                case 310 /* JSDocClassTag */:
-                    return bindJSDocClassTag(node);
-                case 193 /* ObjectLiteralExpression */:
-                    return bindObjectLiteralExpression(node);
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                    return bindFunctionExpression(node);
-                case 196 /* CallExpression */:
-                    var assignmentKind = ts.getAssignmentDeclarationKind(node);
-                    switch (assignmentKind) {
-                        case 7 /* ObjectDefinePropertyValue */:
-                            return bindObjectDefinePropertyAssignment(node);
-                        case 8 /* ObjectDefinePropertyExports */:
-                            return bindObjectDefinePropertyExport(node);
-                        case 9 /* ObjectDefinePrototypeProperty */:
-                            return bindObjectDefinePrototypeProperty(node);
-                        case 0 /* None */:
-                            break; // Nothing to do
-                        default:
-                            return ts.Debug.fail("Unknown call expression assignment declaration kind");
-                    }
-                    if (ts.isInJSFile(node)) {
-                        bindCallExpression(node);
-                    }
-                    break;
-                // Members of classes, interfaces, and modules
-                case 214 /* ClassExpression */:
-                case 245 /* ClassDeclaration */:
-                    // All classes are automatically in strict mode in ES6.
-                    inStrictMode = true;
-                    return bindClassLikeDeclaration(node);
-                case 246 /* InterfaceDeclaration */:
-                    return bindBlockScopedDeclaration(node, 64 /* Interface */, 788872 /* InterfaceExcludes */);
-                case 247 /* TypeAliasDeclaration */:
-                    return bindBlockScopedDeclaration(node, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
-                case 248 /* EnumDeclaration */:
-                    return bindEnumDeclaration(node);
-                case 249 /* ModuleDeclaration */:
-                    return bindModuleDeclaration(node);
-                // Jsx-attributes
-                case 274 /* JsxAttributes */:
-                    return bindJsxAttributes(node);
-                case 273 /* JsxAttribute */:
-                    return bindJsxAttribute(node, 4 /* Property */, 0 /* PropertyExcludes */);
-                // Imports and exports
-                case 253 /* ImportEqualsDeclaration */:
-                case 256 /* NamespaceImport */:
-                case 258 /* ImportSpecifier */:
-                case 263 /* ExportSpecifier */:
-                    return declareSymbolAndAddToSymbolTable(node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
-                case 252 /* NamespaceExportDeclaration */:
-                    return bindNamespaceExportDeclaration(node);
-                case 255 /* ImportClause */:
-                    return bindImportClause(node);
-                case 260 /* ExportDeclaration */:
-                    return bindExportDeclaration(node);
-                case 259 /* ExportAssignment */:
-                    return bindExportAssignment(node);
-                case 290 /* SourceFile */:
-                    updateStrictModeStatementList(node.statements);
-                    return bindSourceFileIfExternalModule();
-                case 223 /* Block */:
-                    if (!ts.isFunctionLike(node.parent)) {
-                        return;
-                    }
-                // falls through
-                case 250 /* ModuleBlock */:
-                    return updateStrictModeStatementList(node.statements);
-                case 317 /* JSDocParameterTag */:
-                    if (node.parent.kind === 305 /* JSDocSignature */) {
-                        return bindParameter(node);
-                    }
-                    if (node.parent.kind !== 304 /* JSDocTypeLiteral */) {
-                        break;
-                    }
-                // falls through
-                case 323 /* JSDocPropertyTag */:
-                    var propTag = node;
-                    var flags = propTag.isBracketed || propTag.typeExpression && propTag.typeExpression.type.kind === 299 /* JSDocOptionalType */ ?
-                        4 /* Property */ | 16777216 /* Optional */ :
-                        4 /* Property */;
-                    return declareSymbolAndAddToSymbolTable(propTag, flags, 0 /* PropertyExcludes */);
-                case 322 /* JSDocTypedefTag */:
-                case 315 /* JSDocCallbackTag */:
-                case 316 /* JSDocEnumTag */:
-                    return (delayedTypeAliases || (delayedTypeAliases = [])).push(node);
+                    };
+                    var raw_2 = extendedConfig.raw;
+                    mapPropertiesInRawIfNotUndefined("include");
+                    mapPropertiesInRawIfNotUndefined("exclude");
+                    mapPropertiesInRawIfNotUndefined("files");
+                }
+            }
+            if (extendedConfigCache) {
+                extendedConfigCache.set(path, { extendedResult: extendedResult, extendedConfig: extendedConfig });
             }
         }
-        function bindPropertyWorker(node) {
-            return bindPropertyOrMethodOrAccessor(node, 4 /* Property */ | (node.questionToken ? 16777216 /* Optional */ : 0 /* None */), 0 /* PropertyExcludes */);
+        if (sourceFile) {
+            sourceFile.extendedSourceFiles = [extendedResult.fileName];
+            if (extendedResult.extendedSourceFiles) {
+                (_a = sourceFile.extendedSourceFiles).push.apply(_a, extendedResult.extendedSourceFiles);
+            }
         }
-        function bindAnonymousTypeWorker(node) {
-            return bindAnonymousDeclaration(node, 2048 /* TypeLiteral */, "__type" /* Type */);
+        if (extendedResult.parseDiagnostics.length) {
+            errors.push.apply(errors, extendedResult.parseDiagnostics);
+            return undefined;
         }
-        function bindSourceFileIfExternalModule() {
-            setExportContextFlag(file);
-            if (ts.isExternalModule(file)) {
-                bindSourceFileAsExternalModule();
-            }
-            else if (ts.isJsonSourceFile(file)) {
-                bindSourceFileAsExternalModule();
-                // Create symbol equivalent for the module.exports = {}
-                var originalSymbol = file.symbol;
-                declareSymbol(file.symbol.exports, file.symbol, file, 4 /* Property */, 67108863 /* All */);
-                file.symbol = originalSymbol;
-            }
+        return extendedConfig;
+    }
+    function convertCompileOnSaveOptionFromJson(jsonOption, basePath, errors) {
+        if (!ts.hasProperty(jsonOption, ts.compileOnSaveCommandLineOption.name)) {
+            return false;
         }
-        function bindSourceFileAsExternalModule() {
-            bindAnonymousDeclaration(file, 512 /* ValueModule */, "\"" + ts.removeFileExtension(file.fileName) + "\"");
+        var result = convertJsonOption(ts.compileOnSaveCommandLineOption, jsonOption.compileOnSave, basePath, errors);
+        return typeof result === "boolean" && result;
+    }
+    function convertCompilerOptionsFromJson(jsonOptions, basePath, configFileName) {
+        var errors = [];
+        var options = convertCompilerOptionsFromJsonWorker(jsonOptions, basePath, errors, configFileName);
+        return { options: options, errors: errors };
+    }
+    ts.convertCompilerOptionsFromJson = convertCompilerOptionsFromJson;
+    function convertTypeAcquisitionFromJson(jsonOptions, basePath, configFileName) {
+        var errors = [];
+        var options = convertTypeAcquisitionFromJsonWorker(jsonOptions, basePath, errors, configFileName);
+        return { options: options, errors: errors };
+    }
+    ts.convertTypeAcquisitionFromJson = convertTypeAcquisitionFromJson;
+    function getDefaultCompilerOptions(configFileName) {
+        var options = configFileName && ts.getBaseFileName(configFileName) === "jsconfig.json"
+            ? { allowJs: true, maxNodeModuleJsDepth: 2, allowSyntheticDefaultImports: true, skipLibCheck: true, noEmit: true }
+            : {};
+        return options;
+    }
+    function convertCompilerOptionsFromJsonWorker(jsonOptions, basePath, errors, configFileName) {
+        var options = getDefaultCompilerOptions(configFileName);
+        convertOptionsFromJson(getCommandLineCompilerOptionsMap(), jsonOptions, basePath, options, ts.compilerOptionsDidYouMeanDiagnostics, errors);
+        if (configFileName) {
+            options.configFilePath = ts.normalizeSlashes(configFileName);
         }
-        function bindExportAssignment(node) {
-            if (!container.symbol || !container.symbol.exports) {
-                // Export assignment in some sort of block construct
-                bindAnonymousDeclaration(node, 2097152 /* Alias */, getDeclarationName(node));
-            }
-            else {
-                var flags = ts.exportAssignmentIsAlias(node)
-                    // An export default clause with an EntityNameExpression or a class expression exports all meanings of that identifier or expression;
-                    ? 2097152 /* Alias */
-                    // An export default clause with any other expression exports a value
-                    : 4 /* Property */;
-                // If there is an `export default x;` alias declaration, can't `export default` anything else.
-                // (In contrast, you can still have `export default function f() {}` and `export default interface I {}`.)
-                var symbol = declareSymbol(container.symbol.exports, container.symbol, node, flags, 67108863 /* All */);
-                if (node.isExportEquals) {
-                    // Will be an error later, since the module already has other exports. Just make sure this has a valueDeclaration set.
-                    ts.setValueDeclaration(symbol, node);
-                }
-            }
+        return options;
+    }
+    function getDefaultTypeAcquisition(configFileName) {
+        return { enable: !!configFileName && ts.getBaseFileName(configFileName) === "jsconfig.json", include: [], exclude: [] };
+    }
+    function convertTypeAcquisitionFromJsonWorker(jsonOptions, basePath, errors, configFileName) {
+        var options = getDefaultTypeAcquisition(configFileName);
+        var typeAcquisition = convertEnableAutoDiscoveryToEnable(jsonOptions);
+        convertOptionsFromJson(getCommandLineTypeAcquisitionMap(), typeAcquisition, basePath, options, typeAcquisitionDidYouMeanDiagnostics, errors);
+        return options;
+    }
+    function convertWatchOptionsFromJsonWorker(jsonOptions, basePath, errors) {
+        return convertOptionsFromJson(getCommandLineWatchOptionsMap(), jsonOptions, basePath, /*defaultOptions*/ undefined, watchOptionsDidYouMeanDiagnostics, errors);
+    }
+    function convertOptionsFromJson(optionsNameMap, jsonOptions, basePath, defaultOptions, diagnostics, errors) {
+        if (!jsonOptions) {
+            return;
         }
-        function bindNamespaceExportDeclaration(node) {
-            if (node.modifiers && node.modifiers.length) {
-                file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Modifiers_cannot_appear_here));
-            }
-            var diag = !ts.isSourceFile(node.parent) ? ts.Diagnostics.Global_module_exports_may_only_appear_at_top_level
-                : !ts.isExternalModule(node.parent) ? ts.Diagnostics.Global_module_exports_may_only_appear_in_module_files
-                    : !node.parent.isDeclarationFile ? ts.Diagnostics.Global_module_exports_may_only_appear_in_declaration_files
-                        : undefined;
-            if (diag) {
-                file.bindDiagnostics.push(createDiagnosticForNode(node, diag));
+        for (var id in jsonOptions) {
+            var opt = optionsNameMap.get(id);
+            if (opt) {
+                (defaultOptions || (defaultOptions = {}))[opt.name] = convertJsonOption(opt, jsonOptions[id], basePath, errors);
             }
             else {
-                file.symbol.globalExports = file.symbol.globalExports || ts.createSymbolTable();
-                declareSymbol(file.symbol.globalExports, file.symbol, node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
+                errors.push(createUnknownOptionError(id, diagnostics, ts.createCompilerDiagnostic));
             }
         }
-        function bindExportDeclaration(node) {
-            if (!container.symbol || !container.symbol.exports) {
-                // Export * in some sort of block construct
-                bindAnonymousDeclaration(node, 8388608 /* ExportStar */, getDeclarationName(node));
-            }
-            else if (!node.exportClause) {
-                // All export * declarations are collected in an __export symbol
-                declareSymbol(container.symbol.exports, container.symbol, node, 8388608 /* ExportStar */, 0 /* None */);
+        return defaultOptions;
+    }
+    function convertJsonOption(opt, value, basePath, errors) {
+        if (isCompilerOptionsValue(opt, value)) {
+            var optType = opt.type;
+            if (optType === "list" && ts.isArray(value)) {
+                return convertJsonOptionOfListType(opt, value, basePath, errors);
             }
-            else if (ts.isNamespaceExport(node.exportClause)) {
-                // declareSymbol walks up parents to find name text, parent _must_ be set
-                // but won't be set by the normal binder walk until `bindChildren` later on.
-                node.exportClause.parent = node;
-                declareSymbol(container.symbol.exports, container.symbol, node.exportClause, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
+            else if (!ts.isString(optType)) {
+                return convertJsonOptionOfCustomType(opt, value, errors);
             }
+            return normalizeNonListOptionValue(opt, basePath, value);
         }
-        function bindImportClause(node) {
-            if (node.name) {
-                declareSymbolAndAddToSymbolTable(node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
-            }
+        else {
+            errors.push(ts.createCompilerDiagnostic(ts.Diagnostics.Compiler_option_0_requires_a_value_of_type_1, opt.name, getCompilerOptionValueTypeString(opt)));
         }
-        function setCommonJsModuleIndicator(node) {
-            if (file.externalModuleIndicator) {
-                return false;
-            }
-            if (!file.commonJsModuleIndicator) {
-                file.commonJsModuleIndicator = node;
-                bindSourceFileAsExternalModule();
+    }
+    function normalizeOptionValue(option, basePath, value) {
+        if (isNullOrUndefined(value))
+            return undefined;
+        if (option.type === "list") {
+            var listOption_1 = option;
+            if (listOption_1.element.isFilePath || !ts.isString(listOption_1.element.type)) {
+                return ts.filter(ts.map(value, function (v) { return normalizeOptionValue(listOption_1.element, basePath, v); }), function (v) { return !!v; });
             }
-            return true;
+            return value;
         }
-        function bindObjectDefinePropertyExport(node) {
-            if (!setCommonJsModuleIndicator(node)) {
-                return;
-            }
-            var symbol = forEachIdentifierInEntityName(node.arguments[0], /*parent*/ undefined, function (id, symbol) {
-                if (symbol) {
-                    addDeclarationToSymbol(symbol, id, 1536 /* Module */ | 67108864 /* Assignment */);
-                }
-                return symbol;
-            });
-            if (symbol) {
-                var flags = 4 /* Property */ | 1048576 /* ExportValue */;
-                declareSymbol(symbol.exports, symbol, node, flags, 0 /* None */);
+        else if (!ts.isString(option.type)) {
+            return option.type.get(ts.isString(value) ? value.toLowerCase() : value);
+        }
+        return normalizeNonListOptionValue(option, basePath, value);
+    }
+    function normalizeNonListOptionValue(option, basePath, value) {
+        if (option.isFilePath) {
+            value = ts.getNormalizedAbsolutePath(value, basePath);
+            if (value === "") {
+                value = ".";
             }
         }
-        function bindExportsPropertyAssignment(node) {
-            // When we create a property via 'exports.foo = bar', the 'exports.foo' property access
-            // expression is the declaration
-            if (!setCommonJsModuleIndicator(node)) {
-                return;
+        return value;
+    }
+    function convertJsonOptionOfCustomType(opt, value, errors) {
+        if (isNullOrUndefined(value))
+            return undefined;
+        var key = value.toLowerCase();
+        var val = opt.type.get(key);
+        if (val !== undefined) {
+            return val;
+        }
+        else {
+            errors.push(createCompilerDiagnosticForInvalidCustomType(opt));
+        }
+    }
+    function convertJsonOptionOfListType(option, values, basePath, errors) {
+        return ts.filter(ts.map(values, function (v) { return convertJsonOption(option.element, v, basePath, errors); }), function (v) { return !!v; });
+    }
+    function trimString(s) {
+        return typeof s.trim === "function" ? s.trim() : s.replace(/^[\s]+|[\s]+$/g, "");
+    }
+    /**
+     * Tests for a path that ends in a recursive directory wildcard.
+     * Matches **, \**, **\, and \**\, but not a**b.
+     *
+     * NOTE: used \ in place of / above to avoid issues with multiline comments.
+     *
+     * Breakdown:
+     *  (^|\/)      # matches either the beginning of the string or a directory separator.
+     *  \*\*        # matches the recursive directory wildcard "**".
+     *  \/?$        # matches an optional trailing directory separator at the end of the string.
+     */
+    var invalidTrailingRecursionPattern = /(^|\/)\*\*\/?$/;
+    /**
+     * Tests for a path where .. appears after a recursive directory wildcard.
+     * Matches **\..\*, **\a\..\*, and **\.., but not ..\**\*
+     *
+     * NOTE: used \ in place of / above to avoid issues with multiline comments.
+     *
+     * Breakdown:
+     *  (^|\/)      # matches either the beginning of the string or a directory separator.
+     *  \*\*\/      # matches a recursive directory wildcard "**" followed by a directory separator.
+     *  (.*\/)?     # optionally matches any number of characters followed by a directory separator.
+     *  \.\.        # matches a parent directory path component ".."
+     *  ($|\/)      # matches either the end of the string or a directory separator.
+     */
+    var invalidDotDotAfterRecursiveWildcardPattern = /(^|\/)\*\*\/(.*\/)?\.\.($|\/)/;
+    /**
+     * Tests for a path containing a wildcard character in a directory component of the path.
+     * Matches \*\, \?\, and \a*b\, but not \a\ or \a\*.
+     *
+     * NOTE: used \ in place of / above to avoid issues with multiline comments.
+     *
+     * Breakdown:
+     *  \/          # matches a directory separator.
+     *  [^/]*?      # matches any number of characters excluding directory separators (non-greedy).
+     *  [*?]        # matches either a wildcard character (* or ?)
+     *  [^/]*       # matches any number of characters excluding directory separators (greedy).
+     *  \/          # matches a directory separator.
+     */
+    var watchRecursivePattern = /\/[^/]*?[*?][^/]*\//;
+    /**
+     * Matches the portion of a wildcard path that does not contain wildcards.
+     * Matches \a of \a\*, or \a\b\c of \a\b\c\?\d.
+     *
+     * NOTE: used \ in place of / above to avoid issues with multiline comments.
+     *
+     * Breakdown:
+     *  ^                   # matches the beginning of the string
+     *  [^*?]*              # matches any number of non-wildcard characters
+     *  (?=\/[^/]*[*?])     # lookahead that matches a directory separator followed by
+     *                      # a path component that contains at least one wildcard character (* or ?).
+     */
+    var wildcardDirectoryPattern = /^[^*?]*(?=\/[^/]*[*?])/;
+    /**
+     * Expands an array of file specifications.
+     *
+     * @param filesSpecs The literal file names to include.
+     * @param includeSpecs The wildcard file specifications to include.
+     * @param excludeSpecs The wildcard file specifications to exclude.
+     * @param basePath The base path for any relative file specifications.
+     * @param options Compiler options.
+     * @param host The host used to resolve files and directories.
+     * @param errors An array for diagnostic reporting.
+     */
+    function matchFileNames(filesSpecs, includeSpecs, excludeSpecs, basePath, options, host, errors, extraFileExtensions, jsonSourceFile) {
+        basePath = ts.normalizePath(basePath);
+        var validatedIncludeSpecs, validatedExcludeSpecs;
+        // The exclude spec list is converted into a regular expression, which allows us to quickly
+        // test whether a file or directory should be excluded before recursively traversing the
+        // file system.
+        if (includeSpecs) {
+            validatedIncludeSpecs = validateSpecs(includeSpecs, errors, /*allowTrailingRecursion*/ false, jsonSourceFile, "include");
+        }
+        if (excludeSpecs) {
+            validatedExcludeSpecs = validateSpecs(excludeSpecs, errors, /*allowTrailingRecursion*/ true, jsonSourceFile, "exclude");
+        }
+        // Wildcard directories (provided as part of a wildcard path) are stored in a
+        // file map that marks whether it was a regular wildcard match (with a `*` or `?` token),
+        // or a recursive directory. This information is used by filesystem watchers to monitor for
+        // new entries in these paths.
+        var wildcardDirectories = getWildcardDirectories(validatedIncludeSpecs, validatedExcludeSpecs, basePath, host.useCaseSensitiveFileNames);
+        var spec = {
+            filesSpecs: filesSpecs,
+            includeSpecs: includeSpecs,
+            excludeSpecs: excludeSpecs,
+            validatedFilesSpec: ts.filter(filesSpecs, ts.isString),
+            validatedIncludeSpecs: validatedIncludeSpecs,
+            validatedExcludeSpecs: validatedExcludeSpecs,
+            wildcardDirectories: wildcardDirectories
+        };
+        return getFileNamesFromConfigSpecs(spec, basePath, options, host, extraFileExtensions);
+    }
+    /**
+     * Gets the file names from the provided config file specs that contain, files, include, exclude and
+     * other properties needed to resolve the file names
+     * @param spec The config file specs extracted with file names to include, wildcards to include/exclude and other details
+     * @param basePath The base path for any relative file specifications.
+     * @param options Compiler options.
+     * @param host The host used to resolve files and directories.
+     * @param extraFileExtensions optionaly file extra file extension information from host
+     */
+    /* @internal */
+    function getFileNamesFromConfigSpecs(spec, basePath, options, host, extraFileExtensions) {
+        if (extraFileExtensions === void 0) { extraFileExtensions = ts.emptyArray; }
+        basePath = ts.normalizePath(basePath);
+        var keyMapper = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames);
+        // Literal file names (provided via the "files" array in tsconfig.json) are stored in a
+        // file map with a possibly case insensitive key. We use this map later when when including
+        // wildcard paths.
+        var literalFileMap = new ts.Map();
+        // Wildcard paths (provided via the "includes" array in tsconfig.json) are stored in a
+        // file map with a possibly case insensitive key. We use this map to store paths matched
+        // via wildcard, and to handle extension priority.
+        var wildcardFileMap = new ts.Map();
+        // Wildcard paths of json files (provided via the "includes" array in tsconfig.json) are stored in a
+        // file map with a possibly case insensitive key. We use this map to store paths matched
+        // via wildcard of *.json kind
+        var wildCardJsonFileMap = new ts.Map();
+        var validatedFilesSpec = spec.validatedFilesSpec, validatedIncludeSpecs = spec.validatedIncludeSpecs, validatedExcludeSpecs = spec.validatedExcludeSpecs, wildcardDirectories = spec.wildcardDirectories;
+        // Rather than requery this for each file and filespec, we query the supported extensions
+        // once and store it on the expansion context.
+        var supportedExtensions = ts.getSupportedExtensions(options, extraFileExtensions);
+        var supportedExtensionsWithJsonIfResolveJsonModule = ts.getSuppoertedExtensionsWithJsonIfResolveJsonModule(options, supportedExtensions);
+        // Literal files are always included verbatim. An "include" or "exclude" specification cannot
+        // remove a literal file.
+        if (validatedFilesSpec) {
+            for (var _i = 0, validatedFilesSpec_1 = validatedFilesSpec; _i < validatedFilesSpec_1.length; _i++) {
+                var fileName = validatedFilesSpec_1[_i];
+                var file = ts.getNormalizedAbsolutePath(fileName, basePath);
+                literalFileMap.set(keyMapper(file), file);
             }
-            var symbol = forEachIdentifierInEntityName(node.left.expression, /*parent*/ undefined, function (id, symbol) {
-                if (symbol) {
-                    addDeclarationToSymbol(symbol, id, 1536 /* Module */ | 67108864 /* Assignment */);
+        }
+        var jsonOnlyIncludeRegexes;
+        if (validatedIncludeSpecs && validatedIncludeSpecs.length > 0) {
+            var _loop_6 = function (file) {
+                if (ts.fileExtensionIs(file, ".json" /* Json */)) {
+                    // Valid only if *.json specified
+                    if (!jsonOnlyIncludeRegexes) {
+                        var includes = validatedIncludeSpecs.filter(function (s) { return ts.endsWith(s, ".json" /* Json */); });
+                        var includeFilePatterns = ts.map(ts.getRegularExpressionsForWildcards(includes, basePath, "files"), function (pattern) { return "^" + pattern + "$"; });
+                        jsonOnlyIncludeRegexes = includeFilePatterns ? includeFilePatterns.map(function (pattern) { return ts.getRegexFromPattern(pattern, host.useCaseSensitiveFileNames); }) : ts.emptyArray;
+                    }
+                    var includeIndex = ts.findIndex(jsonOnlyIncludeRegexes, function (re) { return re.test(file); });
+                    if (includeIndex !== -1) {
+                        var key_1 = keyMapper(file);
+                        if (!literalFileMap.has(key_1) && !wildCardJsonFileMap.has(key_1)) {
+                            wildCardJsonFileMap.set(key_1, file);
+                        }
+                    }
+                    return "continue";
                 }
-                return symbol;
-            });
-            if (symbol) {
-                var flags = ts.isClassExpression(node.right) ?
-                    4 /* Property */ | 1048576 /* ExportValue */ | 32 /* Class */ :
-                    4 /* Property */ | 1048576 /* ExportValue */;
-                declareSymbol(symbol.exports, symbol, node.left, flags, 0 /* None */);
+                // If we have already included a literal or wildcard path with a
+                // higher priority extension, we should skip this file.
+                //
+                // This handles cases where we may encounter both <file>.ts and
+                // <file>.d.ts (or <file>.js if "allowJs" is enabled) in the same
+                // directory when they are compilation outputs.
+                if (hasFileWithHigherPriorityExtension(file, literalFileMap, wildcardFileMap, supportedExtensions, keyMapper)) {
+                    return "continue";
+                }
+                // We may have included a wildcard path with a lower priority
+                // extension due to the user-defined order of entries in the
+                // "include" array. If there is a lower priority extension in the
+                // same directory, we should remove it.
+                removeWildcardFilesWithLowerPriorityExtension(file, wildcardFileMap, supportedExtensions, keyMapper);
+                var key = keyMapper(file);
+                if (!literalFileMap.has(key) && !wildcardFileMap.has(key)) {
+                    wildcardFileMap.set(key, file);
+                }
+            };
+            for (var _a = 0, _b = host.readDirectory(basePath, supportedExtensionsWithJsonIfResolveJsonModule, validatedExcludeSpecs, validatedIncludeSpecs, /*depth*/ undefined); _a < _b.length; _a++) {
+                var file = _b[_a];
+                _loop_6(file);
             }
         }
-        function bindModuleExportsAssignment(node) {
-            // A common practice in node modules is to set 'export = module.exports = {}', this ensures that 'exports'
-            // is still pointing to 'module.exports'.
-            // We do not want to consider this as 'export=' since a module can have only one of these.
-            // Similarly we do not want to treat 'module.exports = exports' as an 'export='.
-            if (!setCommonJsModuleIndicator(node)) {
-                return;
-            }
-            var assignedExpression = ts.getRightMostAssignedExpression(node.right);
-            if (ts.isEmptyObjectLiteral(assignedExpression) || container === file && isExportsOrModuleExportsOrAlias(file, assignedExpression)) {
-                return;
+        var literalFiles = ts.arrayFrom(literalFileMap.values());
+        var wildcardFiles = ts.arrayFrom(wildcardFileMap.values());
+        return {
+            fileNames: literalFiles.concat(wildcardFiles, ts.arrayFrom(wildCardJsonFileMap.values())),
+            wildcardDirectories: wildcardDirectories,
+            spec: spec
+        };
+    }
+    ts.getFileNamesFromConfigSpecs = getFileNamesFromConfigSpecs;
+    /* @internal */
+    function isExcludedFile(pathToCheck, spec, basePath, useCaseSensitiveFileNames, currentDirectory) {
+        var validatedFilesSpec = spec.validatedFilesSpec, validatedIncludeSpecs = spec.validatedIncludeSpecs, validatedExcludeSpecs = spec.validatedExcludeSpecs;
+        if (!ts.length(validatedIncludeSpecs) || !ts.length(validatedExcludeSpecs))
+            return false;
+        basePath = ts.normalizePath(basePath);
+        var keyMapper = ts.createGetCanonicalFileName(useCaseSensitiveFileNames);
+        if (validatedFilesSpec) {
+            for (var _i = 0, validatedFilesSpec_2 = validatedFilesSpec; _i < validatedFilesSpec_2.length; _i++) {
+                var fileName = validatedFilesSpec_2[_i];
+                if (keyMapper(ts.getNormalizedAbsolutePath(fileName, basePath)) === pathToCheck)
+                    return false;
             }
-            // 'module.exports = expr' assignment
-            var flags = ts.exportAssignmentIsAlias(node)
-                ? 2097152 /* Alias */
-                : 4 /* Property */ | 1048576 /* ExportValue */ | 512 /* ValueModule */;
-            var symbol = declareSymbol(file.symbol.exports, file.symbol, node, flags | 67108864 /* Assignment */, 0 /* None */);
-            ts.setValueDeclaration(symbol, node);
         }
-        function bindThisPropertyAssignment(node) {
-            ts.Debug.assert(ts.isInJSFile(node));
-            // private identifiers *must* be declared (even in JS files)
-            var hasPrivateIdentifier = (ts.isBinaryExpression(node) && ts.isPropertyAccessExpression(node.left) && ts.isPrivateIdentifier(node.left.name))
-                || (ts.isPropertyAccessExpression(node) && ts.isPrivateIdentifier(node.name));
-            if (hasPrivateIdentifier) {
-                return;
+        var excludePattern = ts.getRegularExpressionForWildcard(validatedExcludeSpecs, ts.combinePaths(ts.normalizePath(currentDirectory), basePath), "exclude");
+        var excludeRegex = excludePattern && ts.getRegexFromPattern(excludePattern, useCaseSensitiveFileNames);
+        if (!excludeRegex)
+            return false;
+        if (excludeRegex.test(pathToCheck))
+            return true;
+        return !ts.hasExtension(pathToCheck) && excludeRegex.test(ts.ensureTrailingDirectorySeparator(pathToCheck));
+    }
+    ts.isExcludedFile = isExcludedFile;
+    function validateSpecs(specs, errors, allowTrailingRecursion, jsonSourceFile, specKey) {
+        return specs.filter(function (spec) {
+            if (!ts.isString(spec))
+                return false;
+            var diag = specToDiagnostic(spec, allowTrailingRecursion);
+            if (diag !== undefined) {
+                errors.push(createDiagnostic(diag, spec));
             }
-            var thisContainer = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
-            switch (thisContainer.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                    var constructorSymbol = thisContainer.symbol;
-                    // For `f.prototype.m = function() { this.x = 0; }`, `this.x = 0` should modify `f`'s members, not the function expression.
-                    if (ts.isBinaryExpression(thisContainer.parent) && thisContainer.parent.operatorToken.kind === 62 /* EqualsToken */) {
-                        var l = thisContainer.parent.left;
-                        if (ts.isBindableStaticAccessExpression(l) && ts.isPrototypeAccess(l.expression)) {
-                            constructorSymbol = lookupSymbolForPropertyAccess(l.expression.expression, thisParentContainer);
+            return diag === undefined;
+        });
+        function createDiagnostic(message, spec) {
+            var element = ts.getTsConfigPropArrayElementValue(jsonSourceFile, specKey, spec);
+            return element ?
+                ts.createDiagnosticForNodeInSourceFile(jsonSourceFile, element, message, spec) :
+                ts.createCompilerDiagnostic(message, spec);
+        }
+    }
+    function specToDiagnostic(spec, allowTrailingRecursion) {
+        if (!allowTrailingRecursion && invalidTrailingRecursionPattern.test(spec)) {
+            return ts.Diagnostics.File_specification_cannot_end_in_a_recursive_directory_wildcard_Asterisk_Asterisk_Colon_0;
+        }
+        else if (invalidDotDotAfterRecursiveWildcardPattern.test(spec)) {
+            return ts.Diagnostics.File_specification_cannot_contain_a_parent_directory_that_appears_after_a_recursive_directory_wildcard_Asterisk_Asterisk_Colon_0;
+        }
+    }
+    /**
+     * Gets directories in a set of include patterns that should be watched for changes.
+     */
+    function getWildcardDirectories(include, exclude, path, useCaseSensitiveFileNames) {
+        // We watch a directory recursively if it contains a wildcard anywhere in a directory segment
+        // of the pattern:
+        //
+        //  /a/b/**/d   - Watch /a/b recursively to catch changes to any d in any subfolder recursively
+        //  /a/b/*/d    - Watch /a/b recursively to catch any d in any immediate subfolder, even if a new subfolder is added
+        //  /a/b        - Watch /a/b recursively to catch changes to anything in any recursive subfoler
+        //
+        // We watch a directory without recursion if it contains a wildcard in the file segment of
+        // the pattern:
+        //
+        //  /a/b/*      - Watch /a/b directly to catch any new file
+        //  /a/b/a?z    - Watch /a/b directly to catch any new file matching a?z
+        var rawExcludeRegex = ts.getRegularExpressionForWildcard(exclude, path, "exclude");
+        var excludeRegex = rawExcludeRegex && new RegExp(rawExcludeRegex, useCaseSensitiveFileNames ? "" : "i");
+        var wildcardDirectories = {};
+        if (include !== undefined) {
+            var recursiveKeys = [];
+            for (var _i = 0, include_1 = include; _i < include_1.length; _i++) {
+                var file = include_1[_i];
+                var spec = ts.normalizePath(ts.combinePaths(path, file));
+                if (excludeRegex && excludeRegex.test(spec)) {
+                    continue;
+                }
+                var match = getWildcardDirectoryFromSpec(spec, useCaseSensitiveFileNames);
+                if (match) {
+                    var key = match.key, flags = match.flags;
+                    var existingFlags = wildcardDirectories[key];
+                    if (existingFlags === undefined || existingFlags < flags) {
+                        wildcardDirectories[key] = flags;
+                        if (flags === 1 /* Recursive */) {
+                            recursiveKeys.push(key);
                         }
                     }
-                    if (constructorSymbol && constructorSymbol.valueDeclaration) {
-                        // Declare a 'member' if the container is an ES5 class or ES6 constructor
-                        constructorSymbol.members = constructorSymbol.members || ts.createSymbolTable();
-                        // It's acceptable for multiple 'this' assignments of the same identifier to occur
-                        if (ts.hasDynamicName(node)) {
-                            bindDynamicallyNamedThisPropertyAssignment(node, constructorSymbol);
-                        }
-                        else {
-                            declareSymbol(constructorSymbol.members, constructorSymbol, node, 4 /* Property */ | 67108864 /* Assignment */, 0 /* PropertyExcludes */ & ~4 /* Property */);
+                }
+            }
+            // Remove any subpaths under an existing recursively watched directory.
+            for (var key in wildcardDirectories) {
+                if (ts.hasProperty(wildcardDirectories, key)) {
+                    for (var _a = 0, recursiveKeys_1 = recursiveKeys; _a < recursiveKeys_1.length; _a++) {
+                        var recursiveKey = recursiveKeys_1[_a];
+                        if (key !== recursiveKey && ts.containsPath(recursiveKey, key, path, !useCaseSensitiveFileNames)) {
+                            delete wildcardDirectories[key];
                         }
-                        addDeclarationToSymbol(constructorSymbol, constructorSymbol.valueDeclaration, 32 /* Class */);
-                    }
-                    break;
-                case 162 /* Constructor */:
-                case 159 /* PropertyDeclaration */:
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    // this.foo assignment in a JavaScript class
-                    // Bind this property to the containing class
-                    var containingClass = thisContainer.parent;
-                    var symbolTable = ts.hasModifier(thisContainer, 32 /* Static */) ? containingClass.symbol.exports : containingClass.symbol.members;
-                    if (ts.hasDynamicName(node)) {
-                        bindDynamicallyNamedThisPropertyAssignment(node, containingClass.symbol);
-                    }
-                    else {
-                        declareSymbol(symbolTable, containingClass.symbol, node, 4 /* Property */ | 67108864 /* Assignment */, 0 /* None */, /*isReplaceableByMethod*/ true);
-                    }
-                    break;
-                case 290 /* SourceFile */:
-                    // this.property = assignment in a source file -- declare symbol in exports for a module, in locals for a script
-                    if (ts.hasDynamicName(node)) {
-                        break;
-                    }
-                    else if (thisContainer.commonJsModuleIndicator) {
-                        declareSymbol(thisContainer.symbol.exports, thisContainer.symbol, node, 4 /* Property */ | 1048576 /* ExportValue */, 0 /* None */);
-                    }
-                    else {
-                        declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111550 /* FunctionScopedVariableExcludes */);
                     }
-                    break;
-                default:
-                    ts.Debug.failBadSyntaxKind(thisContainer);
+                }
             }
         }
-        function bindDynamicallyNamedThisPropertyAssignment(node, symbol) {
-            bindAnonymousDeclaration(node, 4 /* Property */, "__computed" /* Computed */);
-            addLateBoundAssignmentDeclarationToSymbol(node, symbol);
+        return wildcardDirectories;
+    }
+    function getWildcardDirectoryFromSpec(spec, useCaseSensitiveFileNames) {
+        var match = wildcardDirectoryPattern.exec(spec);
+        if (match) {
+            return {
+                key: useCaseSensitiveFileNames ? match[0] : ts.toFileNameLowerCase(match[0]),
+                flags: watchRecursivePattern.test(spec) ? 1 /* Recursive */ : 0 /* None */
+            };
         }
-        function addLateBoundAssignmentDeclarationToSymbol(node, symbol) {
-            if (symbol) {
-                var members = symbol.assignmentDeclarationMembers || (symbol.assignmentDeclarationMembers = ts.createMap());
-                members.set("" + ts.getNodeId(node), node);
+        if (ts.isImplicitGlob(spec)) {
+            return {
+                key: useCaseSensitiveFileNames ? spec : ts.toFileNameLowerCase(spec),
+                flags: 1 /* Recursive */
+            };
+        }
+        return undefined;
+    }
+    /**
+     * Determines whether a literal or wildcard file has already been included that has a higher
+     * extension priority.
+     *
+     * @param file The path to the file.
+     * @param extensionPriority The priority of the extension.
+     * @param context The expansion context.
+     */
+    function hasFileWithHigherPriorityExtension(file, literalFiles, wildcardFiles, extensions, keyMapper) {
+        var extensionPriority = ts.getExtensionPriority(file, extensions);
+        var adjustedExtensionPriority = ts.adjustExtensionPriority(extensionPriority, extensions);
+        for (var i = 0 /* Highest */; i < adjustedExtensionPriority; i++) {
+            var higherPriorityExtension = extensions[i];
+            var higherPriorityPath = keyMapper(ts.changeExtension(file, higherPriorityExtension));
+            if (literalFiles.has(higherPriorityPath) || wildcardFiles.has(higherPriorityPath)) {
+                return true;
             }
         }
-        function bindSpecialPropertyDeclaration(node) {
-            if (node.expression.kind === 104 /* ThisKeyword */) {
-                bindThisPropertyAssignment(node);
-            }
-            else if (ts.isBindableStaticAccessExpression(node) && node.parent.parent.kind === 290 /* SourceFile */) {
-                if (ts.isPrototypeAccess(node.expression)) {
-                    bindPrototypePropertyAssignment(node, node.parent);
-                }
-                else {
-                    bindStaticPropertyAssignment(node);
+        return false;
+    }
+    /**
+     * Removes files included via wildcard expansion with a lower extension priority that have
+     * already been included.
+     *
+     * @param file The path to the file.
+     * @param extensionPriority The priority of the extension.
+     * @param context The expansion context.
+     */
+    function removeWildcardFilesWithLowerPriorityExtension(file, wildcardFiles, extensions, keyMapper) {
+        var extensionPriority = ts.getExtensionPriority(file, extensions);
+        var nextExtensionPriority = ts.getNextLowestExtensionPriority(extensionPriority, extensions);
+        for (var i = nextExtensionPriority; i < extensions.length; i++) {
+            var lowerPriorityExtension = extensions[i];
+            var lowerPriorityPath = keyMapper(ts.changeExtension(file, lowerPriorityExtension));
+            wildcardFiles.delete(lowerPriorityPath);
+        }
+    }
+    /**
+     * Produces a cleaned version of compiler options with personally identifying info (aka, paths) removed.
+     * Also converts enum values back to strings.
+     */
+    /* @internal */
+    function convertCompilerOptionsForTelemetry(opts) {
+        var out = {};
+        for (var key in opts) {
+            if (opts.hasOwnProperty(key)) {
+                var type = getOptionFromName(key);
+                if (type !== undefined) { // Ignore unknown options
+                    out[key] = getOptionValueWithEmptyStrings(opts[key], type);
                 }
             }
         }
-        /** For `x.prototype = { p, ... }`, declare members p,... if `x` is function/class/{}, or not declared. */
-        function bindPrototypeAssignment(node) {
-            node.left.parent = node;
-            node.right.parent = node;
-            bindPropertyAssignment(node.left.expression, node.left, /*isPrototypeProperty*/ false, /*containerIsClass*/ true);
+        return out;
+    }
+    ts.convertCompilerOptionsForTelemetry = convertCompilerOptionsForTelemetry;
+    function getOptionValueWithEmptyStrings(value, option) {
+        switch (option.type) {
+            case "object": // "paths". Can't get any useful information from the value since we blank out strings, so just return "".
+                return "";
+            case "string": // Could be any arbitrary string -- use empty string instead.
+                return "";
+            case "number": // Allow numbers, but be sure to check it's actually a number.
+                return typeof value === "number" ? value : "";
+            case "boolean":
+                return typeof value === "boolean" ? value : "";
+            case "list":
+                var elementType_1 = option.element;
+                return ts.isArray(value) ? value.map(function (v) { return getOptionValueWithEmptyStrings(v, elementType_1); }) : "";
+            default:
+                return ts.forEachEntry(option.type, function (optionEnumValue, optionStringValue) {
+                    if (optionEnumValue === value) {
+                        return optionStringValue;
+                    }
+                }); // TODO: GH#18217
         }
-        function bindObjectDefinePrototypeProperty(node) {
-            var namespaceSymbol = lookupSymbolForPropertyAccess(node.arguments[0].expression);
-            if (namespaceSymbol && namespaceSymbol.valueDeclaration) {
-                // Ensure the namespace symbol becomes class-like
-                addDeclarationToSymbol(namespaceSymbol, namespaceSymbol.valueDeclaration, 32 /* Class */);
+    }
+})(ts || (ts = {}));
+var ts;
+(function (ts) {
+    function trace(host) {
+        host.trace(ts.formatMessage.apply(undefined, arguments));
+    }
+    ts.trace = trace;
+    /* @internal */
+    function isTraceEnabled(compilerOptions, host) {
+        return !!compilerOptions.traceResolution && host.trace !== undefined;
+    }
+    ts.isTraceEnabled = isTraceEnabled;
+    function withPackageId(packageInfo, r) {
+        var packageId;
+        if (r && packageInfo) {
+            var packageJsonContent = packageInfo.packageJsonContent;
+            if (typeof packageJsonContent.name === "string" && typeof packageJsonContent.version === "string") {
+                packageId = {
+                    name: packageJsonContent.name,
+                    subModuleName: r.path.slice(packageInfo.packageDirectory.length + ts.directorySeparator.length),
+                    version: packageJsonContent.version
+                };
             }
-            bindPotentiallyNewExpandoMemberToNamespace(node, namespaceSymbol, /*isPrototypeProperty*/ true);
-        }
-        /**
-         * For `x.prototype.y = z`, declare a member `y` on `x` if `x` is a function or class, or not declared.
-         * Note that jsdoc preceding an ExpressionStatement like `x.prototype.y;` is also treated as a declaration.
-         */
-        function bindPrototypePropertyAssignment(lhs, parent) {
-            // Look up the function in the local scope, since prototype assignments should
-            // follow the function declaration
-            var classPrototype = lhs.expression;
-            var constructorFunction = classPrototype.expression;
-            // Fix up parent pointers since we're going to use these nodes before we bind into them
-            lhs.parent = parent;
-            constructorFunction.parent = classPrototype;
-            classPrototype.parent = lhs;
-            bindPropertyAssignment(constructorFunction, lhs, /*isPrototypeProperty*/ true, /*containerIsClass*/ true);
         }
-        function bindObjectDefinePropertyAssignment(node) {
-            var namespaceSymbol = lookupSymbolForPropertyAccess(node.arguments[0]);
-            var isToplevel = node.parent.parent.kind === 290 /* SourceFile */;
-            namespaceSymbol = bindPotentiallyMissingNamespaces(namespaceSymbol, node.arguments[0], isToplevel, /*isPrototypeProperty*/ false, /*containerIsClass*/ false);
-            bindPotentiallyNewExpandoMemberToNamespace(node, namespaceSymbol, /*isPrototypeProperty*/ false);
+        return r && { path: r.path, extension: r.ext, packageId: packageId };
+    }
+    function noPackageId(r) {
+        return withPackageId(/*packageInfo*/ undefined, r);
+    }
+    function removeIgnoredPackageId(r) {
+        if (r) {
+            ts.Debug.assert(r.packageId === undefined);
+            return { path: r.path, ext: r.extension };
         }
-        function bindSpecialPropertyAssignment(node) {
-            // Class declarations in Typescript do not allow property declarations
-            var parentSymbol = lookupSymbolForPropertyAccess(node.left.expression);
-            if (!ts.isInJSFile(node) && !ts.isFunctionSymbol(parentSymbol)) {
-                return;
-            }
-            // Fix up parent pointers since we're going to use these nodes before we bind into them
-            node.left.parent = node;
-            node.right.parent = node;
-            if (ts.isIdentifier(node.left.expression) && container === file && isExportsOrModuleExportsOrAlias(file, node.left.expression)) {
-                // This can be an alias for the 'exports' or 'module.exports' names, e.g.
-                //    var util = module.exports;
-                //    util.property = function ...
-                bindExportsPropertyAssignment(node);
-            }
-            else if (ts.hasDynamicName(node)) {
-                bindAnonymousDeclaration(node, 4 /* Property */ | 67108864 /* Assignment */, "__computed" /* Computed */);
-                var sym = bindPotentiallyMissingNamespaces(parentSymbol, node.left.expression, isTopLevelNamespaceAssignment(node.left), /*isPrototype*/ false, /*containerIsClass*/ false);
-                addLateBoundAssignmentDeclarationToSymbol(node, sym);
-            }
-            else {
-                bindStaticPropertyAssignment(ts.cast(node.left, ts.isBindableStaticNameExpression));
-            }
+    }
+    /**
+     * Kinds of file that we are currently looking for.
+     * Typically there is one pass with Extensions.TypeScript, then a second pass with Extensions.JavaScript.
+     */
+    var Extensions;
+    (function (Extensions) {
+        Extensions[Extensions["TypeScript"] = 0] = "TypeScript";
+        Extensions[Extensions["JavaScript"] = 1] = "JavaScript";
+        Extensions[Extensions["Json"] = 2] = "Json";
+        Extensions[Extensions["TSConfig"] = 3] = "TSConfig";
+        Extensions[Extensions["DtsOnly"] = 4] = "DtsOnly"; /** Only '.d.ts' */
+    })(Extensions || (Extensions = {}));
+    /** Used with `Extensions.DtsOnly` to extract the path from TypeScript results. */
+    function resolvedTypeScriptOnly(resolved) {
+        if (!resolved) {
+            return undefined;
         }
-        /**
-         * For nodes like `x.y = z`, declare a member 'y' on 'x' if x is a function (or IIFE) or class or {}, or not declared.
-         * Also works for expression statements preceded by JSDoc, like / ** @type number * / x.y;
-         */
-        function bindStaticPropertyAssignment(node) {
-            ts.Debug.assert(!ts.isIdentifier(node));
-            node.expression.parent = node;
-            bindPropertyAssignment(node.expression, node, /*isPrototypeProperty*/ false, /*containerIsClass*/ false);
+        ts.Debug.assert(ts.extensionIsTS(resolved.extension));
+        return { fileName: resolved.path, packageId: resolved.packageId };
+    }
+    function createResolvedModuleWithFailedLookupLocations(resolved, isExternalLibraryImport, failedLookupLocations, resultFromCache) {
+        var _a;
+        if (resultFromCache) {
+            (_a = resultFromCache.failedLookupLocations).push.apply(_a, failedLookupLocations);
+            return resultFromCache;
         }
-        function bindPotentiallyMissingNamespaces(namespaceSymbol, entityName, isToplevel, isPrototypeProperty, containerIsClass) {
-            if (isToplevel && !isPrototypeProperty) {
-                // make symbols or add declarations for intermediate containers
-                var flags_1 = 1536 /* Module */ | 67108864 /* Assignment */;
-                var excludeFlags_1 = 110735 /* ValueModuleExcludes */ & ~67108864 /* Assignment */;
-                namespaceSymbol = forEachIdentifierInEntityName(entityName, namespaceSymbol, function (id, symbol, parent) {
-                    if (symbol) {
-                        addDeclarationToSymbol(symbol, id, flags_1);
-                        return symbol;
-                    }
-                    else {
-                        var table = parent ? parent.exports :
-                            file.jsGlobalAugmentations || (file.jsGlobalAugmentations = ts.createSymbolTable());
-                        return declareSymbol(table, parent, id, flags_1, excludeFlags_1);
-                    }
-                });
-            }
-            if (containerIsClass && namespaceSymbol && namespaceSymbol.valueDeclaration) {
-                addDeclarationToSymbol(namespaceSymbol, namespaceSymbol.valueDeclaration, 32 /* Class */);
+        return {
+            resolvedModule: resolved && { resolvedFileName: resolved.path, originalPath: resolved.originalPath === true ? undefined : resolved.originalPath, extension: resolved.extension, isExternalLibraryImport: isExternalLibraryImport, packageId: resolved.packageId },
+            failedLookupLocations: failedLookupLocations
+        };
+    }
+    function readPackageJsonField(jsonContent, fieldName, typeOfTag, state) {
+        if (!ts.hasProperty(jsonContent, fieldName)) {
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.package_json_does_not_have_a_0_field, fieldName);
             }
-            return namespaceSymbol;
+            return;
         }
-        function bindPotentiallyNewExpandoMemberToNamespace(declaration, namespaceSymbol, isPrototypeProperty) {
-            if (!namespaceSymbol || !isExpandoSymbol(namespaceSymbol)) {
-                return;
-            }
-            // Set up the members collection if it doesn't exist already
-            var symbolTable = isPrototypeProperty ?
-                (namespaceSymbol.members || (namespaceSymbol.members = ts.createSymbolTable())) :
-                (namespaceSymbol.exports || (namespaceSymbol.exports = ts.createSymbolTable()));
-            var includes = 0 /* None */;
-            var excludes = 0 /* None */;
-            // Method-like
-            if (ts.isFunctionLikeDeclaration(ts.getAssignedExpandoInitializer(declaration))) {
-                includes = 8192 /* Method */;
-                excludes = 103359 /* MethodExcludes */;
-            }
-            // Maybe accessor-like
-            else if (ts.isCallExpression(declaration) && ts.isBindableObjectDefinePropertyCall(declaration)) {
-                if (ts.some(declaration.arguments[2].properties, function (p) {
-                    var id = ts.getNameOfDeclaration(p);
-                    return !!id && ts.isIdentifier(id) && ts.idText(id) === "set";
-                })) {
-                    // We mix in `SymbolFLags.Property` so in the checker `getTypeOfVariableParameterOrProperty` is used for this
-                    // symbol, instead of `getTypeOfAccessor` (which will assert as there is no real accessor declaration)
-                    includes |= 65536 /* SetAccessor */ | 4 /* Property */;
-                    excludes |= 78783 /* SetAccessorExcludes */;
-                }
-                if (ts.some(declaration.arguments[2].properties, function (p) {
-                    var id = ts.getNameOfDeclaration(p);
-                    return !!id && ts.isIdentifier(id) && ts.idText(id) === "get";
-                })) {
-                    includes |= 32768 /* GetAccessor */ | 4 /* Property */;
-                    excludes |= 46015 /* GetAccessorExcludes */;
-                }
+        var value = jsonContent[fieldName];
+        if (typeof value !== typeOfTag || value === null) { // eslint-disable-line no-null/no-null
+            if (state.traceEnabled) {
+                // eslint-disable-next-line no-null/no-null
+                trace(state.host, ts.Diagnostics.Expected_type_of_0_field_in_package_json_to_be_1_got_2, fieldName, typeOfTag, value === null ? "null" : typeof value);
             }
-            if (includes === 0 /* None */) {
-                includes = 4 /* Property */;
-                excludes = 0 /* PropertyExcludes */;
+            return;
+        }
+        return value;
+    }
+    function readPackageJsonPathField(jsonContent, fieldName, baseDirectory, state) {
+        var fileName = readPackageJsonField(jsonContent, fieldName, "string", state);
+        if (fileName === undefined) {
+            return;
+        }
+        if (!fileName) {
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.package_json_had_a_falsy_0_field, fieldName);
             }
-            declareSymbol(symbolTable, namespaceSymbol, declaration, includes | 67108864 /* Assignment */, excludes & ~67108864 /* Assignment */);
+            return;
         }
-        function isTopLevelNamespaceAssignment(propertyAccess) {
-            return ts.isBinaryExpression(propertyAccess.parent)
-                ? getParentOfBinaryExpression(propertyAccess.parent).parent.kind === 290 /* SourceFile */
-                : propertyAccess.parent.parent.kind === 290 /* SourceFile */;
+        var path = ts.normalizePath(ts.combinePaths(baseDirectory, fileName));
+        if (state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.package_json_has_0_field_1_that_references_2, fieldName, fileName, path);
         }
-        function bindPropertyAssignment(name, propertyAccess, isPrototypeProperty, containerIsClass) {
-            var namespaceSymbol = lookupSymbolForPropertyAccess(name);
-            var isToplevel = isTopLevelNamespaceAssignment(propertyAccess);
-            namespaceSymbol = bindPotentiallyMissingNamespaces(namespaceSymbol, propertyAccess.expression, isToplevel, isPrototypeProperty, containerIsClass);
-            bindPotentiallyNewExpandoMemberToNamespace(propertyAccess, namespaceSymbol, isPrototypeProperty);
+        return path;
+    }
+    function readPackageJsonTypesFields(jsonContent, baseDirectory, state) {
+        return readPackageJsonPathField(jsonContent, "typings", baseDirectory, state)
+            || readPackageJsonPathField(jsonContent, "types", baseDirectory, state);
+    }
+    function readPackageJsonTSConfigField(jsonContent, baseDirectory, state) {
+        return readPackageJsonPathField(jsonContent, "tsconfig", baseDirectory, state);
+    }
+    function readPackageJsonMainField(jsonContent, baseDirectory, state) {
+        return readPackageJsonPathField(jsonContent, "main", baseDirectory, state);
+    }
+    function readPackageJsonTypesVersionsField(jsonContent, state) {
+        var typesVersions = readPackageJsonField(jsonContent, "typesVersions", "object", state);
+        if (typesVersions === undefined)
+            return;
+        if (state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_field_with_version_specific_path_mappings);
         }
-        /**
-         * Javascript expando values are:
-         * - Functions
-         * - classes
-         * - namespaces
-         * - variables initialized with function expressions
-         * -                       with class expressions
-         * -                       with empty object literals
-         * -                       with non-empty object literals if assigned to the prototype property
-         */
-        function isExpandoSymbol(symbol) {
-            if (symbol.flags & (16 /* Function */ | 32 /* Class */ | 1024 /* NamespaceModule */)) {
-                return true;
-            }
-            var node = symbol.valueDeclaration;
-            if (node && ts.isCallExpression(node)) {
-                return !!ts.getAssignedExpandoInitializer(node);
-            }
-            var init = !node ? undefined :
-                ts.isVariableDeclaration(node) ? node.initializer :
-                    ts.isBinaryExpression(node) ? node.right :
-                        ts.isPropertyAccessExpression(node) && ts.isBinaryExpression(node.parent) ? node.parent.right :
-                            undefined;
-            init = init && ts.getRightMostAssignedExpression(init);
-            if (init) {
-                var isPrototypeAssignment = ts.isPrototypeAccess(ts.isVariableDeclaration(node) ? node.name : ts.isBinaryExpression(node) ? node.left : node);
-                return !!ts.getExpandoInitializer(ts.isBinaryExpression(init) && (init.operatorToken.kind === 56 /* BarBarToken */ || init.operatorToken.kind === 60 /* QuestionQuestionToken */) ? init.right : init, isPrototypeAssignment);
+        return typesVersions;
+    }
+    function readPackageJsonTypesVersionPaths(jsonContent, state) {
+        var typesVersions = readPackageJsonTypesVersionsField(jsonContent, state);
+        if (typesVersions === undefined)
+            return;
+        if (state.traceEnabled) {
+            for (var key in typesVersions) {
+                if (ts.hasProperty(typesVersions, key) && !ts.VersionRange.tryParse(key)) {
+                    trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_entry_0_that_is_not_a_valid_semver_range, key);
+                }
             }
-            return false;
         }
-        function getParentOfBinaryExpression(expr) {
-            while (ts.isBinaryExpression(expr.parent)) {
-                expr = expr.parent;
+        var result = getPackageJsonTypesVersionsPaths(typesVersions);
+        if (!result) {
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.package_json_does_not_have_a_typesVersions_entry_that_matches_version_0, ts.versionMajorMinor);
             }
-            return expr.parent;
+            return;
         }
-        function lookupSymbolForPropertyAccess(node, lookupContainer) {
-            if (lookupContainer === void 0) { lookupContainer = container; }
-            if (ts.isIdentifier(node)) {
-                return lookupSymbolForNameWorker(lookupContainer, node.escapedText);
-            }
-            else {
-                var symbol = lookupSymbolForPropertyAccess(node.expression);
-                return symbol && symbol.exports && symbol.exports.get(ts.getElementOrPropertyAccessName(node));
+        var bestVersionKey = result.version, bestVersionPaths = result.paths;
+        if (typeof bestVersionPaths !== "object") {
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Expected_type_of_0_field_in_package_json_to_be_1_got_2, "typesVersions['" + bestVersionKey + "']", "object", typeof bestVersionPaths);
             }
+            return;
         }
-        function forEachIdentifierInEntityName(e, parent, action) {
-            if (isExportsOrModuleExportsOrAlias(file, e)) {
-                return file.symbol;
-            }
-            else if (ts.isIdentifier(e)) {
-                return action(e, lookupSymbolForPropertyAccess(e), parent);
+        return result;
+    }
+    var typeScriptVersion;
+    /* @internal */
+    function getPackageJsonTypesVersionsPaths(typesVersions) {
+        if (!typeScriptVersion)
+            typeScriptVersion = new ts.Version(ts.version);
+        for (var key in typesVersions) {
+            if (!ts.hasProperty(typesVersions, key))
+                continue;
+            var keyRange = ts.VersionRange.tryParse(key);
+            if (keyRange === undefined) {
+                continue;
             }
-            else {
-                var s = forEachIdentifierInEntityName(e.expression, parent, action);
-                var name = ts.getNameOrArgument(e);
-                // unreachable
-                if (ts.isPrivateIdentifier(name)) {
-                    ts.Debug.fail("unexpected PrivateIdentifier");
-                }
-                return action(name, s && s.exports && s.exports.get(ts.getElementOrPropertyAccessName(e)), s);
+            // return the first entry whose range matches the current compiler version.
+            if (keyRange.test(typeScriptVersion)) {
+                return { version: key, paths: typesVersions[key] };
             }
         }
-        function bindCallExpression(node) {
-            // We're only inspecting call expressions to detect CommonJS modules, so we can skip
-            // this check if we've already seen the module indicator
-            if (!file.commonJsModuleIndicator && ts.isRequireCall(node, /*checkArgumentIsStringLiteralLike*/ false)) {
-                setCommonJsModuleIndicator(node);
+    }
+    ts.getPackageJsonTypesVersionsPaths = getPackageJsonTypesVersionsPaths;
+    function getEffectiveTypeRoots(options, host) {
+        if (options.typeRoots) {
+            return options.typeRoots;
+        }
+        var currentDirectory;
+        if (options.configFilePath) {
+            currentDirectory = ts.getDirectoryPath(options.configFilePath);
+        }
+        else if (host.getCurrentDirectory) {
+            currentDirectory = host.getCurrentDirectory();
+        }
+        if (currentDirectory !== undefined) {
+            return getDefaultTypeRoots(currentDirectory, host);
+        }
+    }
+    ts.getEffectiveTypeRoots = getEffectiveTypeRoots;
+    /**
+     * Returns the path to every node_modules/@types directory from some ancestor directory.
+     * Returns undefined if there are none.
+     */
+    function getDefaultTypeRoots(currentDirectory, host) {
+        if (!host.directoryExists) {
+            return [ts.combinePaths(currentDirectory, nodeModulesAtTypes)];
+            // And if it doesn't exist, tough.
+        }
+        var typeRoots;
+        ts.forEachAncestorDirectory(ts.normalizePath(currentDirectory), function (directory) {
+            var atTypes = ts.combinePaths(directory, nodeModulesAtTypes);
+            if (host.directoryExists(atTypes)) {
+                (typeRoots || (typeRoots = [])).push(atTypes);
             }
+            return undefined;
+        });
+        return typeRoots;
+    }
+    var nodeModulesAtTypes = ts.combinePaths("node_modules", "@types");
+    /**
+     * @param {string | undefined} containingFile - file that contains type reference directive, can be undefined if containing file is unknown.
+     * This is possible in case if resolution is performed for directives specified via 'types' parameter. In this case initial path for secondary lookups
+     * is assumed to be the same as root directory of the project.
+     */
+    function resolveTypeReferenceDirective(typeReferenceDirectiveName, containingFile, options, host, redirectedReference) {
+        var traceEnabled = isTraceEnabled(options, host);
+        if (redirectedReference) {
+            options = redirectedReference.commandLine.options;
         }
-        function bindClassLikeDeclaration(node) {
-            if (node.kind === 245 /* ClassDeclaration */) {
-                bindBlockScopedDeclaration(node, 32 /* Class */, 899503 /* ClassExcludes */);
+        var failedLookupLocations = [];
+        var moduleResolutionState = { compilerOptions: options, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
+        var typeRoots = getEffectiveTypeRoots(options, host);
+        if (traceEnabled) {
+            if (containingFile === undefined) {
+                if (typeRoots === undefined) {
+                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_not_set_root_directory_not_set, typeReferenceDirectiveName);
+                }
+                else {
+                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_not_set_root_directory_1, typeReferenceDirectiveName, typeRoots);
+                }
             }
             else {
-                var bindingName = node.name ? node.name.escapedText : "__class" /* Class */;
-                bindAnonymousDeclaration(node, 32 /* Class */, bindingName);
-                // Add name of class expression into the map for semantic classifier
-                if (node.name) {
-                    classifiableNames.set(node.name.escapedText, true);
+                if (typeRoots === undefined) {
+                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_1_root_directory_not_set, typeReferenceDirectiveName, containingFile);
                 }
-            }
-            var symbol = node.symbol;
-            // TypeScript 1.0 spec (April 2014): 8.4
-            // Every class automatically contains a static property member named 'prototype', the
-            // type of which is an instantiation of the class type with type Any supplied as a type
-            // argument for each type parameter. It is an error to explicitly declare a static
-            // property member with the name 'prototype'.
-            //
-            // Note: we check for this here because this class may be merging into a module.  The
-            // module might have an exported variable called 'prototype'.  We can't allow that as
-            // that would clash with the built-in 'prototype' for the class.
-            var prototypeSymbol = createSymbol(4 /* Property */ | 4194304 /* Prototype */, "prototype");
-            var symbolExport = symbol.exports.get(prototypeSymbol.escapedName);
-            if (symbolExport) {
-                if (node.name) {
-                    node.name.parent = node;
+                else {
+                    trace(host, ts.Diagnostics.Resolving_type_reference_directive_0_containing_file_1_root_directory_2, typeReferenceDirectiveName, containingFile, typeRoots);
                 }
-                file.bindDiagnostics.push(createDiagnosticForNode(symbolExport.declarations[0], ts.Diagnostics.Duplicate_identifier_0, ts.symbolName(prototypeSymbol)));
             }
-            symbol.exports.set(prototypeSymbol.escapedName, prototypeSymbol);
-            prototypeSymbol.parent = symbol;
+            if (redirectedReference) {
+                trace(host, ts.Diagnostics.Using_compiler_options_of_project_reference_redirect_0, redirectedReference.sourceFile.fileName);
+            }
         }
-        function bindEnumDeclaration(node) {
-            return ts.isEnumConst(node)
-                ? bindBlockScopedDeclaration(node, 128 /* ConstEnum */, 899967 /* ConstEnumExcludes */)
-                : bindBlockScopedDeclaration(node, 256 /* RegularEnum */, 899327 /* RegularEnumExcludes */);
+        var resolved = primaryLookup();
+        var primary = true;
+        if (!resolved) {
+            resolved = secondaryLookup();
+            primary = false;
         }
-        function bindVariableDeclarationOrBindingElement(node) {
-            if (inStrictMode) {
-                checkStrictModeEvalOrArguments(node, node.name);
-            }
-            if (!ts.isBindingPattern(node.name)) {
-                if (ts.isBlockOrCatchScoped(node)) {
-                    bindBlockScopedDeclaration(node, 2 /* BlockScopedVariable */, 111551 /* BlockScopedVariableExcludes */);
-                }
-                else if (ts.isParameterDeclaration(node)) {
-                    // It is safe to walk up parent chain to find whether the node is a destructuring parameter declaration
-                    // because its parent chain has already been set up, since parents are set before descending into children.
-                    //
-                    // If node is a binding element in parameter declaration, we need to use ParameterExcludes.
-                    // Using ParameterExcludes flag allows the compiler to report an error on duplicate identifiers in Parameter Declaration
-                    // For example:
-                    //      function foo([a,a]) {} // Duplicate Identifier error
-                    //      function bar(a,a) {}   // Duplicate Identifier error, parameter declaration in this case is handled in bindParameter
-                    //                             // which correctly set excluded symbols
-                    declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111551 /* ParameterExcludes */);
+        var resolvedTypeReferenceDirective;
+        if (resolved) {
+            var fileName = resolved.fileName, packageId = resolved.packageId;
+            var resolvedFileName = options.preserveSymlinks ? fileName : realPath(fileName, host, traceEnabled);
+            if (traceEnabled) {
+                if (packageId) {
+                    trace(host, ts.Diagnostics.Type_reference_directive_0_was_successfully_resolved_to_1_with_Package_ID_2_primary_Colon_3, typeReferenceDirectiveName, resolvedFileName, ts.packageIdToString(packageId), primary);
                 }
                 else {
-                    declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111550 /* FunctionScopedVariableExcludes */);
+                    trace(host, ts.Diagnostics.Type_reference_directive_0_was_successfully_resolved_to_1_primary_Colon_2, typeReferenceDirectiveName, resolvedFileName, primary);
                 }
             }
+            resolvedTypeReferenceDirective = { primary: primary, resolvedFileName: resolvedFileName, packageId: packageId, isExternalLibraryImport: pathContainsNodeModules(fileName) };
         }
-        function bindParameter(node) {
-            if (node.kind === 317 /* JSDocParameterTag */ && container.kind !== 305 /* JSDocSignature */) {
-                return;
-            }
-            if (inStrictMode && !(node.flags & 8388608 /* Ambient */)) {
-                // It is a SyntaxError if the identifier eval or arguments appears within a FormalParameterList of a
-                // strict mode FunctionLikeDeclaration or FunctionExpression(13.1)
-                checkStrictModeEvalOrArguments(node, node.name);
-            }
-            if (ts.isBindingPattern(node.name)) {
-                bindAnonymousDeclaration(node, 1 /* FunctionScopedVariable */, "__" + node.parent.parameters.indexOf(node));
-            }
-            else {
-                declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111551 /* ParameterExcludes */);
-            }
-            // If this is a property-parameter, then also declare the property symbol into the
-            // containing class.
-            if (ts.isParameterPropertyDeclaration(node, node.parent)) {
-                var classDeclaration = node.parent.parent;
-                declareSymbol(classDeclaration.symbol.members, classDeclaration.symbol, node, 4 /* Property */ | (node.questionToken ? 16777216 /* Optional */ : 0 /* None */), 0 /* PropertyExcludes */);
-            }
-        }
-        function bindFunctionDeclaration(node) {
-            if (!file.isDeclarationFile && !(node.flags & 8388608 /* Ambient */)) {
-                if (ts.isAsyncFunction(node)) {
-                    emitFlags |= 2048 /* HasAsyncFunctions */;
+        return { resolvedTypeReferenceDirective: resolvedTypeReferenceDirective, failedLookupLocations: failedLookupLocations };
+        function primaryLookup() {
+            // Check primary library paths
+            if (typeRoots && typeRoots.length) {
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Resolving_with_primary_search_path_0, typeRoots.join(", "));
                 }
-            }
-            checkStrictModeFunctionName(node);
-            if (inStrictMode) {
-                checkStrictModeFunctionDeclaration(node);
-                bindBlockScopedDeclaration(node, 16 /* Function */, 110991 /* FunctionExcludes */);
+                return ts.firstDefined(typeRoots, function (typeRoot) {
+                    var candidate = ts.combinePaths(typeRoot, typeReferenceDirectiveName);
+                    var candidateDirectory = ts.getDirectoryPath(candidate);
+                    var directoryExists = ts.directoryProbablyExists(candidateDirectory, host);
+                    if (!directoryExists && traceEnabled) {
+                        trace(host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, candidateDirectory);
+                    }
+                    return resolvedTypeScriptOnly(loadNodeModuleFromDirectory(Extensions.DtsOnly, candidate, !directoryExists, moduleResolutionState));
+                });
             }
             else {
-                declareSymbolAndAddToSymbolTable(node, 16 /* Function */, 110991 /* FunctionExcludes */);
-            }
-        }
-        function bindFunctionExpression(node) {
-            if (!file.isDeclarationFile && !(node.flags & 8388608 /* Ambient */)) {
-                if (ts.isAsyncFunction(node)) {
-                    emitFlags |= 2048 /* HasAsyncFunctions */;
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Root_directory_cannot_be_determined_skipping_primary_search_paths);
                 }
             }
-            if (currentFlow) {
-                node.flowNode = currentFlow;
-            }
-            checkStrictModeFunctionName(node);
-            var bindingName = node.name ? node.name.escapedText : "__function" /* Function */;
-            return bindAnonymousDeclaration(node, 16 /* Function */, bindingName);
-        }
-        function bindPropertyOrMethodOrAccessor(node, symbolFlags, symbolExcludes) {
-            if (!file.isDeclarationFile && !(node.flags & 8388608 /* Ambient */) && ts.isAsyncFunction(node)) {
-                emitFlags |= 2048 /* HasAsyncFunctions */;
-            }
-            if (currentFlow && ts.isObjectLiteralOrClassExpressionMethod(node)) {
-                node.flowNode = currentFlow;
-            }
-            return ts.hasDynamicName(node)
-                ? bindAnonymousDeclaration(node, symbolFlags, "__computed" /* Computed */)
-                : declareSymbolAndAddToSymbolTable(node, symbolFlags, symbolExcludes);
-        }
-        function getInferTypeContainer(node) {
-            var extendsType = ts.findAncestor(node, function (n) { return n.parent && ts.isConditionalTypeNode(n.parent) && n.parent.extendsType === n; });
-            return extendsType && extendsType.parent;
         }
-        function bindTypeParameter(node) {
-            if (ts.isJSDocTemplateTag(node.parent)) {
-                var container_1 = ts.find(node.parent.parent.tags, ts.isJSDocTypeAlias) || ts.getHostSignatureFromJSDoc(node.parent); // TODO: GH#18217
-                if (container_1) {
-                    if (!container_1.locals) {
-                        container_1.locals = ts.createSymbolTable();
-                    }
-                    declareSymbol(container_1.locals, /*parent*/ undefined, node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
-                }
-                else {
-                    declareSymbolAndAddToSymbolTable(node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
+        function secondaryLookup() {
+            var initialLocationForSecondaryLookup = containingFile && ts.getDirectoryPath(containingFile);
+            if (initialLocationForSecondaryLookup !== undefined) {
+                // check secondary locations
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Looking_up_in_node_modules_folder_initial_location_0, initialLocationForSecondaryLookup);
                 }
-            }
-            else if (node.parent.kind === 181 /* InferType */) {
-                var container_2 = getInferTypeContainer(node.parent);
-                if (container_2) {
-                    if (!container_2.locals) {
-                        container_2.locals = ts.createSymbolTable();
-                    }
-                    declareSymbol(container_2.locals, /*parent*/ undefined, node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
+                var result = void 0;
+                if (!ts.isExternalModuleNameRelative(typeReferenceDirectiveName)) {
+                    var searchResult = loadModuleFromNearestNodeModulesDirectory(Extensions.DtsOnly, typeReferenceDirectiveName, initialLocationForSecondaryLookup, moduleResolutionState, /*cache*/ undefined, /*redirectedReference*/ undefined);
+                    result = searchResult && searchResult.value;
                 }
                 else {
-                    bindAnonymousDeclaration(node, 262144 /* TypeParameter */, getDeclarationName(node)); // TODO: GH#18217
+                    var candidate = ts.normalizePathAndParts(ts.combinePaths(initialLocationForSecondaryLookup, typeReferenceDirectiveName)).path;
+                    result = nodeLoadModuleByRelativeName(Extensions.DtsOnly, candidate, /*onlyRecordFailures*/ false, moduleResolutionState, /*considerPackageJson*/ true);
+                }
+                var resolvedFile = resolvedTypeScriptOnly(result);
+                if (!resolvedFile && traceEnabled) {
+                    trace(host, ts.Diagnostics.Type_reference_directive_0_was_not_resolved, typeReferenceDirectiveName);
                 }
+                return resolvedFile;
             }
             else {
-                declareSymbolAndAddToSymbolTable(node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
-            }
-        }
-        // reachability checks
-        function shouldReportErrorOnModuleDeclaration(node) {
-            var instanceState = getModuleInstanceState(node);
-            return instanceState === 1 /* Instantiated */ || (instanceState === 2 /* ConstEnumOnly */ && !!options.preserveConstEnums);
-        }
-        function checkUnreachable(node) {
-            if (!(currentFlow.flags & 1 /* Unreachable */)) {
-                return false;
-            }
-            if (currentFlow === unreachableFlow) {
-                var reportError = 
-                // report error on all statements except empty ones
-                (ts.isStatementButNotDeclaration(node) && node.kind !== 224 /* EmptyStatement */) ||
-                    // report error on class declarations
-                    node.kind === 245 /* ClassDeclaration */ ||
-                    // report error on instantiated modules or const-enums only modules if preserveConstEnums is set
-                    (node.kind === 249 /* ModuleDeclaration */ && shouldReportErrorOnModuleDeclaration(node));
-                if (reportError) {
-                    currentFlow = reportedUnreachableFlow;
-                    if (!options.allowUnreachableCode) {
-                        // unreachable code is reported if
-                        // - user has explicitly asked about it AND
-                        // - statement is in not ambient context (statements in ambient context is already an error
-                        //   so we should not report extras) AND
-                        //   - node is not variable statement OR
-                        //   - node is block scoped variable statement OR
-                        //   - node is not block scoped variable statement and at least one variable declaration has initializer
-                        //   Rationale: we don't want to report errors on non-initialized var's since they are hoisted
-                        //   On the other side we do want to report errors on non-initialized 'lets' because of TDZ
-                        var isError_1 = ts.unreachableCodeIsError(options) &&
-                            !(node.flags & 8388608 /* Ambient */) &&
-                            (!ts.isVariableStatement(node) ||
-                                !!(ts.getCombinedNodeFlags(node.declarationList) & 3 /* BlockScoped */) ||
-                                node.declarationList.declarations.some(function (d) { return !!d.initializer; }));
-                        eachUnreachableRange(node, function (start, end) { return errorOrSuggestionOnRange(isError_1, start, end, ts.Diagnostics.Unreachable_code_detected); });
-                    }
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Containing_file_is_not_specified_and_root_directory_cannot_be_determined_skipping_lookup_in_node_modules_folder);
                 }
             }
-            return true;
-        }
-    }
-    function eachUnreachableRange(node, cb) {
-        if (ts.isStatement(node) && isExecutableStatement(node) && ts.isBlock(node.parent)) {
-            var statements = node.parent.statements;
-            var slice_1 = ts.sliceAfter(statements, node);
-            ts.getRangesWhere(slice_1, isExecutableStatement, function (start, afterEnd) { return cb(slice_1[start], slice_1[afterEnd - 1]); });
-        }
-        else {
-            cb(node, node);
         }
     }
-    // As opposed to a pure declaration like an `interface`
-    function isExecutableStatement(s) {
-        // Don't remove statements that can validly be used before they appear.
-        return !ts.isFunctionDeclaration(s) && !isPurelyTypeDeclaration(s) && !ts.isEnumDeclaration(s) &&
-            // `var x;` may declare a variable used above
-            !(ts.isVariableStatement(s) && !(ts.getCombinedNodeFlags(s) & (1 /* Let */ | 2 /* Const */)) && s.declarationList.declarations.some(function (d) { return !d.initializer; }));
-    }
-    function isPurelyTypeDeclaration(s) {
-        switch (s.kind) {
-            case 246 /* InterfaceDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-                return true;
-            case 249 /* ModuleDeclaration */:
-                return getModuleInstanceState(s) !== 1 /* Instantiated */;
-            case 248 /* EnumDeclaration */:
-                return ts.hasModifier(s, 2048 /* Const */);
-            default:
-                return false;
+    ts.resolveTypeReferenceDirective = resolveTypeReferenceDirective;
+    /**
+     * Given a set of options, returns the set of type directive names
+     *   that should be included for this program automatically.
+     * This list could either come from the config file,
+     *   or from enumerating the types root + initial secondary types lookup location.
+     * More type directives might appear in the program later as a result of loading actual source files;
+     *   this list is only the set of defaults that are implicitly included.
+     */
+    function getAutomaticTypeDirectiveNames(options, host) {
+        // Use explicit type list from tsconfig.json
+        if (options.types) {
+            return options.types;
         }
-    }
-    function isExportsOrModuleExportsOrAlias(sourceFile, node) {
-        var i = 0;
-        var q = [node];
-        while (q.length && i < 100) {
-            i++;
-            node = q.shift();
-            if (ts.isExportsIdentifier(node) || ts.isModuleExportsAccessExpression(node)) {
-                return true;
-            }
-            else if (ts.isIdentifier(node)) {
-                var symbol = lookupSymbolForNameWorker(sourceFile, node.escapedText);
-                if (!!symbol && !!symbol.valueDeclaration && ts.isVariableDeclaration(symbol.valueDeclaration) && !!symbol.valueDeclaration.initializer) {
-                    var init = symbol.valueDeclaration.initializer;
-                    q.push(init);
-                    if (ts.isAssignmentExpression(init, /*excludeCompoundAssignment*/ true)) {
-                        q.push(init.left);
-                        q.push(init.right);
+        // Walk the primary type lookup locations
+        var result = [];
+        if (host.directoryExists && host.getDirectories) {
+            var typeRoots = getEffectiveTypeRoots(options, host);
+            if (typeRoots) {
+                for (var _i = 0, typeRoots_1 = typeRoots; _i < typeRoots_1.length; _i++) {
+                    var root = typeRoots_1[_i];
+                    if (host.directoryExists(root)) {
+                        for (var _a = 0, _b = host.getDirectories(root); _a < _b.length; _a++) {
+                            var typeDirectivePath = _b[_a];
+                            var normalized = ts.normalizePath(typeDirectivePath);
+                            var packageJsonPath = ts.combinePaths(root, normalized, "package.json");
+                            // `types-publisher` sometimes creates packages with `"typings": null` for packages that don't provide their own types.
+                            // See `createNotNeededPackageJSON` in the types-publisher` repo.
+                            // eslint-disable-next-line no-null/no-null
+                            var isNotNeededPackage = host.fileExists(packageJsonPath) && ts.readJson(packageJsonPath, host).typings === null;
+                            if (!isNotNeededPackage) {
+                                var baseFileName = ts.getBaseFileName(normalized);
+                                // At this stage, skip results with leading dot.
+                                if (baseFileName.charCodeAt(0) !== 46 /* dot */) {
+                                    // Return just the type directive names
+                                    result.push(baseFileName);
+                                }
+                            }
+                        }
                     }
                 }
             }
         }
-        return false;
-    }
-    ts.isExportsOrModuleExportsOrAlias = isExportsOrModuleExportsOrAlias;
-    function lookupSymbolForNameWorker(container, name) {
-        var local = container.locals && container.locals.get(name);
-        if (local) {
-            return local.exportSymbol || local;
-        }
-        if (ts.isSourceFile(container) && container.jsGlobalAugmentations && container.jsGlobalAugmentations.has(name)) {
-            return container.jsGlobalAugmentations.get(name);
-        }
-        return container.symbol && container.symbol.exports && container.symbol.exports.get(name);
+        return result;
     }
-    /**
-     * Computes the transform flags for a node, given the transform flags of its subtree
-     *
-     * @param node The node to analyze
-     * @param subtreeFlags Transform flags computed for this node's subtree
-     */
-    function computeTransformFlagsForNode(node, subtreeFlags) {
-        var kind = node.kind;
-        switch (kind) {
-            case 196 /* CallExpression */:
-                return computeCallExpression(node, subtreeFlags);
-            case 197 /* NewExpression */:
-                return computeNewExpression(node, subtreeFlags);
-            case 249 /* ModuleDeclaration */:
-                return computeModuleDeclaration(node, subtreeFlags);
-            case 200 /* ParenthesizedExpression */:
-                return computeParenthesizedExpression(node, subtreeFlags);
-            case 209 /* BinaryExpression */:
-                return computeBinaryExpression(node, subtreeFlags);
-            case 226 /* ExpressionStatement */:
-                return computeExpressionStatement(node, subtreeFlags);
-            case 156 /* Parameter */:
-                return computeParameter(node, subtreeFlags);
-            case 202 /* ArrowFunction */:
-                return computeArrowFunction(node, subtreeFlags);
-            case 201 /* FunctionExpression */:
-                return computeFunctionExpression(node, subtreeFlags);
-            case 244 /* FunctionDeclaration */:
-                return computeFunctionDeclaration(node, subtreeFlags);
-            case 242 /* VariableDeclaration */:
-                return computeVariableDeclaration(node, subtreeFlags);
-            case 243 /* VariableDeclarationList */:
-                return computeVariableDeclarationList(node, subtreeFlags);
-            case 225 /* VariableStatement */:
-                return computeVariableStatement(node, subtreeFlags);
-            case 238 /* LabeledStatement */:
-                return computeLabeledStatement(node, subtreeFlags);
-            case 245 /* ClassDeclaration */:
-                return computeClassDeclaration(node, subtreeFlags);
-            case 214 /* ClassExpression */:
-                return computeClassExpression(node, subtreeFlags);
-            case 279 /* HeritageClause */:
-                return computeHeritageClause(node, subtreeFlags);
-            case 280 /* CatchClause */:
-                return computeCatchClause(node, subtreeFlags);
-            case 216 /* ExpressionWithTypeArguments */:
-                return computeExpressionWithTypeArguments(node, subtreeFlags);
-            case 162 /* Constructor */:
-                return computeConstructor(node, subtreeFlags);
-            case 159 /* PropertyDeclaration */:
-                return computePropertyDeclaration(node, subtreeFlags);
-            case 161 /* MethodDeclaration */:
-                return computeMethod(node, subtreeFlags);
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-                return computeAccessor(node, subtreeFlags);
-            case 253 /* ImportEqualsDeclaration */:
-                return computeImportEquals(node, subtreeFlags);
-            case 194 /* PropertyAccessExpression */:
-                return computePropertyAccess(node, subtreeFlags);
-            case 195 /* ElementAccessExpression */:
-                return computeElementAccess(node, subtreeFlags);
-            case 267 /* JsxSelfClosingElement */:
-            case 268 /* JsxOpeningElement */:
-                return computeJsxOpeningLikeElement(node, subtreeFlags);
-            default:
-                return computeOther(node, kind, subtreeFlags);
-        }
+    ts.getAutomaticTypeDirectiveNames = getAutomaticTypeDirectiveNames;
+    function createModuleResolutionCache(currentDirectory, getCanonicalFileName, options) {
+        return createModuleResolutionCacheWithMaps(createCacheWithRedirects(options), createCacheWithRedirects(options), currentDirectory, getCanonicalFileName);
     }
-    ts.computeTransformFlagsForNode = computeTransformFlagsForNode;
-    function computeCallExpression(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        var callee = ts.skipOuterExpressions(node.expression);
-        var expression = node.expression;
-        if (node.flags & 32 /* OptionalChain */) {
-            transformFlags |= 8 /* ContainsES2020 */;
-        }
-        if (node.typeArguments) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        if (subtreeFlags & 8192 /* ContainsRestOrSpread */ || ts.isSuperOrSuperProperty(callee)) {
-            // If the this node contains a SpreadExpression, or is a super call, then it is an ES6
-            // node.
-            transformFlags |= 256 /* AssertES2015 */;
-            if (ts.isSuperProperty(callee)) {
-                transformFlags |= 4096 /* ContainsLexicalThis */;
-            }
-        }
-        if (expression.kind === 96 /* ImportKeyword */) {
-            transformFlags |= 2097152 /* ContainsDynamicImport */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536879104 /* ArrayLiteralOrCallOrNewExcludes */;
-    }
-    function computeNewExpression(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        if (node.typeArguments) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        if (subtreeFlags & 8192 /* ContainsRestOrSpread */) {
-            // If the this node contains a SpreadElementExpression then it is an ES6
-            // node.
-            transformFlags |= 256 /* AssertES2015 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536879104 /* ArrayLiteralOrCallOrNewExcludes */;
-    }
-    function computeJsxOpeningLikeElement(node, subtreeFlags) {
-        var transformFlags = subtreeFlags | 2 /* AssertJsx */;
-        if (node.typeArguments) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeBinaryExpression(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        var operatorTokenKind = node.operatorToken.kind;
-        var leftKind = node.left.kind;
-        if (operatorTokenKind === 60 /* QuestionQuestionToken */) {
-            transformFlags |= 8 /* AssertES2020 */;
-        }
-        else if (operatorTokenKind === 62 /* EqualsToken */ && leftKind === 193 /* ObjectLiteralExpression */) {
-            // Destructuring object assignments with are ES2015 syntax
-            // and possibly ES2018 if they contain rest
-            transformFlags |= 32 /* AssertES2018 */ | 256 /* AssertES2015 */ | 1024 /* AssertDestructuringAssignment */;
-        }
-        else if (operatorTokenKind === 62 /* EqualsToken */ && leftKind === 192 /* ArrayLiteralExpression */) {
-            // Destructuring assignments are ES2015 syntax.
-            transformFlags |= 256 /* AssertES2015 */ | 1024 /* AssertDestructuringAssignment */;
-        }
-        else if (operatorTokenKind === 42 /* AsteriskAsteriskToken */
-            || operatorTokenKind === 66 /* AsteriskAsteriskEqualsToken */) {
-            // Exponentiation is ES2016 syntax.
-            transformFlags |= 128 /* AssertES2016 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeParameter(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        var name = node.name;
-        var initializer = node.initializer;
-        var dotDotDotToken = node.dotDotDotToken;
-        // The '?' token, type annotations, decorators, and 'this' parameters are TypeSCript
-        // syntax.
-        if (node.questionToken
-            || node.type
-            || (subtreeFlags & 2048 /* ContainsTypeScriptClassSyntax */ && ts.some(node.decorators))
-            || ts.isThisIdentifier(name)) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // If a parameter has an accessibility modifier, then it is TypeScript syntax.
-        if (ts.hasModifier(node, 92 /* ParameterPropertyModifier */)) {
-            transformFlags |= 1 /* AssertTypeScript */ | 2048 /* ContainsTypeScriptClassSyntax */;
-        }
-        // parameters with object rest destructuring are ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        // If a parameter has an initializer, a binding pattern or a dotDotDot token, then
-        // it is ES6 syntax and its container must emit default value assignments or parameter destructuring downlevel.
-        if (subtreeFlags & 131072 /* ContainsBindingPattern */ || initializer || dotDotDotToken) {
-            transformFlags |= 256 /* AssertES2015 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* ParameterExcludes */;
-    }
-    function computeParenthesizedExpression(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        var expression = node.expression;
-        var expressionKind = expression.kind;
-        // If the node is synthesized, it means the emitter put the parentheses there,
-        // not the user. If we didn't want them, the emitter would not have put them
-        // there.
-        if (expressionKind === 217 /* AsExpression */
-            || expressionKind === 199 /* TypeAssertionExpression */) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* OuterExpressionExcludes */;
-    }
-    function computeClassDeclaration(node, subtreeFlags) {
-        var transformFlags;
-        if (ts.hasModifier(node, 2 /* Ambient */)) {
-            // An ambient declaration is TypeScript syntax.
-            transformFlags = 1 /* AssertTypeScript */;
-        }
-        else {
-            // A ClassDeclaration is ES6 syntax.
-            transformFlags = subtreeFlags | 256 /* AssertES2015 */;
-            // A class with a parameter property assignment or decorator is TypeScript syntax.
-            // An exported declaration may be TypeScript syntax, but is handled by the visitor
-            // for a namespace declaration.
-            if ((subtreeFlags & 2048 /* ContainsTypeScriptClassSyntax */)
-                || node.typeParameters) {
-                transformFlags |= 1 /* AssertTypeScript */;
-            }
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536905728 /* ClassExcludes */;
-    }
-    function computeClassExpression(node, subtreeFlags) {
-        // A ClassExpression is ES6 syntax.
-        var transformFlags = subtreeFlags | 256 /* AssertES2015 */;
-        // A class with a parameter property assignment or decorator is TypeScript syntax.
-        if (subtreeFlags & 2048 /* ContainsTypeScriptClassSyntax */
-            || node.typeParameters) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536905728 /* ClassExcludes */;
-    }
-    function computeHeritageClause(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        switch (node.token) {
-            case 90 /* ExtendsKeyword */:
-                // An `extends` HeritageClause is ES6 syntax.
-                transformFlags |= 256 /* AssertES2015 */;
-                break;
-            case 113 /* ImplementsKeyword */:
-                // An `implements` HeritageClause is TypeScript syntax.
-                transformFlags |= 1 /* AssertTypeScript */;
-                break;
-            default:
-                ts.Debug.fail("Unexpected token for heritage clause");
-                break;
+    ts.createModuleResolutionCache = createModuleResolutionCache;
+    /*@internal*/
+    function createCacheWithRedirects(options) {
+        var ownMap = new ts.Map();
+        var redirectsMap = new ts.Map();
+        return {
+            ownMap: ownMap,
+            redirectsMap: redirectsMap,
+            getOrCreateMapOfCacheRedirects: getOrCreateMapOfCacheRedirects,
+            clear: clear,
+            setOwnOptions: setOwnOptions,
+            setOwnMap: setOwnMap
+        };
+        function setOwnOptions(newOptions) {
+            options = newOptions;
         }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeCatchClause(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        if (!node.variableDeclaration) {
-            transformFlags |= 16 /* AssertES2019 */;
-        }
-        else if (ts.isBindingPattern(node.variableDeclaration.name)) {
-            transformFlags |= 256 /* AssertES2015 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536887296 /* CatchClauseExcludes */;
-    }
-    function computeExpressionWithTypeArguments(node, subtreeFlags) {
-        // An ExpressionWithTypeArguments is ES6 syntax, as it is used in the
-        // extends clause of a class.
-        var transformFlags = subtreeFlags | 256 /* AssertES2015 */;
-        // If an ExpressionWithTypeArguments contains type arguments, then it
-        // is TypeScript syntax.
-        if (node.typeArguments) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeConstructor(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        // TypeScript-specific modifiers and overloads are TypeScript syntax
-        if (ts.hasModifier(node, 2270 /* TypeScriptModifier */)
-            || !node.body) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // function declarations with object rest destructuring are ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~538923008 /* ConstructorExcludes */;
-    }
-    function computeMethod(node, subtreeFlags) {
-        // A MethodDeclaration is ES6 syntax.
-        var transformFlags = subtreeFlags | 256 /* AssertES2015 */;
-        // Decorators, TypeScript-specific modifiers, type parameters, type annotations, and
-        // overloads are TypeScript syntax.
-        if (node.decorators
-            || ts.hasModifier(node, 2270 /* TypeScriptModifier */)
-            || node.typeParameters
-            || node.type
-            || !node.body
-            || node.questionToken) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // function declarations with object rest destructuring are ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        // An async method declaration is ES2017 syntax.
-        if (ts.hasModifier(node, 256 /* Async */)) {
-            transformFlags |= node.asteriskToken ? 32 /* AssertES2018 */ : 64 /* AssertES2017 */;
-        }
-        if (node.asteriskToken) {
-            transformFlags |= 512 /* AssertGenerator */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return propagatePropertyNameFlags(node.name, transformFlags & ~538923008 /* MethodOrAccessorExcludes */);
-    }
-    function computeAccessor(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        // Decorators, TypeScript-specific modifiers, type annotations, and overloads are
-        // TypeScript syntax.
-        if (node.decorators
-            || ts.hasModifier(node, 2270 /* TypeScriptModifier */)
-            || node.type
-            || !node.body) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // function declarations with object rest destructuring are ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return propagatePropertyNameFlags(node.name, transformFlags & ~538923008 /* MethodOrAccessorExcludes */);
-    }
-    function computePropertyDeclaration(node, subtreeFlags) {
-        var transformFlags = subtreeFlags | 4194304 /* ContainsClassFields */;
-        // Decorators, TypeScript-specific modifiers, and type annotations are TypeScript syntax.
-        if (ts.some(node.decorators) || ts.hasModifier(node, 2270 /* TypeScriptModifier */) || node.type || node.questionToken || node.exclamationToken) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // Hoisted variables related to class properties should live within the TypeScript class wrapper.
-        if (ts.isComputedPropertyName(node.name) || (ts.hasStaticModifier(node) && node.initializer)) {
-            transformFlags |= 2048 /* ContainsTypeScriptClassSyntax */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return propagatePropertyNameFlags(node.name, transformFlags & ~536875008 /* PropertyExcludes */);
-    }
-    function computeFunctionDeclaration(node, subtreeFlags) {
-        var transformFlags;
-        var modifierFlags = ts.getModifierFlags(node);
-        var body = node.body;
-        if (!body || (modifierFlags & 2 /* Ambient */)) {
-            // An ambient declaration is TypeScript syntax.
-            // A FunctionDeclaration without a body is an overload and is TypeScript syntax.
-            transformFlags = 1 /* AssertTypeScript */;
+        function setOwnMap(newOwnMap) {
+            ownMap = newOwnMap;
         }
-        else {
-            transformFlags = subtreeFlags | 1048576 /* ContainsHoistedDeclarationOrCompletion */;
-            // TypeScript-specific modifiers, type parameters, and type annotations are TypeScript
-            // syntax.
-            if (modifierFlags & 2270 /* TypeScriptModifier */
-                || node.typeParameters
-                || node.type) {
-                transformFlags |= 1 /* AssertTypeScript */;
-            }
-            // An async function declaration is ES2017 syntax.
-            if (modifierFlags & 256 /* Async */) {
-                transformFlags |= node.asteriskToken ? 32 /* AssertES2018 */ : 64 /* AssertES2017 */;
-            }
-            // function declarations with object rest destructuring are ES2018 syntax
-            if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-                transformFlags |= 32 /* AssertES2018 */;
-            }
-            // If a FunctionDeclaration is generator function and is the body of a
-            // transformed async function, then this node can be transformed to a
-            // down-level generator.
-            // Currently we do not support transforming any other generator functions
-            // down level.
-            if (node.asteriskToken) {
-                transformFlags |= 512 /* AssertGenerator */;
-            }
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~538925056 /* FunctionExcludes */;
-    }
-    function computeFunctionExpression(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        // TypeScript-specific modifiers, type parameters, and type annotations are TypeScript
-        // syntax.
-        if (ts.hasModifier(node, 2270 /* TypeScriptModifier */)
-            || node.typeParameters
-            || node.type) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // An async function expression is ES2017 syntax.
-        if (ts.hasModifier(node, 256 /* Async */)) {
-            transformFlags |= node.asteriskToken ? 32 /* AssertES2018 */ : 64 /* AssertES2017 */;
-        }
-        // function expressions with object rest destructuring are ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        // If a FunctionExpression is generator function and is the body of a
-        // transformed async function, then this node can be transformed to a
-        // down-level generator.
-        if (node.asteriskToken) {
-            transformFlags |= 512 /* AssertGenerator */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~538925056 /* FunctionExcludes */;
-    }
-    function computeArrowFunction(node, subtreeFlags) {
-        // An ArrowFunction is ES6 syntax, and excludes markers that should not escape the scope of an ArrowFunction.
-        var transformFlags = subtreeFlags | 256 /* AssertES2015 */;
-        // TypeScript-specific modifiers, type parameters, and type annotations are TypeScript
-        // syntax.
-        if (ts.hasModifier(node, 2270 /* TypeScriptModifier */)
-            || node.typeParameters
-            || node.type) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        // An async arrow function is ES2017 syntax.
-        if (ts.hasModifier(node, 256 /* Async */)) {
-            transformFlags |= 64 /* AssertES2017 */;
-        }
-        // arrow functions with object rest destructuring are ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~538920960 /* ArrowFunctionExcludes */;
-    }
-    function computePropertyAccess(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        if (node.flags & 32 /* OptionalChain */) {
-            transformFlags |= 8 /* ContainsES2020 */;
-        }
-        // If a PropertyAccessExpression starts with a super keyword, then it is
-        // ES6 syntax, and requires a lexical `this` binding.
-        if (node.expression.kind === 102 /* SuperKeyword */) {
-            // super inside of an async function requires hoisting the super access (ES2017).
-            // same for super inside of an async generator, which is ES2018.
-            transformFlags |= 64 /* ContainsES2017 */ | 32 /* ContainsES2018 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* PropertyAccessExcludes */;
-    }
-    function computeElementAccess(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        if (node.flags & 32 /* OptionalChain */) {
-            transformFlags |= 8 /* ContainsES2020 */;
-        }
-        // If an ElementAccessExpression starts with a super keyword, then it is
-        // ES6 syntax, and requires a lexical `this` binding.
-        if (node.expression.kind === 102 /* SuperKeyword */) {
-            // super inside of an async function requires hoisting the super access (ES2017).
-            // same for super inside of an async generator, which is ES2018.
-            transformFlags |= 64 /* ContainsES2017 */ | 32 /* ContainsES2018 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* PropertyAccessExcludes */;
-    }
-    function computeVariableDeclaration(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        transformFlags |= 256 /* AssertES2015 */ | 131072 /* ContainsBindingPattern */; // TODO(rbuckton): Why are these set unconditionally?
-        // A VariableDeclaration containing ObjectRest is ES2018 syntax
-        if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-            transformFlags |= 32 /* AssertES2018 */;
-        }
-        // Type annotations are TypeScript syntax.
-        if (node.type || node.exclamationToken) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeVariableStatement(node, subtreeFlags) {
-        var transformFlags;
-        var declarationListTransformFlags = node.declarationList.transformFlags;
-        // An ambient declaration is TypeScript syntax.
-        if (ts.hasModifier(node, 2 /* Ambient */)) {
-            transformFlags = 1 /* AssertTypeScript */;
+        function getOrCreateMapOfCacheRedirects(redirectedReference) {
+            if (!redirectedReference) {
+                return ownMap;
+            }
+            var path = redirectedReference.sourceFile.path;
+            var redirects = redirectsMap.get(path);
+            if (!redirects) {
+                // Reuse map if redirected reference map uses same resolution
+                redirects = !options || ts.optionsHaveModuleResolutionChanges(options, redirectedReference.commandLine.options) ? new ts.Map() : ownMap;
+                redirectsMap.set(path, redirects);
+            }
+            return redirects;
         }
-        else {
-            transformFlags = subtreeFlags;
-            if (declarationListTransformFlags & 131072 /* ContainsBindingPattern */) {
-                transformFlags |= 256 /* AssertES2015 */;
-            }
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeLabeledStatement(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        // A labeled statement containing a block scoped binding *may* need to be transformed from ES6.
-        if (subtreeFlags & 65536 /* ContainsBlockScopedBinding */
-            && ts.isIterationStatement(node, /*lookInLabeledStatements*/ true)) {
-            transformFlags |= 256 /* AssertES2015 */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeImportEquals(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        // An ImportEqualsDeclaration with a namespace reference is TypeScript.
-        if (!ts.isExternalModuleImportEqualsDeclaration(node)) {
-            transformFlags |= 1 /* AssertTypeScript */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeExpressionStatement(node, subtreeFlags) {
-        var transformFlags = subtreeFlags;
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~536870912 /* NodeExcludes */;
-    }
-    function computeModuleDeclaration(node, subtreeFlags) {
-        var transformFlags = 1 /* AssertTypeScript */;
-        var modifierFlags = ts.getModifierFlags(node);
-        if ((modifierFlags & 2 /* Ambient */) === 0) {
-            transformFlags |= subtreeFlags;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~537991168 /* ModuleExcludes */;
-    }
-    function computeVariableDeclarationList(node, subtreeFlags) {
-        var transformFlags = subtreeFlags | 1048576 /* ContainsHoistedDeclarationOrCompletion */;
-        if (subtreeFlags & 131072 /* ContainsBindingPattern */) {
-            transformFlags |= 256 /* AssertES2015 */;
-        }
-        // If a VariableDeclarationList is `let` or `const`, then it is ES6 syntax.
-        if (node.flags & 3 /* BlockScoped */) {
-            transformFlags |= 256 /* AssertES2015 */ | 65536 /* ContainsBlockScopedBinding */;
-        }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~537018368 /* VariableDeclarationListExcludes */;
-    }
-    function computeOther(node, kind, subtreeFlags) {
-        // Mark transformations needed for each node
-        var transformFlags = subtreeFlags;
-        var excludeFlags = 536870912 /* NodeExcludes */;
-        switch (kind) {
-            case 126 /* AsyncKeyword */:
-                // async is ES2017 syntax, but may be ES2018 syntax (for async generators)
-                transformFlags |= 32 /* AssertES2018 */ | 64 /* AssertES2017 */;
-                break;
-            case 206 /* AwaitExpression */:
-                // await is ES2017 syntax, but may be ES2018 syntax (for async generators)
-                transformFlags |= 32 /* AssertES2018 */ | 64 /* AssertES2017 */ | 524288 /* ContainsAwait */;
-                break;
-            case 199 /* TypeAssertionExpression */:
-            case 217 /* AsExpression */:
-            case 326 /* PartiallyEmittedExpression */:
-                // These nodes are TypeScript syntax.
-                transformFlags |= 1 /* AssertTypeScript */;
-                excludeFlags = 536870912 /* OuterExpressionExcludes */;
-                break;
-            case 119 /* PublicKeyword */:
-            case 117 /* PrivateKeyword */:
-            case 118 /* ProtectedKeyword */:
-            case 122 /* AbstractKeyword */:
-            case 130 /* DeclareKeyword */:
-            case 81 /* ConstKeyword */:
-            case 248 /* EnumDeclaration */:
-            case 284 /* EnumMember */:
-            case 218 /* NonNullExpression */:
-            case 138 /* ReadonlyKeyword */:
-                // These nodes are TypeScript syntax.
-                transformFlags |= 1 /* AssertTypeScript */;
-                break;
-            case 266 /* JsxElement */:
-            case 11 /* JsxText */:
-            case 269 /* JsxClosingElement */:
-            case 270 /* JsxFragment */:
-            case 271 /* JsxOpeningFragment */:
-            case 272 /* JsxClosingFragment */:
-            case 273 /* JsxAttribute */:
-            case 274 /* JsxAttributes */:
-            case 275 /* JsxSpreadAttribute */:
-            case 276 /* JsxExpression */:
-                // These nodes are Jsx syntax.
-                transformFlags |= 2 /* AssertJsx */;
-                break;
-            case 14 /* NoSubstitutionTemplateLiteral */:
-            case 15 /* TemplateHead */:
-            case 16 /* TemplateMiddle */:
-            case 17 /* TemplateTail */:
-                if (node.templateFlags) {
-                    transformFlags |= 32 /* AssertES2018 */;
-                    break;
-                }
-            // falls through
-            case 198 /* TaggedTemplateExpression */:
-                if (ts.hasInvalidEscape(node.template)) {
-                    transformFlags |= 32 /* AssertES2018 */;
-                    break;
-                }
-            // falls through
-            case 211 /* TemplateExpression */:
-            case 282 /* ShorthandPropertyAssignment */:
-            case 120 /* StaticKeyword */:
-            case 219 /* MetaProperty */:
-                // These nodes are ES6 syntax.
-                transformFlags |= 256 /* AssertES2015 */;
-                break;
-            case 10 /* StringLiteral */:
-                if (node.hasExtendedUnicodeEscape) {
-                    transformFlags |= 256 /* AssertES2015 */;
-                }
-                break;
-            case 8 /* NumericLiteral */:
-                if (node.numericLiteralFlags & 384 /* BinaryOrOctalSpecifier */) {
-                    transformFlags |= 256 /* AssertES2015 */;
-                }
-                break;
-            case 9 /* BigIntLiteral */:
-                transformFlags |= 4 /* AssertESNext */;
-                break;
-            case 232 /* ForOfStatement */:
-                // This node is either ES2015 syntax or ES2017 syntax (if it is a for-await-of).
-                if (node.awaitModifier) {
-                    transformFlags |= 32 /* AssertES2018 */;
-                }
-                transformFlags |= 256 /* AssertES2015 */;
-                break;
-            case 212 /* YieldExpression */:
-                // This node is either ES2015 syntax (in a generator) or ES2017 syntax (in an async
-                // generator).
-                transformFlags |= 32 /* AssertES2018 */ | 256 /* AssertES2015 */ | 262144 /* ContainsYield */;
-                break;
-            case 125 /* AnyKeyword */:
-            case 140 /* NumberKeyword */:
-            case 151 /* BigIntKeyword */:
-            case 137 /* NeverKeyword */:
-            case 141 /* ObjectKeyword */:
-            case 143 /* StringKeyword */:
-            case 128 /* BooleanKeyword */:
-            case 144 /* SymbolKeyword */:
-            case 110 /* VoidKeyword */:
-            case 155 /* TypeParameter */:
-            case 158 /* PropertySignature */:
-            case 160 /* MethodSignature */:
-            case 165 /* CallSignature */:
-            case 166 /* ConstructSignature */:
-            case 167 /* IndexSignature */:
-            case 168 /* TypePredicate */:
-            case 169 /* TypeReference */:
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
-            case 172 /* TypeQuery */:
-            case 173 /* TypeLiteral */:
-            case 174 /* ArrayType */:
-            case 175 /* TupleType */:
-            case 176 /* OptionalType */:
-            case 177 /* RestType */:
-            case 178 /* UnionType */:
-            case 179 /* IntersectionType */:
-            case 180 /* ConditionalType */:
-            case 181 /* InferType */:
-            case 182 /* ParenthesizedType */:
-            case 246 /* InterfaceDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-            case 183 /* ThisType */:
-            case 184 /* TypeOperator */:
-            case 185 /* IndexedAccessType */:
-            case 186 /* MappedType */:
-            case 187 /* LiteralType */:
-            case 252 /* NamespaceExportDeclaration */:
-                // Types and signatures are TypeScript syntax, and exclude all other facts.
-                transformFlags = 1 /* AssertTypeScript */;
-                excludeFlags = -2 /* TypeExcludes */;
-                break;
-            case 154 /* ComputedPropertyName */:
-                // Even though computed property names are ES6, we don't treat them as such.
-                // This is so that they can flow through PropertyName transforms unaffected.
-                // Instead, we mark the container as ES6, so that it can properly handle the transform.
-                transformFlags |= 32768 /* ContainsComputedPropertyName */;
-                break;
-            case 213 /* SpreadElement */:
-                transformFlags |= 256 /* AssertES2015 */ | 8192 /* ContainsRestOrSpread */;
-                break;
-            case 283 /* SpreadAssignment */:
-                transformFlags |= 32 /* AssertES2018 */ | 16384 /* ContainsObjectRestOrSpread */;
-                break;
-            case 102 /* SuperKeyword */:
-                // This node is ES6 syntax.
-                transformFlags |= 256 /* AssertES2015 */;
-                excludeFlags = 536870912 /* OuterExpressionExcludes */; // must be set to persist `Super`
-                break;
-            case 104 /* ThisKeyword */:
-                // Mark this node and its ancestors as containing a lexical `this` keyword.
-                transformFlags |= 4096 /* ContainsLexicalThis */;
-                break;
-            case 189 /* ObjectBindingPattern */:
-                transformFlags |= 256 /* AssertES2015 */ | 131072 /* ContainsBindingPattern */;
-                if (subtreeFlags & 8192 /* ContainsRestOrSpread */) {
-                    transformFlags |= 32 /* AssertES2018 */ | 16384 /* ContainsObjectRestOrSpread */;
-                }
-                excludeFlags = 536879104 /* BindingPatternExcludes */;
-                break;
-            case 190 /* ArrayBindingPattern */:
-                transformFlags |= 256 /* AssertES2015 */ | 131072 /* ContainsBindingPattern */;
-                excludeFlags = 536879104 /* BindingPatternExcludes */;
-                break;
-            case 191 /* BindingElement */:
-                transformFlags |= 256 /* AssertES2015 */;
-                if (node.dotDotDotToken) {
-                    transformFlags |= 8192 /* ContainsRestOrSpread */;
-                }
-                break;
-            case 157 /* Decorator */:
-                // This node is TypeScript syntax, and marks its container as also being TypeScript syntax.
-                transformFlags |= 1 /* AssertTypeScript */ | 2048 /* ContainsTypeScriptClassSyntax */;
-                break;
-            case 193 /* ObjectLiteralExpression */:
-                excludeFlags = 536922112 /* ObjectLiteralExcludes */;
-                if (subtreeFlags & 32768 /* ContainsComputedPropertyName */) {
-                    // If an ObjectLiteralExpression contains a ComputedPropertyName, then it
-                    // is an ES6 node.
-                    transformFlags |= 256 /* AssertES2015 */;
-                }
-                if (subtreeFlags & 16384 /* ContainsObjectRestOrSpread */) {
-                    // If an ObjectLiteralExpression contains a spread element, then it
-                    // is an ES2018 node.
-                    transformFlags |= 32 /* AssertES2018 */;
-                }
-                break;
-            case 192 /* ArrayLiteralExpression */:
-                excludeFlags = 536879104 /* ArrayLiteralOrCallOrNewExcludes */;
-                break;
-            case 228 /* DoStatement */:
-            case 229 /* WhileStatement */:
-            case 230 /* ForStatement */:
-            case 231 /* ForInStatement */:
-                // A loop containing a block scoped binding *may* need to be transformed from ES6.
-                if (subtreeFlags & 65536 /* ContainsBlockScopedBinding */) {
-                    transformFlags |= 256 /* AssertES2015 */;
-                }
-                break;
-            case 290 /* SourceFile */:
-                break;
-            case 262 /* NamespaceExport */:
-                transformFlags |= 4 /* AssertESNext */;
-                break;
-            case 235 /* ReturnStatement */:
-                // Return statements may require an `await` in ES2018.
-                transformFlags |= 1048576 /* ContainsHoistedDeclarationOrCompletion */ | 32 /* AssertES2018 */;
-                break;
-            case 233 /* ContinueStatement */:
-            case 234 /* BreakStatement */:
-                transformFlags |= 1048576 /* ContainsHoistedDeclarationOrCompletion */;
-                break;
-            case 76 /* PrivateIdentifier */:
-                transformFlags |= 4194304 /* ContainsClassFields */;
-                break;
+        function clear() {
+            ownMap.clear();
+            redirectsMap.clear();
         }
-        node.transformFlags = transformFlags | 536870912 /* HasComputedFlags */;
-        return transformFlags & ~excludeFlags;
-    }
-    function propagatePropertyNameFlags(node, transformFlags) {
-        return transformFlags | (node.transformFlags & 4096 /* PropertyNamePropagatingFlags */);
     }
-    /**
-     * Gets the transform flags to exclude when unioning the transform flags of a subtree.
-     *
-     * NOTE: This needs to be kept up-to-date with the exclusions used in `computeTransformFlagsForNode`.
-     *       For performance reasons, `computeTransformFlagsForNode` uses local constant values rather
-     *       than calling this function.
-     */
-    function getTransformFlagsSubtreeExclusions(kind) {
-        if (kind >= 168 /* FirstTypeNode */ && kind <= 188 /* LastTypeNode */) {
-            return -2 /* TypeExcludes */;
+    ts.createCacheWithRedirects = createCacheWithRedirects;
+    /*@internal*/
+    function createModuleResolutionCacheWithMaps(directoryToModuleNameMap, moduleNameToDirectoryMap, currentDirectory, getCanonicalFileName) {
+        return { getOrCreateCacheForDirectory: getOrCreateCacheForDirectory, getOrCreateCacheForModuleName: getOrCreateCacheForModuleName, directoryToModuleNameMap: directoryToModuleNameMap, moduleNameToDirectoryMap: moduleNameToDirectoryMap };
+        function getOrCreateCacheForDirectory(directoryName, redirectedReference) {
+            var path = ts.toPath(directoryName, currentDirectory, getCanonicalFileName);
+            return getOrCreateCache(directoryToModuleNameMap, redirectedReference, path, function () { return new ts.Map(); });
         }
-        switch (kind) {
-            case 196 /* CallExpression */:
-            case 197 /* NewExpression */:
-            case 192 /* ArrayLiteralExpression */:
-                return 536879104 /* ArrayLiteralOrCallOrNewExcludes */;
-            case 249 /* ModuleDeclaration */:
-                return 537991168 /* ModuleExcludes */;
-            case 156 /* Parameter */:
-                return 536870912 /* ParameterExcludes */;
-            case 202 /* ArrowFunction */:
-                return 538920960 /* ArrowFunctionExcludes */;
-            case 201 /* FunctionExpression */:
-            case 244 /* FunctionDeclaration */:
-                return 538925056 /* FunctionExcludes */;
-            case 243 /* VariableDeclarationList */:
-                return 537018368 /* VariableDeclarationListExcludes */;
-            case 245 /* ClassDeclaration */:
-            case 214 /* ClassExpression */:
-                return 536905728 /* ClassExcludes */;
-            case 162 /* Constructor */:
-                return 538923008 /* ConstructorExcludes */;
-            case 161 /* MethodDeclaration */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-                return 538923008 /* MethodOrAccessorExcludes */;
-            case 125 /* AnyKeyword */:
-            case 140 /* NumberKeyword */:
-            case 151 /* BigIntKeyword */:
-            case 137 /* NeverKeyword */:
-            case 143 /* StringKeyword */:
-            case 141 /* ObjectKeyword */:
-            case 128 /* BooleanKeyword */:
-            case 144 /* SymbolKeyword */:
-            case 110 /* VoidKeyword */:
-            case 155 /* TypeParameter */:
-            case 158 /* PropertySignature */:
-            case 160 /* MethodSignature */:
-            case 165 /* CallSignature */:
-            case 166 /* ConstructSignature */:
-            case 167 /* IndexSignature */:
-            case 246 /* InterfaceDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-                return -2 /* TypeExcludes */;
-            case 193 /* ObjectLiteralExpression */:
-                return 536922112 /* ObjectLiteralExcludes */;
-            case 280 /* CatchClause */:
-                return 536887296 /* CatchClauseExcludes */;
-            case 189 /* ObjectBindingPattern */:
-            case 190 /* ArrayBindingPattern */:
-                return 536879104 /* BindingPatternExcludes */;
-            case 199 /* TypeAssertionExpression */:
-            case 217 /* AsExpression */:
-            case 326 /* PartiallyEmittedExpression */:
-            case 200 /* ParenthesizedExpression */:
-            case 102 /* SuperKeyword */:
-                return 536870912 /* OuterExpressionExcludes */;
-            case 194 /* PropertyAccessExpression */:
-            case 195 /* ElementAccessExpression */:
-                return 536870912 /* PropertyAccessExcludes */;
-            default:
-                return 536870912 /* NodeExcludes */;
+        function getOrCreateCacheForModuleName(nonRelativeModuleName, redirectedReference) {
+            ts.Debug.assert(!ts.isExternalModuleNameRelative(nonRelativeModuleName));
+            return getOrCreateCache(moduleNameToDirectoryMap, redirectedReference, nonRelativeModuleName, createPerModuleNameCache);
         }
-    }
-    ts.getTransformFlagsSubtreeExclusions = getTransformFlagsSubtreeExclusions;
-    /**
-     * "Binds" JSDoc nodes in TypeScript code.
-     * Since we will never create symbols for JSDoc, we just set parent pointers instead.
-     */
-    function setParentPointers(parent, child) {
-        child.parent = parent;
-        ts.forEachChild(child, function (grandchild) { return setParentPointers(child, grandchild); });
-    }
-})(ts || (ts = {}));
-/** @internal */
-var ts;
-(function (ts) {
-    function createGetSymbolWalker(getRestTypeOfSignature, getTypePredicateOfSignature, getReturnTypeOfSignature, getBaseTypes, resolveStructuredTypeMembers, getTypeOfSymbol, getResolvedSymbol, getIndexTypeOfStructuredType, getConstraintOfTypeParameter, getFirstIdentifier, getTypeArguments) {
-        return getSymbolWalker;
-        function getSymbolWalker(accept) {
-            if (accept === void 0) { accept = function () { return true; }; }
-            var visitedTypes = []; // Sparse array from id to type
-            var visitedSymbols = []; // Sparse array from id to symbol
-            return {
-                walkType: function (type) {
-                    try {
-                        visitType(type);
-                        return { visitedTypes: ts.getOwnValues(visitedTypes), visitedSymbols: ts.getOwnValues(visitedSymbols) };
-                    }
-                    finally {
-                        ts.clear(visitedTypes);
-                        ts.clear(visitedSymbols);
-                    }
-                },
-                walkSymbol: function (symbol) {
-                    try {
-                        visitSymbol(symbol);
-                        return { visitedTypes: ts.getOwnValues(visitedTypes), visitedSymbols: ts.getOwnValues(visitedSymbols) };
-                    }
-                    finally {
-                        ts.clear(visitedTypes);
-                        ts.clear(visitedSymbols);
-                    }
-                },
-            };
-            function visitType(type) {
-                if (!type) {
-                    return;
-                }
-                if (visitedTypes[type.id]) {
-                    return;
-                }
-                visitedTypes[type.id] = type;
-                // Reuse visitSymbol to visit the type's symbol,
-                //  but be sure to bail on recuring into the type if accept declines the symbol.
-                var shouldBail = visitSymbol(type.symbol);
-                if (shouldBail)
-                    return;
-                // Visit the type's related types, if any
-                if (type.flags & 524288 /* Object */) {
-                    var objectType = type;
-                    var objectFlags = objectType.objectFlags;
-                    if (objectFlags & 4 /* Reference */) {
-                        visitTypeReference(type);
-                    }
-                    if (objectFlags & 32 /* Mapped */) {
-                        visitMappedType(type);
-                    }
-                    if (objectFlags & (1 /* Class */ | 2 /* Interface */)) {
-                        visitInterfaceType(type);
-                    }
-                    if (objectFlags & (8 /* Tuple */ | 16 /* Anonymous */)) {
-                        visitObjectType(objectType);
-                    }
-                }
-                if (type.flags & 262144 /* TypeParameter */) {
-                    visitTypeParameter(type);
-                }
-                if (type.flags & 3145728 /* UnionOrIntersection */) {
-                    visitUnionOrIntersectionType(type);
-                }
-                if (type.flags & 4194304 /* Index */) {
-                    visitIndexType(type);
-                }
-                if (type.flags & 8388608 /* IndexedAccess */) {
-                    visitIndexedAccessType(type);
-                }
-            }
-            function visitTypeReference(type) {
-                visitType(type.target);
-                ts.forEach(getTypeArguments(type), visitType);
-            }
-            function visitTypeParameter(type) {
-                visitType(getConstraintOfTypeParameter(type));
-            }
-            function visitUnionOrIntersectionType(type) {
-                ts.forEach(type.types, visitType);
-            }
-            function visitIndexType(type) {
-                visitType(type.type);
-            }
-            function visitIndexedAccessType(type) {
-                visitType(type.objectType);
-                visitType(type.indexType);
-                visitType(type.constraint);
-            }
-            function visitMappedType(type) {
-                visitType(type.typeParameter);
-                visitType(type.constraintType);
-                visitType(type.templateType);
-                visitType(type.modifiersType);
-            }
-            function visitSignature(signature) {
-                var typePredicate = getTypePredicateOfSignature(signature);
-                if (typePredicate) {
-                    visitType(typePredicate.type);
-                }
-                ts.forEach(signature.typeParameters, visitType);
-                for (var _i = 0, _a = signature.parameters; _i < _a.length; _i++) {
-                    var parameter = _a[_i];
-                    visitSymbol(parameter);
-                }
-                visitType(getRestTypeOfSignature(signature));
-                visitType(getReturnTypeOfSignature(signature));
+        function getOrCreateCache(cacheWithRedirects, redirectedReference, key, create) {
+            var cache = cacheWithRedirects.getOrCreateMapOfCacheRedirects(redirectedReference);
+            var result = cache.get(key);
+            if (!result) {
+                result = create();
+                cache.set(key, result);
             }
-            function visitInterfaceType(interfaceT) {
-                visitObjectType(interfaceT);
-                ts.forEach(interfaceT.typeParameters, visitType);
-                ts.forEach(getBaseTypes(interfaceT), visitType);
-                visitType(interfaceT.thisType);
+            return result;
+        }
+        function createPerModuleNameCache() {
+            var directoryPathMap = new ts.Map();
+            return { get: get, set: set };
+            function get(directory) {
+                return directoryPathMap.get(ts.toPath(directory, currentDirectory, getCanonicalFileName));
             }
-            function visitObjectType(type) {
-                var stringIndexType = getIndexTypeOfStructuredType(type, 0 /* String */);
-                visitType(stringIndexType);
-                var numberIndexType = getIndexTypeOfStructuredType(type, 1 /* Number */);
-                visitType(numberIndexType);
-                // The two checks above *should* have already resolved the type (if needed), so this should be cached
-                var resolved = resolveStructuredTypeMembers(type);
-                for (var _i = 0, _a = resolved.callSignatures; _i < _a.length; _i++) {
-                    var signature = _a[_i];
-                    visitSignature(signature);
-                }
-                for (var _b = 0, _c = resolved.constructSignatures; _b < _c.length; _b++) {
-                    var signature = _c[_b];
-                    visitSignature(signature);
+            /**
+             * At first this function add entry directory -> module resolution result to the table.
+             * Then it computes the set of parent folders for 'directory' that should have the same module resolution result
+             * and for every parent folder in set it adds entry: parent -> module resolution. .
+             * Lets say we first directory name: /a/b/c/d/e and resolution result is: /a/b/bar.ts.
+             * Set of parent folders that should have the same result will be:
+             * [
+             *     /a/b/c/d, /a/b/c, /a/b
+             * ]
+             * this means that request for module resolution from file in any of these folder will be immediately found in cache.
+             */
+            function set(directory, result) {
+                var path = ts.toPath(directory, currentDirectory, getCanonicalFileName);
+                // if entry is already in cache do nothing
+                if (directoryPathMap.has(path)) {
+                    return;
                 }
-                for (var _d = 0, _e = resolved.properties; _d < _e.length; _d++) {
-                    var p = _e[_d];
-                    visitSymbol(p);
+                directoryPathMap.set(path, result);
+                var resolvedFileName = result.resolvedModule &&
+                    (result.resolvedModule.originalPath || result.resolvedModule.resolvedFileName);
+                // find common prefix between directory and resolved file name
+                // this common prefix should be the shortest path that has the same resolution
+                // directory: /a/b/c/d/e
+                // resolvedFileName: /a/b/foo.d.ts
+                // commonPrefix: /a/b
+                // for failed lookups cache the result for every directory up to root
+                var commonPrefix = resolvedFileName && getCommonPrefix(path, resolvedFileName);
+                var current = path;
+                while (current !== commonPrefix) {
+                    var parent = ts.getDirectoryPath(current);
+                    if (parent === current || directoryPathMap.has(parent)) {
+                        break;
+                    }
+                    directoryPathMap.set(parent, result);
+                    current = parent;
                 }
             }
-            function visitSymbol(symbol) {
-                if (!symbol) {
-                    return false;
+            function getCommonPrefix(directory, resolution) {
+                var resolutionDirectory = ts.toPath(ts.getDirectoryPath(resolution), currentDirectory, getCanonicalFileName);
+                // find first position where directory and resolution differs
+                var i = 0;
+                var limit = Math.min(directory.length, resolutionDirectory.length);
+                while (i < limit && directory.charCodeAt(i) === resolutionDirectory.charCodeAt(i)) {
+                    i++;
                 }
-                var symbolId = ts.getSymbolId(symbol);
-                if (visitedSymbols[symbolId]) {
-                    return false;
+                if (i === directory.length && (resolutionDirectory.length === i || resolutionDirectory[i] === ts.directorySeparator)) {
+                    return directory;
                 }
-                visitedSymbols[symbolId] = symbol;
-                if (!accept(symbol)) {
-                    return true;
+                var rootLength = ts.getRootLength(directory);
+                if (i < rootLength) {
+                    return undefined;
                 }
-                var t = getTypeOfSymbol(symbol);
-                visitType(t); // Should handle members on classes and such
-                if (symbol.exports) {
-                    symbol.exports.forEach(visitSymbol);
+                var sep = directory.lastIndexOf(ts.directorySeparator, i - 1);
+                if (sep === -1) {
+                    return undefined;
                 }
-                ts.forEach(symbol.declarations, function (d) {
-                    // Type queries are too far resolved when we just visit the symbol's type
-                    //  (their type resolved directly to the member deeply referenced)
-                    // So to get the intervening symbols, we need to check if there's a type
-                    // query node on any of the symbol's declarations and get symbols there
-                    if (d.type && d.type.kind === 172 /* TypeQuery */) {
-                        var query = d.type;
-                        var entity = getResolvedSymbol(getFirstIdentifier(query.exprName));
-                        visitSymbol(entity);
-                    }
-                });
-                return false;
+                return directory.substr(0, Math.max(sep, rootLength));
             }
         }
     }
-    ts.createGetSymbolWalker = createGetSymbolWalker;
-})(ts || (ts = {}));
-/* @internal */
-var ts;
-(function (ts) {
-    var ambientModuleSymbolRegex = /^".+"$/;
-    var anon = "(anonymous)";
-    var nextSymbolId = 1;
-    var nextNodeId = 1;
-    var nextMergeId = 1;
-    var nextFlowId = 1;
-    var IterationUse;
-    (function (IterationUse) {
-        IterationUse[IterationUse["AllowsSyncIterablesFlag"] = 1] = "AllowsSyncIterablesFlag";
-        IterationUse[IterationUse["AllowsAsyncIterablesFlag"] = 2] = "AllowsAsyncIterablesFlag";
-        IterationUse[IterationUse["AllowsStringInputFlag"] = 4] = "AllowsStringInputFlag";
-        IterationUse[IterationUse["ForOfFlag"] = 8] = "ForOfFlag";
-        IterationUse[IterationUse["YieldStarFlag"] = 16] = "YieldStarFlag";
-        IterationUse[IterationUse["SpreadFlag"] = 32] = "SpreadFlag";
-        IterationUse[IterationUse["DestructuringFlag"] = 64] = "DestructuringFlag";
-        // Spread, Destructuring, Array element assignment
-        IterationUse[IterationUse["Element"] = 1] = "Element";
-        IterationUse[IterationUse["Spread"] = 33] = "Spread";
-        IterationUse[IterationUse["Destructuring"] = 65] = "Destructuring";
-        IterationUse[IterationUse["ForOf"] = 13] = "ForOf";
-        IterationUse[IterationUse["ForAwaitOf"] = 15] = "ForAwaitOf";
-        IterationUse[IterationUse["YieldStar"] = 17] = "YieldStar";
-        IterationUse[IterationUse["AsyncYieldStar"] = 19] = "AsyncYieldStar";
-        IterationUse[IterationUse["GeneratorReturnType"] = 1] = "GeneratorReturnType";
-        IterationUse[IterationUse["AsyncGeneratorReturnType"] = 2] = "AsyncGeneratorReturnType";
-    })(IterationUse || (IterationUse = {}));
-    var IterationTypeKind;
-    (function (IterationTypeKind) {
-        IterationTypeKind[IterationTypeKind["Yield"] = 0] = "Yield";
-        IterationTypeKind[IterationTypeKind["Return"] = 1] = "Return";
-        IterationTypeKind[IterationTypeKind["Next"] = 2] = "Next";
-    })(IterationTypeKind || (IterationTypeKind = {}));
-    var WideningKind;
-    (function (WideningKind) {
-        WideningKind[WideningKind["Normal"] = 0] = "Normal";
-        WideningKind[WideningKind["FunctionReturn"] = 1] = "FunctionReturn";
-        WideningKind[WideningKind["GeneratorNext"] = 2] = "GeneratorNext";
-        WideningKind[WideningKind["GeneratorYield"] = 3] = "GeneratorYield";
-    })(WideningKind || (WideningKind = {}));
-    var TypeFacts;
-    (function (TypeFacts) {
-        TypeFacts[TypeFacts["None"] = 0] = "None";
-        TypeFacts[TypeFacts["TypeofEQString"] = 1] = "TypeofEQString";
-        TypeFacts[TypeFacts["TypeofEQNumber"] = 2] = "TypeofEQNumber";
-        TypeFacts[TypeFacts["TypeofEQBigInt"] = 4] = "TypeofEQBigInt";
-        TypeFacts[TypeFacts["TypeofEQBoolean"] = 8] = "TypeofEQBoolean";
-        TypeFacts[TypeFacts["TypeofEQSymbol"] = 16] = "TypeofEQSymbol";
-        TypeFacts[TypeFacts["TypeofEQObject"] = 32] = "TypeofEQObject";
-        TypeFacts[TypeFacts["TypeofEQFunction"] = 64] = "TypeofEQFunction";
-        TypeFacts[TypeFacts["TypeofEQHostObject"] = 128] = "TypeofEQHostObject";
-        TypeFacts[TypeFacts["TypeofNEString"] = 256] = "TypeofNEString";
-        TypeFacts[TypeFacts["TypeofNENumber"] = 512] = "TypeofNENumber";
-        TypeFacts[TypeFacts["TypeofNEBigInt"] = 1024] = "TypeofNEBigInt";
-        TypeFacts[TypeFacts["TypeofNEBoolean"] = 2048] = "TypeofNEBoolean";
-        TypeFacts[TypeFacts["TypeofNESymbol"] = 4096] = "TypeofNESymbol";
-        TypeFacts[TypeFacts["TypeofNEObject"] = 8192] = "TypeofNEObject";
-        TypeFacts[TypeFacts["TypeofNEFunction"] = 16384] = "TypeofNEFunction";
-        TypeFacts[TypeFacts["TypeofNEHostObject"] = 32768] = "TypeofNEHostObject";
-        TypeFacts[TypeFacts["EQUndefined"] = 65536] = "EQUndefined";
-        TypeFacts[TypeFacts["EQNull"] = 131072] = "EQNull";
-        TypeFacts[TypeFacts["EQUndefinedOrNull"] = 262144] = "EQUndefinedOrNull";
-        TypeFacts[TypeFacts["NEUndefined"] = 524288] = "NEUndefined";
-        TypeFacts[TypeFacts["NENull"] = 1048576] = "NENull";
-        TypeFacts[TypeFacts["NEUndefinedOrNull"] = 2097152] = "NEUndefinedOrNull";
-        TypeFacts[TypeFacts["Truthy"] = 4194304] = "Truthy";
-        TypeFacts[TypeFacts["Falsy"] = 8388608] = "Falsy";
-        TypeFacts[TypeFacts["All"] = 16777215] = "All";
-        // The following members encode facts about particular kinds of types for use in the getTypeFacts function.
-        // The presence of a particular fact means that the given test is true for some (and possibly all) values
-        // of that kind of type.
-        TypeFacts[TypeFacts["BaseStringStrictFacts"] = 3735041] = "BaseStringStrictFacts";
-        TypeFacts[TypeFacts["BaseStringFacts"] = 12582401] = "BaseStringFacts";
-        TypeFacts[TypeFacts["StringStrictFacts"] = 16317953] = "StringStrictFacts";
-        TypeFacts[TypeFacts["StringFacts"] = 16776705] = "StringFacts";
-        TypeFacts[TypeFacts["EmptyStringStrictFacts"] = 12123649] = "EmptyStringStrictFacts";
-        TypeFacts[TypeFacts["EmptyStringFacts"] = 12582401] = "EmptyStringFacts";
-        TypeFacts[TypeFacts["NonEmptyStringStrictFacts"] = 7929345] = "NonEmptyStringStrictFacts";
-        TypeFacts[TypeFacts["NonEmptyStringFacts"] = 16776705] = "NonEmptyStringFacts";
-        TypeFacts[TypeFacts["BaseNumberStrictFacts"] = 3734786] = "BaseNumberStrictFacts";
-        TypeFacts[TypeFacts["BaseNumberFacts"] = 12582146] = "BaseNumberFacts";
-        TypeFacts[TypeFacts["NumberStrictFacts"] = 16317698] = "NumberStrictFacts";
-        TypeFacts[TypeFacts["NumberFacts"] = 16776450] = "NumberFacts";
-        TypeFacts[TypeFacts["ZeroNumberStrictFacts"] = 12123394] = "ZeroNumberStrictFacts";
-        TypeFacts[TypeFacts["ZeroNumberFacts"] = 12582146] = "ZeroNumberFacts";
-        TypeFacts[TypeFacts["NonZeroNumberStrictFacts"] = 7929090] = "NonZeroNumberStrictFacts";
-        TypeFacts[TypeFacts["NonZeroNumberFacts"] = 16776450] = "NonZeroNumberFacts";
-        TypeFacts[TypeFacts["BaseBigIntStrictFacts"] = 3734276] = "BaseBigIntStrictFacts";
-        TypeFacts[TypeFacts["BaseBigIntFacts"] = 12581636] = "BaseBigIntFacts";
-        TypeFacts[TypeFacts["BigIntStrictFacts"] = 16317188] = "BigIntStrictFacts";
-        TypeFacts[TypeFacts["BigIntFacts"] = 16775940] = "BigIntFacts";
-        TypeFacts[TypeFacts["ZeroBigIntStrictFacts"] = 12122884] = "ZeroBigIntStrictFacts";
-        TypeFacts[TypeFacts["ZeroBigIntFacts"] = 12581636] = "ZeroBigIntFacts";
-        TypeFacts[TypeFacts["NonZeroBigIntStrictFacts"] = 7928580] = "NonZeroBigIntStrictFacts";
-        TypeFacts[TypeFacts["NonZeroBigIntFacts"] = 16775940] = "NonZeroBigIntFacts";
-        TypeFacts[TypeFacts["BaseBooleanStrictFacts"] = 3733256] = "BaseBooleanStrictFacts";
-        TypeFacts[TypeFacts["BaseBooleanFacts"] = 12580616] = "BaseBooleanFacts";
-        TypeFacts[TypeFacts["BooleanStrictFacts"] = 16316168] = "BooleanStrictFacts";
-        TypeFacts[TypeFacts["BooleanFacts"] = 16774920] = "BooleanFacts";
-        TypeFacts[TypeFacts["FalseStrictFacts"] = 12121864] = "FalseStrictFacts";
-        TypeFacts[TypeFacts["FalseFacts"] = 12580616] = "FalseFacts";
-        TypeFacts[TypeFacts["TrueStrictFacts"] = 7927560] = "TrueStrictFacts";
-        TypeFacts[TypeFacts["TrueFacts"] = 16774920] = "TrueFacts";
-        TypeFacts[TypeFacts["SymbolStrictFacts"] = 7925520] = "SymbolStrictFacts";
-        TypeFacts[TypeFacts["SymbolFacts"] = 16772880] = "SymbolFacts";
-        TypeFacts[TypeFacts["ObjectStrictFacts"] = 7888800] = "ObjectStrictFacts";
-        TypeFacts[TypeFacts["ObjectFacts"] = 16736160] = "ObjectFacts";
-        TypeFacts[TypeFacts["FunctionStrictFacts"] = 7880640] = "FunctionStrictFacts";
-        TypeFacts[TypeFacts["FunctionFacts"] = 16728000] = "FunctionFacts";
-        TypeFacts[TypeFacts["UndefinedFacts"] = 9830144] = "UndefinedFacts";
-        TypeFacts[TypeFacts["NullFacts"] = 9363232] = "NullFacts";
-        TypeFacts[TypeFacts["EmptyObjectStrictFacts"] = 16318463] = "EmptyObjectStrictFacts";
-        TypeFacts[TypeFacts["EmptyObjectFacts"] = 16777215] = "EmptyObjectFacts";
-    })(TypeFacts || (TypeFacts = {}));
-    var typeofEQFacts = ts.createMapFromTemplate({
-        string: 1 /* TypeofEQString */,
-        number: 2 /* TypeofEQNumber */,
-        bigint: 4 /* TypeofEQBigInt */,
-        boolean: 8 /* TypeofEQBoolean */,
-        symbol: 16 /* TypeofEQSymbol */,
-        undefined: 65536 /* EQUndefined */,
-        object: 32 /* TypeofEQObject */,
-        function: 64 /* TypeofEQFunction */
-    });
-    var typeofNEFacts = ts.createMapFromTemplate({
-        string: 256 /* TypeofNEString */,
-        number: 512 /* TypeofNENumber */,
-        bigint: 1024 /* TypeofNEBigInt */,
-        boolean: 2048 /* TypeofNEBoolean */,
-        symbol: 4096 /* TypeofNESymbol */,
-        undefined: 524288 /* NEUndefined */,
-        object: 8192 /* TypeofNEObject */,
-        function: 16384 /* TypeofNEFunction */
-    });
-    var TypeSystemPropertyName;
-    (function (TypeSystemPropertyName) {
-        TypeSystemPropertyName[TypeSystemPropertyName["Type"] = 0] = "Type";
-        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedBaseConstructorType"] = 1] = "ResolvedBaseConstructorType";
-        TypeSystemPropertyName[TypeSystemPropertyName["DeclaredType"] = 2] = "DeclaredType";
-        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedReturnType"] = 3] = "ResolvedReturnType";
-        TypeSystemPropertyName[TypeSystemPropertyName["ImmediateBaseConstraint"] = 4] = "ImmediateBaseConstraint";
-        TypeSystemPropertyName[TypeSystemPropertyName["EnumTagType"] = 5] = "EnumTagType";
-        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedTypeArguments"] = 6] = "ResolvedTypeArguments";
-    })(TypeSystemPropertyName || (TypeSystemPropertyName = {}));
-    var CheckMode;
-    (function (CheckMode) {
-        CheckMode[CheckMode["Normal"] = 0] = "Normal";
-        CheckMode[CheckMode["Contextual"] = 1] = "Contextual";
-        CheckMode[CheckMode["Inferential"] = 2] = "Inferential";
-        CheckMode[CheckMode["SkipContextSensitive"] = 4] = "SkipContextSensitive";
-        CheckMode[CheckMode["SkipGenericFunctions"] = 8] = "SkipGenericFunctions";
-        CheckMode[CheckMode["IsForSignatureHelp"] = 16] = "IsForSignatureHelp";
-    })(CheckMode || (CheckMode = {}));
-    var AccessFlags;
-    (function (AccessFlags) {
-        AccessFlags[AccessFlags["None"] = 0] = "None";
-        AccessFlags[AccessFlags["NoIndexSignatures"] = 1] = "NoIndexSignatures";
-        AccessFlags[AccessFlags["Writing"] = 2] = "Writing";
-        AccessFlags[AccessFlags["CacheSymbol"] = 4] = "CacheSymbol";
-        AccessFlags[AccessFlags["NoTupleBoundsCheck"] = 8] = "NoTupleBoundsCheck";
-    })(AccessFlags || (AccessFlags = {}));
-    var SignatureCheckMode;
-    (function (SignatureCheckMode) {
-        SignatureCheckMode[SignatureCheckMode["BivariantCallback"] = 1] = "BivariantCallback";
-        SignatureCheckMode[SignatureCheckMode["StrictCallback"] = 2] = "StrictCallback";
-        SignatureCheckMode[SignatureCheckMode["IgnoreReturnTypes"] = 4] = "IgnoreReturnTypes";
-        SignatureCheckMode[SignatureCheckMode["StrictArity"] = 8] = "StrictArity";
-        SignatureCheckMode[SignatureCheckMode["Callback"] = 3] = "Callback";
-    })(SignatureCheckMode || (SignatureCheckMode = {}));
-    var IntersectionState;
-    (function (IntersectionState) {
-        IntersectionState[IntersectionState["None"] = 0] = "None";
-        IntersectionState[IntersectionState["Source"] = 1] = "Source";
-        IntersectionState[IntersectionState["Target"] = 2] = "Target";
-        IntersectionState[IntersectionState["PropertyCheck"] = 4] = "PropertyCheck";
-        IntersectionState[IntersectionState["InPropertyCheck"] = 8] = "InPropertyCheck";
-    })(IntersectionState || (IntersectionState = {}));
-    var MappedTypeModifiers;
-    (function (MappedTypeModifiers) {
-        MappedTypeModifiers[MappedTypeModifiers["IncludeReadonly"] = 1] = "IncludeReadonly";
-        MappedTypeModifiers[MappedTypeModifiers["ExcludeReadonly"] = 2] = "ExcludeReadonly";
-        MappedTypeModifiers[MappedTypeModifiers["IncludeOptional"] = 4] = "IncludeOptional";
-        MappedTypeModifiers[MappedTypeModifiers["ExcludeOptional"] = 8] = "ExcludeOptional";
-    })(MappedTypeModifiers || (MappedTypeModifiers = {}));
-    var ExpandingFlags;
-    (function (ExpandingFlags) {
-        ExpandingFlags[ExpandingFlags["None"] = 0] = "None";
-        ExpandingFlags[ExpandingFlags["Source"] = 1] = "Source";
-        ExpandingFlags[ExpandingFlags["Target"] = 2] = "Target";
-        ExpandingFlags[ExpandingFlags["Both"] = 3] = "Both";
-    })(ExpandingFlags || (ExpandingFlags = {}));
-    var MembersOrExportsResolutionKind;
-    (function (MembersOrExportsResolutionKind) {
-        MembersOrExportsResolutionKind["resolvedExports"] = "resolvedExports";
-        MembersOrExportsResolutionKind["resolvedMembers"] = "resolvedMembers";
-    })(MembersOrExportsResolutionKind || (MembersOrExportsResolutionKind = {}));
-    var UnusedKind;
-    (function (UnusedKind) {
-        UnusedKind[UnusedKind["Local"] = 0] = "Local";
-        UnusedKind[UnusedKind["Parameter"] = 1] = "Parameter";
-    })(UnusedKind || (UnusedKind = {}));
-    var isNotOverloadAndNotAccessor = ts.and(isNotOverload, isNotAccessor);
-    var DeclarationMeaning;
-    (function (DeclarationMeaning) {
-        DeclarationMeaning[DeclarationMeaning["GetAccessor"] = 1] = "GetAccessor";
-        DeclarationMeaning[DeclarationMeaning["SetAccessor"] = 2] = "SetAccessor";
-        DeclarationMeaning[DeclarationMeaning["PropertyAssignment"] = 4] = "PropertyAssignment";
-        DeclarationMeaning[DeclarationMeaning["Method"] = 8] = "Method";
-        DeclarationMeaning[DeclarationMeaning["GetOrSetAccessor"] = 3] = "GetOrSetAccessor";
-        DeclarationMeaning[DeclarationMeaning["PropertyAssignmentOrMethod"] = 12] = "PropertyAssignmentOrMethod";
-    })(DeclarationMeaning || (DeclarationMeaning = {}));
-    var DeclarationSpaces;
-    (function (DeclarationSpaces) {
-        DeclarationSpaces[DeclarationSpaces["None"] = 0] = "None";
-        DeclarationSpaces[DeclarationSpaces["ExportValue"] = 1] = "ExportValue";
-        DeclarationSpaces[DeclarationSpaces["ExportType"] = 2] = "ExportType";
-        DeclarationSpaces[DeclarationSpaces["ExportNamespace"] = 4] = "ExportNamespace";
-    })(DeclarationSpaces || (DeclarationSpaces = {}));
-    function SymbolLinks() {
-    }
-    function NodeLinks() {
-        this.flags = 0;
+    ts.createModuleResolutionCacheWithMaps = createModuleResolutionCacheWithMaps;
+    function resolveModuleNameFromCache(moduleName, containingFile, cache) {
+        var containingDirectory = ts.getDirectoryPath(containingFile);
+        var perFolderCache = cache && cache.getOrCreateCacheForDirectory(containingDirectory);
+        return perFolderCache && perFolderCache.get(moduleName);
     }
-    function getNodeId(node) {
-        if (!node.id) {
-            node.id = nextNodeId;
-            nextNodeId++;
+    ts.resolveModuleNameFromCache = resolveModuleNameFromCache;
+    function resolveModuleName(moduleName, containingFile, compilerOptions, host, cache, redirectedReference) {
+        var traceEnabled = isTraceEnabled(compilerOptions, host);
+        if (redirectedReference) {
+            compilerOptions = redirectedReference.commandLine.options;
         }
-        return node.id;
-    }
-    ts.getNodeId = getNodeId;
-    function getSymbolId(symbol) {
-        if (!symbol.id) {
-            symbol.id = nextSymbolId;
-            nextSymbolId++;
+        if (traceEnabled) {
+            trace(host, ts.Diagnostics.Resolving_module_0_from_1, moduleName, containingFile);
+            if (redirectedReference) {
+                trace(host, ts.Diagnostics.Using_compiler_options_of_project_reference_redirect_0, redirectedReference.sourceFile.fileName);
+            }
         }
-        return symbol.id;
-    }
-    ts.getSymbolId = getSymbolId;
-    function isInstantiatedModule(node, preserveConstEnums) {
-        var moduleState = ts.getModuleInstanceState(node);
-        return moduleState === 1 /* Instantiated */ ||
-            (preserveConstEnums && moduleState === 2 /* ConstEnumOnly */);
-    }
-    ts.isInstantiatedModule = isInstantiatedModule;
-    function createTypeChecker(host, produceDiagnostics) {
-        var getPackagesSet = ts.memoize(function () {
-            var set = ts.createMap();
-            host.getSourceFiles().forEach(function (sf) {
-                if (!sf.resolvedModules)
-                    return;
-                ts.forEachEntry(sf.resolvedModules, function (r) {
-                    if (r && r.packageId)
-                        set.set(r.packageId.name, true);
-                });
-            });
-            return set;
-        });
-        // Cancellation that controls whether or not we can cancel in the middle of type checking.
-        // In general cancelling is *not* safe for the type checker.  We might be in the middle of
-        // computing something, and we will leave our internals in an inconsistent state.  Callers
-        // who set the cancellation token should catch if a cancellation exception occurs, and
-        // should throw away and create a new TypeChecker.
-        //
-        // Currently we only support setting the cancellation token when getting diagnostics.  This
-        // is because diagnostics can be quite expensive, and we want to allow hosts to bail out if
-        // they no longer need the information (for example, if the user started editing again).
-        var cancellationToken;
-        var requestedExternalEmitHelpers;
-        var externalHelpersModule;
-        var Symbol = ts.objectAllocator.getSymbolConstructor();
-        var Type = ts.objectAllocator.getTypeConstructor();
-        var Signature = ts.objectAllocator.getSignatureConstructor();
-        var typeCount = 0;
-        var symbolCount = 0;
-        var enumCount = 0;
-        var totalInstantiationCount = 0;
-        var instantiationCount = 0;
-        var instantiationDepth = 0;
-        var constraintDepth = 0;
-        var currentNode;
-        var emptySymbols = ts.createSymbolTable();
-        var arrayVariances = [1 /* Covariant */];
-        var compilerOptions = host.getCompilerOptions();
-        var languageVersion = ts.getEmitScriptTarget(compilerOptions);
-        var moduleKind = ts.getEmitModuleKind(compilerOptions);
-        var allowSyntheticDefaultImports = ts.getAllowSyntheticDefaultImports(compilerOptions);
-        var strictNullChecks = ts.getStrictOptionValue(compilerOptions, "strictNullChecks");
-        var strictFunctionTypes = ts.getStrictOptionValue(compilerOptions, "strictFunctionTypes");
-        var strictBindCallApply = ts.getStrictOptionValue(compilerOptions, "strictBindCallApply");
-        var strictPropertyInitialization = ts.getStrictOptionValue(compilerOptions, "strictPropertyInitialization");
-        var noImplicitAny = ts.getStrictOptionValue(compilerOptions, "noImplicitAny");
-        var noImplicitThis = ts.getStrictOptionValue(compilerOptions, "noImplicitThis");
-        var keyofStringsOnly = !!compilerOptions.keyofStringsOnly;
-        var freshObjectLiteralFlag = compilerOptions.suppressExcessPropertyErrors ? 0 : 32768 /* FreshLiteral */;
-        var emitResolver = createResolver();
-        var nodeBuilder = createNodeBuilder();
-        var globals = ts.createSymbolTable();
-        var undefinedSymbol = createSymbol(4 /* Property */, "undefined");
-        undefinedSymbol.declarations = [];
-        var globalThisSymbol = createSymbol(1536 /* Module */, "globalThis", 8 /* Readonly */);
-        globalThisSymbol.exports = globals;
-        globalThisSymbol.declarations = [];
-        globals.set(globalThisSymbol.escapedName, globalThisSymbol);
-        var argumentsSymbol = createSymbol(4 /* Property */, "arguments");
-        var requireSymbol = createSymbol(4 /* Property */, "require");
-        /** This will be set during calls to `getResolvedSignature` where services determines an apparent number of arguments greater than what is actually provided. */
-        var apparentArgumentCount;
-        // for public members that accept a Node or one of its subtypes, we must guard against
-        // synthetic nodes created during transformations by calling `getParseTreeNode`.
-        // for most of these, we perform the guard only on `checker` to avoid any possible
-        // extra cost of calling `getParseTreeNode` when calling these functions from inside the
-        // checker.
-        var checker = {
-            getNodeCount: function () { return ts.sum(host.getSourceFiles(), "nodeCount"); },
-            getIdentifierCount: function () { return ts.sum(host.getSourceFiles(), "identifierCount"); },
-            getSymbolCount: function () { return ts.sum(host.getSourceFiles(), "symbolCount") + symbolCount; },
-            getTypeCount: function () { return typeCount; },
-            getInstantiationCount: function () { return totalInstantiationCount; },
-            getRelationCacheSizes: function () { return ({
-                assignable: assignableRelation.size,
-                identity: identityRelation.size,
-                subtype: subtypeRelation.size,
-                strictSubtype: strictSubtypeRelation.size,
-            }); },
-            isUndefinedSymbol: function (symbol) { return symbol === undefinedSymbol; },
-            isArgumentsSymbol: function (symbol) { return symbol === argumentsSymbol; },
-            isUnknownSymbol: function (symbol) { return symbol === unknownSymbol; },
-            getMergedSymbol: getMergedSymbol,
-            getDiagnostics: getDiagnostics,
-            getGlobalDiagnostics: getGlobalDiagnostics,
-            getTypeOfSymbolAtLocation: function (symbol, location) {
-                location = ts.getParseTreeNode(location);
-                return location ? getTypeOfSymbolAtLocation(symbol, location) : errorType;
-            },
-            getSymbolsOfParameterPropertyDeclaration: function (parameterIn, parameterName) {
-                var parameter = ts.getParseTreeNode(parameterIn, ts.isParameter);
-                if (parameter === undefined)
-                    return ts.Debug.fail("Cannot get symbols of a synthetic parameter that cannot be resolved to a parse-tree node.");
-                return getSymbolsOfParameterPropertyDeclaration(parameter, ts.escapeLeadingUnderscores(parameterName));
-            },
-            getDeclaredTypeOfSymbol: getDeclaredTypeOfSymbol,
-            getPropertiesOfType: getPropertiesOfType,
-            getPropertyOfType: function (type, name) { return getPropertyOfType(type, ts.escapeLeadingUnderscores(name)); },
-            getPrivateIdentifierPropertyOfType: function (leftType, name, location) {
-                var node = ts.getParseTreeNode(location);
-                if (!node) {
-                    return undefined;
-                }
-                var propName = ts.escapeLeadingUnderscores(name);
-                var lexicallyScopedIdentifier = lookupSymbolForPrivateIdentifierDeclaration(propName, node);
-                return lexicallyScopedIdentifier ? getPrivateIdentifierPropertyOfType(leftType, lexicallyScopedIdentifier) : undefined;
-            },
-            getTypeOfPropertyOfType: function (type, name) { return getTypeOfPropertyOfType(type, ts.escapeLeadingUnderscores(name)); },
-            getIndexInfoOfType: getIndexInfoOfType,
-            getSignaturesOfType: getSignaturesOfType,
-            getIndexTypeOfType: getIndexTypeOfType,
-            getBaseTypes: getBaseTypes,
-            getBaseTypeOfLiteralType: getBaseTypeOfLiteralType,
-            getWidenedType: getWidenedType,
-            getTypeFromTypeNode: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isTypeNode);
-                return node ? getTypeFromTypeNode(node) : errorType;
-            },
-            getParameterType: getTypeAtPosition,
-            getPromisedTypeOfPromise: getPromisedTypeOfPromise,
-            getReturnTypeOfSignature: getReturnTypeOfSignature,
-            isNullableType: isNullableType,
-            getNullableType: getNullableType,
-            getNonNullableType: getNonNullableType,
-            getNonOptionalType: removeOptionalTypeMarker,
-            getTypeArguments: getTypeArguments,
-            typeToTypeNode: nodeBuilder.typeToTypeNode,
-            indexInfoToIndexSignatureDeclaration: nodeBuilder.indexInfoToIndexSignatureDeclaration,
-            signatureToSignatureDeclaration: nodeBuilder.signatureToSignatureDeclaration,
-            symbolToEntityName: nodeBuilder.symbolToEntityName,
-            symbolToExpression: nodeBuilder.symbolToExpression,
-            symbolToTypeParameterDeclarations: nodeBuilder.symbolToTypeParameterDeclarations,
-            symbolToParameterDeclaration: nodeBuilder.symbolToParameterDeclaration,
-            typeParameterToDeclaration: nodeBuilder.typeParameterToDeclaration,
-            getSymbolsInScope: function (location, meaning) {
-                location = ts.getParseTreeNode(location);
-                return location ? getSymbolsInScope(location, meaning) : [];
-            },
-            getSymbolAtLocation: function (node) {
-                node = ts.getParseTreeNode(node);
-                // set ignoreErrors: true because any lookups invoked by the API shouldn't cause any new errors
-                return node ? getSymbolAtLocation(node, /*ignoreErrors*/ true) : undefined;
-            },
-            getShorthandAssignmentValueSymbol: function (node) {
-                node = ts.getParseTreeNode(node);
-                return node ? getShorthandAssignmentValueSymbol(node) : undefined;
-            },
-            getExportSpecifierLocalTargetSymbol: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isExportSpecifier);
-                return node ? getExportSpecifierLocalTargetSymbol(node) : undefined;
-            },
-            getExportSymbolOfSymbol: function (symbol) {
-                return getMergedSymbol(symbol.exportSymbol || symbol);
-            },
-            getTypeAtLocation: function (node) {
-                node = ts.getParseTreeNode(node);
-                return node ? getTypeOfNode(node) : errorType;
-            },
-            getTypeOfAssignmentPattern: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isAssignmentPattern);
-                return node && getTypeOfAssignmentPattern(node) || errorType;
-            },
-            getPropertySymbolOfDestructuringAssignment: function (locationIn) {
-                var location = ts.getParseTreeNode(locationIn, ts.isIdentifier);
-                return location ? getPropertySymbolOfDestructuringAssignment(location) : undefined;
-            },
-            signatureToString: function (signature, enclosingDeclaration, flags, kind) {
-                return signatureToString(signature, ts.getParseTreeNode(enclosingDeclaration), flags, kind);
-            },
-            typeToString: function (type, enclosingDeclaration, flags) {
-                return typeToString(type, ts.getParseTreeNode(enclosingDeclaration), flags);
-            },
-            symbolToString: function (symbol, enclosingDeclaration, meaning, flags) {
-                return symbolToString(symbol, ts.getParseTreeNode(enclosingDeclaration), meaning, flags);
-            },
-            typePredicateToString: function (predicate, enclosingDeclaration, flags) {
-                return typePredicateToString(predicate, ts.getParseTreeNode(enclosingDeclaration), flags);
-            },
-            writeSignature: function (signature, enclosingDeclaration, flags, kind, writer) {
-                return signatureToString(signature, ts.getParseTreeNode(enclosingDeclaration), flags, kind, writer);
-            },
-            writeType: function (type, enclosingDeclaration, flags, writer) {
-                return typeToString(type, ts.getParseTreeNode(enclosingDeclaration), flags, writer);
-            },
-            writeSymbol: function (symbol, enclosingDeclaration, meaning, flags, writer) {
-                return symbolToString(symbol, ts.getParseTreeNode(enclosingDeclaration), meaning, flags, writer);
-            },
-            writeTypePredicate: function (predicate, enclosingDeclaration, flags, writer) {
-                return typePredicateToString(predicate, ts.getParseTreeNode(enclosingDeclaration), flags, writer);
-            },
-            getAugmentedPropertiesOfType: getAugmentedPropertiesOfType,
-            getRootSymbols: getRootSymbols,
-            getContextualType: function (nodeIn, contextFlags) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isExpression);
-                if (!node) {
-                    return undefined;
-                }
-                var containingCall = ts.findAncestor(node, ts.isCallLikeExpression);
-                var containingCallResolvedSignature = containingCall && getNodeLinks(containingCall).resolvedSignature;
-                if (contextFlags & 4 /* Completions */ && containingCall) {
-                    var toMarkSkip = node;
-                    do {
-                        getNodeLinks(toMarkSkip).skipDirectInference = true;
-                        toMarkSkip = toMarkSkip.parent;
-                    } while (toMarkSkip && toMarkSkip !== containingCall);
-                    getNodeLinks(containingCall).resolvedSignature = undefined;
-                }
-                var result = getContextualType(node, contextFlags);
-                if (contextFlags & 4 /* Completions */ && containingCall) {
-                    var toMarkSkip = node;
-                    do {
-                        getNodeLinks(toMarkSkip).skipDirectInference = undefined;
-                        toMarkSkip = toMarkSkip.parent;
-                    } while (toMarkSkip && toMarkSkip !== containingCall);
-                    getNodeLinks(containingCall).resolvedSignature = containingCallResolvedSignature;
-                }
-                return result;
-            },
-            getContextualTypeForObjectLiteralElement: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isObjectLiteralElementLike);
-                return node ? getContextualTypeForObjectLiteralElement(node) : undefined;
-            },
-            getContextualTypeForArgumentAtIndex: function (nodeIn, argIndex) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isCallLikeExpression);
-                return node && getContextualTypeForArgumentAtIndex(node, argIndex);
-            },
-            getContextualTypeForJsxAttribute: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isJsxAttributeLike);
-                return node && getContextualTypeForJsxAttribute(node);
-            },
-            isContextSensitive: isContextSensitive,
-            getFullyQualifiedName: getFullyQualifiedName,
-            getResolvedSignature: function (node, candidatesOutArray, argumentCount) {
-                return getResolvedSignatureWorker(node, candidatesOutArray, argumentCount, 0 /* Normal */);
-            },
-            getResolvedSignatureForSignatureHelp: function (node, candidatesOutArray, argumentCount) {
-                return getResolvedSignatureWorker(node, candidatesOutArray, argumentCount, 16 /* IsForSignatureHelp */);
-            },
-            getExpandedParameters: getExpandedParameters,
-            hasEffectiveRestParameter: hasEffectiveRestParameter,
-            getConstantValue: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, canHaveConstantValue);
-                return node ? getConstantValue(node) : undefined;
-            },
-            isValidPropertyAccess: function (nodeIn, propertyName) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isPropertyAccessOrQualifiedNameOrImportTypeNode);
-                return !!node && isValidPropertyAccess(node, ts.escapeLeadingUnderscores(propertyName));
-            },
-            isValidPropertyAccessForCompletions: function (nodeIn, type, property) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isPropertyAccessExpression);
-                return !!node && isValidPropertyAccessForCompletions(node, type, property);
-            },
-            getSignatureFromDeclaration: function (declarationIn) {
-                var declaration = ts.getParseTreeNode(declarationIn, ts.isFunctionLike);
-                return declaration ? getSignatureFromDeclaration(declaration) : undefined;
-            },
-            isImplementationOfOverload: function (node) {
-                var parsed = ts.getParseTreeNode(node, ts.isFunctionLike);
-                return parsed ? isImplementationOfOverload(parsed) : undefined;
-            },
-            getImmediateAliasedSymbol: getImmediateAliasedSymbol,
-            getAliasedSymbol: resolveAlias,
-            getEmitResolver: getEmitResolver,
-            getExportsOfModule: getExportsOfModuleAsArray,
-            getExportsAndPropertiesOfModule: getExportsAndPropertiesOfModule,
-            getSymbolWalker: ts.createGetSymbolWalker(getRestTypeOfSignature, getTypePredicateOfSignature, getReturnTypeOfSignature, getBaseTypes, resolveStructuredTypeMembers, getTypeOfSymbol, getResolvedSymbol, getIndexTypeOfStructuredType, getConstraintOfTypeParameter, ts.getFirstIdentifier, getTypeArguments),
-            getAmbientModules: getAmbientModules,
-            getJsxIntrinsicTagNamesAt: getJsxIntrinsicTagNamesAt,
-            isOptionalParameter: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isParameter);
-                return node ? isOptionalParameter(node) : false;
-            },
-            tryGetMemberInModuleExports: function (name, symbol) { return tryGetMemberInModuleExports(ts.escapeLeadingUnderscores(name), symbol); },
-            tryGetMemberInModuleExportsAndProperties: function (name, symbol) { return tryGetMemberInModuleExportsAndProperties(ts.escapeLeadingUnderscores(name), symbol); },
-            tryFindAmbientModuleWithoutAugmentations: function (moduleName) {
-                // we deliberately exclude augmentations
-                // since we are only interested in declarations of the module itself
-                return tryFindAmbientModule(moduleName, /*withAugmentations*/ false);
-            },
-            getApparentType: getApparentType,
-            getUnionType: getUnionType,
-            isTypeAssignableTo: isTypeAssignableTo,
-            createAnonymousType: createAnonymousType,
-            createSignature: createSignature,
-            createSymbol: createSymbol,
-            createIndexInfo: createIndexInfo,
-            getAnyType: function () { return anyType; },
-            getStringType: function () { return stringType; },
-            getNumberType: function () { return numberType; },
-            createPromiseType: createPromiseType,
-            createArrayType: createArrayType,
-            getElementTypeOfArrayType: getElementTypeOfArrayType,
-            getBooleanType: function () { return booleanType; },
-            getFalseType: function (fresh) { return fresh ? falseType : regularFalseType; },
-            getTrueType: function (fresh) { return fresh ? trueType : regularTrueType; },
-            getVoidType: function () { return voidType; },
-            getUndefinedType: function () { return undefinedType; },
-            getNullType: function () { return nullType; },
-            getESSymbolType: function () { return esSymbolType; },
-            getNeverType: function () { return neverType; },
-            getOptionalType: function () { return optionalType; },
-            isSymbolAccessible: isSymbolAccessible,
-            isArrayType: isArrayType,
-            isTupleType: isTupleType,
-            isArrayLikeType: isArrayLikeType,
-            isTypeInvalidDueToUnionDiscriminant: isTypeInvalidDueToUnionDiscriminant,
-            getAllPossiblePropertiesOfTypes: getAllPossiblePropertiesOfTypes,
-            getSuggestedSymbolForNonexistentProperty: getSuggestedSymbolForNonexistentProperty,
-            getSuggestionForNonexistentProperty: getSuggestionForNonexistentProperty,
-            getSuggestedSymbolForNonexistentSymbol: function (location, name, meaning) { return getSuggestedSymbolForNonexistentSymbol(location, ts.escapeLeadingUnderscores(name), meaning); },
-            getSuggestionForNonexistentSymbol: function (location, name, meaning) { return getSuggestionForNonexistentSymbol(location, ts.escapeLeadingUnderscores(name), meaning); },
-            getSuggestedSymbolForNonexistentModule: getSuggestedSymbolForNonexistentModule,
-            getSuggestionForNonexistentExport: getSuggestionForNonexistentExport,
-            getBaseConstraintOfType: getBaseConstraintOfType,
-            getDefaultFromTypeParameter: function (type) { return type && type.flags & 262144 /* TypeParameter */ ? getDefaultFromTypeParameter(type) : undefined; },
-            resolveName: function (name, location, meaning, excludeGlobals) {
-                return resolveName(location, ts.escapeLeadingUnderscores(name), meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false, excludeGlobals);
-            },
-            getJsxNamespace: function (n) { return ts.unescapeLeadingUnderscores(getJsxNamespace(n)); },
-            getAccessibleSymbolChain: getAccessibleSymbolChain,
-            getTypePredicateOfSignature: getTypePredicateOfSignature,
-            resolveExternalModuleName: function (moduleSpecifier) {
-                return resolveExternalModuleName(moduleSpecifier, moduleSpecifier, /*ignoreErrors*/ true);
-            },
-            resolveExternalModuleSymbol: resolveExternalModuleSymbol,
-            tryGetThisTypeAt: function (node, includeGlobalThis) {
-                node = ts.getParseTreeNode(node);
-                return node && tryGetThisTypeAt(node, includeGlobalThis);
-            },
-            getTypeArgumentConstraint: function (nodeIn) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isTypeNode);
-                return node && getTypeArgumentConstraint(node);
-            },
-            getSuggestionDiagnostics: function (file, ct) {
-                if (ts.skipTypeChecking(file, compilerOptions, host)) {
-                    return ts.emptyArray;
+        var containingDirectory = ts.getDirectoryPath(containingFile);
+        var perFolderCache = cache && cache.getOrCreateCacheForDirectory(containingDirectory, redirectedReference);
+        var result = perFolderCache && perFolderCache.get(moduleName);
+        if (result) {
+            if (traceEnabled) {
+                trace(host, ts.Diagnostics.Resolution_for_module_0_was_found_in_cache_from_location_1, moduleName, containingDirectory);
+            }
+        }
+        else {
+            var moduleResolution = compilerOptions.moduleResolution;
+            if (moduleResolution === undefined) {
+                moduleResolution = ts.getEmitModuleKind(compilerOptions) === ts.ModuleKind.CommonJS ? ts.ModuleResolutionKind.NodeJs : ts.ModuleResolutionKind.Classic;
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Module_resolution_kind_is_not_specified_using_0, ts.ModuleResolutionKind[moduleResolution]);
                 }
-                var diagnostics;
-                try {
-                    // Record the cancellation token so it can be checked later on during checkSourceElement.
-                    // Do this in a finally block so we can ensure that it gets reset back to nothing after
-                    // this call is done.
-                    cancellationToken = ct;
-                    // Ensure file is type checked
-                    checkSourceFile(file);
-                    ts.Debug.assert(!!(getNodeLinks(file).flags & 1 /* TypeChecked */));
-                    diagnostics = ts.addRange(diagnostics, suggestionDiagnostics.getDiagnostics(file.fileName));
-                    checkUnusedIdentifiers(getPotentiallyUnusedIdentifiers(file), function (containingNode, kind, diag) {
-                        if (!ts.containsParseError(containingNode) && !unusedIsError(kind, !!(containingNode.flags & 8388608 /* Ambient */))) {
-                            (diagnostics || (diagnostics = [])).push(__assign(__assign({}, diag), { category: ts.DiagnosticCategory.Suggestion }));
-                        }
-                    });
-                    return diagnostics || ts.emptyArray;
+            }
+            else {
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Explicitly_specified_module_resolution_kind_Colon_0, ts.ModuleResolutionKind[moduleResolution]);
                 }
-                finally {
-                    cancellationToken = undefined;
+            }
+            ts.perfLogger.logStartResolveModule(moduleName /* , containingFile, ModuleResolutionKind[moduleResolution]*/);
+            switch (moduleResolution) {
+                case ts.ModuleResolutionKind.NodeJs:
+                    result = nodeModuleNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference);
+                    break;
+                case ts.ModuleResolutionKind.Classic:
+                    result = classicNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference);
+                    break;
+                default:
+                    return ts.Debug.fail("Unexpected moduleResolution: " + moduleResolution);
+            }
+            if (result && result.resolvedModule)
+                ts.perfLogger.logInfoEvent("Module \"" + moduleName + "\" resolved to \"" + result.resolvedModule.resolvedFileName + "\"");
+            ts.perfLogger.logStopResolveModule((result && result.resolvedModule) ? "" + result.resolvedModule.resolvedFileName : "null");
+            if (perFolderCache) {
+                perFolderCache.set(moduleName, result);
+                if (!ts.isExternalModuleNameRelative(moduleName)) {
+                    // put result in per-module name cache
+                    cache.getOrCreateCacheForModuleName(moduleName, redirectedReference).set(containingDirectory, result);
                 }
-            },
-            runWithCancellationToken: function (token, callback) {
-                try {
-                    cancellationToken = token;
-                    return callback(checker);
+            }
+        }
+        if (traceEnabled) {
+            if (result.resolvedModule) {
+                if (result.resolvedModule.packageId) {
+                    trace(host, ts.Diagnostics.Module_name_0_was_successfully_resolved_to_1_with_Package_ID_2, moduleName, result.resolvedModule.resolvedFileName, ts.packageIdToString(result.resolvedModule.packageId));
                 }
-                finally {
-                    cancellationToken = undefined;
+                else {
+                    trace(host, ts.Diagnostics.Module_name_0_was_successfully_resolved_to_1, moduleName, result.resolvedModule.resolvedFileName);
                 }
-            },
-            getLocalTypeParametersOfClassOrInterfaceOrTypeAlias: getLocalTypeParametersOfClassOrInterfaceOrTypeAlias,
-            isDeclarationVisible: isDeclarationVisible,
-        };
-        function getResolvedSignatureWorker(nodeIn, candidatesOutArray, argumentCount, checkMode) {
-            var node = ts.getParseTreeNode(nodeIn, ts.isCallLikeExpression);
-            apparentArgumentCount = argumentCount;
-            var res = node ? getResolvedSignature(node, candidatesOutArray, checkMode) : undefined;
-            apparentArgumentCount = undefined;
-            return res;
+            }
+            else {
+                trace(host, ts.Diagnostics.Module_name_0_was_not_resolved, moduleName);
+            }
         }
-        var tupleTypes = ts.createMap();
-        var unionTypes = ts.createMap();
-        var intersectionTypes = ts.createMap();
-        var literalTypes = ts.createMap();
-        var indexedAccessTypes = ts.createMap();
-        var substitutionTypes = ts.createMap();
-        var evolvingArrayTypes = [];
-        var undefinedProperties = ts.createMap();
-        var unknownSymbol = createSymbol(4 /* Property */, "unknown");
-        var resolvingSymbol = createSymbol(0, "__resolving__" /* Resolving */);
-        var anyType = createIntrinsicType(1 /* Any */, "any");
-        var autoType = createIntrinsicType(1 /* Any */, "any");
-        var wildcardType = createIntrinsicType(1 /* Any */, "any");
-        var errorType = createIntrinsicType(1 /* Any */, "error");
-        var nonInferrableAnyType = createIntrinsicType(1 /* Any */, "any", 524288 /* ContainsWideningType */);
-        var unknownType = createIntrinsicType(2 /* Unknown */, "unknown");
-        var undefinedType = createIntrinsicType(32768 /* Undefined */, "undefined");
-        var undefinedWideningType = strictNullChecks ? undefinedType : createIntrinsicType(32768 /* Undefined */, "undefined", 524288 /* ContainsWideningType */);
-        var optionalType = createIntrinsicType(32768 /* Undefined */, "undefined");
-        var nullType = createIntrinsicType(65536 /* Null */, "null");
-        var nullWideningType = strictNullChecks ? nullType : createIntrinsicType(65536 /* Null */, "null", 524288 /* ContainsWideningType */);
-        var stringType = createIntrinsicType(4 /* String */, "string");
-        var numberType = createIntrinsicType(8 /* Number */, "number");
-        var bigintType = createIntrinsicType(64 /* BigInt */, "bigint");
-        var falseType = createIntrinsicType(512 /* BooleanLiteral */, "false");
-        var regularFalseType = createIntrinsicType(512 /* BooleanLiteral */, "false");
-        var trueType = createIntrinsicType(512 /* BooleanLiteral */, "true");
-        var regularTrueType = createIntrinsicType(512 /* BooleanLiteral */, "true");
-        trueType.regularType = regularTrueType;
-        trueType.freshType = trueType;
-        regularTrueType.regularType = regularTrueType;
-        regularTrueType.freshType = trueType;
-        falseType.regularType = regularFalseType;
-        falseType.freshType = falseType;
-        regularFalseType.regularType = regularFalseType;
-        regularFalseType.freshType = falseType;
-        var booleanType = createBooleanType([regularFalseType, regularTrueType]);
-        // Also mark all combinations of fresh/regular booleans as "Boolean" so they print as `boolean` instead of `true | false`
-        // (The union is cached, so simply doing the marking here is sufficient)
-        createBooleanType([regularFalseType, trueType]);
-        createBooleanType([falseType, regularTrueType]);
-        createBooleanType([falseType, trueType]);
-        var esSymbolType = createIntrinsicType(4096 /* ESSymbol */, "symbol");
-        var voidType = createIntrinsicType(16384 /* Void */, "void");
-        var neverType = createIntrinsicType(131072 /* Never */, "never");
-        var silentNeverType = createIntrinsicType(131072 /* Never */, "never");
-        var nonInferrableType = createIntrinsicType(131072 /* Never */, "never", 2097152 /* NonInferrableType */);
-        var implicitNeverType = createIntrinsicType(131072 /* Never */, "never");
-        var unreachableNeverType = createIntrinsicType(131072 /* Never */, "never");
-        var nonPrimitiveType = createIntrinsicType(67108864 /* NonPrimitive */, "object");
-        var stringNumberSymbolType = getUnionType([stringType, numberType, esSymbolType]);
-        var keyofConstraintType = keyofStringsOnly ? stringType : stringNumberSymbolType;
-        var numberOrBigIntType = getUnionType([numberType, bigintType]);
-        var restrictiveMapper = makeFunctionTypeMapper(function (t) { return t.flags & 262144 /* TypeParameter */ ? getRestrictiveTypeParameter(t) : t; });
-        var permissiveMapper = makeFunctionTypeMapper(function (t) { return t.flags & 262144 /* TypeParameter */ ? wildcardType : t; });
-        var emptyObjectType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        var emptyJsxObjectType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        emptyJsxObjectType.objectFlags |= 4096 /* JsxAttributes */;
-        var emptyTypeLiteralSymbol = createSymbol(2048 /* TypeLiteral */, "__type" /* Type */);
-        emptyTypeLiteralSymbol.members = ts.createSymbolTable();
-        var emptyTypeLiteralType = createAnonymousType(emptyTypeLiteralSymbol, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        var emptyGenericType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        emptyGenericType.instantiations = ts.createMap();
-        var anyFunctionType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        // The anyFunctionType contains the anyFunctionType by definition. The flag is further propagated
-        // in getPropagatingFlagsOfTypes, and it is checked in inferFromTypes.
-        anyFunctionType.objectFlags |= 2097152 /* NonInferrableType */;
-        var noConstraintType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        var circularConstraintType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        var resolvingDefaultType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-        var markerSuperType = createTypeParameter();
-        var markerSubType = createTypeParameter();
-        markerSubType.constraint = markerSuperType;
-        var markerOtherType = createTypeParameter();
-        var noTypePredicate = createTypePredicate(1 /* Identifier */, "<<unresolved>>", 0, anyType);
-        var anySignature = createSignature(undefined, undefined, undefined, ts.emptyArray, anyType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
-        var unknownSignature = createSignature(undefined, undefined, undefined, ts.emptyArray, errorType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
-        var resolvingSignature = createSignature(undefined, undefined, undefined, ts.emptyArray, anyType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
-        var silentNeverSignature = createSignature(undefined, undefined, undefined, ts.emptyArray, silentNeverType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
-        var enumNumberIndexInfo = createIndexInfo(stringType, /*isReadonly*/ true);
-        var iterationTypesCache = ts.createMap(); // cache for common IterationTypes instances
-        var noIterationTypes = {
-            get yieldType() { return ts.Debug.fail("Not supported"); },
-            get returnType() { return ts.Debug.fail("Not supported"); },
-            get nextType() { return ts.Debug.fail("Not supported"); },
-        };
-        var anyIterationTypes = createIterationTypes(anyType, anyType, anyType);
-        var anyIterationTypesExceptNext = createIterationTypes(anyType, anyType, unknownType);
-        var defaultIterationTypes = createIterationTypes(neverType, anyType, undefinedType); // default iteration types for `Iterator`.
-        var asyncIterationTypesResolver = {
-            iterableCacheKey: "iterationTypesOfAsyncIterable",
-            iteratorCacheKey: "iterationTypesOfAsyncIterator",
-            iteratorSymbolName: "asyncIterator",
-            getGlobalIteratorType: getGlobalAsyncIteratorType,
-            getGlobalIterableType: getGlobalAsyncIterableType,
-            getGlobalIterableIteratorType: getGlobalAsyncIterableIteratorType,
-            getGlobalGeneratorType: getGlobalAsyncGeneratorType,
-            resolveIterationType: getAwaitedType,
-            mustHaveANextMethodDiagnostic: ts.Diagnostics.An_async_iterator_must_have_a_next_method,
-            mustBeAMethodDiagnostic: ts.Diagnostics.The_0_property_of_an_async_iterator_must_be_a_method,
-            mustHaveAValueDiagnostic: ts.Diagnostics.The_type_returned_by_the_0_method_of_an_async_iterator_must_be_a_promise_for_a_type_with_a_value_property,
-        };
-        var syncIterationTypesResolver = {
-            iterableCacheKey: "iterationTypesOfIterable",
-            iteratorCacheKey: "iterationTypesOfIterator",
-            iteratorSymbolName: "iterator",
-            getGlobalIteratorType: getGlobalIteratorType,
-            getGlobalIterableType: getGlobalIterableType,
-            getGlobalIterableIteratorType: getGlobalIterableIteratorType,
-            getGlobalGeneratorType: getGlobalGeneratorType,
-            resolveIterationType: function (type, _errorNode) { return type; },
-            mustHaveANextMethodDiagnostic: ts.Diagnostics.An_iterator_must_have_a_next_method,
-            mustBeAMethodDiagnostic: ts.Diagnostics.The_0_property_of_an_iterator_must_be_a_method,
-            mustHaveAValueDiagnostic: ts.Diagnostics.The_type_returned_by_the_0_method_of_an_iterator_must_have_a_value_property,
-        };
-        /** Key is "/path/to/a.ts|/path/to/b.ts". */
-        var amalgamatedDuplicates;
-        var reverseMappedCache = ts.createMap();
-        var ambientModulesCache;
-        /**
-         * List of every ambient module with a "*" wildcard.
-         * Unlike other ambient modules, these can't be stored in `globals` because symbol tables only deal with exact matches.
-         * This is only used if there is no exact match.
-         */
-        var patternAmbientModules;
-        var patternAmbientModuleAugmentations;
-        var globalObjectType;
-        var globalFunctionType;
-        var globalCallableFunctionType;
-        var globalNewableFunctionType;
-        var globalArrayType;
-        var globalReadonlyArrayType;
-        var globalStringType;
-        var globalNumberType;
-        var globalBooleanType;
-        var globalRegExpType;
-        var globalThisType;
-        var anyArrayType;
-        var autoArrayType;
-        var anyReadonlyArrayType;
-        var deferredGlobalNonNullableTypeAlias;
-        // The library files are only loaded when the feature is used.
-        // This allows users to just specify library files they want to used through --lib
-        // and they will not get an error from not having unrelated library files
-        var deferredGlobalESSymbolConstructorSymbol;
-        var deferredGlobalESSymbolType;
-        var deferredGlobalTypedPropertyDescriptorType;
-        var deferredGlobalPromiseType;
-        var deferredGlobalPromiseLikeType;
-        var deferredGlobalPromiseConstructorSymbol;
-        var deferredGlobalPromiseConstructorLikeType;
-        var deferredGlobalIterableType;
-        var deferredGlobalIteratorType;
-        var deferredGlobalIterableIteratorType;
-        var deferredGlobalGeneratorType;
-        var deferredGlobalIteratorYieldResultType;
-        var deferredGlobalIteratorReturnResultType;
-        var deferredGlobalAsyncIterableType;
-        var deferredGlobalAsyncIteratorType;
-        var deferredGlobalAsyncIterableIteratorType;
-        var deferredGlobalAsyncGeneratorType;
-        var deferredGlobalTemplateStringsArrayType;
-        var deferredGlobalImportMetaType;
-        var deferredGlobalExtractSymbol;
-        var deferredGlobalOmitSymbol;
-        var deferredGlobalBigIntType;
-        var allPotentiallyUnusedIdentifiers = ts.createMap(); // key is file name
-        var flowLoopStart = 0;
-        var flowLoopCount = 0;
-        var sharedFlowCount = 0;
-        var flowAnalysisDisabled = false;
-        var flowInvocationCount = 0;
-        var lastFlowNode;
-        var lastFlowNodeReachable;
-        var flowTypeCache;
-        var emptyStringType = getLiteralType("");
-        var zeroType = getLiteralType(0);
-        var zeroBigIntType = getLiteralType({ negative: false, base10Value: "0" });
-        var resolutionTargets = [];
-        var resolutionResults = [];
-        var resolutionPropertyNames = [];
-        var suggestionCount = 0;
-        var maximumSuggestionCount = 10;
-        var mergedSymbols = [];
-        var symbolLinks = [];
-        var nodeLinks = [];
-        var flowLoopCaches = [];
-        var flowLoopNodes = [];
-        var flowLoopKeys = [];
-        var flowLoopTypes = [];
-        var sharedFlowNodes = [];
-        var sharedFlowTypes = [];
-        var flowNodeReachable = [];
-        var potentialThisCollisions = [];
-        var potentialNewTargetCollisions = [];
-        var potentialWeakMapCollisions = [];
-        var awaitedTypeStack = [];
-        var diagnostics = ts.createDiagnosticCollection();
-        var suggestionDiagnostics = ts.createDiagnosticCollection();
-        var typeofTypesByName = ts.createMapFromTemplate({
-            string: stringType,
-            number: numberType,
-            bigint: bigintType,
-            boolean: booleanType,
-            symbol: esSymbolType,
-            undefined: undefinedType
-        });
-        var typeofType = createTypeofType();
-        var _jsxNamespace;
-        var _jsxFactoryEntity;
-        var outofbandVarianceMarkerHandler;
-        var subtypeRelation = ts.createMap();
-        var strictSubtypeRelation = ts.createMap();
-        var assignableRelation = ts.createMap();
-        var comparableRelation = ts.createMap();
-        var identityRelation = ts.createMap();
-        var enumRelation = ts.createMap();
-        var builtinGlobals = ts.createSymbolTable();
-        builtinGlobals.set(undefinedSymbol.escapedName, undefinedSymbol);
-        initializeTypeChecker();
-        return checker;
-        function getJsxNamespace(location) {
-            if (location) {
-                var file = ts.getSourceFileOfNode(location);
-                if (file) {
-                    if (file.localJsxNamespace) {
-                        return file.localJsxNamespace;
-                    }
-                    var jsxPragma = file.pragmas.get("jsx");
-                    if (jsxPragma) {
-                        var chosenpragma = ts.isArray(jsxPragma) ? jsxPragma[0] : jsxPragma;
-                        file.localJsxFactory = ts.parseIsolatedEntityName(chosenpragma.arguments.factory, languageVersion);
-                        ts.visitNode(file.localJsxFactory, markAsSynthetic);
-                        if (file.localJsxFactory) {
-                            return file.localJsxNamespace = ts.getFirstIdentifier(file.localJsxFactory).escapedText;
-                        }
-                    }
+        return result;
+    }
+    ts.resolveModuleName = resolveModuleName;
+    /**
+     * Any module resolution kind can be augmented with optional settings: 'baseUrl', 'paths' and 'rootDirs' - they are used to
+     * mitigate differences between design time structure of the project and its runtime counterpart so the same import name
+     * can be resolved successfully by TypeScript compiler and runtime module loader.
+     * If these settings are set then loading procedure will try to use them to resolve module name and it can of failure it will
+     * fallback to standard resolution routine.
+     *
+     * - baseUrl - this setting controls how non-relative module names are resolved. If this setting is specified then non-relative
+     * names will be resolved relative to baseUrl: i.e. if baseUrl is '/a/b' then candidate location to resolve module name 'c/d' will
+     * be '/a/b/c/d'
+     * - paths - this setting can only be used when baseUrl is specified. allows to tune how non-relative module names
+     * will be resolved based on the content of the module name.
+     * Structure of 'paths' compiler options
+     * 'paths': {
+     *    pattern-1: [...substitutions],
+     *    pattern-2: [...substitutions],
+     *    ...
+     *    pattern-n: [...substitutions]
+     * }
+     * Pattern here is a string that can contain zero or one '*' character. During module resolution module name will be matched against
+     * all patterns in the list. Matching for patterns that don't contain '*' means that module name must be equal to pattern respecting the case.
+     * If pattern contains '*' then to match pattern "<prefix>*<suffix>" module name must start with the <prefix> and end with <suffix>.
+     * <MatchedStar> denotes part of the module name between <prefix> and <suffix>.
+     * If module name can be matches with multiple patterns then pattern with the longest prefix will be picked.
+     * After selecting pattern we'll use list of substitutions to get candidate locations of the module and the try to load module
+     * from the candidate location.
+     * Substitution is a string that can contain zero or one '*'. To get candidate location from substitution we'll pick every
+     * substitution in the list and replace '*' with <MatchedStar> string. If candidate location is not rooted it
+     * will be converted to absolute using baseUrl.
+     * For example:
+     * baseUrl: /a/b/c
+     * "paths": {
+     *     // match all module names
+     *     "*": [
+     *         "*",        // use matched name as is,
+     *                     // <matched name> will be looked as /a/b/c/<matched name>
+     *
+     *         "folder1/*" // substitution will convert matched name to 'folder1/<matched name>',
+     *                     // since it is not rooted then final candidate location will be /a/b/c/folder1/<matched name>
+     *     ],
+     *     // match module names that start with 'components/'
+     *     "components/*": [ "/root/components/*" ] // substitution will convert /components/folder1/<matched name> to '/root/components/folder1/<matched name>',
+     *                                              // it is rooted so it will be final candidate location
+     * }
+     *
+     * 'rootDirs' allows the project to be spreaded across multiple locations and resolve modules with relative names as if
+     * they were in the same location. For example lets say there are two files
+     * '/local/src/content/file1.ts'
+     * '/shared/components/contracts/src/content/protocols/file2.ts'
+     * After bundling content of '/shared/components/contracts/src' will be merged with '/local/src' so
+     * if file1 has the following import 'import {x} from "./protocols/file2"' it will be resolved successfully in runtime.
+     * 'rootDirs' provides the way to tell compiler that in order to get the whole project it should behave as if content of all
+     * root dirs were merged together.
+     * I.e. for the example above 'rootDirs' will have two entries: [ '/local/src', '/shared/components/contracts/src' ].
+     * Compiler will first convert './protocols/file2' into absolute path relative to the location of containing file:
+     * '/local/src/content/protocols/file2' and try to load it - failure.
+     * Then it will search 'rootDirs' looking for a longest matching prefix of this absolute path and if such prefix is found - absolute path will
+     * be converted to a path relative to found rootDir entry './content/protocols/file2' (*). As a last step compiler will check all remaining
+     * entries in 'rootDirs', use them to build absolute path out of (*) and try to resolve module from this location.
+     */
+    function tryLoadModuleUsingOptionalResolutionSettings(extensions, moduleName, containingDirectory, loader, state) {
+        var resolved = tryLoadModuleUsingPathsIfEligible(extensions, moduleName, loader, state);
+        if (resolved)
+            return resolved.value;
+        if (!ts.isExternalModuleNameRelative(moduleName)) {
+            return tryLoadModuleUsingBaseUrl(extensions, moduleName, loader, state);
+        }
+        else {
+            return tryLoadModuleUsingRootDirs(extensions, moduleName, containingDirectory, loader, state);
+        }
+    }
+    function tryLoadModuleUsingPathsIfEligible(extensions, moduleName, loader, state) {
+        var _a = state.compilerOptions, baseUrl = _a.baseUrl, paths = _a.paths;
+        if (paths && !ts.pathIsRelative(moduleName)) {
+            if (state.traceEnabled) {
+                if (baseUrl) {
+                    trace(state.host, ts.Diagnostics.baseUrl_option_is_set_to_0_using_this_value_to_resolve_non_relative_module_name_1, baseUrl, moduleName);
                 }
+                trace(state.host, ts.Diagnostics.paths_option_is_specified_looking_for_a_pattern_to_match_module_name_0, moduleName);
             }
-            if (!_jsxNamespace) {
-                _jsxNamespace = "React";
-                if (compilerOptions.jsxFactory) {
-                    _jsxFactoryEntity = ts.parseIsolatedEntityName(compilerOptions.jsxFactory, languageVersion);
-                    ts.visitNode(_jsxFactoryEntity, markAsSynthetic);
-                    if (_jsxFactoryEntity) {
-                        _jsxNamespace = ts.getFirstIdentifier(_jsxFactoryEntity).escapedText;
-                    }
+            var baseDirectory = ts.getPathsBasePath(state.compilerOptions, state.host); // Always defined when 'paths' is defined
+            return tryLoadModuleUsingPaths(extensions, moduleName, baseDirectory, paths, loader, /*onlyRecordFailures*/ false, state);
+        }
+    }
+    function tryLoadModuleUsingRootDirs(extensions, moduleName, containingDirectory, loader, state) {
+        if (!state.compilerOptions.rootDirs) {
+            return undefined;
+        }
+        if (state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.rootDirs_option_is_set_using_it_to_resolve_relative_module_name_0, moduleName);
+        }
+        var candidate = ts.normalizePath(ts.combinePaths(containingDirectory, moduleName));
+        var matchedRootDir;
+        var matchedNormalizedPrefix;
+        for (var _i = 0, _a = state.compilerOptions.rootDirs; _i < _a.length; _i++) {
+            var rootDir = _a[_i];
+            // rootDirs are expected to be absolute
+            // in case of tsconfig.json this will happen automatically - compiler will expand relative names
+            // using location of tsconfig.json as base location
+            var normalizedRoot = ts.normalizePath(rootDir);
+            if (!ts.endsWith(normalizedRoot, ts.directorySeparator)) {
+                normalizedRoot += ts.directorySeparator;
+            }
+            var isLongestMatchingPrefix = ts.startsWith(candidate, normalizedRoot) &&
+                (matchedNormalizedPrefix === undefined || matchedNormalizedPrefix.length < normalizedRoot.length);
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Checking_if_0_is_the_longest_matching_prefix_for_1_2, normalizedRoot, candidate, isLongestMatchingPrefix);
+            }
+            if (isLongestMatchingPrefix) {
+                matchedNormalizedPrefix = normalizedRoot;
+                matchedRootDir = rootDir;
+            }
+        }
+        if (matchedNormalizedPrefix) {
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Longest_matching_prefix_for_0_is_1, candidate, matchedNormalizedPrefix);
+            }
+            var suffix = candidate.substr(matchedNormalizedPrefix.length);
+            // first - try to load from a initial location
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Loading_0_from_the_root_dir_1_candidate_location_2, suffix, matchedNormalizedPrefix, candidate);
+            }
+            var resolvedFileName = loader(extensions, candidate, !ts.directoryProbablyExists(containingDirectory, state.host), state);
+            if (resolvedFileName) {
+                return resolvedFileName;
+            }
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Trying_other_entries_in_rootDirs);
+            }
+            // then try to resolve using remaining entries in rootDirs
+            for (var _b = 0, _c = state.compilerOptions.rootDirs; _b < _c.length; _b++) {
+                var rootDir = _c[_b];
+                if (rootDir === matchedRootDir) {
+                    // skip the initially matched entry
+                    continue;
                 }
-                else if (compilerOptions.reactNamespace) {
-                    _jsxNamespace = ts.escapeLeadingUnderscores(compilerOptions.reactNamespace);
+                var candidate_1 = ts.combinePaths(ts.normalizePath(rootDir), suffix);
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.Loading_0_from_the_root_dir_1_candidate_location_2, suffix, rootDir, candidate_1);
+                }
+                var baseDirectory = ts.getDirectoryPath(candidate_1);
+                var resolvedFileName_1 = loader(extensions, candidate_1, !ts.directoryProbablyExists(baseDirectory, state.host), state);
+                if (resolvedFileName_1) {
+                    return resolvedFileName_1;
                 }
             }
-            if (!_jsxFactoryEntity) {
-                _jsxFactoryEntity = ts.createQualifiedName(ts.createIdentifier(ts.unescapeLeadingUnderscores(_jsxNamespace)), "createElement");
-            }
-            return _jsxNamespace;
-            function markAsSynthetic(node) {
-                node.pos = -1;
-                node.end = -1;
-                return ts.visitEachChild(node, markAsSynthetic, ts.nullTransformationContext);
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Module_resolution_using_rootDirs_has_failed);
             }
         }
-        function getEmitResolver(sourceFile, cancellationToken) {
-            // Ensure we have all the type information in place for this file so that all the
-            // emitter questions of this resolver will return the right information.
-            getDiagnostics(sourceFile, cancellationToken);
-            return emitResolver;
+        return undefined;
+    }
+    function tryLoadModuleUsingBaseUrl(extensions, moduleName, loader, state) {
+        var baseUrl = state.compilerOptions.baseUrl;
+        if (!baseUrl) {
+            return undefined;
         }
-        function lookupOrIssueError(location, message, arg0, arg1, arg2, arg3) {
-            var diagnostic = location
-                ? ts.createDiagnosticForNode(location, message, arg0, arg1, arg2, arg3)
-                : ts.createCompilerDiagnostic(message, arg0, arg1, arg2, arg3);
-            var existing = diagnostics.lookup(diagnostic);
-            if (existing) {
-                return existing;
+        if (state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.baseUrl_option_is_set_to_0_using_this_value_to_resolve_non_relative_module_name_1, baseUrl, moduleName);
+        }
+        var candidate = ts.normalizePath(ts.combinePaths(baseUrl, moduleName));
+        if (state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.Resolving_module_name_0_relative_to_base_url_1_2, moduleName, baseUrl, candidate);
+        }
+        return loader(extensions, candidate, !ts.directoryProbablyExists(ts.getDirectoryPath(candidate), state.host), state);
+    }
+    /**
+     * Expose resolution logic to allow us to use Node module resolution logic from arbitrary locations.
+     * No way to do this with `require()`: https://github.com/nodejs/node/issues/5963
+     * Throws an error if the module can't be resolved.
+     */
+    /* @internal */
+    function resolveJSModule(moduleName, initialDir, host) {
+        var _a = tryResolveJSModuleWorker(moduleName, initialDir, host), resolvedModule = _a.resolvedModule, failedLookupLocations = _a.failedLookupLocations;
+        if (!resolvedModule) {
+            throw new Error("Could not resolve JS module '" + moduleName + "' starting at '" + initialDir + "'. Looked in: " + failedLookupLocations.join(", "));
+        }
+        return resolvedModule.resolvedFileName;
+    }
+    ts.resolveJSModule = resolveJSModule;
+    /* @internal */
+    function tryResolveJSModule(moduleName, initialDir, host) {
+        var resolvedModule = tryResolveJSModuleWorker(moduleName, initialDir, host).resolvedModule;
+        return resolvedModule && resolvedModule.resolvedFileName;
+    }
+    ts.tryResolveJSModule = tryResolveJSModule;
+    var jsOnlyExtensions = [Extensions.JavaScript];
+    var tsExtensions = [Extensions.TypeScript, Extensions.JavaScript];
+    var tsPlusJsonExtensions = __spreadArrays(tsExtensions, [Extensions.Json]);
+    var tsconfigExtensions = [Extensions.TSConfig];
+    function tryResolveJSModuleWorker(moduleName, initialDir, host) {
+        return nodeModuleNameResolverWorker(moduleName, initialDir, { moduleResolution: ts.ModuleResolutionKind.NodeJs, allowJs: true }, host, /*cache*/ undefined, jsOnlyExtensions, /*redirectedReferences*/ undefined);
+    }
+    function nodeModuleNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference, lookupConfig) {
+        return nodeModuleNameResolverWorker(moduleName, ts.getDirectoryPath(containingFile), compilerOptions, host, cache, lookupConfig ? tsconfigExtensions : (compilerOptions.resolveJsonModule ? tsPlusJsonExtensions : tsExtensions), redirectedReference);
+    }
+    ts.nodeModuleNameResolver = nodeModuleNameResolver;
+    function nodeModuleNameResolverWorker(moduleName, containingDirectory, compilerOptions, host, cache, extensions, redirectedReference) {
+        var _a, _b;
+        var traceEnabled = isTraceEnabled(compilerOptions, host);
+        var failedLookupLocations = [];
+        var state = { compilerOptions: compilerOptions, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
+        var result = ts.forEach(extensions, function (ext) { return tryResolve(ext); });
+        return createResolvedModuleWithFailedLookupLocations((_a = result === null || result === void 0 ? void 0 : result.value) === null || _a === void 0 ? void 0 : _a.resolved, (_b = result === null || result === void 0 ? void 0 : result.value) === null || _b === void 0 ? void 0 : _b.isExternalLibraryImport, failedLookupLocations, state.resultFromCache);
+        function tryResolve(extensions) {
+            var loader = function (extensions, candidate, onlyRecordFailures, state) { return nodeLoadModuleByRelativeName(extensions, candidate, onlyRecordFailures, state, /*considerPackageJson*/ true); };
+            var resolved = tryLoadModuleUsingOptionalResolutionSettings(extensions, moduleName, containingDirectory, loader, state);
+            if (resolved) {
+                return toSearchResult({ resolved: resolved, isExternalLibraryImport: pathContainsNodeModules(resolved.path) });
+            }
+            if (!ts.isExternalModuleNameRelative(moduleName)) {
+                if (traceEnabled) {
+                    trace(host, ts.Diagnostics.Loading_module_0_from_node_modules_folder_target_file_type_1, moduleName, Extensions[extensions]);
+                }
+                var resolved_1 = loadModuleFromNearestNodeModulesDirectory(extensions, moduleName, containingDirectory, state, cache, redirectedReference);
+                if (!resolved_1)
+                    return undefined;
+                var resolvedValue = resolved_1.value;
+                if (!compilerOptions.preserveSymlinks && resolvedValue && !resolvedValue.originalPath) {
+                    var path = realPath(resolvedValue.path, host, traceEnabled);
+                    var originalPath = path === resolvedValue.path ? undefined : resolvedValue.path;
+                    resolvedValue = __assign(__assign({}, resolvedValue), { path: path, originalPath: originalPath });
+                }
+                // For node_modules lookups, get the real path so that multiple accesses to an `npm link`-ed module do not create duplicate files.
+                return { value: resolvedValue && { resolved: resolvedValue, isExternalLibraryImport: true } };
             }
             else {
-                diagnostics.add(diagnostic);
-                return diagnostic;
+                var _a = ts.normalizePathAndParts(ts.combinePaths(containingDirectory, moduleName)), candidate = _a.path, parts = _a.parts;
+                var resolved_2 = nodeLoadModuleByRelativeName(extensions, candidate, /*onlyRecordFailures*/ false, state, /*considerPackageJson*/ true);
+                // Treat explicit "node_modules" import as an external library import.
+                return resolved_2 && toSearchResult({ resolved: resolved_2, isExternalLibraryImport: ts.contains(parts, "node_modules") });
             }
         }
-        function error(location, message, arg0, arg1, arg2, arg3) {
-            var diagnostic = location
-                ? ts.createDiagnosticForNode(location, message, arg0, arg1, arg2, arg3)
-                : ts.createCompilerDiagnostic(message, arg0, arg1, arg2, arg3);
-            diagnostics.add(diagnostic);
-            return diagnostic;
+    }
+    function realPath(path, host, traceEnabled) {
+        if (!host.realpath) {
+            return path;
         }
-        function addErrorOrSuggestion(isError, diagnostic) {
-            if (isError) {
-                diagnostics.add(diagnostic);
+        var real = ts.normalizePath(host.realpath(path));
+        if (traceEnabled) {
+            trace(host, ts.Diagnostics.Resolving_real_path_for_0_result_1, path, real);
+        }
+        ts.Debug.assert(host.fileExists(real), path + " linked to nonexistent file " + real);
+        return real;
+    }
+    function nodeLoadModuleByRelativeName(extensions, candidate, onlyRecordFailures, state, considerPackageJson) {
+        if (state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.Loading_module_as_file_Slash_folder_candidate_module_location_0_target_file_type_1, candidate, Extensions[extensions]);
+        }
+        if (!ts.hasTrailingDirectorySeparator(candidate)) {
+            if (!onlyRecordFailures) {
+                var parentOfCandidate = ts.getDirectoryPath(candidate);
+                if (!ts.directoryProbablyExists(parentOfCandidate, state.host)) {
+                    if (state.traceEnabled) {
+                        trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, parentOfCandidate);
+                    }
+                    onlyRecordFailures = true;
+                }
             }
-            else {
-                suggestionDiagnostics.add(__assign(__assign({}, diagnostic), { category: ts.DiagnosticCategory.Suggestion }));
+            var resolvedFromFile = loadModuleFromFile(extensions, candidate, onlyRecordFailures, state);
+            if (resolvedFromFile) {
+                var packageDirectory = considerPackageJson ? parseNodeModuleFromPath(resolvedFromFile) : undefined;
+                var packageInfo = packageDirectory ? getPackageJsonInfo(packageDirectory, /*onlyRecordFailures*/ false, state) : undefined;
+                return withPackageId(packageInfo, resolvedFromFile);
             }
         }
-        function errorOrSuggestion(isError, location, message, arg0, arg1, arg2, arg3) {
-            addErrorOrSuggestion(isError, "message" in message ? ts.createDiagnosticForNode(location, message, arg0, arg1, arg2, arg3) : ts.createDiagnosticForNodeFromMessageChain(location, message)); // eslint-disable-line no-in-operator
-        }
-        function errorAndMaybeSuggestAwait(location, maybeMissingAwait, message, arg0, arg1, arg2, arg3) {
-            var diagnostic = error(location, message, arg0, arg1, arg2, arg3);
-            if (maybeMissingAwait) {
-                var related = ts.createDiagnosticForNode(location, ts.Diagnostics.Did_you_forget_to_use_await);
-                ts.addRelatedInfo(diagnostic, related);
+        if (!onlyRecordFailures) {
+            var candidateExists = ts.directoryProbablyExists(candidate, state.host);
+            if (!candidateExists) {
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, candidate);
+                }
+                onlyRecordFailures = true;
             }
-            return diagnostic;
         }
-        function createSymbol(flags, name, checkFlags) {
-            symbolCount++;
-            var symbol = (new Symbol(flags | 33554432 /* Transient */, name));
-            symbol.checkFlags = checkFlags || 0;
-            return symbol;
+        return loadNodeModuleFromDirectory(extensions, candidate, onlyRecordFailures, state, considerPackageJson);
+    }
+    /*@internal*/
+    ts.nodeModulesPathPart = "/node_modules/";
+    /*@internal*/
+    function pathContainsNodeModules(path) {
+        return ts.stringContains(path, ts.nodeModulesPathPart);
+    }
+    ts.pathContainsNodeModules = pathContainsNodeModules;
+    /**
+     * This will be called on the successfully resolved path from `loadModuleFromFile`.
+     * (Not neeeded for `loadModuleFromNodeModules` as that looks up the `package.json` as part of resolution.)
+     *
+     * packageDirectory is the directory of the package itself.
+     *   For `blah/node_modules/foo/index.d.ts` this is packageDirectory: "foo"
+     *   For `/node_modules/foo/bar.d.ts` this is packageDirectory: "foo"
+     *   For `/node_modules/@types/foo/bar/index.d.ts` this is packageDirectory: "@types/foo"
+     *   For `/node_modules/foo/bar/index.d.ts` this is packageDirectory: "foo"
+     */
+    function parseNodeModuleFromPath(resolved) {
+        var path = ts.normalizePath(resolved.path);
+        var idx = path.lastIndexOf(ts.nodeModulesPathPart);
+        if (idx === -1) {
+            return undefined;
         }
-        function getExcludedSymbolFlags(flags) {
-            var result = 0;
-            if (flags & 2 /* BlockScopedVariable */)
-                result |= 111551 /* BlockScopedVariableExcludes */;
-            if (flags & 1 /* FunctionScopedVariable */)
-                result |= 111550 /* FunctionScopedVariableExcludes */;
-            if (flags & 4 /* Property */)
-                result |= 0 /* PropertyExcludes */;
-            if (flags & 8 /* EnumMember */)
-                result |= 900095 /* EnumMemberExcludes */;
-            if (flags & 16 /* Function */)
-                result |= 110991 /* FunctionExcludes */;
-            if (flags & 32 /* Class */)
-                result |= 899503 /* ClassExcludes */;
-            if (flags & 64 /* Interface */)
-                result |= 788872 /* InterfaceExcludes */;
-            if (flags & 256 /* RegularEnum */)
-                result |= 899327 /* RegularEnumExcludes */;
-            if (flags & 128 /* ConstEnum */)
-                result |= 899967 /* ConstEnumExcludes */;
-            if (flags & 512 /* ValueModule */)
-                result |= 110735 /* ValueModuleExcludes */;
-            if (flags & 8192 /* Method */)
-                result |= 103359 /* MethodExcludes */;
-            if (flags & 32768 /* GetAccessor */)
-                result |= 46015 /* GetAccessorExcludes */;
-            if (flags & 65536 /* SetAccessor */)
-                result |= 78783 /* SetAccessorExcludes */;
-            if (flags & 262144 /* TypeParameter */)
-                result |= 526824 /* TypeParameterExcludes */;
-            if (flags & 524288 /* TypeAlias */)
-                result |= 788968 /* TypeAliasExcludes */;
-            if (flags & 2097152 /* Alias */)
-                result |= 2097152 /* AliasExcludes */;
-            return result;
+        var indexAfterNodeModules = idx + ts.nodeModulesPathPart.length;
+        var indexAfterPackageName = moveToNextDirectorySeparatorIfAvailable(path, indexAfterNodeModules);
+        if (path.charCodeAt(indexAfterNodeModules) === 64 /* at */) {
+            indexAfterPackageName = moveToNextDirectorySeparatorIfAvailable(path, indexAfterPackageName);
         }
-        function recordMergedSymbol(target, source) {
-            if (!source.mergeId) {
-                source.mergeId = nextMergeId;
-                nextMergeId++;
+        return path.slice(0, indexAfterPackageName);
+    }
+    function moveToNextDirectorySeparatorIfAvailable(path, prevSeparatorIndex) {
+        var nextSeparatorIndex = path.indexOf(ts.directorySeparator, prevSeparatorIndex + 1);
+        return nextSeparatorIndex === -1 ? prevSeparatorIndex : nextSeparatorIndex;
+    }
+    function loadModuleFromFileNoPackageId(extensions, candidate, onlyRecordFailures, state) {
+        return noPackageId(loadModuleFromFile(extensions, candidate, onlyRecordFailures, state));
+    }
+    /**
+     * @param {boolean} onlyRecordFailures - if true then function won't try to actually load files but instead record all attempts as failures. This flag is necessary
+     * in cases when we know upfront that all load attempts will fail (because containing folder does not exists) however we still need to record all failed lookup locations.
+     */
+    function loadModuleFromFile(extensions, candidate, onlyRecordFailures, state) {
+        if (extensions === Extensions.Json || extensions === Extensions.TSConfig) {
+            var extensionLess = ts.tryRemoveExtension(candidate, ".json" /* Json */);
+            return (extensionLess === undefined && extensions === Extensions.Json) ? undefined : tryAddingExtensions(extensionLess || candidate, extensions, onlyRecordFailures, state);
+        }
+        // First, try adding an extension. An import of "foo" could be matched by a file "foo.ts", or "foo.js" by "foo.js.ts"
+        var resolvedByAddingExtension = tryAddingExtensions(candidate, extensions, onlyRecordFailures, state);
+        if (resolvedByAddingExtension) {
+            return resolvedByAddingExtension;
+        }
+        // If that didn't work, try stripping a ".js" or ".jsx" extension and replacing it with a TypeScript one;
+        // e.g. "./foo.js" can be matched by "./foo.ts" or "./foo.d.ts"
+        if (ts.hasJSFileExtension(candidate)) {
+            var extensionless = ts.removeFileExtension(candidate);
+            if (state.traceEnabled) {
+                var extension = candidate.substring(extensionless.length);
+                trace(state.host, ts.Diagnostics.File_name_0_has_a_1_extension_stripping_it, candidate, extension);
             }
-            mergedSymbols[source.mergeId] = target;
+            return tryAddingExtensions(extensionless, extensions, onlyRecordFailures, state);
         }
-        function cloneSymbol(symbol) {
-            var result = createSymbol(symbol.flags, symbol.escapedName);
-            result.declarations = symbol.declarations ? symbol.declarations.slice() : [];
-            result.parent = symbol.parent;
-            if (symbol.valueDeclaration)
-                result.valueDeclaration = symbol.valueDeclaration;
-            if (symbol.constEnumOnlyModule)
-                result.constEnumOnlyModule = true;
-            if (symbol.members)
-                result.members = ts.cloneMap(symbol.members);
-            if (symbol.exports)
-                result.exports = ts.cloneMap(symbol.exports);
-            recordMergedSymbol(result, symbol);
-            return result;
+    }
+    /** Try to return an existing file that adds one of the `extensions` to `candidate`. */
+    function tryAddingExtensions(candidate, extensions, onlyRecordFailures, state) {
+        if (!onlyRecordFailures) {
+            // check if containing folder exists - if it doesn't then just record failures for all supported extensions without disk probing
+            var directory = ts.getDirectoryPath(candidate);
+            if (directory) {
+                onlyRecordFailures = !ts.directoryProbablyExists(directory, state.host);
+            }
         }
-        /**
-         * Note: if target is transient, then it is mutable, and mergeSymbol with both mutate and return it.
-         * If target is not transient, mergeSymbol will produce a transient clone, mutate that and return it.
-         */
-        function mergeSymbol(target, source, unidirectional) {
-            if (unidirectional === void 0) { unidirectional = false; }
-            if (!(target.flags & getExcludedSymbolFlags(source.flags)) ||
-                (source.flags | target.flags) & 67108864 /* Assignment */) {
-                if (source === target) {
-                    // This can happen when an export assigned namespace exports something also erroneously exported at the top level
-                    // See `declarationFileNoCrashOnExtraExportModifier` for an example
-                    return target;
-                }
-                if (!(target.flags & 33554432 /* Transient */)) {
-                    var resolvedTarget = resolveSymbol(target);
-                    if (resolvedTarget === unknownSymbol) {
-                        return source;
-                    }
-                    target = cloneSymbol(resolvedTarget);
-                }
-                // Javascript static-property-assignment declarations always merge, even though they are also values
-                if (source.flags & 512 /* ValueModule */ && target.flags & 512 /* ValueModule */ && target.constEnumOnlyModule && !source.constEnumOnlyModule) {
-                    // reset flag when merging instantiated module into value module that has only const enums
-                    target.constEnumOnlyModule = false;
-                }
-                target.flags |= source.flags;
-                if (source.valueDeclaration) {
-                    ts.setValueDeclaration(target, source.valueDeclaration);
-                }
-                ts.addRange(target.declarations, source.declarations);
-                if (source.members) {
-                    if (!target.members)
-                        target.members = ts.createSymbolTable();
-                    mergeSymbolTable(target.members, source.members, unidirectional);
-                }
-                if (source.exports) {
-                    if (!target.exports)
-                        target.exports = ts.createSymbolTable();
-                    mergeSymbolTable(target.exports, source.exports, unidirectional);
-                }
-                if (!unidirectional) {
-                    recordMergedSymbol(target, source);
+        switch (extensions) {
+            case Extensions.DtsOnly:
+                return tryExtension(".d.ts" /* Dts */);
+            case Extensions.TypeScript:
+                return tryExtension(".ts" /* Ts */) || tryExtension(".tsx" /* Tsx */) || tryExtension(".d.ts" /* Dts */);
+            case Extensions.JavaScript:
+                return tryExtension(".js" /* Js */) || tryExtension(".jsx" /* Jsx */);
+            case Extensions.TSConfig:
+            case Extensions.Json:
+                return tryExtension(".json" /* Json */);
+        }
+        function tryExtension(ext) {
+            var path = tryFile(candidate + ext, onlyRecordFailures, state);
+            return path === undefined ? undefined : { path: path, ext: ext };
+        }
+    }
+    /** Return the file if it exists. */
+    function tryFile(fileName, onlyRecordFailures, state) {
+        if (!onlyRecordFailures) {
+            if (state.host.fileExists(fileName)) {
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.File_0_exist_use_it_as_a_name_resolution_result, fileName);
                 }
+                return fileName;
             }
-            else if (target.flags & 1024 /* NamespaceModule */) {
-                // Do not report an error when merging `var globalThis` with the built-in `globalThis`,
-                // as we will already report a "Declaration name conflicts..." error, and this error
-                // won't make much sense.
-                if (target !== globalThisSymbol) {
-                    error(ts.getNameOfDeclaration(source.declarations[0]), ts.Diagnostics.Cannot_augment_module_0_with_value_exports_because_it_resolves_to_a_non_module_entity, symbolToString(target));
+            else {
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.File_0_does_not_exist, fileName);
                 }
             }
-            else { // error
-                var isEitherEnum = !!(target.flags & 384 /* Enum */ || source.flags & 384 /* Enum */);
-                var isEitherBlockScoped_1 = !!(target.flags & 2 /* BlockScopedVariable */ || source.flags & 2 /* BlockScopedVariable */);
-                var message = isEitherEnum
-                    ? ts.Diagnostics.Enum_declarations_can_only_merge_with_namespace_or_other_enum_declarations
-                    : isEitherBlockScoped_1
-                        ? ts.Diagnostics.Cannot_redeclare_block_scoped_variable_0
-                        : ts.Diagnostics.Duplicate_identifier_0;
-                var sourceSymbolFile = source.declarations && ts.getSourceFileOfNode(source.declarations[0]);
-                var targetSymbolFile = target.declarations && ts.getSourceFileOfNode(target.declarations[0]);
-                var symbolName_1 = symbolToString(source);
-                // Collect top-level duplicate identifier errors into one mapping, so we can then merge their diagnostics if there are a bunch
-                if (sourceSymbolFile && targetSymbolFile && amalgamatedDuplicates && !isEitherEnum && sourceSymbolFile !== targetSymbolFile) {
-                    var firstFile_1 = ts.comparePaths(sourceSymbolFile.path, targetSymbolFile.path) === -1 /* LessThan */ ? sourceSymbolFile : targetSymbolFile;
-                    var secondFile_1 = firstFile_1 === sourceSymbolFile ? targetSymbolFile : sourceSymbolFile;
-                    var filesDuplicates = ts.getOrUpdate(amalgamatedDuplicates, firstFile_1.path + "|" + secondFile_1.path, function () {
-                        return ({ firstFile: firstFile_1, secondFile: secondFile_1, conflictingSymbols: ts.createMap() });
-                    });
-                    var conflictingSymbolInfo = ts.getOrUpdate(filesDuplicates.conflictingSymbols, symbolName_1, function () {
-                        return ({ isBlockScoped: isEitherBlockScoped_1, firstFileLocations: [], secondFileLocations: [] });
-                    });
-                    addDuplicateLocations(conflictingSymbolInfo.firstFileLocations, source);
-                    addDuplicateLocations(conflictingSymbolInfo.secondFileLocations, target);
+        }
+        state.failedLookupLocations.push(fileName);
+        return undefined;
+    }
+    function loadNodeModuleFromDirectory(extensions, candidate, onlyRecordFailures, state, considerPackageJson) {
+        if (considerPackageJson === void 0) { considerPackageJson = true; }
+        var packageInfo = considerPackageJson ? getPackageJsonInfo(candidate, onlyRecordFailures, state) : undefined;
+        var packageJsonContent = packageInfo && packageInfo.packageJsonContent;
+        var versionPaths = packageInfo && packageInfo.versionPaths;
+        return withPackageId(packageInfo, loadNodeModuleFromDirectoryWorker(extensions, candidate, onlyRecordFailures, state, packageJsonContent, versionPaths));
+    }
+    function getPackageJsonInfo(packageDirectory, onlyRecordFailures, state) {
+        var host = state.host, traceEnabled = state.traceEnabled;
+        var directoryExists = !onlyRecordFailures && ts.directoryProbablyExists(packageDirectory, host);
+        var packageJsonPath = ts.combinePaths(packageDirectory, "package.json");
+        if (directoryExists && host.fileExists(packageJsonPath)) {
+            var packageJsonContent = ts.readJson(packageJsonPath, host);
+            if (traceEnabled) {
+                trace(host, ts.Diagnostics.Found_package_json_at_0, packageJsonPath);
+            }
+            var versionPaths = readPackageJsonTypesVersionPaths(packageJsonContent, state);
+            return { packageDirectory: packageDirectory, packageJsonContent: packageJsonContent, versionPaths: versionPaths };
+        }
+        else {
+            if (directoryExists && traceEnabled) {
+                trace(host, ts.Diagnostics.File_0_does_not_exist, packageJsonPath);
+            }
+            // record package json as one of failed lookup locations - in the future if this file will appear it will invalidate resolution results
+            state.failedLookupLocations.push(packageJsonPath);
+        }
+    }
+    function loadNodeModuleFromDirectoryWorker(extensions, candidate, onlyRecordFailures, state, jsonContent, versionPaths) {
+        var packageFile;
+        if (jsonContent) {
+            switch (extensions) {
+                case Extensions.JavaScript:
+                case Extensions.Json:
+                    packageFile = readPackageJsonMainField(jsonContent, candidate, state);
+                    break;
+                case Extensions.TypeScript:
+                    // When resolving typescript modules, try resolving using main field as well
+                    packageFile = readPackageJsonTypesFields(jsonContent, candidate, state) || readPackageJsonMainField(jsonContent, candidate, state);
+                    break;
+                case Extensions.DtsOnly:
+                    packageFile = readPackageJsonTypesFields(jsonContent, candidate, state);
+                    break;
+                case Extensions.TSConfig:
+                    packageFile = readPackageJsonTSConfigField(jsonContent, candidate, state);
+                    break;
+                default:
+                    return ts.Debug.assertNever(extensions);
+            }
+        }
+        var loader = function (extensions, candidate, onlyRecordFailures, state) {
+            var fromFile = tryFile(candidate, onlyRecordFailures, state);
+            if (fromFile) {
+                var resolved = resolvedIfExtensionMatches(extensions, fromFile);
+                if (resolved) {
+                    return noPackageId(resolved);
                 }
-                else {
-                    addDuplicateDeclarationErrorsForSymbols(source, message, symbolName_1, target);
-                    addDuplicateDeclarationErrorsForSymbols(target, message, symbolName_1, source);
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.File_0_has_an_unsupported_extension_so_skipping_it, fromFile);
                 }
             }
-            return target;
-            function addDuplicateLocations(locs, symbol) {
-                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                    var decl = _a[_i];
-                    ts.pushIfUnique(locs, decl);
-                }
+            // Even if extensions is DtsOnly, we can still look up a .ts file as a result of package.json "types"
+            var nextExtensions = extensions === Extensions.DtsOnly ? Extensions.TypeScript : extensions;
+            // Don't do package.json lookup recursively, because Node.js' package lookup doesn't.
+            return nodeLoadModuleByRelativeName(nextExtensions, candidate, onlyRecordFailures, state, /*considerPackageJson*/ false);
+        };
+        var onlyRecordFailuresForPackageFile = packageFile ? !ts.directoryProbablyExists(ts.getDirectoryPath(packageFile), state.host) : undefined;
+        var onlyRecordFailuresForIndex = onlyRecordFailures || !ts.directoryProbablyExists(candidate, state.host);
+        var indexPath = ts.combinePaths(candidate, extensions === Extensions.TSConfig ? "tsconfig" : "index");
+        if (versionPaths && (!packageFile || ts.containsPath(candidate, packageFile))) {
+            var moduleName = ts.getRelativePathFromDirectory(candidate, packageFile || indexPath, /*ignoreCase*/ false);
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_entry_0_that_matches_compiler_version_1_looking_for_a_pattern_to_match_module_name_2, versionPaths.version, ts.version, moduleName);
+            }
+            var result = tryLoadModuleUsingPaths(extensions, moduleName, candidate, versionPaths.paths, loader, onlyRecordFailuresForPackageFile || onlyRecordFailuresForIndex, state);
+            if (result) {
+                return removeIgnoredPackageId(result.value);
             }
         }
-        function addDuplicateDeclarationErrorsForSymbols(target, message, symbolName, source) {
-            ts.forEach(target.declarations, function (node) {
-                addDuplicateDeclarationError(node, message, symbolName, source.declarations);
-            });
+        // It won't have a `packageId` set, because we disabled `considerPackageJson`.
+        var packageFileResult = packageFile && removeIgnoredPackageId(loader(extensions, packageFile, onlyRecordFailuresForPackageFile, state));
+        if (packageFileResult)
+            return packageFileResult;
+        return loadModuleFromFile(extensions, indexPath, onlyRecordFailuresForIndex, state);
+    }
+    /** Resolve from an arbitrarily specified file. Return `undefined` if it has an unsupported extension. */
+    function resolvedIfExtensionMatches(extensions, path) {
+        var ext = ts.tryGetExtensionFromPath(path);
+        return ext !== undefined && extensionIsOk(extensions, ext) ? { path: path, ext: ext } : undefined;
+    }
+    /** True if `extension` is one of the supported `extensions`. */
+    function extensionIsOk(extensions, extension) {
+        switch (extensions) {
+            case Extensions.JavaScript:
+                return extension === ".js" /* Js */ || extension === ".jsx" /* Jsx */;
+            case Extensions.TSConfig:
+            case Extensions.Json:
+                return extension === ".json" /* Json */;
+            case Extensions.TypeScript:
+                return extension === ".ts" /* Ts */ || extension === ".tsx" /* Tsx */ || extension === ".d.ts" /* Dts */;
+            case Extensions.DtsOnly:
+                return extension === ".d.ts" /* Dts */;
         }
-        function addDuplicateDeclarationError(node, message, symbolName, relatedNodes) {
-            var errorNode = (ts.getExpandoInitializer(node, /*isPrototypeAssignment*/ false) ? ts.getNameOfExpando(node) : ts.getNameOfDeclaration(node)) || node;
-            var err = lookupOrIssueError(errorNode, message, symbolName);
-            var _loop_6 = function (relatedNode) {
-                var adjustedNode = (ts.getExpandoInitializer(relatedNode, /*isPrototypeAssignment*/ false) ? ts.getNameOfExpando(relatedNode) : ts.getNameOfDeclaration(relatedNode)) || relatedNode;
-                if (adjustedNode === errorNode)
-                    return "continue";
-                err.relatedInformation = err.relatedInformation || [];
-                var leadingMessage = ts.createDiagnosticForNode(adjustedNode, ts.Diagnostics._0_was_also_declared_here, symbolName);
-                var followOnMessage = ts.createDiagnosticForNode(adjustedNode, ts.Diagnostics.and_here);
-                if (ts.length(err.relatedInformation) >= 5 || ts.some(err.relatedInformation, function (r) { return ts.compareDiagnostics(r, followOnMessage) === 0 /* EqualTo */ || ts.compareDiagnostics(r, leadingMessage) === 0 /* EqualTo */; }))
-                    return "continue";
-                ts.addRelatedInfo(err, !ts.length(err.relatedInformation) ? leadingMessage : followOnMessage);
-            };
-            for (var _i = 0, _a = relatedNodes || ts.emptyArray; _i < _a.length; _i++) {
-                var relatedNode = _a[_i];
-                _loop_6(relatedNode);
+    }
+    /* @internal */
+    function parsePackageName(moduleName) {
+        var idx = moduleName.indexOf(ts.directorySeparator);
+        if (moduleName[0] === "@") {
+            idx = moduleName.indexOf(ts.directorySeparator, idx + 1);
+        }
+        return idx === -1 ? { packageName: moduleName, rest: "" } : { packageName: moduleName.slice(0, idx), rest: moduleName.slice(idx + 1) };
+    }
+    ts.parsePackageName = parsePackageName;
+    function loadModuleFromNearestNodeModulesDirectory(extensions, moduleName, directory, state, cache, redirectedReference) {
+        return loadModuleFromNearestNodeModulesDirectoryWorker(extensions, moduleName, directory, state, /*typesScopeOnly*/ false, cache, redirectedReference);
+    }
+    function loadModuleFromNearestNodeModulesDirectoryTypesScope(moduleName, directory, state) {
+        // Extensions parameter here doesn't actually matter, because typesOnly ensures we're just doing @types lookup, which is always DtsOnly.
+        return loadModuleFromNearestNodeModulesDirectoryWorker(Extensions.DtsOnly, moduleName, directory, state, /*typesScopeOnly*/ true, /*cache*/ undefined, /*redirectedReference*/ undefined);
+    }
+    function loadModuleFromNearestNodeModulesDirectoryWorker(extensions, moduleName, directory, state, typesScopeOnly, cache, redirectedReference) {
+        var perModuleNameCache = cache && cache.getOrCreateCacheForModuleName(moduleName, redirectedReference);
+        return ts.forEachAncestorDirectory(ts.normalizeSlashes(directory), function (ancestorDirectory) {
+            if (ts.getBaseFileName(ancestorDirectory) !== "node_modules") {
+                var resolutionFromCache = tryFindNonRelativeModuleNameInCache(perModuleNameCache, moduleName, ancestorDirectory, state);
+                if (resolutionFromCache) {
+                    return resolutionFromCache;
+                }
+                return toSearchResult(loadModuleFromImmediateNodeModulesDirectory(extensions, moduleName, ancestorDirectory, state, typesScopeOnly));
             }
+        });
+    }
+    function loadModuleFromImmediateNodeModulesDirectory(extensions, moduleName, directory, state, typesScopeOnly) {
+        var nodeModulesFolder = ts.combinePaths(directory, "node_modules");
+        var nodeModulesFolderExists = ts.directoryProbablyExists(nodeModulesFolder, state.host);
+        if (!nodeModulesFolderExists && state.traceEnabled) {
+            trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, nodeModulesFolder);
         }
-        function combineSymbolTables(first, second) {
-            if (!ts.hasEntries(first))
-                return second;
-            if (!ts.hasEntries(second))
-                return first;
-            var combined = ts.createSymbolTable();
-            mergeSymbolTable(combined, first);
-            mergeSymbolTable(combined, second);
-            return combined;
+        var packageResult = typesScopeOnly ? undefined : loadModuleFromSpecificNodeModulesDirectory(extensions, moduleName, nodeModulesFolder, nodeModulesFolderExists, state);
+        if (packageResult) {
+            return packageResult;
         }
-        function mergeSymbolTable(target, source, unidirectional) {
-            if (unidirectional === void 0) { unidirectional = false; }
-            source.forEach(function (sourceSymbol, id) {
-                var targetSymbol = target.get(id);
-                target.set(id, targetSymbol ? mergeSymbol(targetSymbol, sourceSymbol, unidirectional) : sourceSymbol);
-            });
+        if (extensions === Extensions.TypeScript || extensions === Extensions.DtsOnly) {
+            var nodeModulesAtTypes_1 = ts.combinePaths(nodeModulesFolder, "@types");
+            var nodeModulesAtTypesExists = nodeModulesFolderExists;
+            if (nodeModulesFolderExists && !ts.directoryProbablyExists(nodeModulesAtTypes_1, state.host)) {
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.Directory_0_does_not_exist_skipping_all_lookups_in_it, nodeModulesAtTypes_1);
+                }
+                nodeModulesAtTypesExists = false;
+            }
+            return loadModuleFromSpecificNodeModulesDirectory(Extensions.DtsOnly, mangleScopedPackageNameWithTrace(moduleName, state), nodeModulesAtTypes_1, nodeModulesAtTypesExists, state);
         }
-        function mergeModuleAugmentation(moduleName) {
-            var _a, _b;
-            var moduleAugmentation = moduleName.parent;
-            if (moduleAugmentation.symbol.declarations[0] !== moduleAugmentation) {
-                // this is a combined symbol for multiple augmentations within the same file.
-                // its symbol already has accumulated information for all declarations
-                // so we need to add it just once - do the work only for first declaration
-                ts.Debug.assert(moduleAugmentation.symbol.declarations.length > 1);
-                return;
+    }
+    function loadModuleFromSpecificNodeModulesDirectory(extensions, moduleName, nodeModulesDirectory, nodeModulesDirectoryExists, state) {
+        var candidate = ts.normalizePath(ts.combinePaths(nodeModulesDirectory, moduleName));
+        // First look for a nested package.json, as in `node_modules/foo/bar/package.json`.
+        var packageInfo = getPackageJsonInfo(candidate, !nodeModulesDirectoryExists, state);
+        if (packageInfo) {
+            var fromFile = loadModuleFromFile(extensions, candidate, !nodeModulesDirectoryExists, state);
+            if (fromFile) {
+                return noPackageId(fromFile);
             }
-            if (ts.isGlobalScopeAugmentation(moduleAugmentation)) {
-                mergeSymbolTable(globals, moduleAugmentation.symbol.exports);
+            var fromDirectory = loadNodeModuleFromDirectoryWorker(extensions, candidate, !nodeModulesDirectoryExists, state, packageInfo.packageJsonContent, packageInfo.versionPaths);
+            return withPackageId(packageInfo, fromDirectory);
+        }
+        var loader = function (extensions, candidate, onlyRecordFailures, state) {
+            var pathAndExtension = loadModuleFromFile(extensions, candidate, onlyRecordFailures, state) ||
+                loadNodeModuleFromDirectoryWorker(extensions, candidate, onlyRecordFailures, state, packageInfo && packageInfo.packageJsonContent, packageInfo && packageInfo.versionPaths);
+            return withPackageId(packageInfo, pathAndExtension);
+        };
+        var _a = parsePackageName(moduleName), packageName = _a.packageName, rest = _a.rest;
+        if (rest !== "") { // If "rest" is empty, we just did this search above.
+            var packageDirectory = ts.combinePaths(nodeModulesDirectory, packageName);
+            // Don't use a "types" or "main" from here because we're not loading the root, but a subdirectory -- just here for the packageId and path mappings.
+            packageInfo = getPackageJsonInfo(packageDirectory, !nodeModulesDirectoryExists, state);
+            if (packageInfo && packageInfo.versionPaths) {
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.package_json_has_a_typesVersions_entry_0_that_matches_compiler_version_1_looking_for_a_pattern_to_match_module_name_2, packageInfo.versionPaths.version, ts.version, rest);
+                }
+                var packageDirectoryExists = nodeModulesDirectoryExists && ts.directoryProbablyExists(packageDirectory, state.host);
+                var fromPaths = tryLoadModuleUsingPaths(extensions, rest, packageDirectory, packageInfo.versionPaths.paths, loader, !packageDirectoryExists, state);
+                if (fromPaths) {
+                    return fromPaths.value;
+                }
             }
-            else {
-                // find a module that about to be augmented
-                // do not validate names of augmentations that are defined in ambient context
-                var moduleNotFoundError = !(moduleName.parent.parent.flags & 8388608 /* Ambient */)
-                    ? ts.Diagnostics.Invalid_module_name_in_augmentation_module_0_cannot_be_found
-                    : undefined;
-                var mainModule_1 = resolveExternalModuleNameWorker(moduleName, moduleName, moduleNotFoundError, /*isForAugmentation*/ true);
-                if (!mainModule_1) {
-                    return;
+        }
+        return loader(extensions, candidate, !nodeModulesDirectoryExists, state);
+    }
+    function tryLoadModuleUsingPaths(extensions, moduleName, baseDirectory, paths, loader, onlyRecordFailures, state) {
+        var matchedPattern = ts.matchPatternOrExact(ts.getOwnKeys(paths), moduleName);
+        if (matchedPattern) {
+            var matchedStar_1 = ts.isString(matchedPattern) ? undefined : ts.matchedText(matchedPattern, moduleName);
+            var matchedPatternText = ts.isString(matchedPattern) ? matchedPattern : ts.patternText(matchedPattern);
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Module_name_0_matched_pattern_1, moduleName, matchedPatternText);
+            }
+            var resolved = ts.forEach(paths[matchedPatternText], function (subst) {
+                var path = matchedStar_1 ? subst.replace("*", matchedStar_1) : subst;
+                // When baseUrl is not specified, the command line parser resolves relative paths to the config file location.
+                var candidate = ts.normalizePath(ts.combinePaths(baseDirectory, path));
+                if (state.traceEnabled) {
+                    trace(state.host, ts.Diagnostics.Trying_substitution_0_candidate_module_location_Colon_1, subst, path);
                 }
-                // obtain item referenced by 'export='
-                mainModule_1 = resolveExternalModuleSymbol(mainModule_1);
-                if (mainModule_1.flags & 1920 /* Namespace */) {
-                    // If we're merging an augmentation to a pattern ambient module, we want to
-                    // perform the merge unidirectionally from the augmentation ('a.foo') to
-                    // the pattern ('*.foo'), so that 'getMergedSymbol()' on a.foo gives you
-                    // all the exports both from the pattern and from the augmentation, but
-                    // 'getMergedSymbol()' on *.foo only gives you exports from *.foo.
-                    if (ts.some(patternAmbientModules, function (module) { return mainModule_1 === module.symbol; })) {
-                        var merged = mergeSymbol(moduleAugmentation.symbol, mainModule_1, /*unidirectional*/ true);
-                        if (!patternAmbientModuleAugmentations) {
-                            patternAmbientModuleAugmentations = ts.createMap();
-                        }
-                        // moduleName will be a StringLiteral since this is not `declare global`.
-                        patternAmbientModuleAugmentations.set(moduleName.text, merged);
-                    }
-                    else {
-                        if (((_a = mainModule_1.exports) === null || _a === void 0 ? void 0 : _a.get("__export" /* ExportStar */)) && ((_b = moduleAugmentation.symbol.exports) === null || _b === void 0 ? void 0 : _b.size)) {
-                            // We may need to merge the module augmentation's exports into the target symbols of the resolved exports
-                            var resolvedExports = getResolvedMembersOrExportsOfSymbol(mainModule_1, "resolvedExports" /* resolvedExports */);
-                            for (var _i = 0, _c = ts.arrayFrom(moduleAugmentation.symbol.exports.entries()); _i < _c.length; _i++) {
-                                var _d = _c[_i], key = _d[0], value = _d[1];
-                                if (resolvedExports.has(key) && !mainModule_1.exports.has(key)) {
-                                    mergeSymbol(resolvedExports.get(key), value);
-                                }
-                            }
-                        }
-                        mergeSymbol(mainModule_1, moduleAugmentation.symbol);
+                // A path mapping may have an extension, in contrast to an import, which should omit it.
+                var extension = ts.tryGetExtensionFromPath(candidate);
+                if (extension !== undefined) {
+                    var path_1 = tryFile(candidate, onlyRecordFailures, state);
+                    if (path_1 !== undefined) {
+                        return noPackageId({ path: path_1, ext: extension });
                     }
                 }
-                else {
-                    // moduleName will be a StringLiteral since this is not `declare global`.
-                    error(moduleName, ts.Diagnostics.Cannot_augment_module_0_because_it_resolves_to_a_non_module_entity, moduleName.text);
-                }
+                return loader(extensions, candidate, onlyRecordFailures || !ts.directoryProbablyExists(ts.getDirectoryPath(candidate), state.host), state);
+            });
+            return { value: resolved };
+        }
+    }
+    /** Double underscores are used in DefinitelyTyped to delimit scoped packages. */
+    var mangledScopedPackageSeparator = "__";
+    /** For a scoped package, we must look in `@types/foo__bar` instead of `@types/@foo/bar`. */
+    function mangleScopedPackageNameWithTrace(packageName, state) {
+        var mangled = mangleScopedPackageName(packageName);
+        if (state.traceEnabled && mangled !== packageName) {
+            trace(state.host, ts.Diagnostics.Scoped_package_detected_looking_in_0, mangled);
+        }
+        return mangled;
+    }
+    /* @internal */
+    function getTypesPackageName(packageName) {
+        return "@types/" + mangleScopedPackageName(packageName);
+    }
+    ts.getTypesPackageName = getTypesPackageName;
+    /* @internal */
+    function mangleScopedPackageName(packageName) {
+        if (ts.startsWith(packageName, "@")) {
+            var replaceSlash = packageName.replace(ts.directorySeparator, mangledScopedPackageSeparator);
+            if (replaceSlash !== packageName) {
+                return replaceSlash.slice(1); // Take off the "@"
             }
         }
-        function addToSymbolTable(target, source, message) {
-            source.forEach(function (sourceSymbol, id) {
-                var targetSymbol = target.get(id);
-                if (targetSymbol) {
-                    // Error on redeclarations
-                    ts.forEach(targetSymbol.declarations, addDeclarationDiagnostic(ts.unescapeLeadingUnderscores(id), message));
+        return packageName;
+    }
+    ts.mangleScopedPackageName = mangleScopedPackageName;
+    /* @internal */
+    function getPackageNameFromTypesPackageName(mangledName) {
+        var withoutAtTypePrefix = ts.removePrefix(mangledName, "@types/");
+        if (withoutAtTypePrefix !== mangledName) {
+            return unmangleScopedPackageName(withoutAtTypePrefix);
+        }
+        return mangledName;
+    }
+    ts.getPackageNameFromTypesPackageName = getPackageNameFromTypesPackageName;
+    /* @internal */
+    function unmangleScopedPackageName(typesPackageName) {
+        return ts.stringContains(typesPackageName, mangledScopedPackageSeparator) ?
+            "@" + typesPackageName.replace(mangledScopedPackageSeparator, ts.directorySeparator) :
+            typesPackageName;
+    }
+    ts.unmangleScopedPackageName = unmangleScopedPackageName;
+    function tryFindNonRelativeModuleNameInCache(cache, moduleName, containingDirectory, state) {
+        var result = cache && cache.get(containingDirectory);
+        if (result) {
+            if (state.traceEnabled) {
+                trace(state.host, ts.Diagnostics.Resolution_for_module_0_was_found_in_cache_from_location_1, moduleName, containingDirectory);
+            }
+            state.resultFromCache = result;
+            return { value: result.resolvedModule && { path: result.resolvedModule.resolvedFileName, originalPath: result.resolvedModule.originalPath || true, extension: result.resolvedModule.extension, packageId: result.resolvedModule.packageId } };
+        }
+    }
+    function classicNameResolver(moduleName, containingFile, compilerOptions, host, cache, redirectedReference) {
+        var traceEnabled = isTraceEnabled(compilerOptions, host);
+        var failedLookupLocations = [];
+        var state = { compilerOptions: compilerOptions, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
+        var containingDirectory = ts.getDirectoryPath(containingFile);
+        var resolved = tryResolve(Extensions.TypeScript) || tryResolve(Extensions.JavaScript);
+        // No originalPath because classic resolution doesn't resolve realPath
+        return createResolvedModuleWithFailedLookupLocations(resolved && resolved.value, /*isExternalLibraryImport*/ false, failedLookupLocations, state.resultFromCache);
+        function tryResolve(extensions) {
+            var resolvedUsingSettings = tryLoadModuleUsingOptionalResolutionSettings(extensions, moduleName, containingDirectory, loadModuleFromFileNoPackageId, state);
+            if (resolvedUsingSettings) {
+                return { value: resolvedUsingSettings };
+            }
+            if (!ts.isExternalModuleNameRelative(moduleName)) {
+                var perModuleNameCache_1 = cache && cache.getOrCreateCacheForModuleName(moduleName, redirectedReference);
+                // Climb up parent directories looking for a module.
+                var resolved_3 = ts.forEachAncestorDirectory(containingDirectory, function (directory) {
+                    var resolutionFromCache = tryFindNonRelativeModuleNameInCache(perModuleNameCache_1, moduleName, directory, state);
+                    if (resolutionFromCache) {
+                        return resolutionFromCache;
+                    }
+                    var searchName = ts.normalizePath(ts.combinePaths(directory, moduleName));
+                    return toSearchResult(loadModuleFromFileNoPackageId(extensions, searchName, /*onlyRecordFailures*/ false, state));
+                });
+                if (resolved_3) {
+                    return resolved_3;
                 }
-                else {
-                    target.set(id, sourceSymbol);
+                if (extensions === Extensions.TypeScript) {
+                    // If we didn't find the file normally, look it up in @types.
+                    return loadModuleFromNearestNodeModulesDirectoryTypesScope(moduleName, containingDirectory, state);
                 }
-            });
-            function addDeclarationDiagnostic(id, message) {
-                return function (declaration) { return diagnostics.add(ts.createDiagnosticForNode(declaration, message, id)); };
+            }
+            else {
+                var candidate = ts.normalizePath(ts.combinePaths(containingDirectory, moduleName));
+                return toSearchResult(loadModuleFromFileNoPackageId(extensions, candidate, /*onlyRecordFailures*/ false, state));
             }
         }
-        function getSymbolLinks(symbol) {
-            if (symbol.flags & 33554432 /* Transient */)
-                return symbol;
-            var id = getSymbolId(symbol);
-            return symbolLinks[id] || (symbolLinks[id] = new SymbolLinks());
+    }
+    ts.classicNameResolver = classicNameResolver;
+    /**
+     * A host may load a module from a global cache of typings.
+     * This is the minumum code needed to expose that functionality; the rest is in the host.
+     */
+    /* @internal */
+    function loadModuleFromGlobalCache(moduleName, projectName, compilerOptions, host, globalCache) {
+        var traceEnabled = isTraceEnabled(compilerOptions, host);
+        if (traceEnabled) {
+            trace(host, ts.Diagnostics.Auto_discovery_for_typings_is_enabled_in_project_0_Running_extra_resolution_pass_for_module_1_using_cache_location_2, projectName, moduleName, globalCache);
         }
-        function getNodeLinks(node) {
-            var nodeId = getNodeId(node);
-            return nodeLinks[nodeId] || (nodeLinks[nodeId] = new NodeLinks());
+        var failedLookupLocations = [];
+        var state = { compilerOptions: compilerOptions, host: host, traceEnabled: traceEnabled, failedLookupLocations: failedLookupLocations };
+        var resolved = loadModuleFromImmediateNodeModulesDirectory(Extensions.DtsOnly, moduleName, globalCache, state, /*typesScopeOnly*/ false);
+        return createResolvedModuleWithFailedLookupLocations(resolved, /*isExternalLibraryImport*/ true, failedLookupLocations, state.resultFromCache);
+    }
+    ts.loadModuleFromGlobalCache = loadModuleFromGlobalCache;
+    /**
+     * Wraps value to SearchResult.
+     * @returns undefined if value is undefined or { value } otherwise
+     */
+    function toSearchResult(value) {
+        return value !== undefined ? { value: value } : undefined;
+    }
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var ModuleInstanceState;
+    (function (ModuleInstanceState) {
+        ModuleInstanceState[ModuleInstanceState["NonInstantiated"] = 0] = "NonInstantiated";
+        ModuleInstanceState[ModuleInstanceState["Instantiated"] = 1] = "Instantiated";
+        ModuleInstanceState[ModuleInstanceState["ConstEnumOnly"] = 2] = "ConstEnumOnly";
+    })(ModuleInstanceState = ts.ModuleInstanceState || (ts.ModuleInstanceState = {}));
+    function getModuleInstanceState(node, visited) {
+        if (node.body && !node.body.parent) {
+            // getModuleInstanceStateForAliasTarget needs to walk up the parent chain, so parent pointers must be set on this tree already
+            ts.setParent(node.body, node);
+            ts.setParentRecursive(node.body, /*incremental*/ false);
         }
-        function isGlobalSourceFile(node) {
-            return node.kind === 290 /* SourceFile */ && !ts.isExternalOrCommonJsModule(node);
+        return node.body ? getModuleInstanceStateCached(node.body, visited) : 1 /* Instantiated */;
+    }
+    ts.getModuleInstanceState = getModuleInstanceState;
+    function getModuleInstanceStateCached(node, visited) {
+        if (visited === void 0) { visited = new ts.Map(); }
+        var nodeId = ts.getNodeId(node);
+        if (visited.has(nodeId)) {
+            return visited.get(nodeId) || 0 /* NonInstantiated */;
         }
-        function getSymbol(symbols, name, meaning) {
-            if (meaning) {
-                var symbol = getMergedSymbol(symbols.get(name));
-                if (symbol) {
-                    ts.Debug.assert((ts.getCheckFlags(symbol) & 1 /* Instantiated */) === 0, "Should never get an instantiated symbol here.");
-                    if (symbol.flags & meaning) {
-                        return symbol;
+        visited.set(nodeId, undefined);
+        var result = getModuleInstanceStateWorker(node, visited);
+        visited.set(nodeId, result);
+        return result;
+    }
+    function getModuleInstanceStateWorker(node, visited) {
+        // A module is uninstantiated if it contains only
+        switch (node.kind) {
+            // 1. interface declarations, type alias declarations
+            case 253 /* InterfaceDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+                return 0 /* NonInstantiated */;
+            // 2. const enum declarations
+            case 255 /* EnumDeclaration */:
+                if (ts.isEnumConst(node)) {
+                    return 2 /* ConstEnumOnly */;
+                }
+                break;
+            // 3. non-exported import declarations
+            case 261 /* ImportDeclaration */:
+            case 260 /* ImportEqualsDeclaration */:
+                if (!(ts.hasSyntacticModifier(node, 1 /* Export */))) {
+                    return 0 /* NonInstantiated */;
+                }
+                break;
+            // 4. Export alias declarations pointing at only uninstantiated modules or things uninstantiated modules contain
+            case 267 /* ExportDeclaration */:
+                var exportDeclaration = node;
+                if (!exportDeclaration.moduleSpecifier && exportDeclaration.exportClause && exportDeclaration.exportClause.kind === 268 /* NamedExports */) {
+                    var state = 0 /* NonInstantiated */;
+                    for (var _i = 0, _a = exportDeclaration.exportClause.elements; _i < _a.length; _i++) {
+                        var specifier = _a[_i];
+                        var specifierState = getModuleInstanceStateForAliasTarget(specifier, visited);
+                        if (specifierState > state) {
+                            state = specifierState;
+                        }
+                        if (state === 1 /* Instantiated */) {
+                            return state;
+                        }
                     }
-                    if (symbol.flags & 2097152 /* Alias */) {
-                        var target = resolveAlias(symbol);
-                        // Unknown symbol means an error occurred in alias resolution, treat it as positive answer to avoid cascading errors
-                        if (target === unknownSymbol || target.flags & meaning) {
-                            return symbol;
+                    return state;
+                }
+                break;
+            // 5. other uninstantiated module declarations.
+            case 257 /* ModuleBlock */: {
+                var state_1 = 0 /* NonInstantiated */;
+                ts.forEachChild(node, function (n) {
+                    var childState = getModuleInstanceStateCached(n, visited);
+                    switch (childState) {
+                        case 0 /* NonInstantiated */:
+                            // child is non-instantiated - continue searching
+                            return;
+                        case 2 /* ConstEnumOnly */:
+                            // child is const enum only - record state and continue searching
+                            state_1 = 2 /* ConstEnumOnly */;
+                            return;
+                        case 1 /* Instantiated */:
+                            // child is instantiated - record state and stop
+                            state_1 = 1 /* Instantiated */;
+                            return true;
+                        default:
+                            ts.Debug.assertNever(childState);
+                    }
+                });
+                return state_1;
+            }
+            case 256 /* ModuleDeclaration */:
+                return getModuleInstanceState(node, visited);
+            case 78 /* Identifier */:
+                // Only jsdoc typedef definition can exist in jsdoc namespace, and it should
+                // be considered the same as type alias
+                if (node.isInJSDocNamespace) {
+                    return 0 /* NonInstantiated */;
+                }
+        }
+        return 1 /* Instantiated */;
+    }
+    function getModuleInstanceStateForAliasTarget(specifier, visited) {
+        var name = specifier.propertyName || specifier.name;
+        var p = specifier.parent;
+        while (p) {
+            if (ts.isBlock(p) || ts.isModuleBlock(p) || ts.isSourceFile(p)) {
+                var statements = p.statements;
+                var found = void 0;
+                for (var _i = 0, statements_2 = statements; _i < statements_2.length; _i++) {
+                    var statement = statements_2[_i];
+                    if (ts.nodeHasName(statement, name)) {
+                        if (!statement.parent) {
+                            ts.setParent(statement, p);
+                            ts.setParentRecursive(statement, /*incremental*/ false);
+                        }
+                        var state = getModuleInstanceStateCached(statement, visited);
+                        if (found === undefined || state > found) {
+                            found = state;
+                        }
+                        if (found === 1 /* Instantiated */) {
+                            return found;
                         }
                     }
                 }
+                if (found !== undefined) {
+                    return found;
+                }
             }
-            // return undefined if we can't find a symbol.
+            p = p.parent;
         }
+        return 1 /* Instantiated */; // Couldn't locate, assume could refer to a value
+    }
+    var ContainerFlags;
+    (function (ContainerFlags) {
+        // The current node is not a container, and no container manipulation should happen before
+        // recursing into it.
+        ContainerFlags[ContainerFlags["None"] = 0] = "None";
+        // The current node is a container.  It should be set as the current container (and block-
+        // container) before recursing into it.  The current node does not have locals.  Examples:
+        //
+        //      Classes, ObjectLiterals, TypeLiterals, Interfaces...
+        ContainerFlags[ContainerFlags["IsContainer"] = 1] = "IsContainer";
+        // The current node is a block-scoped-container.  It should be set as the current block-
+        // container before recursing into it.  Examples:
+        //
+        //      Blocks (when not parented by functions), Catch clauses, For/For-in/For-of statements...
+        ContainerFlags[ContainerFlags["IsBlockScopedContainer"] = 2] = "IsBlockScopedContainer";
+        // The current node is the container of a control flow path. The current control flow should
+        // be saved and restored, and a new control flow initialized within the container.
+        ContainerFlags[ContainerFlags["IsControlFlowContainer"] = 4] = "IsControlFlowContainer";
+        ContainerFlags[ContainerFlags["IsFunctionLike"] = 8] = "IsFunctionLike";
+        ContainerFlags[ContainerFlags["IsFunctionExpression"] = 16] = "IsFunctionExpression";
+        ContainerFlags[ContainerFlags["HasLocals"] = 32] = "HasLocals";
+        ContainerFlags[ContainerFlags["IsInterface"] = 64] = "IsInterface";
+        ContainerFlags[ContainerFlags["IsObjectLiteralOrClassExpressionMethod"] = 128] = "IsObjectLiteralOrClassExpressionMethod";
+    })(ContainerFlags || (ContainerFlags = {}));
+    function initFlowNode(node) {
+        ts.Debug.attachFlowNodeDebugInfo(node);
+        return node;
+    }
+    var binder = createBinder();
+    function bindSourceFile(file, options) {
+        var tracingData = ["bind" /* Bind */, "bindSourceFile", { path: file.path }];
+        ts.tracing.begin.apply(ts.tracing, tracingData);
+        ts.performance.mark("beforeBind");
+        ts.perfLogger.logStartBindFile("" + file.fileName);
+        binder(file, options);
+        ts.perfLogger.logStopBindFile();
+        ts.performance.mark("afterBind");
+        ts.performance.measure("Bind", "beforeBind", "afterBind");
+        ts.tracing.end.apply(ts.tracing, tracingData);
+    }
+    ts.bindSourceFile = bindSourceFile;
+    function createBinder() {
+        var file;
+        var options;
+        var languageVersion;
+        var parent;
+        var container;
+        var thisParentContainer; // Container one level up
+        var blockScopeContainer;
+        var lastContainer;
+        var delayedTypeAliases;
+        var seenThisKeyword;
+        // state used by control flow analysis
+        var currentFlow;
+        var currentBreakTarget;
+        var currentContinueTarget;
+        var currentReturnTarget;
+        var currentTrueTarget;
+        var currentFalseTarget;
+        var currentExceptionTarget;
+        var preSwitchCaseFlow;
+        var activeLabelList;
+        var hasExplicitReturn;
+        // state used for emit helpers
+        var emitFlags;
+        // If this file is an external module, then it is automatically in strict-mode according to
+        // ES6.  If it is not an external module, then we'll determine if it is in strict mode or
+        // not depending on if we see "use strict" in certain places or if we hit a class/namespace
+        // or if compiler options contain alwaysStrict.
+        var inStrictMode;
+        // If we are binding an assignment pattern, we will bind certain expressions differently.
+        var inAssignmentPattern = false;
+        var symbolCount = 0;
+        var Symbol;
+        var classifiableNames;
+        var unreachableFlow = { flags: 1 /* Unreachable */ };
+        var reportedUnreachableFlow = { flags: 1 /* Unreachable */ };
         /**
-         * Get symbols that represent parameter-property-declaration as parameter and as property declaration
-         * @param parameter a parameterDeclaration node
-         * @param parameterName a name of the parameter to get the symbols for.
-         * @return a tuple of two symbols
+         * Inside the binder, we may create a diagnostic for an as-yet unbound node (with potentially no parent pointers, implying no accessible source file)
+         * If so, the node _must_ be in the current file (as that's the only way anything could have traversed to it to yield it as the error node)
+         * This version of `createDiagnosticForNode` uses the binder's context to account for this, and always yields correct diagnostics even in these situations.
          */
-        function getSymbolsOfParameterPropertyDeclaration(parameter, parameterName) {
-            var constructorDeclaration = parameter.parent;
-            var classDeclaration = parameter.parent.parent;
-            var parameterSymbol = getSymbol(constructorDeclaration.locals, parameterName, 111551 /* Value */);
-            var propertySymbol = getSymbol(getMembersOfSymbol(classDeclaration.symbol), parameterName, 111551 /* Value */);
-            if (parameterSymbol && propertySymbol) {
-                return [parameterSymbol, propertySymbol];
-            }
-            return ts.Debug.fail("There should exist two symbols, one as property declaration and one as parameter declaration");
+        function createDiagnosticForNode(node, message, arg0, arg1, arg2) {
+            return ts.createDiagnosticForNodeInSourceFile(ts.getSourceFileOfNode(node) || file, node, message, arg0, arg1, arg2);
         }
-        function isBlockScopedNameDeclaredBeforeUse(declaration, usage) {
-            var declarationFile = ts.getSourceFileOfNode(declaration);
-            var useFile = ts.getSourceFileOfNode(usage);
-            var declContainer = ts.getEnclosingBlockScopeContainer(declaration);
-            if (declarationFile !== useFile) {
-                if ((moduleKind && (declarationFile.externalModuleIndicator || useFile.externalModuleIndicator)) ||
-                    (!compilerOptions.outFile && !compilerOptions.out) ||
-                    isInTypeQuery(usage) ||
-                    declaration.flags & 8388608 /* Ambient */) {
-                    // nodes are in different files and order cannot be determined
-                    return true;
-                }
-                // declaration is after usage
-                // can be legal if usage is deferred (i.e. inside function or in initializer of instance property)
-                if (isUsedInFunctionOrInstanceProperty(usage, declaration)) {
-                    return true;
-                }
-                var sourceFiles = host.getSourceFiles();
-                return sourceFiles.indexOf(declarationFile) <= sourceFiles.indexOf(useFile);
-            }
-            if (declaration.pos <= usage.pos) {
-                // declaration is before usage
-                if (declaration.kind === 191 /* BindingElement */) {
-                    // still might be illegal if declaration and usage are both binding elements (eg var [a = b, b = b] = [1, 2])
-                    var errorBindingElement = ts.getAncestor(usage, 191 /* BindingElement */);
-                    if (errorBindingElement) {
-                        return ts.findAncestor(errorBindingElement, ts.isBindingElement) !== ts.findAncestor(declaration, ts.isBindingElement) ||
-                            declaration.pos < errorBindingElement.pos;
-                    }
-                    // or it might be illegal if usage happens before parent variable is declared (eg var [a] = a)
-                    return isBlockScopedNameDeclaredBeforeUse(ts.getAncestor(declaration, 242 /* VariableDeclaration */), usage);
-                }
-                else if (declaration.kind === 242 /* VariableDeclaration */) {
-                    // still might be illegal if usage is in the initializer of the variable declaration (eg var a = a)
-                    return !isImmediatelyUsedInInitializerOfBlockScopedVariable(declaration, usage);
-                }
-                else if (ts.isClassDeclaration(declaration)) {
-                    // still might be illegal if the usage is within a computed property name in the class (eg class A { static p = "a"; [A.p]() {} })
-                    return !ts.findAncestor(usage, function (n) { return ts.isComputedPropertyName(n) && n.parent.parent === declaration; });
-                }
-                else if (ts.isPropertyDeclaration(declaration)) {
-                    // still might be illegal if a self-referencing property initializer (eg private x = this.x)
-                    return !isPropertyImmediatelyReferencedWithinDeclaration(declaration, usage, /*stopAtAnyPropertyDeclaration*/ false);
-                }
-                else if (ts.isParameterPropertyDeclaration(declaration, declaration.parent)) {
-                    // foo = this.bar is illegal in esnext+useDefineForClassFields when bar is a parameter property
-                    return !(compilerOptions.target === 99 /* ESNext */ && !!compilerOptions.useDefineForClassFields
-                        && ts.getContainingClass(declaration) === ts.getContainingClass(usage)
-                        && isUsedInFunctionOrInstanceProperty(usage, declaration));
-                }
-                return true;
+        function bindSourceFile(f, opts) {
+            file = f;
+            options = opts;
+            languageVersion = ts.getEmitScriptTarget(options);
+            inStrictMode = bindInStrictMode(file, opts);
+            classifiableNames = new ts.Set();
+            symbolCount = 0;
+            Symbol = ts.objectAllocator.getSymbolConstructor();
+            // Attach debugging information if necessary
+            ts.Debug.attachFlowNodeDebugInfo(unreachableFlow);
+            ts.Debug.attachFlowNodeDebugInfo(reportedUnreachableFlow);
+            if (!file.locals) {
+                bind(file);
+                file.symbolCount = symbolCount;
+                file.classifiableNames = classifiableNames;
+                delayedBindJSDocTypedefTag();
             }
-            // declaration is after usage, but it can still be legal if usage is deferred:
-            // 1. inside an export specifier
-            // 2. inside a function
-            // 3. inside an instance property initializer, a reference to a non-instance property
-            //    (except when target: "esnext" and useDefineForClassFields: true and the reference is to a parameter property)
-            // 4. inside a static property initializer, a reference to a static method in the same class
-            // 5. inside a TS export= declaration (since we will move the export statement during emit to avoid TDZ)
-            // or if usage is in a type context:
-            // 1. inside a type query (typeof in type position)
-            // 2. inside a jsdoc comment
-            if (usage.parent.kind === 263 /* ExportSpecifier */ || (usage.parent.kind === 259 /* ExportAssignment */ && usage.parent.isExportEquals)) {
-                // export specifiers do not use the variable, they only make it available for use
+            file = undefined;
+            options = undefined;
+            languageVersion = undefined;
+            parent = undefined;
+            container = undefined;
+            thisParentContainer = undefined;
+            blockScopeContainer = undefined;
+            lastContainer = undefined;
+            delayedTypeAliases = undefined;
+            seenThisKeyword = false;
+            currentFlow = undefined;
+            currentBreakTarget = undefined;
+            currentContinueTarget = undefined;
+            currentReturnTarget = undefined;
+            currentTrueTarget = undefined;
+            currentFalseTarget = undefined;
+            currentExceptionTarget = undefined;
+            activeLabelList = undefined;
+            hasExplicitReturn = false;
+            inAssignmentPattern = false;
+            emitFlags = 0 /* None */;
+        }
+        return bindSourceFile;
+        function bindInStrictMode(file, opts) {
+            if (ts.getStrictOptionValue(opts, "alwaysStrict") && !file.isDeclarationFile) {
+                // bind in strict mode source files with alwaysStrict option
                 return true;
             }
-            // When resolving symbols for exports, the `usage` location passed in can be the export site directly
-            if (usage.kind === 259 /* ExportAssignment */ && usage.isExportEquals) {
-                return true;
+            else {
+                return !!file.externalModuleIndicator;
             }
-            if (!!(usage.flags & 4194304 /* JSDoc */) || isInTypeQuery(usage) || usageInTypeDeclaration()) {
-                return true;
+        }
+        function createSymbol(flags, name) {
+            symbolCount++;
+            return new Symbol(flags, name);
+        }
+        function addDeclarationToSymbol(symbol, node, symbolFlags) {
+            symbol.flags |= symbolFlags;
+            node.symbol = symbol;
+            symbol.declarations = ts.appendIfUnique(symbol.declarations, node);
+            if (symbolFlags & (32 /* Class */ | 384 /* Enum */ | 1536 /* Module */ | 3 /* Variable */) && !symbol.exports) {
+                symbol.exports = ts.createSymbolTable();
             }
-            if (isUsedInFunctionOrInstanceProperty(usage, declaration)) {
-                if (compilerOptions.target === 99 /* ESNext */ && !!compilerOptions.useDefineForClassFields
-                    && ts.getContainingClass(declaration)
-                    && (ts.isPropertyDeclaration(declaration) || ts.isParameterPropertyDeclaration(declaration, declaration.parent))) {
-                    return !isPropertyImmediatelyReferencedWithinDeclaration(declaration, usage, /*stopAtAnyPropertyDeclaration*/ true);
-                }
-                else {
-                    return true;
-                }
+            if (symbolFlags & (32 /* Class */ | 64 /* Interface */ | 2048 /* TypeLiteral */ | 4096 /* ObjectLiteral */) && !symbol.members) {
+                symbol.members = ts.createSymbolTable();
             }
-            return false;
-            function usageInTypeDeclaration() {
-                return !!ts.findAncestor(usage, function (node) { return ts.isInterfaceDeclaration(node) || ts.isTypeAliasDeclaration(node); });
+            // On merge of const enum module with class or function, reset const enum only flag (namespaces will already recalculate)
+            if (symbol.constEnumOnlyModule && (symbol.flags & (16 /* Function */ | 32 /* Class */ | 256 /* RegularEnum */))) {
+                symbol.constEnumOnlyModule = false;
             }
-            function isImmediatelyUsedInInitializerOfBlockScopedVariable(declaration, usage) {
-                switch (declaration.parent.parent.kind) {
-                    case 225 /* VariableStatement */:
-                    case 230 /* ForStatement */:
-                    case 232 /* ForOfStatement */:
-                        // variable statement/for/for-of statement case,
-                        // use site should not be inside variable declaration (initializer of declaration or binding element)
-                        if (isSameScopeDescendentOf(usage, declaration, declContainer)) {
-                            return true;
-                        }
-                        break;
-                }
-                // ForIn/ForOf case - use site should not be used in expression part
-                var grandparent = declaration.parent.parent;
-                return ts.isForInOrOfStatement(grandparent) && isSameScopeDescendentOf(usage, grandparent.expression, declContainer);
+            if (symbolFlags & 111551 /* Value */) {
+                ts.setValueDeclaration(symbol, node);
             }
-            function isUsedInFunctionOrInstanceProperty(usage, declaration) {
-                return !!ts.findAncestor(usage, function (current) {
-                    if (current === declContainer) {
-                        return "quit";
-                    }
-                    if (ts.isFunctionLike(current)) {
-                        return true;
-                    }
-                    var initializerOfProperty = current.parent &&
-                        current.parent.kind === 159 /* PropertyDeclaration */ &&
-                        current.parent.initializer === current;
-                    if (initializerOfProperty) {
-                        if (ts.hasModifier(current.parent, 32 /* Static */)) {
-                            if (declaration.kind === 161 /* MethodDeclaration */) {
-                                return true;
-                            }
-                        }
-                        else {
-                            var isDeclarationInstanceProperty = declaration.kind === 159 /* PropertyDeclaration */ && !ts.hasModifier(declaration, 32 /* Static */);
-                            if (!isDeclarationInstanceProperty || ts.getContainingClass(usage) !== ts.getContainingClass(declaration)) {
-                                return true;
-                            }
-                        }
-                    }
-                    return false;
-                });
+        }
+        // Should not be called on a declaration with a computed property name,
+        // unless it is a well known Symbol.
+        function getDeclarationName(node) {
+            if (node.kind === 266 /* ExportAssignment */) {
+                return node.isExportEquals ? "export=" /* ExportEquals */ : "default" /* Default */;
             }
-            /** stopAtAnyPropertyDeclaration is used for detecting ES-standard class field use-before-def errors */
-            function isPropertyImmediatelyReferencedWithinDeclaration(declaration, usage, stopAtAnyPropertyDeclaration) {
-                // always legal if usage is after declaration
-                if (usage.end > declaration.end) {
-                    return false;
+            var name = ts.getNameOfDeclaration(node);
+            if (name) {
+                if (ts.isAmbientModule(node)) {
+                    var moduleName = ts.getTextOfIdentifierOrLiteral(name);
+                    return (ts.isGlobalScopeAugmentation(node) ? "__global" : "\"" + moduleName + "\"");
                 }
-                // still might be legal if usage is deferred (e.g. x: any = () => this.x)
-                // otherwise illegal if immediately referenced within the declaration (e.g. x: any = this.x)
-                var ancestorChangingReferenceScope = ts.findAncestor(usage, function (node) {
-                    if (node === declaration) {
-                        return "quit";
+                if (name.kind === 158 /* ComputedPropertyName */) {
+                    var nameExpression = name.expression;
+                    // treat computed property names where expression is string/numeric literal as just string/numeric literal
+                    if (ts.isStringOrNumericLiteralLike(nameExpression)) {
+                        return ts.escapeLeadingUnderscores(nameExpression.text);
                     }
-                    switch (node.kind) {
-                        case 202 /* ArrowFunction */:
-                            return true;
-                        case 159 /* PropertyDeclaration */:
-                            // even when stopping at any property declaration, they need to come from the same class
-                            return stopAtAnyPropertyDeclaration &&
-                                (ts.isPropertyDeclaration(declaration) && node.parent === declaration.parent
-                                    || ts.isParameterPropertyDeclaration(declaration, declaration.parent) && node.parent === declaration.parent.parent)
-                                ? "quit" : true;
-                        case 223 /* Block */:
-                            switch (node.parent.kind) {
-                                case 163 /* GetAccessor */:
-                                case 161 /* MethodDeclaration */:
-                                case 164 /* SetAccessor */:
-                                    return true;
-                                default:
-                                    return false;
-                            }
-                        default:
-                            return false;
+                    if (ts.isSignedNumericLiteral(nameExpression)) {
+                        return ts.tokenToString(nameExpression.operator) + nameExpression.operand.text;
                     }
-                });
-                return ancestorChangingReferenceScope === undefined;
-            }
-        }
-        function useOuterVariableScopeInParameter(result, location, lastLocation) {
-            var target = ts.getEmitScriptTarget(compilerOptions);
-            var functionLocation = location;
-            if (ts.isParameter(lastLocation) && functionLocation.body && result.valueDeclaration.pos >= functionLocation.body.pos && result.valueDeclaration.end <= functionLocation.body.end) {
-                // check for several cases where we introduce temporaries that require moving the name/initializer of the parameter to the body
-                // - static field in a class expression
-                // - optional chaining pre-es2020
-                // - nullish coalesce pre-es2020
-                // - spread assignment in binding pattern pre-es2017
-                if (target >= 2 /* ES2015 */) {
-                    var links = getNodeLinks(functionLocation);
-                    if (links.declarationRequiresScopeChange === undefined) {
-                        links.declarationRequiresScopeChange = ts.forEach(functionLocation.parameters, requiresScopeChange) || false;
+                    ts.Debug.assert(ts.isWellKnownSymbolSyntactically(nameExpression));
+                    return ts.getPropertyNameForKnownSymbolName(ts.idText(nameExpression.name));
+                }
+                if (ts.isWellKnownSymbolSyntactically(name)) {
+                    return ts.getPropertyNameForKnownSymbolName(ts.idText(name.name));
+                }
+                if (ts.isPrivateIdentifier(name)) {
+                    // containingClass exists because private names only allowed inside classes
+                    var containingClass = ts.getContainingClass(node);
+                    if (!containingClass) {
+                        // we can get here in cases where there is already a parse error.
+                        return undefined;
                     }
-                    return !links.declarationRequiresScopeChange;
+                    var containingClassSymbol = containingClass.symbol;
+                    return ts.getSymbolNameForPrivateIdentifier(containingClassSymbol, name.escapedText);
                 }
+                return ts.isPropertyNameLiteral(name) ? ts.getEscapedTextOfIdentifierOrLiteral(name) : undefined;
             }
-            return false;
-            function requiresScopeChange(node) {
-                return requiresScopeChangeWorker(node.name)
-                    || !!node.initializer && requiresScopeChangeWorker(node.initializer);
-            }
-            function requiresScopeChangeWorker(node) {
-                switch (node.kind) {
-                    case 202 /* ArrowFunction */:
-                    case 201 /* FunctionExpression */:
-                    case 244 /* FunctionDeclaration */:
-                    case 162 /* Constructor */:
-                        // do not descend into these
-                        return false;
-                    case 161 /* MethodDeclaration */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                    case 281 /* PropertyAssignment */:
-                        return requiresScopeChangeWorker(node.name);
-                    case 159 /* PropertyDeclaration */:
-                        // static properties in classes introduce temporary variables
-                        if (ts.hasStaticModifier(node)) {
-                            return target < 99 /* ESNext */ || !compilerOptions.useDefineForClassFields;
-                        }
-                        return requiresScopeChangeWorker(node.name);
-                    default:
-                        // null coalesce and optional chain pre-es2020 produce temporary variables
-                        if (ts.isNullishCoalesce(node) || ts.isOptionalChain(node)) {
-                            return target < 7 /* ES2020 */;
-                        }
-                        if (ts.isBindingElement(node) && node.dotDotDotToken && ts.isObjectBindingPattern(node.parent)) {
-                            return target < 4 /* ES2017 */;
-                        }
-                        if (ts.isTypeNode(node))
-                            return false;
-                        return ts.forEachChild(node, requiresScopeChangeWorker) || false;
-                }
+            switch (node.kind) {
+                case 166 /* Constructor */:
+                    return "__constructor" /* Constructor */;
+                case 174 /* FunctionType */:
+                case 169 /* CallSignature */:
+                case 313 /* JSDocSignature */:
+                    return "__call" /* Call */;
+                case 175 /* ConstructorType */:
+                case 170 /* ConstructSignature */:
+                    return "__new" /* New */;
+                case 171 /* IndexSignature */:
+                    return "__index" /* Index */;
+                case 267 /* ExportDeclaration */:
+                    return "__export" /* ExportStar */;
+                case 297 /* SourceFile */:
+                    // json file should behave as
+                    // module.exports = ...
+                    return "export=" /* ExportEquals */;
+                case 216 /* BinaryExpression */:
+                    if (ts.getAssignmentDeclarationKind(node) === 2 /* ModuleExports */) {
+                        // module.exports = ...
+                        return "export=" /* ExportEquals */;
+                    }
+                    ts.Debug.fail("Unknown binary declaration kind");
+                    break;
+                case 308 /* JSDocFunctionType */:
+                    return (ts.isJSDocConstructSignature(node) ? "__new" /* New */ : "__call" /* Call */);
+                case 160 /* Parameter */:
+                    // Parameters with names are handled at the top of this function.  Parameters
+                    // without names can only come from JSDocFunctionTypes.
+                    ts.Debug.assert(node.parent.kind === 308 /* JSDocFunctionType */, "Impossible parameter parent kind", function () { return "parent is: " + (ts.SyntaxKind ? ts.SyntaxKind[node.parent.kind] : node.parent.kind) + ", expected JSDocFunctionType"; });
+                    var functionType = node.parent;
+                    var index = functionType.parameters.indexOf(node);
+                    return "arg" + index;
             }
         }
+        function getDisplayName(node) {
+            return ts.isNamedDeclaration(node) ? ts.declarationNameToString(node.name) : ts.unescapeLeadingUnderscores(ts.Debug.checkDefined(getDeclarationName(node)));
+        }
         /**
-         * Resolve a given name for a given meaning at a given location. An error is reported if the name was not found and
-         * the nameNotFoundMessage argument is not undefined. Returns the resolved symbol, or undefined if no symbol with
-         * the given name can be found.
-         *
-         * @param isUse If true, this will count towards --noUnusedLocals / --noUnusedParameters.
+         * Declares a Symbol for the node and adds it to symbols. Reports errors for conflicting identifier names.
+         * @param symbolTable - The symbol table which node will be added to.
+         * @param parent - node's parent declaration.
+         * @param node - The declaration to be added to the symbol table
+         * @param includes - The SymbolFlags that node has in addition to its declaration type (eg: export, ambient, etc.)
+         * @param excludes - The flags which node cannot be declared alongside in a symbol table. Used to report forbidden declarations.
          */
-        function resolveName(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, suggestedNameNotFoundMessage) {
-            if (excludeGlobals === void 0) { excludeGlobals = false; }
-            return resolveNameHelper(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, getSymbol, suggestedNameNotFoundMessage);
-        }
-        function resolveNameHelper(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, lookup, suggestedNameNotFoundMessage) {
-            var originalLocation = location; // needed for did-you-mean error reporting, which gathers candidates starting from the original location
-            var result;
-            var lastLocation;
-            var lastSelfReferenceLocation;
-            var propertyWithInvalidInitializer;
-            var associatedDeclarationForContainingInitializerOrBindingName;
-            var withinDeferredContext = false;
-            var errorLocation = location;
-            var grandparent;
-            var isInExternalModule = false;
-            loop: while (location) {
-                // Locals of a source file are not in scope (because they get merged into the global symbol table)
-                if (location.locals && !isGlobalSourceFile(location)) {
-                    if (result = lookup(location.locals, name, meaning)) {
-                        var useResult = true;
-                        if (ts.isFunctionLike(location) && lastLocation && lastLocation !== location.body) {
-                            // symbol lookup restrictions for function-like declarations
-                            // - Type parameters of a function are in scope in the entire function declaration, including the parameter
-                            //   list and return type. However, local types are only in scope in the function body.
-                            // - parameters are only in the scope of function body
-                            // This restriction does not apply to JSDoc comment types because they are parented
-                            // at a higher level than type parameters would normally be
-                            if (meaning & result.flags & 788968 /* Type */ && lastLocation.kind !== 303 /* JSDocComment */) {
-                                useResult = result.flags & 262144 /* TypeParameter */
-                                    // type parameters are visible in parameter list, return type and type parameter list
-                                    ? lastLocation === location.type ||
-                                        lastLocation.kind === 156 /* Parameter */ ||
-                                        lastLocation.kind === 155 /* TypeParameter */
-                                    // local types not visible outside the function body
-                                    : false;
-                            }
-                            if (meaning & result.flags & 3 /* Variable */) {
-                                // expression inside parameter will lookup as normal variable scope when targeting es2015+
-                                if (useOuterVariableScopeInParameter(result, location, lastLocation)) {
-                                    useResult = false;
-                                }
-                                else if (result.flags & 1 /* FunctionScopedVariable */) {
-                                    // parameters are visible only inside function body, parameter list and return type
-                                    // technically for parameter list case here we might mix parameters and variables declared in function,
-                                    // however it is detected separately when checking initializers of parameters
-                                    // to make sure that they reference no variables declared after them.
-                                    useResult =
-                                        lastLocation.kind === 156 /* Parameter */ ||
-                                            (lastLocation === location.type &&
-                                                !!ts.findAncestor(result.valueDeclaration, ts.isParameter));
-                                }
-                            }
-                        }
-                        else if (location.kind === 180 /* ConditionalType */) {
-                            // A type parameter declared using 'infer T' in a conditional type is visible only in
-                            // the true branch of the conditional type.
-                            useResult = lastLocation === location.trueType;
-                        }
-                        if (useResult) {
-                            break loop;
-                        }
-                        else {
-                            result = undefined;
-                        }
-                    }
+        function declareSymbol(symbolTable, parent, node, includes, excludes, isReplaceableByMethod) {
+            ts.Debug.assert(!ts.hasDynamicName(node));
+            var isDefaultExport = ts.hasSyntacticModifier(node, 512 /* Default */) || ts.isExportSpecifier(node) && node.name.escapedText === "default";
+            // The exported symbol for an export default function/class node is always named "default"
+            var name = isDefaultExport && parent ? "default" /* Default */ : getDeclarationName(node);
+            var symbol;
+            if (name === undefined) {
+                symbol = createSymbol(0 /* None */, "__missing" /* Missing */);
+            }
+            else {
+                // Check and see if the symbol table already has a symbol with this name.  If not,
+                // create a new symbol with this name and add it to the table.  Note that we don't
+                // give the new symbol any flags *yet*.  This ensures that it will not conflict
+                // with the 'excludes' flags we pass in.
+                //
+                // If we do get an existing symbol, see if it conflicts with the new symbol we're
+                // creating.  For example, a 'var' symbol and a 'class' symbol will conflict within
+                // the same symbol table.  If we have a conflict, report the issue on each
+                // declaration we have for this symbol, and then create a new symbol for this
+                // declaration.
+                //
+                // Note that when properties declared in Javascript constructors
+                // (marked by isReplaceableByMethod) conflict with another symbol, the property loses.
+                // Always. This allows the common Javascript pattern of overwriting a prototype method
+                // with an bound instance method of the same type: `this.method = this.method.bind(this)`
+                //
+                // If we created a new symbol, either because we didn't have a symbol with this name
+                // in the symbol table, or we conflicted with an existing symbol, then just add this
+                // node as the sole declaration of the new symbol.
+                //
+                // Otherwise, we'll be merging into a compatible existing symbol (for example when
+                // you have multiple 'vars' with the same name in the same container).  In this case
+                // just add this node into the declarations list of the symbol.
+                symbol = symbolTable.get(name);
+                if (includes & 2885600 /* Classifiable */) {
+                    classifiableNames.add(name);
                 }
-                withinDeferredContext = withinDeferredContext || getIsDeferredContext(location, lastLocation);
-                switch (location.kind) {
-                    case 290 /* SourceFile */:
-                        if (!ts.isExternalOrCommonJsModule(location))
-                            break;
-                        isInExternalModule = true;
-                    // falls through
-                    case 249 /* ModuleDeclaration */:
-                        var moduleExports = getSymbolOfNode(location).exports || emptySymbols;
-                        if (location.kind === 290 /* SourceFile */ || (ts.isModuleDeclaration(location) && location.flags & 8388608 /* Ambient */ && !ts.isGlobalScopeAugmentation(location))) {
-                            // It's an external module. First see if the module has an export default and if the local
-                            // name of that export default matches.
-                            if (result = moduleExports.get("default" /* Default */)) {
-                                var localSymbol = ts.getLocalSymbolForExportDefault(result);
-                                if (localSymbol && (result.flags & meaning) && localSymbol.escapedName === name) {
-                                    break loop;
-                                }
-                                result = undefined;
-                            }
-                            // Because of module/namespace merging, a module's exports are in scope,
-                            // yet we never want to treat an export specifier as putting a member in scope.
-                            // Therefore, if the name we find is purely an export specifier, it is not actually considered in scope.
-                            // Two things to note about this:
-                            //     1. We have to check this without calling getSymbol. The problem with calling getSymbol
-                            //        on an export specifier is that it might find the export specifier itself, and try to
-                            //        resolve it as an alias. This will cause the checker to consider the export specifier
-                            //        a circular alias reference when it might not be.
-                            //     2. We check === SymbolFlags.Alias in order to check that the symbol is *purely*
-                            //        an alias. If we used &, we'd be throwing out symbols that have non alias aspects,
-                            //        which is not the desired behavior.
-                            var moduleExport = moduleExports.get(name);
-                            if (moduleExport &&
-                                moduleExport.flags === 2097152 /* Alias */ &&
-                                (ts.getDeclarationOfKind(moduleExport, 263 /* ExportSpecifier */) || ts.getDeclarationOfKind(moduleExport, 262 /* NamespaceExport */))) {
-                                break;
-                            }
-                        }
-                        // ES6 exports are also visible locally (except for 'default'), but commonjs exports are not (except typedefs)
-                        if (name !== "default" /* Default */ && (result = lookup(moduleExports, name, meaning & 2623475 /* ModuleMember */))) {
-                            if (ts.isSourceFile(location) && location.commonJsModuleIndicator && !result.declarations.some(ts.isJSDocTypeAlias)) {
-                                result = undefined;
-                            }
-                            else {
-                                break loop;
-                            }
-                        }
-                        break;
-                    case 248 /* EnumDeclaration */:
-                        if (result = lookup(getSymbolOfNode(location).exports, name, meaning & 8 /* EnumMember */)) {
-                            break loop;
-                        }
-                        break;
-                    case 159 /* PropertyDeclaration */:
-                        // TypeScript 1.0 spec (April 2014): 8.4.1
-                        // Initializer expressions for instance member variables are evaluated in the scope
-                        // of the class constructor body but are not permitted to reference parameters or
-                        // local variables of the constructor. This effectively means that entities from outer scopes
-                        // by the same name as a constructor parameter or local variable are inaccessible
-                        // in initializer expressions for instance member variables.
-                        if (!ts.hasModifier(location, 32 /* Static */)) {
-                            var ctor = findConstructorDeclaration(location.parent);
-                            if (ctor && ctor.locals) {
-                                if (lookup(ctor.locals, name, meaning & 111551 /* Value */)) {
-                                    // Remember the property node, it will be used later to report appropriate error
-                                    propertyWithInvalidInitializer = location;
-                                }
-                            }
+                if (!symbol) {
+                    symbolTable.set(name, symbol = createSymbol(0 /* None */, name));
+                    if (isReplaceableByMethod)
+                        symbol.isReplaceableByMethod = true;
+                }
+                else if (isReplaceableByMethod && !symbol.isReplaceableByMethod) {
+                    // A symbol already exists, so don't add this as a declaration.
+                    return symbol;
+                }
+                else if (symbol.flags & excludes) {
+                    if (symbol.isReplaceableByMethod) {
+                        // Javascript constructor-declared symbols can be discarded in favor of
+                        // prototype symbols like methods.
+                        symbolTable.set(name, symbol = createSymbol(0 /* None */, name));
+                    }
+                    else if (!(includes & 3 /* Variable */ && symbol.flags & 67108864 /* Assignment */)) {
+                        // Assignment declarations are allowed to merge with variables, no matter what other flags they have.
+                        if (ts.isNamedDeclaration(node)) {
+                            ts.setParent(node.name, node);
                         }
-                        break;
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 246 /* InterfaceDeclaration */:
-                        // The below is used to lookup type parameters within a class or interface, as they are added to the class/interface locals
-                        // These can never be latebound, so the symbol's raw members are sufficient. `getMembersOfNode` cannot be used, as it would
-                        // trigger resolving late-bound names, which we may already be in the process of doing while we're here!
-                        if (result = lookup(getSymbolOfNode(location).members || emptySymbols, name, meaning & 788968 /* Type */)) {
-                            if (!isTypeParameterSymbolDeclaredInContainer(result, location)) {
-                                // ignore type parameters not declared in this container
-                                result = undefined;
-                                break;
-                            }
-                            if (lastLocation && ts.hasModifier(lastLocation, 32 /* Static */)) {
-                                // TypeScript 1.0 spec (April 2014): 3.4.1
-                                // The scope of a type parameter extends over the entire declaration with which the type
-                                // parameter list is associated, with the exception of static member declarations in classes.
-                                error(errorLocation, ts.Diagnostics.Static_members_cannot_reference_class_type_parameters);
-                                return undefined;
-                            }
-                            break loop;
+                        // Report errors every position with duplicate declaration
+                        // Report errors on previous encountered declarations
+                        var message_1 = symbol.flags & 2 /* BlockScopedVariable */
+                            ? ts.Diagnostics.Cannot_redeclare_block_scoped_variable_0
+                            : ts.Diagnostics.Duplicate_identifier_0;
+                        var messageNeedsName_1 = true;
+                        if (symbol.flags & 384 /* Enum */ || includes & 384 /* Enum */) {
+                            message_1 = ts.Diagnostics.Enum_declarations_can_only_merge_with_namespace_or_other_enum_declarations;
+                            messageNeedsName_1 = false;
                         }
-                        if (location.kind === 214 /* ClassExpression */ && meaning & 32 /* Class */) {
-                            var className = location.name;
-                            if (className && name === className.escapedText) {
-                                result = location.symbol;
-                                break loop;
+                        var multipleDefaultExports_1 = false;
+                        if (ts.length(symbol.declarations)) {
+                            // If the current node is a default export of some sort, then check if
+                            // there are any other default exports that we need to error on.
+                            // We'll know whether we have other default exports depending on if `symbol` already has a declaration list set.
+                            if (isDefaultExport) {
+                                message_1 = ts.Diagnostics.A_module_cannot_have_multiple_default_exports;
+                                messageNeedsName_1 = false;
+                                multipleDefaultExports_1 = true;
                             }
-                        }
-                        break;
-                    case 216 /* ExpressionWithTypeArguments */:
-                        // The type parameters of a class are not in scope in the base class expression.
-                        if (lastLocation === location.expression && location.parent.token === 90 /* ExtendsKeyword */) {
-                            var container = location.parent.parent;
-                            if (ts.isClassLike(container) && (result = lookup(getSymbolOfNode(container).members, name, meaning & 788968 /* Type */))) {
-                                if (nameNotFoundMessage) {
-                                    error(errorLocation, ts.Diagnostics.Base_class_expressions_cannot_reference_class_type_parameters);
+                            else {
+                                // This is to properly report an error in the case "export default { }" is after export default of class declaration or function declaration.
+                                // Error on multiple export default in the following case:
+                                // 1. multiple export default of class declaration or function declaration by checking NodeFlags.Default
+                                // 2. multiple export default of export assignment. This one doesn't have NodeFlags.Default on (as export default doesn't considered as modifiers)
+                                if (symbol.declarations && symbol.declarations.length &&
+                                    (node.kind === 266 /* ExportAssignment */ && !node.isExportEquals)) {
+                                    message_1 = ts.Diagnostics.A_module_cannot_have_multiple_default_exports;
+                                    messageNeedsName_1 = false;
+                                    multipleDefaultExports_1 = true;
                                 }
-                                return undefined;
-                            }
-                        }
-                        break;
-                    // It is not legal to reference a class's own type parameters from a computed property name that
-                    // belongs to the class. For example:
-                    //
-                    //   function foo<T>() { return '' }
-                    //   class C<T> { // <-- Class's own type parameter T
-                    //       [foo<T>()]() { } // <-- Reference to T from class's own computed property
-                    //   }
-                    //
-                    case 154 /* ComputedPropertyName */:
-                        grandparent = location.parent.parent;
-                        if (ts.isClassLike(grandparent) || grandparent.kind === 246 /* InterfaceDeclaration */) {
-                            // A reference to this grandparent's type parameters would be an error
-                            if (result = lookup(getSymbolOfNode(grandparent).members, name, meaning & 788968 /* Type */)) {
-                                error(errorLocation, ts.Diagnostics.A_computed_property_name_cannot_reference_a_type_parameter_from_its_containing_type);
-                                return undefined;
-                            }
-                        }
-                        break;
-                    case 202 /* ArrowFunction */:
-                        // when targeting ES6 or higher there is no 'arguments' in an arrow function
-                        // for lower compile targets the resolved symbol is used to emit an error
-                        if (compilerOptions.target >= 2 /* ES2015 */) {
-                            break;
-                        }
-                    // falls through
-                    case 161 /* MethodDeclaration */:
-                    case 162 /* Constructor */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                    case 244 /* FunctionDeclaration */:
-                        if (meaning & 3 /* Variable */ && name === "arguments") {
-                            result = argumentsSymbol;
-                            break loop;
-                        }
-                        break;
-                    case 201 /* FunctionExpression */:
-                        if (meaning & 3 /* Variable */ && name === "arguments") {
-                            result = argumentsSymbol;
-                            break loop;
-                        }
-                        if (meaning & 16 /* Function */) {
-                            var functionName = location.name;
-                            if (functionName && name === functionName.escapedText) {
-                                result = location.symbol;
-                                break loop;
                             }
                         }
-                        break;
-                    case 157 /* Decorator */:
-                        // Decorators are resolved at the class declaration. Resolving at the parameter
-                        // or member would result in looking up locals in the method.
-                        //
-                        //   function y() {}
-                        //   class C {
-                        //       method(@y x, y) {} // <-- decorator y should be resolved at the class declaration, not the parameter.
-                        //   }
-                        //
-                        if (location.parent && location.parent.kind === 156 /* Parameter */) {
-                            location = location.parent;
-                        }
-                        //
-                        //   function y() {}
-                        //   class C {
-                        //       @y method(x, y) {} // <-- decorator y should be resolved at the class declaration, not the method.
-                        //   }
-                        //
-                        // class Decorators are resolved outside of the class to avoid referencing type parameters of that class.
-                        //
-                        //   type T = number;
-                        //   declare function y(x: T): any;
-                        //   @param(1 as T) // <-- T should resolve to the type alias outside of class C
-                        //   class C<T> {}
-                        if (location.parent && (ts.isClassElement(location.parent) || location.parent.kind === 245 /* ClassDeclaration */)) {
-                            location = location.parent;
-                        }
-                        break;
-                    case 322 /* JSDocTypedefTag */:
-                    case 315 /* JSDocCallbackTag */:
-                    case 316 /* JSDocEnumTag */:
-                        // js type aliases do not resolve names from their host, so skip past it
-                        location = ts.getJSDocHost(location);
-                        break;
-                    case 156 /* Parameter */:
-                        if (lastLocation && (lastLocation === location.initializer ||
-                            lastLocation === location.name && ts.isBindingPattern(lastLocation))) {
-                            if (!associatedDeclarationForContainingInitializerOrBindingName) {
-                                associatedDeclarationForContainingInitializerOrBindingName = location;
-                            }
+                        var relatedInformation_1 = [];
+                        if (ts.isTypeAliasDeclaration(node) && ts.nodeIsMissing(node.type) && ts.hasSyntacticModifier(node, 1 /* Export */) && symbol.flags & (2097152 /* Alias */ | 788968 /* Type */ | 1920 /* Namespace */)) {
+                            // export type T; - may have meant export type { T }?
+                            relatedInformation_1.push(createDiagnosticForNode(node, ts.Diagnostics.Did_you_mean_0, "export type { " + ts.unescapeLeadingUnderscores(node.name.escapedText) + " }"));
                         }
-                        break;
-                    case 191 /* BindingElement */:
-                        if (lastLocation && (lastLocation === location.initializer ||
-                            lastLocation === location.name && ts.isBindingPattern(lastLocation))) {
-                            var root = ts.getRootDeclaration(location);
-                            if (root.kind === 156 /* Parameter */) {
-                                if (!associatedDeclarationForContainingInitializerOrBindingName) {
-                                    associatedDeclarationForContainingInitializerOrBindingName = location;
-                                }
+                        var declarationName_1 = ts.getNameOfDeclaration(node) || node;
+                        ts.forEach(symbol.declarations, function (declaration, index) {
+                            var decl = ts.getNameOfDeclaration(declaration) || declaration;
+                            var diag = createDiagnosticForNode(decl, message_1, messageNeedsName_1 ? getDisplayName(declaration) : undefined);
+                            file.bindDiagnostics.push(multipleDefaultExports_1 ? ts.addRelatedInfo(diag, createDiagnosticForNode(declarationName_1, index === 0 ? ts.Diagnostics.Another_export_default_is_here : ts.Diagnostics.and_here)) : diag);
+                            if (multipleDefaultExports_1) {
+                                relatedInformation_1.push(createDiagnosticForNode(decl, ts.Diagnostics.The_first_export_default_is_here));
                             }
-                        }
-                        break;
-                }
-                if (isSelfReferenceLocation(location)) {
-                    lastSelfReferenceLocation = location;
+                        });
+                        var diag = createDiagnosticForNode(declarationName_1, message_1, messageNeedsName_1 ? getDisplayName(node) : undefined);
+                        file.bindDiagnostics.push(ts.addRelatedInfo.apply(void 0, __spreadArrays([diag], relatedInformation_1)));
+                        symbol = createSymbol(0 /* None */, name);
+                    }
                 }
-                lastLocation = location;
-                location = location.parent;
             }
-            // We just climbed up parents looking for the name, meaning that we started in a descendant node of `lastLocation`.
-            // If `result === lastSelfReferenceLocation.symbol`, that means that we are somewhere inside `lastSelfReferenceLocation` looking up a name, and resolving to `lastLocation` itself.
-            // That means that this is a self-reference of `lastLocation`, and shouldn't count this when considering whether `lastLocation` is used.
-            if (isUse && result && (!lastSelfReferenceLocation || result !== lastSelfReferenceLocation.symbol)) {
-                result.isReferenced |= meaning;
+            addDeclarationToSymbol(symbol, node, includes);
+            if (symbol.parent) {
+                ts.Debug.assert(symbol.parent === parent, "Existing symbol parent should match new one");
             }
-            if (!result) {
-                if (lastLocation) {
-                    ts.Debug.assert(lastLocation.kind === 290 /* SourceFile */);
-                    if (lastLocation.commonJsModuleIndicator && name === "exports" && meaning & lastLocation.symbol.flags) {
-                        return lastLocation.symbol;
-                    }
+            else {
+                symbol.parent = parent;
+            }
+            return symbol;
+        }
+        function declareModuleMember(node, symbolFlags, symbolExcludes) {
+            var hasExportModifier = !!(ts.getCombinedModifierFlags(node) & 1 /* Export */) || jsdocTreatAsExported(node);
+            if (symbolFlags & 2097152 /* Alias */) {
+                if (node.kind === 270 /* ExportSpecifier */ || (node.kind === 260 /* ImportEqualsDeclaration */ && hasExportModifier)) {
+                    return declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes);
                 }
-                if (!excludeGlobals) {
-                    result = lookup(globals, name, meaning);
+                else {
+                    return declareSymbol(container.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
                 }
             }
-            if (!result) {
-                if (originalLocation && ts.isInJSFile(originalLocation) && originalLocation.parent) {
-                    if (ts.isRequireCall(originalLocation.parent, /*checkArgumentIsStringLiteralLike*/ false)) {
-                        return requireSymbol;
+            else {
+                // Exported module members are given 2 symbols: A local symbol that is classified with an ExportValue flag,
+                // and an associated export symbol with all the correct flags set on it. There are 2 main reasons:
+                //
+                //   1. We treat locals and exports of the same name as mutually exclusive within a container.
+                //      That means the binder will issue a Duplicate Identifier error if you mix locals and exports
+                //      with the same name in the same container.
+                //      TODO: Make this a more specific error and decouple it from the exclusion logic.
+                //   2. When we checkIdentifier in the checker, we set its resolved symbol to the local symbol,
+                //      but return the export symbol (by calling getExportSymbolOfValueSymbolIfExported). That way
+                //      when the emitter comes back to it, it knows not to qualify the name if it was found in a containing scope.
+                // NOTE: Nested ambient modules always should go to to 'locals' table to prevent their automatic merge
+                //       during global merging in the checker. Why? The only case when ambient module is permitted inside another module is module augmentation
+                //       and this case is specially handled. Module augmentations should only be merged with original module definition
+                //       and should never be merged directly with other augmentation, and the latter case would be possible if automatic merge is allowed.
+                if (ts.isJSDocTypeAlias(node))
+                    ts.Debug.assert(ts.isInJSFile(node)); // We shouldn't add symbols for JSDoc nodes if not in a JS file.
+                if (!ts.isAmbientModule(node) && (hasExportModifier || container.flags & 64 /* ExportContext */)) {
+                    if (!container.locals || (ts.hasSyntacticModifier(node, 512 /* Default */) && !getDeclarationName(node))) {
+                        return declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes); // No local symbol for an unnamed default!
                     }
+                    var exportKind = symbolFlags & 111551 /* Value */ ? 1048576 /* ExportValue */ : 0;
+                    var local = declareSymbol(container.locals, /*parent*/ undefined, node, exportKind, symbolExcludes);
+                    local.exportSymbol = declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes);
+                    node.localSymbol = local;
+                    return local;
                 }
-            }
-            if (!result) {
-                if (nameNotFoundMessage) {
-                    if (!errorLocation ||
-                        !checkAndReportErrorForMissingPrefix(errorLocation, name, nameArg) && // TODO: GH#18217
-                            !checkAndReportErrorForExtendingInterface(errorLocation) &&
-                            !checkAndReportErrorForUsingTypeAsNamespace(errorLocation, name, meaning) &&
-                            !checkAndReportErrorForExportingPrimitiveType(errorLocation, name) &&
-                            !checkAndReportErrorForUsingTypeAsValue(errorLocation, name, meaning) &&
-                            !checkAndReportErrorForUsingNamespaceModuleAsValue(errorLocation, name, meaning) &&
-                            !checkAndReportErrorForUsingValueAsType(errorLocation, name, meaning)) {
-                        var suggestion = void 0;
-                        if (suggestedNameNotFoundMessage && suggestionCount < maximumSuggestionCount) {
-                            suggestion = getSuggestedSymbolForNonexistentSymbol(originalLocation, name, meaning);
-                            if (suggestion) {
-                                var suggestionName = symbolToString(suggestion);
-                                var diagnostic = error(errorLocation, suggestedNameNotFoundMessage, diagnosticName(nameArg), suggestionName);
-                                if (suggestion.valueDeclaration) {
-                                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(suggestion.valueDeclaration, ts.Diagnostics._0_is_declared_here, suggestionName));
-                                }
-                            }
-                        }
-                        if (!suggestion) {
-                            error(errorLocation, nameNotFoundMessage, diagnosticName(nameArg));
-                        }
-                        suggestionCount++;
-                    }
+                else {
+                    return declareSymbol(container.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
                 }
-                return undefined;
             }
-            // Perform extra checks only if error reporting was requested
-            if (nameNotFoundMessage) {
-                if (propertyWithInvalidInitializer && !(compilerOptions.target === 99 /* ESNext */ && compilerOptions.useDefineForClassFields)) {
-                    // We have a match, but the reference occurred within a property initializer and the identifier also binds
-                    // to a local variable in the constructor where the code will be emitted. Note that this is actually allowed
-                    // with ESNext+useDefineForClassFields because the scope semantics are different.
-                    var propertyName = propertyWithInvalidInitializer.name;
-                    error(errorLocation, ts.Diagnostics.Initializer_of_instance_member_variable_0_cannot_reference_identifier_1_declared_in_the_constructor, ts.declarationNameToString(propertyName), diagnosticName(nameArg));
-                    return undefined;
+        }
+        function jsdocTreatAsExported(node) {
+            if (node.parent && ts.isModuleDeclaration(node)) {
+                node = node.parent;
+            }
+            if (!ts.isJSDocTypeAlias(node))
+                return false;
+            // jsdoc typedef handling is a bit of a doozy, but to summarize, treat the typedef as exported if:
+            // 1. It has an explicit name (since by default typedefs are always directly exported, either at the top level or in a container), or
+            if (!ts.isJSDocEnumTag(node) && !!node.fullName)
+                return true;
+            // 2. The thing a nameless typedef pulls its name from is implicitly a direct export (either by assignment or actual export flag).
+            var declName = ts.getNameOfDeclaration(node);
+            if (!declName)
+                return false;
+            if (ts.isPropertyAccessEntityNameExpression(declName.parent) && isTopLevelNamespaceAssignment(declName.parent))
+                return true;
+            if (ts.isDeclaration(declName.parent) && ts.getCombinedModifierFlags(declName.parent) & 1 /* Export */)
+                return true;
+            // This could potentially be simplified by having `delayedBindJSDocTypedefTag` pass in an override for `hasExportModifier`, since it should
+            // already have calculated and branched on most of this.
+            return false;
+        }
+        // All container nodes are kept on a linked list in declaration order. This list is used by
+        // the getLocalNameOfContainer function in the type checker to validate that the local name
+        // used for a container is unique.
+        function bindContainer(node, containerFlags) {
+            // Before we recurse into a node's children, we first save the existing parent, container
+            // and block-container.  Then after we pop out of processing the children, we restore
+            // these saved values.
+            var saveContainer = container;
+            var saveThisParentContainer = thisParentContainer;
+            var savedBlockScopeContainer = blockScopeContainer;
+            // Depending on what kind of node this is, we may have to adjust the current container
+            // and block-container.   If the current node is a container, then it is automatically
+            // considered the current block-container as well.  Also, for containers that we know
+            // may contain locals, we eagerly initialize the .locals field. We do this because
+            // it's highly likely that the .locals will be needed to place some child in (for example,
+            // a parameter, or variable declaration).
+            //
+            // However, we do not proactively create the .locals for block-containers because it's
+            // totally normal and common for block-containers to never actually have a block-scoped
+            // variable in them.  We don't want to end up allocating an object for every 'block' we
+            // run into when most of them won't be necessary.
+            //
+            // Finally, if this is a block-container, then we clear out any existing .locals object
+            // it may contain within it.  This happens in incremental scenarios.  Because we can be
+            // reusing a node from a previous compilation, that node may have had 'locals' created
+            // for it.  We must clear this so we don't accidentally move any stale data forward from
+            // a previous compilation.
+            if (containerFlags & 1 /* IsContainer */) {
+                if (node.kind !== 209 /* ArrowFunction */) {
+                    thisParentContainer = container;
                 }
-                // Only check for block-scoped variable if we have an error location and are looking for the
-                // name with variable meaning
-                //      For example,
-                //          declare module foo {
-                //              interface bar {}
-                //          }
-                //      const foo/*1*/: foo/*2*/.bar;
-                // The foo at /*1*/ and /*2*/ will share same symbol with two meanings:
-                // block-scoped variable and namespace module. However, only when we
-                // try to resolve name in /*1*/ which is used in variable position,
-                // we want to check for block-scoped
-                if (errorLocation &&
-                    (meaning & 2 /* BlockScopedVariable */ ||
-                        ((meaning & 32 /* Class */ || meaning & 384 /* Enum */) && (meaning & 111551 /* Value */) === 111551 /* Value */))) {
-                    var exportOrLocalSymbol = getExportSymbolOfValueSymbolIfExported(result);
-                    if (exportOrLocalSymbol.flags & 2 /* BlockScopedVariable */ || exportOrLocalSymbol.flags & 32 /* Class */ || exportOrLocalSymbol.flags & 384 /* Enum */) {
-                        checkResolvedBlockScopedVariable(exportOrLocalSymbol, errorLocation);
-                    }
+                container = blockScopeContainer = node;
+                if (containerFlags & 32 /* HasLocals */) {
+                    container.locals = ts.createSymbolTable();
                 }
-                // If we're in an external module, we can't reference value symbols created from UMD export declarations
-                if (result && isInExternalModule && (meaning & 111551 /* Value */) === 111551 /* Value */ && !(originalLocation.flags & 4194304 /* JSDoc */)) {
-                    var merged = getMergedSymbol(result);
-                    if (ts.length(merged.declarations) && ts.every(merged.declarations, function (d) { return ts.isNamespaceExportDeclaration(d) || ts.isSourceFile(d) && !!d.symbol.globalExports; })) {
-                        errorOrSuggestion(!compilerOptions.allowUmdGlobalAccess, errorLocation, ts.Diagnostics._0_refers_to_a_UMD_global_but_the_current_file_is_a_module_Consider_adding_an_import_instead, ts.unescapeLeadingUnderscores(name));
+                addToContainerChain(container);
+            }
+            else if (containerFlags & 2 /* IsBlockScopedContainer */) {
+                blockScopeContainer = node;
+                blockScopeContainer.locals = undefined;
+            }
+            if (containerFlags & 4 /* IsControlFlowContainer */) {
+                var saveCurrentFlow = currentFlow;
+                var saveBreakTarget = currentBreakTarget;
+                var saveContinueTarget = currentContinueTarget;
+                var saveReturnTarget = currentReturnTarget;
+                var saveExceptionTarget = currentExceptionTarget;
+                var saveActiveLabelList = activeLabelList;
+                var saveHasExplicitReturn = hasExplicitReturn;
+                var isIIFE = containerFlags & 16 /* IsFunctionExpression */ && !ts.hasSyntacticModifier(node, 256 /* Async */) &&
+                    !node.asteriskToken && !!ts.getImmediatelyInvokedFunctionExpression(node);
+                // A non-async, non-generator IIFE is considered part of the containing control flow. Return statements behave
+                // similarly to break statements that exit to a label just past the statement body.
+                if (!isIIFE) {
+                    currentFlow = initFlowNode({ flags: 2 /* Start */ });
+                    if (containerFlags & (16 /* IsFunctionExpression */ | 128 /* IsObjectLiteralOrClassExpressionMethod */)) {
+                        currentFlow.node = node;
                     }
                 }
-                // If we're in a parameter initializer or binding name, we can't reference the values of the parameter whose initializer we're within or parameters to the right
-                if (result && associatedDeclarationForContainingInitializerOrBindingName && !withinDeferredContext && (meaning & 111551 /* Value */) === 111551 /* Value */) {
-                    var candidate = getMergedSymbol(getLateBoundSymbol(result));
-                    var root = ts.getRootDeclaration(associatedDeclarationForContainingInitializerOrBindingName);
-                    // A parameter initializer or binding pattern initializer within a parameter cannot refer to itself
-                    if (candidate === getSymbolOfNode(associatedDeclarationForContainingInitializerOrBindingName)) {
-                        error(errorLocation, ts.Diagnostics.Parameter_0_cannot_reference_itself, ts.declarationNameToString(associatedDeclarationForContainingInitializerOrBindingName.name));
-                    }
-                    // And it cannot refer to any declarations which come after it
-                    else if (candidate.valueDeclaration && candidate.valueDeclaration.pos > associatedDeclarationForContainingInitializerOrBindingName.pos && root.parent.locals && lookup(root.parent.locals, candidate.escapedName, meaning) === candidate) {
-                        error(errorLocation, ts.Diagnostics.Parameter_0_cannot_reference_identifier_1_declared_after_it, ts.declarationNameToString(associatedDeclarationForContainingInitializerOrBindingName.name), ts.declarationNameToString(errorLocation));
-                    }
+                // We create a return control flow graph for IIFEs and constructors. For constructors
+                // we use the return control flow graph in strict property initialization checks.
+                currentReturnTarget = isIIFE || node.kind === 166 /* Constructor */ || (ts.isInJSFile && (node.kind === 251 /* FunctionDeclaration */ || node.kind === 208 /* FunctionExpression */)) ? createBranchLabel() : undefined;
+                currentExceptionTarget = undefined;
+                currentBreakTarget = undefined;
+                currentContinueTarget = undefined;
+                activeLabelList = undefined;
+                hasExplicitReturn = false;
+                bindChildren(node);
+                // Reset all reachability check related flags on node (for incremental scenarios)
+                node.flags &= ~2816 /* ReachabilityAndEmitFlags */;
+                if (!(currentFlow.flags & 1 /* Unreachable */) && containerFlags & 8 /* IsFunctionLike */ && ts.nodeIsPresent(node.body)) {
+                    node.flags |= 256 /* HasImplicitReturn */;
+                    if (hasExplicitReturn)
+                        node.flags |= 512 /* HasExplicitReturn */;
+                    node.endFlowNode = currentFlow;
                 }
-                if (result && errorLocation && meaning & 111551 /* Value */ && result.flags & 2097152 /* Alias */) {
-                    checkSymbolUsageInExpressionContext(result, name, errorLocation);
+                if (node.kind === 297 /* SourceFile */) {
+                    node.flags |= emitFlags;
                 }
-            }
-            return result;
-        }
-        function checkSymbolUsageInExpressionContext(symbol, name, useSite) {
-            if (!ts.isValidTypeOnlyAliasUseSite(useSite)) {
-                var typeOnlyDeclaration = getTypeOnlyAliasDeclaration(symbol);
-                if (typeOnlyDeclaration) {
-                    var isExport = ts.typeOnlyDeclarationIsExport(typeOnlyDeclaration);
-                    var message = isExport
-                        ? ts.Diagnostics._0_cannot_be_used_as_a_value_because_it_was_exported_using_export_type
-                        : ts.Diagnostics._0_cannot_be_used_as_a_value_because_it_was_imported_using_import_type;
-                    var relatedMessage = isExport
-                        ? ts.Diagnostics._0_was_exported_here
-                        : ts.Diagnostics._0_was_imported_here;
-                    var unescapedName = ts.unescapeLeadingUnderscores(name);
-                    ts.addRelatedInfo(error(useSite, message, unescapedName), ts.createDiagnosticForNode(typeOnlyDeclaration, relatedMessage, unescapedName));
+                if (currentReturnTarget) {
+                    addAntecedent(currentReturnTarget, currentFlow);
+                    currentFlow = finishFlowLabel(currentReturnTarget);
+                    if (node.kind === 166 /* Constructor */ || (ts.isInJSFile && (node.kind === 251 /* FunctionDeclaration */ || node.kind === 208 /* FunctionExpression */))) {
+                        node.returnFlowNode = currentFlow;
+                    }
                 }
+                if (!isIIFE) {
+                    currentFlow = saveCurrentFlow;
+                }
+                currentBreakTarget = saveBreakTarget;
+                currentContinueTarget = saveContinueTarget;
+                currentReturnTarget = saveReturnTarget;
+                currentExceptionTarget = saveExceptionTarget;
+                activeLabelList = saveActiveLabelList;
+                hasExplicitReturn = saveHasExplicitReturn;
             }
-        }
-        function getIsDeferredContext(location, lastLocation) {
-            if (location.kind !== 202 /* ArrowFunction */ && location.kind !== 201 /* FunctionExpression */) {
-                // initializers in instance property declaration of class like entities are executed in constructor and thus deferred
-                return ts.isTypeQueryNode(location) || ((ts.isFunctionLikeDeclaration(location) ||
-                    (location.kind === 159 /* PropertyDeclaration */ && !ts.hasModifier(location, 32 /* Static */))) && (!lastLocation || lastLocation !== location.name)); // A name is evaluated within the enclosing scope - so it shouldn't count as deferred
-            }
-            if (lastLocation && lastLocation === location.name) {
-                return false;
+            else if (containerFlags & 64 /* IsInterface */) {
+                seenThisKeyword = false;
+                bindChildren(node);
+                node.flags = seenThisKeyword ? node.flags | 128 /* ContainsThis */ : node.flags & ~128 /* ContainsThis */;
             }
-            // generator functions and async functions are not inlined in control flow when immediately invoked
-            if (location.asteriskToken || ts.hasModifier(location, 256 /* Async */)) {
-                return true;
+            else {
+                bindChildren(node);
             }
-            return !ts.getImmediatelyInvokedFunctionExpression(location);
+            container = saveContainer;
+            thisParentContainer = saveThisParentContainer;
+            blockScopeContainer = savedBlockScopeContainer;
         }
-        function isSelfReferenceLocation(node) {
-            switch (node.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 249 /* ModuleDeclaration */: // For `namespace N { N; }`
-                    return true;
-                default:
-                    return false;
+        function bindEachFunctionsFirst(nodes) {
+            bindEach(nodes, function (n) { return n.kind === 251 /* FunctionDeclaration */ ? bind(n) : undefined; });
+            bindEach(nodes, function (n) { return n.kind !== 251 /* FunctionDeclaration */ ? bind(n) : undefined; });
+        }
+        function bindEach(nodes, bindFunction) {
+            if (bindFunction === void 0) { bindFunction = bind; }
+            if (nodes === undefined) {
+                return;
             }
+            ts.forEach(nodes, bindFunction);
         }
-        function diagnosticName(nameArg) {
-            return ts.isString(nameArg) ? ts.unescapeLeadingUnderscores(nameArg) : ts.declarationNameToString(nameArg);
+        function bindEachChild(node) {
+            ts.forEachChild(node, bind, bindEach);
         }
-        function isTypeParameterSymbolDeclaredInContainer(symbol, container) {
-            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                var decl = _a[_i];
-                if (decl.kind === 155 /* TypeParameter */) {
-                    var parent = ts.isJSDocTemplateTag(decl.parent) ? ts.getJSDocHost(decl.parent) : decl.parent;
-                    if (parent === container) {
-                        return !(ts.isJSDocTemplateTag(decl.parent) && ts.find(decl.parent.parent.tags, ts.isJSDocTypeAlias)); // TODO: GH#18217
-                    }
-                }
+        function bindChildren(node) {
+            var saveInAssignmentPattern = inAssignmentPattern;
+            // Most nodes aren't valid in an assignment pattern, so we clear the value here
+            // and set it before we descend into nodes that could actually be part of an assignment pattern.
+            inAssignmentPattern = false;
+            if (checkUnreachable(node)) {
+                bindEachChild(node);
+                bindJSDoc(node);
+                inAssignmentPattern = saveInAssignmentPattern;
+                return;
             }
-            return false;
-        }
-        function checkAndReportErrorForMissingPrefix(errorLocation, name, nameArg) {
-            if (!ts.isIdentifier(errorLocation) || errorLocation.escapedText !== name || isTypeReferenceIdentifier(errorLocation) || isInTypeQuery(errorLocation)) {
-                return false;
+            if (node.kind >= 232 /* FirstStatement */ && node.kind <= 248 /* LastStatement */ && !options.allowUnreachableCode) {
+                node.flowNode = currentFlow;
             }
-            var container = ts.getThisContainer(errorLocation, /*includeArrowFunctions*/ false);
-            var location = container;
-            while (location) {
-                if (ts.isClassLike(location.parent)) {
-                    var classSymbol = getSymbolOfNode(location.parent);
-                    if (!classSymbol) {
-                        break;
-                    }
-                    // Check to see if a static member exists.
-                    var constructorType = getTypeOfSymbol(classSymbol);
-                    if (getPropertyOfType(constructorType, name)) {
-                        error(errorLocation, ts.Diagnostics.Cannot_find_name_0_Did_you_mean_the_static_member_1_0, diagnosticName(nameArg), symbolToString(classSymbol));
-                        return true;
-                    }
-                    // No static member is present.
-                    // Check if we're in an instance method and look for a relevant instance member.
-                    if (location === container && !ts.hasModifier(location, 32 /* Static */)) {
-                        var instanceType = getDeclaredTypeOfSymbol(classSymbol).thisType; // TODO: GH#18217
-                        if (getPropertyOfType(instanceType, name)) {
-                            error(errorLocation, ts.Diagnostics.Cannot_find_name_0_Did_you_mean_the_instance_member_this_0, diagnosticName(nameArg));
-                            return true;
-                        }
+            switch (node.kind) {
+                case 236 /* WhileStatement */:
+                    bindWhileStatement(node);
+                    break;
+                case 235 /* DoStatement */:
+                    bindDoStatement(node);
+                    break;
+                case 237 /* ForStatement */:
+                    bindForStatement(node);
+                    break;
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                    bindForInOrForOfStatement(node);
+                    break;
+                case 234 /* IfStatement */:
+                    bindIfStatement(node);
+                    break;
+                case 242 /* ReturnStatement */:
+                case 246 /* ThrowStatement */:
+                    bindReturnOrThrow(node);
+                    break;
+                case 241 /* BreakStatement */:
+                case 240 /* ContinueStatement */:
+                    bindBreakOrContinueStatement(node);
+                    break;
+                case 247 /* TryStatement */:
+                    bindTryStatement(node);
+                    break;
+                case 244 /* SwitchStatement */:
+                    bindSwitchStatement(node);
+                    break;
+                case 258 /* CaseBlock */:
+                    bindCaseBlock(node);
+                    break;
+                case 284 /* CaseClause */:
+                    bindCaseClause(node);
+                    break;
+                case 233 /* ExpressionStatement */:
+                    bindExpressionStatement(node);
+                    break;
+                case 245 /* LabeledStatement */:
+                    bindLabeledStatement(node);
+                    break;
+                case 214 /* PrefixUnaryExpression */:
+                    bindPrefixUnaryExpressionFlow(node);
+                    break;
+                case 215 /* PostfixUnaryExpression */:
+                    bindPostfixUnaryExpressionFlow(node);
+                    break;
+                case 216 /* BinaryExpression */:
+                    if (ts.isDestructuringAssignment(node)) {
+                        // Carry over whether we are in an assignment pattern to
+                        // binary expressions that could actually be an initializer
+                        inAssignmentPattern = saveInAssignmentPattern;
+                        bindDestructuringAssignmentFlow(node);
+                        return;
                     }
+                    bindBinaryExpressionFlow(node);
+                    break;
+                case 210 /* DeleteExpression */:
+                    bindDeleteExpressionFlow(node);
+                    break;
+                case 217 /* ConditionalExpression */:
+                    bindConditionalExpressionFlow(node);
+                    break;
+                case 249 /* VariableDeclaration */:
+                    bindVariableDeclarationFlow(node);
+                    break;
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    bindAccessExpressionFlow(node);
+                    break;
+                case 203 /* CallExpression */:
+                    bindCallExpressionFlow(node);
+                    break;
+                case 225 /* NonNullExpression */:
+                    bindNonNullExpressionFlow(node);
+                    break;
+                case 331 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 325 /* JSDocEnumTag */:
+                    bindJSDocTypeAlias(node);
+                    break;
+                // In source files and blocks, bind functions first to match hoisting that occurs at runtime
+                case 297 /* SourceFile */: {
+                    bindEachFunctionsFirst(node.statements);
+                    bind(node.endOfFileToken);
+                    break;
                 }
-                location = location.parent;
-            }
-            return false;
-        }
-        function checkAndReportErrorForExtendingInterface(errorLocation) {
-            var expression = getEntityNameForExtendingInterface(errorLocation);
-            if (expression && resolveEntityName(expression, 64 /* Interface */, /*ignoreErrors*/ true)) {
-                error(errorLocation, ts.Diagnostics.Cannot_extend_an_interface_0_Did_you_mean_implements, ts.getTextOfNode(expression));
-                return true;
-            }
-            return false;
-        }
-        /**
-         * Climbs up parents to an ExpressionWithTypeArguments, and returns its expression,
-         * but returns undefined if that expression is not an EntityNameExpression.
-         */
-        function getEntityNameForExtendingInterface(node) {
-            switch (node.kind) {
-                case 75 /* Identifier */:
-                case 194 /* PropertyAccessExpression */:
-                    return node.parent ? getEntityNameForExtendingInterface(node.parent) : undefined;
-                case 216 /* ExpressionWithTypeArguments */:
-                    if (ts.isEntityNameExpression(node.expression)) {
-                        return node.expression;
-                    }
+                case 230 /* Block */:
+                case 257 /* ModuleBlock */:
+                    bindEachFunctionsFirst(node.statements);
+                    break;
+                case 198 /* BindingElement */:
+                    bindBindingElementFlow(node);
+                    break;
+                case 200 /* ObjectLiteralExpression */:
+                case 199 /* ArrayLiteralExpression */:
+                case 288 /* PropertyAssignment */:
+                case 220 /* SpreadElement */:
+                    // Carry over whether we are in an assignment pattern of Object and Array literals
+                    // as well as their children that are valid assignment targets.
+                    inAssignmentPattern = saveInAssignmentPattern;
                 // falls through
                 default:
-                    return undefined;
+                    bindEachChild(node);
+                    break;
             }
+            bindJSDoc(node);
+            inAssignmentPattern = saveInAssignmentPattern;
         }
-        function checkAndReportErrorForUsingTypeAsNamespace(errorLocation, name, meaning) {
-            var namespaceMeaning = 1920 /* Namespace */ | (ts.isInJSFile(errorLocation) ? 111551 /* Value */ : 0);
-            if (meaning === namespaceMeaning) {
-                var symbol = resolveSymbol(resolveName(errorLocation, name, 788968 /* Type */ & ~namespaceMeaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
-                var parent = errorLocation.parent;
-                if (symbol) {
-                    if (ts.isQualifiedName(parent)) {
-                        ts.Debug.assert(parent.left === errorLocation, "Should only be resolving left side of qualified name as a namespace");
-                        var propName = parent.right.escapedText;
-                        var propType = getPropertyOfType(getDeclaredTypeOfSymbol(symbol), propName);
-                        if (propType) {
-                            error(parent, ts.Diagnostics.Cannot_access_0_1_because_0_is_a_type_but_not_a_namespace_Did_you_mean_to_retrieve_the_type_of_the_property_1_in_0_with_0_1, ts.unescapeLeadingUnderscores(name), ts.unescapeLeadingUnderscores(propName));
-                            return true;
-                        }
-                    }
-                    error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_namespace_here, ts.unescapeLeadingUnderscores(name));
-                    return true;
-                }
+        function isNarrowingExpression(expr) {
+            switch (expr.kind) {
+                case 78 /* Identifier */:
+                case 79 /* PrivateIdentifier */:
+                case 107 /* ThisKeyword */:
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    return containsNarrowableReference(expr);
+                case 203 /* CallExpression */:
+                    return hasNarrowableArgument(expr);
+                case 207 /* ParenthesizedExpression */:
+                case 225 /* NonNullExpression */:
+                    return isNarrowingExpression(expr.expression);
+                case 216 /* BinaryExpression */:
+                    return isNarrowingBinaryExpression(expr);
+                case 214 /* PrefixUnaryExpression */:
+                    return expr.operator === 53 /* ExclamationToken */ && isNarrowingExpression(expr.operand);
+                case 211 /* TypeOfExpression */:
+                    return isNarrowingExpression(expr.expression);
             }
             return false;
         }
-        function checkAndReportErrorForUsingValueAsType(errorLocation, name, meaning) {
-            if (meaning & (788968 /* Type */ & ~1920 /* Namespace */)) {
-                var symbol = resolveSymbol(resolveName(errorLocation, name, ~788968 /* Type */ & 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
-                if (symbol && !(symbol.flags & 1920 /* Namespace */)) {
-                    error(errorLocation, ts.Diagnostics._0_refers_to_a_value_but_is_being_used_as_a_type_here_Did_you_mean_typeof_0, ts.unescapeLeadingUnderscores(name));
-                    return true;
-                }
-            }
-            return false;
+        function isNarrowableReference(expr) {
+            return expr.kind === 78 /* Identifier */ || expr.kind === 79 /* PrivateIdentifier */ || expr.kind === 107 /* ThisKeyword */ || expr.kind === 105 /* SuperKeyword */ ||
+                (ts.isPropertyAccessExpression(expr) || ts.isNonNullExpression(expr) || ts.isParenthesizedExpression(expr)) && isNarrowableReference(expr.expression) ||
+                ts.isBinaryExpression(expr) && expr.operatorToken.kind === 27 /* CommaToken */ && isNarrowableReference(expr.right) ||
+                ts.isElementAccessExpression(expr) && ts.isStringOrNumericLiteralLike(expr.argumentExpression) && isNarrowableReference(expr.expression) ||
+                ts.isAssignmentExpression(expr) && isNarrowableReference(expr.left);
         }
-        function isPrimitiveTypeName(name) {
-            return name === "any" || name === "string" || name === "number" || name === "boolean" || name === "never" || name === "unknown";
+        function containsNarrowableReference(expr) {
+            return isNarrowableReference(expr) || ts.isOptionalChain(expr) && containsNarrowableReference(expr.expression);
         }
-        function checkAndReportErrorForExportingPrimitiveType(errorLocation, name) {
-            if (isPrimitiveTypeName(name) && errorLocation.parent.kind === 263 /* ExportSpecifier */) {
-                error(errorLocation, ts.Diagnostics.Cannot_export_0_Only_local_declarations_can_be_exported_from_a_module, name);
+        function hasNarrowableArgument(expr) {
+            if (expr.arguments) {
+                for (var _i = 0, _a = expr.arguments; _i < _a.length; _i++) {
+                    var argument = _a[_i];
+                    if (containsNarrowableReference(argument)) {
+                        return true;
+                    }
+                }
+            }
+            if (expr.expression.kind === 201 /* PropertyAccessExpression */ &&
+                containsNarrowableReference(expr.expression.expression)) {
                 return true;
             }
             return false;
         }
-        function checkAndReportErrorForUsingTypeAsValue(errorLocation, name, meaning) {
-            if (meaning & (111551 /* Value */ & ~1024 /* NamespaceModule */)) {
-                if (isPrimitiveTypeName(name)) {
-                    error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here, ts.unescapeLeadingUnderscores(name));
-                    return true;
-                }
-                var symbol = resolveSymbol(resolveName(errorLocation, name, 788968 /* Type */ & ~111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
-                if (symbol && !(symbol.flags & 1024 /* NamespaceModule */)) {
-                    var message = isES2015OrLaterConstructorName(name)
-                        ? ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_es2015_or_later
-                        : ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here;
-                    error(errorLocation, message, ts.unescapeLeadingUnderscores(name));
-                    return true;
-                }
-            }
-            return false;
+        function isNarrowingTypeofOperands(expr1, expr2) {
+            return ts.isTypeOfExpression(expr1) && isNarrowableOperand(expr1.expression) && ts.isStringLiteralLike(expr2);
         }
-        function isES2015OrLaterConstructorName(n) {
-            switch (n) {
-                case "Promise":
-                case "Symbol":
-                case "Map":
-                case "WeakMap":
-                case "Set":
-                case "WeakSet":
-                    return true;
-            }
-            return false;
+        function isNarrowableInOperands(left, right) {
+            return ts.isStringLiteralLike(left) && isNarrowingExpression(right);
         }
-        function checkAndReportErrorForUsingNamespaceModuleAsValue(errorLocation, name, meaning) {
-            if (meaning & (111551 /* Value */ & ~1024 /* NamespaceModule */ & ~788968 /* Type */)) {
-                var symbol = resolveSymbol(resolveName(errorLocation, name, 1024 /* NamespaceModule */ & ~111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
-                if (symbol) {
-                    error(errorLocation, ts.Diagnostics.Cannot_use_namespace_0_as_a_value, ts.unescapeLeadingUnderscores(name));
-                    return true;
-                }
-            }
-            else if (meaning & (788968 /* Type */ & ~1024 /* NamespaceModule */ & ~111551 /* Value */)) {
-                var symbol = resolveSymbol(resolveName(errorLocation, name, (512 /* ValueModule */ | 1024 /* NamespaceModule */) & ~788968 /* Type */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
-                if (symbol) {
-                    error(errorLocation, ts.Diagnostics.Cannot_use_namespace_0_as_a_type, ts.unescapeLeadingUnderscores(name));
-                    return true;
-                }
+        function isNarrowingBinaryExpression(expr) {
+            switch (expr.operatorToken.kind) {
+                case 62 /* EqualsToken */:
+                case 74 /* BarBarEqualsToken */:
+                case 75 /* AmpersandAmpersandEqualsToken */:
+                case 76 /* QuestionQuestionEqualsToken */:
+                    return containsNarrowableReference(expr.left);
+                case 34 /* EqualsEqualsToken */:
+                case 35 /* ExclamationEqualsToken */:
+                case 36 /* EqualsEqualsEqualsToken */:
+                case 37 /* ExclamationEqualsEqualsToken */:
+                    return isNarrowableOperand(expr.left) || isNarrowableOperand(expr.right) ||
+                        isNarrowingTypeofOperands(expr.right, expr.left) || isNarrowingTypeofOperands(expr.left, expr.right);
+                case 101 /* InstanceOfKeyword */:
+                    return isNarrowableOperand(expr.left);
+                case 100 /* InKeyword */:
+                    return isNarrowableInOperands(expr.left, expr.right);
+                case 27 /* CommaToken */:
+                    return isNarrowingExpression(expr.right);
             }
             return false;
         }
-        function checkResolvedBlockScopedVariable(result, errorLocation) {
-            ts.Debug.assert(!!(result.flags & 2 /* BlockScopedVariable */ || result.flags & 32 /* Class */ || result.flags & 384 /* Enum */));
-            if (result.flags & (16 /* Function */ | 1 /* FunctionScopedVariable */ | 67108864 /* Assignment */) && result.flags & 32 /* Class */) {
-                // constructor functions aren't block scoped
-                return;
-            }
-            // Block-scoped variables cannot be used before their definition
-            var declaration = ts.find(result.declarations, function (d) { return ts.isBlockOrCatchScoped(d) || ts.isClassLike(d) || (d.kind === 248 /* EnumDeclaration */); });
-            if (declaration === undefined)
-                return ts.Debug.fail("checkResolvedBlockScopedVariable could not find block-scoped declaration");
-            if (!(declaration.flags & 8388608 /* Ambient */) && !isBlockScopedNameDeclaredBeforeUse(declaration, errorLocation)) {
-                var diagnosticMessage = void 0;
-                var declarationName = ts.declarationNameToString(ts.getNameOfDeclaration(declaration));
-                if (result.flags & 2 /* BlockScopedVariable */) {
-                    diagnosticMessage = error(errorLocation, ts.Diagnostics.Block_scoped_variable_0_used_before_its_declaration, declarationName);
-                }
-                else if (result.flags & 32 /* Class */) {
-                    diagnosticMessage = error(errorLocation, ts.Diagnostics.Class_0_used_before_its_declaration, declarationName);
-                }
-                else if (result.flags & 256 /* RegularEnum */) {
-                    diagnosticMessage = error(errorLocation, ts.Diagnostics.Enum_0_used_before_its_declaration, declarationName);
-                }
-                else {
-                    ts.Debug.assert(!!(result.flags & 128 /* ConstEnum */));
-                    if (compilerOptions.preserveConstEnums) {
-                        diagnosticMessage = error(errorLocation, ts.Diagnostics.Class_0_used_before_its_declaration, declarationName);
+        function isNarrowableOperand(expr) {
+            switch (expr.kind) {
+                case 207 /* ParenthesizedExpression */:
+                    return isNarrowableOperand(expr.expression);
+                case 216 /* BinaryExpression */:
+                    switch (expr.operatorToken.kind) {
+                        case 62 /* EqualsToken */:
+                            return isNarrowableOperand(expr.left);
+                        case 27 /* CommaToken */:
+                            return isNarrowableOperand(expr.right);
                     }
-                }
-                if (diagnosticMessage) {
-                    ts.addRelatedInfo(diagnosticMessage, ts.createDiagnosticForNode(declaration, ts.Diagnostics._0_is_declared_here, declarationName));
-                }
-            }
-        }
-        /* Starting from 'initial' node walk up the parent chain until 'stopAt' node is reached.
-         * If at any point current node is equal to 'parent' node - return true.
-         * Return false if 'stopAt' node is reached or isFunctionLike(current) === true.
-         */
-        function isSameScopeDescendentOf(initial, parent, stopAt) {
-            return !!parent && !!ts.findAncestor(initial, function (n) { return n === stopAt || ts.isFunctionLike(n) ? "quit" : n === parent; });
-        }
-        function getAnyImportSyntax(node) {
-            switch (node.kind) {
-                case 253 /* ImportEqualsDeclaration */:
-                    return node;
-                case 255 /* ImportClause */:
-                    return node.parent;
-                case 256 /* NamespaceImport */:
-                    return node.parent.parent;
-                case 258 /* ImportSpecifier */:
-                    return node.parent.parent.parent;
-                default:
-                    return undefined;
             }
+            return containsNarrowableReference(expr);
         }
-        function getDeclarationOfAliasSymbol(symbol) {
-            return ts.find(symbol.declarations, isAliasSymbolDeclaration);
-        }
-        /**
-         * An alias symbol is created by one of the following declarations:
-         * import <symbol> = ...
-         * import <symbol> from ...
-         * import * as <symbol> from ...
-         * import { x as <symbol> } from ...
-         * export { x as <symbol> } from ...
-         * export * as ns <symbol> from ...
-         * export = <EntityNameExpression>
-         * export default <EntityNameExpression>
-         * module.exports = <EntityNameExpression>
-         * {<Identifier>}
-         * {name: <EntityNameExpression>}
-         */
-        function isAliasSymbolDeclaration(node) {
-            return node.kind === 253 /* ImportEqualsDeclaration */ ||
-                node.kind === 252 /* NamespaceExportDeclaration */ ||
-                node.kind === 255 /* ImportClause */ && !!node.name ||
-                node.kind === 256 /* NamespaceImport */ ||
-                node.kind === 262 /* NamespaceExport */ ||
-                node.kind === 258 /* ImportSpecifier */ ||
-                node.kind === 263 /* ExportSpecifier */ ||
-                node.kind === 259 /* ExportAssignment */ && ts.exportAssignmentIsAlias(node) ||
-                ts.isBinaryExpression(node) && ts.getAssignmentDeclarationKind(node) === 2 /* ModuleExports */ && ts.exportAssignmentIsAlias(node) ||
-                ts.isPropertyAccessExpression(node)
-                    && ts.isBinaryExpression(node.parent)
-                    && node.parent.left === node
-                    && node.parent.operatorToken.kind === 62 /* EqualsToken */
-                    && isAliasableOrJsExpression(node.parent.right) ||
-                node.kind === 282 /* ShorthandPropertyAssignment */ ||
-                node.kind === 281 /* PropertyAssignment */ && isAliasableOrJsExpression(node.initializer);
+        function createBranchLabel() {
+            return initFlowNode({ flags: 4 /* BranchLabel */, antecedents: undefined });
         }
-        function isAliasableOrJsExpression(e) {
-            return ts.isAliasableExpression(e) || ts.isFunctionExpression(e) && isJSConstructor(e);
+        function createLoopLabel() {
+            return initFlowNode({ flags: 8 /* LoopLabel */, antecedents: undefined });
         }
-        function getTargetOfImportEqualsDeclaration(node, dontResolveAlias) {
-            if (node.moduleReference.kind === 265 /* ExternalModuleReference */) {
-                var immediate = resolveExternalModuleName(node, ts.getExternalModuleImportEqualsDeclarationExpression(node));
-                var resolved_4 = resolveExternalModuleSymbol(immediate);
-                markSymbolOfAliasDeclarationIfTypeOnly(node, immediate, resolved_4, /*overwriteEmpty*/ false);
-                return resolved_4;
-            }
-            var resolved = getSymbolOfPartOfRightHandSideOfImportEquals(node.moduleReference, dontResolveAlias);
-            checkAndReportErrorForResolvingImportAliasToTypeOnlySymbol(node, resolved);
-            return resolved;
+        function createReduceLabel(target, antecedents, antecedent) {
+            return initFlowNode({ flags: 1024 /* ReduceLabel */, target: target, antecedents: antecedents, antecedent: antecedent });
         }
-        function checkAndReportErrorForResolvingImportAliasToTypeOnlySymbol(node, resolved) {
-            if (markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false)) {
-                var typeOnlyDeclaration = getTypeOnlyAliasDeclaration(getSymbolOfNode(node));
-                var isExport = ts.typeOnlyDeclarationIsExport(typeOnlyDeclaration);
-                var message = isExport
-                    ? ts.Diagnostics.An_import_alias_cannot_reference_a_declaration_that_was_exported_using_export_type
-                    : ts.Diagnostics.An_import_alias_cannot_reference_a_declaration_that_was_imported_using_import_type;
-                var relatedMessage = isExport
-                    ? ts.Diagnostics._0_was_exported_here
-                    : ts.Diagnostics._0_was_imported_here;
-                // Non-null assertion is safe because the optionality comes from ImportClause,
-                // but if an ImportClause was the typeOnlyDeclaration, it had to have a `name`.
-                var name = ts.unescapeLeadingUnderscores(typeOnlyDeclaration.name.escapedText);
-                ts.addRelatedInfo(error(node.moduleReference, message), ts.createDiagnosticForNode(typeOnlyDeclaration, relatedMessage, name));
-            }
+        function setFlowNodeReferenced(flow) {
+            // On first reference we set the Referenced flag, thereafter we set the Shared flag
+            flow.flags |= flow.flags & 2048 /* Referenced */ ? 4096 /* Shared */ : 2048 /* Referenced */;
         }
-        function resolveExportByName(moduleSymbol, name, sourceNode, dontResolveAlias) {
-            var exportValue = moduleSymbol.exports.get("export=" /* ExportEquals */);
-            if (exportValue) {
-                return getPropertyOfType(getTypeOfSymbol(exportValue), name);
+        function addAntecedent(label, antecedent) {
+            if (!(antecedent.flags & 1 /* Unreachable */) && !ts.contains(label.antecedents, antecedent)) {
+                (label.antecedents || (label.antecedents = [])).push(antecedent);
+                setFlowNodeReferenced(antecedent);
             }
-            var exportSymbol = moduleSymbol.exports.get(name);
-            var resolved = resolveSymbol(exportSymbol, dontResolveAlias);
-            markSymbolOfAliasDeclarationIfTypeOnly(sourceNode, exportSymbol, resolved, /*overwriteEmpty*/ false);
-            return resolved;
         }
-        function isSyntacticDefault(node) {
-            return ((ts.isExportAssignment(node) && !node.isExportEquals) || ts.hasModifier(node, 512 /* Default */) || ts.isExportSpecifier(node));
-        }
-        function canHaveSyntheticDefault(file, moduleSymbol, dontResolveAlias) {
-            if (!allowSyntheticDefaultImports) {
-                return false;
-            }
-            // Declaration files (and ambient modules)
-            if (!file || file.isDeclarationFile) {
-                // Definitely cannot have a synthetic default if they have a syntactic default member specified
-                var defaultExportSymbol = resolveExportByName(moduleSymbol, "default" /* Default */, /*sourceNode*/ undefined, /*dontResolveAlias*/ true); // Dont resolve alias because we want the immediately exported symbol's declaration
-                if (defaultExportSymbol && ts.some(defaultExportSymbol.declarations, isSyntacticDefault)) {
-                    return false;
-                }
-                // It _might_ still be incorrect to assume there is no __esModule marker on the import at runtime, even if there is no `default` member
-                // So we check a bit more,
-                if (resolveExportByName(moduleSymbol, ts.escapeLeadingUnderscores("__esModule"), /*sourceNode*/ undefined, dontResolveAlias)) {
-                    // If there is an `__esModule` specified in the declaration (meaning someone explicitly added it or wrote it in their code),
-                    // it definitely is a module and does not have a synthetic default
-                    return false;
-                }
-                // There are _many_ declaration files not written with esmodules in mind that still get compiled into a format with __esModule set
-                // Meaning there may be no default at runtime - however to be on the permissive side, we allow access to a synthetic default member
-                // as there is no marker to indicate if the accompanying JS has `__esModule` or not, or is even native esm
-                return true;
-            }
-            // TypeScript files never have a synthetic default (as they are always emitted with an __esModule marker) _unless_ they contain an export= statement
-            if (!ts.isSourceFileJS(file)) {
-                return hasExportAssignmentSymbol(moduleSymbol);
+        function createFlowCondition(flags, antecedent, expression) {
+            if (antecedent.flags & 1 /* Unreachable */) {
+                return antecedent;
             }
-            // JS files have a synthetic default if they do not contain ES2015+ module syntax (export = is not valid in js) _and_ do not have an __esModule marker
-            return !file.externalModuleIndicator && !resolveExportByName(moduleSymbol, ts.escapeLeadingUnderscores("__esModule"), /*sourceNode*/ undefined, dontResolveAlias);
-        }
-        function getTargetOfImportClause(node, dontResolveAlias) {
-            var moduleSymbol = resolveExternalModuleName(node, node.parent.moduleSpecifier);
-            if (moduleSymbol) {
-                var exportDefaultSymbol = void 0;
-                if (ts.isShorthandAmbientModuleSymbol(moduleSymbol)) {
-                    exportDefaultSymbol = moduleSymbol;
-                }
-                else {
-                    exportDefaultSymbol = resolveExportByName(moduleSymbol, "default" /* Default */, node, dontResolveAlias);
-                }
-                var file = ts.find(moduleSymbol.declarations, ts.isSourceFile);
-                var hasSyntheticDefault = canHaveSyntheticDefault(file, moduleSymbol, dontResolveAlias);
-                if (!exportDefaultSymbol && !hasSyntheticDefault) {
-                    if (hasExportAssignmentSymbol(moduleSymbol)) {
-                        var compilerOptionName = moduleKind >= ts.ModuleKind.ES2015 ? "allowSyntheticDefaultImports" : "esModuleInterop";
-                        var exportEqualsSymbol = moduleSymbol.exports.get("export=" /* ExportEquals */);
-                        var exportAssignment = exportEqualsSymbol.valueDeclaration;
-                        var err = error(node.name, ts.Diagnostics.Module_0_can_only_be_default_imported_using_the_1_flag, symbolToString(moduleSymbol), compilerOptionName);
-                        ts.addRelatedInfo(err, ts.createDiagnosticForNode(exportAssignment, ts.Diagnostics.This_module_is_declared_with_using_export_and_can_only_be_used_with_a_default_import_when_using_the_0_flag, compilerOptionName));
-                    }
-                    else {
-                        reportNonDefaultExport(moduleSymbol, node);
-                    }
-                }
-                else if (hasSyntheticDefault) {
-                    // per emit behavior, a synthetic default overrides a "real" .default member if `__esModule` is not present
-                    var resolved = resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias) || resolveSymbol(moduleSymbol, dontResolveAlias);
-                    markSymbolOfAliasDeclarationIfTypeOnly(node, moduleSymbol, resolved, /*overwriteTypeOnly*/ false);
-                    return resolved;
-                }
-                markSymbolOfAliasDeclarationIfTypeOnly(node, exportDefaultSymbol, /*finalTarget*/ undefined, /*overwriteTypeOnly*/ false);
-                return exportDefaultSymbol;
+            if (!expression) {
+                return flags & 32 /* TrueCondition */ ? antecedent : unreachableFlow;
             }
-        }
-        function reportNonDefaultExport(moduleSymbol, node) {
-            var _a, _b;
-            if ((_a = moduleSymbol.exports) === null || _a === void 0 ? void 0 : _a.has(node.symbol.escapedName)) {
-                error(node.name, ts.Diagnostics.Module_0_has_no_default_export_Did_you_mean_to_use_import_1_from_0_instead, symbolToString(moduleSymbol), symbolToString(node.symbol));
+            if ((expression.kind === 109 /* TrueKeyword */ && flags & 64 /* FalseCondition */ ||
+                expression.kind === 94 /* FalseKeyword */ && flags & 32 /* TrueCondition */) &&
+                !ts.isExpressionOfOptionalChainRoot(expression) && !ts.isNullishCoalesce(expression.parent)) {
+                return unreachableFlow;
             }
-            else {
-                var diagnostic = error(node.name, ts.Diagnostics.Module_0_has_no_default_export, symbolToString(moduleSymbol));
-                var exportStar = (_b = moduleSymbol.exports) === null || _b === void 0 ? void 0 : _b.get("__export" /* ExportStar */);
-                if (exportStar) {
-                    var defaultExport = ts.find(exportStar.declarations, function (decl) {
-                        var _a, _b;
-                        return !!(ts.isExportDeclaration(decl) && decl.moduleSpecifier && ((_b = (_a = resolveExternalModuleName(decl, decl.moduleSpecifier)) === null || _a === void 0 ? void 0 : _a.exports) === null || _b === void 0 ? void 0 : _b.has("default" /* Default */)));
-                    });
-                    if (defaultExport) {
-                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(defaultExport, ts.Diagnostics.export_Asterisk_does_not_re_export_a_default));
-                    }
-                }
+            if (!isNarrowingExpression(expression)) {
+                return antecedent;
             }
+            setFlowNodeReferenced(antecedent);
+            return initFlowNode({ flags: flags, antecedent: antecedent, node: expression });
         }
-        function getTargetOfNamespaceImport(node, dontResolveAlias) {
-            var moduleSpecifier = node.parent.parent.moduleSpecifier;
-            var immediate = resolveExternalModuleName(node, moduleSpecifier);
-            var resolved = resolveESModuleSymbol(immediate, moduleSpecifier, dontResolveAlias, /*suppressUsageError*/ false);
-            markSymbolOfAliasDeclarationIfTypeOnly(node, immediate, resolved, /*overwriteEmpty*/ false);
-            return resolved;
-        }
-        function getTargetOfNamespaceExport(node, dontResolveAlias) {
-            var moduleSpecifier = node.parent.moduleSpecifier;
-            var immediate = moduleSpecifier && resolveExternalModuleName(node, moduleSpecifier);
-            var resolved = moduleSpecifier && resolveESModuleSymbol(immediate, moduleSpecifier, dontResolveAlias, /*suppressUsageError*/ false);
-            markSymbolOfAliasDeclarationIfTypeOnly(node, immediate, resolved, /*overwriteEmpty*/ false);
-            return resolved;
+        function createFlowSwitchClause(antecedent, switchStatement, clauseStart, clauseEnd) {
+            setFlowNodeReferenced(antecedent);
+            return initFlowNode({ flags: 128 /* SwitchClause */, antecedent: antecedent, switchStatement: switchStatement, clauseStart: clauseStart, clauseEnd: clauseEnd });
         }
-        // This function creates a synthetic symbol that combines the value side of one symbol with the
-        // type/namespace side of another symbol. Consider this example:
-        //
-        //   declare module graphics {
-        //       interface Point {
-        //           x: number;
-        //           y: number;
-        //       }
-        //   }
-        //   declare var graphics: {
-        //       Point: new (x: number, y: number) => graphics.Point;
-        //   }
-        //   declare module "graphics" {
-        //       export = graphics;
-        //   }
-        //
-        // An 'import { Point } from "graphics"' needs to create a symbol that combines the value side 'Point'
-        // property with the type/namespace side interface 'Point'.
-        function combineValueAndTypeSymbols(valueSymbol, typeSymbol) {
-            if (valueSymbol === unknownSymbol && typeSymbol === unknownSymbol) {
-                return unknownSymbol;
-            }
-            if (valueSymbol.flags & (788968 /* Type */ | 1920 /* Namespace */)) {
-                return valueSymbol;
+        function createFlowMutation(flags, antecedent, node) {
+            setFlowNodeReferenced(antecedent);
+            var result = initFlowNode({ flags: flags, antecedent: antecedent, node: node });
+            if (currentExceptionTarget) {
+                addAntecedent(currentExceptionTarget, result);
             }
-            var result = createSymbol(valueSymbol.flags | typeSymbol.flags, valueSymbol.escapedName);
-            result.declarations = ts.deduplicate(ts.concatenate(valueSymbol.declarations, typeSymbol.declarations), ts.equateValues);
-            result.parent = valueSymbol.parent || typeSymbol.parent;
-            if (valueSymbol.valueDeclaration)
-                result.valueDeclaration = valueSymbol.valueDeclaration;
-            if (typeSymbol.members)
-                result.members = ts.cloneMap(typeSymbol.members);
-            if (valueSymbol.exports)
-                result.exports = ts.cloneMap(valueSymbol.exports);
             return result;
         }
-        function getExportOfModule(symbol, specifier, dontResolveAlias) {
-            var _a;
-            if (symbol.flags & 1536 /* Module */) {
-                var name = ((_a = specifier.propertyName) !== null && _a !== void 0 ? _a : specifier.name).escapedText;
-                var exportSymbol = getExportsOfSymbol(symbol).get(name);
-                var resolved = resolveSymbol(exportSymbol, dontResolveAlias);
-                markSymbolOfAliasDeclarationIfTypeOnly(specifier, exportSymbol, resolved, /*overwriteEmpty*/ false);
-                return resolved;
-            }
+        function createFlowCall(antecedent, node) {
+            setFlowNodeReferenced(antecedent);
+            return initFlowNode({ flags: 512 /* Call */, antecedent: antecedent, node: node });
         }
-        function getPropertyOfVariable(symbol, name) {
-            if (symbol.flags & 3 /* Variable */) {
-                var typeAnnotation = symbol.valueDeclaration.type;
-                if (typeAnnotation) {
-                    return resolveSymbol(getPropertyOfType(getTypeFromTypeNode(typeAnnotation), name));
-                }
+        function finishFlowLabel(flow) {
+            var antecedents = flow.antecedents;
+            if (!antecedents) {
+                return unreachableFlow;
             }
-        }
-        function getExternalModuleMember(node, specifier, dontResolveAlias) {
-            var _a;
-            if (dontResolveAlias === void 0) { dontResolveAlias = false; }
-            var moduleSymbol = resolveExternalModuleName(node, node.moduleSpecifier); // TODO: GH#18217
-            var name = specifier.propertyName || specifier.name;
-            var suppressInteropError = name.escapedText === "default" /* Default */ && !!(compilerOptions.allowSyntheticDefaultImports || compilerOptions.esModuleInterop);
-            var targetSymbol = resolveESModuleSymbol(moduleSymbol, node.moduleSpecifier, dontResolveAlias, suppressInteropError);
-            if (targetSymbol) {
-                if (name.escapedText) {
-                    if (ts.isShorthandAmbientModuleSymbol(moduleSymbol)) {
-                        return moduleSymbol;
-                    }
-                    var symbolFromVariable = void 0;
-                    // First check if module was specified with "export=". If so, get the member from the resolved type
-                    if (moduleSymbol && moduleSymbol.exports && moduleSymbol.exports.get("export=" /* ExportEquals */)) {
-                        symbolFromVariable = getPropertyOfType(getTypeOfSymbol(targetSymbol), name.escapedText);
-                    }
-                    else {
-                        symbolFromVariable = getPropertyOfVariable(targetSymbol, name.escapedText);
-                    }
-                    // if symbolFromVariable is export - get its final target
-                    symbolFromVariable = resolveSymbol(symbolFromVariable, dontResolveAlias);
-                    var symbolFromModule = getExportOfModule(targetSymbol, specifier, dontResolveAlias);
-                    if (symbolFromModule === undefined && name.escapedText === "default" /* Default */) {
-                        var file = ts.find(moduleSymbol.declarations, ts.isSourceFile);
-                        if (canHaveSyntheticDefault(file, moduleSymbol, dontResolveAlias)) {
-                            symbolFromModule = resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias) || resolveSymbol(moduleSymbol, dontResolveAlias);
-                        }
-                    }
-                    var symbol = symbolFromModule && symbolFromVariable && symbolFromModule !== symbolFromVariable ?
-                        combineValueAndTypeSymbols(symbolFromVariable, symbolFromModule) :
-                        symbolFromModule || symbolFromVariable;
-                    if (!symbol) {
-                        var moduleName = getFullyQualifiedName(moduleSymbol, node);
-                        var declarationName = ts.declarationNameToString(name);
-                        var suggestion = getSuggestedSymbolForNonexistentModule(name, targetSymbol);
-                        if (suggestion !== undefined) {
-                            var suggestionName = symbolToString(suggestion);
-                            var diagnostic = error(name, ts.Diagnostics.Module_0_has_no_exported_member_1_Did_you_mean_2, moduleName, declarationName, suggestionName);
-                            if (suggestion.valueDeclaration) {
-                                ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(suggestion.valueDeclaration, ts.Diagnostics._0_is_declared_here, suggestionName));
-                            }
-                        }
-                        else {
-                            if ((_a = moduleSymbol.exports) === null || _a === void 0 ? void 0 : _a.has("default" /* Default */)) {
-                                error(name, ts.Diagnostics.Module_0_has_no_exported_member_1_Did_you_mean_to_use_import_1_from_0_instead, moduleName, declarationName);
-                            }
-                            else {
-                                reportNonExportedMember(node, name, declarationName, moduleSymbol, moduleName);
-                            }
-                        }
-                    }
-                    return symbol;
-                }
+            if (antecedents.length === 1) {
+                return antecedents[0];
             }
+            return flow;
         }
-        function reportNonExportedMember(node, name, declarationName, moduleSymbol, moduleName) {
-            var _a;
-            var localSymbol = (_a = moduleSymbol.valueDeclaration.locals) === null || _a === void 0 ? void 0 : _a.get(name.escapedText);
-            var exports = moduleSymbol.exports;
-            if (localSymbol) {
-                var exportedEqualsSymbol = exports === null || exports === void 0 ? void 0 : exports.get("export=" /* ExportEquals */);
-                if (exportedEqualsSymbol) {
-                    getSymbolIfSameReference(exportedEqualsSymbol, localSymbol) ? reportInvalidImportEqualsExportMember(node, name, declarationName, moduleName) :
-                        error(name, ts.Diagnostics.Module_0_has_no_exported_member_1, moduleName, declarationName);
-                }
-                else {
-                    var exportedSymbol = exports ? ts.find(symbolsToArray(exports), function (symbol) { return !!getSymbolIfSameReference(symbol, localSymbol); }) : undefined;
-                    var diagnostic = exportedSymbol ? error(name, ts.Diagnostics.Module_0_declares_1_locally_but_it_is_exported_as_2, moduleName, declarationName, symbolToString(exportedSymbol)) :
-                        error(name, ts.Diagnostics.Module_0_declares_1_locally_but_it_is_not_exported, moduleName, declarationName);
-                    ts.addRelatedInfo.apply(void 0, __spreadArrays([diagnostic], ts.map(localSymbol.declarations, function (decl, index) {
-                        return ts.createDiagnosticForNode(decl, index === 0 ? ts.Diagnostics._0_is_declared_here : ts.Diagnostics.and_here, declarationName);
-                    })));
-                }
-            }
-            else {
-                error(name, ts.Diagnostics.Module_0_has_no_exported_member_1, moduleName, declarationName);
+        function isStatementCondition(node) {
+            var parent = node.parent;
+            switch (parent.kind) {
+                case 234 /* IfStatement */:
+                case 236 /* WhileStatement */:
+                case 235 /* DoStatement */:
+                    return parent.expression === node;
+                case 237 /* ForStatement */:
+                case 217 /* ConditionalExpression */:
+                    return parent.condition === node;
             }
+            return false;
         }
-        function reportInvalidImportEqualsExportMember(node, name, declarationName, moduleName) {
-            if (moduleKind >= ts.ModuleKind.ES2015) {
-                var message = compilerOptions.esModuleInterop ? ts.Diagnostics._0_can_only_be_imported_by_using_a_default_import :
-                    ts.Diagnostics._0_can_only_be_imported_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import;
-                error(name, message, declarationName);
-            }
-            else {
-                if (ts.isInJSFile(node)) {
-                    var message = compilerOptions.esModuleInterop ? ts.Diagnostics._0_can_only_be_imported_by_using_a_require_call_or_by_using_a_default_import :
-                        ts.Diagnostics._0_can_only_be_imported_by_using_a_require_call_or_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import;
-                    error(name, message, declarationName);
+        function isLogicalExpression(node) {
+            while (true) {
+                if (node.kind === 207 /* ParenthesizedExpression */) {
+                    node = node.expression;
+                }
+                else if (node.kind === 214 /* PrefixUnaryExpression */ && node.operator === 53 /* ExclamationToken */) {
+                    node = node.operand;
                 }
                 else {
-                    var message = compilerOptions.esModuleInterop ? ts.Diagnostics._0_can_only_be_imported_by_using_import_1_require_2_or_a_default_import :
-                        ts.Diagnostics._0_can_only_be_imported_by_using_import_1_require_2_or_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import;
-                    error(name, message, declarationName, declarationName, moduleName);
+                    return node.kind === 216 /* BinaryExpression */ && (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */ ||
+                        node.operatorToken.kind === 56 /* BarBarToken */ ||
+                        node.operatorToken.kind === 60 /* QuestionQuestionToken */);
                 }
             }
         }
-        function getTargetOfImportSpecifier(node, dontResolveAlias) {
-            var resolved = getExternalModuleMember(node.parent.parent.parent, node, dontResolveAlias);
-            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
-            return resolved;
-        }
-        function getTargetOfNamespaceExportDeclaration(node, dontResolveAlias) {
-            var resolved = resolveExternalModuleSymbol(node.parent.symbol, dontResolveAlias);
-            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
-            return resolved;
-        }
-        function getTargetOfExportSpecifier(node, meaning, dontResolveAlias) {
-            var resolved = node.parent.parent.moduleSpecifier ?
-                getExternalModuleMember(node.parent.parent, node, dontResolveAlias) :
-                resolveEntityName(node.propertyName || node.name, meaning, /*ignoreErrors*/ false, dontResolveAlias);
-            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
-            return resolved;
-        }
-        function getTargetOfExportAssignment(node, dontResolveAlias) {
-            var expression = ts.isExportAssignment(node) ? node.expression : node.right;
-            var resolved = getTargetOfAliasLikeExpression(expression, dontResolveAlias);
-            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
-            return resolved;
+        function isLogicalAssignmentExpression(node) {
+            node = ts.skipParentheses(node);
+            return ts.isBinaryExpression(node) && ts.isLogicalOrCoalescingAssignmentOperator(node.operatorToken.kind);
         }
-        function getTargetOfAliasLikeExpression(expression, dontResolveAlias) {
-            if (ts.isClassExpression(expression)) {
-                return checkExpressionCached(expression).symbol;
-            }
-            if (!ts.isEntityName(expression) && !ts.isEntityNameExpression(expression)) {
-                return undefined;
-            }
-            var aliasLike = resolveEntityName(expression, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, /*ignoreErrors*/ true, dontResolveAlias);
-            if (aliasLike) {
-                return aliasLike;
+        function isTopLevelLogicalExpression(node) {
+            while (ts.isParenthesizedExpression(node.parent) ||
+                ts.isPrefixUnaryExpression(node.parent) && node.parent.operator === 53 /* ExclamationToken */) {
+                node = node.parent;
             }
-            checkExpressionCached(expression);
-            return getNodeLinks(expression).resolvedSymbol;
+            return !isStatementCondition(node) &&
+                !isLogicalAssignmentExpression(node.parent) &&
+                !isLogicalExpression(node.parent) &&
+                !(ts.isOptionalChain(node.parent) && node.parent.expression === node);
         }
-        function getTargetOfPropertyAssignment(node, dontRecursivelyResolve) {
-            var expression = node.initializer;
-            return getTargetOfAliasLikeExpression(expression, dontRecursivelyResolve);
+        function doWithConditionalBranches(action, value, trueTarget, falseTarget) {
+            var savedTrueTarget = currentTrueTarget;
+            var savedFalseTarget = currentFalseTarget;
+            currentTrueTarget = trueTarget;
+            currentFalseTarget = falseTarget;
+            action(value);
+            currentTrueTarget = savedTrueTarget;
+            currentFalseTarget = savedFalseTarget;
         }
-        function getTargetOfPropertyAccessExpression(node, dontRecursivelyResolve) {
-            if (!(ts.isBinaryExpression(node.parent) && node.parent.left === node && node.parent.operatorToken.kind === 62 /* EqualsToken */)) {
-                return undefined;
+        function bindCondition(node, trueTarget, falseTarget) {
+            doWithConditionalBranches(bind, node, trueTarget, falseTarget);
+            if (!node || !isLogicalAssignmentExpression(node) && !isLogicalExpression(node) && !(ts.isOptionalChain(node) && ts.isOutermostOptionalChain(node))) {
+                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
+                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
             }
-            return getTargetOfAliasLikeExpression(node.parent.right, dontRecursivelyResolve);
         }
-        function getTargetOfAliasDeclaration(node, dontRecursivelyResolve) {
-            if (dontRecursivelyResolve === void 0) { dontRecursivelyResolve = false; }
-            switch (node.kind) {
-                case 253 /* ImportEqualsDeclaration */:
-                    return getTargetOfImportEqualsDeclaration(node, dontRecursivelyResolve);
-                case 255 /* ImportClause */:
-                    return getTargetOfImportClause(node, dontRecursivelyResolve);
-                case 256 /* NamespaceImport */:
-                    return getTargetOfNamespaceImport(node, dontRecursivelyResolve);
-                case 262 /* NamespaceExport */:
-                    return getTargetOfNamespaceExport(node, dontRecursivelyResolve);
-                case 258 /* ImportSpecifier */:
-                    return getTargetOfImportSpecifier(node, dontRecursivelyResolve);
-                case 263 /* ExportSpecifier */:
-                    return getTargetOfExportSpecifier(node, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, dontRecursivelyResolve);
-                case 259 /* ExportAssignment */:
-                case 209 /* BinaryExpression */:
-                    return getTargetOfExportAssignment(node, dontRecursivelyResolve);
-                case 252 /* NamespaceExportDeclaration */:
-                    return getTargetOfNamespaceExportDeclaration(node, dontRecursivelyResolve);
-                case 282 /* ShorthandPropertyAssignment */:
-                    return resolveEntityName(node.name, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, /*ignoreErrors*/ true, dontRecursivelyResolve);
-                case 281 /* PropertyAssignment */:
-                    return getTargetOfPropertyAssignment(node, dontRecursivelyResolve);
-                case 194 /* PropertyAccessExpression */:
-                    return getTargetOfPropertyAccessExpression(node, dontRecursivelyResolve);
-                default:
-                    return ts.Debug.fail();
-            }
+        function bindIterativeStatement(node, breakTarget, continueTarget) {
+            var saveBreakTarget = currentBreakTarget;
+            var saveContinueTarget = currentContinueTarget;
+            currentBreakTarget = breakTarget;
+            currentContinueTarget = continueTarget;
+            bind(node);
+            currentBreakTarget = saveBreakTarget;
+            currentContinueTarget = saveContinueTarget;
         }
-        /**
-         * Indicates that a symbol is an alias that does not merge with a local declaration.
-         * OR Is a JSContainer which may merge an alias with a local declaration
-         */
-        function isNonLocalAlias(symbol, excludes) {
-            if (excludes === void 0) { excludes = 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */; }
-            if (!symbol)
-                return false;
-            return (symbol.flags & (2097152 /* Alias */ | excludes)) === 2097152 /* Alias */ || !!(symbol.flags & 2097152 /* Alias */ && symbol.flags & 67108864 /* Assignment */);
+        function setContinueTarget(node, target) {
+            var label = activeLabelList;
+            while (label && node.parent.kind === 245 /* LabeledStatement */) {
+                label.continueTarget = target;
+                label = label.next;
+                node = node.parent;
+            }
+            return target;
         }
-        function resolveSymbol(symbol, dontResolveAlias) {
-            return !dontResolveAlias && isNonLocalAlias(symbol) ? resolveAlias(symbol) : symbol;
+        function bindWhileStatement(node) {
+            var preWhileLabel = setContinueTarget(node, createLoopLabel());
+            var preBodyLabel = createBranchLabel();
+            var postWhileLabel = createBranchLabel();
+            addAntecedent(preWhileLabel, currentFlow);
+            currentFlow = preWhileLabel;
+            bindCondition(node.expression, preBodyLabel, postWhileLabel);
+            currentFlow = finishFlowLabel(preBodyLabel);
+            bindIterativeStatement(node.statement, postWhileLabel, preWhileLabel);
+            addAntecedent(preWhileLabel, currentFlow);
+            currentFlow = finishFlowLabel(postWhileLabel);
         }
-        function resolveAlias(symbol) {
-            ts.Debug.assert((symbol.flags & 2097152 /* Alias */) !== 0, "Should only get Alias here.");
-            var links = getSymbolLinks(symbol);
-            if (!links.target) {
-                links.target = resolvingSymbol;
-                var node = getDeclarationOfAliasSymbol(symbol);
-                if (!node)
-                    return ts.Debug.fail();
-                var target = getTargetOfAliasDeclaration(node);
-                if (links.target === resolvingSymbol) {
-                    links.target = target || unknownSymbol;
-                }
-                else {
-                    error(node, ts.Diagnostics.Circular_definition_of_import_alias_0, symbolToString(symbol));
-                }
-            }
-            else if (links.target === resolvingSymbol) {
-                links.target = unknownSymbol;
-            }
-            return links.target;
+        function bindDoStatement(node) {
+            var preDoLabel = createLoopLabel();
+            var preConditionLabel = setContinueTarget(node, createBranchLabel());
+            var postDoLabel = createBranchLabel();
+            addAntecedent(preDoLabel, currentFlow);
+            currentFlow = preDoLabel;
+            bindIterativeStatement(node.statement, postDoLabel, preConditionLabel);
+            addAntecedent(preConditionLabel, currentFlow);
+            currentFlow = finishFlowLabel(preConditionLabel);
+            bindCondition(node.expression, preDoLabel, postDoLabel);
+            currentFlow = finishFlowLabel(postDoLabel);
         }
-        function tryResolveAlias(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (links.target !== resolvingSymbol) {
-                return resolveAlias(symbol);
-            }
-            return undefined;
+        function bindForStatement(node) {
+            var preLoopLabel = setContinueTarget(node, createLoopLabel());
+            var preBodyLabel = createBranchLabel();
+            var postLoopLabel = createBranchLabel();
+            bind(node.initializer);
+            addAntecedent(preLoopLabel, currentFlow);
+            currentFlow = preLoopLabel;
+            bindCondition(node.condition, preBodyLabel, postLoopLabel);
+            currentFlow = finishFlowLabel(preBodyLabel);
+            bindIterativeStatement(node.statement, postLoopLabel, preLoopLabel);
+            bind(node.incrementor);
+            addAntecedent(preLoopLabel, currentFlow);
+            currentFlow = finishFlowLabel(postLoopLabel);
         }
-        /**
-         * Marks a symbol as type-only if its declaration is syntactically type-only.
-         * If it is not itself marked type-only, but resolves to a type-only alias
-         * somewhere in its resolution chain, save a reference to the type-only alias declaration
-         * so the alias _not_ marked type-only can be identified as _transitively_ type-only.
-         *
-         * This function is called on each alias declaration that could be type-only or resolve to
-         * another type-only alias during `resolveAlias`, so that later, when an alias is used in a
-         * JS-emitting expression, we can quickly determine if that symbol is effectively type-only
-         * and issue an error if so.
-         *
-         * @param aliasDeclaration The alias declaration not marked as type-only
-         * has already been marked as not resolving to a type-only alias. Used when recursively resolving qualified
-         * names of import aliases, e.g. `import C = a.b.C`. If namespace `a` is not found to be type-only, the
-         * import declaration will initially be marked as not resolving to a type-only symbol. But, namespace `b`
-         * must still be checked for a type-only marker, overwriting the previous negative result if found.
-         * @param immediateTarget The symbol to which the alias declaration immediately resolves
-         * @param finalTarget The symbol to which the alias declaration ultimately resolves
-         * @param overwriteEmpty Checks `resolvesToSymbol` for type-only declarations even if `aliasDeclaration`
-         */
-        function markSymbolOfAliasDeclarationIfTypeOnly(aliasDeclaration, immediateTarget, finalTarget, overwriteEmpty) {
-            if (!aliasDeclaration)
-                return false;
-            // If the declaration itself is type-only, mark it and return.
-            // No need to check what it resolves to.
-            var sourceSymbol = getSymbolOfNode(aliasDeclaration);
-            if (ts.isTypeOnlyImportOrExportDeclaration(aliasDeclaration)) {
-                var links_1 = getSymbolLinks(sourceSymbol);
-                links_1.typeOnlyDeclaration = aliasDeclaration;
-                return true;
+        function bindForInOrForOfStatement(node) {
+            var preLoopLabel = setContinueTarget(node, createLoopLabel());
+            var postLoopLabel = createBranchLabel();
+            bind(node.expression);
+            addAntecedent(preLoopLabel, currentFlow);
+            currentFlow = preLoopLabel;
+            if (node.kind === 239 /* ForOfStatement */) {
+                bind(node.awaitModifier);
             }
-            var links = getSymbolLinks(sourceSymbol);
-            return markSymbolOfAliasDeclarationIfTypeOnlyWorker(links, immediateTarget, overwriteEmpty)
-                || markSymbolOfAliasDeclarationIfTypeOnlyWorker(links, finalTarget, overwriteEmpty);
-        }
-        function markSymbolOfAliasDeclarationIfTypeOnlyWorker(aliasDeclarationLinks, target, overwriteEmpty) {
-            var _a, _b, _c;
-            if (target && (aliasDeclarationLinks.typeOnlyDeclaration === undefined || overwriteEmpty && aliasDeclarationLinks.typeOnlyDeclaration === false)) {
-                var exportSymbol = (_b = (_a = target.exports) === null || _a === void 0 ? void 0 : _a.get("export=" /* ExportEquals */)) !== null && _b !== void 0 ? _b : target;
-                var typeOnly = exportSymbol.declarations && ts.find(exportSymbol.declarations, ts.isTypeOnlyImportOrExportDeclaration);
-                aliasDeclarationLinks.typeOnlyDeclaration = (_c = typeOnly !== null && typeOnly !== void 0 ? typeOnly : getSymbolLinks(exportSymbol).typeOnlyDeclaration) !== null && _c !== void 0 ? _c : false;
+            addAntecedent(postLoopLabel, currentFlow);
+            bind(node.initializer);
+            if (node.initializer.kind !== 250 /* VariableDeclarationList */) {
+                bindAssignmentTargetFlow(node.initializer);
             }
-            return !!aliasDeclarationLinks.typeOnlyDeclaration;
+            bindIterativeStatement(node.statement, postLoopLabel, preLoopLabel);
+            addAntecedent(preLoopLabel, currentFlow);
+            currentFlow = finishFlowLabel(postLoopLabel);
         }
-        /** Indicates that a symbol directly or indirectly resolves to a type-only import or export. */
-        function getTypeOnlyAliasDeclaration(symbol) {
-            if (!(symbol.flags & 2097152 /* Alias */)) {
-                return undefined;
-            }
-            var links = getSymbolLinks(symbol);
-            return links.typeOnlyDeclaration || undefined;
+        function bindIfStatement(node) {
+            var thenLabel = createBranchLabel();
+            var elseLabel = createBranchLabel();
+            var postIfLabel = createBranchLabel();
+            bindCondition(node.expression, thenLabel, elseLabel);
+            currentFlow = finishFlowLabel(thenLabel);
+            bind(node.thenStatement);
+            addAntecedent(postIfLabel, currentFlow);
+            currentFlow = finishFlowLabel(elseLabel);
+            bind(node.elseStatement);
+            addAntecedent(postIfLabel, currentFlow);
+            currentFlow = finishFlowLabel(postIfLabel);
         }
-        function markExportAsReferenced(node) {
-            var symbol = getSymbolOfNode(node);
-            var target = resolveAlias(symbol);
-            if (target) {
-                var markAlias = target === unknownSymbol ||
-                    ((target.flags & 111551 /* Value */) && !isConstEnumOrConstEnumOnlyModule(target) && !getTypeOnlyAliasDeclaration(symbol));
-                if (markAlias) {
-                    markAliasSymbolAsReferenced(symbol);
+        function bindReturnOrThrow(node) {
+            bind(node.expression);
+            if (node.kind === 242 /* ReturnStatement */) {
+                hasExplicitReturn = true;
+                if (currentReturnTarget) {
+                    addAntecedent(currentReturnTarget, currentFlow);
                 }
             }
+            currentFlow = unreachableFlow;
         }
-        // When an alias symbol is referenced, we need to mark the entity it references as referenced and in turn repeat that until
-        // we reach a non-alias or an exported entity (which is always considered referenced). We do this by checking the target of
-        // the alias as an expression (which recursively takes us back here if the target references another alias).
-        function markAliasSymbolAsReferenced(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.referenced) {
-                links.referenced = true;
-                var node = getDeclarationOfAliasSymbol(symbol);
-                if (!node)
-                    return ts.Debug.fail();
-                // We defer checking of the reference of an `import =` until the import itself is referenced,
-                // This way a chain of imports can be elided if ultimately the final input is only used in a type
-                // position.
-                if (ts.isInternalModuleImportEqualsDeclaration(node)) {
-                    var target = resolveSymbol(symbol);
-                    if (target === unknownSymbol || target.flags & 111551 /* Value */) {
-                        // import foo = <symbol>
-                        checkExpressionCached(node.moduleReference);
-                    }
+        function findActiveLabel(name) {
+            for (var label = activeLabelList; label; label = label.next) {
+                if (label.name === name) {
+                    return label;
                 }
             }
+            return undefined;
         }
-        // Aliases that resolve to const enums are not marked as referenced because they are not emitted,
-        // but their usage in value positions must be tracked to determine if the import can be type-only.
-        function markConstEnumAliasAsReferenced(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.constEnumReferenced) {
-                links.constEnumReferenced = true;
+        function bindBreakOrContinueFlow(node, breakTarget, continueTarget) {
+            var flowLabel = node.kind === 241 /* BreakStatement */ ? breakTarget : continueTarget;
+            if (flowLabel) {
+                addAntecedent(flowLabel, currentFlow);
+                currentFlow = unreachableFlow;
             }
         }
-        // This function is only for imports with entity names
-        function getSymbolOfPartOfRightHandSideOfImportEquals(entityName, dontResolveAlias) {
-            // There are three things we might try to look for. In the following examples,
-            // the search term is enclosed in |...|:
-            //
-            //     import a = |b|; // Namespace
-            //     import a = |b.c|; // Value, type, namespace
-            //     import a = |b.c|.d; // Namespace
-            if (entityName.kind === 75 /* Identifier */ && ts.isRightSideOfQualifiedNameOrPropertyAccess(entityName)) {
-                entityName = entityName.parent;
-            }
-            // Check for case 1 and 3 in the above example
-            if (entityName.kind === 75 /* Identifier */ || entityName.parent.kind === 153 /* QualifiedName */) {
-                return resolveEntityName(entityName, 1920 /* Namespace */, /*ignoreErrors*/ false, dontResolveAlias);
+        function bindBreakOrContinueStatement(node) {
+            bind(node.label);
+            if (node.label) {
+                var activeLabel = findActiveLabel(node.label.escapedText);
+                if (activeLabel) {
+                    activeLabel.referenced = true;
+                    bindBreakOrContinueFlow(node, activeLabel.breakTarget, activeLabel.continueTarget);
+                }
             }
             else {
-                // Case 2 in above example
-                // entityName.kind could be a QualifiedName or a Missing identifier
-                ts.Debug.assert(entityName.parent.kind === 253 /* ImportEqualsDeclaration */);
-                return resolveEntityName(entityName, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, /*ignoreErrors*/ false, dontResolveAlias);
+                bindBreakOrContinueFlow(node, currentBreakTarget, currentContinueTarget);
             }
         }
-        function getFullyQualifiedName(symbol, containingLocation) {
-            return symbol.parent ? getFullyQualifiedName(symbol.parent, containingLocation) + "." + symbolToString(symbol) : symbolToString(symbol, containingLocation, /*meaning*/ undefined, 16 /* DoNotIncludeSymbolChain */ | 4 /* AllowAnyNodeKind */);
-        }
-        /**
-         * Resolves a qualified name and any involved aliases.
-         */
-        function resolveEntityName(name, meaning, ignoreErrors, dontResolveAlias, location) {
-            if (ts.nodeIsMissing(name)) {
-                return undefined;
+        function bindTryStatement(node) {
+            // We conservatively assume that *any* code in the try block can cause an exception, but we only need
+            // to track code that causes mutations (because only mutations widen the possible control flow type of
+            // a variable). The exceptionLabel is the target label for control flows that result from exceptions.
+            // We add all mutation flow nodes as antecedents of this label such that we can analyze them as possible
+            // antecedents of the start of catch or finally blocks. Furthermore, we add the current control flow to
+            // represent exceptions that occur before any mutations.
+            var saveReturnTarget = currentReturnTarget;
+            var saveExceptionTarget = currentExceptionTarget;
+            var normalExitLabel = createBranchLabel();
+            var returnLabel = createBranchLabel();
+            var exceptionLabel = createBranchLabel();
+            if (node.finallyBlock) {
+                currentReturnTarget = returnLabel;
             }
-            var namespaceMeaning = 1920 /* Namespace */ | (ts.isInJSFile(name) ? meaning & 111551 /* Value */ : 0);
-            var symbol;
-            if (name.kind === 75 /* Identifier */) {
-                var message = meaning === namespaceMeaning || ts.nodeIsSynthesized(name) ? ts.Diagnostics.Cannot_find_namespace_0 : getCannotFindNameDiagnosticForName(ts.getFirstIdentifier(name));
-                var symbolFromJSPrototype = ts.isInJSFile(name) && !ts.nodeIsSynthesized(name) ? resolveEntityNameFromAssignmentDeclaration(name, meaning) : undefined;
-                symbol = getMergedSymbol(resolveName(location || name, name.escapedText, meaning, ignoreErrors || symbolFromJSPrototype ? undefined : message, name, /*isUse*/ true));
-                if (!symbol) {
-                    return getMergedSymbol(symbolFromJSPrototype);
-                }
+            addAntecedent(exceptionLabel, currentFlow);
+            currentExceptionTarget = exceptionLabel;
+            bind(node.tryBlock);
+            addAntecedent(normalExitLabel, currentFlow);
+            if (node.catchClause) {
+                // Start of catch clause is the target of exceptions from try block.
+                currentFlow = finishFlowLabel(exceptionLabel);
+                // The currentExceptionTarget now represents control flows from exceptions in the catch clause.
+                // Effectively, in a try-catch-finally, if an exception occurs in the try block, the catch block
+                // acts like a second try block.
+                exceptionLabel = createBranchLabel();
+                addAntecedent(exceptionLabel, currentFlow);
+                currentExceptionTarget = exceptionLabel;
+                bind(node.catchClause);
+                addAntecedent(normalExitLabel, currentFlow);
             }
-            else if (name.kind === 153 /* QualifiedName */ || name.kind === 194 /* PropertyAccessExpression */) {
-                var left = name.kind === 153 /* QualifiedName */ ? name.left : name.expression;
-                var right = name.kind === 153 /* QualifiedName */ ? name.right : name.name;
-                var namespace = resolveEntityName(left, namespaceMeaning, ignoreErrors, /*dontResolveAlias*/ false, location);
-                if (!namespace || ts.nodeIsMissing(right)) {
-                    return undefined;
-                }
-                else if (namespace === unknownSymbol) {
-                    return namespace;
+            currentReturnTarget = saveReturnTarget;
+            currentExceptionTarget = saveExceptionTarget;
+            if (node.finallyBlock) {
+                // Possible ways control can reach the finally block:
+                // 1) Normal completion of try block of a try-finally or try-catch-finally
+                // 2) Normal completion of catch block (following exception in try block) of a try-catch-finally
+                // 3) Return in try or catch block of a try-finally or try-catch-finally
+                // 4) Exception in try block of a try-finally
+                // 5) Exception in catch block of a try-catch-finally
+                // When analyzing a control flow graph that starts inside a finally block we want to consider all
+                // five possibilities above. However, when analyzing a control flow graph that starts outside (past)
+                // the finally block, we only want to consider the first two (if we're past a finally block then it
+                // must have completed normally). Likewise, when analyzing a control flow graph from return statements
+                // in try or catch blocks in an IIFE, we only want to consider the third. To make this possible, we
+                // inject a ReduceLabel node into the control flow graph. This node contains an alternate reduced
+                // set of antecedents for the pre-finally label. As control flow analysis passes by a ReduceLabel
+                // node, the pre-finally label is temporarily switched to the reduced antecedent set.
+                var finallyLabel = createBranchLabel();
+                finallyLabel.antecedents = ts.concatenate(ts.concatenate(normalExitLabel.antecedents, exceptionLabel.antecedents), returnLabel.antecedents);
+                currentFlow = finallyLabel;
+                bind(node.finallyBlock);
+                if (currentFlow.flags & 1 /* Unreachable */) {
+                    // If the end of the finally block is unreachable, the end of the entire try statement is unreachable.
+                    currentFlow = unreachableFlow;
                 }
-                if (ts.isInJSFile(name)) {
-                    if (namespace.valueDeclaration &&
-                        ts.isVariableDeclaration(namespace.valueDeclaration) &&
-                        namespace.valueDeclaration.initializer &&
-                        isCommonJsRequire(namespace.valueDeclaration.initializer)) {
-                        var moduleName = namespace.valueDeclaration.initializer.arguments[0];
-                        var moduleSym = resolveExternalModuleName(moduleName, moduleName);
-                        if (moduleSym) {
-                            var resolvedModuleSymbol = resolveExternalModuleSymbol(moduleSym);
-                            if (resolvedModuleSymbol) {
-                                namespace = resolvedModuleSymbol;
-                            }
-                        }
+                else {
+                    // If we have an IIFE return target and return statements in the try or catch blocks, add a control
+                    // flow that goes back through the finally block and back through only the return statements.
+                    if (currentReturnTarget && returnLabel.antecedents) {
+                        addAntecedent(currentReturnTarget, createReduceLabel(finallyLabel, returnLabel.antecedents, currentFlow));
                     }
-                }
-                symbol = getMergedSymbol(getSymbol(getExportsOfSymbol(namespace), right.escapedText, meaning));
-                if (!symbol) {
-                    if (!ignoreErrors) {
-                        error(right, ts.Diagnostics.Namespace_0_has_no_exported_member_1, getFullyQualifiedName(namespace), ts.declarationNameToString(right));
+                    // If we have an outer exception target (i.e. a containing try-finally or try-catch-finally), add a
+                    // control flow that goes back through the finally blok and back through each possible exception source.
+                    if (currentExceptionTarget && exceptionLabel.antecedents) {
+                        addAntecedent(currentExceptionTarget, createReduceLabel(finallyLabel, exceptionLabel.antecedents, currentFlow));
                     }
-                    return undefined;
+                    // If the end of the finally block is reachable, but the end of the try and catch blocks are not,
+                    // convert the current flow to unreachable. For example, 'try { return 1; } finally { ... }' should
+                    // result in an unreachable current control flow.
+                    currentFlow = normalExitLabel.antecedents ? createReduceLabel(finallyLabel, normalExitLabel.antecedents, currentFlow) : unreachableFlow;
                 }
             }
             else {
-                throw ts.Debug.assertNever(name, "Unknown entity name kind.");
-            }
-            ts.Debug.assert((ts.getCheckFlags(symbol) & 1 /* Instantiated */) === 0, "Should never get an instantiated symbol here.");
-            if (!ts.nodeIsSynthesized(name) && ts.isEntityName(name) && (symbol.flags & 2097152 /* Alias */ || name.parent.kind === 259 /* ExportAssignment */)) {
-                markSymbolOfAliasDeclarationIfTypeOnly(ts.getAliasDeclarationFromName(name), symbol, /*finalTarget*/ undefined, /*overwriteEmpty*/ true);
+                currentFlow = finishFlowLabel(normalExitLabel);
             }
-            return (symbol.flags & meaning) || dontResolveAlias ? symbol : resolveAlias(symbol);
         }
-        /**
-         * 1. For prototype-property methods like `A.prototype.m = function () ...`, try to resolve names in the scope of `A` too.
-         * Note that prototype-property assignment to locations outside the current file (eg globals) doesn't work, so
-         * name resolution won't work either.
-         * 2. For property assignments like `{ x: function f () { } }`, try to resolve names in the scope of `f` too.
-         */
-        function resolveEntityNameFromAssignmentDeclaration(name, meaning) {
-            if (isJSDocTypeReference(name.parent)) {
-                var secondaryLocation = getAssignmentDeclarationLocation(name.parent);
-                if (secondaryLocation) {
-                    return resolveName(secondaryLocation, name.escapedText, meaning, /*nameNotFoundMessage*/ undefined, name, /*isUse*/ true);
-                }
+        function bindSwitchStatement(node) {
+            var postSwitchLabel = createBranchLabel();
+            bind(node.expression);
+            var saveBreakTarget = currentBreakTarget;
+            var savePreSwitchCaseFlow = preSwitchCaseFlow;
+            currentBreakTarget = postSwitchLabel;
+            preSwitchCaseFlow = currentFlow;
+            bind(node.caseBlock);
+            addAntecedent(postSwitchLabel, currentFlow);
+            var hasDefault = ts.forEach(node.caseBlock.clauses, function (c) { return c.kind === 285 /* DefaultClause */; });
+            // We mark a switch statement as possibly exhaustive if it has no default clause and if all
+            // case clauses have unreachable end points (e.g. they all return). Note, we no longer need
+            // this property in control flow analysis, it's there only for backwards compatibility.
+            node.possiblyExhaustive = !hasDefault && !postSwitchLabel.antecedents;
+            if (!hasDefault) {
+                addAntecedent(postSwitchLabel, createFlowSwitchClause(preSwitchCaseFlow, node, 0, 0));
             }
+            currentBreakTarget = saveBreakTarget;
+            preSwitchCaseFlow = savePreSwitchCaseFlow;
+            currentFlow = finishFlowLabel(postSwitchLabel);
         }
-        function getAssignmentDeclarationLocation(node) {
-            var typeAlias = ts.findAncestor(node, function (node) { return !(ts.isJSDocNode(node) || node.flags & 4194304 /* JSDoc */) ? "quit" : ts.isJSDocTypeAlias(node); });
-            if (typeAlias) {
-                return;
-            }
-            var host = ts.getJSDocHost(node);
-            if (ts.isExpressionStatement(host) &&
-                ts.isBinaryExpression(host.expression) &&
-                ts.getAssignmentDeclarationKind(host.expression) === 3 /* PrototypeProperty */) {
-                // X.prototype.m = /** @param {K} p */ function () { } <-- look for K on X's declaration
-                var symbol = getSymbolOfNode(host.expression.left);
-                if (symbol) {
-                    return getDeclarationOfJSPrototypeContainer(symbol);
+        function bindCaseBlock(node) {
+            var clauses = node.clauses;
+            var isNarrowingSwitch = isNarrowingExpression(node.parent.expression);
+            var fallthroughFlow = unreachableFlow;
+            for (var i = 0; i < clauses.length; i++) {
+                var clauseStart = i;
+                while (!clauses[i].statements.length && i + 1 < clauses.length) {
+                    bind(clauses[i]);
+                    i++;
                 }
-            }
-            if ((ts.isObjectLiteralMethod(host) || ts.isPropertyAssignment(host)) &&
-                ts.isBinaryExpression(host.parent.parent) &&
-                ts.getAssignmentDeclarationKind(host.parent.parent) === 6 /* Prototype */) {
-                // X.prototype = { /** @param {K} p */m() { } } <-- look for K on X's declaration
-                var symbol = getSymbolOfNode(host.parent.parent.left);
-                if (symbol) {
-                    return getDeclarationOfJSPrototypeContainer(symbol);
+                var preCaseLabel = createBranchLabel();
+                addAntecedent(preCaseLabel, isNarrowingSwitch ? createFlowSwitchClause(preSwitchCaseFlow, node.parent, clauseStart, i + 1) : preSwitchCaseFlow);
+                addAntecedent(preCaseLabel, fallthroughFlow);
+                currentFlow = finishFlowLabel(preCaseLabel);
+                var clause = clauses[i];
+                bind(clause);
+                fallthroughFlow = currentFlow;
+                if (!(currentFlow.flags & 1 /* Unreachable */) && i !== clauses.length - 1 && options.noFallthroughCasesInSwitch) {
+                    clause.fallthroughFlowNode = currentFlow;
                 }
             }
-            var sig = ts.getEffectiveJSDocHost(node);
-            if (sig && ts.isFunctionLike(sig)) {
-                var symbol = getSymbolOfNode(sig);
-                return symbol && symbol.valueDeclaration;
-            }
         }
-        function getDeclarationOfJSPrototypeContainer(symbol) {
-            var decl = symbol.parent.valueDeclaration;
-            if (!decl) {
-                return undefined;
-            }
-            var initializer = ts.isAssignmentDeclaration(decl) ? ts.getAssignedExpandoInitializer(decl) :
-                ts.hasOnlyExpressionInitializer(decl) ? ts.getDeclaredExpandoInitializer(decl) :
-                    undefined;
-            return initializer || decl;
+        function bindCaseClause(node) {
+            var saveCurrentFlow = currentFlow;
+            currentFlow = preSwitchCaseFlow;
+            bind(node.expression);
+            currentFlow = saveCurrentFlow;
+            bindEach(node.statements);
         }
-        /**
-         * Get the real symbol of a declaration with an expando initializer.
-         *
-         * Normally, declarations have an associated symbol, but when a declaration has an expando
-         * initializer, the expando's symbol is the one that has all the members merged into it.
-         */
-        function getExpandoSymbol(symbol) {
-            var decl = symbol.valueDeclaration;
-            if (!decl || !ts.isInJSFile(decl) || symbol.flags & 524288 /* TypeAlias */ || ts.getExpandoInitializer(decl, /*isPrototypeAssignment*/ false)) {
-                return undefined;
-            }
-            var init = ts.isVariableDeclaration(decl) ? ts.getDeclaredExpandoInitializer(decl) : ts.getAssignedExpandoInitializer(decl);
-            if (init) {
-                var initSymbol = getSymbolOfNode(init);
-                if (initSymbol) {
-                    return mergeJSSymbols(initSymbol, symbol);
+        function bindExpressionStatement(node) {
+            bind(node.expression);
+            maybeBindExpressionFlowIfCall(node.expression);
+        }
+        function maybeBindExpressionFlowIfCall(node) {
+            // A top level or LHS of comma expression call expression with a dotted function name and at least one argument
+            // is potentially an assertion and is therefore included in the control flow.
+            if (node.kind === 203 /* CallExpression */) {
+                var call = node;
+                if (ts.isDottedName(call.expression) && call.expression.kind !== 105 /* SuperKeyword */) {
+                    currentFlow = createFlowCall(currentFlow, call);
                 }
             }
         }
-        function resolveExternalModuleName(location, moduleReferenceExpression, ignoreErrors) {
-            return resolveExternalModuleNameWorker(location, moduleReferenceExpression, ignoreErrors ? undefined : ts.Diagnostics.Cannot_find_module_0_or_its_corresponding_type_declarations);
-        }
-        function resolveExternalModuleNameWorker(location, moduleReferenceExpression, moduleNotFoundError, isForAugmentation) {
-            if (isForAugmentation === void 0) { isForAugmentation = false; }
-            return ts.isStringLiteralLike(moduleReferenceExpression)
-                ? resolveExternalModule(location, moduleReferenceExpression.text, moduleNotFoundError, moduleReferenceExpression, isForAugmentation)
-                : undefined;
+        function bindLabeledStatement(node) {
+            var postStatementLabel = createBranchLabel();
+            activeLabelList = {
+                next: activeLabelList,
+                name: node.label.escapedText,
+                breakTarget: postStatementLabel,
+                continueTarget: undefined,
+                referenced: false
+            };
+            bind(node.label);
+            bind(node.statement);
+            if (!activeLabelList.referenced && !options.allowUnusedLabels) {
+                errorOrSuggestionOnNode(ts.unusedLabelIsError(options), node.label, ts.Diagnostics.Unused_label);
+            }
+            activeLabelList = activeLabelList.next;
+            addAntecedent(postStatementLabel, currentFlow);
+            currentFlow = finishFlowLabel(postStatementLabel);
         }
-        function resolveExternalModule(location, moduleReference, moduleNotFoundError, errorNode, isForAugmentation) {
-            if (isForAugmentation === void 0) { isForAugmentation = false; }
-            if (ts.startsWith(moduleReference, "@types/")) {
-                var diag = ts.Diagnostics.Cannot_import_type_declaration_files_Consider_importing_0_instead_of_1;
-                var withoutAtTypePrefix = ts.removePrefix(moduleReference, "@types/");
-                error(errorNode, diag, withoutAtTypePrefix, moduleReference);
+        function bindDestructuringTargetFlow(node) {
+            if (node.kind === 216 /* BinaryExpression */ && node.operatorToken.kind === 62 /* EqualsToken */) {
+                bindAssignmentTargetFlow(node.left);
             }
-            var ambientModule = tryFindAmbientModule(moduleReference, /*withAugmentations*/ true);
-            if (ambientModule) {
-                return ambientModule;
+            else {
+                bindAssignmentTargetFlow(node);
             }
-            var currentSourceFile = ts.getSourceFileOfNode(location);
-            var resolvedModule = ts.getResolvedModule(currentSourceFile, moduleReference); // TODO: GH#18217
-            var resolutionDiagnostic = resolvedModule && ts.getResolutionDiagnostic(compilerOptions, resolvedModule);
-            var sourceFile = resolvedModule && !resolutionDiagnostic && host.getSourceFile(resolvedModule.resolvedFileName);
-            if (sourceFile) {
-                if (sourceFile.symbol) {
-                    if (resolvedModule.isExternalLibraryImport && !ts.resolutionExtensionIsTSOrJson(resolvedModule.extension)) {
-                        errorOnImplicitAnyModule(/*isError*/ false, errorNode, resolvedModule, moduleReference);
-                    }
-                    // merged symbol is module declaration symbol combined with all augmentations
-                    return getMergedSymbol(sourceFile.symbol);
-                }
-                if (moduleNotFoundError) {
-                    // report errors only if it was requested
-                    error(errorNode, ts.Diagnostics.File_0_is_not_a_module, sourceFile.fileName);
-                }
-                return undefined;
+        }
+        function bindAssignmentTargetFlow(node) {
+            if (isNarrowableReference(node)) {
+                currentFlow = createFlowMutation(16 /* Assignment */, currentFlow, node);
             }
-            if (patternAmbientModules) {
-                var pattern = ts.findBestPatternMatch(patternAmbientModules, function (_) { return _.pattern; }, moduleReference);
-                if (pattern) {
-                    // If the module reference matched a pattern ambient module ('*.foo') but there's also a
-                    // module augmentation by the specific name requested ('a.foo'), we store the merged symbol
-                    // by the augmentation name ('a.foo'), because asking for *.foo should not give you exports
-                    // from a.foo.
-                    var augmentation = patternAmbientModuleAugmentations && patternAmbientModuleAugmentations.get(moduleReference);
-                    if (augmentation) {
-                        return getMergedSymbol(augmentation);
+            else if (node.kind === 199 /* ArrayLiteralExpression */) {
+                for (var _i = 0, _a = node.elements; _i < _a.length; _i++) {
+                    var e = _a[_i];
+                    if (e.kind === 220 /* SpreadElement */) {
+                        bindAssignmentTargetFlow(e.expression);
                     }
-                    return getMergedSymbol(pattern.symbol);
-                }
-            }
-            // May be an untyped module. If so, ignore resolutionDiagnostic.
-            if (resolvedModule && !ts.resolutionExtensionIsTSOrJson(resolvedModule.extension) && resolutionDiagnostic === undefined || resolutionDiagnostic === ts.Diagnostics.Could_not_find_a_declaration_file_for_module_0_1_implicitly_has_an_any_type) {
-                if (isForAugmentation) {
-                    var diag = ts.Diagnostics.Invalid_module_name_in_augmentation_Module_0_resolves_to_an_untyped_module_at_1_which_cannot_be_augmented;
-                    error(errorNode, diag, moduleReference, resolvedModule.resolvedFileName);
-                }
-                else {
-                    errorOnImplicitAnyModule(/*isError*/ noImplicitAny && !!moduleNotFoundError, errorNode, resolvedModule, moduleReference);
-                }
-                // Failed imports and untyped modules are both treated in an untyped manner; only difference is whether we give a diagnostic first.
-                return undefined;
-            }
-            if (moduleNotFoundError) {
-                // See if this was possibly a projectReference redirect
-                if (resolvedModule) {
-                    var redirect = host.getProjectReferenceRedirect(resolvedModule.resolvedFileName);
-                    if (redirect) {
-                        error(errorNode, ts.Diagnostics.Output_file_0_has_not_been_built_from_source_file_1, redirect, resolvedModule.resolvedFileName);
-                        return undefined;
+                    else {
+                        bindDestructuringTargetFlow(e);
                     }
                 }
-                if (resolutionDiagnostic) {
-                    error(errorNode, resolutionDiagnostic, moduleReference, resolvedModule.resolvedFileName);
-                }
-                else {
-                    var tsExtension = ts.tryExtractTSExtension(moduleReference);
-                    if (tsExtension) {
-                        var diag = ts.Diagnostics.An_import_path_cannot_end_with_a_0_extension_Consider_importing_1_instead;
-                        error(errorNode, diag, tsExtension, ts.removeExtension(moduleReference, tsExtension));
+            }
+            else if (node.kind === 200 /* ObjectLiteralExpression */) {
+                for (var _b = 0, _c = node.properties; _b < _c.length; _b++) {
+                    var p = _c[_b];
+                    if (p.kind === 288 /* PropertyAssignment */) {
+                        bindDestructuringTargetFlow(p.initializer);
                     }
-                    else if (!compilerOptions.resolveJsonModule &&
-                        ts.fileExtensionIs(moduleReference, ".json" /* Json */) &&
-                        ts.getEmitModuleResolutionKind(compilerOptions) === ts.ModuleResolutionKind.NodeJs &&
-                        ts.hasJsonModuleEmitEnabled(compilerOptions)) {
-                        error(errorNode, ts.Diagnostics.Cannot_find_module_0_Consider_using_resolveJsonModule_to_import_module_with_json_extension, moduleReference);
+                    else if (p.kind === 289 /* ShorthandPropertyAssignment */) {
+                        bindAssignmentTargetFlow(p.name);
                     }
-                    else {
-                        error(errorNode, moduleNotFoundError, moduleReference);
+                    else if (p.kind === 290 /* SpreadAssignment */) {
+                        bindAssignmentTargetFlow(p.expression);
                     }
                 }
             }
-            return undefined;
-        }
-        function errorOnImplicitAnyModule(isError, errorNode, _a, moduleReference) {
-            var packageId = _a.packageId, resolvedFileName = _a.resolvedFileName;
-            var errorInfo = !ts.isExternalModuleNameRelative(moduleReference) && packageId
-                ? typesPackageExists(packageId.name)
-                    ? ts.chainDiagnosticMessages(
-                    /*details*/ undefined, ts.Diagnostics.If_the_0_package_actually_exposes_this_module_consider_sending_a_pull_request_to_amend_https_Colon_Slash_Slashgithub_com_SlashDefinitelyTyped_SlashDefinitelyTyped_Slashtree_Slashmaster_Slashtypes_Slash_1, packageId.name, ts.mangleScopedPackageName(packageId.name))
-                    : ts.chainDiagnosticMessages(
-                    /*details*/ undefined, ts.Diagnostics.Try_npm_install_types_Slash_1_if_it_exists_or_add_a_new_declaration_d_ts_file_containing_declare_module_0, moduleReference, ts.mangleScopedPackageName(packageId.name))
-                : undefined;
-            errorOrSuggestion(isError, errorNode, ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Could_not_find_a_declaration_file_for_module_0_1_implicitly_has_an_any_type, moduleReference, resolvedFileName));
-        }
-        function typesPackageExists(packageName) {
-            return getPackagesSet().has(ts.getTypesPackageName(packageName));
         }
-        function resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias) {
-            if (moduleSymbol === null || moduleSymbol === void 0 ? void 0 : moduleSymbol.exports) {
-                var exportEquals = resolveSymbol(moduleSymbol.exports.get("export=" /* ExportEquals */), dontResolveAlias);
-                var exported = getCommonJsExportEquals(getMergedSymbol(exportEquals), getMergedSymbol(moduleSymbol));
-                return getMergedSymbol(exported) || moduleSymbol;
+        function bindLogicalLikeExpression(node, trueTarget, falseTarget) {
+            var preRightLabel = createBranchLabel();
+            if (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */ || node.operatorToken.kind === 75 /* AmpersandAmpersandEqualsToken */) {
+                bindCondition(node.left, preRightLabel, falseTarget);
             }
-            return undefined;
-        }
-        function getCommonJsExportEquals(exported, moduleSymbol) {
-            if (!exported || exported === unknownSymbol || exported === moduleSymbol || moduleSymbol.exports.size === 1 || exported.flags & 2097152 /* Alias */) {
-                return exported;
+            else {
+                bindCondition(node.left, trueTarget, preRightLabel);
             }
-            var links = getSymbolLinks(exported);
-            if (links.cjsExportMerged) {
-                return links.cjsExportMerged;
+            currentFlow = finishFlowLabel(preRightLabel);
+            bind(node.operatorToken);
+            if (ts.isLogicalOrCoalescingAssignmentOperator(node.operatorToken.kind)) {
+                doWithConditionalBranches(bind, node.right, trueTarget, falseTarget);
+                bindAssignmentTargetFlow(node.left);
+                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
+                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
             }
-            var merged = exported.flags & 33554432 /* Transient */ ? exported : cloneSymbol(exported);
-            merged.flags = merged.flags | 512 /* ValueModule */;
-            if (merged.exports === undefined) {
-                merged.exports = ts.createSymbolTable();
+            else {
+                bindCondition(node.right, trueTarget, falseTarget);
             }
-            moduleSymbol.exports.forEach(function (s, name) {
-                if (name === "export=" /* ExportEquals */)
-                    return;
-                merged.exports.set(name, merged.exports.has(name) ? mergeSymbol(merged.exports.get(name), s) : s);
-            });
-            getSymbolLinks(merged).cjsExportMerged = merged;
-            return links.cjsExportMerged = merged;
         }
-        // An external module with an 'export =' declaration may be referenced as an ES6 module provided the 'export ='
-        // references a symbol that is at least declared as a module or a variable. The target of the 'export =' may
-        // combine other declarations with the module or variable (e.g. a class/module, function/module, interface/variable).
-        function resolveESModuleSymbol(moduleSymbol, referencingLocation, dontResolveAlias, suppressInteropError) {
-            var symbol = resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias);
-            if (!dontResolveAlias && symbol) {
-                if (!suppressInteropError && !(symbol.flags & (1536 /* Module */ | 3 /* Variable */)) && !ts.getDeclarationOfKind(symbol, 290 /* SourceFile */)) {
-                    var compilerOptionName = moduleKind >= ts.ModuleKind.ES2015
-                        ? "allowSyntheticDefaultImports"
-                        : "esModuleInterop";
-                    error(referencingLocation, ts.Diagnostics.This_module_can_only_be_referenced_with_ECMAScript_imports_Slashexports_by_turning_on_the_0_flag_and_referencing_its_default_export, compilerOptionName);
-                    return symbol;
-                }
-                if (compilerOptions.esModuleInterop) {
-                    var referenceParent = referencingLocation.parent;
-                    if ((ts.isImportDeclaration(referenceParent) && ts.getNamespaceDeclarationNode(referenceParent)) ||
-                        ts.isImportCall(referenceParent)) {
-                        var type = getTypeOfSymbol(symbol);
-                        var sigs = getSignaturesOfStructuredType(type, 0 /* Call */);
-                        if (!sigs || !sigs.length) {
-                            sigs = getSignaturesOfStructuredType(type, 1 /* Construct */);
-                        }
-                        if (sigs && sigs.length) {
-                            var moduleType = getTypeWithSyntheticDefaultImportType(type, symbol, moduleSymbol);
-                            // Create a new symbol which has the module's type less the call and construct signatures
-                            var result = createSymbol(symbol.flags, symbol.escapedName);
-                            result.declarations = symbol.declarations ? symbol.declarations.slice() : [];
-                            result.parent = symbol.parent;
-                            result.target = symbol;
-                            result.originatingImport = referenceParent;
-                            if (symbol.valueDeclaration)
-                                result.valueDeclaration = symbol.valueDeclaration;
-                            if (symbol.constEnumOnlyModule)
-                                result.constEnumOnlyModule = true;
-                            if (symbol.members)
-                                result.members = ts.cloneMap(symbol.members);
-                            if (symbol.exports)
-                                result.exports = ts.cloneMap(symbol.exports);
-                            var resolvedModuleType = resolveStructuredTypeMembers(moduleType); // Should already be resolved from the signature checks above
-                            result.type = createAnonymousType(result, resolvedModuleType.members, ts.emptyArray, ts.emptyArray, resolvedModuleType.stringIndexInfo, resolvedModuleType.numberIndexInfo);
-                            return result;
-                        }
-                    }
-                }
+        function bindPrefixUnaryExpressionFlow(node) {
+            if (node.operator === 53 /* ExclamationToken */) {
+                var saveTrueTarget = currentTrueTarget;
+                currentTrueTarget = currentFalseTarget;
+                currentFalseTarget = saveTrueTarget;
+                bindEachChild(node);
+                currentFalseTarget = currentTrueTarget;
+                currentTrueTarget = saveTrueTarget;
             }
-            return symbol;
-        }
-        function hasExportAssignmentSymbol(moduleSymbol) {
-            return moduleSymbol.exports.get("export=" /* ExportEquals */) !== undefined;
-        }
-        function getExportsOfModuleAsArray(moduleSymbol) {
-            return symbolsToArray(getExportsOfModule(moduleSymbol));
-        }
-        function getExportsAndPropertiesOfModule(moduleSymbol) {
-            var exports = getExportsOfModuleAsArray(moduleSymbol);
-            var exportEquals = resolveExternalModuleSymbol(moduleSymbol);
-            if (exportEquals !== moduleSymbol) {
-                ts.addRange(exports, getPropertiesOfType(getTypeOfSymbol(exportEquals)));
+            else {
+                bindEachChild(node);
+                if (node.operator === 45 /* PlusPlusToken */ || node.operator === 46 /* MinusMinusToken */) {
+                    bindAssignmentTargetFlow(node.operand);
+                }
             }
-            return exports;
         }
-        function tryGetMemberInModuleExports(memberName, moduleSymbol) {
-            var symbolTable = getExportsOfModule(moduleSymbol);
-            if (symbolTable) {
-                return symbolTable.get(memberName);
+        function bindPostfixUnaryExpressionFlow(node) {
+            bindEachChild(node);
+            if (node.operator === 45 /* PlusPlusToken */ || node.operator === 46 /* MinusMinusToken */) {
+                bindAssignmentTargetFlow(node.operand);
             }
         }
-        function tryGetMemberInModuleExportsAndProperties(memberName, moduleSymbol) {
-            var symbol = tryGetMemberInModuleExports(memberName, moduleSymbol);
-            if (symbol) {
-                return symbol;
+        function bindDestructuringAssignmentFlow(node) {
+            if (inAssignmentPattern) {
+                inAssignmentPattern = false;
+                bind(node.operatorToken);
+                bind(node.right);
+                inAssignmentPattern = true;
+                bind(node.left);
             }
-            var exportEquals = resolveExternalModuleSymbol(moduleSymbol);
-            if (exportEquals === moduleSymbol) {
-                return undefined;
+            else {
+                inAssignmentPattern = true;
+                bind(node.left);
+                inAssignmentPattern = false;
+                bind(node.operatorToken);
+                bind(node.right);
             }
-            var type = getTypeOfSymbol(exportEquals);
-            return type.flags & 131068 /* Primitive */ ||
-                ts.getObjectFlags(type) & 1 /* Class */ ||
-                isArrayOrTupleLikeType(type)
-                ? undefined
-                : getPropertyOfType(type, memberName);
-        }
-        function getExportsOfSymbol(symbol) {
-            return symbol.flags & 6256 /* LateBindingContainer */ ? getResolvedMembersOrExportsOfSymbol(symbol, "resolvedExports" /* resolvedExports */) :
-                symbol.flags & 1536 /* Module */ ? getExportsOfModule(symbol) :
-                    symbol.exports || emptySymbols;
-        }
-        function getExportsOfModule(moduleSymbol) {
-            var links = getSymbolLinks(moduleSymbol);
-            return links.resolvedExports || (links.resolvedExports = getExportsOfModuleWorker(moduleSymbol));
+            bindAssignmentTargetFlow(node.left);
         }
-        /**
-         * Extends one symbol table with another while collecting information on name collisions for error message generation into the `lookupTable` argument
-         * Not passing `lookupTable` and `exportNode` disables this collection, and just extends the tables
-         */
-        function extendExportSymbols(target, source, lookupTable, exportNode) {
-            if (!source)
-                return;
-            source.forEach(function (sourceSymbol, id) {
-                if (id === "default" /* Default */)
-                    return;
-                var targetSymbol = target.get(id);
-                if (!targetSymbol) {
-                    target.set(id, sourceSymbol);
-                    if (lookupTable && exportNode) {
-                        lookupTable.set(id, {
-                            specifierText: ts.getTextOfNode(exportNode.moduleSpecifier)
-                        });
+        var BindBinaryExpressionFlowState;
+        (function (BindBinaryExpressionFlowState) {
+            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["BindThenBindChildren"] = 0] = "BindThenBindChildren";
+            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["MaybeBindLeft"] = 1] = "MaybeBindLeft";
+            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["BindToken"] = 2] = "BindToken";
+            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["BindRight"] = 3] = "BindRight";
+            BindBinaryExpressionFlowState[BindBinaryExpressionFlowState["FinishBind"] = 4] = "FinishBind";
+        })(BindBinaryExpressionFlowState || (BindBinaryExpressionFlowState = {}));
+        function bindBinaryExpressionFlow(node) {
+            var workStacks = {
+                expr: [node],
+                state: [1 /* MaybeBindLeft */],
+                inStrictMode: [undefined],
+                parent: [undefined],
+            };
+            var stackIndex = 0;
+            while (stackIndex >= 0) {
+                node = workStacks.expr[stackIndex];
+                switch (workStacks.state[stackIndex]) {
+                    case 0 /* BindThenBindChildren */: {
+                        // This state is used only when recuring, to emulate the work that `bind` does before
+                        // reaching `bindChildren`. A normal call to `bindBinaryExpressionFlow` will already have done this work.
+                        ts.setParent(node, parent);
+                        var saveInStrictMode = inStrictMode;
+                        bindWorker(node);
+                        var saveParent = parent;
+                        parent = node;
+                        advanceState(1 /* MaybeBindLeft */, saveInStrictMode, saveParent);
+                        break;
                     }
-                }
-                else if (lookupTable && exportNode && targetSymbol && resolveSymbol(targetSymbol) !== resolveSymbol(sourceSymbol)) {
-                    var collisionTracker = lookupTable.get(id);
-                    if (!collisionTracker.exportsWithDuplicate) {
-                        collisionTracker.exportsWithDuplicate = [exportNode];
+                    case 1 /* MaybeBindLeft */: {
+                        var operator = node.operatorToken.kind;
+                        // TODO: bindLogicalExpression is recursive - if we want to handle deeply nested `&&` expressions
+                        // we'll need to handle the `bindLogicalExpression` scenarios in this state machine, too
+                        // For now, though, since the common cases are chained `+`, leaving it recursive is fine
+                        if (operator === 55 /* AmpersandAmpersandToken */ || operator === 56 /* BarBarToken */ || operator === 60 /* QuestionQuestionToken */ ||
+                            ts.isLogicalOrCoalescingAssignmentOperator(operator)) {
+                            if (isTopLevelLogicalExpression(node)) {
+                                var postExpressionLabel = createBranchLabel();
+                                bindLogicalLikeExpression(node, postExpressionLabel, postExpressionLabel);
+                                currentFlow = finishFlowLabel(postExpressionLabel);
+                            }
+                            else {
+                                bindLogicalLikeExpression(node, currentTrueTarget, currentFalseTarget);
+                            }
+                            completeNode();
+                        }
+                        else {
+                            advanceState(2 /* BindToken */);
+                            maybeBind(node.left);
+                        }
+                        break;
                     }
-                    else {
-                        collisionTracker.exportsWithDuplicate.push(exportNode);
+                    case 2 /* BindToken */: {
+                        if (node.operatorToken.kind === 27 /* CommaToken */) {
+                            maybeBindExpressionFlowIfCall(node.left);
+                        }
+                        advanceState(3 /* BindRight */);
+                        maybeBind(node.operatorToken);
+                        break;
                     }
-                }
-            });
-        }
-        function getExportsOfModuleWorker(moduleSymbol) {
-            var visitedSymbols = [];
-            // A module defined by an 'export=' consists of one export that needs to be resolved
-            moduleSymbol = resolveExternalModuleSymbol(moduleSymbol);
-            return visit(moduleSymbol) || emptySymbols;
-            // The ES6 spec permits export * declarations in a module to circularly reference the module itself. For example,
-            // module 'a' can 'export * from "b"' and 'b' can 'export * from "a"' without error.
-            function visit(symbol) {
-                if (!(symbol && symbol.exports && ts.pushIfUnique(visitedSymbols, symbol))) {
-                    return;
-                }
-                var symbols = ts.cloneMap(symbol.exports);
-                // All export * declarations are collected in an __export symbol by the binder
-                var exportStars = symbol.exports.get("__export" /* ExportStar */);
-                if (exportStars) {
-                    var nestedSymbols = ts.createSymbolTable();
-                    var lookupTable_1 = ts.createMap();
-                    for (var _i = 0, _a = exportStars.declarations; _i < _a.length; _i++) {
-                        var node = _a[_i];
-                        var resolvedModule = resolveExternalModuleName(node, node.moduleSpecifier);
-                        var exportedSymbols = visit(resolvedModule);
-                        extendExportSymbols(nestedSymbols, exportedSymbols, lookupTable_1, node);
+                    case 3 /* BindRight */: {
+                        advanceState(4 /* FinishBind */);
+                        maybeBind(node.right);
+                        break;
                     }
-                    lookupTable_1.forEach(function (_a, id) {
-                        var exportsWithDuplicate = _a.exportsWithDuplicate;
-                        // It's not an error if the file with multiple `export *`s with duplicate names exports a member with that name itself
-                        if (id === "export=" || !(exportsWithDuplicate && exportsWithDuplicate.length) || symbols.has(id)) {
-                            return;
-                        }
-                        for (var _i = 0, exportsWithDuplicate_1 = exportsWithDuplicate; _i < exportsWithDuplicate_1.length; _i++) {
-                            var node = exportsWithDuplicate_1[_i];
-                            diagnostics.add(ts.createDiagnosticForNode(node, ts.Diagnostics.Module_0_has_already_exported_a_member_named_1_Consider_explicitly_re_exporting_to_resolve_the_ambiguity, lookupTable_1.get(id).specifierText, ts.unescapeLeadingUnderscores(id)));
+                    case 4 /* FinishBind */: {
+                        var operator = node.operatorToken.kind;
+                        if (ts.isAssignmentOperator(operator) && !ts.isAssignmentTarget(node)) {
+                            bindAssignmentTargetFlow(node.left);
+                            if (operator === 62 /* EqualsToken */ && node.left.kind === 202 /* ElementAccessExpression */) {
+                                var elementAccess = node.left;
+                                if (isNarrowableOperand(elementAccess.expression)) {
+                                    currentFlow = createFlowMutation(256 /* ArrayMutation */, currentFlow, node);
+                                }
+                            }
                         }
-                    });
-                    extendExportSymbols(symbols, nestedSymbols);
+                        completeNode();
+                        break;
+                    }
+                    default: return ts.Debug.fail("Invalid state " + workStacks.state[stackIndex] + " for bindBinaryExpressionFlow");
                 }
-                return symbols;
-            }
-        }
-        function getMergedSymbol(symbol) {
-            var merged;
-            return symbol && symbol.mergeId && (merged = mergedSymbols[symbol.mergeId]) ? merged : symbol;
-        }
-        function getSymbolOfNode(node) {
-            return getMergedSymbol(node.symbol && getLateBoundSymbol(node.symbol));
-        }
-        function getParentOfSymbol(symbol) {
-            return getMergedSymbol(symbol.parent && getLateBoundSymbol(symbol.parent));
-        }
-        function getAlternativeContainingModules(symbol, enclosingDeclaration) {
-            var containingFile = ts.getSourceFileOfNode(enclosingDeclaration);
-            var id = "" + getNodeId(containingFile);
-            var links = getSymbolLinks(symbol);
-            var results;
-            if (links.extendedContainersByFile && (results = links.extendedContainersByFile.get(id))) {
-                return results;
             }
-            if (containingFile && containingFile.imports) {
-                // Try to make an import using an import already in the enclosing file, if possible
-                for (var _i = 0, _a = containingFile.imports; _i < _a.length; _i++) {
-                    var importRef = _a[_i];
-                    if (ts.nodeIsSynthesized(importRef))
-                        continue; // Synthetic names can't be resolved by `resolveExternalModuleName` - they'll cause a debug assert if they error
-                    var resolvedModule = resolveExternalModuleName(enclosingDeclaration, importRef, /*ignoreErrors*/ true);
-                    if (!resolvedModule)
-                        continue;
-                    var ref = getAliasForSymbolInContainer(resolvedModule, symbol);
-                    if (!ref)
-                        continue;
-                    results = ts.append(results, resolvedModule);
+            /**
+             * Note that `advanceState` sets the _current_ head state, and that `maybeBind` potentially pushes on a new
+             * head state; so `advanceState` must be called before any `maybeBind` during a state's execution.
+             */
+            function advanceState(state, isInStrictMode, parent) {
+                workStacks.state[stackIndex] = state;
+                if (isInStrictMode !== undefined) {
+                    workStacks.inStrictMode[stackIndex] = isInStrictMode;
                 }
-                if (ts.length(results)) {
-                    (links.extendedContainersByFile || (links.extendedContainersByFile = ts.createMap())).set(id, results);
-                    return results;
+                if (parent !== undefined) {
+                    workStacks.parent[stackIndex] = parent;
                 }
             }
-            if (links.extendedContainers) {
-                return links.extendedContainers;
-            }
-            // No results from files already being imported by this file - expand search (expensive, but not location-specific, so cached)
-            var otherFiles = host.getSourceFiles();
-            for (var _b = 0, otherFiles_1 = otherFiles; _b < otherFiles_1.length; _b++) {
-                var file = otherFiles_1[_b];
-                if (!ts.isExternalModule(file))
-                    continue;
-                var sym = getSymbolOfNode(file);
-                var ref = getAliasForSymbolInContainer(sym, symbol);
-                if (!ref)
-                    continue;
-                results = ts.append(results, sym);
-            }
-            return links.extendedContainers = results || ts.emptyArray;
-        }
-        /**
-         * Attempts to find the symbol corresponding to the container a symbol is in - usually this
-         * is just its' `.parent`, but for locals, this value is `undefined`
-         */
-        function getContainersOfSymbol(symbol, enclosingDeclaration) {
-            var container = getParentOfSymbol(symbol);
-            // Type parameters end up in the `members` lists but are not externally visible
-            if (container && !(symbol.flags & 262144 /* TypeParameter */)) {
-                var additionalContainers = ts.mapDefined(container.declarations, fileSymbolIfFileSymbolExportEqualsContainer);
-                var reexportContainers = enclosingDeclaration && getAlternativeContainingModules(symbol, enclosingDeclaration);
-                if (enclosingDeclaration && getAccessibleSymbolChain(container, enclosingDeclaration, 1920 /* Namespace */, /*externalOnly*/ false)) {
-                    return ts.concatenate(ts.concatenate([container], additionalContainers), reexportContainers); // This order expresses a preference for the real container if it is in scope
+            function completeNode() {
+                if (workStacks.inStrictMode[stackIndex] !== undefined) {
+                    inStrictMode = workStacks.inStrictMode[stackIndex];
+                    parent = workStacks.parent[stackIndex];
                 }
-                var res = ts.append(additionalContainers, container);
-                return ts.concatenate(res, reexportContainers);
+                stackIndex--;
             }
-            var candidates = ts.mapDefined(symbol.declarations, function (d) {
-                if (!ts.isAmbientModule(d) && d.parent && hasNonGlobalAugmentationExternalModuleSymbol(d.parent)) {
-                    return getSymbolOfNode(d.parent);
+            /**
+             * If `node` is a BinaryExpression, adds it to the local work stack, otherwise recursively binds it
+             */
+            function maybeBind(node) {
+                if (node && ts.isBinaryExpression(node) && !ts.isDestructuringAssignment(node)) {
+                    stackIndex++;
+                    workStacks.expr[stackIndex] = node;
+                    workStacks.state[stackIndex] = 0 /* BindThenBindChildren */;
+                    workStacks.inStrictMode[stackIndex] = undefined;
+                    workStacks.parent[stackIndex] = undefined;
                 }
-                if (ts.isClassExpression(d) && ts.isBinaryExpression(d.parent) && d.parent.operatorToken.kind === 62 /* EqualsToken */ && ts.isAccessExpression(d.parent.left) && ts.isEntityNameExpression(d.parent.left.expression)) {
-                    if (ts.isModuleExportsAccessExpression(d.parent.left) || ts.isExportsIdentifier(d.parent.left.expression)) {
-                        return getSymbolOfNode(ts.getSourceFileOfNode(d));
-                    }
-                    checkExpressionCached(d.parent.left.expression);
-                    return getNodeLinks(d.parent.left.expression).resolvedSymbol;
+                else {
+                    bind(node);
                 }
-            });
-            if (!ts.length(candidates)) {
-                return undefined;
             }
-            return ts.mapDefined(candidates, function (candidate) { return getAliasForSymbolInContainer(candidate, symbol) ? candidate : undefined; });
-            function fileSymbolIfFileSymbolExportEqualsContainer(d) {
-                return container && getFileSymbolIfFileSymbolExportEqualsContainer(d, container);
+        }
+        function bindDeleteExpressionFlow(node) {
+            bindEachChild(node);
+            if (node.expression.kind === 201 /* PropertyAccessExpression */) {
+                bindAssignmentTargetFlow(node.expression);
             }
         }
-        function getFileSymbolIfFileSymbolExportEqualsContainer(d, container) {
-            var fileSymbol = getExternalModuleContainer(d);
-            var exported = fileSymbol && fileSymbol.exports && fileSymbol.exports.get("export=" /* ExportEquals */);
-            return exported && getSymbolIfSameReference(exported, container) ? fileSymbol : undefined;
+        function bindConditionalExpressionFlow(node) {
+            var trueLabel = createBranchLabel();
+            var falseLabel = createBranchLabel();
+            var postExpressionLabel = createBranchLabel();
+            bindCondition(node.condition, trueLabel, falseLabel);
+            currentFlow = finishFlowLabel(trueLabel);
+            bind(node.questionToken);
+            bind(node.whenTrue);
+            addAntecedent(postExpressionLabel, currentFlow);
+            currentFlow = finishFlowLabel(falseLabel);
+            bind(node.colonToken);
+            bind(node.whenFalse);
+            addAntecedent(postExpressionLabel, currentFlow);
+            currentFlow = finishFlowLabel(postExpressionLabel);
         }
-        function getAliasForSymbolInContainer(container, symbol) {
-            if (container === getParentOfSymbol(symbol)) {
-                // fast path, `symbol` is either already the alias or isn't aliased
-                return symbol;
-            }
-            // Check if container is a thing with an `export=` which points directly at `symbol`, and if so, return
-            // the container itself as the alias for the symbol
-            var exportEquals = container.exports && container.exports.get("export=" /* ExportEquals */);
-            if (exportEquals && getSymbolIfSameReference(exportEquals, symbol)) {
-                return container;
+        function bindInitializedVariableFlow(node) {
+            var name = !ts.isOmittedExpression(node) ? node.name : undefined;
+            if (ts.isBindingPattern(name)) {
+                for (var _i = 0, _a = name.elements; _i < _a.length; _i++) {
+                    var child = _a[_i];
+                    bindInitializedVariableFlow(child);
+                }
             }
-            var exports = getExportsOfSymbol(container);
-            var quick = exports.get(symbol.escapedName);
-            if (quick && getSymbolIfSameReference(quick, symbol)) {
-                return quick;
+            else {
+                currentFlow = createFlowMutation(16 /* Assignment */, currentFlow, node);
             }
-            return ts.forEachEntry(exports, function (exported) {
-                if (getSymbolIfSameReference(exported, symbol)) {
-                    return exported;
-                }
-            });
         }
-        /**
-         * Checks if two symbols, through aliasing and/or merging, refer to the same thing
-         */
-        function getSymbolIfSameReference(s1, s2) {
-            if (getMergedSymbol(resolveSymbol(getMergedSymbol(s1))) === getMergedSymbol(resolveSymbol(getMergedSymbol(s2)))) {
-                return s1;
+        function bindVariableDeclarationFlow(node) {
+            bindEachChild(node);
+            if (node.initializer || ts.isForInOrOfStatement(node.parent.parent)) {
+                bindInitializedVariableFlow(node);
             }
         }
-        function getExportSymbolOfValueSymbolIfExported(symbol) {
-            return getMergedSymbol(symbol && (symbol.flags & 1048576 /* ExportValue */) !== 0 ? symbol.exportSymbol : symbol);
-        }
-        function symbolIsValue(symbol) {
-            return !!(symbol.flags & 111551 /* Value */ || symbol.flags & 2097152 /* Alias */ && resolveAlias(symbol).flags & 111551 /* Value */ && !getTypeOnlyAliasDeclaration(symbol));
+        function bindBindingElementFlow(node) {
+            if (ts.isBindingPattern(node.name)) {
+                // When evaluating a binding pattern, the initializer is evaluated before the binding pattern, per:
+                // - https://tc39.es/ecma262/#sec-destructuring-binding-patterns-runtime-semantics-iteratorbindinginitialization
+                //   - `BindingElement: BindingPattern Initializer?`
+                // - https://tc39.es/ecma262/#sec-runtime-semantics-keyedbindinginitialization
+                //   - `BindingElement: BindingPattern Initializer?`
+                bindEach(node.decorators);
+                bindEach(node.modifiers);
+                bind(node.dotDotDotToken);
+                bind(node.propertyName);
+                bind(node.initializer);
+                bind(node.name);
+            }
+            else {
+                bindEachChild(node);
+            }
         }
-        function findConstructorDeclaration(node) {
-            var members = node.members;
-            for (var _i = 0, members_3 = members; _i < members_3.length; _i++) {
-                var member = members_3[_i];
-                if (member.kind === 162 /* Constructor */ && ts.nodeIsPresent(member.body)) {
-                    return member;
-                }
+        function bindJSDocTypeAlias(node) {
+            ts.setParent(node.tagName, node);
+            if (node.kind !== 325 /* JSDocEnumTag */ && node.fullName) {
+                ts.setParent(node.fullName, node);
+                ts.setParentRecursive(node.fullName, /*incremental*/ false);
             }
         }
-        function createType(flags) {
-            var result = new Type(checker, flags);
-            typeCount++;
-            result.id = typeCount;
-            return result;
+        function bindJSDocClassTag(node) {
+            bindEachChild(node);
+            var host = ts.getHostSignatureFromJSDoc(node);
+            if (host && host.kind !== 165 /* MethodDeclaration */) {
+                addDeclarationToSymbol(host.symbol, host, 32 /* Class */);
+            }
         }
-        function createIntrinsicType(kind, intrinsicName, objectFlags) {
-            if (objectFlags === void 0) { objectFlags = 0; }
-            var type = createType(kind);
-            type.intrinsicName = intrinsicName;
-            type.objectFlags = objectFlags;
-            return type;
+        function bindOptionalExpression(node, trueTarget, falseTarget) {
+            doWithConditionalBranches(bind, node, trueTarget, falseTarget);
+            if (!ts.isOptionalChain(node) || ts.isOutermostOptionalChain(node)) {
+                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
+                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
+            }
         }
-        function createBooleanType(trueFalseTypes) {
-            var type = getUnionType(trueFalseTypes);
-            type.flags |= 16 /* Boolean */;
-            type.intrinsicName = "boolean";
-            return type;
+        function bindOptionalChainRest(node) {
+            switch (node.kind) {
+                case 201 /* PropertyAccessExpression */:
+                    bind(node.questionDotToken);
+                    bind(node.name);
+                    break;
+                case 202 /* ElementAccessExpression */:
+                    bind(node.questionDotToken);
+                    bind(node.argumentExpression);
+                    break;
+                case 203 /* CallExpression */:
+                    bind(node.questionDotToken);
+                    bindEach(node.typeArguments);
+                    bindEach(node.arguments);
+                    break;
+            }
         }
-        function createObjectType(objectFlags, symbol) {
-            var type = createType(524288 /* Object */);
-            type.objectFlags = objectFlags;
-            type.symbol = symbol;
-            type.members = undefined;
-            type.properties = undefined;
-            type.callSignatures = undefined;
-            type.constructSignatures = undefined;
-            type.stringIndexInfo = undefined;
-            type.numberIndexInfo = undefined;
-            return type;
+        function bindOptionalChain(node, trueTarget, falseTarget) {
+            // For an optional chain, we emulate the behavior of a logical expression:
+            //
+            // a?.b         -> a && a.b
+            // a?.b.c       -> a && a.b.c
+            // a?.b?.c      -> a && a.b && a.b.c
+            // a?.[x = 1]   -> a && a[x = 1]
+            //
+            // To do this we descend through the chain until we reach the root of a chain (the expression with a `?.`)
+            // and build it's CFA graph as if it were the first condition (`a && ...`). Then we bind the rest
+            // of the node as part of the "true" branch, and continue to do so as we ascend back up to the outermost
+            // chain node. We then treat the entire node as the right side of the expression.
+            var preChainLabel = ts.isOptionalChainRoot(node) ? createBranchLabel() : undefined;
+            bindOptionalExpression(node.expression, preChainLabel || trueTarget, falseTarget);
+            if (preChainLabel) {
+                currentFlow = finishFlowLabel(preChainLabel);
+            }
+            doWithConditionalBranches(bindOptionalChainRest, node, trueTarget, falseTarget);
+            if (ts.isOutermostOptionalChain(node)) {
+                addAntecedent(trueTarget, createFlowCondition(32 /* TrueCondition */, currentFlow, node));
+                addAntecedent(falseTarget, createFlowCondition(64 /* FalseCondition */, currentFlow, node));
+            }
         }
-        function createTypeofType() {
-            return getUnionType(ts.arrayFrom(typeofEQFacts.keys(), getLiteralType));
+        function bindOptionalChainFlow(node) {
+            if (isTopLevelLogicalExpression(node)) {
+                var postExpressionLabel = createBranchLabel();
+                bindOptionalChain(node, postExpressionLabel, postExpressionLabel);
+                currentFlow = finishFlowLabel(postExpressionLabel);
+            }
+            else {
+                bindOptionalChain(node, currentTrueTarget, currentFalseTarget);
+            }
         }
-        function createTypeParameter(symbol) {
-            var type = createType(262144 /* TypeParameter */);
-            if (symbol)
-                type.symbol = symbol;
-            return type;
+        function bindNonNullExpressionFlow(node) {
+            if (ts.isOptionalChain(node)) {
+                bindOptionalChainFlow(node);
+            }
+            else {
+                bindEachChild(node);
+            }
         }
-        // A reserved member name starts with two underscores, but the third character cannot be an underscore,
-        // @, or #. A third underscore indicates an escaped form of an identifier that started
-        // with at least two underscores. The @ character indicates that the name is denoted by a well known ES
-        // Symbol instance and the # character indicates that the name is a PrivateIdentifier.
-        function isReservedMemberName(name) {
-            return name.charCodeAt(0) === 95 /* _ */ &&
-                name.charCodeAt(1) === 95 /* _ */ &&
-                name.charCodeAt(2) !== 95 /* _ */ &&
-                name.charCodeAt(2) !== 64 /* at */ &&
-                name.charCodeAt(2) !== 35 /* hash */;
+        function bindAccessExpressionFlow(node) {
+            if (ts.isOptionalChain(node)) {
+                bindOptionalChainFlow(node);
+            }
+            else {
+                bindEachChild(node);
+            }
         }
-        function getNamedMembers(members) {
-            var result;
-            members.forEach(function (symbol, id) {
-                if (!isReservedMemberName(id) && symbolIsValue(symbol)) {
-                    (result || (result = [])).push(symbol);
+        function bindCallExpressionFlow(node) {
+            if (ts.isOptionalChain(node)) {
+                bindOptionalChainFlow(node);
+            }
+            else {
+                // If the target of the call expression is a function expression or arrow function we have
+                // an immediately invoked function expression (IIFE). Initialize the flowNode property to
+                // the current control flow (which includes evaluation of the IIFE arguments).
+                var expr = ts.skipParentheses(node.expression);
+                if (expr.kind === 208 /* FunctionExpression */ || expr.kind === 209 /* ArrowFunction */) {
+                    bindEach(node.typeArguments);
+                    bindEach(node.arguments);
+                    bind(node.expression);
                 }
-            });
-            return result || ts.emptyArray;
-        }
-        function setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo) {
-            type.members = members;
-            type.properties = members === emptySymbols ? ts.emptyArray : getNamedMembers(members);
-            type.callSignatures = callSignatures;
-            type.constructSignatures = constructSignatures;
-            type.stringIndexInfo = stringIndexInfo;
-            type.numberIndexInfo = numberIndexInfo;
-            return type;
-        }
-        function createAnonymousType(symbol, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo) {
-            return setStructuredTypeMembers(createObjectType(16 /* Anonymous */, symbol), members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
-        }
-        function forEachSymbolTableInScope(enclosingDeclaration, callback) {
-            var result;
-            var _loop_7 = function (location) {
-                // Locals of a source file are not in scope (because they get merged into the global symbol table)
-                if (location.locals && !isGlobalSourceFile(location)) {
-                    if (result = callback(location.locals)) {
-                        return { value: result };
+                else {
+                    bindEachChild(node);
+                    if (node.expression.kind === 105 /* SuperKeyword */) {
+                        currentFlow = createFlowCall(currentFlow, node);
                     }
                 }
-                switch (location.kind) {
-                    case 290 /* SourceFile */:
-                        if (!ts.isExternalOrCommonJsModule(location)) {
-                            break;
-                        }
-                    // falls through
-                    case 249 /* ModuleDeclaration */:
-                        var sym = getSymbolOfNode(location);
-                        // `sym` may not have exports if this module declaration is backed by the symbol for a `const` that's being rewritten
-                        // into a namespace - in such cases, it's best to just let the namespace appear empty (the const members couldn't have referred
-                        // to one another anyway)
-                        if (result = callback((sym === null || sym === void 0 ? void 0 : sym.exports) || emptySymbols)) {
-                            return { value: result };
-                        }
-                        break;
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 246 /* InterfaceDeclaration */:
-                        // Type parameters are bound into `members` lists so they can merge across declarations
-                        // This is troublesome, since in all other respects, they behave like locals :cries:
-                        // TODO: the below is shared with similar code in `resolveName` - in fact, rephrasing all this symbol
-                        // lookup logic in terms of `resolveName` would be nice
-                        // The below is used to lookup type parameters within a class or interface, as they are added to the class/interface locals
-                        // These can never be latebound, so the symbol's raw members are sufficient. `getMembersOfNode` cannot be used, as it would
-                        // trigger resolving late-bound names, which we may already be in the process of doing while we're here!
-                        var table_1;
-                        // TODO: Should this filtered table be cached in some way?
-                        (getSymbolOfNode(location).members || emptySymbols).forEach(function (memberSymbol, key) {
-                            if (memberSymbol.flags & (788968 /* Type */ & ~67108864 /* Assignment */)) {
-                                (table_1 || (table_1 = ts.createSymbolTable())).set(key, memberSymbol);
-                            }
-                        });
-                        if (table_1 && (result = callback(table_1))) {
-                            return { value: result };
-                        }
-                        break;
+            }
+            if (node.expression.kind === 201 /* PropertyAccessExpression */) {
+                var propertyAccess = node.expression;
+                if (ts.isIdentifier(propertyAccess.name) && isNarrowableOperand(propertyAccess.expression) && ts.isPushOrUnshiftIdentifier(propertyAccess.name)) {
+                    currentFlow = createFlowMutation(256 /* ArrayMutation */, currentFlow, node);
                 }
-            };
-            for (var location = enclosingDeclaration; location; location = location.parent) {
-                var state_2 = _loop_7(location);
-                if (typeof state_2 === "object")
-                    return state_2.value;
             }
-            return callback(globals);
-        }
-        function getQualifiedLeftMeaning(rightMeaning) {
-            // If we are looking in value space, the parent meaning is value, other wise it is namespace
-            return rightMeaning === 111551 /* Value */ ? 111551 /* Value */ : 1920 /* Namespace */;
         }
-        function getAccessibleSymbolChain(symbol, enclosingDeclaration, meaning, useOnlyExternalAliasing, visitedSymbolTablesMap) {
-            if (visitedSymbolTablesMap === void 0) { visitedSymbolTablesMap = ts.createMap(); }
-            if (!(symbol && !isPropertyOrMethodDeclarationSymbol(symbol))) {
-                return undefined;
+        function getContainerFlags(node) {
+            switch (node.kind) {
+                case 221 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 200 /* ObjectLiteralExpression */:
+                case 177 /* TypeLiteral */:
+                case 312 /* JSDocTypeLiteral */:
+                case 281 /* JsxAttributes */:
+                    return 1 /* IsContainer */;
+                case 253 /* InterfaceDeclaration */:
+                    return 1 /* IsContainer */ | 64 /* IsInterface */;
+                case 256 /* ModuleDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 190 /* MappedType */:
+                    return 1 /* IsContainer */ | 32 /* HasLocals */;
+                case 297 /* SourceFile */:
+                    return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */;
+                case 165 /* MethodDeclaration */:
+                    if (ts.isObjectLiteralOrClassExpressionMethod(node)) {
+                        return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */ | 8 /* IsFunctionLike */ | 128 /* IsObjectLiteralOrClassExpressionMethod */;
+                    }
+                // falls through
+                case 166 /* Constructor */:
+                case 251 /* FunctionDeclaration */:
+                case 164 /* MethodSignature */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 169 /* CallSignature */:
+                case 313 /* JSDocSignature */:
+                case 308 /* JSDocFunctionType */:
+                case 174 /* FunctionType */:
+                case 170 /* ConstructSignature */:
+                case 171 /* IndexSignature */:
+                case 175 /* ConstructorType */:
+                    return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */ | 8 /* IsFunctionLike */;
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                    return 1 /* IsContainer */ | 4 /* IsControlFlowContainer */ | 32 /* HasLocals */ | 8 /* IsFunctionLike */ | 16 /* IsFunctionExpression */;
+                case 257 /* ModuleBlock */:
+                    return 4 /* IsControlFlowContainer */;
+                case 163 /* PropertyDeclaration */:
+                    return node.initializer ? 4 /* IsControlFlowContainer */ : 0;
+                case 287 /* CatchClause */:
+                case 237 /* ForStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                case 258 /* CaseBlock */:
+                    return 2 /* IsBlockScopedContainer */;
+                case 230 /* Block */:
+                    // do not treat blocks directly inside a function as a block-scoped-container.
+                    // Locals that reside in this block should go to the function locals. Otherwise 'x'
+                    // would not appear to be a redeclaration of a block scoped local in the following
+                    // example:
+                    //
+                    //      function foo() {
+                    //          var x;
+                    //          let x;
+                    //      }
+                    //
+                    // If we placed 'var x' into the function locals and 'let x' into the locals of
+                    // the block, then there would be no collision.
+                    //
+                    // By not creating a new block-scoped-container here, we ensure that both 'var x'
+                    // and 'let x' go into the Function-container's locals, and we do get a collision
+                    // conflict.
+                    return ts.isFunctionLike(node.parent) ? 0 /* None */ : 2 /* IsBlockScopedContainer */;
             }
-            var id = "" + getSymbolId(symbol);
-            var visitedSymbolTables = visitedSymbolTablesMap.get(id);
-            if (!visitedSymbolTables) {
-                visitedSymbolTablesMap.set(id, visitedSymbolTables = []);
+            return 0 /* None */;
+        }
+        function addToContainerChain(next) {
+            if (lastContainer) {
+                lastContainer.nextContainer = next;
             }
-            return forEachSymbolTableInScope(enclosingDeclaration, getAccessibleSymbolChainFromSymbolTable);
-            /**
-             * @param {ignoreQualification} boolean Set when a symbol is being looked for through the exports of another symbol (meaning we have a route to qualify it already)
-             */
-            function getAccessibleSymbolChainFromSymbolTable(symbols, ignoreQualification) {
-                if (!ts.pushIfUnique(visitedSymbolTables, symbols)) {
-                    return undefined;
-                }
-                var result = trySymbolTable(symbols, ignoreQualification);
-                visitedSymbolTables.pop();
-                return result;
+            lastContainer = next;
+        }
+        function declareSymbolAndAddToSymbolTable(node, symbolFlags, symbolExcludes) {
+            switch (container.kind) {
+                // Modules, source files, and classes need specialized handling for how their
+                // members are declared (for example, a member of a class will go into a specific
+                // symbol table depending on if it is static or not). We defer to specialized
+                // handlers to take care of declaring these child members.
+                case 256 /* ModuleDeclaration */:
+                    return declareModuleMember(node, symbolFlags, symbolExcludes);
+                case 297 /* SourceFile */:
+                    return declareSourceFileMember(node, symbolFlags, symbolExcludes);
+                case 221 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                    return declareClassMember(node, symbolFlags, symbolExcludes);
+                case 255 /* EnumDeclaration */:
+                    return declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes);
+                case 177 /* TypeLiteral */:
+                case 312 /* JSDocTypeLiteral */:
+                case 200 /* ObjectLiteralExpression */:
+                case 253 /* InterfaceDeclaration */:
+                case 281 /* JsxAttributes */:
+                    // Interface/Object-types always have their children added to the 'members' of
+                    // their container. They are only accessible through an instance of their
+                    // container, and are never in scope otherwise (even inside the body of the
+                    // object / type / interface declaring them). An exception is type parameters,
+                    // which are in scope without qualification (similar to 'locals').
+                    return declareSymbol(container.symbol.members, container.symbol, node, symbolFlags, symbolExcludes);
+                case 174 /* FunctionType */:
+                case 175 /* ConstructorType */:
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 313 /* JSDocSignature */:
+                case 171 /* IndexSignature */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 166 /* Constructor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 308 /* JSDocFunctionType */:
+                case 331 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 254 /* TypeAliasDeclaration */:
+                case 190 /* MappedType */:
+                    // All the children of these container types are never visible through another
+                    // symbol (i.e. through another symbol's 'exports' or 'members').  Instead,
+                    // they're only accessed 'lexically' (i.e. from code that exists underneath
+                    // their container in the tree). To accomplish this, we simply add their declared
+                    // symbol to the 'locals' of the container.  These symbols can then be found as
+                    // the type checker walks up the containers, checking them for matching names.
+                    return declareSymbol(container.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
             }
-            function canQualifySymbol(symbolFromSymbolTable, meaning) {
-                // If the symbol is equivalent and doesn't need further qualification, this symbol is accessible
-                return !needsQualification(symbolFromSymbolTable, enclosingDeclaration, meaning) ||
-                    // If symbol needs qualification, make sure that parent is accessible, if it is then this symbol is accessible too
-                    !!getAccessibleSymbolChain(symbolFromSymbolTable.parent, enclosingDeclaration, getQualifiedLeftMeaning(meaning), useOnlyExternalAliasing, visitedSymbolTablesMap);
+        }
+        function declareClassMember(node, symbolFlags, symbolExcludes) {
+            return ts.hasSyntacticModifier(node, 32 /* Static */)
+                ? declareSymbol(container.symbol.exports, container.symbol, node, symbolFlags, symbolExcludes)
+                : declareSymbol(container.symbol.members, container.symbol, node, symbolFlags, symbolExcludes);
+        }
+        function declareSourceFileMember(node, symbolFlags, symbolExcludes) {
+            return ts.isExternalModule(file)
+                ? declareModuleMember(node, symbolFlags, symbolExcludes)
+                : declareSymbol(file.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
+        }
+        function hasExportDeclarations(node) {
+            var body = ts.isSourceFile(node) ? node : ts.tryCast(node.body, ts.isModuleBlock);
+            return !!body && body.statements.some(function (s) { return ts.isExportDeclaration(s) || ts.isExportAssignment(s); });
+        }
+        function setExportContextFlag(node) {
+            // A declaration source file or ambient module declaration that contains no export declarations (but possibly regular
+            // declarations with export modifiers) is an export context in which declarations are implicitly exported.
+            if (node.flags & 8388608 /* Ambient */ && !hasExportDeclarations(node)) {
+                node.flags |= 64 /* ExportContext */;
             }
-            function isAccessible(symbolFromSymbolTable, resolvedAliasSymbol, ignoreQualification) {
-                return (symbol === (resolvedAliasSymbol || symbolFromSymbolTable) || getMergedSymbol(symbol) === getMergedSymbol(resolvedAliasSymbol || symbolFromSymbolTable)) &&
-                    // if the symbolFromSymbolTable is not external module (it could be if it was determined as ambient external module and would be in globals table)
-                    // and if symbolFromSymbolTable or alias resolution matches the symbol,
-                    // check the symbol can be qualified, it is only then this symbol is accessible
-                    !ts.some(symbolFromSymbolTable.declarations, hasNonGlobalAugmentationExternalModuleSymbol) &&
-                    (ignoreQualification || canQualifySymbol(getMergedSymbol(symbolFromSymbolTable), meaning));
+            else {
+                node.flags &= ~64 /* ExportContext */;
             }
-            function trySymbolTable(symbols, ignoreQualification) {
-                // If symbol is directly available by its name in the symbol table
-                if (isAccessible(symbols.get(symbol.escapedName), /*resolvedAliasSymbol*/ undefined, ignoreQualification)) {
-                    return [symbol];
+        }
+        function bindModuleDeclaration(node) {
+            setExportContextFlag(node);
+            if (ts.isAmbientModule(node)) {
+                if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                    errorOnFirstToken(node, ts.Diagnostics.export_modifier_cannot_be_applied_to_ambient_modules_and_module_augmentations_since_they_are_always_visible);
                 }
-                // Check if symbol is any of the aliases in scope
-                var result = ts.forEachEntry(symbols, function (symbolFromSymbolTable) {
-                    if (symbolFromSymbolTable.flags & 2097152 /* Alias */
-                        && symbolFromSymbolTable.escapedName !== "export=" /* ExportEquals */
-                        && symbolFromSymbolTable.escapedName !== "default" /* Default */
-                        && !(ts.isUMDExportSymbol(symbolFromSymbolTable) && enclosingDeclaration && ts.isExternalModule(ts.getSourceFileOfNode(enclosingDeclaration)))
-                        // If `!useOnlyExternalAliasing`, we can use any type of alias to get the name
-                        && (!useOnlyExternalAliasing || ts.some(symbolFromSymbolTable.declarations, ts.isExternalModuleImportEqualsDeclaration))
-                        // While exports are generally considered to be in scope, export-specifier declared symbols are _not_
-                        // See similar comment in `resolveName` for details
-                        && (ignoreQualification || !ts.getDeclarationOfKind(symbolFromSymbolTable, 263 /* ExportSpecifier */))) {
-                        var resolvedImportedSymbol = resolveAlias(symbolFromSymbolTable);
-                        var candidate = getCandidateListForSymbol(symbolFromSymbolTable, resolvedImportedSymbol, ignoreQualification);
-                        if (candidate) {
-                            return candidate;
+                if (ts.isModuleAugmentationExternal(node)) {
+                    declareModuleSymbol(node);
+                }
+                else {
+                    var pattern = void 0;
+                    if (node.name.kind === 10 /* StringLiteral */) {
+                        var text = node.name.text;
+                        if (ts.hasZeroOrOneAsteriskCharacter(text)) {
+                            pattern = ts.tryParsePattern(text);
                         }
-                    }
-                    if (symbolFromSymbolTable.escapedName === symbol.escapedName && symbolFromSymbolTable.exportSymbol) {
-                        if (isAccessible(getMergedSymbol(symbolFromSymbolTable.exportSymbol), /*aliasSymbol*/ undefined, ignoreQualification)) {
-                            return [symbol];
+                        else {
+                            errorOnFirstToken(node.name, ts.Diagnostics.Pattern_0_can_have_at_most_one_Asterisk_character, text);
                         }
                     }
-                });
-                // If there's no result and we're looking at the global symbol table, treat `globalThis` like an alias and try to lookup thru that
-                return result || (symbols === globals ? getCandidateListForSymbol(globalThisSymbol, globalThisSymbol, ignoreQualification) : undefined);
-            }
-            function getCandidateListForSymbol(symbolFromSymbolTable, resolvedImportedSymbol, ignoreQualification) {
-                if (isAccessible(symbolFromSymbolTable, resolvedImportedSymbol, ignoreQualification)) {
-                    return [symbolFromSymbolTable];
+                    var symbol = declareSymbolAndAddToSymbolTable(node, 512 /* ValueModule */, 110735 /* ValueModuleExcludes */);
+                    file.patternAmbientModules = ts.append(file.patternAmbientModules, pattern && { pattern: pattern, symbol: symbol });
                 }
-                // Look in the exported members, if we can find accessibleSymbolChain, symbol is accessible using this chain
-                // but only if the symbolFromSymbolTable can be qualified
-                var candidateTable = getExportsOfSymbol(resolvedImportedSymbol);
-                var accessibleSymbolsFromExports = candidateTable && getAccessibleSymbolChainFromSymbolTable(candidateTable, /*ignoreQualification*/ true);
-                if (accessibleSymbolsFromExports && canQualifySymbol(symbolFromSymbolTable, getQualifiedLeftMeaning(meaning))) {
-                    return [symbolFromSymbolTable].concat(accessibleSymbolsFromExports);
+            }
+            else {
+                var state = declareModuleSymbol(node);
+                if (state !== 0 /* NonInstantiated */) {
+                    var symbol = node.symbol;
+                    // if module was already merged with some function, class or non-const enum, treat it as non-const-enum-only
+                    symbol.constEnumOnlyModule = (!(symbol.flags & (16 /* Function */ | 32 /* Class */ | 256 /* RegularEnum */)))
+                        // Current must be `const enum` only
+                        && state === 2 /* ConstEnumOnly */
+                        // Can't have been set to 'false' in a previous merged symbol. ('undefined' OK)
+                        && symbol.constEnumOnlyModule !== false;
                 }
             }
         }
-        function needsQualification(symbol, enclosingDeclaration, meaning) {
-            var qualify = false;
-            forEachSymbolTableInScope(enclosingDeclaration, function (symbolTable) {
-                // If symbol of this name is not available in the symbol table we are ok
-                var symbolFromSymbolTable = getMergedSymbol(symbolTable.get(symbol.escapedName));
-                if (!symbolFromSymbolTable) {
-                    // Continue to the next symbol table
-                    return false;
-                }
-                // If the symbol with this name is present it should refer to the symbol
-                if (symbolFromSymbolTable === symbol) {
-                    // No need to qualify
-                    return true;
-                }
-                // Qualify if the symbol from symbol table has same meaning as expected
-                symbolFromSymbolTable = (symbolFromSymbolTable.flags & 2097152 /* Alias */ && !ts.getDeclarationOfKind(symbolFromSymbolTable, 263 /* ExportSpecifier */)) ? resolveAlias(symbolFromSymbolTable) : symbolFromSymbolTable;
-                if (symbolFromSymbolTable.flags & meaning) {
-                    qualify = true;
-                    return true;
-                }
-                // Continue to the next symbol table
-                return false;
-            });
-            return qualify;
+        function declareModuleSymbol(node) {
+            var state = getModuleInstanceState(node);
+            var instantiated = state !== 0 /* NonInstantiated */;
+            declareSymbolAndAddToSymbolTable(node, instantiated ? 512 /* ValueModule */ : 1024 /* NamespaceModule */, instantiated ? 110735 /* ValueModuleExcludes */ : 0 /* NamespaceModuleExcludes */);
+            return state;
+        }
+        function bindFunctionOrConstructorType(node) {
+            // For a given function symbol "<...>(...) => T" we want to generate a symbol identical
+            // to the one we would get for: { <...>(...): T }
+            //
+            // We do that by making an anonymous type literal symbol, and then setting the function
+            // symbol as its sole member. To the rest of the system, this symbol will be indistinguishable
+            // from an actual type literal symbol you would have gotten had you used the long form.
+            var symbol = createSymbol(131072 /* Signature */, getDeclarationName(node)); // TODO: GH#18217
+            addDeclarationToSymbol(symbol, node, 131072 /* Signature */);
+            var typeLiteralSymbol = createSymbol(2048 /* TypeLiteral */, "__type" /* Type */);
+            addDeclarationToSymbol(typeLiteralSymbol, node, 2048 /* TypeLiteral */);
+            typeLiteralSymbol.members = ts.createSymbolTable();
+            typeLiteralSymbol.members.set(symbol.escapedName, symbol);
         }
-        function isPropertyOrMethodDeclarationSymbol(symbol) {
-            if (symbol.declarations && symbol.declarations.length) {
-                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                    var declaration = _a[_i];
-                    switch (declaration.kind) {
-                        case 159 /* PropertyDeclaration */:
-                        case 161 /* MethodDeclaration */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
-                            continue;
-                        default:
-                            return false;
+        function bindObjectLiteralExpression(node) {
+            var ElementKind;
+            (function (ElementKind) {
+                ElementKind[ElementKind["Property"] = 1] = "Property";
+                ElementKind[ElementKind["Accessor"] = 2] = "Accessor";
+            })(ElementKind || (ElementKind = {}));
+            if (inStrictMode && !ts.isAssignmentTarget(node)) {
+                var seen = new ts.Map();
+                for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
+                    var prop = _a[_i];
+                    if (prop.kind === 290 /* SpreadAssignment */ || prop.name.kind !== 78 /* Identifier */) {
+                        continue;
+                    }
+                    var identifier = prop.name;
+                    // ECMA-262 11.1.5 Object Initializer
+                    // If previous is not undefined then throw a SyntaxError exception if any of the following conditions are true
+                    // a.This production is contained in strict code and IsDataDescriptor(previous) is true and
+                    // IsDataDescriptor(propId.descriptor) is true.
+                    //    b.IsDataDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true.
+                    //    c.IsAccessorDescriptor(previous) is true and IsDataDescriptor(propId.descriptor) is true.
+                    //    d.IsAccessorDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true
+                    // and either both previous and propId.descriptor have[[Get]] fields or both previous and propId.descriptor have[[Set]] fields
+                    var currentKind = prop.kind === 288 /* PropertyAssignment */ || prop.kind === 289 /* ShorthandPropertyAssignment */ || prop.kind === 165 /* MethodDeclaration */
+                        ? 1 /* Property */
+                        : 2 /* Accessor */;
+                    var existingKind = seen.get(identifier.escapedText);
+                    if (!existingKind) {
+                        seen.set(identifier.escapedText, currentKind);
+                        continue;
+                    }
+                    if (currentKind === 1 /* Property */ && existingKind === 1 /* Property */) {
+                        var span = ts.getErrorSpanForNode(file, identifier);
+                        file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, ts.Diagnostics.An_object_literal_cannot_have_multiple_properties_with_the_same_name_in_strict_mode));
                     }
                 }
-                return true;
             }
-            return false;
+            return bindAnonymousDeclaration(node, 4096 /* ObjectLiteral */, "__object" /* Object */);
         }
-        function isTypeSymbolAccessible(typeSymbol, enclosingDeclaration) {
-            var access = isSymbolAccessible(typeSymbol, enclosingDeclaration, 788968 /* Type */, /*shouldComputeAliasesToMakeVisible*/ false);
-            return access.accessibility === 0 /* Accessible */;
+        function bindJsxAttributes(node) {
+            return bindAnonymousDeclaration(node, 4096 /* ObjectLiteral */, "__jsxAttributes" /* JSXAttributes */);
         }
-        function isValueSymbolAccessible(typeSymbol, enclosingDeclaration) {
-            var access = isSymbolAccessible(typeSymbol, enclosingDeclaration, 111551 /* Value */, /*shouldComputeAliasesToMakeVisible*/ false);
-            return access.accessibility === 0 /* Accessible */;
+        function bindJsxAttribute(node, symbolFlags, symbolExcludes) {
+            return declareSymbolAndAddToSymbolTable(node, symbolFlags, symbolExcludes);
         }
-        function isAnySymbolAccessible(symbols, enclosingDeclaration, initialSymbol, meaning, shouldComputeAliasesToMakeVisible) {
-            if (!ts.length(symbols))
-                return;
-            var hadAccessibleChain;
-            var earlyModuleBail = false;
-            for (var _i = 0, _a = symbols; _i < _a.length; _i++) {
-                var symbol = _a[_i];
-                // Symbol is accessible if it by itself is accessible
-                var accessibleSymbolChain = getAccessibleSymbolChain(symbol, enclosingDeclaration, meaning, /*useOnlyExternalAliasing*/ false);
-                if (accessibleSymbolChain) {
-                    hadAccessibleChain = symbol;
-                    var hasAccessibleDeclarations = hasVisibleDeclarations(accessibleSymbolChain[0], shouldComputeAliasesToMakeVisible);
-                    if (hasAccessibleDeclarations) {
-                        return hasAccessibleDeclarations;
+        function bindAnonymousDeclaration(node, symbolFlags, name) {
+            var symbol = createSymbol(symbolFlags, name);
+            if (symbolFlags & (8 /* EnumMember */ | 106500 /* ClassMember */)) {
+                symbol.parent = container.symbol;
+            }
+            addDeclarationToSymbol(symbol, node, symbolFlags);
+            return symbol;
+        }
+        function bindBlockScopedDeclaration(node, symbolFlags, symbolExcludes) {
+            switch (blockScopeContainer.kind) {
+                case 256 /* ModuleDeclaration */:
+                    declareModuleMember(node, symbolFlags, symbolExcludes);
+                    break;
+                case 297 /* SourceFile */:
+                    if (ts.isExternalOrCommonJsModule(container)) {
+                        declareModuleMember(node, symbolFlags, symbolExcludes);
+                        break;
                     }
-                }
-                else {
-                    if (ts.some(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
-                        if (shouldComputeAliasesToMakeVisible) {
-                            earlyModuleBail = true;
-                            // Generally speaking, we want to use the aliases that already exist to refer to a module, if present
-                            // In order to do so, we need to find those aliases in order to retain them in declaration emit; so
-                            // if we are in declaration emit, we cannot use the fast path for module visibility until we've exhausted
-                            // all other visibility options (in order to capture the possible aliases used to reference the module)
-                            continue;
+                // falls through
+                default:
+                    if (!blockScopeContainer.locals) {
+                        blockScopeContainer.locals = ts.createSymbolTable();
+                        addToContainerChain(blockScopeContainer);
+                    }
+                    declareSymbol(blockScopeContainer.locals, /*parent*/ undefined, node, symbolFlags, symbolExcludes);
+            }
+        }
+        function delayedBindJSDocTypedefTag() {
+            if (!delayedTypeAliases) {
+                return;
+            }
+            var saveContainer = container;
+            var saveLastContainer = lastContainer;
+            var saveBlockScopeContainer = blockScopeContainer;
+            var saveParent = parent;
+            var saveCurrentFlow = currentFlow;
+            for (var _i = 0, delayedTypeAliases_1 = delayedTypeAliases; _i < delayedTypeAliases_1.length; _i++) {
+                var typeAlias = delayedTypeAliases_1[_i];
+                var host = ts.getJSDocHost(typeAlias);
+                container = ts.findAncestor(host.parent, function (n) { return !!(getContainerFlags(n) & 1 /* IsContainer */); }) || file;
+                blockScopeContainer = ts.getEnclosingBlockScopeContainer(host) || file;
+                currentFlow = initFlowNode({ flags: 2 /* Start */ });
+                parent = typeAlias;
+                bind(typeAlias.typeExpression);
+                var declName = ts.getNameOfDeclaration(typeAlias);
+                if ((ts.isJSDocEnumTag(typeAlias) || !typeAlias.fullName) && declName && ts.isPropertyAccessEntityNameExpression(declName.parent)) {
+                    // typedef anchored to an A.B.C assignment - we need to bind into B's namespace under name C
+                    var isTopLevel = isTopLevelNamespaceAssignment(declName.parent);
+                    if (isTopLevel) {
+                        bindPotentiallyMissingNamespaces(file.symbol, declName.parent, isTopLevel, !!ts.findAncestor(declName, function (d) { return ts.isPropertyAccessExpression(d) && d.name.escapedText === "prototype"; }), /*containerIsClass*/ false);
+                        var oldContainer = container;
+                        switch (ts.getAssignmentDeclarationPropertyAccessKind(declName.parent)) {
+                            case 1 /* ExportsProperty */:
+                            case 2 /* ModuleExports */:
+                                if (!ts.isExternalOrCommonJsModule(file)) {
+                                    container = undefined;
+                                }
+                                else {
+                                    container = file;
+                                }
+                                break;
+                            case 4 /* ThisProperty */:
+                                container = declName.parent.expression;
+                                break;
+                            case 3 /* PrototypeProperty */:
+                                container = declName.parent.expression.name;
+                                break;
+                            case 5 /* Property */:
+                                container = isExportsOrModuleExportsOrAlias(file, declName.parent.expression) ? file
+                                    : ts.isPropertyAccessExpression(declName.parent.expression) ? declName.parent.expression.name
+                                        : declName.parent.expression;
+                                break;
+                            case 0 /* None */:
+                                return ts.Debug.fail("Shouldn't have detected typedef or enum on non-assignment declaration");
                         }
-                        // Any meaning of a module symbol is always accessible via an `import` type
-                        return {
-                            accessibility: 0 /* Accessible */
-                        };
+                        if (container) {
+                            declareModuleMember(typeAlias, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
+                        }
+                        container = oldContainer;
                     }
                 }
-                // If we haven't got the accessible symbol, it doesn't mean the symbol is actually inaccessible.
-                // It could be a qualified symbol and hence verify the path
-                // e.g.:
-                // module m {
-                //     export class c {
-                //     }
-                // }
-                // const x: typeof m.c
-                // In the above example when we start with checking if typeof m.c symbol is accessible,
-                // we are going to see if c can be accessed in scope directly.
-                // But it can't, hence the accessible is going to be undefined, but that doesn't mean m.c is inaccessible
-                // It is accessible if the parent m is accessible because then m.c can be accessed through qualification
-                var containers = getContainersOfSymbol(symbol, enclosingDeclaration);
-                // If we're trying to reference some object literal in, eg `var a = { x: 1 }`, the symbol for the literal, `__object`, is distinct
-                // from the symbol of the declaration it is being assigned to. Since we can use the declaration to refer to the literal, however,
-                // we'd like to make that connection here - potentially causing us to paint the declaration's visibility, and therefore the literal.
-                var firstDecl = !!ts.length(symbol.declarations) && ts.first(symbol.declarations);
-                if (!ts.length(containers) && meaning & 111551 /* Value */ && firstDecl && ts.isObjectLiteralExpression(firstDecl)) {
-                    if (firstDecl.parent && ts.isVariableDeclaration(firstDecl.parent) && firstDecl === firstDecl.parent.initializer) {
-                        containers = [getSymbolOfNode(firstDecl.parent)];
-                    }
+                else if (ts.isJSDocEnumTag(typeAlias) || !typeAlias.fullName || typeAlias.fullName.kind === 78 /* Identifier */) {
+                    parent = typeAlias.parent;
+                    bindBlockScopedDeclaration(typeAlias, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
                 }
-                var parentResult = isAnySymbolAccessible(containers, enclosingDeclaration, initialSymbol, initialSymbol === symbol ? getQualifiedLeftMeaning(meaning) : meaning, shouldComputeAliasesToMakeVisible);
-                if (parentResult) {
-                    return parentResult;
+                else {
+                    bind(typeAlias.fullName);
                 }
             }
-            if (earlyModuleBail) {
-                return {
-                    accessibility: 0 /* Accessible */
-                };
-            }
-            if (hadAccessibleChain) {
-                return {
-                    accessibility: 1 /* NotAccessible */,
-                    errorSymbolName: symbolToString(initialSymbol, enclosingDeclaration, meaning),
-                    errorModuleName: hadAccessibleChain !== initialSymbol ? symbolToString(hadAccessibleChain, enclosingDeclaration, 1920 /* Namespace */) : undefined,
-                };
-            }
+            container = saveContainer;
+            lastContainer = saveLastContainer;
+            blockScopeContainer = saveBlockScopeContainer;
+            parent = saveParent;
+            currentFlow = saveCurrentFlow;
         }
-        /**
-         * Check if the given symbol in given enclosing declaration is accessible and mark all associated alias to be visible if requested
-         *
-         * @param symbol a Symbol to check if accessible
-         * @param enclosingDeclaration a Node containing reference to the symbol
-         * @param meaning a SymbolFlags to check if such meaning of the symbol is accessible
-         * @param shouldComputeAliasToMakeVisible a boolean value to indicate whether to return aliases to be mark visible in case the symbol is accessible
-         */
-        function isSymbolAccessible(symbol, enclosingDeclaration, meaning, shouldComputeAliasesToMakeVisible) {
-            if (symbol && enclosingDeclaration) {
-                var result = isAnySymbolAccessible([symbol], enclosingDeclaration, symbol, meaning, shouldComputeAliasesToMakeVisible);
-                if (result) {
-                    return result;
+        // The binder visits every node in the syntax tree so it is a convenient place to perform a single localized
+        // check for reserved words used as identifiers in strict mode code, as well as `yield` or `await` in
+        // [Yield] or [Await] contexts, respectively.
+        function checkContextualIdentifier(node) {
+            // Report error only if there are no parse errors in file
+            if (!file.parseDiagnostics.length &&
+                !(node.flags & 8388608 /* Ambient */) &&
+                !(node.flags & 4194304 /* JSDoc */) &&
+                !ts.isIdentifierName(node)) {
+                // strict mode identifiers
+                if (inStrictMode &&
+                    node.originalKeywordKind >= 116 /* FirstFutureReservedWord */ &&
+                    node.originalKeywordKind <= 124 /* LastFutureReservedWord */) {
+                    file.bindDiagnostics.push(createDiagnosticForNode(node, getStrictModeIdentifierMessage(node), ts.declarationNameToString(node)));
                 }
-                // This could be a symbol that is not exported in the external module
-                // or it could be a symbol from different external module that is not aliased and hence cannot be named
-                var symbolExternalModule = ts.forEach(symbol.declarations, getExternalModuleContainer);
-                if (symbolExternalModule) {
-                    var enclosingExternalModule = getExternalModuleContainer(enclosingDeclaration);
-                    if (symbolExternalModule !== enclosingExternalModule) {
-                        // name from different external module that is not visible
-                        return {
-                            accessibility: 2 /* CannotBeNamed */,
-                            errorSymbolName: symbolToString(symbol, enclosingDeclaration, meaning),
-                            errorModuleName: symbolToString(symbolExternalModule)
-                        };
+                else if (node.originalKeywordKind === 130 /* AwaitKeyword */) {
+                    if (ts.isExternalModule(file) && ts.isInTopLevelContext(node)) {
+                        file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_at_the_top_level_of_a_module, ts.declarationNameToString(node)));
+                    }
+                    else if (node.flags & 32768 /* AwaitContext */) {
+                        file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_that_cannot_be_used_here, ts.declarationNameToString(node)));
                     }
                 }
-                // Just a local name that is not accessible
-                return {
-                    accessibility: 1 /* NotAccessible */,
-                    errorSymbolName: symbolToString(symbol, enclosingDeclaration, meaning),
-                };
+                else if (node.originalKeywordKind === 124 /* YieldKeyword */ && node.flags & 8192 /* YieldContext */) {
+                    file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_that_cannot_be_used_here, ts.declarationNameToString(node)));
+                }
             }
-            return { accessibility: 0 /* Accessible */ };
         }
-        function getExternalModuleContainer(declaration) {
-            var node = ts.findAncestor(declaration, hasExternalModuleSymbol);
-            return node && getSymbolOfNode(node);
+        function getStrictModeIdentifierMessage(node) {
+            // Provide specialized messages to help the user understand why we think they're in
+            // strict mode.
+            if (ts.getContainingClass(node)) {
+                return ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode_Class_definitions_are_automatically_in_strict_mode;
+            }
+            if (file.externalModuleIndicator) {
+                return ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode_Modules_are_automatically_in_strict_mode;
+            }
+            return ts.Diagnostics.Identifier_expected_0_is_a_reserved_word_in_strict_mode;
         }
-        function hasExternalModuleSymbol(declaration) {
-            return ts.isAmbientModule(declaration) || (declaration.kind === 290 /* SourceFile */ && ts.isExternalOrCommonJsModule(declaration));
+        // The binder visits every node, so this is a good place to check for
+        // the reserved private name (there is only one)
+        function checkPrivateIdentifier(node) {
+            if (node.escapedText === "#constructor") {
+                // Report error only if there are no parse errors in file
+                if (!file.parseDiagnostics.length) {
+                    file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.constructor_is_a_reserved_word, ts.declarationNameToString(node)));
+                }
+            }
         }
-        function hasNonGlobalAugmentationExternalModuleSymbol(declaration) {
-            return ts.isModuleWithStringLiteralName(declaration) || (declaration.kind === 290 /* SourceFile */ && ts.isExternalOrCommonJsModule(declaration));
+        function checkStrictModeBinaryExpression(node) {
+            if (inStrictMode && ts.isLeftHandSideExpression(node.left) && ts.isAssignmentOperator(node.operatorToken.kind)) {
+                // ECMA 262 (Annex C) The identifier eval or arguments may not appear as the LeftHandSideExpression of an
+                // Assignment operator(11.13) or of a PostfixExpression(11.3)
+                checkStrictModeEvalOrArguments(node, node.left);
+            }
         }
-        function hasVisibleDeclarations(symbol, shouldComputeAliasToMakeVisible) {
-            var aliasesToMakeVisible;
-            if (!ts.every(ts.filter(symbol.declarations, function (d) { return d.kind !== 75 /* Identifier */; }), getIsDeclarationVisible)) {
-                return undefined;
+        function checkStrictModeCatchClause(node) {
+            // It is a SyntaxError if a TryStatement with a Catch occurs within strict code and the Identifier of the
+            // Catch production is eval or arguments
+            if (inStrictMode && node.variableDeclaration) {
+                checkStrictModeEvalOrArguments(node, node.variableDeclaration.name);
             }
-            return { accessibility: 0 /* Accessible */, aliasesToMakeVisible: aliasesToMakeVisible };
-            function getIsDeclarationVisible(declaration) {
-                if (!isDeclarationVisible(declaration)) {
-                    // Mark the unexported alias as visible if its parent is visible
-                    // because these kind of aliases can be used to name types in declaration file
-                    var anyImportSyntax = getAnyImportSyntax(declaration);
-                    if (anyImportSyntax &&
-                        !ts.hasModifier(anyImportSyntax, 1 /* Export */) && // import clause without export
-                        isDeclarationVisible(anyImportSyntax.parent)) {
-                        return addVisibleAlias(declaration, anyImportSyntax);
-                    }
-                    else if (ts.isVariableDeclaration(declaration) && ts.isVariableStatement(declaration.parent.parent) &&
-                        !ts.hasModifier(declaration.parent.parent, 1 /* Export */) && // unexported variable statement
-                        isDeclarationVisible(declaration.parent.parent.parent)) {
-                        return addVisibleAlias(declaration, declaration.parent.parent);
-                    }
-                    else if (ts.isLateVisibilityPaintedStatement(declaration) // unexported top-level statement
-                        && !ts.hasModifier(declaration, 1 /* Export */)
-                        && isDeclarationVisible(declaration.parent)) {
-                        return addVisibleAlias(declaration, declaration);
-                    }
-                    // Declaration is not visible
-                    return false;
-                }
-                return true;
+        }
+        function checkStrictModeDeleteExpression(node) {
+            // Grammar checking
+            if (inStrictMode && node.expression.kind === 78 /* Identifier */) {
+                // When a delete operator occurs within strict mode code, a SyntaxError is thrown if its
+                // UnaryExpression is a direct reference to a variable, function argument, or function name
+                var span = ts.getErrorSpanForNode(file, node.expression);
+                file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, ts.Diagnostics.delete_cannot_be_called_on_an_identifier_in_strict_mode));
             }
-            function addVisibleAlias(declaration, aliasingStatement) {
-                // In function "buildTypeDisplay" where we decide whether to write type-alias or serialize types,
-                // we want to just check if type- alias is accessible or not but we don't care about emitting those alias at that time
-                // since we will do the emitting later in trackSymbol.
-                if (shouldComputeAliasToMakeVisible) {
-                    getNodeLinks(declaration).isVisible = true;
-                    aliasesToMakeVisible = ts.appendIfUnique(aliasesToMakeVisible, aliasingStatement);
+        }
+        function isEvalOrArgumentsIdentifier(node) {
+            return ts.isIdentifier(node) && (node.escapedText === "eval" || node.escapedText === "arguments");
+        }
+        function checkStrictModeEvalOrArguments(contextNode, name) {
+            if (name && name.kind === 78 /* Identifier */) {
+                var identifier = name;
+                if (isEvalOrArgumentsIdentifier(identifier)) {
+                    // We check first if the name is inside class declaration or class expression; if so give explicit message
+                    // otherwise report generic error message.
+                    var span = ts.getErrorSpanForNode(file, name);
+                    file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, getStrictModeEvalOrArgumentsMessage(contextNode), ts.idText(identifier)));
                 }
-                return true;
             }
         }
-        function isEntityNameVisible(entityName, enclosingDeclaration) {
-            // get symbol of the first identifier of the entityName
-            var meaning;
-            if (entityName.parent.kind === 172 /* TypeQuery */ ||
-                ts.isExpressionWithTypeArgumentsInClassExtendsClause(entityName.parent) ||
-                entityName.parent.kind === 154 /* ComputedPropertyName */) {
-                // Typeof value
-                meaning = 111551 /* Value */ | 1048576 /* ExportValue */;
+        function getStrictModeEvalOrArgumentsMessage(node) {
+            // Provide specialized messages to help the user understand why we think they're in
+            // strict mode.
+            if (ts.getContainingClass(node)) {
+                return ts.Diagnostics.Invalid_use_of_0_Class_definitions_are_automatically_in_strict_mode;
             }
-            else if (entityName.kind === 153 /* QualifiedName */ || entityName.kind === 194 /* PropertyAccessExpression */ ||
-                entityName.parent.kind === 253 /* ImportEqualsDeclaration */) {
-                // Left identifier from type reference or TypeAlias
-                // Entity name of the import declaration
-                meaning = 1920 /* Namespace */;
+            if (file.externalModuleIndicator) {
+                return ts.Diagnostics.Invalid_use_of_0_Modules_are_automatically_in_strict_mode;
             }
-            else {
-                // Type Reference or TypeAlias entity = Identifier
-                meaning = 788968 /* Type */;
+            return ts.Diagnostics.Invalid_use_of_0_in_strict_mode;
+        }
+        function checkStrictModeFunctionName(node) {
+            if (inStrictMode) {
+                // It is a SyntaxError if the identifier eval or arguments appears within a FormalParameterList of a strict mode FunctionDeclaration or FunctionExpression (13.1))
+                checkStrictModeEvalOrArguments(node, node.name);
             }
-            var firstIdentifier = ts.getFirstIdentifier(entityName);
-            var symbol = resolveName(enclosingDeclaration, firstIdentifier.escapedText, meaning, /*nodeNotFoundErrorMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
-            // Verify if the symbol is accessible
-            return (symbol && hasVisibleDeclarations(symbol, /*shouldComputeAliasToMakeVisible*/ true)) || {
-                accessibility: 1 /* NotAccessible */,
-                errorSymbolName: ts.getTextOfNode(firstIdentifier),
-                errorNode: firstIdentifier
-            };
         }
-        function symbolToString(symbol, enclosingDeclaration, meaning, flags, writer) {
-            if (flags === void 0) { flags = 4 /* AllowAnyNodeKind */; }
-            var nodeFlags = 70221824 /* IgnoreErrors */;
-            if (flags & 2 /* UseOnlyExternalAliasing */) {
-                nodeFlags |= 128 /* UseOnlyExternalAliasing */;
+        function getStrictModeBlockScopeFunctionDeclarationMessage(node) {
+            // Provide specialized messages to help the user understand why we think they're in
+            // strict mode.
+            if (ts.getContainingClass(node)) {
+                return ts.Diagnostics.Function_declarations_are_not_allowed_inside_blocks_in_strict_mode_when_targeting_ES3_or_ES5_Class_definitions_are_automatically_in_strict_mode;
             }
-            if (flags & 1 /* WriteTypeParametersOrArguments */) {
-                nodeFlags |= 512 /* WriteTypeParametersInQualifiedName */;
+            if (file.externalModuleIndicator) {
+                return ts.Diagnostics.Function_declarations_are_not_allowed_inside_blocks_in_strict_mode_when_targeting_ES3_or_ES5_Modules_are_automatically_in_strict_mode;
             }
-            if (flags & 8 /* UseAliasDefinedOutsideCurrentScope */) {
-                nodeFlags |= 16384 /* UseAliasDefinedOutsideCurrentScope */;
+            return ts.Diagnostics.Function_declarations_are_not_allowed_inside_blocks_in_strict_mode_when_targeting_ES3_or_ES5;
+        }
+        function checkStrictModeFunctionDeclaration(node) {
+            if (languageVersion < 2 /* ES2015 */) {
+                // Report error if function is not top level function declaration
+                if (blockScopeContainer.kind !== 297 /* SourceFile */ &&
+                    blockScopeContainer.kind !== 256 /* ModuleDeclaration */ &&
+                    !ts.isFunctionLike(blockScopeContainer)) {
+                    // We check first if the name is inside class declaration or class expression; if so give explicit message
+                    // otherwise report generic error message.
+                    var errorSpan = ts.getErrorSpanForNode(file, node);
+                    file.bindDiagnostics.push(ts.createFileDiagnostic(file, errorSpan.start, errorSpan.length, getStrictModeBlockScopeFunctionDeclarationMessage(node)));
+                }
             }
-            if (flags & 16 /* DoNotIncludeSymbolChain */) {
-                nodeFlags |= 134217728 /* DoNotIncludeSymbolChain */;
+        }
+        function checkStrictModeNumericLiteral(node) {
+            if (inStrictMode && node.numericLiteralFlags & 32 /* Octal */) {
+                file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Octal_literals_are_not_allowed_in_strict_mode));
             }
-            var builder = flags & 4 /* AllowAnyNodeKind */ ? nodeBuilder.symbolToExpression : nodeBuilder.symbolToEntityName;
-            return writer ? symbolToStringWorker(writer).getText() : ts.usingSingleLineStringWriter(symbolToStringWorker);
-            function symbolToStringWorker(writer) {
-                var entity = builder(symbol, meaning, enclosingDeclaration, nodeFlags); // TODO: GH#18217
-                var printer = ts.createPrinter({ removeComments: true });
-                var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
-                printer.writeNode(4 /* Unspecified */, entity, /*sourceFile*/ sourceFile, writer);
-                return writer;
+        }
+        function checkStrictModePostfixUnaryExpression(node) {
+            // Grammar checking
+            // The identifier eval or arguments may not appear as the LeftHandSideExpression of an
+            // Assignment operator(11.13) or of a PostfixExpression(11.3) or as the UnaryExpression
+            // operated upon by a Prefix Increment(11.4.4) or a Prefix Decrement(11.4.5) operator.
+            if (inStrictMode) {
+                checkStrictModeEvalOrArguments(node, node.operand);
             }
         }
-        function signatureToString(signature, enclosingDeclaration, flags, kind, writer) {
-            if (flags === void 0) { flags = 0 /* None */; }
-            return writer ? signatureToStringWorker(writer).getText() : ts.usingSingleLineStringWriter(signatureToStringWorker);
-            function signatureToStringWorker(writer) {
-                var sigOutput;
-                if (flags & 262144 /* WriteArrowStyleSignature */) {
-                    sigOutput = kind === 1 /* Construct */ ? 171 /* ConstructorType */ : 170 /* FunctionType */;
-                }
-                else {
-                    sigOutput = kind === 1 /* Construct */ ? 166 /* ConstructSignature */ : 165 /* CallSignature */;
+        function checkStrictModePrefixUnaryExpression(node) {
+            // Grammar checking
+            if (inStrictMode) {
+                if (node.operator === 45 /* PlusPlusToken */ || node.operator === 46 /* MinusMinusToken */) {
+                    checkStrictModeEvalOrArguments(node, node.operand);
                 }
-                var sig = nodeBuilder.signatureToSignatureDeclaration(signature, sigOutput, enclosingDeclaration, toNodeBuilderFlags(flags) | 70221824 /* IgnoreErrors */ | 512 /* WriteTypeParametersInQualifiedName */);
-                var printer = ts.createPrinter({ removeComments: true, omitTrailingSemicolon: true });
-                var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
-                printer.writeNode(4 /* Unspecified */, sig, /*sourceFile*/ sourceFile, ts.getTrailingSemicolonDeferringWriter(writer)); // TODO: GH#18217
-                return writer;
             }
         }
-        function typeToString(type, enclosingDeclaration, flags, writer) {
-            if (flags === void 0) { flags = 1048576 /* AllowUniqueESSymbolType */ | 16384 /* UseAliasDefinedOutsideCurrentScope */; }
-            if (writer === void 0) { writer = ts.createTextWriter(""); }
-            var noTruncation = compilerOptions.noErrorTruncation || flags & 1 /* NoTruncation */;
-            var typeNode = nodeBuilder.typeToTypeNode(type, enclosingDeclaration, toNodeBuilderFlags(flags) | 70221824 /* IgnoreErrors */ | (noTruncation ? 1 /* NoTruncation */ : 0), writer);
-            if (typeNode === undefined)
-                return ts.Debug.fail("should always get typenode");
-            var options = { removeComments: true };
-            var printer = ts.createPrinter(options);
-            var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
-            printer.writeNode(4 /* Unspecified */, typeNode, /*sourceFile*/ sourceFile, writer);
-            var result = writer.getText();
-            var maxLength = noTruncation ? ts.noTruncationMaximumTruncationLength * 2 : ts.defaultMaximumTruncationLength * 2;
-            if (maxLength && result && result.length >= maxLength) {
-                return result.substr(0, maxLength - "...".length) + "...";
+        function checkStrictModeWithStatement(node) {
+            // Grammar checking for withStatement
+            if (inStrictMode) {
+                errorOnFirstToken(node, ts.Diagnostics.with_statements_are_not_allowed_in_strict_mode);
             }
-            return result;
         }
-        function getTypeNamesForErrorDisplay(left, right) {
-            var leftStr = symbolValueDeclarationIsContextSensitive(left.symbol) ? typeToString(left, left.symbol.valueDeclaration) : typeToString(left);
-            var rightStr = symbolValueDeclarationIsContextSensitive(right.symbol) ? typeToString(right, right.symbol.valueDeclaration) : typeToString(right);
-            if (leftStr === rightStr) {
-                leftStr = typeToString(left, /*enclosingDeclaration*/ undefined, 64 /* UseFullyQualifiedType */);
-                rightStr = typeToString(right, /*enclosingDeclaration*/ undefined, 64 /* UseFullyQualifiedType */);
+        function checkStrictModeLabeledStatement(node) {
+            // Grammar checking for labeledStatement
+            if (inStrictMode && options.target >= 2 /* ES2015 */) {
+                if (ts.isDeclarationStatement(node.statement) || ts.isVariableStatement(node.statement)) {
+                    errorOnFirstToken(node.label, ts.Diagnostics.A_label_is_not_allowed_here);
+                }
             }
-            return [leftStr, rightStr];
         }
-        function symbolValueDeclarationIsContextSensitive(symbol) {
-            return symbol && symbol.valueDeclaration && ts.isExpression(symbol.valueDeclaration) && !isContextSensitive(symbol.valueDeclaration);
+        function errorOnFirstToken(node, message, arg0, arg1, arg2) {
+            var span = ts.getSpanOfTokenAtPosition(file, node.pos);
+            file.bindDiagnostics.push(ts.createFileDiagnostic(file, span.start, span.length, message, arg0, arg1, arg2));
         }
-        function toNodeBuilderFlags(flags) {
-            if (flags === void 0) { flags = 0 /* None */; }
-            return flags & 814775659 /* NodeBuilderFlagsMask */;
+        function errorOrSuggestionOnNode(isError, node, message) {
+            errorOrSuggestionOnRange(isError, node, node, message);
         }
-        function createNodeBuilder() {
-            return {
-                typeToTypeNode: function (type, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return typeToTypeNodeHelper(type, context); });
-                },
-                indexInfoToIndexSignatureDeclaration: function (indexInfo, kind, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return indexInfoToIndexSignatureDeclarationHelper(indexInfo, kind, context); });
-                },
-                signatureToSignatureDeclaration: function (signature, kind, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return signatureToSignatureDeclarationHelper(signature, kind, context); });
-                },
-                symbolToEntityName: function (symbol, meaning, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolToName(symbol, context, meaning, /*expectsIdentifier*/ false); });
-                },
-                symbolToExpression: function (symbol, meaning, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolToExpression(symbol, context, meaning); });
-                },
-                symbolToTypeParameterDeclarations: function (symbol, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return typeParametersToTypeParameterDeclarations(symbol, context); });
-                },
-                symbolToParameterDeclaration: function (symbol, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolToParameterDeclaration(symbol, context); });
-                },
-                typeParameterToDeclaration: function (parameter, enclosingDeclaration, flags, tracker) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return typeParameterToDeclaration(parameter, context); });
-                },
-                symbolTableToDeclarationStatements: function (symbolTable, enclosingDeclaration, flags, tracker, bundled) {
-                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolTableToDeclarationStatements(symbolTable, context, bundled); });
-                },
-            };
-            function withContext(enclosingDeclaration, flags, tracker, cb) {
-                ts.Debug.assert(enclosingDeclaration === undefined || (enclosingDeclaration.flags & 8 /* Synthesized */) === 0);
-                var context = {
-                    enclosingDeclaration: enclosingDeclaration,
-                    flags: flags || 0 /* None */,
-                    // If no full tracker is provided, fake up a dummy one with a basic limited-functionality moduleResolverHost
-                    tracker: tracker && tracker.trackSymbol ? tracker : { trackSymbol: ts.noop, moduleResolverHost: flags & 134217728 /* DoNotIncludeSymbolChain */ ? {
-                            getCommonSourceDirectory: !!host.getCommonSourceDirectory ? function () { return host.getCommonSourceDirectory(); } : function () { return ""; },
-                            getSourceFiles: function () { return host.getSourceFiles(); },
-                            getCurrentDirectory: function () { return host.getCurrentDirectory(); },
-                            getProbableSymlinks: ts.maybeBind(host, host.getProbableSymlinks),
-                            useCaseSensitiveFileNames: ts.maybeBind(host, host.useCaseSensitiveFileNames),
-                            redirectTargetsMap: host.redirectTargetsMap,
-                            getProjectReferenceRedirect: function (fileName) { return host.getProjectReferenceRedirect(fileName); },
-                            isSourceOfProjectReferenceRedirect: function (fileName) { return host.isSourceOfProjectReferenceRedirect(fileName); },
-                            fileExists: function (fileName) { return host.fileExists(fileName); },
-                        } : undefined },
-                    encounteredError: false,
-                    visitedTypes: undefined,
-                    symbolDepth: undefined,
-                    inferTypeParameters: undefined,
-                    approximateLength: 0
-                };
-                var resultingNode = cb(context);
-                return context.encounteredError ? undefined : resultingNode;
+        function errorOrSuggestionOnRange(isError, startNode, endNode, message) {
+            addErrorOrSuggestionDiagnostic(isError, { pos: ts.getTokenPosOfNode(startNode, file), end: endNode.end }, message);
+        }
+        function addErrorOrSuggestionDiagnostic(isError, range, message) {
+            var diag = ts.createFileDiagnostic(file, range.pos, range.end - range.pos, message);
+            if (isError) {
+                file.bindDiagnostics.push(diag);
             }
-            function checkTruncationLength(context) {
-                if (context.truncating)
-                    return context.truncating;
-                return context.truncating = context.approximateLength > ((context.flags & 1 /* NoTruncation */) ? ts.noTruncationMaximumTruncationLength : ts.defaultMaximumTruncationLength);
+            else {
+                file.bindSuggestionDiagnostics = ts.append(file.bindSuggestionDiagnostics, __assign(__assign({}, diag), { category: ts.DiagnosticCategory.Suggestion }));
             }
-            function typeToTypeNodeHelper(type, context) {
-                if (cancellationToken && cancellationToken.throwIfCancellationRequested) {
-                    cancellationToken.throwIfCancellationRequested();
-                }
-                var inTypeAlias = context.flags & 8388608 /* InTypeAlias */;
-                context.flags &= ~8388608 /* InTypeAlias */;
-                if (!type) {
-                    if (!(context.flags & 262144 /* AllowEmptyUnionOrIntersection */)) {
-                        context.encounteredError = true;
-                        return undefined; // TODO: GH#18217
-                    }
-                    context.approximateLength += 3;
-                    return ts.createKeywordTypeNode(125 /* AnyKeyword */);
-                }
-                if (!(context.flags & 536870912 /* NoTypeReduction */)) {
-                    type = getReducedType(type);
-                }
-                if (type.flags & 1 /* Any */) {
-                    context.approximateLength += 3;
-                    return ts.createKeywordTypeNode(125 /* AnyKeyword */);
-                }
-                if (type.flags & 2 /* Unknown */) {
-                    return ts.createKeywordTypeNode(148 /* UnknownKeyword */);
-                }
-                if (type.flags & 4 /* String */) {
-                    context.approximateLength += 6;
-                    return ts.createKeywordTypeNode(143 /* StringKeyword */);
-                }
-                if (type.flags & 8 /* Number */) {
-                    context.approximateLength += 6;
-                    return ts.createKeywordTypeNode(140 /* NumberKeyword */);
-                }
-                if (type.flags & 64 /* BigInt */) {
-                    context.approximateLength += 6;
-                    return ts.createKeywordTypeNode(151 /* BigIntKeyword */);
-                }
-                if (type.flags & 16 /* Boolean */) {
-                    context.approximateLength += 7;
-                    return ts.createKeywordTypeNode(128 /* BooleanKeyword */);
-                }
-                if (type.flags & 1024 /* EnumLiteral */ && !(type.flags & 1048576 /* Union */)) {
-                    var parentSymbol = getParentOfSymbol(type.symbol);
-                    var parentName = symbolToTypeNode(parentSymbol, context, 788968 /* Type */);
-                    var enumLiteralName = getDeclaredTypeOfSymbol(parentSymbol) === type
-                        ? parentName
-                        : appendReferenceToType(parentName, ts.createTypeReferenceNode(ts.symbolName(type.symbol), /*typeArguments*/ undefined));
-                    return enumLiteralName;
-                }
-                if (type.flags & 1056 /* EnumLike */) {
-                    return symbolToTypeNode(type.symbol, context, 788968 /* Type */);
-                }
-                if (type.flags & 128 /* StringLiteral */) {
-                    context.approximateLength += (type.value.length + 2);
-                    return ts.createLiteralTypeNode(ts.setEmitFlags(ts.createLiteral(type.value, !!(context.flags & 268435456 /* UseSingleQuotesForStringLiteralType */)), 16777216 /* NoAsciiEscaping */));
-                }
-                if (type.flags & 256 /* NumberLiteral */) {
-                    var value = type.value;
-                    context.approximateLength += ("" + value).length;
-                    return ts.createLiteralTypeNode(value < 0 ? ts.createPrefix(40 /* MinusToken */, ts.createLiteral(-value)) : ts.createLiteral(value));
-                }
-                if (type.flags & 2048 /* BigIntLiteral */) {
-                    context.approximateLength += (ts.pseudoBigIntToString(type.value).length) + 1;
-                    return ts.createLiteralTypeNode((ts.createLiteral(type.value)));
+        }
+        function bind(node) {
+            if (!node) {
+                return;
+            }
+            ts.setParent(node, parent);
+            var saveInStrictMode = inStrictMode;
+            // Even though in the AST the jsdoc @typedef node belongs to the current node,
+            // its symbol might be in the same scope with the current node's symbol. Consider:
+            //
+            //     /** @typedef {string | number} MyType */
+            //     function foo();
+            //
+            // Here the current node is "foo", which is a container, but the scope of "MyType" should
+            // not be inside "foo". Therefore we always bind @typedef before bind the parent node,
+            // and skip binding this tag later when binding all the other jsdoc tags.
+            // First we bind declaration nodes to a symbol if possible. We'll both create a symbol
+            // and then potentially add the symbol to an appropriate symbol table. Possible
+            // destination symbol tables are:
+            //
+            //  1) The 'exports' table of the current container's symbol.
+            //  2) The 'members' table of the current container's symbol.
+            //  3) The 'locals' table of the current container.
+            //
+            // However, not all symbols will end up in any of these tables. 'Anonymous' symbols
+            // (like TypeLiterals for example) will not be put in any table.
+            bindWorker(node);
+            // Then we recurse into the children of the node to bind them as well. For certain
+            // symbols we do specialized work when we recurse. For example, we'll keep track of
+            // the current 'container' node when it changes. This helps us know which symbol table
+            // a local should go into for example. Since terminal nodes are known not to have
+            // children, as an optimization we don't process those.
+            if (node.kind > 156 /* LastToken */) {
+                var saveParent = parent;
+                parent = node;
+                var containerFlags = getContainerFlags(node);
+                if (containerFlags === 0 /* None */) {
+                    bindChildren(node);
                 }
-                if (type.flags & 512 /* BooleanLiteral */) {
-                    context.approximateLength += type.intrinsicName.length;
-                    return type.intrinsicName === "true" ? ts.createTrue() : ts.createFalse();
+                else {
+                    bindContainer(node, containerFlags);
                 }
-                if (type.flags & 8192 /* UniqueESSymbol */) {
-                    if (!(context.flags & 1048576 /* AllowUniqueESSymbolType */)) {
-                        if (isValueSymbolAccessible(type.symbol, context.enclosingDeclaration)) {
-                            context.approximateLength += 6;
-                            return symbolToTypeNode(type.symbol, context, 111551 /* Value */);
-                        }
-                        if (context.tracker.reportInaccessibleUniqueSymbolError) {
-                            context.tracker.reportInaccessibleUniqueSymbolError();
-                        }
+                parent = saveParent;
+            }
+            else {
+                var saveParent = parent;
+                if (node.kind === 1 /* EndOfFileToken */)
+                    parent = node;
+                bindJSDoc(node);
+                parent = saveParent;
+            }
+            inStrictMode = saveInStrictMode;
+        }
+        function bindJSDoc(node) {
+            if (ts.hasJSDocNodes(node)) {
+                if (ts.isInJSFile(node)) {
+                    for (var _i = 0, _a = node.jsDoc; _i < _a.length; _i++) {
+                        var j = _a[_i];
+                        bind(j);
                     }
-                    context.approximateLength += 13;
-                    return ts.createTypeOperatorNode(147 /* UniqueKeyword */, ts.createKeywordTypeNode(144 /* SymbolKeyword */));
-                }
-                if (type.flags & 16384 /* Void */) {
-                    context.approximateLength += 4;
-                    return ts.createKeywordTypeNode(110 /* VoidKeyword */);
-                }
-                if (type.flags & 32768 /* Undefined */) {
-                    context.approximateLength += 9;
-                    return ts.createKeywordTypeNode(146 /* UndefinedKeyword */);
-                }
-                if (type.flags & 65536 /* Null */) {
-                    context.approximateLength += 4;
-                    return ts.createKeywordTypeNode(100 /* NullKeyword */);
-                }
-                if (type.flags & 131072 /* Never */) {
-                    context.approximateLength += 5;
-                    return ts.createKeywordTypeNode(137 /* NeverKeyword */);
                 }
-                if (type.flags & 4096 /* ESSymbol */) {
-                    context.approximateLength += 6;
-                    return ts.createKeywordTypeNode(144 /* SymbolKeyword */);
+                else {
+                    for (var _b = 0, _c = node.jsDoc; _b < _c.length; _b++) {
+                        var j = _c[_b];
+                        ts.setParent(j, node);
+                        ts.setParentRecursive(j, /*incremental*/ false);
+                    }
                 }
-                if (type.flags & 67108864 /* NonPrimitive */) {
-                    context.approximateLength += 6;
-                    return ts.createKeywordTypeNode(141 /* ObjectKeyword */);
+            }
+        }
+        function updateStrictModeStatementList(statements) {
+            if (!inStrictMode) {
+                for (var _i = 0, statements_3 = statements; _i < statements_3.length; _i++) {
+                    var statement = statements_3[_i];
+                    if (!ts.isPrologueDirective(statement)) {
+                        return;
+                    }
+                    if (isUseStrictPrologueDirective(statement)) {
+                        inStrictMode = true;
+                        return;
+                    }
                 }
-                if (isThisTypeParameter(type)) {
-                    if (context.flags & 4194304 /* InObjectTypeLiteral */) {
-                        if (!context.encounteredError && !(context.flags & 32768 /* AllowThisInObjectLiteral */)) {
-                            context.encounteredError = true;
-                        }
-                        if (context.tracker.reportInaccessibleThisError) {
-                            context.tracker.reportInaccessibleThisError();
+            }
+        }
+        /// Should be called only on prologue directives (isPrologueDirective(node) should be true)
+        function isUseStrictPrologueDirective(node) {
+            var nodeText = ts.getSourceTextOfNodeFromSourceFile(file, node.expression);
+            // Note: the node text must be exactly "use strict" or 'use strict'.  It is not ok for the
+            // string to contain unicode escapes (as per ES5).
+            return nodeText === '"use strict"' || nodeText === "'use strict'";
+        }
+        function bindWorker(node) {
+            switch (node.kind) {
+                /* Strict mode checks */
+                case 78 /* Identifier */:
+                    // for typedef type names with namespaces, bind the new jsdoc type symbol here
+                    // because it requires all containing namespaces to be in effect, namely the
+                    // current "blockScopeContainer" needs to be set to its immediate namespace parent.
+                    if (node.isInJSDocNamespace) {
+                        var parentNode = node.parent;
+                        while (parentNode && !ts.isJSDocTypeAlias(parentNode)) {
+                            parentNode = parentNode.parent;
                         }
+                        bindBlockScopedDeclaration(parentNode, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
+                        break;
                     }
-                    context.approximateLength += 4;
-                    return ts.createThis();
-                }
-                if (!inTypeAlias && type.aliasSymbol && (context.flags & 16384 /* UseAliasDefinedOutsideCurrentScope */ || isTypeSymbolAccessible(type.aliasSymbol, context.enclosingDeclaration))) {
-                    var typeArgumentNodes = mapToTypeNodes(type.aliasTypeArguments, context);
-                    if (isReservedMemberName(type.aliasSymbol.escapedName) && !(type.aliasSymbol.flags & 32 /* Class */))
-                        return ts.createTypeReferenceNode(ts.createIdentifier(""), typeArgumentNodes);
-                    return symbolToTypeNode(type.aliasSymbol, context, 788968 /* Type */, typeArgumentNodes);
-                }
-                var objectFlags = ts.getObjectFlags(type);
-                if (objectFlags & 4 /* Reference */) {
-                    ts.Debug.assert(!!(type.flags & 524288 /* Object */));
-                    return type.node ? visitAndTransformType(type, typeReferenceToTypeNode) : typeReferenceToTypeNode(type);
-                }
-                if (type.flags & 262144 /* TypeParameter */ || objectFlags & 3 /* ClassOrInterface */) {
-                    if (type.flags & 262144 /* TypeParameter */ && ts.contains(context.inferTypeParameters, type)) {
-                        context.approximateLength += (ts.symbolName(type.symbol).length + 6);
-                        return ts.createInferTypeNode(typeParameterToDeclarationWithConstraint(type, context, /*constraintNode*/ undefined));
+                // falls through
+                case 107 /* ThisKeyword */:
+                    if (currentFlow && (ts.isExpression(node) || parent.kind === 289 /* ShorthandPropertyAssignment */)) {
+                        node.flowNode = currentFlow;
                     }
-                    if (context.flags & 4 /* GenerateNamesForShadowedTypeParams */ &&
-                        type.flags & 262144 /* TypeParameter */ &&
-                        !isTypeSymbolAccessible(type.symbol, context.enclosingDeclaration)) {
-                        var name = typeParameterToName(type, context);
-                        context.approximateLength += ts.idText(name).length;
-                        return ts.createTypeReferenceNode(ts.createIdentifier(ts.idText(name)), /*typeArguments*/ undefined);
+                    return checkContextualIdentifier(node);
+                case 105 /* SuperKeyword */:
+                    node.flowNode = currentFlow;
+                    break;
+                case 79 /* PrivateIdentifier */:
+                    return checkPrivateIdentifier(node);
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    var expr = node;
+                    if (currentFlow && isNarrowableReference(expr)) {
+                        expr.flowNode = currentFlow;
                     }
-                    // Ignore constraint/default when creating a usage (as opposed to declaration) of a type parameter.
-                    return type.symbol
-                        ? symbolToTypeNode(type.symbol, context, 788968 /* Type */)
-                        : ts.createTypeReferenceNode(ts.createIdentifier("?"), /*typeArguments*/ undefined);
-                }
-                if (type.flags & (1048576 /* Union */ | 2097152 /* Intersection */)) {
-                    var types = type.flags & 1048576 /* Union */ ? formatUnionTypes(type.types) : type.types;
-                    if (ts.length(types) === 1) {
-                        return typeToTypeNodeHelper(types[0], context);
+                    if (ts.isSpecialPropertyDeclaration(expr)) {
+                        bindSpecialPropertyDeclaration(expr);
                     }
-                    var typeNodes = mapToTypeNodes(types, context, /*isBareList*/ true);
-                    if (typeNodes && typeNodes.length > 0) {
-                        var unionOrIntersectionTypeNode = ts.createUnionOrIntersectionTypeNode(type.flags & 1048576 /* Union */ ? 178 /* UnionType */ : 179 /* IntersectionType */, typeNodes);
-                        return unionOrIntersectionTypeNode;
+                    if (ts.isInJSFile(expr) &&
+                        file.commonJsModuleIndicator &&
+                        ts.isModuleExportsAccessExpression(expr) &&
+                        !lookupSymbolForName(blockScopeContainer, "module")) {
+                        declareSymbol(file.locals, /*parent*/ undefined, expr.expression, 1 /* FunctionScopedVariable */ | 134217728 /* ModuleExports */, 111550 /* FunctionScopedVariableExcludes */);
                     }
-                    else {
-                        if (!context.encounteredError && !(context.flags & 262144 /* AllowEmptyUnionOrIntersection */)) {
-                            context.encounteredError = true;
-                        }
-                        return undefined; // TODO: GH#18217
+                    break;
+                case 216 /* BinaryExpression */:
+                    var specialKind = ts.getAssignmentDeclarationKind(node);
+                    switch (specialKind) {
+                        case 1 /* ExportsProperty */:
+                            bindExportsPropertyAssignment(node);
+                            break;
+                        case 2 /* ModuleExports */:
+                            bindModuleExportsAssignment(node);
+                            break;
+                        case 3 /* PrototypeProperty */:
+                            bindPrototypePropertyAssignment(node.left, node);
+                            break;
+                        case 6 /* Prototype */:
+                            bindPrototypeAssignment(node);
+                            break;
+                        case 4 /* ThisProperty */:
+                            bindThisPropertyAssignment(node);
+                            break;
+                        case 5 /* Property */:
+                            var expression = node.left.expression;
+                            if (ts.isInJSFile(node) && ts.isIdentifier(expression)) {
+                                var symbol = lookupSymbolForName(blockScopeContainer, expression.escapedText);
+                                if (ts.isThisInitializedDeclaration(symbol === null || symbol === void 0 ? void 0 : symbol.valueDeclaration)) {
+                                    bindThisPropertyAssignment(node);
+                                    break;
+                                }
+                            }
+                            bindSpecialPropertyAssignment(node);
+                            break;
+                        case 0 /* None */:
+                            // Nothing to do
+                            break;
+                        default:
+                            ts.Debug.fail("Unknown binary expression special property assignment kind");
                     }
-                }
-                if (objectFlags & (16 /* Anonymous */ | 32 /* Mapped */)) {
-                    ts.Debug.assert(!!(type.flags & 524288 /* Object */));
-                    // The type is an object literal type.
-                    return createAnonymousTypeNode(type);
-                }
-                if (type.flags & 4194304 /* Index */) {
-                    var indexedType = type.type;
-                    context.approximateLength += 6;
-                    var indexTypeNode = typeToTypeNodeHelper(indexedType, context);
-                    return ts.createTypeOperatorNode(indexTypeNode);
-                }
-                if (type.flags & 8388608 /* IndexedAccess */) {
-                    var objectTypeNode = typeToTypeNodeHelper(type.objectType, context);
-                    var indexTypeNode = typeToTypeNodeHelper(type.indexType, context);
-                    context.approximateLength += 2;
-                    return ts.createIndexedAccessTypeNode(objectTypeNode, indexTypeNode);
-                }
-                if (type.flags & 16777216 /* Conditional */) {
-                    var checkTypeNode = typeToTypeNodeHelper(type.checkType, context);
-                    var saveInferTypeParameters = context.inferTypeParameters;
-                    context.inferTypeParameters = type.root.inferTypeParameters;
-                    var extendsTypeNode = typeToTypeNodeHelper(type.extendsType, context);
-                    context.inferTypeParameters = saveInferTypeParameters;
-                    var trueTypeNode = typeToTypeNodeHelper(getTrueTypeFromConditionalType(type), context);
-                    var falseTypeNode = typeToTypeNodeHelper(getFalseTypeFromConditionalType(type), context);
-                    context.approximateLength += 15;
-                    return ts.createConditionalTypeNode(checkTypeNode, extendsTypeNode, trueTypeNode, falseTypeNode);
-                }
-                if (type.flags & 33554432 /* Substitution */) {
-                    return typeToTypeNodeHelper(type.baseType, context);
-                }
-                return ts.Debug.fail("Should be unreachable.");
-                function createMappedTypeNodeFromType(type) {
-                    ts.Debug.assert(!!(type.flags & 524288 /* Object */));
-                    var readonlyToken = type.declaration.readonlyToken ? ts.createToken(type.declaration.readonlyToken.kind) : undefined;
-                    var questionToken = type.declaration.questionToken ? ts.createToken(type.declaration.questionToken.kind) : undefined;
-                    var appropriateConstraintTypeNode;
-                    if (isMappedTypeWithKeyofConstraintDeclaration(type)) {
-                        // We have a { [P in keyof T]: X }
-                        // We do this to ensure we retain the toplevel keyof-ness of the type which may be lost due to keyof distribution during `getConstraintTypeFromMappedType`
-                        appropriateConstraintTypeNode = ts.createTypeOperatorNode(typeToTypeNodeHelper(getModifiersTypeFromMappedType(type), context));
+                    return checkStrictModeBinaryExpression(node);
+                case 287 /* CatchClause */:
+                    return checkStrictModeCatchClause(node);
+                case 210 /* DeleteExpression */:
+                    return checkStrictModeDeleteExpression(node);
+                case 8 /* NumericLiteral */:
+                    return checkStrictModeNumericLiteral(node);
+                case 215 /* PostfixUnaryExpression */:
+                    return checkStrictModePostfixUnaryExpression(node);
+                case 214 /* PrefixUnaryExpression */:
+                    return checkStrictModePrefixUnaryExpression(node);
+                case 243 /* WithStatement */:
+                    return checkStrictModeWithStatement(node);
+                case 245 /* LabeledStatement */:
+                    return checkStrictModeLabeledStatement(node);
+                case 187 /* ThisType */:
+                    seenThisKeyword = true;
+                    return;
+                case 172 /* TypePredicate */:
+                    break; // Binding the children will handle everything
+                case 159 /* TypeParameter */:
+                    return bindTypeParameter(node);
+                case 160 /* Parameter */:
+                    return bindParameter(node);
+                case 249 /* VariableDeclaration */:
+                    return bindVariableDeclarationOrBindingElement(node);
+                case 198 /* BindingElement */:
+                    node.flowNode = currentFlow;
+                    return bindVariableDeclarationOrBindingElement(node);
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                    return bindPropertyWorker(node);
+                case 288 /* PropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
+                    return bindPropertyOrMethodOrAccessor(node, 4 /* Property */, 0 /* PropertyExcludes */);
+                case 291 /* EnumMember */:
+                    return bindPropertyOrMethodOrAccessor(node, 8 /* EnumMember */, 900095 /* EnumMemberExcludes */);
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 171 /* IndexSignature */:
+                    return declareSymbolAndAddToSymbolTable(node, 131072 /* Signature */, 0 /* None */);
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    // If this is an ObjectLiteralExpression method, then it sits in the same space
+                    // as other properties in the object literal.  So we use SymbolFlags.PropertyExcludes
+                    // so that it will conflict with any other object literal members with the same
+                    // name.
+                    return bindPropertyOrMethodOrAccessor(node, 8192 /* Method */ | (node.questionToken ? 16777216 /* Optional */ : 0 /* None */), ts.isObjectLiteralMethod(node) ? 0 /* PropertyExcludes */ : 103359 /* MethodExcludes */);
+                case 251 /* FunctionDeclaration */:
+                    return bindFunctionDeclaration(node);
+                case 166 /* Constructor */:
+                    return declareSymbolAndAddToSymbolTable(node, 16384 /* Constructor */, /*symbolExcludes:*/ 0 /* None */);
+                case 167 /* GetAccessor */:
+                    return bindPropertyOrMethodOrAccessor(node, 32768 /* GetAccessor */, 46015 /* GetAccessorExcludes */);
+                case 168 /* SetAccessor */:
+                    return bindPropertyOrMethodOrAccessor(node, 65536 /* SetAccessor */, 78783 /* SetAccessorExcludes */);
+                case 174 /* FunctionType */:
+                case 308 /* JSDocFunctionType */:
+                case 313 /* JSDocSignature */:
+                case 175 /* ConstructorType */:
+                    return bindFunctionOrConstructorType(node);
+                case 177 /* TypeLiteral */:
+                case 312 /* JSDocTypeLiteral */:
+                case 190 /* MappedType */:
+                    return bindAnonymousTypeWorker(node);
+                case 319 /* JSDocClassTag */:
+                    return bindJSDocClassTag(node);
+                case 200 /* ObjectLiteralExpression */:
+                    return bindObjectLiteralExpression(node);
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                    return bindFunctionExpression(node);
+                case 203 /* CallExpression */:
+                    var assignmentKind = ts.getAssignmentDeclarationKind(node);
+                    switch (assignmentKind) {
+                        case 7 /* ObjectDefinePropertyValue */:
+                            return bindObjectDefinePropertyAssignment(node);
+                        case 8 /* ObjectDefinePropertyExports */:
+                            return bindObjectDefinePropertyExport(node);
+                        case 9 /* ObjectDefinePrototypeProperty */:
+                            return bindObjectDefinePrototypeProperty(node);
+                        case 0 /* None */:
+                            break; // Nothing to do
+                        default:
+                            return ts.Debug.fail("Unknown call expression assignment declaration kind");
                     }
-                    else {
-                        appropriateConstraintTypeNode = typeToTypeNodeHelper(getConstraintTypeFromMappedType(type), context);
+                    if (ts.isInJSFile(node)) {
+                        bindCallExpression(node);
                     }
-                    var typeParameterNode = typeParameterToDeclarationWithConstraint(getTypeParameterFromMappedType(type), context, appropriateConstraintTypeNode);
-                    var templateTypeNode = typeToTypeNodeHelper(getTemplateTypeFromMappedType(type), context);
-                    var mappedTypeNode = ts.createMappedTypeNode(readonlyToken, typeParameterNode, questionToken, templateTypeNode);
-                    context.approximateLength += 10;
-                    return ts.setEmitFlags(mappedTypeNode, 1 /* SingleLine */);
+                    break;
+                // Members of classes, interfaces, and modules
+                case 221 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                    // All classes are automatically in strict mode in ES6.
+                    inStrictMode = true;
+                    return bindClassLikeDeclaration(node);
+                case 253 /* InterfaceDeclaration */:
+                    return bindBlockScopedDeclaration(node, 64 /* Interface */, 788872 /* InterfaceExcludes */);
+                case 254 /* TypeAliasDeclaration */:
+                    return bindBlockScopedDeclaration(node, 524288 /* TypeAlias */, 788968 /* TypeAliasExcludes */);
+                case 255 /* EnumDeclaration */:
+                    return bindEnumDeclaration(node);
+                case 256 /* ModuleDeclaration */:
+                    return bindModuleDeclaration(node);
+                // Jsx-attributes
+                case 281 /* JsxAttributes */:
+                    return bindJsxAttributes(node);
+                case 280 /* JsxAttribute */:
+                    return bindJsxAttribute(node, 4 /* Property */, 0 /* PropertyExcludes */);
+                // Imports and exports
+                case 260 /* ImportEqualsDeclaration */:
+                case 263 /* NamespaceImport */:
+                case 265 /* ImportSpecifier */:
+                case 270 /* ExportSpecifier */:
+                    return declareSymbolAndAddToSymbolTable(node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
+                case 259 /* NamespaceExportDeclaration */:
+                    return bindNamespaceExportDeclaration(node);
+                case 262 /* ImportClause */:
+                    return bindImportClause(node);
+                case 267 /* ExportDeclaration */:
+                    return bindExportDeclaration(node);
+                case 266 /* ExportAssignment */:
+                    return bindExportAssignment(node);
+                case 297 /* SourceFile */:
+                    updateStrictModeStatementList(node.statements);
+                    return bindSourceFileIfExternalModule();
+                case 230 /* Block */:
+                    if (!ts.isFunctionLike(node.parent)) {
+                        return;
+                    }
+                // falls through
+                case 257 /* ModuleBlock */:
+                    return updateStrictModeStatementList(node.statements);
+                case 326 /* JSDocParameterTag */:
+                    if (node.parent.kind === 313 /* JSDocSignature */) {
+                        return bindParameter(node);
+                    }
+                    if (node.parent.kind !== 312 /* JSDocTypeLiteral */) {
+                        break;
+                    }
+                // falls through
+                case 333 /* JSDocPropertyTag */:
+                    var propTag = node;
+                    var flags = propTag.isBracketed || propTag.typeExpression && propTag.typeExpression.type.kind === 307 /* JSDocOptionalType */ ?
+                        4 /* Property */ | 16777216 /* Optional */ :
+                        4 /* Property */;
+                    return declareSymbolAndAddToSymbolTable(propTag, flags, 0 /* PropertyExcludes */);
+                case 331 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 325 /* JSDocEnumTag */:
+                    return (delayedTypeAliases || (delayedTypeAliases = [])).push(node);
+            }
+        }
+        function bindPropertyWorker(node) {
+            return bindPropertyOrMethodOrAccessor(node, 4 /* Property */ | (node.questionToken ? 16777216 /* Optional */ : 0 /* None */), 0 /* PropertyExcludes */);
+        }
+        function bindAnonymousTypeWorker(node) {
+            return bindAnonymousDeclaration(node, 2048 /* TypeLiteral */, "__type" /* Type */);
+        }
+        function bindSourceFileIfExternalModule() {
+            setExportContextFlag(file);
+            if (ts.isExternalModule(file)) {
+                bindSourceFileAsExternalModule();
+            }
+            else if (ts.isJsonSourceFile(file)) {
+                bindSourceFileAsExternalModule();
+                // Create symbol equivalent for the module.exports = {}
+                var originalSymbol = file.symbol;
+                declareSymbol(file.symbol.exports, file.symbol, file, 4 /* Property */, 67108863 /* All */);
+                file.symbol = originalSymbol;
+            }
+        }
+        function bindSourceFileAsExternalModule() {
+            bindAnonymousDeclaration(file, 512 /* ValueModule */, "\"" + ts.removeFileExtension(file.fileName) + "\"");
+        }
+        function bindExportAssignment(node) {
+            if (!container.symbol || !container.symbol.exports) {
+                // Export assignment in some sort of block construct
+                bindAnonymousDeclaration(node, 2097152 /* Alias */, getDeclarationName(node));
+            }
+            else {
+                var flags = ts.exportAssignmentIsAlias(node)
+                    // An export default clause with an EntityNameExpression or a class expression exports all meanings of that identifier or expression;
+                    ? 2097152 /* Alias */
+                    // An export default clause with any other expression exports a value
+                    : 4 /* Property */;
+                // If there is an `export default x;` alias declaration, can't `export default` anything else.
+                // (In contrast, you can still have `export default function f() {}` and `export default interface I {}`.)
+                var symbol = declareSymbol(container.symbol.exports, container.symbol, node, flags, 67108863 /* All */);
+                if (node.isExportEquals) {
+                    // Will be an error later, since the module already has other exports. Just make sure this has a valueDeclaration set.
+                    ts.setValueDeclaration(symbol, node);
                 }
-                function createAnonymousTypeNode(type) {
-                    var typeId = "" + type.id;
-                    var symbol = type.symbol;
-                    if (symbol) {
-                        if (isJSConstructor(symbol.valueDeclaration)) {
-                            // Instance and static types share the same symbol; only add 'typeof' for the static side.
-                            var isInstanceType = type === getDeclaredTypeOfClassOrInterface(symbol) ? 788968 /* Type */ : 111551 /* Value */;
-                            return symbolToTypeNode(symbol, context, isInstanceType);
-                        }
-                        // Always use 'typeof T' for type of class, enum, and module objects
-                        else if (symbol.flags & 32 /* Class */ && !getBaseTypeVariableOfClass(symbol) && !(symbol.valueDeclaration.kind === 214 /* ClassExpression */ && context.flags & 2048 /* WriteClassExpressionAsTypeLiteral */) ||
-                            symbol.flags & (384 /* Enum */ | 512 /* ValueModule */) ||
-                            shouldWriteTypeOfFunctionSymbol()) {
-                            return symbolToTypeNode(symbol, context, 111551 /* Value */);
+            }
+        }
+        function bindNamespaceExportDeclaration(node) {
+            if (node.modifiers && node.modifiers.length) {
+                file.bindDiagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Modifiers_cannot_appear_here));
+            }
+            var diag = !ts.isSourceFile(node.parent) ? ts.Diagnostics.Global_module_exports_may_only_appear_at_top_level
+                : !ts.isExternalModule(node.parent) ? ts.Diagnostics.Global_module_exports_may_only_appear_in_module_files
+                    : !node.parent.isDeclarationFile ? ts.Diagnostics.Global_module_exports_may_only_appear_in_declaration_files
+                        : undefined;
+            if (diag) {
+                file.bindDiagnostics.push(createDiagnosticForNode(node, diag));
+            }
+            else {
+                file.symbol.globalExports = file.symbol.globalExports || ts.createSymbolTable();
+                declareSymbol(file.symbol.globalExports, file.symbol, node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
+            }
+        }
+        function bindExportDeclaration(node) {
+            if (!container.symbol || !container.symbol.exports) {
+                // Export * in some sort of block construct
+                bindAnonymousDeclaration(node, 8388608 /* ExportStar */, getDeclarationName(node));
+            }
+            else if (!node.exportClause) {
+                // All export * declarations are collected in an __export symbol
+                declareSymbol(container.symbol.exports, container.symbol, node, 8388608 /* ExportStar */, 0 /* None */);
+            }
+            else if (ts.isNamespaceExport(node.exportClause)) {
+                // declareSymbol walks up parents to find name text, parent _must_ be set
+                // but won't be set by the normal binder walk until `bindChildren` later on.
+                ts.setParent(node.exportClause, node);
+                declareSymbol(container.symbol.exports, container.symbol, node.exportClause, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
+            }
+        }
+        function bindImportClause(node) {
+            if (node.name) {
+                declareSymbolAndAddToSymbolTable(node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
+            }
+        }
+        function setCommonJsModuleIndicator(node) {
+            if (file.externalModuleIndicator) {
+                return false;
+            }
+            if (!file.commonJsModuleIndicator) {
+                file.commonJsModuleIndicator = node;
+                bindSourceFileAsExternalModule();
+            }
+            return true;
+        }
+        function bindObjectDefinePropertyExport(node) {
+            if (!setCommonJsModuleIndicator(node)) {
+                return;
+            }
+            var symbol = forEachIdentifierInEntityName(node.arguments[0], /*parent*/ undefined, function (id, symbol) {
+                if (symbol) {
+                    addDeclarationToSymbol(symbol, id, 1536 /* Module */ | 67108864 /* Assignment */);
+                }
+                return symbol;
+            });
+            if (symbol) {
+                var flags = 4 /* Property */ | 1048576 /* ExportValue */;
+                declareSymbol(symbol.exports, symbol, node, flags, 0 /* None */);
+            }
+        }
+        function bindExportsPropertyAssignment(node) {
+            // When we create a property via 'exports.foo = bar', the 'exports.foo' property access
+            // expression is the declaration
+            if (!setCommonJsModuleIndicator(node)) {
+                return;
+            }
+            var symbol = forEachIdentifierInEntityName(node.left.expression, /*parent*/ undefined, function (id, symbol) {
+                if (symbol) {
+                    addDeclarationToSymbol(symbol, id, 1536 /* Module */ | 67108864 /* Assignment */);
+                }
+                return symbol;
+            });
+            if (symbol) {
+                var isAlias = ts.isAliasableExpression(node.right) && (ts.isExportsIdentifier(node.left.expression) || ts.isModuleExportsAccessExpression(node.left.expression));
+                var flags = isAlias ? 2097152 /* Alias */ : 4 /* Property */ | 1048576 /* ExportValue */;
+                ts.setParent(node.left, node);
+                declareSymbol(symbol.exports, symbol, node.left, flags, 0 /* None */);
+            }
+        }
+        function bindModuleExportsAssignment(node) {
+            // A common practice in node modules is to set 'export = module.exports = {}', this ensures that 'exports'
+            // is still pointing to 'module.exports'.
+            // We do not want to consider this as 'export=' since a module can have only one of these.
+            // Similarly we do not want to treat 'module.exports = exports' as an 'export='.
+            if (!setCommonJsModuleIndicator(node)) {
+                return;
+            }
+            var assignedExpression = ts.getRightMostAssignedExpression(node.right);
+            if (ts.isEmptyObjectLiteral(assignedExpression) || container === file && isExportsOrModuleExportsOrAlias(file, assignedExpression)) {
+                return;
+            }
+            if (ts.isObjectLiteralExpression(assignedExpression) && ts.every(assignedExpression.properties, ts.isShorthandPropertyAssignment)) {
+                ts.forEach(assignedExpression.properties, bindExportAssignedObjectMemberAlias);
+                return;
+            }
+            // 'module.exports = expr' assignment
+            var flags = ts.exportAssignmentIsAlias(node)
+                ? 2097152 /* Alias */
+                : 4 /* Property */ | 1048576 /* ExportValue */ | 512 /* ValueModule */;
+            var symbol = declareSymbol(file.symbol.exports, file.symbol, node, flags | 67108864 /* Assignment */, 0 /* None */);
+            ts.setValueDeclaration(symbol, node);
+        }
+        function bindExportAssignedObjectMemberAlias(node) {
+            declareSymbol(file.symbol.exports, file.symbol, node, 2097152 /* Alias */ | 67108864 /* Assignment */, 0 /* None */);
+        }
+        function bindThisPropertyAssignment(node) {
+            ts.Debug.assert(ts.isInJSFile(node));
+            // private identifiers *must* be declared (even in JS files)
+            var hasPrivateIdentifier = (ts.isBinaryExpression(node) && ts.isPropertyAccessExpression(node.left) && ts.isPrivateIdentifier(node.left.name))
+                || (ts.isPropertyAccessExpression(node) && ts.isPrivateIdentifier(node.name));
+            if (hasPrivateIdentifier) {
+                return;
+            }
+            var thisContainer = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
+            switch (thisContainer.kind) {
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                    var constructorSymbol = thisContainer.symbol;
+                    // For `f.prototype.m = function() { this.x = 0; }`, `this.x = 0` should modify `f`'s members, not the function expression.
+                    if (ts.isBinaryExpression(thisContainer.parent) && thisContainer.parent.operatorToken.kind === 62 /* EqualsToken */) {
+                        var l = thisContainer.parent.left;
+                        if (ts.isBindableStaticAccessExpression(l) && ts.isPrototypeAccess(l.expression)) {
+                            constructorSymbol = lookupSymbolForPropertyAccess(l.expression.expression, thisParentContainer);
                         }
-                        else if (context.visitedTypes && context.visitedTypes.has(typeId)) {
-                            // If type is an anonymous type literal in a type alias declaration, use type alias name
-                            var typeAlias = getTypeAliasForTypeLiteral(type);
-                            if (typeAlias) {
-                                // The specified symbol flags need to be reinterpreted as type flags
-                                return symbolToTypeNode(typeAlias, context, 788968 /* Type */);
-                            }
-                            else {
-                                return createElidedInformationPlaceholder(context);
-                            }
+                    }
+                    if (constructorSymbol && constructorSymbol.valueDeclaration) {
+                        // Declare a 'member' if the container is an ES5 class or ES6 constructor
+                        constructorSymbol.members = constructorSymbol.members || ts.createSymbolTable();
+                        // It's acceptable for multiple 'this' assignments of the same identifier to occur
+                        if (ts.hasDynamicName(node)) {
+                            bindDynamicallyNamedThisPropertyAssignment(node, constructorSymbol);
                         }
                         else {
-                            return visitAndTransformType(type, createTypeNodeFromObjectType);
+                            declareSymbol(constructorSymbol.members, constructorSymbol, node, 4 /* Property */ | 67108864 /* Assignment */, 0 /* PropertyExcludes */ & ~4 /* Property */);
                         }
+                        addDeclarationToSymbol(constructorSymbol, constructorSymbol.valueDeclaration, 32 /* Class */);
                     }
-                    else {
-                        // Anonymous types without a symbol are never circular.
-                        return createTypeNodeFromObjectType(type);
-                    }
-                    function shouldWriteTypeOfFunctionSymbol() {
-                        var isStaticMethodSymbol = !!(symbol.flags & 8192 /* Method */) && // typeof static method
-                            ts.some(symbol.declarations, function (declaration) { return ts.hasModifier(declaration, 32 /* Static */); });
-                        var isNonLocalFunctionSymbol = !!(symbol.flags & 16 /* Function */) &&
-                            (symbol.parent || // is exported function symbol
-                                ts.forEach(symbol.declarations, function (declaration) {
-                                    return declaration.parent.kind === 290 /* SourceFile */ || declaration.parent.kind === 250 /* ModuleBlock */;
-                                }));
-                        if (isStaticMethodSymbol || isNonLocalFunctionSymbol) {
-                            // typeof is allowed only for static/non local functions
-                            return (!!(context.flags & 4096 /* UseTypeOfFunction */) || (context.visitedTypes && context.visitedTypes.has(typeId))) && // it is type of the symbol uses itself recursively
-                                (!(context.flags & 8 /* UseStructuralFallback */) || isValueSymbolAccessible(symbol, context.enclosingDeclaration)); // And the build is going to succeed without visibility error or there is no structural fallback allowed
-                        }
+                    break;
+                case 166 /* Constructor */:
+                case 163 /* PropertyDeclaration */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    // this.foo assignment in a JavaScript class
+                    // Bind this property to the containing class
+                    var containingClass = thisContainer.parent;
+                    var symbolTable = ts.hasSyntacticModifier(thisContainer, 32 /* Static */) ? containingClass.symbol.exports : containingClass.symbol.members;
+                    if (ts.hasDynamicName(node)) {
+                        bindDynamicallyNamedThisPropertyAssignment(node, containingClass.symbol);
                     }
-                }
-                function visitAndTransformType(type, transform) {
-                    var typeId = "" + type.id;
-                    var isConstructorObject = ts.getObjectFlags(type) & 16 /* Anonymous */ && type.symbol && type.symbol.flags & 32 /* Class */;
-                    var id = ts.getObjectFlags(type) & 4 /* Reference */ && type.node ? "N" + getNodeId(type.node) :
-                        type.symbol ? (isConstructorObject ? "+" : "") + getSymbolId(type.symbol) :
-                            undefined;
-                    // Since instantiations of the same anonymous type have the same symbol, tracking symbols instead
-                    // of types allows us to catch circular references to instantiations of the same anonymous type
-                    if (!context.visitedTypes) {
-                        context.visitedTypes = ts.createMap();
+                    else {
+                        declareSymbol(symbolTable, containingClass.symbol, node, 4 /* Property */ | 67108864 /* Assignment */, 0 /* None */, /*isReplaceableByMethod*/ true);
                     }
-                    if (id && !context.symbolDepth) {
-                        context.symbolDepth = ts.createMap();
+                    break;
+                case 297 /* SourceFile */:
+                    // this.property = assignment in a source file -- declare symbol in exports for a module, in locals for a script
+                    if (ts.hasDynamicName(node)) {
+                        break;
                     }
-                    var depth;
-                    if (id) {
-                        depth = context.symbolDepth.get(id) || 0;
-                        if (depth > 10) {
-                            return createElidedInformationPlaceholder(context);
-                        }
-                        context.symbolDepth.set(id, depth + 1);
+                    else if (thisContainer.commonJsModuleIndicator) {
+                        declareSymbol(thisContainer.symbol.exports, thisContainer.symbol, node, 4 /* Property */ | 1048576 /* ExportValue */, 0 /* None */);
                     }
-                    context.visitedTypes.set(typeId, true);
-                    var result = transform(type);
-                    context.visitedTypes.delete(typeId);
-                    if (id) {
-                        context.symbolDepth.set(id, depth);
+                    else {
+                        declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111550 /* FunctionScopedVariableExcludes */);
                     }
-                    return result;
+                    break;
+                default:
+                    ts.Debug.failBadSyntaxKind(thisContainer);
+            }
+        }
+        function bindDynamicallyNamedThisPropertyAssignment(node, symbol) {
+            bindAnonymousDeclaration(node, 4 /* Property */, "__computed" /* Computed */);
+            addLateBoundAssignmentDeclarationToSymbol(node, symbol);
+        }
+        function addLateBoundAssignmentDeclarationToSymbol(node, symbol) {
+            if (symbol) {
+                (symbol.assignmentDeclarationMembers || (symbol.assignmentDeclarationMembers = new ts.Map())).set(ts.getNodeId(node), node);
+            }
+        }
+        function bindSpecialPropertyDeclaration(node) {
+            if (node.expression.kind === 107 /* ThisKeyword */) {
+                bindThisPropertyAssignment(node);
+            }
+            else if (ts.isBindableStaticAccessExpression(node) && node.parent.parent.kind === 297 /* SourceFile */) {
+                if (ts.isPrototypeAccess(node.expression)) {
+                    bindPrototypePropertyAssignment(node, node.parent);
                 }
-                function createTypeNodeFromObjectType(type) {
-                    if (isGenericMappedType(type)) {
-                        return createMappedTypeNodeFromType(type);
-                    }
-                    var resolved = resolveStructuredTypeMembers(type);
-                    if (!resolved.properties.length && !resolved.stringIndexInfo && !resolved.numberIndexInfo) {
-                        if (!resolved.callSignatures.length && !resolved.constructSignatures.length) {
-                            context.approximateLength += 2;
-                            return ts.setEmitFlags(ts.createTypeLiteralNode(/*members*/ undefined), 1 /* SingleLine */);
-                        }
-                        if (resolved.callSignatures.length === 1 && !resolved.constructSignatures.length) {
-                            var signature = resolved.callSignatures[0];
-                            var signatureNode = signatureToSignatureDeclarationHelper(signature, 170 /* FunctionType */, context);
-                            return signatureNode;
-                        }
-                        if (resolved.constructSignatures.length === 1 && !resolved.callSignatures.length) {
-                            var signature = resolved.constructSignatures[0];
-                            var signatureNode = signatureToSignatureDeclarationHelper(signature, 171 /* ConstructorType */, context);
-                            return signatureNode;
-                        }
-                    }
-                    var savedFlags = context.flags;
-                    context.flags |= 4194304 /* InObjectTypeLiteral */;
-                    var members = createTypeNodesFromResolvedType(resolved);
-                    context.flags = savedFlags;
-                    var typeLiteralNode = ts.createTypeLiteralNode(members);
-                    context.approximateLength += 2;
-                    return ts.setEmitFlags(typeLiteralNode, (context.flags & 1024 /* MultilineObjectLiterals */) ? 0 : 1 /* SingleLine */);
+                else {
+                    bindStaticPropertyAssignment(node);
                 }
-                function typeReferenceToTypeNode(type) {
-                    var typeArguments = getTypeArguments(type);
-                    if (type.target === globalArrayType || type.target === globalReadonlyArrayType) {
-                        if (context.flags & 2 /* WriteArrayAsGenericType */) {
-                            var typeArgumentNode = typeToTypeNodeHelper(typeArguments[0], context);
-                            return ts.createTypeReferenceNode(type.target === globalArrayType ? "Array" : "ReadonlyArray", [typeArgumentNode]);
-                        }
-                        var elementType = typeToTypeNodeHelper(typeArguments[0], context);
-                        var arrayType = ts.createArrayTypeNode(elementType);
-                        return type.target === globalArrayType ? arrayType : ts.createTypeOperatorNode(138 /* ReadonlyKeyword */, arrayType);
-                    }
-                    else if (type.target.objectFlags & 8 /* Tuple */) {
-                        if (typeArguments.length > 0) {
-                            var arity = getTypeReferenceArity(type);
-                            var tupleConstituentNodes = mapToTypeNodes(typeArguments.slice(0, arity), context);
-                            var hasRestElement = type.target.hasRestElement;
-                            if (tupleConstituentNodes) {
-                                for (var i = type.target.minLength; i < Math.min(arity, tupleConstituentNodes.length); i++) {
-                                    tupleConstituentNodes[i] = hasRestElement && i === arity - 1 ?
-                                        ts.createRestTypeNode(ts.createArrayTypeNode(tupleConstituentNodes[i])) :
-                                        ts.createOptionalTypeNode(tupleConstituentNodes[i]);
-                                }
-                                var tupleTypeNode = ts.createTupleTypeNode(tupleConstituentNodes);
-                                return type.target.readonly ? ts.createTypeOperatorNode(138 /* ReadonlyKeyword */, tupleTypeNode) : tupleTypeNode;
-                            }
-                        }
-                        if (context.encounteredError || (context.flags & 524288 /* AllowEmptyTuple */)) {
-                            var tupleTypeNode = ts.createTupleTypeNode([]);
-                            return type.target.readonly ? ts.createTypeOperatorNode(138 /* ReadonlyKeyword */, tupleTypeNode) : tupleTypeNode;
-                        }
-                        context.encounteredError = true;
-                        return undefined; // TODO: GH#18217
-                    }
-                    else if (context.flags & 2048 /* WriteClassExpressionAsTypeLiteral */ &&
-                        type.symbol.valueDeclaration &&
-                        ts.isClassLike(type.symbol.valueDeclaration) &&
-                        !isValueSymbolAccessible(type.symbol, context.enclosingDeclaration)) {
-                        return createAnonymousTypeNode(type);
+            }
+        }
+        /** For `x.prototype = { p, ... }`, declare members p,... if `x` is function/class/{}, or not declared. */
+        function bindPrototypeAssignment(node) {
+            ts.setParent(node.left, node);
+            ts.setParent(node.right, node);
+            bindPropertyAssignment(node.left.expression, node.left, /*isPrototypeProperty*/ false, /*containerIsClass*/ true);
+        }
+        function bindObjectDefinePrototypeProperty(node) {
+            var namespaceSymbol = lookupSymbolForPropertyAccess(node.arguments[0].expression);
+            if (namespaceSymbol && namespaceSymbol.valueDeclaration) {
+                // Ensure the namespace symbol becomes class-like
+                addDeclarationToSymbol(namespaceSymbol, namespaceSymbol.valueDeclaration, 32 /* Class */);
+            }
+            bindPotentiallyNewExpandoMemberToNamespace(node, namespaceSymbol, /*isPrototypeProperty*/ true);
+        }
+        /**
+         * For `x.prototype.y = z`, declare a member `y` on `x` if `x` is a function or class, or not declared.
+         * Note that jsdoc preceding an ExpressionStatement like `x.prototype.y;` is also treated as a declaration.
+         */
+        function bindPrototypePropertyAssignment(lhs, parent) {
+            // Look up the function in the local scope, since prototype assignments should
+            // follow the function declaration
+            var classPrototype = lhs.expression;
+            var constructorFunction = classPrototype.expression;
+            // Fix up parent pointers since we're going to use these nodes before we bind into them
+            ts.setParent(constructorFunction, classPrototype);
+            ts.setParent(classPrototype, lhs);
+            ts.setParent(lhs, parent);
+            bindPropertyAssignment(constructorFunction, lhs, /*isPrototypeProperty*/ true, /*containerIsClass*/ true);
+        }
+        function bindObjectDefinePropertyAssignment(node) {
+            var namespaceSymbol = lookupSymbolForPropertyAccess(node.arguments[0]);
+            var isToplevel = node.parent.parent.kind === 297 /* SourceFile */;
+            namespaceSymbol = bindPotentiallyMissingNamespaces(namespaceSymbol, node.arguments[0], isToplevel, /*isPrototypeProperty*/ false, /*containerIsClass*/ false);
+            bindPotentiallyNewExpandoMemberToNamespace(node, namespaceSymbol, /*isPrototypeProperty*/ false);
+        }
+        function bindSpecialPropertyAssignment(node) {
+            var _a;
+            // Class declarations in Typescript do not allow property declarations
+            var parentSymbol = lookupSymbolForPropertyAccess(node.left.expression, container) || lookupSymbolForPropertyAccess(node.left.expression, blockScopeContainer);
+            if (!ts.isInJSFile(node) && !ts.isFunctionSymbol(parentSymbol)) {
+                return;
+            }
+            var rootExpr = ts.getLeftmostAccessExpression(node.left);
+            if (ts.isIdentifier(rootExpr) && ((_a = lookupSymbolForName(container, rootExpr.escapedText)) === null || _a === void 0 ? void 0 : _a.flags) & 2097152 /* Alias */) {
+                return;
+            }
+            // Fix up parent pointers since we're going to use these nodes before we bind into them
+            ts.setParent(node.left, node);
+            ts.setParent(node.right, node);
+            if (ts.isIdentifier(node.left.expression) && container === file && isExportsOrModuleExportsOrAlias(file, node.left.expression)) {
+                // This can be an alias for the 'exports' or 'module.exports' names, e.g.
+                //    var util = module.exports;
+                //    util.property = function ...
+                bindExportsPropertyAssignment(node);
+            }
+            else if (ts.hasDynamicName(node)) {
+                bindAnonymousDeclaration(node, 4 /* Property */ | 67108864 /* Assignment */, "__computed" /* Computed */);
+                var sym = bindPotentiallyMissingNamespaces(parentSymbol, node.left.expression, isTopLevelNamespaceAssignment(node.left), /*isPrototype*/ false, /*containerIsClass*/ false);
+                addLateBoundAssignmentDeclarationToSymbol(node, sym);
+            }
+            else {
+                bindStaticPropertyAssignment(ts.cast(node.left, ts.isBindableStaticNameExpression));
+            }
+        }
+        /**
+         * For nodes like `x.y = z`, declare a member 'y' on 'x' if x is a function (or IIFE) or class or {}, or not declared.
+         * Also works for expression statements preceded by JSDoc, like / ** @type number * / x.y;
+         */
+        function bindStaticPropertyAssignment(node) {
+            ts.Debug.assert(!ts.isIdentifier(node));
+            ts.setParent(node.expression, node);
+            bindPropertyAssignment(node.expression, node, /*isPrototypeProperty*/ false, /*containerIsClass*/ false);
+        }
+        function bindPotentiallyMissingNamespaces(namespaceSymbol, entityName, isToplevel, isPrototypeProperty, containerIsClass) {
+            if ((namespaceSymbol === null || namespaceSymbol === void 0 ? void 0 : namespaceSymbol.flags) & 2097152 /* Alias */) {
+                return namespaceSymbol;
+            }
+            if (isToplevel && !isPrototypeProperty) {
+                // make symbols or add declarations for intermediate containers
+                var flags_2 = 1536 /* Module */ | 67108864 /* Assignment */;
+                var excludeFlags_1 = 110735 /* ValueModuleExcludes */ & ~67108864 /* Assignment */;
+                namespaceSymbol = forEachIdentifierInEntityName(entityName, namespaceSymbol, function (id, symbol, parent) {
+                    if (symbol) {
+                        addDeclarationToSymbol(symbol, id, flags_2);
+                        return symbol;
                     }
                     else {
-                        var outerTypeParameters = type.target.outerTypeParameters;
-                        var i = 0;
-                        var resultType = void 0;
-                        if (outerTypeParameters) {
-                            var length_2 = outerTypeParameters.length;
-                            while (i < length_2) {
-                                // Find group of type arguments for type parameters with the same declaring container.
-                                var start = i;
-                                var parent = getParentSymbolOfTypeParameter(outerTypeParameters[i]);
-                                do {
-                                    i++;
-                                } while (i < length_2 && getParentSymbolOfTypeParameter(outerTypeParameters[i]) === parent);
-                                // When type parameters are their own type arguments for the whole group (i.e. we have
-                                // the default outer type arguments), we don't show the group.
-                                if (!ts.rangeEquals(outerTypeParameters, typeArguments, start, i)) {
-                                    var typeArgumentSlice = mapToTypeNodes(typeArguments.slice(start, i), context);
-                                    var flags_2 = context.flags;
-                                    context.flags |= 16 /* ForbidIndexedAccessSymbolReferences */;
-                                    var ref = symbolToTypeNode(parent, context, 788968 /* Type */, typeArgumentSlice);
-                                    context.flags = flags_2;
-                                    resultType = !resultType ? ref : appendReferenceToType(resultType, ref);
-                                }
-                            }
-                        }
-                        var typeArgumentNodes = void 0;
-                        if (typeArguments.length > 0) {
-                            var typeParameterCount = (type.target.typeParameters || ts.emptyArray).length;
-                            typeArgumentNodes = mapToTypeNodes(typeArguments.slice(i, typeParameterCount), context);
-                        }
-                        var flags = context.flags;
-                        context.flags |= 16 /* ForbidIndexedAccessSymbolReferences */;
-                        var finalRef = symbolToTypeNode(type.symbol, context, 788968 /* Type */, typeArgumentNodes);
-                        context.flags = flags;
-                        return !resultType ? finalRef : appendReferenceToType(resultType, finalRef);
+                        var table = parent ? parent.exports :
+                            file.jsGlobalAugmentations || (file.jsGlobalAugmentations = ts.createSymbolTable());
+                        return declareSymbol(table, parent, id, flags_2, excludeFlags_1);
                     }
+                });
+            }
+            if (containerIsClass && namespaceSymbol && namespaceSymbol.valueDeclaration) {
+                addDeclarationToSymbol(namespaceSymbol, namespaceSymbol.valueDeclaration, 32 /* Class */);
+            }
+            return namespaceSymbol;
+        }
+        function bindPotentiallyNewExpandoMemberToNamespace(declaration, namespaceSymbol, isPrototypeProperty) {
+            if (!namespaceSymbol || !isExpandoSymbol(namespaceSymbol)) {
+                return;
+            }
+            // Set up the members collection if it doesn't exist already
+            var symbolTable = isPrototypeProperty ?
+                (namespaceSymbol.members || (namespaceSymbol.members = ts.createSymbolTable())) :
+                (namespaceSymbol.exports || (namespaceSymbol.exports = ts.createSymbolTable()));
+            var includes = 0 /* None */;
+            var excludes = 0 /* None */;
+            // Method-like
+            if (ts.isFunctionLikeDeclaration(ts.getAssignedExpandoInitializer(declaration))) {
+                includes = 8192 /* Method */;
+                excludes = 103359 /* MethodExcludes */;
+            }
+            // Maybe accessor-like
+            else if (ts.isCallExpression(declaration) && ts.isBindableObjectDefinePropertyCall(declaration)) {
+                if (ts.some(declaration.arguments[2].properties, function (p) {
+                    var id = ts.getNameOfDeclaration(p);
+                    return !!id && ts.isIdentifier(id) && ts.idText(id) === "set";
+                })) {
+                    // We mix in `SymbolFLags.Property` so in the checker `getTypeOfVariableParameterOrProperty` is used for this
+                    // symbol, instead of `getTypeOfAccessor` (which will assert as there is no real accessor declaration)
+                    includes |= 65536 /* SetAccessor */ | 4 /* Property */;
+                    excludes |= 78783 /* SetAccessorExcludes */;
                 }
-                function appendReferenceToType(root, ref) {
-                    if (ts.isImportTypeNode(root)) {
-                        // first shift type arguments
-                        var innerParams = root.typeArguments;
-                        if (root.qualifier) {
-                            (ts.isIdentifier(root.qualifier) ? root.qualifier : root.qualifier.right).typeArguments = innerParams;
-                        }
-                        root.typeArguments = ref.typeArguments;
-                        // then move qualifiers
-                        var ids = getAccessStack(ref);
-                        for (var _i = 0, ids_1 = ids; _i < ids_1.length; _i++) {
-                            var id = ids_1[_i];
-                            root.qualifier = root.qualifier ? ts.createQualifiedName(root.qualifier, id) : id;
-                        }
-                        return root;
-                    }
-                    else {
-                        // first shift type arguments
-                        var innerParams = root.typeArguments;
-                        (ts.isIdentifier(root.typeName) ? root.typeName : root.typeName.right).typeArguments = innerParams;
-                        root.typeArguments = ref.typeArguments;
-                        // then move qualifiers
-                        var ids = getAccessStack(ref);
-                        for (var _a = 0, ids_2 = ids; _a < ids_2.length; _a++) {
-                            var id = ids_2[_a];
-                            root.typeName = ts.createQualifiedName(root.typeName, id);
-                        }
-                        return root;
-                    }
+                if (ts.some(declaration.arguments[2].properties, function (p) {
+                    var id = ts.getNameOfDeclaration(p);
+                    return !!id && ts.isIdentifier(id) && ts.idText(id) === "get";
+                })) {
+                    includes |= 32768 /* GetAccessor */ | 4 /* Property */;
+                    excludes |= 46015 /* GetAccessorExcludes */;
                 }
-                function getAccessStack(ref) {
-                    var state = ref.typeName;
-                    var ids = [];
-                    while (!ts.isIdentifier(state)) {
-                        ids.unshift(state.right);
-                        state = state.left;
-                    }
-                    ids.unshift(state);
-                    return ids;
+            }
+            if (includes === 0 /* None */) {
+                includes = 4 /* Property */;
+                excludes = 0 /* PropertyExcludes */;
+            }
+            declareSymbol(symbolTable, namespaceSymbol, declaration, includes | 67108864 /* Assignment */, excludes & ~67108864 /* Assignment */);
+        }
+        function isTopLevelNamespaceAssignment(propertyAccess) {
+            return ts.isBinaryExpression(propertyAccess.parent)
+                ? getParentOfBinaryExpression(propertyAccess.parent).parent.kind === 297 /* SourceFile */
+                : propertyAccess.parent.parent.kind === 297 /* SourceFile */;
+        }
+        function bindPropertyAssignment(name, propertyAccess, isPrototypeProperty, containerIsClass) {
+            var namespaceSymbol = lookupSymbolForPropertyAccess(name, container) || lookupSymbolForPropertyAccess(name, blockScopeContainer);
+            var isToplevel = isTopLevelNamespaceAssignment(propertyAccess);
+            namespaceSymbol = bindPotentiallyMissingNamespaces(namespaceSymbol, propertyAccess.expression, isToplevel, isPrototypeProperty, containerIsClass);
+            bindPotentiallyNewExpandoMemberToNamespace(propertyAccess, namespaceSymbol, isPrototypeProperty);
+        }
+        /**
+         * Javascript expando values are:
+         * - Functions
+         * - classes
+         * - namespaces
+         * - variables initialized with function expressions
+         * -                       with class expressions
+         * -                       with empty object literals
+         * -                       with non-empty object literals if assigned to the prototype property
+         */
+        function isExpandoSymbol(symbol) {
+            if (symbol.flags & (16 /* Function */ | 32 /* Class */ | 1024 /* NamespaceModule */)) {
+                return true;
+            }
+            var node = symbol.valueDeclaration;
+            if (node && ts.isCallExpression(node)) {
+                return !!ts.getAssignedExpandoInitializer(node);
+            }
+            var init = !node ? undefined :
+                ts.isVariableDeclaration(node) ? node.initializer :
+                    ts.isBinaryExpression(node) ? node.right :
+                        ts.isPropertyAccessExpression(node) && ts.isBinaryExpression(node.parent) ? node.parent.right :
+                            undefined;
+            init = init && ts.getRightMostAssignedExpression(init);
+            if (init) {
+                var isPrototypeAssignment = ts.isPrototypeAccess(ts.isVariableDeclaration(node) ? node.name : ts.isBinaryExpression(node) ? node.left : node);
+                return !!ts.getExpandoInitializer(ts.isBinaryExpression(init) && (init.operatorToken.kind === 56 /* BarBarToken */ || init.operatorToken.kind === 60 /* QuestionQuestionToken */) ? init.right : init, isPrototypeAssignment);
+            }
+            return false;
+        }
+        function getParentOfBinaryExpression(expr) {
+            while (ts.isBinaryExpression(expr.parent)) {
+                expr = expr.parent;
+            }
+            return expr.parent;
+        }
+        function lookupSymbolForPropertyAccess(node, lookupContainer) {
+            if (lookupContainer === void 0) { lookupContainer = container; }
+            if (ts.isIdentifier(node)) {
+                return lookupSymbolForName(lookupContainer, node.escapedText);
+            }
+            else {
+                var symbol = lookupSymbolForPropertyAccess(node.expression);
+                return symbol && symbol.exports && symbol.exports.get(ts.getElementOrPropertyAccessName(node));
+            }
+        }
+        function forEachIdentifierInEntityName(e, parent, action) {
+            if (isExportsOrModuleExportsOrAlias(file, e)) {
+                return file.symbol;
+            }
+            else if (ts.isIdentifier(e)) {
+                return action(e, lookupSymbolForPropertyAccess(e), parent);
+            }
+            else {
+                var s = forEachIdentifierInEntityName(e.expression, parent, action);
+                var name = ts.getNameOrArgument(e);
+                // unreachable
+                if (ts.isPrivateIdentifier(name)) {
+                    ts.Debug.fail("unexpected PrivateIdentifier");
                 }
-                function createTypeNodesFromResolvedType(resolvedType) {
-                    if (checkTruncationLength(context)) {
-                        return [ts.createPropertySignature(/*modifiers*/ undefined, "...", /*questionToken*/ undefined, /*type*/ undefined, /*initializer*/ undefined)];
-                    }
-                    var typeElements = [];
-                    for (var _i = 0, _a = resolvedType.callSignatures; _i < _a.length; _i++) {
-                        var signature = _a[_i];
-                        typeElements.push(signatureToSignatureDeclarationHelper(signature, 165 /* CallSignature */, context));
-                    }
-                    for (var _b = 0, _c = resolvedType.constructSignatures; _b < _c.length; _b++) {
-                        var signature = _c[_b];
-                        typeElements.push(signatureToSignatureDeclarationHelper(signature, 166 /* ConstructSignature */, context));
-                    }
-                    if (resolvedType.stringIndexInfo) {
-                        var indexSignature = void 0;
-                        if (resolvedType.objectFlags & 2048 /* ReverseMapped */) {
-                            indexSignature = indexInfoToIndexSignatureDeclarationHelper(createIndexInfo(anyType, resolvedType.stringIndexInfo.isReadonly, resolvedType.stringIndexInfo.declaration), 0 /* String */, context);
-                            indexSignature.type = createElidedInformationPlaceholder(context);
-                        }
-                        else {
-                            indexSignature = indexInfoToIndexSignatureDeclarationHelper(resolvedType.stringIndexInfo, 0 /* String */, context);
-                        }
-                        typeElements.push(indexSignature);
-                    }
-                    if (resolvedType.numberIndexInfo) {
-                        typeElements.push(indexInfoToIndexSignatureDeclarationHelper(resolvedType.numberIndexInfo, 1 /* Number */, context));
-                    }
-                    var properties = resolvedType.properties;
-                    if (!properties) {
-                        return typeElements;
-                    }
-                    var i = 0;
-                    for (var _d = 0, properties_1 = properties; _d < properties_1.length; _d++) {
-                        var propertySymbol = properties_1[_d];
-                        i++;
-                        if (context.flags & 2048 /* WriteClassExpressionAsTypeLiteral */) {
-                            if (propertySymbol.flags & 4194304 /* Prototype */) {
-                                continue;
-                            }
-                            if (ts.getDeclarationModifierFlagsFromSymbol(propertySymbol) & (8 /* Private */ | 16 /* Protected */) && context.tracker.reportPrivateInBaseOfClassExpression) {
-                                context.tracker.reportPrivateInBaseOfClassExpression(ts.unescapeLeadingUnderscores(propertySymbol.escapedName));
-                            }
-                        }
-                        if (checkTruncationLength(context) && (i + 2 < properties.length - 1)) {
-                            typeElements.push(ts.createPropertySignature(/*modifiers*/ undefined, "... " + (properties.length - i) + " more ...", /*questionToken*/ undefined, /*type*/ undefined, /*initializer*/ undefined));
-                            addPropertyToElementList(properties[properties.length - 1], context, typeElements);
-                            break;
-                        }
-                        addPropertyToElementList(propertySymbol, context, typeElements);
-                    }
-                    return typeElements.length ? typeElements : undefined;
+                return action(name, s && s.exports && s.exports.get(ts.getElementOrPropertyAccessName(e)), s);
+            }
+        }
+        function bindCallExpression(node) {
+            // We're only inspecting call expressions to detect CommonJS modules, so we can skip
+            // this check if we've already seen the module indicator
+            if (!file.commonJsModuleIndicator && ts.isRequireCall(node, /*checkArgumentIsStringLiteralLike*/ false)) {
+                setCommonJsModuleIndicator(node);
+            }
+        }
+        function bindClassLikeDeclaration(node) {
+            if (node.kind === 252 /* ClassDeclaration */) {
+                bindBlockScopedDeclaration(node, 32 /* Class */, 899503 /* ClassExcludes */);
+            }
+            else {
+                var bindingName = node.name ? node.name.escapedText : "__class" /* Class */;
+                bindAnonymousDeclaration(node, 32 /* Class */, bindingName);
+                // Add name of class expression into the map for semantic classifier
+                if (node.name) {
+                    classifiableNames.add(node.name.escapedText);
                 }
             }
-            function createElidedInformationPlaceholder(context) {
-                context.approximateLength += 3;
-                if (!(context.flags & 1 /* NoTruncation */)) {
-                    return ts.createTypeReferenceNode(ts.createIdentifier("..."), /*typeArguments*/ undefined);
+            var symbol = node.symbol;
+            // TypeScript 1.0 spec (April 2014): 8.4
+            // Every class automatically contains a static property member named 'prototype', the
+            // type of which is an instantiation of the class type with type Any supplied as a type
+            // argument for each type parameter. It is an error to explicitly declare a static
+            // property member with the name 'prototype'.
+            //
+            // Note: we check for this here because this class may be merging into a module.  The
+            // module might have an exported variable called 'prototype'.  We can't allow that as
+            // that would clash with the built-in 'prototype' for the class.
+            var prototypeSymbol = createSymbol(4 /* Property */ | 4194304 /* Prototype */, "prototype");
+            var symbolExport = symbol.exports.get(prototypeSymbol.escapedName);
+            if (symbolExport) {
+                if (node.name) {
+                    ts.setParent(node.name, node);
                 }
-                return ts.createKeywordTypeNode(125 /* AnyKeyword */);
+                file.bindDiagnostics.push(createDiagnosticForNode(symbolExport.declarations[0], ts.Diagnostics.Duplicate_identifier_0, ts.symbolName(prototypeSymbol)));
             }
-            function addPropertyToElementList(propertySymbol, context, typeElements) {
-                var propertyIsReverseMapped = !!(ts.getCheckFlags(propertySymbol) & 8192 /* ReverseMapped */);
-                var propertyType = propertyIsReverseMapped && context.flags & 33554432 /* InReverseMappedType */ ?
-                    anyType : getTypeOfSymbol(propertySymbol);
-                var saveEnclosingDeclaration = context.enclosingDeclaration;
-                context.enclosingDeclaration = undefined;
-                if (context.tracker.trackSymbol && ts.getCheckFlags(propertySymbol) & 4096 /* Late */) {
-                    var decl = ts.first(propertySymbol.declarations);
-                    if (hasLateBindableName(decl)) {
-                        if (ts.isBinaryExpression(decl)) {
-                            var name = ts.getNameOfDeclaration(decl);
-                            if (name && ts.isElementAccessExpression(name) && ts.isPropertyAccessEntityNameExpression(name.argumentExpression)) {
-                                trackComputedName(name.argumentExpression, saveEnclosingDeclaration, context);
-                            }
-                        }
-                        else {
-                            trackComputedName(decl.name.expression, saveEnclosingDeclaration, context);
-                        }
-                    }
+            symbol.exports.set(prototypeSymbol.escapedName, prototypeSymbol);
+            prototypeSymbol.parent = symbol;
+        }
+        function bindEnumDeclaration(node) {
+            return ts.isEnumConst(node)
+                ? bindBlockScopedDeclaration(node, 128 /* ConstEnum */, 899967 /* ConstEnumExcludes */)
+                : bindBlockScopedDeclaration(node, 256 /* RegularEnum */, 899327 /* RegularEnumExcludes */);
+        }
+        function bindVariableDeclarationOrBindingElement(node) {
+            if (inStrictMode) {
+                checkStrictModeEvalOrArguments(node, node.name);
+            }
+            if (!ts.isBindingPattern(node.name)) {
+                if (ts.isInJSFile(node) && ts.isRequireVariableDeclaration(node, /*requireStringLiteralLikeArgument*/ true) && !ts.getJSDocTypeTag(node)) {
+                    declareSymbolAndAddToSymbolTable(node, 2097152 /* Alias */, 2097152 /* AliasExcludes */);
                 }
-                context.enclosingDeclaration = saveEnclosingDeclaration;
-                var propertyName = getPropertyNameNodeForSymbol(propertySymbol, context);
-                context.approximateLength += (ts.symbolName(propertySymbol).length + 1);
-                var optionalToken = propertySymbol.flags & 16777216 /* Optional */ ? ts.createToken(57 /* QuestionToken */) : undefined;
-                if (propertySymbol.flags & (16 /* Function */ | 8192 /* Method */) && !getPropertiesOfObjectType(propertyType).length && !isReadonlySymbol(propertySymbol)) {
-                    var signatures = getSignaturesOfType(filterType(propertyType, function (t) { return !(t.flags & 32768 /* Undefined */); }), 0 /* Call */);
-                    for (var _i = 0, signatures_1 = signatures; _i < signatures_1.length; _i++) {
-                        var signature = signatures_1[_i];
-                        var methodDeclaration = signatureToSignatureDeclarationHelper(signature, 160 /* MethodSignature */, context);
-                        methodDeclaration.name = propertyName;
-                        methodDeclaration.questionToken = optionalToken;
-                        typeElements.push(preserveCommentsOn(methodDeclaration));
-                    }
+                else if (ts.isBlockOrCatchScoped(node)) {
+                    bindBlockScopedDeclaration(node, 2 /* BlockScopedVariable */, 111551 /* BlockScopedVariableExcludes */);
                 }
-                else {
-                    var savedFlags = context.flags;
-                    context.flags |= propertyIsReverseMapped ? 33554432 /* InReverseMappedType */ : 0;
-                    var propertyTypeNode = void 0;
-                    if (propertyIsReverseMapped && !!(savedFlags & 33554432 /* InReverseMappedType */)) {
-                        propertyTypeNode = createElidedInformationPlaceholder(context);
-                    }
-                    else {
-                        propertyTypeNode = propertyType ? serializeTypeForDeclaration(context, propertyType, propertySymbol, saveEnclosingDeclaration) : ts.createKeywordTypeNode(125 /* AnyKeyword */);
-                    }
-                    context.flags = savedFlags;
-                    var modifiers = isReadonlySymbol(propertySymbol) ? [ts.createToken(138 /* ReadonlyKeyword */)] : undefined;
-                    if (modifiers) {
-                        context.approximateLength += 9;
-                    }
-                    var propertySignature = ts.createPropertySignature(modifiers, propertyName, optionalToken, propertyTypeNode, 
-                    /*initializer*/ undefined);
-                    typeElements.push(preserveCommentsOn(propertySignature));
+                else if (ts.isParameterDeclaration(node)) {
+                    // It is safe to walk up parent chain to find whether the node is a destructuring parameter declaration
+                    // because its parent chain has already been set up, since parents are set before descending into children.
+                    //
+                    // If node is a binding element in parameter declaration, we need to use ParameterExcludes.
+                    // Using ParameterExcludes flag allows the compiler to report an error on duplicate identifiers in Parameter Declaration
+                    // For example:
+                    //      function foo([a,a]) {} // Duplicate Identifier error
+                    //      function bar(a,a) {}   // Duplicate Identifier error, parameter declaration in this case is handled in bindParameter
+                    //                             // which correctly set excluded symbols
+                    declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111551 /* ParameterExcludes */);
                 }
-                function preserveCommentsOn(node) {
-                    if (ts.some(propertySymbol.declarations, function (d) { return d.kind === 323 /* JSDocPropertyTag */; })) {
-                        var d = ts.find(propertySymbol.declarations, function (d) { return d.kind === 323 /* JSDocPropertyTag */; });
-                        var commentText = d.comment;
-                        if (commentText) {
-                            ts.setSyntheticLeadingComments(node, [{ kind: 3 /* MultiLineCommentTrivia */, text: "*\n * " + commentText.replace(/\n/g, "\n * ") + "\n ", pos: -1, end: -1, hasTrailingNewLine: true }]);
-                        }
-                    }
-                    else if (propertySymbol.valueDeclaration) {
-                        // Copy comments to node for declaration emit
-                        ts.setCommentRange(node, propertySymbol.valueDeclaration);
-                    }
-                    return node;
+                else {
+                    declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111550 /* FunctionScopedVariableExcludes */);
                 }
             }
-            function mapToTypeNodes(types, context, isBareList) {
-                if (ts.some(types)) {
-                    if (checkTruncationLength(context)) {
-                        if (!isBareList) {
-                            return [ts.createTypeReferenceNode("...", /*typeArguments*/ undefined)];
-                        }
-                        else if (types.length > 2) {
-                            return [
-                                typeToTypeNodeHelper(types[0], context),
-                                ts.createTypeReferenceNode("... " + (types.length - 2) + " more ...", /*typeArguments*/ undefined),
-                                typeToTypeNodeHelper(types[types.length - 1], context)
-                            ];
-                        }
-                    }
-                    var mayHaveNameCollisions = !(context.flags & 64 /* UseFullyQualifiedType */);
-                    /** Map from type reference identifier text to [type, index in `result` where the type node is] */
-                    var seenNames = mayHaveNameCollisions ? ts.createUnderscoreEscapedMultiMap() : undefined;
-                    var result_3 = [];
-                    var i = 0;
-                    for (var _i = 0, types_1 = types; _i < types_1.length; _i++) {
-                        var type = types_1[_i];
-                        i++;
-                        if (checkTruncationLength(context) && (i + 2 < types.length - 1)) {
-                            result_3.push(ts.createTypeReferenceNode("... " + (types.length - i) + " more ...", /*typeArguments*/ undefined));
-                            var typeNode_1 = typeToTypeNodeHelper(types[types.length - 1], context);
-                            if (typeNode_1) {
-                                result_3.push(typeNode_1);
-                            }
-                            break;
-                        }
-                        context.approximateLength += 2; // Account for whitespace + separator
-                        var typeNode = typeToTypeNodeHelper(type, context);
-                        if (typeNode) {
-                            result_3.push(typeNode);
-                            if (seenNames && ts.isIdentifierTypeReference(typeNode)) {
-                                seenNames.add(typeNode.typeName.escapedText, [type, result_3.length - 1]);
-                            }
-                        }
-                    }
-                    if (seenNames) {
-                        // To avoid printing types like `[Foo, Foo]` or `Bar & Bar` where
-                        // occurrences of the same name actually come from different
-                        // namespaces, go through the single-identifier type reference nodes
-                        // we just generated, and see if any names were generated more than
-                        // once while referring to different types. If so, regenerate the
-                        // type node for each entry by that name with the
-                        // `UseFullyQualifiedType` flag enabled.
-                        var saveContextFlags = context.flags;
-                        context.flags |= 64 /* UseFullyQualifiedType */;
-                        seenNames.forEach(function (types) {
-                            if (!ts.arrayIsHomogeneous(types, function (_a, _b) {
-                                var a = _a[0];
-                                var b = _b[0];
-                                return typesAreSameReference(a, b);
-                            })) {
-                                for (var _i = 0, types_2 = types; _i < types_2.length; _i++) {
-                                    var _a = types_2[_i], type = _a[0], resultIndex = _a[1];
-                                    result_3[resultIndex] = typeToTypeNodeHelper(type, context);
-                                }
-                            }
-                        });
-                        context.flags = saveContextFlags;
-                    }
-                    return result_3;
+        }
+        function bindParameter(node) {
+            if (node.kind === 326 /* JSDocParameterTag */ && container.kind !== 313 /* JSDocSignature */) {
+                return;
+            }
+            if (inStrictMode && !(node.flags & 8388608 /* Ambient */)) {
+                // It is a SyntaxError if the identifier eval or arguments appears within a FormalParameterList of a
+                // strict mode FunctionLikeDeclaration or FunctionExpression(13.1)
+                checkStrictModeEvalOrArguments(node, node.name);
+            }
+            if (ts.isBindingPattern(node.name)) {
+                bindAnonymousDeclaration(node, 1 /* FunctionScopedVariable */, "__" + node.parent.parameters.indexOf(node));
+            }
+            else {
+                declareSymbolAndAddToSymbolTable(node, 1 /* FunctionScopedVariable */, 111551 /* ParameterExcludes */);
+            }
+            // If this is a property-parameter, then also declare the property symbol into the
+            // containing class.
+            if (ts.isParameterPropertyDeclaration(node, node.parent)) {
+                var classDeclaration = node.parent.parent;
+                declareSymbol(classDeclaration.symbol.members, classDeclaration.symbol, node, 4 /* Property */ | (node.questionToken ? 16777216 /* Optional */ : 0 /* None */), 0 /* PropertyExcludes */);
+            }
+        }
+        function bindFunctionDeclaration(node) {
+            if (!file.isDeclarationFile && !(node.flags & 8388608 /* Ambient */)) {
+                if (ts.isAsyncFunction(node)) {
+                    emitFlags |= 2048 /* HasAsyncFunctions */;
                 }
             }
-            function typesAreSameReference(a, b) {
-                return a === b
-                    || !!a.symbol && a.symbol === b.symbol
-                    || !!a.aliasSymbol && a.aliasSymbol === b.aliasSymbol;
+            checkStrictModeFunctionName(node);
+            if (inStrictMode) {
+                checkStrictModeFunctionDeclaration(node);
+                bindBlockScopedDeclaration(node, 16 /* Function */, 110991 /* FunctionExcludes */);
             }
-            function indexInfoToIndexSignatureDeclarationHelper(indexInfo, kind, context) {
-                var name = ts.getNameFromIndexInfo(indexInfo) || "x";
-                var indexerTypeNode = ts.createKeywordTypeNode(kind === 0 /* String */ ? 143 /* StringKeyword */ : 140 /* NumberKeyword */);
-                var indexingParameter = ts.createParameter(
-                /*decorators*/ undefined, 
-                /*modifiers*/ undefined, 
-                /*dotDotDotToken*/ undefined, name, 
-                /*questionToken*/ undefined, indexerTypeNode, 
-                /*initializer*/ undefined);
-                var typeNode = typeToTypeNodeHelper(indexInfo.type || anyType, context);
-                if (!indexInfo.type && !(context.flags & 2097152 /* AllowEmptyIndexInfoType */)) {
-                    context.encounteredError = true;
+            else {
+                declareSymbolAndAddToSymbolTable(node, 16 /* Function */, 110991 /* FunctionExcludes */);
+            }
+        }
+        function bindFunctionExpression(node) {
+            if (!file.isDeclarationFile && !(node.flags & 8388608 /* Ambient */)) {
+                if (ts.isAsyncFunction(node)) {
+                    emitFlags |= 2048 /* HasAsyncFunctions */;
                 }
-                context.approximateLength += (name.length + 4);
-                return ts.createIndexSignature(
-                /*decorators*/ undefined, indexInfo.isReadonly ? [ts.createToken(138 /* ReadonlyKeyword */)] : undefined, [indexingParameter], typeNode);
             }
-            function signatureToSignatureDeclarationHelper(signature, kind, context, privateSymbolVisitor, bundledImports) {
-                var suppressAny = context.flags & 256 /* SuppressAnyReturnType */;
-                if (suppressAny)
-                    context.flags &= ~256 /* SuppressAnyReturnType */; // suppress only toplevel `any`s
-                var typeParameters;
-                var typeArguments;
-                if (context.flags & 32 /* WriteTypeArgumentsOfSignature */ && signature.target && signature.mapper && signature.target.typeParameters) {
-                    typeArguments = signature.target.typeParameters.map(function (parameter) { return typeToTypeNodeHelper(instantiateType(parameter, signature.mapper), context); });
+            if (currentFlow) {
+                node.flowNode = currentFlow;
+            }
+            checkStrictModeFunctionName(node);
+            var bindingName = node.name ? node.name.escapedText : "__function" /* Function */;
+            return bindAnonymousDeclaration(node, 16 /* Function */, bindingName);
+        }
+        function bindPropertyOrMethodOrAccessor(node, symbolFlags, symbolExcludes) {
+            if (!file.isDeclarationFile && !(node.flags & 8388608 /* Ambient */) && ts.isAsyncFunction(node)) {
+                emitFlags |= 2048 /* HasAsyncFunctions */;
+            }
+            if (currentFlow && ts.isObjectLiteralOrClassExpressionMethod(node)) {
+                node.flowNode = currentFlow;
+            }
+            return ts.hasDynamicName(node)
+                ? bindAnonymousDeclaration(node, symbolFlags, "__computed" /* Computed */)
+                : declareSymbolAndAddToSymbolTable(node, symbolFlags, symbolExcludes);
+        }
+        function getInferTypeContainer(node) {
+            var extendsType = ts.findAncestor(node, function (n) { return n.parent && ts.isConditionalTypeNode(n.parent) && n.parent.extendsType === n; });
+            return extendsType && extendsType.parent;
+        }
+        function bindTypeParameter(node) {
+            if (ts.isJSDocTemplateTag(node.parent)) {
+                var container_1 = ts.find(node.parent.parent.tags, ts.isJSDocTypeAlias) || ts.getHostSignatureFromJSDoc(node.parent); // TODO: GH#18217
+                if (container_1) {
+                    if (!container_1.locals) {
+                        container_1.locals = ts.createSymbolTable();
+                    }
+                    declareSymbol(container_1.locals, /*parent*/ undefined, node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
                 }
                 else {
-                    typeParameters = signature.typeParameters && signature.typeParameters.map(function (parameter) { return typeParameterToDeclaration(parameter, context); });
-                }
-                var parameters = getExpandedParameters(signature).map(function (parameter) { return symbolToParameterDeclaration(parameter, context, kind === 162 /* Constructor */, privateSymbolVisitor, bundledImports); });
-                if (signature.thisParameter) {
-                    var thisParameter = symbolToParameterDeclaration(signature.thisParameter, context);
-                    parameters.unshift(thisParameter);
+                    declareSymbolAndAddToSymbolTable(node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
                 }
-                var returnTypeNode;
-                var typePredicate = getTypePredicateOfSignature(signature);
-                if (typePredicate) {
-                    var assertsModifier = typePredicate.kind === 2 /* AssertsThis */ || typePredicate.kind === 3 /* AssertsIdentifier */ ?
-                        ts.createToken(124 /* AssertsKeyword */) :
-                        undefined;
-                    var parameterName = typePredicate.kind === 1 /* Identifier */ || typePredicate.kind === 3 /* AssertsIdentifier */ ?
-                        ts.setEmitFlags(ts.createIdentifier(typePredicate.parameterName), 16777216 /* NoAsciiEscaping */) :
-                        ts.createThisTypeNode();
-                    var typeNode = typePredicate.type && typeToTypeNodeHelper(typePredicate.type, context);
-                    returnTypeNode = ts.createTypePredicateNodeWithModifier(assertsModifier, parameterName, typeNode);
+            }
+            else if (node.parent.kind === 185 /* InferType */) {
+                var container_2 = getInferTypeContainer(node.parent);
+                if (container_2) {
+                    if (!container_2.locals) {
+                        container_2.locals = ts.createSymbolTable();
+                    }
+                    declareSymbol(container_2.locals, /*parent*/ undefined, node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
                 }
                 else {
-                    var returnType = getReturnTypeOfSignature(signature);
-                    if (returnType && !(suppressAny && isTypeAny(returnType))) {
-                        returnTypeNode = serializeReturnTypeForSignature(context, returnType, signature, privateSymbolVisitor, bundledImports);
-                    }
-                    else if (!suppressAny) {
-                        returnTypeNode = ts.createKeywordTypeNode(125 /* AnyKeyword */);
-                    }
+                    bindAnonymousDeclaration(node, 262144 /* TypeParameter */, getDeclarationName(node)); // TODO: GH#18217
                 }
-                context.approximateLength += 3; // Usually a signature contributes a few more characters than this, but 3 is the minimum
-                return ts.createSignatureDeclaration(kind, typeParameters, parameters, returnTypeNode, typeArguments);
             }
-            function typeParameterToDeclarationWithConstraint(type, context, constraintNode) {
-                var savedContextFlags = context.flags;
-                context.flags &= ~512 /* WriteTypeParametersInQualifiedName */; // Avoids potential infinite loop when building for a claimspace with a generic
-                var name = typeParameterToName(type, context);
-                var defaultParameter = getDefaultFromTypeParameter(type);
-                var defaultParameterNode = defaultParameter && typeToTypeNodeHelper(defaultParameter, context);
-                context.flags = savedContextFlags;
-                return ts.createTypeParameterDeclaration(name, constraintNode, defaultParameterNode);
+            else {
+                declareSymbolAndAddToSymbolTable(node, 262144 /* TypeParameter */, 526824 /* TypeParameterExcludes */);
             }
-            function typeParameterToDeclaration(type, context, constraint) {
-                if (constraint === void 0) { constraint = getConstraintOfTypeParameter(type); }
-                var constraintNode = constraint && typeToTypeNodeHelper(constraint, context);
-                return typeParameterToDeclarationWithConstraint(type, context, constraintNode);
+        }
+        // reachability checks
+        function shouldReportErrorOnModuleDeclaration(node) {
+            var instanceState = getModuleInstanceState(node);
+            return instanceState === 1 /* Instantiated */ || (instanceState === 2 /* ConstEnumOnly */ && !!options.preserveConstEnums);
+        }
+        function checkUnreachable(node) {
+            if (!(currentFlow.flags & 1 /* Unreachable */)) {
+                return false;
             }
-            function symbolToParameterDeclaration(parameterSymbol, context, preserveModifierFlags, privateSymbolVisitor, bundledImports) {
-                var parameterDeclaration = ts.getDeclarationOfKind(parameterSymbol, 156 /* Parameter */);
-                if (!parameterDeclaration && !ts.isTransientSymbol(parameterSymbol)) {
-                    parameterDeclaration = ts.getDeclarationOfKind(parameterSymbol, 317 /* JSDocParameterTag */);
-                }
-                var parameterType = getTypeOfSymbol(parameterSymbol);
-                if (parameterDeclaration && isRequiredInitializedParameter(parameterDeclaration)) {
-                    parameterType = getOptionalType(parameterType);
-                }
-                var parameterTypeNode = serializeTypeForDeclaration(context, parameterType, parameterSymbol, context.enclosingDeclaration, privateSymbolVisitor, bundledImports);
-                var modifiers = !(context.flags & 8192 /* OmitParameterModifiers */) && preserveModifierFlags && parameterDeclaration && parameterDeclaration.modifiers ? parameterDeclaration.modifiers.map(ts.getSynthesizedClone) : undefined;
-                var isRest = parameterDeclaration && ts.isRestParameter(parameterDeclaration) || ts.getCheckFlags(parameterSymbol) & 32768 /* RestParameter */;
-                var dotDotDotToken = isRest ? ts.createToken(25 /* DotDotDotToken */) : undefined;
-                var name = parameterDeclaration ? parameterDeclaration.name ?
-                    parameterDeclaration.name.kind === 75 /* Identifier */ ? ts.setEmitFlags(ts.getSynthesizedClone(parameterDeclaration.name), 16777216 /* NoAsciiEscaping */) :
-                        parameterDeclaration.name.kind === 153 /* QualifiedName */ ? ts.setEmitFlags(ts.getSynthesizedClone(parameterDeclaration.name.right), 16777216 /* NoAsciiEscaping */) :
-                            cloneBindingName(parameterDeclaration.name) :
-                    ts.symbolName(parameterSymbol) :
-                    ts.symbolName(parameterSymbol);
-                var isOptional = parameterDeclaration && isOptionalParameter(parameterDeclaration) || ts.getCheckFlags(parameterSymbol) & 16384 /* OptionalParameter */;
-                var questionToken = isOptional ? ts.createToken(57 /* QuestionToken */) : undefined;
-                var parameterNode = ts.createParameter(
-                /*decorators*/ undefined, modifiers, dotDotDotToken, name, questionToken, parameterTypeNode, 
-                /*initializer*/ undefined);
-                context.approximateLength += ts.symbolName(parameterSymbol).length + 3;
-                return parameterNode;
-                function cloneBindingName(node) {
-                    return elideInitializerAndSetEmitFlags(node);
-                    function elideInitializerAndSetEmitFlags(node) {
-                        if (context.tracker.trackSymbol && ts.isComputedPropertyName(node) && isLateBindableName(node)) {
-                            trackComputedName(node.expression, context.enclosingDeclaration, context);
-                        }
-                        var visited = ts.visitEachChild(node, elideInitializerAndSetEmitFlags, ts.nullTransformationContext, /*nodesVisitor*/ undefined, elideInitializerAndSetEmitFlags);
-                        var clone = ts.nodeIsSynthesized(visited) ? visited : ts.getSynthesizedClone(visited);
-                        if (clone.kind === 191 /* BindingElement */) {
-                            clone.initializer = undefined;
-                        }
-                        return ts.setEmitFlags(clone, 1 /* SingleLine */ | 16777216 /* NoAsciiEscaping */);
+            if (currentFlow === unreachableFlow) {
+                var reportError = 
+                // report error on all statements except empty ones
+                (ts.isStatementButNotDeclaration(node) && node.kind !== 231 /* EmptyStatement */) ||
+                    // report error on class declarations
+                    node.kind === 252 /* ClassDeclaration */ ||
+                    // report error on instantiated modules or const-enums only modules if preserveConstEnums is set
+                    (node.kind === 256 /* ModuleDeclaration */ && shouldReportErrorOnModuleDeclaration(node));
+                if (reportError) {
+                    currentFlow = reportedUnreachableFlow;
+                    if (!options.allowUnreachableCode) {
+                        // unreachable code is reported if
+                        // - user has explicitly asked about it AND
+                        // - statement is in not ambient context (statements in ambient context is already an error
+                        //   so we should not report extras) AND
+                        //   - node is not variable statement OR
+                        //   - node is block scoped variable statement OR
+                        //   - node is not block scoped variable statement and at least one variable declaration has initializer
+                        //   Rationale: we don't want to report errors on non-initialized var's since they are hoisted
+                        //   On the other side we do want to report errors on non-initialized 'lets' because of TDZ
+                        var isError_1 = ts.unreachableCodeIsError(options) &&
+                            !(node.flags & 8388608 /* Ambient */) &&
+                            (!ts.isVariableStatement(node) ||
+                                !!(ts.getCombinedNodeFlags(node.declarationList) & 3 /* BlockScoped */) ||
+                                node.declarationList.declarations.some(function (d) { return !!d.initializer; }));
+                        eachUnreachableRange(node, function (start, end) { return errorOrSuggestionOnRange(isError_1, start, end, ts.Diagnostics.Unreachable_code_detected); });
                     }
                 }
             }
-            function trackComputedName(accessExpression, enclosingDeclaration, context) {
-                if (!context.tracker.trackSymbol)
-                    return;
-                // get symbol of the first identifier of the entityName
-                var firstIdentifier = ts.getFirstIdentifier(accessExpression);
-                var name = resolveName(firstIdentifier, firstIdentifier.escapedText, 111551 /* Value */ | 1048576 /* ExportValue */, /*nodeNotFoundErrorMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true);
-                if (name) {
-                    context.tracker.trackSymbol(name, enclosingDeclaration, 111551 /* Value */);
-                }
+            return true;
+        }
+    }
+    function eachUnreachableRange(node, cb) {
+        if (ts.isStatement(node) && isExecutableStatement(node) && ts.isBlock(node.parent)) {
+            var statements = node.parent.statements;
+            var slice_1 = ts.sliceAfter(statements, node);
+            ts.getRangesWhere(slice_1, isExecutableStatement, function (start, afterEnd) { return cb(slice_1[start], slice_1[afterEnd - 1]); });
+        }
+        else {
+            cb(node, node);
+        }
+    }
+    // As opposed to a pure declaration like an `interface`
+    function isExecutableStatement(s) {
+        // Don't remove statements that can validly be used before they appear.
+        return !ts.isFunctionDeclaration(s) && !isPurelyTypeDeclaration(s) && !ts.isEnumDeclaration(s) &&
+            // `var x;` may declare a variable used above
+            !(ts.isVariableStatement(s) && !(ts.getCombinedNodeFlags(s) & (1 /* Let */ | 2 /* Const */)) && s.declarationList.declarations.some(function (d) { return !d.initializer; }));
+    }
+    function isPurelyTypeDeclaration(s) {
+        switch (s.kind) {
+            case 253 /* InterfaceDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+                return true;
+            case 256 /* ModuleDeclaration */:
+                return getModuleInstanceState(s) !== 1 /* Instantiated */;
+            case 255 /* EnumDeclaration */:
+                return ts.hasSyntacticModifier(s, 2048 /* Const */);
+            default:
+                return false;
+        }
+    }
+    function isExportsOrModuleExportsOrAlias(sourceFile, node) {
+        var i = 0;
+        var q = [node];
+        while (q.length && i < 100) {
+            i++;
+            node = q.shift();
+            if (ts.isExportsIdentifier(node) || ts.isModuleExportsAccessExpression(node)) {
+                return true;
             }
-            function lookupSymbolChain(symbol, context, meaning, yieldModuleSymbol) {
-                context.tracker.trackSymbol(symbol, context.enclosingDeclaration, meaning); // TODO: GH#18217
-                return lookupSymbolChainWorker(symbol, context, meaning, yieldModuleSymbol);
+            else if (ts.isIdentifier(node)) {
+                var symbol = lookupSymbolForName(sourceFile, node.escapedText);
+                if (!!symbol && !!symbol.valueDeclaration && ts.isVariableDeclaration(symbol.valueDeclaration) && !!symbol.valueDeclaration.initializer) {
+                    var init = symbol.valueDeclaration.initializer;
+                    q.push(init);
+                    if (ts.isAssignmentExpression(init, /*excludeCompoundAssignment*/ true)) {
+                        q.push(init.left);
+                        q.push(init.right);
+                    }
+                }
             }
-            function lookupSymbolChainWorker(symbol, context, meaning, yieldModuleSymbol) {
-                // Try to get qualified name if the symbol is not a type parameter and there is an enclosing declaration.
-                var chain;
-                var isTypeParameter = symbol.flags & 262144 /* TypeParameter */;
-                if (!isTypeParameter && (context.enclosingDeclaration || context.flags & 64 /* UseFullyQualifiedType */) && !(context.flags & 134217728 /* DoNotIncludeSymbolChain */)) {
-                    chain = ts.Debug.checkDefined(getSymbolChain(symbol, meaning, /*endOfChain*/ true));
-                    ts.Debug.assert(chain && chain.length > 0);
+        }
+        return false;
+    }
+    ts.isExportsOrModuleExportsOrAlias = isExportsOrModuleExportsOrAlias;
+    function lookupSymbolForName(container, name) {
+        var local = container.locals && container.locals.get(name);
+        if (local) {
+            return local.exportSymbol || local;
+        }
+        if (ts.isSourceFile(container) && container.jsGlobalAugmentations && container.jsGlobalAugmentations.has(name)) {
+            return container.jsGlobalAugmentations.get(name);
+        }
+        return container.symbol && container.symbol.exports && container.symbol.exports.get(name);
+    }
+})(ts || (ts = {}));
+/** @internal */
+var ts;
+(function (ts) {
+    function createGetSymbolWalker(getRestTypeOfSignature, getTypePredicateOfSignature, getReturnTypeOfSignature, getBaseTypes, resolveStructuredTypeMembers, getTypeOfSymbol, getResolvedSymbol, getIndexTypeOfStructuredType, getConstraintOfTypeParameter, getFirstIdentifier, getTypeArguments) {
+        return getSymbolWalker;
+        function getSymbolWalker(accept) {
+            if (accept === void 0) { accept = function () { return true; }; }
+            var visitedTypes = []; // Sparse array from id to type
+            var visitedSymbols = []; // Sparse array from id to symbol
+            return {
+                walkType: function (type) {
+                    try {
+                        visitType(type);
+                        return { visitedTypes: ts.getOwnValues(visitedTypes), visitedSymbols: ts.getOwnValues(visitedSymbols) };
+                    }
+                    finally {
+                        ts.clear(visitedTypes);
+                        ts.clear(visitedSymbols);
+                    }
+                },
+                walkSymbol: function (symbol) {
+                    try {
+                        visitSymbol(symbol);
+                        return { visitedTypes: ts.getOwnValues(visitedTypes), visitedSymbols: ts.getOwnValues(visitedSymbols) };
+                    }
+                    finally {
+                        ts.clear(visitedTypes);
+                        ts.clear(visitedSymbols);
+                    }
+                },
+            };
+            function visitType(type) {
+                if (!type) {
+                    return;
                 }
-                else {
-                    chain = [symbol];
+                if (visitedTypes[type.id]) {
+                    return;
                 }
-                return chain;
-                /** @param endOfChain Set to false for recursive calls; non-recursive calls should always output something. */
-                function getSymbolChain(symbol, meaning, endOfChain) {
-                    var accessibleSymbolChain = getAccessibleSymbolChain(symbol, context.enclosingDeclaration, meaning, !!(context.flags & 128 /* UseOnlyExternalAliasing */));
-                    var parentSpecifiers;
-                    if (!accessibleSymbolChain ||
-                        needsQualification(accessibleSymbolChain[0], context.enclosingDeclaration, accessibleSymbolChain.length === 1 ? meaning : getQualifiedLeftMeaning(meaning))) {
-                        // Go up and add our parent.
-                        var parents_1 = getContainersOfSymbol(accessibleSymbolChain ? accessibleSymbolChain[0] : symbol, context.enclosingDeclaration);
-                        if (ts.length(parents_1)) {
-                            parentSpecifiers = parents_1.map(function (symbol) {
-                                return ts.some(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)
-                                    ? getSpecifierForModuleSymbol(symbol, context)
-                                    : undefined;
-                            });
-                            var indices = parents_1.map(function (_, i) { return i; });
-                            indices.sort(sortByBestName);
-                            var sortedParents = indices.map(function (i) { return parents_1[i]; });
-                            for (var _i = 0, sortedParents_1 = sortedParents; _i < sortedParents_1.length; _i++) {
-                                var parent = sortedParents_1[_i];
-                                var parentChain = getSymbolChain(parent, getQualifiedLeftMeaning(meaning), /*endOfChain*/ false);
-                                if (parentChain) {
-                                    if (parent.exports && parent.exports.get("export=" /* ExportEquals */) &&
-                                        getSymbolIfSameReference(parent.exports.get("export=" /* ExportEquals */), symbol)) {
-                                        // parentChain root _is_ symbol - symbol is a module export=, so it kinda looks like it's own parent
-                                        // No need to lookup an alias for the symbol in itself
-                                        accessibleSymbolChain = parentChain;
-                                        break;
-                                    }
-                                    accessibleSymbolChain = parentChain.concat(accessibleSymbolChain || [getAliasForSymbolInContainer(parent, symbol) || symbol]);
-                                    break;
-                                }
-                            }
-                        }
+                visitedTypes[type.id] = type;
+                // Reuse visitSymbol to visit the type's symbol,
+                //  but be sure to bail on recuring into the type if accept declines the symbol.
+                var shouldBail = visitSymbol(type.symbol);
+                if (shouldBail)
+                    return;
+                // Visit the type's related types, if any
+                if (type.flags & 524288 /* Object */) {
+                    var objectType = type;
+                    var objectFlags = objectType.objectFlags;
+                    if (objectFlags & 4 /* Reference */) {
+                        visitTypeReference(type);
                     }
-                    if (accessibleSymbolChain) {
-                        return accessibleSymbolChain;
+                    if (objectFlags & 32 /* Mapped */) {
+                        visitMappedType(type);
                     }
-                    if (
-                    // If this is the last part of outputting the symbol, always output. The cases apply only to parent symbols.
-                    endOfChain ||
-                        // If a parent symbol is an anonymous type, don't write it.
-                        !(symbol.flags & (2048 /* TypeLiteral */ | 4096 /* ObjectLiteral */))) {
-                        // If a parent symbol is an external module, don't write it. (We prefer just `x` vs `"foo/bar".x`.)
-                        if (!endOfChain && !yieldModuleSymbol && !!ts.forEach(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
-                            return;
-                        }
-                        return [symbol];
+                    if (objectFlags & (1 /* Class */ | 2 /* Interface */)) {
+                        visitInterfaceType(type);
                     }
-                    function sortByBestName(a, b) {
-                        var specifierA = parentSpecifiers[a];
-                        var specifierB = parentSpecifiers[b];
-                        if (specifierA && specifierB) {
-                            var isBRelative = ts.pathIsRelative(specifierB);
-                            if (ts.pathIsRelative(specifierA) === isBRelative) {
-                                // Both relative or both non-relative, sort by number of parts
-                                return ts.moduleSpecifiers.countPathComponents(specifierA) - ts.moduleSpecifiers.countPathComponents(specifierB);
-                            }
-                            if (isBRelative) {
-                                // A is non-relative, B is relative: prefer A
-                                return -1;
-                            }
-                            // A is relative, B is non-relative: prefer B
-                            return 1;
-                        }
-                        return 0;
+                    if (objectFlags & (8 /* Tuple */ | 16 /* Anonymous */)) {
+                        visitObjectType(objectType);
                     }
                 }
+                if (type.flags & 262144 /* TypeParameter */) {
+                    visitTypeParameter(type);
+                }
+                if (type.flags & 3145728 /* UnionOrIntersection */) {
+                    visitUnionOrIntersectionType(type);
+                }
+                if (type.flags & 4194304 /* Index */) {
+                    visitIndexType(type);
+                }
+                if (type.flags & 8388608 /* IndexedAccess */) {
+                    visitIndexedAccessType(type);
+                }
             }
-            function typeParametersToTypeParameterDeclarations(symbol, context) {
-                var typeParameterNodes;
-                var targetSymbol = getTargetSymbol(symbol);
-                if (targetSymbol.flags & (32 /* Class */ | 64 /* Interface */ | 524288 /* TypeAlias */)) {
-                    typeParameterNodes = ts.createNodeArray(ts.map(getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol), function (tp) { return typeParameterToDeclaration(tp, context); }));
-                }
-                return typeParameterNodes;
+            function visitTypeReference(type) {
+                visitType(type.target);
+                ts.forEach(getTypeArguments(type), visitType);
             }
-            function lookupTypeParameterNodes(chain, index, context) {
-                ts.Debug.assert(chain && 0 <= index && index < chain.length);
-                var symbol = chain[index];
-                var symbolId = "" + getSymbolId(symbol);
-                if (context.typeParameterSymbolList && context.typeParameterSymbolList.get(symbolId)) {
-                    return undefined;
+            function visitTypeParameter(type) {
+                visitType(getConstraintOfTypeParameter(type));
+            }
+            function visitUnionOrIntersectionType(type) {
+                ts.forEach(type.types, visitType);
+            }
+            function visitIndexType(type) {
+                visitType(type.type);
+            }
+            function visitIndexedAccessType(type) {
+                visitType(type.objectType);
+                visitType(type.indexType);
+                visitType(type.constraint);
+            }
+            function visitMappedType(type) {
+                visitType(type.typeParameter);
+                visitType(type.constraintType);
+                visitType(type.templateType);
+                visitType(type.modifiersType);
+            }
+            function visitSignature(signature) {
+                var typePredicate = getTypePredicateOfSignature(signature);
+                if (typePredicate) {
+                    visitType(typePredicate.type);
                 }
-                (context.typeParameterSymbolList || (context.typeParameterSymbolList = ts.createMap())).set(symbolId, true);
-                var typeParameterNodes;
-                if (context.flags & 512 /* WriteTypeParametersInQualifiedName */ && index < (chain.length - 1)) {
-                    var parentSymbol = symbol;
-                    var nextSymbol_1 = chain[index + 1];
-                    if (ts.getCheckFlags(nextSymbol_1) & 1 /* Instantiated */) {
-                        var params = getTypeParametersOfClassOrInterface(parentSymbol.flags & 2097152 /* Alias */ ? resolveAlias(parentSymbol) : parentSymbol);
-                        typeParameterNodes = mapToTypeNodes(ts.map(params, function (t) { return getMappedType(t, nextSymbol_1.mapper); }), context);
-                    }
-                    else {
-                        typeParameterNodes = typeParametersToTypeParameterDeclarations(symbol, context);
-                    }
+                ts.forEach(signature.typeParameters, visitType);
+                for (var _i = 0, _a = signature.parameters; _i < _a.length; _i++) {
+                    var parameter = _a[_i];
+                    visitSymbol(parameter);
                 }
-                return typeParameterNodes;
+                visitType(getRestTypeOfSignature(signature));
+                visitType(getReturnTypeOfSignature(signature));
             }
-            /**
-             * Given A[B][C][D], finds A[B]
-             */
-            function getTopmostIndexedAccessType(top) {
-                if (ts.isIndexedAccessTypeNode(top.objectType)) {
-                    return getTopmostIndexedAccessType(top.objectType);
+            function visitInterfaceType(interfaceT) {
+                visitObjectType(interfaceT);
+                ts.forEach(interfaceT.typeParameters, visitType);
+                ts.forEach(getBaseTypes(interfaceT), visitType);
+                visitType(interfaceT.thisType);
+            }
+            function visitObjectType(type) {
+                var stringIndexType = getIndexTypeOfStructuredType(type, 0 /* String */);
+                visitType(stringIndexType);
+                var numberIndexType = getIndexTypeOfStructuredType(type, 1 /* Number */);
+                visitType(numberIndexType);
+                // The two checks above *should* have already resolved the type (if needed), so this should be cached
+                var resolved = resolveStructuredTypeMembers(type);
+                for (var _i = 0, _a = resolved.callSignatures; _i < _a.length; _i++) {
+                    var signature = _a[_i];
+                    visitSignature(signature);
+                }
+                for (var _b = 0, _c = resolved.constructSignatures; _b < _c.length; _b++) {
+                    var signature = _c[_b];
+                    visitSignature(signature);
+                }
+                for (var _d = 0, _e = resolved.properties; _d < _e.length; _d++) {
+                    var p = _e[_d];
+                    visitSymbol(p);
                 }
-                return top;
             }
-            function getSpecifierForModuleSymbol(symbol, context) {
-                var file = ts.getDeclarationOfKind(symbol, 290 /* SourceFile */);
-                if (!file) {
-                    var equivalentFileSymbol = ts.firstDefined(symbol.declarations, function (d) { return getFileSymbolIfFileSymbolExportEqualsContainer(d, symbol); });
-                    if (equivalentFileSymbol) {
-                        file = ts.getDeclarationOfKind(equivalentFileSymbol, 290 /* SourceFile */);
-                    }
+            function visitSymbol(symbol) {
+                if (!symbol) {
+                    return false;
                 }
-                if (file && file.moduleName !== undefined) {
-                    // Use the amd name if it is available
-                    return file.moduleName;
+                var symbolId = ts.getSymbolId(symbol);
+                if (visitedSymbols[symbolId]) {
+                    return false;
                 }
-                if (!file) {
-                    if (context.tracker.trackReferencedAmbientModule) {
-                        var ambientDecls = ts.filter(symbol.declarations, ts.isAmbientModule);
-                        if (ts.length(ambientDecls)) {
-                            for (var _i = 0, ambientDecls_1 = ambientDecls; _i < ambientDecls_1.length; _i++) {
-                                var decl = ambientDecls_1[_i];
-                                context.tracker.trackReferencedAmbientModule(decl, symbol);
-                            }
-                        }
-                    }
-                    if (ambientModuleSymbolRegex.test(symbol.escapedName)) {
-                        return symbol.escapedName.substring(1, symbol.escapedName.length - 1);
-                    }
+                visitedSymbols[symbolId] = symbol;
+                if (!accept(symbol)) {
+                    return true;
                 }
-                if (!context.enclosingDeclaration || !context.tracker.moduleResolverHost) {
-                    // If there's no context declaration, we can't lookup a non-ambient specifier, so we just use the symbol name
-                    if (ambientModuleSymbolRegex.test(symbol.escapedName)) {
-                        return symbol.escapedName.substring(1, symbol.escapedName.length - 1);
+                var t = getTypeOfSymbol(symbol);
+                visitType(t); // Should handle members on classes and such
+                if (symbol.exports) {
+                    symbol.exports.forEach(visitSymbol);
+                }
+                ts.forEach(symbol.declarations, function (d) {
+                    // Type queries are too far resolved when we just visit the symbol's type
+                    //  (their type resolved directly to the member deeply referenced)
+                    // So to get the intervening symbols, we need to check if there's a type
+                    // query node on any of the symbol's declarations and get symbols there
+                    if (d.type && d.type.kind === 176 /* TypeQuery */) {
+                        var query = d.type;
+                        var entity = getResolvedSymbol(getFirstIdentifier(query.exprName));
+                        visitSymbol(entity);
                     }
-                    return ts.getSourceFileOfNode(ts.getNonAugmentationDeclaration(symbol)).fileName; // A resolver may not be provided for baselines and errors - in those cases we use the fileName in full
+                });
+                return false;
+            }
+        }
+    }
+    ts.createGetSymbolWalker = createGetSymbolWalker;
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var ambientModuleSymbolRegex = /^".+"$/;
+    var anon = "(anonymous)";
+    var nextSymbolId = 1;
+    var nextNodeId = 1;
+    var nextMergeId = 1;
+    var nextFlowId = 1;
+    var IterationUse;
+    (function (IterationUse) {
+        IterationUse[IterationUse["AllowsSyncIterablesFlag"] = 1] = "AllowsSyncIterablesFlag";
+        IterationUse[IterationUse["AllowsAsyncIterablesFlag"] = 2] = "AllowsAsyncIterablesFlag";
+        IterationUse[IterationUse["AllowsStringInputFlag"] = 4] = "AllowsStringInputFlag";
+        IterationUse[IterationUse["ForOfFlag"] = 8] = "ForOfFlag";
+        IterationUse[IterationUse["YieldStarFlag"] = 16] = "YieldStarFlag";
+        IterationUse[IterationUse["SpreadFlag"] = 32] = "SpreadFlag";
+        IterationUse[IterationUse["DestructuringFlag"] = 64] = "DestructuringFlag";
+        IterationUse[IterationUse["PossiblyOutOfBounds"] = 128] = "PossiblyOutOfBounds";
+        // Spread, Destructuring, Array element assignment
+        IterationUse[IterationUse["Element"] = 1] = "Element";
+        IterationUse[IterationUse["Spread"] = 33] = "Spread";
+        IterationUse[IterationUse["Destructuring"] = 65] = "Destructuring";
+        IterationUse[IterationUse["ForOf"] = 13] = "ForOf";
+        IterationUse[IterationUse["ForAwaitOf"] = 15] = "ForAwaitOf";
+        IterationUse[IterationUse["YieldStar"] = 17] = "YieldStar";
+        IterationUse[IterationUse["AsyncYieldStar"] = 19] = "AsyncYieldStar";
+        IterationUse[IterationUse["GeneratorReturnType"] = 1] = "GeneratorReturnType";
+        IterationUse[IterationUse["AsyncGeneratorReturnType"] = 2] = "AsyncGeneratorReturnType";
+    })(IterationUse || (IterationUse = {}));
+    var IterationTypeKind;
+    (function (IterationTypeKind) {
+        IterationTypeKind[IterationTypeKind["Yield"] = 0] = "Yield";
+        IterationTypeKind[IterationTypeKind["Return"] = 1] = "Return";
+        IterationTypeKind[IterationTypeKind["Next"] = 2] = "Next";
+    })(IterationTypeKind || (IterationTypeKind = {}));
+    var WideningKind;
+    (function (WideningKind) {
+        WideningKind[WideningKind["Normal"] = 0] = "Normal";
+        WideningKind[WideningKind["FunctionReturn"] = 1] = "FunctionReturn";
+        WideningKind[WideningKind["GeneratorNext"] = 2] = "GeneratorNext";
+        WideningKind[WideningKind["GeneratorYield"] = 3] = "GeneratorYield";
+    })(WideningKind || (WideningKind = {}));
+    var TypeFacts;
+    (function (TypeFacts) {
+        TypeFacts[TypeFacts["None"] = 0] = "None";
+        TypeFacts[TypeFacts["TypeofEQString"] = 1] = "TypeofEQString";
+        TypeFacts[TypeFacts["TypeofEQNumber"] = 2] = "TypeofEQNumber";
+        TypeFacts[TypeFacts["TypeofEQBigInt"] = 4] = "TypeofEQBigInt";
+        TypeFacts[TypeFacts["TypeofEQBoolean"] = 8] = "TypeofEQBoolean";
+        TypeFacts[TypeFacts["TypeofEQSymbol"] = 16] = "TypeofEQSymbol";
+        TypeFacts[TypeFacts["TypeofEQObject"] = 32] = "TypeofEQObject";
+        TypeFacts[TypeFacts["TypeofEQFunction"] = 64] = "TypeofEQFunction";
+        TypeFacts[TypeFacts["TypeofEQHostObject"] = 128] = "TypeofEQHostObject";
+        TypeFacts[TypeFacts["TypeofNEString"] = 256] = "TypeofNEString";
+        TypeFacts[TypeFacts["TypeofNENumber"] = 512] = "TypeofNENumber";
+        TypeFacts[TypeFacts["TypeofNEBigInt"] = 1024] = "TypeofNEBigInt";
+        TypeFacts[TypeFacts["TypeofNEBoolean"] = 2048] = "TypeofNEBoolean";
+        TypeFacts[TypeFacts["TypeofNESymbol"] = 4096] = "TypeofNESymbol";
+        TypeFacts[TypeFacts["TypeofNEObject"] = 8192] = "TypeofNEObject";
+        TypeFacts[TypeFacts["TypeofNEFunction"] = 16384] = "TypeofNEFunction";
+        TypeFacts[TypeFacts["TypeofNEHostObject"] = 32768] = "TypeofNEHostObject";
+        TypeFacts[TypeFacts["EQUndefined"] = 65536] = "EQUndefined";
+        TypeFacts[TypeFacts["EQNull"] = 131072] = "EQNull";
+        TypeFacts[TypeFacts["EQUndefinedOrNull"] = 262144] = "EQUndefinedOrNull";
+        TypeFacts[TypeFacts["NEUndefined"] = 524288] = "NEUndefined";
+        TypeFacts[TypeFacts["NENull"] = 1048576] = "NENull";
+        TypeFacts[TypeFacts["NEUndefinedOrNull"] = 2097152] = "NEUndefinedOrNull";
+        TypeFacts[TypeFacts["Truthy"] = 4194304] = "Truthy";
+        TypeFacts[TypeFacts["Falsy"] = 8388608] = "Falsy";
+        TypeFacts[TypeFacts["All"] = 16777215] = "All";
+        // The following members encode facts about particular kinds of types for use in the getTypeFacts function.
+        // The presence of a particular fact means that the given test is true for some (and possibly all) values
+        // of that kind of type.
+        TypeFacts[TypeFacts["BaseStringStrictFacts"] = 3735041] = "BaseStringStrictFacts";
+        TypeFacts[TypeFacts["BaseStringFacts"] = 12582401] = "BaseStringFacts";
+        TypeFacts[TypeFacts["StringStrictFacts"] = 16317953] = "StringStrictFacts";
+        TypeFacts[TypeFacts["StringFacts"] = 16776705] = "StringFacts";
+        TypeFacts[TypeFacts["EmptyStringStrictFacts"] = 12123649] = "EmptyStringStrictFacts";
+        TypeFacts[TypeFacts["EmptyStringFacts"] = 12582401] = "EmptyStringFacts";
+        TypeFacts[TypeFacts["NonEmptyStringStrictFacts"] = 7929345] = "NonEmptyStringStrictFacts";
+        TypeFacts[TypeFacts["NonEmptyStringFacts"] = 16776705] = "NonEmptyStringFacts";
+        TypeFacts[TypeFacts["BaseNumberStrictFacts"] = 3734786] = "BaseNumberStrictFacts";
+        TypeFacts[TypeFacts["BaseNumberFacts"] = 12582146] = "BaseNumberFacts";
+        TypeFacts[TypeFacts["NumberStrictFacts"] = 16317698] = "NumberStrictFacts";
+        TypeFacts[TypeFacts["NumberFacts"] = 16776450] = "NumberFacts";
+        TypeFacts[TypeFacts["ZeroNumberStrictFacts"] = 12123394] = "ZeroNumberStrictFacts";
+        TypeFacts[TypeFacts["ZeroNumberFacts"] = 12582146] = "ZeroNumberFacts";
+        TypeFacts[TypeFacts["NonZeroNumberStrictFacts"] = 7929090] = "NonZeroNumberStrictFacts";
+        TypeFacts[TypeFacts["NonZeroNumberFacts"] = 16776450] = "NonZeroNumberFacts";
+        TypeFacts[TypeFacts["BaseBigIntStrictFacts"] = 3734276] = "BaseBigIntStrictFacts";
+        TypeFacts[TypeFacts["BaseBigIntFacts"] = 12581636] = "BaseBigIntFacts";
+        TypeFacts[TypeFacts["BigIntStrictFacts"] = 16317188] = "BigIntStrictFacts";
+        TypeFacts[TypeFacts["BigIntFacts"] = 16775940] = "BigIntFacts";
+        TypeFacts[TypeFacts["ZeroBigIntStrictFacts"] = 12122884] = "ZeroBigIntStrictFacts";
+        TypeFacts[TypeFacts["ZeroBigIntFacts"] = 12581636] = "ZeroBigIntFacts";
+        TypeFacts[TypeFacts["NonZeroBigIntStrictFacts"] = 7928580] = "NonZeroBigIntStrictFacts";
+        TypeFacts[TypeFacts["NonZeroBigIntFacts"] = 16775940] = "NonZeroBigIntFacts";
+        TypeFacts[TypeFacts["BaseBooleanStrictFacts"] = 3733256] = "BaseBooleanStrictFacts";
+        TypeFacts[TypeFacts["BaseBooleanFacts"] = 12580616] = "BaseBooleanFacts";
+        TypeFacts[TypeFacts["BooleanStrictFacts"] = 16316168] = "BooleanStrictFacts";
+        TypeFacts[TypeFacts["BooleanFacts"] = 16774920] = "BooleanFacts";
+        TypeFacts[TypeFacts["FalseStrictFacts"] = 12121864] = "FalseStrictFacts";
+        TypeFacts[TypeFacts["FalseFacts"] = 12580616] = "FalseFacts";
+        TypeFacts[TypeFacts["TrueStrictFacts"] = 7927560] = "TrueStrictFacts";
+        TypeFacts[TypeFacts["TrueFacts"] = 16774920] = "TrueFacts";
+        TypeFacts[TypeFacts["SymbolStrictFacts"] = 7925520] = "SymbolStrictFacts";
+        TypeFacts[TypeFacts["SymbolFacts"] = 16772880] = "SymbolFacts";
+        TypeFacts[TypeFacts["ObjectStrictFacts"] = 7888800] = "ObjectStrictFacts";
+        TypeFacts[TypeFacts["ObjectFacts"] = 16736160] = "ObjectFacts";
+        TypeFacts[TypeFacts["FunctionStrictFacts"] = 7880640] = "FunctionStrictFacts";
+        TypeFacts[TypeFacts["FunctionFacts"] = 16728000] = "FunctionFacts";
+        TypeFacts[TypeFacts["UndefinedFacts"] = 9830144] = "UndefinedFacts";
+        TypeFacts[TypeFacts["NullFacts"] = 9363232] = "NullFacts";
+        TypeFacts[TypeFacts["EmptyObjectStrictFacts"] = 16318463] = "EmptyObjectStrictFacts";
+        TypeFacts[TypeFacts["AllTypeofNE"] = 556800] = "AllTypeofNE";
+        TypeFacts[TypeFacts["EmptyObjectFacts"] = 16777215] = "EmptyObjectFacts";
+    })(TypeFacts || (TypeFacts = {}));
+    var typeofEQFacts = new ts.Map(ts.getEntries({
+        string: 1 /* TypeofEQString */,
+        number: 2 /* TypeofEQNumber */,
+        bigint: 4 /* TypeofEQBigInt */,
+        boolean: 8 /* TypeofEQBoolean */,
+        symbol: 16 /* TypeofEQSymbol */,
+        undefined: 65536 /* EQUndefined */,
+        object: 32 /* TypeofEQObject */,
+        function: 64 /* TypeofEQFunction */
+    }));
+    var typeofNEFacts = new ts.Map(ts.getEntries({
+        string: 256 /* TypeofNEString */,
+        number: 512 /* TypeofNENumber */,
+        bigint: 1024 /* TypeofNEBigInt */,
+        boolean: 2048 /* TypeofNEBoolean */,
+        symbol: 4096 /* TypeofNESymbol */,
+        undefined: 524288 /* NEUndefined */,
+        object: 8192 /* TypeofNEObject */,
+        function: 16384 /* TypeofNEFunction */
+    }));
+    var TypeSystemPropertyName;
+    (function (TypeSystemPropertyName) {
+        TypeSystemPropertyName[TypeSystemPropertyName["Type"] = 0] = "Type";
+        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedBaseConstructorType"] = 1] = "ResolvedBaseConstructorType";
+        TypeSystemPropertyName[TypeSystemPropertyName["DeclaredType"] = 2] = "DeclaredType";
+        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedReturnType"] = 3] = "ResolvedReturnType";
+        TypeSystemPropertyName[TypeSystemPropertyName["ImmediateBaseConstraint"] = 4] = "ImmediateBaseConstraint";
+        TypeSystemPropertyName[TypeSystemPropertyName["EnumTagType"] = 5] = "EnumTagType";
+        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedTypeArguments"] = 6] = "ResolvedTypeArguments";
+        TypeSystemPropertyName[TypeSystemPropertyName["ResolvedBaseTypes"] = 7] = "ResolvedBaseTypes";
+    })(TypeSystemPropertyName || (TypeSystemPropertyName = {}));
+    var CheckMode;
+    (function (CheckMode) {
+        CheckMode[CheckMode["Normal"] = 0] = "Normal";
+        CheckMode[CheckMode["Contextual"] = 1] = "Contextual";
+        CheckMode[CheckMode["Inferential"] = 2] = "Inferential";
+        CheckMode[CheckMode["SkipContextSensitive"] = 4] = "SkipContextSensitive";
+        CheckMode[CheckMode["SkipGenericFunctions"] = 8] = "SkipGenericFunctions";
+        CheckMode[CheckMode["IsForSignatureHelp"] = 16] = "IsForSignatureHelp";
+    })(CheckMode || (CheckMode = {}));
+    var AccessFlags;
+    (function (AccessFlags) {
+        AccessFlags[AccessFlags["None"] = 0] = "None";
+        AccessFlags[AccessFlags["NoIndexSignatures"] = 1] = "NoIndexSignatures";
+        AccessFlags[AccessFlags["Writing"] = 2] = "Writing";
+        AccessFlags[AccessFlags["CacheSymbol"] = 4] = "CacheSymbol";
+        AccessFlags[AccessFlags["NoTupleBoundsCheck"] = 8] = "NoTupleBoundsCheck";
+        AccessFlags[AccessFlags["ExpressionPosition"] = 16] = "ExpressionPosition";
+    })(AccessFlags || (AccessFlags = {}));
+    var SignatureCheckMode;
+    (function (SignatureCheckMode) {
+        SignatureCheckMode[SignatureCheckMode["BivariantCallback"] = 1] = "BivariantCallback";
+        SignatureCheckMode[SignatureCheckMode["StrictCallback"] = 2] = "StrictCallback";
+        SignatureCheckMode[SignatureCheckMode["IgnoreReturnTypes"] = 4] = "IgnoreReturnTypes";
+        SignatureCheckMode[SignatureCheckMode["StrictArity"] = 8] = "StrictArity";
+        SignatureCheckMode[SignatureCheckMode["Callback"] = 3] = "Callback";
+    })(SignatureCheckMode || (SignatureCheckMode = {}));
+    var IntersectionState;
+    (function (IntersectionState) {
+        IntersectionState[IntersectionState["None"] = 0] = "None";
+        IntersectionState[IntersectionState["Source"] = 1] = "Source";
+        IntersectionState[IntersectionState["Target"] = 2] = "Target";
+        IntersectionState[IntersectionState["PropertyCheck"] = 4] = "PropertyCheck";
+        IntersectionState[IntersectionState["InPropertyCheck"] = 8] = "InPropertyCheck";
+    })(IntersectionState || (IntersectionState = {}));
+    var MappedTypeModifiers;
+    (function (MappedTypeModifiers) {
+        MappedTypeModifiers[MappedTypeModifiers["IncludeReadonly"] = 1] = "IncludeReadonly";
+        MappedTypeModifiers[MappedTypeModifiers["ExcludeReadonly"] = 2] = "ExcludeReadonly";
+        MappedTypeModifiers[MappedTypeModifiers["IncludeOptional"] = 4] = "IncludeOptional";
+        MappedTypeModifiers[MappedTypeModifiers["ExcludeOptional"] = 8] = "ExcludeOptional";
+    })(MappedTypeModifiers || (MappedTypeModifiers = {}));
+    var ExpandingFlags;
+    (function (ExpandingFlags) {
+        ExpandingFlags[ExpandingFlags["None"] = 0] = "None";
+        ExpandingFlags[ExpandingFlags["Source"] = 1] = "Source";
+        ExpandingFlags[ExpandingFlags["Target"] = 2] = "Target";
+        ExpandingFlags[ExpandingFlags["Both"] = 3] = "Both";
+    })(ExpandingFlags || (ExpandingFlags = {}));
+    var MembersOrExportsResolutionKind;
+    (function (MembersOrExportsResolutionKind) {
+        MembersOrExportsResolutionKind["resolvedExports"] = "resolvedExports";
+        MembersOrExportsResolutionKind["resolvedMembers"] = "resolvedMembers";
+    })(MembersOrExportsResolutionKind || (MembersOrExportsResolutionKind = {}));
+    var UnusedKind;
+    (function (UnusedKind) {
+        UnusedKind[UnusedKind["Local"] = 0] = "Local";
+        UnusedKind[UnusedKind["Parameter"] = 1] = "Parameter";
+    })(UnusedKind || (UnusedKind = {}));
+    var isNotOverloadAndNotAccessor = ts.and(isNotOverload, isNotAccessor);
+    var DeclarationMeaning;
+    (function (DeclarationMeaning) {
+        DeclarationMeaning[DeclarationMeaning["GetAccessor"] = 1] = "GetAccessor";
+        DeclarationMeaning[DeclarationMeaning["SetAccessor"] = 2] = "SetAccessor";
+        DeclarationMeaning[DeclarationMeaning["PropertyAssignment"] = 4] = "PropertyAssignment";
+        DeclarationMeaning[DeclarationMeaning["Method"] = 8] = "Method";
+        DeclarationMeaning[DeclarationMeaning["GetOrSetAccessor"] = 3] = "GetOrSetAccessor";
+        DeclarationMeaning[DeclarationMeaning["PropertyAssignmentOrMethod"] = 12] = "PropertyAssignmentOrMethod";
+    })(DeclarationMeaning || (DeclarationMeaning = {}));
+    var DeclarationSpaces;
+    (function (DeclarationSpaces) {
+        DeclarationSpaces[DeclarationSpaces["None"] = 0] = "None";
+        DeclarationSpaces[DeclarationSpaces["ExportValue"] = 1] = "ExportValue";
+        DeclarationSpaces[DeclarationSpaces["ExportType"] = 2] = "ExportType";
+        DeclarationSpaces[DeclarationSpaces["ExportNamespace"] = 4] = "ExportNamespace";
+    })(DeclarationSpaces || (DeclarationSpaces = {}));
+    var MinArgumentCountFlags;
+    (function (MinArgumentCountFlags) {
+        MinArgumentCountFlags[MinArgumentCountFlags["None"] = 0] = "None";
+        MinArgumentCountFlags[MinArgumentCountFlags["StrongArityForUntypedJS"] = 1] = "StrongArityForUntypedJS";
+        MinArgumentCountFlags[MinArgumentCountFlags["VoidIsNonOptional"] = 2] = "VoidIsNonOptional";
+    })(MinArgumentCountFlags || (MinArgumentCountFlags = {}));
+    var IntrinsicTypeKind;
+    (function (IntrinsicTypeKind) {
+        IntrinsicTypeKind[IntrinsicTypeKind["Uppercase"] = 0] = "Uppercase";
+        IntrinsicTypeKind[IntrinsicTypeKind["Lowercase"] = 1] = "Lowercase";
+        IntrinsicTypeKind[IntrinsicTypeKind["Capitalize"] = 2] = "Capitalize";
+        IntrinsicTypeKind[IntrinsicTypeKind["Uncapitalize"] = 3] = "Uncapitalize";
+    })(IntrinsicTypeKind || (IntrinsicTypeKind = {}));
+    var intrinsicTypeKinds = new ts.Map(ts.getEntries({
+        Uppercase: 0 /* Uppercase */,
+        Lowercase: 1 /* Lowercase */,
+        Capitalize: 2 /* Capitalize */,
+        Uncapitalize: 3 /* Uncapitalize */
+    }));
+    function SymbolLinks() {
+    }
+    function NodeLinks() {
+        this.flags = 0;
+    }
+    function getNodeId(node) {
+        if (!node.id) {
+            node.id = nextNodeId;
+            nextNodeId++;
+        }
+        return node.id;
+    }
+    ts.getNodeId = getNodeId;
+    function getSymbolId(symbol) {
+        if (!symbol.id) {
+            symbol.id = nextSymbolId;
+            nextSymbolId++;
+        }
+        return symbol.id;
+    }
+    ts.getSymbolId = getSymbolId;
+    function isInstantiatedModule(node, preserveConstEnums) {
+        var moduleState = ts.getModuleInstanceState(node);
+        return moduleState === 1 /* Instantiated */ ||
+            (preserveConstEnums && moduleState === 2 /* ConstEnumOnly */);
+    }
+    ts.isInstantiatedModule = isInstantiatedModule;
+    function createTypeChecker(host, produceDiagnostics) {
+        var getPackagesSet = ts.memoize(function () {
+            var set = new ts.Set();
+            host.getSourceFiles().forEach(function (sf) {
+                if (!sf.resolvedModules)
+                    return;
+                ts.forEachEntry(sf.resolvedModules, function (r) {
+                    if (r && r.packageId)
+                        set.add(r.packageId.name);
+                });
+            });
+            return set;
+        });
+        // Cancellation that controls whether or not we can cancel in the middle of type checking.
+        // In general cancelling is *not* safe for the type checker.  We might be in the middle of
+        // computing something, and we will leave our internals in an inconsistent state.  Callers
+        // who set the cancellation token should catch if a cancellation exception occurs, and
+        // should throw away and create a new TypeChecker.
+        //
+        // Currently we only support setting the cancellation token when getting diagnostics.  This
+        // is because diagnostics can be quite expensive, and we want to allow hosts to bail out if
+        // they no longer need the information (for example, if the user started editing again).
+        var cancellationToken;
+        var requestedExternalEmitHelpers;
+        var externalHelpersModule;
+        var Symbol = ts.objectAllocator.getSymbolConstructor();
+        var Type = ts.objectAllocator.getTypeConstructor();
+        var Signature = ts.objectAllocator.getSignatureConstructor();
+        var typeCount = 0;
+        var symbolCount = 0;
+        var enumCount = 0;
+        var totalInstantiationCount = 0;
+        var instantiationCount = 0;
+        var instantiationDepth = 0;
+        var constraintDepth = 0;
+        var currentNode;
+        var typeCatalog = []; // NB: id is index + 1
+        var emptySymbols = ts.createSymbolTable();
+        var arrayVariances = [1 /* Covariant */];
+        var compilerOptions = host.getCompilerOptions();
+        var languageVersion = ts.getEmitScriptTarget(compilerOptions);
+        var moduleKind = ts.getEmitModuleKind(compilerOptions);
+        var allowSyntheticDefaultImports = ts.getAllowSyntheticDefaultImports(compilerOptions);
+        var strictNullChecks = ts.getStrictOptionValue(compilerOptions, "strictNullChecks");
+        var strictFunctionTypes = ts.getStrictOptionValue(compilerOptions, "strictFunctionTypes");
+        var strictBindCallApply = ts.getStrictOptionValue(compilerOptions, "strictBindCallApply");
+        var strictPropertyInitialization = ts.getStrictOptionValue(compilerOptions, "strictPropertyInitialization");
+        var noImplicitAny = ts.getStrictOptionValue(compilerOptions, "noImplicitAny");
+        var noImplicitThis = ts.getStrictOptionValue(compilerOptions, "noImplicitThis");
+        var keyofStringsOnly = !!compilerOptions.keyofStringsOnly;
+        var freshObjectLiteralFlag = compilerOptions.suppressExcessPropertyErrors ? 0 : 32768 /* FreshLiteral */;
+        var emitResolver = createResolver();
+        var nodeBuilder = createNodeBuilder();
+        var globals = ts.createSymbolTable();
+        var undefinedSymbol = createSymbol(4 /* Property */, "undefined");
+        undefinedSymbol.declarations = [];
+        var globalThisSymbol = createSymbol(1536 /* Module */, "globalThis", 8 /* Readonly */);
+        globalThisSymbol.exports = globals;
+        globalThisSymbol.declarations = [];
+        globals.set(globalThisSymbol.escapedName, globalThisSymbol);
+        var argumentsSymbol = createSymbol(4 /* Property */, "arguments");
+        var requireSymbol = createSymbol(4 /* Property */, "require");
+        /** This will be set during calls to `getResolvedSignature` where services determines an apparent number of arguments greater than what is actually provided. */
+        var apparentArgumentCount;
+        // for public members that accept a Node or one of its subtypes, we must guard against
+        // synthetic nodes created during transformations by calling `getParseTreeNode`.
+        // for most of these, we perform the guard only on `checker` to avoid any possible
+        // extra cost of calling `getParseTreeNode` when calling these functions from inside the
+        // checker.
+        var checker = {
+            getNodeCount: function () { return ts.sum(host.getSourceFiles(), "nodeCount"); },
+            getIdentifierCount: function () { return ts.sum(host.getSourceFiles(), "identifierCount"); },
+            getSymbolCount: function () { return ts.sum(host.getSourceFiles(), "symbolCount") + symbolCount; },
+            getTypeCatalog: function () { return typeCatalog; },
+            getTypeCount: function () { return typeCount; },
+            getInstantiationCount: function () { return totalInstantiationCount; },
+            getRelationCacheSizes: function () { return ({
+                assignable: assignableRelation.size,
+                identity: identityRelation.size,
+                subtype: subtypeRelation.size,
+                strictSubtype: strictSubtypeRelation.size,
+            }); },
+            isUndefinedSymbol: function (symbol) { return symbol === undefinedSymbol; },
+            isArgumentsSymbol: function (symbol) { return symbol === argumentsSymbol; },
+            isUnknownSymbol: function (symbol) { return symbol === unknownSymbol; },
+            getMergedSymbol: getMergedSymbol,
+            getDiagnostics: getDiagnostics,
+            getGlobalDiagnostics: getGlobalDiagnostics,
+            getRecursionIdentity: getRecursionIdentity,
+            getTypeOfSymbolAtLocation: function (symbol, locationIn) {
+                var location = ts.getParseTreeNode(locationIn);
+                return location ? getTypeOfSymbolAtLocation(symbol, location) : errorType;
+            },
+            getSymbolsOfParameterPropertyDeclaration: function (parameterIn, parameterName) {
+                var parameter = ts.getParseTreeNode(parameterIn, ts.isParameter);
+                if (parameter === undefined)
+                    return ts.Debug.fail("Cannot get symbols of a synthetic parameter that cannot be resolved to a parse-tree node.");
+                return getSymbolsOfParameterPropertyDeclaration(parameter, ts.escapeLeadingUnderscores(parameterName));
+            },
+            getDeclaredTypeOfSymbol: getDeclaredTypeOfSymbol,
+            getPropertiesOfType: getPropertiesOfType,
+            getPropertyOfType: function (type, name) { return getPropertyOfType(type, ts.escapeLeadingUnderscores(name)); },
+            getPrivateIdentifierPropertyOfType: function (leftType, name, location) {
+                var node = ts.getParseTreeNode(location);
+                if (!node) {
+                    return undefined;
                 }
-                var contextFile = ts.getSourceFileOfNode(ts.getOriginalNode(context.enclosingDeclaration));
-                var links = getSymbolLinks(symbol);
-                var specifier = links.specifierCache && links.specifierCache.get(contextFile.path);
-                if (!specifier) {
-                    var isBundle_1 = (compilerOptions.out || compilerOptions.outFile);
-                    // For declaration bundles, we need to generate absolute paths relative to the common source dir for imports,
-                    // just like how the declaration emitter does for the ambient module declarations - we can easily accomplish this
-                    // using the `baseUrl` compiler option (which we would otherwise never use in declaration emit) and a non-relative
-                    // specifier preference
-                    var moduleResolverHost = context.tracker.moduleResolverHost;
-                    var specifierCompilerOptions = isBundle_1 ? __assign(__assign({}, compilerOptions), { baseUrl: moduleResolverHost.getCommonSourceDirectory() }) : compilerOptions;
-                    specifier = ts.first(ts.moduleSpecifiers.getModuleSpecifiers(symbol, specifierCompilerOptions, contextFile, moduleResolverHost, { importModuleSpecifierPreference: isBundle_1 ? "non-relative" : "relative" }));
-                    links.specifierCache = links.specifierCache || ts.createMap();
-                    links.specifierCache.set(contextFile.path, specifier);
+                var propName = ts.escapeLeadingUnderscores(name);
+                var lexicallyScopedIdentifier = lookupSymbolForPrivateIdentifierDeclaration(propName, node);
+                return lexicallyScopedIdentifier ? getPrivateIdentifierPropertyOfType(leftType, lexicallyScopedIdentifier) : undefined;
+            },
+            getTypeOfPropertyOfType: function (type, name) { return getTypeOfPropertyOfType(type, ts.escapeLeadingUnderscores(name)); },
+            getIndexInfoOfType: getIndexInfoOfType,
+            getSignaturesOfType: getSignaturesOfType,
+            getIndexTypeOfType: getIndexTypeOfType,
+            getBaseTypes: getBaseTypes,
+            getBaseTypeOfLiteralType: getBaseTypeOfLiteralType,
+            getWidenedType: getWidenedType,
+            getTypeFromTypeNode: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isTypeNode);
+                return node ? getTypeFromTypeNode(node) : errorType;
+            },
+            getParameterType: getTypeAtPosition,
+            getPromisedTypeOfPromise: getPromisedTypeOfPromise,
+            getAwaitedType: function (type) { return getAwaitedType(type); },
+            getReturnTypeOfSignature: getReturnTypeOfSignature,
+            isNullableType: isNullableType,
+            getNullableType: getNullableType,
+            getNonNullableType: getNonNullableType,
+            getNonOptionalType: removeOptionalTypeMarker,
+            getTypeArguments: getTypeArguments,
+            typeToTypeNode: nodeBuilder.typeToTypeNode,
+            indexInfoToIndexSignatureDeclaration: nodeBuilder.indexInfoToIndexSignatureDeclaration,
+            signatureToSignatureDeclaration: nodeBuilder.signatureToSignatureDeclaration,
+            symbolToEntityName: nodeBuilder.symbolToEntityName,
+            symbolToExpression: nodeBuilder.symbolToExpression,
+            symbolToTypeParameterDeclarations: nodeBuilder.symbolToTypeParameterDeclarations,
+            symbolToParameterDeclaration: nodeBuilder.symbolToParameterDeclaration,
+            typeParameterToDeclaration: nodeBuilder.typeParameterToDeclaration,
+            getSymbolsInScope: function (locationIn, meaning) {
+                var location = ts.getParseTreeNode(locationIn);
+                return location ? getSymbolsInScope(location, meaning) : [];
+            },
+            getSymbolAtLocation: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn);
+                // set ignoreErrors: true because any lookups invoked by the API shouldn't cause any new errors
+                return node ? getSymbolAtLocation(node, /*ignoreErrors*/ true) : undefined;
+            },
+            getShorthandAssignmentValueSymbol: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn);
+                return node ? getShorthandAssignmentValueSymbol(node) : undefined;
+            },
+            getExportSpecifierLocalTargetSymbol: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isExportSpecifier);
+                return node ? getExportSpecifierLocalTargetSymbol(node) : undefined;
+            },
+            getExportSymbolOfSymbol: function (symbol) {
+                return getMergedSymbol(symbol.exportSymbol || symbol);
+            },
+            getTypeAtLocation: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn);
+                return node ? getTypeOfNode(node) : errorType;
+            },
+            getTypeOfAssignmentPattern: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isAssignmentPattern);
+                return node && getTypeOfAssignmentPattern(node) || errorType;
+            },
+            getPropertySymbolOfDestructuringAssignment: function (locationIn) {
+                var location = ts.getParseTreeNode(locationIn, ts.isIdentifier);
+                return location ? getPropertySymbolOfDestructuringAssignment(location) : undefined;
+            },
+            signatureToString: function (signature, enclosingDeclaration, flags, kind) {
+                return signatureToString(signature, ts.getParseTreeNode(enclosingDeclaration), flags, kind);
+            },
+            typeToString: function (type, enclosingDeclaration, flags) {
+                return typeToString(type, ts.getParseTreeNode(enclosingDeclaration), flags);
+            },
+            symbolToString: function (symbol, enclosingDeclaration, meaning, flags) {
+                return symbolToString(symbol, ts.getParseTreeNode(enclosingDeclaration), meaning, flags);
+            },
+            typePredicateToString: function (predicate, enclosingDeclaration, flags) {
+                return typePredicateToString(predicate, ts.getParseTreeNode(enclosingDeclaration), flags);
+            },
+            writeSignature: function (signature, enclosingDeclaration, flags, kind, writer) {
+                return signatureToString(signature, ts.getParseTreeNode(enclosingDeclaration), flags, kind, writer);
+            },
+            writeType: function (type, enclosingDeclaration, flags, writer) {
+                return typeToString(type, ts.getParseTreeNode(enclosingDeclaration), flags, writer);
+            },
+            writeSymbol: function (symbol, enclosingDeclaration, meaning, flags, writer) {
+                return symbolToString(symbol, ts.getParseTreeNode(enclosingDeclaration), meaning, flags, writer);
+            },
+            writeTypePredicate: function (predicate, enclosingDeclaration, flags, writer) {
+                return typePredicateToString(predicate, ts.getParseTreeNode(enclosingDeclaration), flags, writer);
+            },
+            getAugmentedPropertiesOfType: getAugmentedPropertiesOfType,
+            getRootSymbols: getRootSymbols,
+            getSymbolOfExpando: getSymbolOfExpando,
+            getContextualType: function (nodeIn, contextFlags) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isExpression);
+                if (!node) {
+                    return undefined;
                 }
-                return specifier;
-            }
-            function symbolToTypeNode(symbol, context, meaning, overrideTypeArguments) {
-                var chain = lookupSymbolChain(symbol, context, meaning, !(context.flags & 16384 /* UseAliasDefinedOutsideCurrentScope */)); // If we're using aliases outside the current scope, dont bother with the module
-                var isTypeOf = meaning === 111551 /* Value */;
-                if (ts.some(chain[0].declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
-                    // module is root, must use `ImportTypeNode`
-                    var nonRootParts = chain.length > 1 ? createAccessFromSymbolChain(chain, chain.length - 1, 1) : undefined;
-                    var typeParameterNodes = overrideTypeArguments || lookupTypeParameterNodes(chain, 0, context);
-                    var specifier = getSpecifierForModuleSymbol(chain[0], context);
-                    if (!(context.flags & 67108864 /* AllowNodeModulesRelativePaths */) && ts.getEmitModuleResolutionKind(compilerOptions) === ts.ModuleResolutionKind.NodeJs && specifier.indexOf("/node_modules/") >= 0) {
-                        // If ultimately we can only name the symbol with a reference that dives into a `node_modules` folder, we should error
-                        // since declaration files with these kinds of references are liable to fail when published :(
-                        context.encounteredError = true;
-                        if (context.tracker.reportLikelyUnsafeImportRequiredError) {
-                            context.tracker.reportLikelyUnsafeImportRequiredError(specifier);
-                        }
-                    }
-                    var lit = ts.createLiteralTypeNode(ts.createLiteral(specifier));
-                    if (context.tracker.trackExternalModuleSymbolOfImportTypeNode)
-                        context.tracker.trackExternalModuleSymbolOfImportTypeNode(chain[0]);
-                    context.approximateLength += specifier.length + 10; // specifier + import("")
-                    if (!nonRootParts || ts.isEntityName(nonRootParts)) {
-                        if (nonRootParts) {
-                            var lastId = ts.isIdentifier(nonRootParts) ? nonRootParts : nonRootParts.right;
-                            lastId.typeArguments = undefined;
+                var containingCall = ts.findAncestor(node, ts.isCallLikeExpression);
+                var containingCallResolvedSignature = containingCall && getNodeLinks(containingCall).resolvedSignature;
+                if (contextFlags & 4 /* Completions */ && containingCall) {
+                    var toMarkSkip = node;
+                    do {
+                        getNodeLinks(toMarkSkip).skipDirectInference = true;
+                        toMarkSkip = toMarkSkip.parent;
+                    } while (toMarkSkip && toMarkSkip !== containingCall);
+                    getNodeLinks(containingCall).resolvedSignature = undefined;
+                }
+                var result = getContextualType(node, contextFlags);
+                if (contextFlags & 4 /* Completions */ && containingCall) {
+                    var toMarkSkip = node;
+                    do {
+                        getNodeLinks(toMarkSkip).skipDirectInference = undefined;
+                        toMarkSkip = toMarkSkip.parent;
+                    } while (toMarkSkip && toMarkSkip !== containingCall);
+                    getNodeLinks(containingCall).resolvedSignature = containingCallResolvedSignature;
+                }
+                return result;
+            },
+            getContextualTypeForObjectLiteralElement: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isObjectLiteralElementLike);
+                return node ? getContextualTypeForObjectLiteralElement(node) : undefined;
+            },
+            getContextualTypeForArgumentAtIndex: function (nodeIn, argIndex) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isCallLikeExpression);
+                return node && getContextualTypeForArgumentAtIndex(node, argIndex);
+            },
+            getContextualTypeForJsxAttribute: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isJsxAttributeLike);
+                return node && getContextualTypeForJsxAttribute(node);
+            },
+            isContextSensitive: isContextSensitive,
+            getFullyQualifiedName: getFullyQualifiedName,
+            getResolvedSignature: function (node, candidatesOutArray, argumentCount) {
+                return getResolvedSignatureWorker(node, candidatesOutArray, argumentCount, 0 /* Normal */);
+            },
+            getResolvedSignatureForSignatureHelp: function (node, candidatesOutArray, argumentCount) {
+                return getResolvedSignatureWorker(node, candidatesOutArray, argumentCount, 16 /* IsForSignatureHelp */);
+            },
+            getExpandedParameters: getExpandedParameters,
+            hasEffectiveRestParameter: hasEffectiveRestParameter,
+            getConstantValue: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, canHaveConstantValue);
+                return node ? getConstantValue(node) : undefined;
+            },
+            isValidPropertyAccess: function (nodeIn, propertyName) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isPropertyAccessOrQualifiedNameOrImportTypeNode);
+                return !!node && isValidPropertyAccess(node, ts.escapeLeadingUnderscores(propertyName));
+            },
+            isValidPropertyAccessForCompletions: function (nodeIn, type, property) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isPropertyAccessExpression);
+                return !!node && isValidPropertyAccessForCompletions(node, type, property);
+            },
+            getSignatureFromDeclaration: function (declarationIn) {
+                var declaration = ts.getParseTreeNode(declarationIn, ts.isFunctionLike);
+                return declaration ? getSignatureFromDeclaration(declaration) : undefined;
+            },
+            isImplementationOfOverload: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isFunctionLike);
+                return node ? isImplementationOfOverload(node) : undefined;
+            },
+            getImmediateAliasedSymbol: getImmediateAliasedSymbol,
+            getAliasedSymbol: resolveAlias,
+            getEmitResolver: getEmitResolver,
+            getExportsOfModule: getExportsOfModuleAsArray,
+            getExportsAndPropertiesOfModule: getExportsAndPropertiesOfModule,
+            getSymbolWalker: ts.createGetSymbolWalker(getRestTypeOfSignature, getTypePredicateOfSignature, getReturnTypeOfSignature, getBaseTypes, resolveStructuredTypeMembers, getTypeOfSymbol, getResolvedSymbol, getIndexTypeOfStructuredType, getConstraintOfTypeParameter, ts.getFirstIdentifier, getTypeArguments),
+            getAmbientModules: getAmbientModules,
+            getJsxIntrinsicTagNamesAt: getJsxIntrinsicTagNamesAt,
+            isOptionalParameter: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isParameter);
+                return node ? isOptionalParameter(node) : false;
+            },
+            tryGetMemberInModuleExports: function (name, symbol) { return tryGetMemberInModuleExports(ts.escapeLeadingUnderscores(name), symbol); },
+            tryGetMemberInModuleExportsAndProperties: function (name, symbol) { return tryGetMemberInModuleExportsAndProperties(ts.escapeLeadingUnderscores(name), symbol); },
+            tryFindAmbientModuleWithoutAugmentations: function (moduleName) {
+                // we deliberately exclude augmentations
+                // since we are only interested in declarations of the module itself
+                return tryFindAmbientModule(moduleName, /*withAugmentations*/ false);
+            },
+            getApparentType: getApparentType,
+            getUnionType: getUnionType,
+            isTypeAssignableTo: isTypeAssignableTo,
+            createAnonymousType: createAnonymousType,
+            createSignature: createSignature,
+            createSymbol: createSymbol,
+            createIndexInfo: createIndexInfo,
+            getAnyType: function () { return anyType; },
+            getStringType: function () { return stringType; },
+            getNumberType: function () { return numberType; },
+            createPromiseType: createPromiseType,
+            createArrayType: createArrayType,
+            getElementTypeOfArrayType: getElementTypeOfArrayType,
+            getBooleanType: function () { return booleanType; },
+            getFalseType: function (fresh) { return fresh ? falseType : regularFalseType; },
+            getTrueType: function (fresh) { return fresh ? trueType : regularTrueType; },
+            getVoidType: function () { return voidType; },
+            getUndefinedType: function () { return undefinedType; },
+            getNullType: function () { return nullType; },
+            getESSymbolType: function () { return esSymbolType; },
+            getNeverType: function () { return neverType; },
+            getOptionalType: function () { return optionalType; },
+            isSymbolAccessible: isSymbolAccessible,
+            isArrayType: isArrayType,
+            isTupleType: isTupleType,
+            isArrayLikeType: isArrayLikeType,
+            isTypeInvalidDueToUnionDiscriminant: isTypeInvalidDueToUnionDiscriminant,
+            getAllPossiblePropertiesOfTypes: getAllPossiblePropertiesOfTypes,
+            getSuggestedSymbolForNonexistentProperty: getSuggestedSymbolForNonexistentProperty,
+            getSuggestionForNonexistentProperty: getSuggestionForNonexistentProperty,
+            getSuggestedSymbolForNonexistentJSXAttribute: getSuggestedSymbolForNonexistentJSXAttribute,
+            getSuggestedSymbolForNonexistentSymbol: function (location, name, meaning) { return getSuggestedSymbolForNonexistentSymbol(location, ts.escapeLeadingUnderscores(name), meaning); },
+            getSuggestionForNonexistentSymbol: function (location, name, meaning) { return getSuggestionForNonexistentSymbol(location, ts.escapeLeadingUnderscores(name), meaning); },
+            getSuggestedSymbolForNonexistentModule: getSuggestedSymbolForNonexistentModule,
+            getSuggestionForNonexistentExport: getSuggestionForNonexistentExport,
+            getBaseConstraintOfType: getBaseConstraintOfType,
+            getDefaultFromTypeParameter: function (type) { return type && type.flags & 262144 /* TypeParameter */ ? getDefaultFromTypeParameter(type) : undefined; },
+            resolveName: function (name, location, meaning, excludeGlobals) {
+                return resolveName(location, ts.escapeLeadingUnderscores(name), meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false, excludeGlobals);
+            },
+            getJsxNamespace: function (n) { return ts.unescapeLeadingUnderscores(getJsxNamespace(n)); },
+            getAccessibleSymbolChain: getAccessibleSymbolChain,
+            getTypePredicateOfSignature: getTypePredicateOfSignature,
+            resolveExternalModuleName: function (moduleSpecifierIn) {
+                var moduleSpecifier = ts.getParseTreeNode(moduleSpecifierIn, ts.isExpression);
+                return moduleSpecifier && resolveExternalModuleName(moduleSpecifier, moduleSpecifier, /*ignoreErrors*/ true);
+            },
+            resolveExternalModuleSymbol: resolveExternalModuleSymbol,
+            tryGetThisTypeAt: function (nodeIn, includeGlobalThis) {
+                var node = ts.getParseTreeNode(nodeIn);
+                return node && tryGetThisTypeAt(node, includeGlobalThis);
+            },
+            getTypeArgumentConstraint: function (nodeIn) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isTypeNode);
+                return node && getTypeArgumentConstraint(node);
+            },
+            getSuggestionDiagnostics: function (fileIn, ct) {
+                var file = ts.getParseTreeNode(fileIn, ts.isSourceFile) || ts.Debug.fail("Could not determine parsed source file.");
+                if (ts.skipTypeChecking(file, compilerOptions, host)) {
+                    return ts.emptyArray;
+                }
+                var diagnostics;
+                try {
+                    // Record the cancellation token so it can be checked later on during checkSourceElement.
+                    // Do this in a finally block so we can ensure that it gets reset back to nothing after
+                    // this call is done.
+                    cancellationToken = ct;
+                    // Ensure file is type checked
+                    checkSourceFile(file);
+                    ts.Debug.assert(!!(getNodeLinks(file).flags & 1 /* TypeChecked */));
+                    diagnostics = ts.addRange(diagnostics, suggestionDiagnostics.getDiagnostics(file.fileName));
+                    checkUnusedIdentifiers(getPotentiallyUnusedIdentifiers(file), function (containingNode, kind, diag) {
+                        if (!ts.containsParseError(containingNode) && !unusedIsError(kind, !!(containingNode.flags & 8388608 /* Ambient */))) {
+                            (diagnostics || (diagnostics = [])).push(__assign(__assign({}, diag), { category: ts.DiagnosticCategory.Suggestion }));
                         }
-                        return ts.createImportTypeNode(lit, nonRootParts, typeParameterNodes, isTypeOf);
-                    }
-                    else {
-                        var splitNode = getTopmostIndexedAccessType(nonRootParts);
-                        var qualifier = splitNode.objectType.typeName;
-                        return ts.createIndexedAccessTypeNode(ts.createImportTypeNode(lit, qualifier, typeParameterNodes, isTypeOf), splitNode.indexType);
-                    }
+                    });
+                    return diagnostics || ts.emptyArray;
                 }
-                var entityName = createAccessFromSymbolChain(chain, chain.length - 1, 0);
-                if (ts.isIndexedAccessTypeNode(entityName)) {
-                    return entityName; // Indexed accesses can never be `typeof`
+                finally {
+                    cancellationToken = undefined;
                 }
-                if (isTypeOf) {
-                    return ts.createTypeQueryNode(entityName);
+            },
+            runWithCancellationToken: function (token, callback) {
+                try {
+                    cancellationToken = token;
+                    return callback(checker);
                 }
-                else {
-                    var lastId = ts.isIdentifier(entityName) ? entityName : entityName.right;
-                    var lastTypeArgs = lastId.typeArguments;
-                    lastId.typeArguments = undefined;
-                    return ts.createTypeReferenceNode(entityName, lastTypeArgs);
+                finally {
+                    cancellationToken = undefined;
                 }
-                function createAccessFromSymbolChain(chain, index, stopper) {
-                    var typeParameterNodes = index === (chain.length - 1) ? overrideTypeArguments : lookupTypeParameterNodes(chain, index, context);
-                    var symbol = chain[index];
-                    var parent = chain[index - 1];
-                    var symbolName;
-                    if (index === 0) {
-                        context.flags |= 16777216 /* InInitialEntityName */;
-                        symbolName = getNameOfSymbolAsWritten(symbol, context);
-                        context.approximateLength += (symbolName ? symbolName.length : 0) + 1;
-                        context.flags ^= 16777216 /* InInitialEntityName */;
-                    }
-                    else {
-                        if (parent && getExportsOfSymbol(parent)) {
-                            var exports_1 = getExportsOfSymbol(parent);
-                            ts.forEachEntry(exports_1, function (ex, name) {
-                                if (getSymbolIfSameReference(ex, symbol) && !isLateBoundName(name) && name !== "export=" /* ExportEquals */) {
-                                    symbolName = ts.unescapeLeadingUnderscores(name);
-                                    return true;
-                                }
-                            });
-                        }
-                    }
-                    if (!symbolName) {
-                        symbolName = getNameOfSymbolAsWritten(symbol, context);
-                    }
-                    context.approximateLength += symbolName.length + 1;
-                    if (!(context.flags & 16 /* ForbidIndexedAccessSymbolReferences */) && parent &&
-                        getMembersOfSymbol(parent) && getMembersOfSymbol(parent).get(symbol.escapedName) &&
-                        getSymbolIfSameReference(getMembersOfSymbol(parent).get(symbol.escapedName), symbol)) {
-                        // Should use an indexed access
-                        var LHS = createAccessFromSymbolChain(chain, index - 1, stopper);
-                        if (ts.isIndexedAccessTypeNode(LHS)) {
-                            return ts.createIndexedAccessTypeNode(LHS, ts.createLiteralTypeNode(ts.createLiteral(symbolName)));
+            },
+            getLocalTypeParametersOfClassOrInterfaceOrTypeAlias: getLocalTypeParametersOfClassOrInterfaceOrTypeAlias,
+            isDeclarationVisible: isDeclarationVisible,
+        };
+        function getResolvedSignatureWorker(nodeIn, candidatesOutArray, argumentCount, checkMode) {
+            var node = ts.getParseTreeNode(nodeIn, ts.isCallLikeExpression);
+            apparentArgumentCount = argumentCount;
+            var res = node ? getResolvedSignature(node, candidatesOutArray, checkMode) : undefined;
+            apparentArgumentCount = undefined;
+            return res;
+        }
+        var tupleTypes = new ts.Map();
+        var unionTypes = new ts.Map();
+        var intersectionTypes = new ts.Map();
+        var literalTypes = new ts.Map();
+        var indexedAccessTypes = new ts.Map();
+        var templateLiteralTypes = new ts.Map();
+        var stringMappingTypes = new ts.Map();
+        var substitutionTypes = new ts.Map();
+        var evolvingArrayTypes = [];
+        var undefinedProperties = new ts.Map();
+        var unknownSymbol = createSymbol(4 /* Property */, "unknown");
+        var resolvingSymbol = createSymbol(0, "__resolving__" /* Resolving */);
+        var anyType = createIntrinsicType(1 /* Any */, "any");
+        var autoType = createIntrinsicType(1 /* Any */, "any");
+        var wildcardType = createIntrinsicType(1 /* Any */, "any");
+        var errorType = createIntrinsicType(1 /* Any */, "error");
+        var nonInferrableAnyType = createIntrinsicType(1 /* Any */, "any", 524288 /* ContainsWideningType */);
+        var intrinsicMarkerType = createIntrinsicType(1 /* Any */, "intrinsic");
+        var unknownType = createIntrinsicType(2 /* Unknown */, "unknown");
+        var undefinedType = createIntrinsicType(32768 /* Undefined */, "undefined");
+        var undefinedWideningType = strictNullChecks ? undefinedType : createIntrinsicType(32768 /* Undefined */, "undefined", 524288 /* ContainsWideningType */);
+        var optionalType = createIntrinsicType(32768 /* Undefined */, "undefined");
+        var nullType = createIntrinsicType(65536 /* Null */, "null");
+        var nullWideningType = strictNullChecks ? nullType : createIntrinsicType(65536 /* Null */, "null", 524288 /* ContainsWideningType */);
+        var stringType = createIntrinsicType(4 /* String */, "string");
+        var numberType = createIntrinsicType(8 /* Number */, "number");
+        var bigintType = createIntrinsicType(64 /* BigInt */, "bigint");
+        var falseType = createIntrinsicType(512 /* BooleanLiteral */, "false");
+        var regularFalseType = createIntrinsicType(512 /* BooleanLiteral */, "false");
+        var trueType = createIntrinsicType(512 /* BooleanLiteral */, "true");
+        var regularTrueType = createIntrinsicType(512 /* BooleanLiteral */, "true");
+        trueType.regularType = regularTrueType;
+        trueType.freshType = trueType;
+        regularTrueType.regularType = regularTrueType;
+        regularTrueType.freshType = trueType;
+        falseType.regularType = regularFalseType;
+        falseType.freshType = falseType;
+        regularFalseType.regularType = regularFalseType;
+        regularFalseType.freshType = falseType;
+        var booleanType = createBooleanType([regularFalseType, regularTrueType]);
+        // Also mark all combinations of fresh/regular booleans as "Boolean" so they print as `boolean` instead of `true | false`
+        // (The union is cached, so simply doing the marking here is sufficient)
+        createBooleanType([regularFalseType, trueType]);
+        createBooleanType([falseType, regularTrueType]);
+        createBooleanType([falseType, trueType]);
+        var esSymbolType = createIntrinsicType(4096 /* ESSymbol */, "symbol");
+        var voidType = createIntrinsicType(16384 /* Void */, "void");
+        var neverType = createIntrinsicType(131072 /* Never */, "never");
+        var silentNeverType = createIntrinsicType(131072 /* Never */, "never");
+        var nonInferrableType = createIntrinsicType(131072 /* Never */, "never", 2097152 /* NonInferrableType */);
+        var implicitNeverType = createIntrinsicType(131072 /* Never */, "never");
+        var unreachableNeverType = createIntrinsicType(131072 /* Never */, "never");
+        var nonPrimitiveType = createIntrinsicType(67108864 /* NonPrimitive */, "object");
+        var stringNumberSymbolType = getUnionType([stringType, numberType, esSymbolType]);
+        var keyofConstraintType = keyofStringsOnly ? stringType : stringNumberSymbolType;
+        var numberOrBigIntType = getUnionType([numberType, bigintType]);
+        var templateConstraintType = getUnionType([stringType, numberType, booleanType, bigintType, nullType, undefinedType]);
+        var restrictiveMapper = makeFunctionTypeMapper(function (t) { return t.flags & 262144 /* TypeParameter */ ? getRestrictiveTypeParameter(t) : t; });
+        var permissiveMapper = makeFunctionTypeMapper(function (t) { return t.flags & 262144 /* TypeParameter */ ? wildcardType : t; });
+        var emptyObjectType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        var emptyJsxObjectType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        emptyJsxObjectType.objectFlags |= 4096 /* JsxAttributes */;
+        var emptyTypeLiteralSymbol = createSymbol(2048 /* TypeLiteral */, "__type" /* Type */);
+        emptyTypeLiteralSymbol.members = ts.createSymbolTable();
+        var emptyTypeLiteralType = createAnonymousType(emptyTypeLiteralSymbol, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        var emptyGenericType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        emptyGenericType.instantiations = new ts.Map();
+        var anyFunctionType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        // The anyFunctionType contains the anyFunctionType by definition. The flag is further propagated
+        // in getPropagatingFlagsOfTypes, and it is checked in inferFromTypes.
+        anyFunctionType.objectFlags |= 2097152 /* NonInferrableType */;
+        var noConstraintType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        var circularConstraintType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        var resolvingDefaultType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        var markerSuperType = createTypeParameter();
+        var markerSubType = createTypeParameter();
+        markerSubType.constraint = markerSuperType;
+        var markerOtherType = createTypeParameter();
+        var noTypePredicate = createTypePredicate(1 /* Identifier */, "<<unresolved>>", 0, anyType);
+        var anySignature = createSignature(undefined, undefined, undefined, ts.emptyArray, anyType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
+        var unknownSignature = createSignature(undefined, undefined, undefined, ts.emptyArray, errorType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
+        var resolvingSignature = createSignature(undefined, undefined, undefined, ts.emptyArray, anyType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
+        var silentNeverSignature = createSignature(undefined, undefined, undefined, ts.emptyArray, silentNeverType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
+        var enumNumberIndexInfo = createIndexInfo(stringType, /*isReadonly*/ true);
+        var iterationTypesCache = new ts.Map(); // cache for common IterationTypes instances
+        var noIterationTypes = {
+            get yieldType() { return ts.Debug.fail("Not supported"); },
+            get returnType() { return ts.Debug.fail("Not supported"); },
+            get nextType() { return ts.Debug.fail("Not supported"); },
+        };
+        var anyIterationTypes = createIterationTypes(anyType, anyType, anyType);
+        var anyIterationTypesExceptNext = createIterationTypes(anyType, anyType, unknownType);
+        var defaultIterationTypes = createIterationTypes(neverType, anyType, undefinedType); // default iteration types for `Iterator`.
+        var asyncIterationTypesResolver = {
+            iterableCacheKey: "iterationTypesOfAsyncIterable",
+            iteratorCacheKey: "iterationTypesOfAsyncIterator",
+            iteratorSymbolName: "asyncIterator",
+            getGlobalIteratorType: getGlobalAsyncIteratorType,
+            getGlobalIterableType: getGlobalAsyncIterableType,
+            getGlobalIterableIteratorType: getGlobalAsyncIterableIteratorType,
+            getGlobalGeneratorType: getGlobalAsyncGeneratorType,
+            resolveIterationType: getAwaitedType,
+            mustHaveANextMethodDiagnostic: ts.Diagnostics.An_async_iterator_must_have_a_next_method,
+            mustBeAMethodDiagnostic: ts.Diagnostics.The_0_property_of_an_async_iterator_must_be_a_method,
+            mustHaveAValueDiagnostic: ts.Diagnostics.The_type_returned_by_the_0_method_of_an_async_iterator_must_be_a_promise_for_a_type_with_a_value_property,
+        };
+        var syncIterationTypesResolver = {
+            iterableCacheKey: "iterationTypesOfIterable",
+            iteratorCacheKey: "iterationTypesOfIterator",
+            iteratorSymbolName: "iterator",
+            getGlobalIteratorType: getGlobalIteratorType,
+            getGlobalIterableType: getGlobalIterableType,
+            getGlobalIterableIteratorType: getGlobalIterableIteratorType,
+            getGlobalGeneratorType: getGlobalGeneratorType,
+            resolveIterationType: function (type, _errorNode) { return type; },
+            mustHaveANextMethodDiagnostic: ts.Diagnostics.An_iterator_must_have_a_next_method,
+            mustBeAMethodDiagnostic: ts.Diagnostics.The_0_property_of_an_iterator_must_be_a_method,
+            mustHaveAValueDiagnostic: ts.Diagnostics.The_type_returned_by_the_0_method_of_an_iterator_must_have_a_value_property,
+        };
+        /** Key is "/path/to/a.ts|/path/to/b.ts". */
+        var amalgamatedDuplicates;
+        var reverseMappedCache = new ts.Map();
+        var inInferTypeForHomomorphicMappedType = false;
+        var ambientModulesCache;
+        /**
+         * List of every ambient module with a "*" wildcard.
+         * Unlike other ambient modules, these can't be stored in `globals` because symbol tables only deal with exact matches.
+         * This is only used if there is no exact match.
+         */
+        var patternAmbientModules;
+        var patternAmbientModuleAugmentations;
+        var globalObjectType;
+        var globalFunctionType;
+        var globalCallableFunctionType;
+        var globalNewableFunctionType;
+        var globalArrayType;
+        var globalReadonlyArrayType;
+        var globalStringType;
+        var globalNumberType;
+        var globalBooleanType;
+        var globalRegExpType;
+        var globalThisType;
+        var anyArrayType;
+        var autoArrayType;
+        var anyReadonlyArrayType;
+        var deferredGlobalNonNullableTypeAlias;
+        // The library files are only loaded when the feature is used.
+        // This allows users to just specify library files they want to used through --lib
+        // and they will not get an error from not having unrelated library files
+        var deferredGlobalESSymbolConstructorSymbol;
+        var deferredGlobalESSymbolType;
+        var deferredGlobalTypedPropertyDescriptorType;
+        var deferredGlobalPromiseType;
+        var deferredGlobalPromiseLikeType;
+        var deferredGlobalPromiseConstructorSymbol;
+        var deferredGlobalPromiseConstructorLikeType;
+        var deferredGlobalIterableType;
+        var deferredGlobalIteratorType;
+        var deferredGlobalIterableIteratorType;
+        var deferredGlobalGeneratorType;
+        var deferredGlobalIteratorYieldResultType;
+        var deferredGlobalIteratorReturnResultType;
+        var deferredGlobalAsyncIterableType;
+        var deferredGlobalAsyncIteratorType;
+        var deferredGlobalAsyncIterableIteratorType;
+        var deferredGlobalAsyncGeneratorType;
+        var deferredGlobalTemplateStringsArrayType;
+        var deferredGlobalImportMetaType;
+        var deferredGlobalExtractSymbol;
+        var deferredGlobalOmitSymbol;
+        var deferredGlobalBigIntType;
+        var allPotentiallyUnusedIdentifiers = new ts.Map(); // key is file name
+        var flowLoopStart = 0;
+        var flowLoopCount = 0;
+        var sharedFlowCount = 0;
+        var flowAnalysisDisabled = false;
+        var flowInvocationCount = 0;
+        var lastFlowNode;
+        var lastFlowNodeReachable;
+        var flowTypeCache;
+        var emptyStringType = getLiteralType("");
+        var zeroType = getLiteralType(0);
+        var zeroBigIntType = getLiteralType({ negative: false, base10Value: "0" });
+        var resolutionTargets = [];
+        var resolutionResults = [];
+        var resolutionPropertyNames = [];
+        var suggestionCount = 0;
+        var maximumSuggestionCount = 10;
+        var mergedSymbols = [];
+        var symbolLinks = [];
+        var nodeLinks = [];
+        var flowLoopCaches = [];
+        var flowLoopNodes = [];
+        var flowLoopKeys = [];
+        var flowLoopTypes = [];
+        var sharedFlowNodes = [];
+        var sharedFlowTypes = [];
+        var flowNodeReachable = [];
+        var flowNodePostSuper = [];
+        var potentialThisCollisions = [];
+        var potentialNewTargetCollisions = [];
+        var potentialWeakMapCollisions = [];
+        var awaitedTypeStack = [];
+        var diagnostics = ts.createDiagnosticCollection();
+        var suggestionDiagnostics = ts.createDiagnosticCollection();
+        var typeofTypesByName = new ts.Map(ts.getEntries({
+            string: stringType,
+            number: numberType,
+            bigint: bigintType,
+            boolean: booleanType,
+            symbol: esSymbolType,
+            undefined: undefinedType
+        }));
+        var typeofType = createTypeofType();
+        var _jsxNamespace;
+        var _jsxFactoryEntity;
+        var outofbandVarianceMarkerHandler;
+        var subtypeRelation = new ts.Map();
+        var strictSubtypeRelation = new ts.Map();
+        var assignableRelation = new ts.Map();
+        var comparableRelation = new ts.Map();
+        var identityRelation = new ts.Map();
+        var enumRelation = new ts.Map();
+        var builtinGlobals = ts.createSymbolTable();
+        builtinGlobals.set(undefinedSymbol.escapedName, undefinedSymbol);
+        initializeTypeChecker();
+        return checker;
+        function getJsxNamespace(location) {
+            if (location) {
+                var file = ts.getSourceFileOfNode(location);
+                if (file) {
+                    if (ts.isJsxOpeningFragment(location)) {
+                        if (file.localJsxFragmentNamespace) {
+                            return file.localJsxFragmentNamespace;
                         }
-                        else {
-                            return ts.createIndexedAccessTypeNode(ts.createTypeReferenceNode(LHS, typeParameterNodes), ts.createLiteralTypeNode(ts.createLiteral(symbolName)));
+                        var jsxFragmentPragma = file.pragmas.get("jsxfrag");
+                        if (jsxFragmentPragma) {
+                            var chosenPragma = ts.isArray(jsxFragmentPragma) ? jsxFragmentPragma[0] : jsxFragmentPragma;
+                            file.localJsxFragmentFactory = ts.parseIsolatedEntityName(chosenPragma.arguments.factory, languageVersion);
+                            ts.visitNode(file.localJsxFragmentFactory, markAsSynthetic);
+                            if (file.localJsxFragmentFactory) {
+                                return file.localJsxFragmentNamespace = ts.getFirstIdentifier(file.localJsxFragmentFactory).escapedText;
+                            }
                         }
-                    }
-                    var identifier = ts.setEmitFlags(ts.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
-                    identifier.symbol = symbol;
-                    if (index > stopper) {
-                        var LHS = createAccessFromSymbolChain(chain, index - 1, stopper);
-                        if (!ts.isEntityName(LHS)) {
-                            return ts.Debug.fail("Impossible construct - an export of an indexed access cannot be reachable");
+                        var entity = getJsxFragmentFactoryEntity(location);
+                        if (entity) {
+                            file.localJsxFragmentFactory = entity;
+                            return file.localJsxFragmentNamespace = ts.getFirstIdentifier(entity).escapedText;
                         }
-                        return ts.createQualifiedName(LHS, identifier);
-                    }
-                    return identifier;
-                }
-            }
-            function typeParameterShadowsNameInScope(escapedName, context, type) {
-                var result = resolveName(context.enclosingDeclaration, escapedName, 788968 /* Type */, /*nameNotFoundArg*/ undefined, escapedName, /*isUse*/ false);
-                if (result) {
-                    if (result.flags & 262144 /* TypeParameter */ && result === type.symbol) {
-                        return false;
-                    }
-                    return true;
-                }
-                return false;
-            }
-            function typeParameterToName(type, context) {
-                if (context.flags & 4 /* GenerateNamesForShadowedTypeParams */ && context.typeParameterNames) {
-                    var cached = context.typeParameterNames.get("" + getTypeId(type));
-                    if (cached) {
-                        return cached;
-                    }
-                }
-                var result = symbolToName(type.symbol, context, 788968 /* Type */, /*expectsIdentifier*/ true);
-                if (!(result.kind & 75 /* Identifier */)) {
-                    return ts.createIdentifier("(Missing type parameter)");
-                }
-                if (context.flags & 4 /* GenerateNamesForShadowedTypeParams */) {
-                    var rawtext = result.escapedText;
-                    var i = 0;
-                    var text = rawtext;
-                    while ((context.typeParameterNamesByText && context.typeParameterNamesByText.get(text)) || typeParameterShadowsNameInScope(text, context, type)) {
-                        i++;
-                        text = rawtext + "_" + i;
-                    }
-                    if (text !== rawtext) {
-                        result = ts.createIdentifier(text, result.typeArguments);
-                    }
-                    (context.typeParameterNames || (context.typeParameterNames = ts.createMap())).set("" + getTypeId(type), result);
-                    (context.typeParameterNamesByText || (context.typeParameterNamesByText = ts.createMap())).set(result.escapedText, true);
-                }
-                return result;
-            }
-            function symbolToName(symbol, context, meaning, expectsIdentifier) {
-                var chain = lookupSymbolChain(symbol, context, meaning);
-                if (expectsIdentifier && chain.length !== 1
-                    && !context.encounteredError
-                    && !(context.flags & 65536 /* AllowQualifedNameInPlaceOfIdentifier */)) {
-                    context.encounteredError = true;
-                }
-                return createEntityNameFromSymbolChain(chain, chain.length - 1);
-                function createEntityNameFromSymbolChain(chain, index) {
-                    var typeParameterNodes = lookupTypeParameterNodes(chain, index, context);
-                    var symbol = chain[index];
-                    if (index === 0) {
-                        context.flags |= 16777216 /* InInitialEntityName */;
-                    }
-                    var symbolName = getNameOfSymbolAsWritten(symbol, context);
-                    if (index === 0) {
-                        context.flags ^= 16777216 /* InInitialEntityName */;
-                    }
-                    var identifier = ts.setEmitFlags(ts.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
-                    identifier.symbol = symbol;
-                    return index > 0 ? ts.createQualifiedName(createEntityNameFromSymbolChain(chain, index - 1), identifier) : identifier;
-                }
-            }
-            function symbolToExpression(symbol, context, meaning) {
-                var chain = lookupSymbolChain(symbol, context, meaning);
-                return createExpressionFromSymbolChain(chain, chain.length - 1);
-                function createExpressionFromSymbolChain(chain, index) {
-                    var typeParameterNodes = lookupTypeParameterNodes(chain, index, context);
-                    var symbol = chain[index];
-                    if (index === 0) {
-                        context.flags |= 16777216 /* InInitialEntityName */;
-                    }
-                    var symbolName = getNameOfSymbolAsWritten(symbol, context);
-                    if (index === 0) {
-                        context.flags ^= 16777216 /* InInitialEntityName */;
-                    }
-                    var firstChar = symbolName.charCodeAt(0);
-                    if (ts.isSingleOrDoubleQuote(firstChar) && ts.some(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
-                        return ts.createLiteral(getSpecifierForModuleSymbol(symbol, context));
-                    }
-                    var canUsePropertyAccess = firstChar === 35 /* hash */ ?
-                        symbolName.length > 1 && ts.isIdentifierStart(symbolName.charCodeAt(1), languageVersion) :
-                        ts.isIdentifierStart(firstChar, languageVersion);
-                    if (index === 0 || canUsePropertyAccess) {
-                        var identifier = ts.setEmitFlags(ts.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
-                        identifier.symbol = symbol;
-                        return index > 0 ? ts.createPropertyAccess(createExpressionFromSymbolChain(chain, index - 1), identifier) : identifier;
                     }
                     else {
-                        if (firstChar === 91 /* openBracket */) {
-                            symbolName = symbolName.substring(1, symbolName.length - 1);
-                            firstChar = symbolName.charCodeAt(0);
+                        if (file.localJsxNamespace) {
+                            return file.localJsxNamespace;
                         }
-                        var expression = void 0;
-                        if (ts.isSingleOrDoubleQuote(firstChar)) {
-                            expression = ts.createLiteral(symbolName.substring(1, symbolName.length - 1).replace(/\\./g, function (s) { return s.substring(1); }));
-                            expression.singleQuote = firstChar === 39 /* singleQuote */;
-                        }
-                        else if (("" + +symbolName) === symbolName) {
-                            expression = ts.createLiteral(+symbolName);
-                        }
-                        if (!expression) {
-                            expression = ts.setEmitFlags(ts.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
-                            expression.symbol = symbol;
+                        var jsxPragma = file.pragmas.get("jsx");
+                        if (jsxPragma) {
+                            var chosenPragma = ts.isArray(jsxPragma) ? jsxPragma[0] : jsxPragma;
+                            file.localJsxFactory = ts.parseIsolatedEntityName(chosenPragma.arguments.factory, languageVersion);
+                            ts.visitNode(file.localJsxFactory, markAsSynthetic);
+                            if (file.localJsxFactory) {
+                                return file.localJsxNamespace = ts.getFirstIdentifier(file.localJsxFactory).escapedText;
+                            }
                         }
-                        return ts.createElementAccess(createExpressionFromSymbolChain(chain, index - 1), expression);
                     }
                 }
             }
-            function isSingleQuotedStringNamed(d) {
-                var name = ts.getNameOfDeclaration(d);
-                if (name && ts.isStringLiteral(name) && (name.singleQuote ||
-                    (!ts.nodeIsSynthesized(name) && ts.startsWith(ts.getTextOfNode(name, /*includeTrivia*/ false), "'")))) {
-                    return true;
-                }
-                return false;
-            }
-            function getPropertyNameNodeForSymbol(symbol, context) {
-                var singleQuote = !!ts.length(symbol.declarations) && ts.every(symbol.declarations, isSingleQuotedStringNamed);
-                var fromNameType = getPropertyNameNodeForSymbolFromNameType(symbol, context, singleQuote);
-                if (fromNameType) {
-                    return fromNameType;
+            if (!_jsxNamespace) {
+                _jsxNamespace = "React";
+                if (compilerOptions.jsxFactory) {
+                    _jsxFactoryEntity = ts.parseIsolatedEntityName(compilerOptions.jsxFactory, languageVersion);
+                    ts.visitNode(_jsxFactoryEntity, markAsSynthetic);
+                    if (_jsxFactoryEntity) {
+                        _jsxNamespace = ts.getFirstIdentifier(_jsxFactoryEntity).escapedText;
+                    }
                 }
-                if (ts.isKnownSymbol(symbol)) {
-                    return ts.createComputedPropertyName(ts.createPropertyAccess(ts.createIdentifier("Symbol"), symbol.escapedName.substr(3)));
+                else if (compilerOptions.reactNamespace) {
+                    _jsxNamespace = ts.escapeLeadingUnderscores(compilerOptions.reactNamespace);
                 }
-                var rawName = ts.unescapeLeadingUnderscores(symbol.escapedName);
-                return createPropertyNameNodeForIdentifierOrLiteral(rawName, singleQuote);
             }
-            // See getNameForSymbolFromNameType for a stringy equivalent
-            function getPropertyNameNodeForSymbolFromNameType(symbol, context, singleQuote) {
-                var nameType = getSymbolLinks(symbol).nameType;
-                if (nameType) {
-                    if (nameType.flags & 384 /* StringOrNumberLiteral */) {
-                        var name = "" + nameType.value;
-                        if (!ts.isIdentifierText(name, compilerOptions.target) && !isNumericLiteralName(name)) {
-                            return ts.createLiteral(name, !!singleQuote);
-                        }
-                        if (isNumericLiteralName(name) && ts.startsWith(name, "-")) {
-                            return ts.createComputedPropertyName(ts.createLiteral(+name));
-                        }
-                        return createPropertyNameNodeForIdentifierOrLiteral(name);
-                    }
-                    if (nameType.flags & 8192 /* UniqueESSymbol */) {
-                        return ts.createComputedPropertyName(symbolToExpression(nameType.symbol, context, 111551 /* Value */));
-                    }
-                }
+            if (!_jsxFactoryEntity) {
+                _jsxFactoryEntity = ts.factory.createQualifiedName(ts.factory.createIdentifier(ts.unescapeLeadingUnderscores(_jsxNamespace)), "createElement");
             }
-            function createPropertyNameNodeForIdentifierOrLiteral(name, singleQuote) {
-                return ts.isIdentifierText(name, compilerOptions.target) ? ts.createIdentifier(name) : ts.createLiteral(isNumericLiteralName(name) && +name >= 0 ? +name : name, !!singleQuote);
+            return _jsxNamespace;
+            function markAsSynthetic(node) {
+                ts.setTextRangePosEnd(node, -1, -1);
+                return ts.visitEachChild(node, markAsSynthetic, ts.nullTransformationContext);
             }
-            function cloneNodeBuilderContext(context) {
-                var initial = __assign({}, context);
-                // Make type parameters created within this context not consume the name outside this context
-                // The symbol serializer ends up creating many sibling scopes that all need "separate" contexts when
-                // it comes to naming things - within a normal `typeToTypeNode` call, the node builder only ever descends
-                // through the type tree, so the only cases where we could have used distinct sibling scopes was when there
-                // were multiple generic overloads with similar generated type parameter names
-                // The effect:
-                // When we write out
-                // export const x: <T>(x: T) => T
-                // export const y: <T>(x: T) => T
-                // we write it out like that, rather than as
-                // export const x: <T>(x: T) => T
-                // export const y: <T_1>(x: T_1) => T_1
-                if (initial.typeParameterNames) {
-                    initial.typeParameterNames = ts.cloneMap(initial.typeParameterNames);
-                }
-                if (initial.typeParameterNamesByText) {
-                    initial.typeParameterNamesByText = ts.cloneMap(initial.typeParameterNamesByText);
-                }
-                if (initial.typeParameterSymbolList) {
-                    initial.typeParameterSymbolList = ts.cloneMap(initial.typeParameterSymbolList);
-                }
-                return initial;
+        }
+        function getEmitResolver(sourceFile, cancellationToken) {
+            // Ensure we have all the type information in place for this file so that all the
+            // emitter questions of this resolver will return the right information.
+            getDiagnostics(sourceFile, cancellationToken);
+            return emitResolver;
+        }
+        function lookupOrIssueError(location, message, arg0, arg1, arg2, arg3) {
+            var diagnostic = location
+                ? ts.createDiagnosticForNode(location, message, arg0, arg1, arg2, arg3)
+                : ts.createCompilerDiagnostic(message, arg0, arg1, arg2, arg3);
+            var existing = diagnostics.lookup(diagnostic);
+            if (existing) {
+                return existing;
             }
-            function getDeclarationWithTypeAnnotation(symbol, enclosingDeclaration) {
-                return symbol.declarations && ts.find(symbol.declarations, function (s) { return !!ts.getEffectiveTypeAnnotationNode(s) && (!enclosingDeclaration || !!ts.findAncestor(s, function (n) { return n === enclosingDeclaration; })); });
+            else {
+                diagnostics.add(diagnostic);
+                return diagnostic;
             }
-            function existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(existing, type) {
-                return !(ts.getObjectFlags(type) & 4 /* Reference */) || !ts.isTypeReferenceNode(existing) || ts.length(existing.typeArguments) >= getMinTypeArgumentCount(type.target.typeParameters);
+        }
+        function errorSkippedOn(key, location, message, arg0, arg1, arg2, arg3) {
+            var diagnostic = error(location, message, arg0, arg1, arg2, arg3);
+            diagnostic.skippedOn = key;
+            return diagnostic;
+        }
+        function error(location, message, arg0, arg1, arg2, arg3) {
+            var diagnostic = location
+                ? ts.createDiagnosticForNode(location, message, arg0, arg1, arg2, arg3)
+                : ts.createCompilerDiagnostic(message, arg0, arg1, arg2, arg3);
+            diagnostics.add(diagnostic);
+            return diagnostic;
+        }
+        function addErrorOrSuggestion(isError, diagnostic) {
+            if (isError) {
+                diagnostics.add(diagnostic);
             }
-            /**
-             * Unlike `typeToTypeNodeHelper`, this handles setting up the `AllowUniqueESSymbolType` flag
-             * so a `unique symbol` is returned when appropriate for the input symbol, rather than `typeof sym`
-             */
-            function serializeTypeForDeclaration(context, type, symbol, enclosingDeclaration, includePrivateSymbol, bundled) {
-                if (type !== errorType && enclosingDeclaration) {
-                    var declWithExistingAnnotation = getDeclarationWithTypeAnnotation(symbol, enclosingDeclaration);
-                    if (declWithExistingAnnotation && !ts.isFunctionLikeDeclaration(declWithExistingAnnotation)) {
-                        // try to reuse the existing annotation
-                        var existing = ts.getEffectiveTypeAnnotationNode(declWithExistingAnnotation);
-                        if (getTypeFromTypeNode(existing) === type && existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(existing, type)) {
-                            var result_4 = serializeExistingTypeNode(context, existing, includePrivateSymbol, bundled);
-                            if (result_4) {
-                                return result_4;
-                            }
-                        }
-                    }
-                }
-                var oldFlags = context.flags;
-                if (type.flags & 8192 /* UniqueESSymbol */ &&
-                    type.symbol === symbol) {
-                    context.flags |= 1048576 /* AllowUniqueESSymbolType */;
-                }
-                var result = typeToTypeNodeHelper(type, context);
-                context.flags = oldFlags;
-                return result;
+            else {
+                suggestionDiagnostics.add(__assign(__assign({}, diagnostic), { category: ts.DiagnosticCategory.Suggestion }));
             }
-            function serializeReturnTypeForSignature(context, type, signature, includePrivateSymbol, bundled) {
-                if (type !== errorType && context.enclosingDeclaration) {
-                    var annotation = signature.declaration && ts.getEffectiveReturnTypeNode(signature.declaration);
-                    if (!!ts.findAncestor(annotation, function (n) { return n === context.enclosingDeclaration; }) && annotation && instantiateType(getTypeFromTypeNode(annotation), signature.mapper) === type && existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(annotation, type)) {
-                        var result = serializeExistingTypeNode(context, annotation, includePrivateSymbol, bundled);
-                        if (result) {
-                            return result;
-                        }
-                    }
+        }
+        function errorOrSuggestion(isError, location, message, arg0, arg1, arg2, arg3) {
+            // Pseudo-synthesized input node
+            if (location.pos < 0 || location.end < 0) {
+                if (!isError) {
+                    return; // Drop suggestions (we have no span to suggest on)
                 }
-                return typeToTypeNodeHelper(type, context);
+                // Issue errors globally
+                var file = ts.getSourceFileOfNode(location);
+                addErrorOrSuggestion(isError, "message" in message ? ts.createFileDiagnostic(file, 0, 0, message, arg0, arg1, arg2, arg3) : ts.createDiagnosticForFileFromMessageChain(file, message)); // eslint-disable-line no-in-operator
+                return;
             }
-            function serializeExistingTypeNode(context, existing, includePrivateSymbol, bundled) {
-                if (cancellationToken && cancellationToken.throwIfCancellationRequested) {
-                    cancellationToken.throwIfCancellationRequested();
-                }
-                var hadError = false;
-                var transformed = ts.visitNode(existing, visitExistingNodeTreeSymbols);
-                if (hadError) {
-                    return undefined;
-                }
-                return transformed === existing ? ts.getMutableClone(existing) : transformed;
-                function visitExistingNodeTreeSymbols(node) {
-                    var _a, _b;
-                    // We don't _actually_ support jsdoc namepath types, emit `any` instead
-                    if (ts.isJSDocAllType(node) || node.kind === 302 /* JSDocNamepathType */) {
-                        return ts.createKeywordTypeNode(125 /* AnyKeyword */);
-                    }
-                    if (ts.isJSDocUnknownType(node)) {
-                        return ts.createKeywordTypeNode(148 /* UnknownKeyword */);
-                    }
-                    if (ts.isJSDocNullableType(node)) {
-                        return ts.createUnionTypeNode([ts.visitNode(node.type, visitExistingNodeTreeSymbols), ts.createKeywordTypeNode(100 /* NullKeyword */)]);
-                    }
-                    if (ts.isJSDocOptionalType(node)) {
-                        return ts.createUnionTypeNode([ts.visitNode(node.type, visitExistingNodeTreeSymbols), ts.createKeywordTypeNode(146 /* UndefinedKeyword */)]);
-                    }
-                    if (ts.isJSDocNonNullableType(node)) {
-                        return ts.visitNode(node.type, visitExistingNodeTreeSymbols);
-                    }
-                    if (ts.isJSDocVariadicType(node)) {
-                        return ts.createArrayTypeNode(ts.visitNode(node.type, visitExistingNodeTreeSymbols));
-                    }
-                    if (ts.isJSDocTypeLiteral(node)) {
-                        return ts.createTypeLiteralNode(ts.map(node.jsDocPropertyTags, function (t) {
-                            var name = ts.isIdentifier(t.name) ? t.name : t.name.right;
-                            var typeViaParent = getTypeOfPropertyOfType(getTypeFromTypeNode(node), name.escapedText);
-                            var overrideTypeNode = typeViaParent && t.typeExpression && getTypeFromTypeNode(t.typeExpression.type) !== typeViaParent ? typeToTypeNodeHelper(typeViaParent, context) : undefined;
-                            return ts.createPropertySignature(
-                            /*modifiers*/ undefined, name, t.typeExpression && ts.isJSDocOptionalType(t.typeExpression.type) ? ts.createToken(57 /* QuestionToken */) : undefined, overrideTypeNode || (t.typeExpression && ts.visitNode(t.typeExpression.type, visitExistingNodeTreeSymbols)) || ts.createKeywordTypeNode(125 /* AnyKeyword */), 
-                            /*initializer*/ undefined);
-                        }));
-                    }
-                    if (ts.isTypeReferenceNode(node) && ts.isIdentifier(node.typeName) && node.typeName.escapedText === "") {
-                        return ts.setOriginalNode(ts.createKeywordTypeNode(125 /* AnyKeyword */), node);
-                    }
-                    if ((ts.isExpressionWithTypeArguments(node) || ts.isTypeReferenceNode(node)) && ts.isJSDocIndexSignature(node)) {
-                        return ts.createTypeLiteralNode([ts.createIndexSignature(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, [ts.createParameter(
-                                /*decorators*/ undefined, 
-                                /*modifiers*/ undefined, 
-                                /*dotdotdotToken*/ undefined, "x", 
-                                /*questionToken*/ undefined, ts.visitNode(node.typeArguments[0], visitExistingNodeTreeSymbols))], ts.visitNode(node.typeArguments[1], visitExistingNodeTreeSymbols))]);
-                    }
-                    if (ts.isJSDocFunctionType(node)) {
-                        if (ts.isJSDocConstructSignature(node)) {
-                            var newTypeNode_1;
-                            return ts.createConstructorTypeNode(ts.visitNodes(node.typeParameters, visitExistingNodeTreeSymbols), ts.mapDefined(node.parameters, function (p, i) { return p.name && ts.isIdentifier(p.name) && p.name.escapedText === "new" ? (newTypeNode_1 = p.type, undefined) : ts.createParameter(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, getEffectiveDotDotDotForParameter(p), p.name || getEffectiveDotDotDotForParameter(p) ? "args" : "arg" + i, p.questionToken, ts.visitNode(p.type, visitExistingNodeTreeSymbols), 
-                            /*initializer*/ undefined); }), ts.visitNode(newTypeNode_1 || node.type, visitExistingNodeTreeSymbols));
-                        }
-                        else {
-                            return ts.createFunctionTypeNode(ts.visitNodes(node.typeParameters, visitExistingNodeTreeSymbols), ts.map(node.parameters, function (p, i) { return ts.createParameter(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, getEffectiveDotDotDotForParameter(p), p.name || getEffectiveDotDotDotForParameter(p) ? "args" : "arg" + i, p.questionToken, ts.visitNode(p.type, visitExistingNodeTreeSymbols), 
-                            /*initializer*/ undefined); }), ts.visitNode(node.type, visitExistingNodeTreeSymbols));
-                        }
-                    }
-                    if (ts.isTypeReferenceNode(node) && ts.isInJSDoc(node) && (getIntendedTypeFromJSDocTypeReference(node) || unknownSymbol === resolveTypeReferenceName(getTypeReferenceName(node), 788968 /* Type */, /*ignoreErrors*/ true))) {
-                        return ts.setOriginalNode(typeToTypeNodeHelper(getTypeFromTypeNode(node), context), node);
-                    }
-                    if (ts.isLiteralImportTypeNode(node)) {
-                        return ts.updateImportTypeNode(node, ts.updateLiteralTypeNode(node.argument, rewriteModuleSpecifier(node, node.argument.literal)), node.qualifier, ts.visitNodes(node.typeArguments, visitExistingNodeTreeSymbols, ts.isTypeNode), node.isTypeOf);
-                    }
-                    if (ts.isEntityName(node) || ts.isEntityNameExpression(node)) {
-                        var leftmost = ts.getFirstIdentifier(node);
-                        if (ts.isInJSFile(node) && (ts.isExportsIdentifier(leftmost) || ts.isModuleExportsAccessExpression(leftmost.parent) || (ts.isQualifiedName(leftmost.parent) && ts.isModuleIdentifier(leftmost.parent.left) && ts.isExportsIdentifier(leftmost.parent.right)))) {
-                            hadError = true;
-                            return node;
-                        }
-                        var sym = resolveEntityName(leftmost, 67108863 /* All */, /*ignoreErrors*/ true, /*dontResolveALias*/ true);
-                        if (sym) {
-                            if (isSymbolAccessible(sym, context.enclosingDeclaration, 67108863 /* All */, /*shouldComputeAliasesToMakeVisible*/ false).accessibility !== 0 /* Accessible */) {
-                                hadError = true;
-                            }
-                            else {
-                                (_b = (_a = context.tracker) === null || _a === void 0 ? void 0 : _a.trackSymbol) === null || _b === void 0 ? void 0 : _b.call(_a, sym, context.enclosingDeclaration, 67108863 /* All */);
-                                includePrivateSymbol === null || includePrivateSymbol === void 0 ? void 0 : includePrivateSymbol(sym);
-                            }
-                            if (ts.isIdentifier(node)) {
-                                var name = sym.flags & 262144 /* TypeParameter */ ? typeParameterToName(getDeclaredTypeOfSymbol(sym), context) : ts.getMutableClone(node);
-                                name.symbol = sym; // for quickinfo, which uses identifier symbol information
-                                return ts.setEmitFlags(ts.setOriginalNode(name, node), 16777216 /* NoAsciiEscaping */);
-                            }
-                        }
-                    }
-                    return ts.visitEachChild(node, visitExistingNodeTreeSymbols, ts.nullTransformationContext);
-                    function getEffectiveDotDotDotForParameter(p) {
-                        return p.dotDotDotToken || (p.type && ts.isJSDocVariadicType(p.type) ? ts.createToken(25 /* DotDotDotToken */) : undefined);
-                    }
-                    function rewriteModuleSpecifier(parent, lit) {
-                        if (bundled) {
-                            if (context.tracker && context.tracker.moduleResolverHost) {
-                                var targetFile = getExternalModuleFileFromDeclaration(parent);
-                                if (targetFile) {
-                                    var getCanonicalFileName = ts.createGetCanonicalFileName(!!host.useCaseSensitiveFileNames);
-                                    var resolverHost = {
-                                        getCanonicalFileName: getCanonicalFileName,
-                                        getCurrentDirectory: function () { return context.tracker.moduleResolverHost.getCurrentDirectory(); },
-                                        getCommonSourceDirectory: function () { return context.tracker.moduleResolverHost.getCommonSourceDirectory(); }
-                                    };
-                                    var newName = ts.getResolvedExternalModuleName(resolverHost, targetFile);
-                                    return ts.createLiteral(newName);
-                                }
-                            }
-                        }
-                        else {
-                            if (context.tracker && context.tracker.trackExternalModuleSymbolOfImportTypeNode) {
-                                var moduleSym = resolveExternalModuleNameWorker(lit, lit, /*moduleNotFoundError*/ undefined);
-                                if (moduleSym) {
-                                    context.tracker.trackExternalModuleSymbolOfImportTypeNode(moduleSym);
-                                }
-                            }
-                        }
-                        return lit;
-                    }
-                }
+            addErrorOrSuggestion(isError, "message" in message ? ts.createDiagnosticForNode(location, message, arg0, arg1, arg2, arg3) : ts.createDiagnosticForNodeFromMessageChain(location, message)); // eslint-disable-line no-in-operator
+        }
+        function errorAndMaybeSuggestAwait(location, maybeMissingAwait, message, arg0, arg1, arg2, arg3) {
+            var diagnostic = error(location, message, arg0, arg1, arg2, arg3);
+            if (maybeMissingAwait) {
+                var related = ts.createDiagnosticForNode(location, ts.Diagnostics.Did_you_forget_to_use_await);
+                ts.addRelatedInfo(diagnostic, related);
             }
-            function symbolTableToDeclarationStatements(symbolTable, context, bundled) {
-                var serializePropertySymbolForClass = makeSerializePropertySymbol(ts.createProperty, 161 /* MethodDeclaration */, /*useAcessors*/ true);
-                var serializePropertySymbolForInterfaceWorker = makeSerializePropertySymbol(function (_decorators, mods, name, question, type, initializer) { return ts.createPropertySignature(mods, name, question, type, initializer); }, 160 /* MethodSignature */, /*useAcessors*/ false);
-                // TODO: Use `setOriginalNode` on original declaration names where possible so these declarations see some kind of
-                // declaration mapping
-                // We save the enclosing declaration off here so it's not adjusted by well-meaning declaration
-                // emit codepaths which want to apply more specific contexts (so we can still refer to the root real declaration
-                // we're trying to emit from later on)
-                var enclosingDeclaration = context.enclosingDeclaration;
-                var results = [];
-                var visitedSymbols = ts.createMap();
-                var deferredPrivates;
-                var oldcontext = context;
-                context = __assign(__assign({}, oldcontext), { usedSymbolNames: ts.createMap(), remappedSymbolNames: ts.createMap(), tracker: __assign(__assign({}, oldcontext.tracker), { trackSymbol: function (sym, decl, meaning) {
-                            var accessibleResult = isSymbolAccessible(sym, decl, meaning, /*computeALiases*/ false);
-                            if (accessibleResult.accessibility === 0 /* Accessible */) {
-                                // Lookup the root symbol of the chain of refs we'll use to access it and serialize it
-                                var chain = lookupSymbolChainWorker(sym, context, meaning);
-                                if (!(sym.flags & 4 /* Property */)) {
-                                    includePrivateSymbol(chain[0]);
-                                }
-                            }
-                            else if (oldcontext.tracker && oldcontext.tracker.trackSymbol) {
-                                oldcontext.tracker.trackSymbol(sym, decl, meaning);
-                            }
-                        } }) });
-                if (oldcontext.usedSymbolNames) {
-                    oldcontext.usedSymbolNames.forEach(function (_, name) {
-                        context.usedSymbolNames.set(name, true);
-                    });
-                }
-                ts.forEachEntry(symbolTable, function (symbol, name) {
-                    var baseName = ts.unescapeLeadingUnderscores(name);
-                    void getInternalSymbolName(symbol, baseName); // Called to cache values into `usedSymbolNames` and `remappedSymbolNames`
-                });
-                var addingDeclare = !bundled;
-                var exportEquals = symbolTable.get("export=" /* ExportEquals */);
-                if (exportEquals && symbolTable.size > 1 && exportEquals.flags & 2097152 /* Alias */) {
-                    symbolTable = ts.createSymbolTable();
-                    // Remove extraneous elements from root symbol table (they'll be mixed back in when the target of the `export=` is looked up)
-                    symbolTable.set("export=" /* ExportEquals */, exportEquals);
-                }
-                visitSymbolTable(symbolTable);
-                return mergeRedundantStatements(results);
-                function isIdentifierAndNotUndefined(node) {
-                    return !!node && node.kind === 75 /* Identifier */;
-                }
-                function getNamesOfDeclaration(statement) {
-                    if (ts.isVariableStatement(statement)) {
-                        return ts.filter(ts.map(statement.declarationList.declarations, ts.getNameOfDeclaration), isIdentifierAndNotUndefined);
-                    }
-                    return ts.filter([ts.getNameOfDeclaration(statement)], isIdentifierAndNotUndefined);
-                }
-                function flattenExportAssignedNamespace(statements) {
-                    var exportAssignment = ts.find(statements, ts.isExportAssignment);
-                    var ns = ts.find(statements, ts.isModuleDeclaration);
-                    if (ns && exportAssignment && exportAssignment.isExportEquals &&
-                        ts.isIdentifier(exportAssignment.expression) && ts.isIdentifier(ns.name) && ts.idText(ns.name) === ts.idText(exportAssignment.expression) &&
-                        ns.body && ts.isModuleBlock(ns.body)) {
-                        // Pass 0: Correct situations where a module has both an `export = ns` and multiple top-level exports by stripping the export modifiers from
-                        //  the top-level exports and exporting them in the targeted ns, as can occur when a js file has both typedefs and `module.export` assignments
-                        var excessExports = ts.filter(statements, function (s) { return !!(ts.getModifierFlags(s) & 1 /* Export */); });
-                        if (ts.length(excessExports)) {
-                            ns.body.statements = ts.createNodeArray(__spreadArrays(ns.body.statements, [ts.createExportDeclaration(
-                                /*decorators*/ undefined, 
-                                /*modifiers*/ undefined, ts.createNamedExports(ts.map(ts.flatMap(excessExports, function (e) { return getNamesOfDeclaration(e); }), function (id) { return ts.createExportSpecifier(/*alias*/ undefined, id); })), 
-                                /*moduleSpecifier*/ undefined)]));
-                        }
-                        // Pass 1: Flatten `export namespace _exports {} export = _exports;` so long as the `export=` only points at a single namespace declaration
-                        if (!ts.find(statements, function (s) { return s !== ns && ts.nodeHasName(s, ns.name); })) {
-                            results = [];
-                            ts.forEach(ns.body.statements, function (s) {
-                                addResult(s, 0 /* None */); // Recalculates the ambient (and export, if applicable from above) flag
-                            });
-                            statements = __spreadArrays(ts.filter(statements, function (s) { return s !== ns && s !== exportAssignment; }), results);
-                        }
-                    }
-                    return statements;
-                }
-                function mergeExportDeclarations(statements) {
-                    // Pass 2: Combine all `export {}` declarations
-                    var exports = ts.filter(statements, function (d) { return ts.isExportDeclaration(d) && !d.moduleSpecifier && !!d.exportClause && ts.isNamedExports(d.exportClause); });
-                    if (ts.length(exports) > 1) {
-                        var nonExports = ts.filter(statements, function (d) { return !ts.isExportDeclaration(d) || !!d.moduleSpecifier || !d.exportClause; });
-                        statements = __spreadArrays(nonExports, [ts.createExportDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createNamedExports(ts.flatMap(exports, function (e) { return ts.cast(e.exportClause, ts.isNamedExports).elements; })), 
-                            /*moduleSpecifier*/ undefined)]);
-                    }
-                    // Pass 2b: Also combine all `export {} from "..."` declarations as needed
-                    var reexports = ts.filter(statements, function (d) { return ts.isExportDeclaration(d) && !!d.moduleSpecifier && !!d.exportClause && ts.isNamedExports(d.exportClause); });
-                    if (ts.length(reexports) > 1) {
-                        var groups = ts.group(reexports, function (decl) { return ts.isStringLiteral(decl.moduleSpecifier) ? ">" + decl.moduleSpecifier.text : ">"; });
-                        if (groups.length !== reexports.length) {
-                            var _loop_8 = function (group_1) {
-                                if (group_1.length > 1) {
-                                    // remove group members from statements and then merge group members and add back to statements
-                                    statements = __spreadArrays(ts.filter(statements, function (s) { return group_1.indexOf(s) === -1; }), [
-                                        ts.createExportDeclaration(
-                                        /*decorators*/ undefined, 
-                                        /*modifiers*/ undefined, ts.createNamedExports(ts.flatMap(group_1, function (e) { return ts.cast(e.exportClause, ts.isNamedExports).elements; })), group_1[0].moduleSpecifier)
-                                    ]);
-                                }
-                            };
-                            for (var _i = 0, groups_1 = groups; _i < groups_1.length; _i++) {
-                                var group_1 = groups_1[_i];
-                                _loop_8(group_1);
-                            }
-                        }
-                    }
-                    return statements;
-                }
-                function inlineExportModifiers(statements) {
-                    // Pass 3: Move all `export {}`'s to `export` modifiers where possible
-                    var exportDecl = ts.find(statements, function (d) { return ts.isExportDeclaration(d) && !d.moduleSpecifier && !!d.exportClause; });
-                    if (exportDecl && exportDecl.exportClause && ts.isNamedExports(exportDecl.exportClause)) {
-                        var replacements = ts.mapDefined(exportDecl.exportClause.elements, function (e) {
-                            if (!e.propertyName) {
-                                // export {name} - look thru `statements` for `name`, and if all results can take an `export` modifier, do so and filter it
-                                var associated = ts.filter(statements, function (s) { return ts.nodeHasName(s, e.name); });
-                                if (ts.length(associated) && ts.every(associated, canHaveExportModifier)) {
-                                    ts.forEach(associated, addExportModifier);
-                                    return undefined;
-                                }
-                            }
-                            return e;
-                        });
-                        if (!ts.length(replacements)) {
-                            // all clauses removed, filter the export declaration
-                            statements = ts.filter(statements, function (s) { return s !== exportDecl; });
-                        }
-                        else {
-                            // some items filtered, others not - update the export declaration
-                            // (mutating because why not, we're building a whole new tree here anyway)
-                            exportDecl.exportClause.elements = ts.createNodeArray(replacements);
-                        }
-                    }
-                    return statements;
-                }
-                function mergeRedundantStatements(statements) {
-                    statements = flattenExportAssignedNamespace(statements);
-                    statements = mergeExportDeclarations(statements);
-                    statements = inlineExportModifiers(statements);
-                    // Not a cleanup, but as a final step: If there is a mix of `export` and non-`export` declarations, but no `export =` or `export {}` add a `export {};` so
-                    // declaration privacy is respected.
-                    if (enclosingDeclaration &&
-                        ((ts.isSourceFile(enclosingDeclaration) && ts.isExternalOrCommonJsModule(enclosingDeclaration)) || ts.isModuleDeclaration(enclosingDeclaration)) &&
-                        (!ts.some(statements, ts.isExternalModuleIndicator) || (!ts.hasScopeMarker(statements) && ts.some(statements, ts.needsScopeMarker)))) {
-                        statements.push(ts.createEmptyExports());
-                    }
-                    return statements;
-                }
-                function canHaveExportModifier(node) {
-                    return ts.isEnumDeclaration(node) ||
-                        ts.isVariableStatement(node) ||
-                        ts.isFunctionDeclaration(node) ||
-                        ts.isClassDeclaration(node) ||
-                        (ts.isModuleDeclaration(node) && !ts.isExternalModuleAugmentation(node) && !ts.isGlobalScopeAugmentation(node)) ||
-                        ts.isInterfaceDeclaration(node) ||
-                        isTypeDeclaration(node);
-                }
-                function addExportModifier(statement) {
-                    var flags = (ts.getModifierFlags(statement) | 1 /* Export */) & ~2 /* Ambient */;
-                    statement.modifiers = ts.createNodeArray(ts.createModifiersFromModifierFlags(flags));
-                    statement.modifierFlagsCache = 0;
-                }
-                function visitSymbolTable(symbolTable, suppressNewPrivateContext, propertyAsAlias) {
-                    var oldDeferredPrivates = deferredPrivates;
-                    if (!suppressNewPrivateContext) {
-                        deferredPrivates = ts.createMap();
-                    }
-                    symbolTable.forEach(function (symbol) {
-                        serializeSymbol(symbol, /*isPrivate*/ false, !!propertyAsAlias);
-                    });
-                    if (!suppressNewPrivateContext) {
-                        // deferredPrivates will be filled up by visiting the symbol table
-                        // And will continue to iterate as elements are added while visited `deferredPrivates`
-                        // (As that's how a map iterator is defined to work)
-                        deferredPrivates.forEach(function (symbol) {
-                            serializeSymbol(symbol, /*isPrivate*/ true, !!propertyAsAlias);
-                        });
-                    }
-                    deferredPrivates = oldDeferredPrivates;
-                }
-                function serializeSymbol(symbol, isPrivate, propertyAsAlias) {
-                    // cache visited list based on merged symbol, since we want to use the unmerged top-level symbol, but
-                    // still skip reserializing it if we encounter the merged product later on
-                    var visitedSym = getMergedSymbol(symbol);
-                    if (visitedSymbols.has("" + getSymbolId(visitedSym))) {
-                        return; // Already printed
-                    }
-                    visitedSymbols.set("" + getSymbolId(visitedSym), true);
-                    // Only actually serialize symbols within the correct enclosing declaration, otherwise do nothing with the out-of-context symbol
-                    var skipMembershipCheck = !isPrivate; // We only call this on exported symbols when we know they're in the correct scope
-                    if (skipMembershipCheck || (!!ts.length(symbol.declarations) && ts.some(symbol.declarations, function (d) { return !!ts.findAncestor(d, function (n) { return n === enclosingDeclaration; }); }))) {
-                        var oldContext = context;
-                        context = cloneNodeBuilderContext(context);
-                        var result = serializeSymbolWorker(symbol, isPrivate, propertyAsAlias);
-                        context = oldContext;
-                        return result;
-                    }
-                }
-                // Synthesize declarations for a symbol - might be an Interface, a Class, a Namespace, a Type, a Variable (const, let, or var), an Alias
-                // or a merge of some number of those.
-                // An interesting challenge is ensuring that when classes merge with namespaces and interfaces, is keeping
-                // each symbol in only one of the representations
-                // Also, synthesizing a default export of some kind
-                // If it's an alias: emit `export default ref`
-                // If it's a property: emit `export default _default` with a `_default` prop
-                // If it's a class/interface/function: emit a class/interface/function with a `default` modifier
-                // These forms can merge, eg (`export default 12; export default interface A {}`)
-                function serializeSymbolWorker(symbol, isPrivate, propertyAsAlias) {
-                    var symbolName = ts.unescapeLeadingUnderscores(symbol.escapedName);
-                    var isDefault = symbol.escapedName === "default" /* Default */;
-                    if (!(context.flags & 131072 /* AllowAnonymousIdentifier */) && ts.isStringANonContextualKeyword(symbolName) && !isDefault) {
-                        // Oh no. We cannot use this symbol's name as it's name... It's likely some jsdoc had an invalid name like `export` or `default` :(
-                        context.encounteredError = true;
-                        // TODO: Issue error via symbol tracker?
-                        return; // If we need to emit a private with a keyword name, we're done for, since something else will try to refer to it by that name
-                    }
-                    var needsPostExportDefault = isDefault && !!(symbol.flags & -113 /* ExportDoesNotSupportDefaultModifier */
-                        || (symbol.flags & 16 /* Function */ && ts.length(getPropertiesOfType(getTypeOfSymbol(symbol))))) && !(symbol.flags & 2097152 /* Alias */); // An alias symbol should preclude needing to make an alias ourselves
-                    if (needsPostExportDefault) {
-                        isPrivate = true;
-                    }
-                    var modifierFlags = (!isPrivate ? 1 /* Export */ : 0) | (isDefault && !needsPostExportDefault ? 512 /* Default */ : 0);
-                    var isConstMergedWithNS = symbol.flags & 1536 /* Module */ &&
-                        symbol.flags & (2 /* BlockScopedVariable */ | 1 /* FunctionScopedVariable */ | 4 /* Property */) &&
-                        symbol.escapedName !== "export=" /* ExportEquals */;
-                    var isConstMergedWithNSPrintableAsSignatureMerge = isConstMergedWithNS && isTypeRepresentableAsFunctionNamespaceMerge(getTypeOfSymbol(symbol), symbol);
-                    if (symbol.flags & (16 /* Function */ | 8192 /* Method */) || isConstMergedWithNSPrintableAsSignatureMerge) {
-                        serializeAsFunctionNamespaceMerge(getTypeOfSymbol(symbol), symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
-                    }
-                    if (symbol.flags & 524288 /* TypeAlias */) {
-                        serializeTypeAlias(symbol, symbolName, modifierFlags);
-                    }
-                    // Need to skip over export= symbols below - json source files get a single `Property` flagged
-                    // symbol of name `export=` which needs to be handled like an alias. It's not great, but it is what it is.
-                    if (symbol.flags & (2 /* BlockScopedVariable */ | 1 /* FunctionScopedVariable */ | 4 /* Property */)
-                        && symbol.escapedName !== "export=" /* ExportEquals */
-                        && !(symbol.flags & 4194304 /* Prototype */)
-                        && !(symbol.flags & 32 /* Class */)
-                        && !isConstMergedWithNSPrintableAsSignatureMerge) {
-                        serializeVariableOrProperty(symbol, symbolName, isPrivate, needsPostExportDefault, propertyAsAlias, modifierFlags);
-                    }
-                    if (symbol.flags & 384 /* Enum */) {
-                        serializeEnum(symbol, symbolName, modifierFlags);
-                    }
-                    if (symbol.flags & 32 /* Class */) {
-                        if (symbol.flags & 4 /* Property */ && ts.isBinaryExpression(symbol.valueDeclaration.parent) && ts.isClassExpression(symbol.valueDeclaration.parent.right)) {
-                            // Looks like a `module.exports.Sub = class {}` - if we serialize `symbol` as a class, the result will have no members,
-                            // since the classiness is actually from the target of the effective alias the symbol is. yes. A BlockScopedVariable|Class|Property
-                            // _really_ acts like an Alias, and none of a BlockScopedVariable, Class, or Property. This is the travesty of JS binding today.
-                            serializeAsAlias(symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
-                        }
-                        else {
-                            serializeAsClass(symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
-                        }
-                    }
-                    if ((symbol.flags & (512 /* ValueModule */ | 1024 /* NamespaceModule */) && (!isConstMergedWithNS || isTypeOnlyNamespace(symbol))) || isConstMergedWithNSPrintableAsSignatureMerge) {
-                        serializeModule(symbol, symbolName, modifierFlags);
-                    }
-                    if (symbol.flags & 64 /* Interface */) {
-                        serializeInterface(symbol, symbolName, modifierFlags);
-                    }
-                    if (symbol.flags & 2097152 /* Alias */) {
-                        serializeAsAlias(symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
-                    }
-                    if (symbol.flags & 4 /* Property */ && symbol.escapedName === "export=" /* ExportEquals */) {
-                        serializeMaybeAliasAssignment(symbol);
-                    }
-                    if (symbol.flags & 8388608 /* ExportStar */) {
-                        // synthesize export * from "moduleReference"
-                        // Straightforward - only one thing to do - make an export declaration
-                        for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                            var node = _a[_i];
-                            var resolvedModule = resolveExternalModuleName(node, node.moduleSpecifier);
-                            if (!resolvedModule)
-                                continue;
-                            addResult(ts.createExportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*exportClause*/ undefined, ts.createLiteral(getSpecifierForModuleSymbol(resolvedModule, context))), 0 /* None */);
-                        }
-                    }
-                    if (needsPostExportDefault) {
-                        addResult(ts.createExportAssignment(/*decorators*/ undefined, /*modifiers*/ undefined, /*isExportAssignment*/ false, ts.createIdentifier(getInternalSymbolName(symbol, symbolName))), 0 /* None */);
-                    }
-                }
-                function includePrivateSymbol(symbol) {
-                    if (ts.some(symbol.declarations, ts.isParameterDeclaration))
-                        return;
-                    ts.Debug.assertIsDefined(deferredPrivates);
-                    getUnusedName(ts.unescapeLeadingUnderscores(symbol.escapedName), symbol); // Call to cache unique name for symbol
-                    deferredPrivates.set("" + getSymbolId(symbol), symbol);
-                }
-                function isExportingScope(enclosingDeclaration) {
-                    return ((ts.isSourceFile(enclosingDeclaration) && (ts.isExternalOrCommonJsModule(enclosingDeclaration) || ts.isJsonSourceFile(enclosingDeclaration))) ||
-                        (ts.isAmbientModule(enclosingDeclaration) && !ts.isGlobalScopeAugmentation(enclosingDeclaration)));
-                }
-                // Prepends a `declare` and/or `export` modifier if the context requires it, and then adds `node` to `result` and returns `node`
-                // Note: This _mutates_ `node` without using `updateNode` - the assumption being that all nodes should be manufactured fresh by the node builder
-                function addResult(node, additionalModifierFlags) {
-                    var newModifierFlags = 0 /* None */;
-                    if (additionalModifierFlags & 1 /* Export */ &&
-                        enclosingDeclaration &&
-                        isExportingScope(enclosingDeclaration) &&
-                        canHaveExportModifier(node)) {
-                        // Classes, namespaces, variables, functions, interfaces, and types should all be `export`ed in a module context if not private
-                        newModifierFlags |= 1 /* Export */;
-                    }
-                    if (addingDeclare && !(newModifierFlags & 1 /* Export */) &&
-                        (!enclosingDeclaration || !(enclosingDeclaration.flags & 8388608 /* Ambient */)) &&
-                        (ts.isEnumDeclaration(node) || ts.isVariableStatement(node) || ts.isFunctionDeclaration(node) || ts.isClassDeclaration(node) || ts.isModuleDeclaration(node))) {
-                        // Classes, namespaces, variables, enums, and functions all need `declare` modifiers to be valid in a declaration file top-level scope
-                        newModifierFlags |= 2 /* Ambient */;
-                    }
-                    if ((additionalModifierFlags & 512 /* Default */) && (ts.isClassDeclaration(node) || ts.isInterfaceDeclaration(node) || ts.isFunctionDeclaration(node))) {
-                        newModifierFlags |= 512 /* Default */;
-                    }
-                    if (newModifierFlags) {
-                        node.modifiers = ts.createNodeArray(ts.createModifiersFromModifierFlags(newModifierFlags | ts.getModifierFlags(node)));
-                        node.modifierFlagsCache = 0; // Reset computed flags cache
-                    }
-                    results.push(node);
-                }
-                function serializeTypeAlias(symbol, symbolName, modifierFlags) {
-                    var aliasType = getDeclaredTypeOfTypeAlias(symbol);
-                    var typeParams = getSymbolLinks(symbol).typeParameters;
-                    var typeParamDecls = ts.map(typeParams, function (p) { return typeParameterToDeclaration(p, context); });
-                    var jsdocAliasDecl = ts.find(symbol.declarations, ts.isJSDocTypeAlias);
-                    var commentText = jsdocAliasDecl ? jsdocAliasDecl.comment || jsdocAliasDecl.parent.comment : undefined;
-                    var oldFlags = context.flags;
-                    context.flags |= 8388608 /* InTypeAlias */;
-                    addResult(ts.setSyntheticLeadingComments(ts.createTypeAliasDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, getInternalSymbolName(symbol, symbolName), typeParamDecls, typeToTypeNodeHelper(aliasType, context)), !commentText ? [] : [{ kind: 3 /* MultiLineCommentTrivia */, text: "*\n * " + commentText.replace(/\n/g, "\n * ") + "\n ", pos: -1, end: -1, hasTrailingNewLine: true }]), modifierFlags);
-                    context.flags = oldFlags;
-                }
-                function serializeInterface(symbol, symbolName, modifierFlags) {
-                    var interfaceType = getDeclaredTypeOfClassOrInterface(symbol);
-                    var localParams = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
-                    var typeParamDecls = ts.map(localParams, function (p) { return typeParameterToDeclaration(p, context); });
-                    var baseTypes = getBaseTypes(interfaceType);
-                    var baseType = ts.length(baseTypes) ? getIntersectionType(baseTypes) : undefined;
-                    var members = ts.flatMap(getPropertiesOfType(interfaceType), function (p) { return serializePropertySymbolForInterface(p, baseType); });
-                    var callSignatures = serializeSignatures(0 /* Call */, interfaceType, baseType, 165 /* CallSignature */);
-                    var constructSignatures = serializeSignatures(1 /* Construct */, interfaceType, baseType, 166 /* ConstructSignature */);
-                    var indexSignatures = serializeIndexSignatures(interfaceType, baseType);
-                    var heritageClauses = !ts.length(baseTypes) ? undefined : [ts.createHeritageClause(90 /* ExtendsKeyword */, ts.mapDefined(baseTypes, function (b) { return trySerializeAsTypeReference(b); }))];
-                    addResult(ts.createInterfaceDeclaration(
-                    /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, getInternalSymbolName(symbol, symbolName), typeParamDecls, heritageClauses, __spreadArrays(indexSignatures, constructSignatures, callSignatures, members)), modifierFlags);
-                }
-                function getNamespaceMembersForSerialization(symbol) {
-                    return !symbol.exports ? [] : ts.filter(ts.arrayFrom(symbol.exports.values()), isNamespaceMember);
-                }
-                function isTypeOnlyNamespace(symbol) {
-                    return ts.every(getNamespaceMembersForSerialization(symbol), function (m) { return !(resolveSymbol(m).flags & 111551 /* Value */); });
-                }
-                function serializeModule(symbol, symbolName, modifierFlags) {
-                    var members = getNamespaceMembersForSerialization(symbol);
-                    // Split NS members up by declaration - members whose parent symbol is the ns symbol vs those whose is not (but were added in later via merging)
-                    var locationMap = ts.arrayToMultiMap(members, function (m) { return m.parent && m.parent === symbol ? "real" : "merged"; });
-                    var realMembers = locationMap.get("real") || ts.emptyArray;
-                    var mergedMembers = locationMap.get("merged") || ts.emptyArray;
-                    // TODO: `suppressNewPrivateContext` is questionable -we need to simply be emitting privates in whatever scope they were declared in, rather
-                    // than whatever scope we traverse to them in. That's a bit of a complex rewrite, since we're not _actually_ tracking privates at all in advance,
-                    // so we don't even have placeholders to fill in.
-                    if (ts.length(realMembers)) {
-                        var localName = getInternalSymbolName(symbol, symbolName);
-                        serializeAsNamespaceDeclaration(realMembers, localName, modifierFlags, !!(symbol.flags & (16 /* Function */ | 67108864 /* Assignment */)));
-                    }
-                    if (ts.length(mergedMembers)) {
-                        var containingFile_1 = ts.getSourceFileOfNode(context.enclosingDeclaration);
-                        var localName = getInternalSymbolName(symbol, symbolName);
-                        var nsBody = ts.createModuleBlock([ts.createExportDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createNamedExports(ts.mapDefined(ts.filter(mergedMembers, function (n) { return n.escapedName !== "export=" /* ExportEquals */; }), function (s) {
-                                var _a, _b;
-                                var name = ts.unescapeLeadingUnderscores(s.escapedName);
-                                var localName = getInternalSymbolName(s, name);
-                                var aliasDecl = s.declarations && getDeclarationOfAliasSymbol(s);
-                                if (containingFile_1 && (aliasDecl ? containingFile_1 !== ts.getSourceFileOfNode(aliasDecl) : !ts.some(s.declarations, function (d) { return ts.getSourceFileOfNode(d) === containingFile_1; }))) {
-                                    (_b = (_a = context.tracker) === null || _a === void 0 ? void 0 : _a.reportNonlocalAugmentation) === null || _b === void 0 ? void 0 : _b.call(_a, containingFile_1, symbol, s);
-                                    return undefined;
-                                }
-                                var target = aliasDecl && getTargetOfAliasDeclaration(aliasDecl, /*dontRecursivelyResolve*/ true);
-                                includePrivateSymbol(target || s);
-                                var targetName = target ? getInternalSymbolName(target, ts.unescapeLeadingUnderscores(target.escapedName)) : localName;
-                                return ts.createExportSpecifier(name === targetName ? undefined : targetName, name);
-                            })))]);
-                        addResult(ts.createModuleDeclaration(
-                        /*decorators*/ undefined, 
-                        /*modifiers*/ undefined, ts.createIdentifier(localName), nsBody, 16 /* Namespace */), 0 /* None */);
-                    }
-                }
-                function serializeEnum(symbol, symbolName, modifierFlags) {
-                    addResult(ts.createEnumDeclaration(
-                    /*decorators*/ undefined, ts.createModifiersFromModifierFlags(isConstEnumSymbol(symbol) ? 2048 /* Const */ : 0), getInternalSymbolName(symbol, symbolName), ts.map(ts.filter(getPropertiesOfType(getTypeOfSymbol(symbol)), function (p) { return !!(p.flags & 8 /* EnumMember */); }), function (p) {
-                        // TODO: Handle computed names
-                        // I hate that to get the initialized value we need to walk back to the declarations here; but there's no
-                        // other way to get the possible const value of an enum member that I'm aware of, as the value is cached
-                        // _on the declaration_, not on the declaration's symbol...
-                        var initializedValue = p.declarations && p.declarations[0] && ts.isEnumMember(p.declarations[0]) && getConstantValue(p.declarations[0]);
-                        return ts.createEnumMember(ts.unescapeLeadingUnderscores(p.escapedName), initializedValue === undefined ? undefined : ts.createLiteral(initializedValue));
-                    })), modifierFlags);
-                }
-                function serializeVariableOrProperty(symbol, symbolName, isPrivate, needsPostExportDefault, propertyAsAlias, modifierFlags) {
-                    if (propertyAsAlias) {
-                        serializeMaybeAliasAssignment(symbol);
-                    }
-                    else {
-                        var type = getTypeOfSymbol(symbol);
-                        var localName = getInternalSymbolName(symbol, symbolName);
-                        if (!(symbol.flags & 16 /* Function */) && isTypeRepresentableAsFunctionNamespaceMerge(type, symbol)) {
-                            // If the type looks like a function declaration + ns could represent it, and it's type is sourced locally, rewrite it into a function declaration + ns
-                            serializeAsFunctionNamespaceMerge(type, symbol, localName, modifierFlags);
-                        }
-                        else {
-                            // A Class + Property merge is made for a `module.exports.Member = class {}`, and it doesn't serialize well as either a class _or_ a property symbol - in fact, _it behaves like an alias!_
-                            // `var` is `FunctionScopedVariable`, `const` and `let` are `BlockScopedVariable`, and `module.exports.thing =` is `Property`
-                            var flags = !(symbol.flags & 2 /* BlockScopedVariable */) ? undefined
-                                : isConstVariable(symbol) ? 2 /* Const */
-                                    : 1 /* Let */;
-                            var name = (needsPostExportDefault || !(symbol.flags & 4 /* Property */)) ? localName : getUnusedName(localName, symbol);
-                            var textRange = symbol.declarations && ts.find(symbol.declarations, function (d) { return ts.isVariableDeclaration(d); });
-                            if (textRange && ts.isVariableDeclarationList(textRange.parent) && textRange.parent.declarations.length === 1) {
-                                textRange = textRange.parent.parent;
-                            }
-                            var statement = ts.setTextRange(ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([
-                                ts.createVariableDeclaration(name, serializeTypeForDeclaration(context, type, symbol, enclosingDeclaration, includePrivateSymbol, bundled))
-                            ], flags)), textRange);
-                            addResult(statement, name !== localName ? modifierFlags & ~1 /* Export */ : modifierFlags);
-                            if (name !== localName && !isPrivate) {
-                                // We rename the variable declaration we generate for Property symbols since they may have a name which
-                                // conflicts with a local declaration. For example, given input:
-                                // ```
-                                // function g() {}
-                                // module.exports.g = g
-                                // ```
-                                // In such a situation, we have a local variable named `g`, and a separate exported variable named `g`.
-                                // Naively, we would emit
-                                // ```
-                                // function g() {}
-                                // export const g: typeof g;
-                                // ```
-                                // That's obviously incorrect - the `g` in the type annotation needs to refer to the local `g`, but
-                                // the export declaration shadows it.
-                                // To work around that, we instead write
-                                // ```
-                                // function g() {}
-                                // const g_1: typeof g;
-                                // export { g_1 as g };
-                                // ```
-                                // To create an export named `g` that does _not_ shadow the local `g`
-                                addResult(ts.createExportDeclaration(
-                                /*decorators*/ undefined, 
-                                /*modifiers*/ undefined, ts.createNamedExports([ts.createExportSpecifier(name, localName)])), 0 /* None */);
-                            }
-                        }
-                    }
-                }
-                function serializeAsFunctionNamespaceMerge(type, symbol, localName, modifierFlags) {
-                    var signatures = getSignaturesOfType(type, 0 /* Call */);
-                    for (var _i = 0, signatures_2 = signatures; _i < signatures_2.length; _i++) {
-                        var sig = signatures_2[_i];
-                        // Each overload becomes a separate function declaration, in order
-                        var decl = signatureToSignatureDeclarationHelper(sig, 244 /* FunctionDeclaration */, context, includePrivateSymbol, bundled);
-                        decl.name = ts.createIdentifier(localName);
-                        // for expressions assigned to `var`s, use the `var` as the text range
-                        addResult(ts.setTextRange(decl, sig.declaration && ts.isVariableDeclaration(sig.declaration.parent) && sig.declaration.parent.parent || sig.declaration), modifierFlags);
-                    }
-                    // Module symbol emit will take care of module-y members, provided it has exports
-                    if (!(symbol.flags & (512 /* ValueModule */ | 1024 /* NamespaceModule */) && !!symbol.exports && !!symbol.exports.size)) {
-                        var props = ts.filter(getPropertiesOfType(type), isNamespaceMember);
-                        serializeAsNamespaceDeclaration(props, localName, modifierFlags, /*suppressNewPrivateContext*/ true);
-                    }
-                }
-                function serializeAsNamespaceDeclaration(props, localName, modifierFlags, suppressNewPrivateContext) {
-                    if (ts.length(props)) {
-                        var localVsRemoteMap = ts.arrayToMultiMap(props, function (p) {
-                            return !ts.length(p.declarations) || ts.some(p.declarations, function (d) {
-                                return ts.getSourceFileOfNode(d) === ts.getSourceFileOfNode(context.enclosingDeclaration);
-                            }) ? "local" : "remote";
-                        });
-                        var localProps = localVsRemoteMap.get("local") || ts.emptyArray;
-                        // handle remote props first - we need to make an `import` declaration that points at the module containing each remote
-                        // prop in the outermost scope (TODO: a namespace within a namespace would need to be appropriately handled by this)
-                        // Example:
-                        // import Foo_1 = require("./exporter");
-                        // export namespace ns {
-                        //     import Foo = Foo_1.Foo;
-                        //     export { Foo };
-                        //     export const c: number;
-                        // }
-                        // This is needed because in JS, statements like `const x = require("./f")` support both type and value lookup, even if they're
-                        // normally just value lookup (so it functions kinda like an alias even when it's not an alias)
-                        // _Usually_, we'll simply print the top-level as an alias instead of a `var` in such situations, however is is theoretically
-                        // possible to encounter a situation where a type has members from both the current file and other files - in those situations,
-                        // emit akin to the above would be needed.
-                        // Add a namespace
-                        var fakespace = ts.createModuleDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.createIdentifier(localName), ts.createModuleBlock([]), 16 /* Namespace */);
-                        fakespace.flags ^= 8 /* Synthesized */; // unset synthesized so it is usable as an enclosing declaration
-                        fakespace.parent = enclosingDeclaration;
-                        fakespace.locals = ts.createSymbolTable(props);
-                        fakespace.symbol = props[0].parent;
-                        var oldResults = results;
-                        results = [];
-                        var oldAddingDeclare = addingDeclare;
-                        addingDeclare = false;
-                        var subcontext = __assign(__assign({}, context), { enclosingDeclaration: fakespace });
-                        var oldContext = context;
-                        context = subcontext;
-                        // TODO: implement handling for the localVsRemoteMap.get("remote") - should be difficult to trigger (see comment above), as only interesting cross-file js merges should make this possible
-                        visitSymbolTable(ts.createSymbolTable(localProps), suppressNewPrivateContext, /*propertyAsAlias*/ true);
-                        context = oldContext;
-                        addingDeclare = oldAddingDeclare;
-                        var declarations = results;
-                        results = oldResults;
-                        fakespace.flags ^= 8 /* Synthesized */; // reset synthesized
-                        fakespace.parent = undefined;
-                        fakespace.locals = undefined;
-                        fakespace.symbol = undefined;
-                        fakespace.body = ts.createModuleBlock(declarations);
-                        addResult(fakespace, modifierFlags); // namespaces can never be default exported
-                    }
-                }
-                function isNamespaceMember(p) {
-                    return !(p.flags & 4194304 /* Prototype */ || p.escapedName === "prototype" || p.valueDeclaration && ts.isClassLike(p.valueDeclaration.parent));
-                }
-                function serializeAsClass(symbol, localName, modifierFlags) {
-                    var localParams = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
-                    var typeParamDecls = ts.map(localParams, function (p) { return typeParameterToDeclaration(p, context); });
-                    var classType = getDeclaredTypeOfClassOrInterface(symbol);
-                    var baseTypes = getBaseTypes(classType);
-                    var implementsTypes = getImplementsTypes(classType);
-                    var staticType = getTypeOfSymbol(symbol);
-                    var staticBaseType = getBaseConstructorTypeOfClass(staticType);
-                    var heritageClauses = __spreadArrays(!ts.length(baseTypes) ? [] : [ts.createHeritageClause(90 /* ExtendsKeyword */, ts.map(baseTypes, function (b) { return serializeBaseType(b, staticBaseType, localName); }))], !ts.length(implementsTypes) ? [] : [ts.createHeritageClause(113 /* ImplementsKeyword */, ts.map(implementsTypes, function (b) { return serializeBaseType(b, staticBaseType, localName); }))]);
-                    var symbolProps = getNonInterhitedProperties(classType, baseTypes, getPropertiesOfType(classType));
-                    var publicSymbolProps = ts.filter(symbolProps, function (s) {
-                        // `valueDeclaration` could be undefined if inherited from
-                        // a union/intersection base type, but inherited properties
-                        // don't matter here.
-                        var valueDecl = s.valueDeclaration;
-                        return valueDecl && !(ts.isNamedDeclaration(valueDecl) && ts.isPrivateIdentifier(valueDecl.name));
-                    });
-                    var hasPrivateIdentifier = ts.some(symbolProps, function (s) {
-                        // `valueDeclaration` could be undefined if inherited from
-                        // a union/intersection base type, but inherited properties
-                        // don't matter here.
-                        var valueDecl = s.valueDeclaration;
-                        return valueDecl && ts.isNamedDeclaration(valueDecl) && ts.isPrivateIdentifier(valueDecl.name);
-                    });
-                    // Boil down all private properties into a single one.
-                    var privateProperties = hasPrivateIdentifier ?
-                        [ts.createProperty(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createPrivateIdentifier("#private"), 
-                            /*questionOrExclamationToken*/ undefined, 
-                            /*type*/ undefined, 
-                            /*initializer*/ undefined)] :
-                        ts.emptyArray;
-                    var publicProperties = ts.flatMap(publicSymbolProps, function (p) { return serializePropertySymbolForClass(p, /*isStatic*/ false, baseTypes[0]); });
-                    // Consider static members empty if symbol also has function or module meaning - function namespacey emit will handle statics
-                    var staticMembers = ts.flatMap(ts.filter(getPropertiesOfType(staticType), function (p) { return !(p.flags & 4194304 /* Prototype */) && p.escapedName !== "prototype" && !isNamespaceMember(p); }), function (p) { return serializePropertySymbolForClass(p, /*isStatic*/ true, staticBaseType); });
-                    var constructors = serializeSignatures(1 /* Construct */, staticType, baseTypes[0], 162 /* Constructor */);
-                    for (var _i = 0, constructors_1 = constructors; _i < constructors_1.length; _i++) {
-                        var c = constructors_1[_i];
-                        // A constructor's return type and type parameters are supposed to be controlled by the enclosing class declaration
-                        // `signatureToSignatureDeclarationHelper` appends them regardless, so for now we delete them here
-                        c.type = undefined;
-                        c.typeParameters = undefined;
-                    }
-                    var indexSignatures = serializeIndexSignatures(classType, baseTypes[0]);
-                    addResult(ts.setTextRange(ts.createClassDeclaration(
-                    /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, localName, typeParamDecls, heritageClauses, __spreadArrays(indexSignatures, staticMembers, constructors, publicProperties, privateProperties)), symbol.declarations && ts.filter(symbol.declarations, function (d) { return ts.isClassDeclaration(d) || ts.isClassExpression(d); })[0]), modifierFlags);
-                }
-                function serializeAsAlias(symbol, localName, modifierFlags) {
-                    // synthesize an alias, eg `export { symbolName as Name }`
-                    // need to mark the alias `symbol` points at
-                    // as something we need to serialize as a private declaration as well
-                    var node = getDeclarationOfAliasSymbol(symbol);
-                    if (!node)
-                        return ts.Debug.fail();
-                    var target = getMergedSymbol(getTargetOfAliasDeclaration(node, /*dontRecursivelyResolve*/ true));
-                    if (!target) {
-                        return;
-                    }
-                    var verbatimTargetName = ts.unescapeLeadingUnderscores(target.escapedName);
-                    if (verbatimTargetName === "export=" /* ExportEquals */ && (compilerOptions.esModuleInterop || compilerOptions.allowSyntheticDefaultImports)) {
-                        // target refers to an `export=` symbol that was hoisted into a synthetic default - rename here to match
-                        verbatimTargetName = "default" /* Default */;
-                    }
-                    var targetName = getInternalSymbolName(target, verbatimTargetName);
-                    includePrivateSymbol(target); // the target may be within the same scope - attempt to serialize it first
-                    switch (node.kind) {
-                        case 253 /* ImportEqualsDeclaration */:
-                            // Could be a local `import localName = ns.member` or
-                            // an external `import localName = require("whatever")`
-                            var isLocalImport = !(target.flags & 512 /* ValueModule */);
-                            addResult(ts.createImportEqualsDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createIdentifier(localName), isLocalImport
-                                ? symbolToName(target, context, 67108863 /* All */, /*expectsIdentifier*/ false)
-                                : ts.createExternalModuleReference(ts.createLiteral(getSpecifierForModuleSymbol(symbol, context)))), isLocalImport ? modifierFlags : 0 /* None */);
-                            break;
-                        case 252 /* NamespaceExportDeclaration */:
-                            // export as namespace foo
-                            // TODO: Not part of a file's local or export symbol tables
-                            // Is bound into file.symbol.globalExports instead, which we don't currently traverse
-                            addResult(ts.createNamespaceExportDeclaration(ts.idText(node.name)), 0 /* None */);
-                            break;
-                        case 255 /* ImportClause */:
-                            addResult(ts.createImportDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createImportClause(ts.createIdentifier(localName), /*namedBindings*/ undefined), 
-                            // We use `target.parent || target` below as `target.parent` is unset when the target is a module which has been export assigned
-                            // And then made into a default by the `esModuleInterop` or `allowSyntheticDefaultImports` flag
-                            // In such cases, the `target` refers to the module itself already
-                            ts.createLiteral(getSpecifierForModuleSymbol(target.parent || target, context))), 0 /* None */);
-                            break;
-                        case 256 /* NamespaceImport */:
-                            addResult(ts.createImportDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createImportClause(/*importClause*/ undefined, ts.createNamespaceImport(ts.createIdentifier(localName))), ts.createLiteral(getSpecifierForModuleSymbol(target, context))), 0 /* None */);
-                            break;
-                        case 262 /* NamespaceExport */:
-                            addResult(ts.createExportDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createNamespaceExport(ts.createIdentifier(localName)), ts.createLiteral(getSpecifierForModuleSymbol(target, context))), 0 /* None */);
-                            break;
-                        case 258 /* ImportSpecifier */:
-                            addResult(ts.createImportDeclaration(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, ts.createImportClause(/*importClause*/ undefined, ts.createNamedImports([
-                                ts.createImportSpecifier(localName !== verbatimTargetName ? ts.createIdentifier(verbatimTargetName) : undefined, ts.createIdentifier(localName))
-                            ])), ts.createLiteral(getSpecifierForModuleSymbol(target.parent || target, context))), 0 /* None */);
-                            break;
-                        case 263 /* ExportSpecifier */:
-                            // does not use localName because the symbol name in this case refers to the name in the exports table,
-                            // which we must exactly preserve
-                            var specifier = node.parent.parent.moduleSpecifier;
-                            // targetName is only used when the target is local, as otherwise the target is an alias that points at
-                            // another file
-                            serializeExportSpecifier(ts.unescapeLeadingUnderscores(symbol.escapedName), specifier ? verbatimTargetName : targetName, specifier && ts.isStringLiteralLike(specifier) ? ts.createLiteral(specifier.text) : undefined);
-                            break;
-                        case 259 /* ExportAssignment */:
-                            serializeMaybeAliasAssignment(symbol);
-                            break;
-                        case 209 /* BinaryExpression */:
-                        case 194 /* PropertyAccessExpression */:
-                            // Could be best encoded as though an export specifier or as though an export assignment
-                            // If name is default or export=, do an export assignment
-                            // Otherwise do an export specifier
-                            if (symbol.escapedName === "default" /* Default */ || symbol.escapedName === "export=" /* ExportEquals */) {
-                                serializeMaybeAliasAssignment(symbol);
-                            }
-                            else {
-                                serializeExportSpecifier(localName, targetName);
-                            }
-                            break;
-                        default:
-                            return ts.Debug.failBadSyntaxKind(node, "Unhandled alias declaration kind in symbol serializer!");
-                    }
-                }
-                function serializeExportSpecifier(localName, targetName, specifier) {
-                    addResult(ts.createExportDeclaration(
-                    /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, ts.createNamedExports([ts.createExportSpecifier(localName !== targetName ? targetName : undefined, localName)]), specifier), 0 /* None */);
-                }
-                function serializeMaybeAliasAssignment(symbol) {
-                    if (symbol.flags & 4194304 /* Prototype */) {
-                        return;
-                    }
-                    var name = ts.unescapeLeadingUnderscores(symbol.escapedName);
-                    var isExportEquals = name === "export=" /* ExportEquals */;
-                    var isDefault = name === "default" /* Default */;
-                    var isExportAssignment = isExportEquals || isDefault;
-                    // synthesize export = ref
-                    // ref should refer to either be a locally scoped symbol which we need to emit, or
-                    // a reference to another namespace/module which we may need to emit an `import` statement for
-                    var aliasDecl = symbol.declarations && getDeclarationOfAliasSymbol(symbol);
-                    // serialize what the alias points to, preserve the declaration's initializer
-                    var target = aliasDecl && getTargetOfAliasDeclaration(aliasDecl, /*dontRecursivelyResolve*/ true);
-                    // If the target resolves and resolves to a thing defined in this file, emit as an alias, otherwise emit as a const
-                    if (target && ts.length(target.declarations) && ts.some(target.declarations, function (d) { return ts.getSourceFileOfNode(d) === ts.getSourceFileOfNode(enclosingDeclaration); })) {
-                        // In case `target` refers to a namespace member, look at the declaration and serialize the leftmost symbol in it
-                        // eg, `namespace A { export class B {} }; exports = A.B;`
-                        // Technically, this is all that's required in the case where the assignment is an entity name expression
-                        var expr = isExportAssignment ? ts.getExportAssignmentExpression(aliasDecl) : ts.getPropertyAssignmentAliasLikeExpression(aliasDecl);
-                        var first_1 = ts.isEntityNameExpression(expr) ? getFirstNonModuleExportsIdentifier(expr) : undefined;
-                        var referenced = first_1 && resolveEntityName(first_1, 67108863 /* All */, /*ignoreErrors*/ true, /*dontResolveAlias*/ true, enclosingDeclaration);
-                        if (referenced || target) {
-                            includePrivateSymbol(referenced || target);
-                        }
-                        // We disable the context's symbol tracker for the duration of this name serialization
-                        // as, by virtue of being here, the name is required to print something, and we don't want to
-                        // issue a visibility error on it. Only anonymous classes that an alias points at _would_ issue
-                        // a visibility error here (as they're not visible within any scope), but we want to hoist them
-                        // into the containing scope anyway, so we want to skip the visibility checks.
-                        var oldTrack = context.tracker.trackSymbol;
-                        context.tracker.trackSymbol = ts.noop;
-                        if (isExportAssignment) {
-                            results.push(ts.createExportAssignment(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, isExportEquals, symbolToExpression(target, context, 67108863 /* All */)));
-                        }
-                        else {
-                            if (first_1 === expr) {
-                                // serialize as `export {target as name}`
-                                serializeExportSpecifier(name, ts.idText(first_1));
-                            }
-                            else if (ts.isClassExpression(expr)) {
-                                serializeExportSpecifier(name, getInternalSymbolName(target, ts.symbolName(target)));
-                            }
-                            else {
-                                // serialize as `import _Ref = t.arg.et; export { _Ref as name }`
-                                var varName = getUnusedName(name, symbol);
-                                addResult(ts.createImportEqualsDeclaration(
-                                /*decorators*/ undefined, 
-                                /*modifiers*/ undefined, ts.createIdentifier(varName), symbolToName(target, context, 67108863 /* All */, /*expectsIdentifier*/ false)), 0 /* None */);
-                                serializeExportSpecifier(name, varName);
-                            }
-                        }
-                        context.tracker.trackSymbol = oldTrack;
-                    }
-                    else {
-                        // serialize as an anonymous property declaration
-                        var varName = getUnusedName(name, symbol);
-                        // We have to use `getWidenedType` here since the object within a json file is unwidened within the file
-                        // (Unwidened types can only exist in expression contexts and should never be serialized)
-                        var typeToSerialize = getWidenedType(getTypeOfSymbol(getMergedSymbol(symbol)));
-                        if (isTypeRepresentableAsFunctionNamespaceMerge(typeToSerialize, symbol)) {
-                            // If there are no index signatures and `typeToSerialize` is an object type, emit as a namespace instead of a const
-                            serializeAsFunctionNamespaceMerge(typeToSerialize, symbol, varName, isExportAssignment ? 0 /* None */ : 1 /* Export */);
-                        }
-                        else {
-                            var statement = ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([
-                                ts.createVariableDeclaration(varName, serializeTypeForDeclaration(context, typeToSerialize, symbol, enclosingDeclaration, includePrivateSymbol, bundled))
-                            ], 2 /* Const */));
-                            addResult(statement, name === varName ? 1 /* Export */ : 0 /* None */);
-                        }
-                        if (isExportAssignment) {
-                            results.push(ts.createExportAssignment(
-                            /*decorators*/ undefined, 
-                            /*modifiers*/ undefined, isExportEquals, ts.createIdentifier(varName)));
-                        }
-                        else if (name !== varName) {
-                            serializeExportSpecifier(name, varName);
-                        }
-                    }
-                }
-                function isTypeRepresentableAsFunctionNamespaceMerge(typeToSerialize, hostSymbol) {
-                    // Only object types which are not constructable, or indexable, whose members all come from the
-                    // context source file, and whose property names are all valid identifiers and not late-bound, _and_
-                    // whose input is not type annotated (if the input symbol has an annotation we can reuse, we should prefer it)
-                    var ctxSrc = ts.getSourceFileOfNode(context.enclosingDeclaration);
-                    return ts.getObjectFlags(typeToSerialize) & (16 /* Anonymous */ | 32 /* Mapped */) &&
-                        !getIndexInfoOfType(typeToSerialize, 0 /* String */) &&
-                        !getIndexInfoOfType(typeToSerialize, 1 /* Number */) &&
-                        !!(ts.length(getPropertiesOfType(typeToSerialize)) || ts.length(getSignaturesOfType(typeToSerialize, 0 /* Call */))) &&
-                        !ts.length(getSignaturesOfType(typeToSerialize, 1 /* Construct */)) && // TODO: could probably serialize as function + ns + class, now that that's OK
-                        !getDeclarationWithTypeAnnotation(hostSymbol, enclosingDeclaration) &&
-                        !(typeToSerialize.symbol && ts.some(typeToSerialize.symbol.declarations, function (d) { return ts.getSourceFileOfNode(d) !== ctxSrc; })) &&
-                        !ts.some(getPropertiesOfType(typeToSerialize), function (p) { return isLateBoundName(p.escapedName); }) &&
-                        !ts.some(getPropertiesOfType(typeToSerialize), function (p) { return ts.some(p.declarations, function (d) { return ts.getSourceFileOfNode(d) !== ctxSrc; }); }) &&
-                        ts.every(getPropertiesOfType(typeToSerialize), function (p) { return ts.isIdentifierText(ts.symbolName(p), languageVersion) && !ts.isStringAKeyword(ts.symbolName(p)); });
-                }
-                function makeSerializePropertySymbol(createProperty, methodKind, useAccessors) {
-                    return function serializePropertySymbol(p, isStatic, baseType) {
-                        var modifierFlags = ts.getDeclarationModifierFlagsFromSymbol(p);
-                        var isPrivate = !!(modifierFlags & 8 /* Private */);
-                        if (isStatic && (p.flags & (788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */))) {
-                            // Only value-only-meaning symbols can be correctly encoded as class statics, type/namespace/alias meaning symbols
-                            // need to be merged namespace members
-                            return [];
-                        }
-                        if (p.flags & 4194304 /* Prototype */ ||
-                            (baseType && getPropertyOfType(baseType, p.escapedName)
-                                && isReadonlySymbol(getPropertyOfType(baseType, p.escapedName)) === isReadonlySymbol(p)
-                                && (p.flags & 16777216 /* Optional */) === (getPropertyOfType(baseType, p.escapedName).flags & 16777216 /* Optional */)
-                                && isTypeIdenticalTo(getTypeOfSymbol(p), getTypeOfPropertyOfType(baseType, p.escapedName)))) {
-                            return [];
-                        }
-                        var flag = (modifierFlags & ~256 /* Async */) | (isStatic ? 32 /* Static */ : 0);
-                        var name = getPropertyNameNodeForSymbol(p, context);
-                        var firstPropertyLikeDecl = ts.find(p.declarations, ts.or(ts.isPropertyDeclaration, ts.isAccessor, ts.isVariableDeclaration, ts.isPropertySignature, ts.isBinaryExpression, ts.isPropertyAccessExpression));
-                        if (p.flags & 98304 /* Accessor */ && useAccessors) {
-                            var result = [];
-                            if (p.flags & 65536 /* SetAccessor */) {
-                                result.push(ts.setTextRange(ts.createSetAccessor(
-                                /*decorators*/ undefined, ts.createModifiersFromModifierFlags(flag), name, [ts.createParameter(
-                                    /*decorators*/ undefined, 
-                                    /*modifiers*/ undefined, 
-                                    /*dotDotDotToken*/ undefined, "arg", 
-                                    /*questionToken*/ undefined, isPrivate ? undefined : serializeTypeForDeclaration(context, getTypeOfSymbol(p), p, enclosingDeclaration, includePrivateSymbol, bundled))], 
-                                /*body*/ undefined), ts.find(p.declarations, ts.isSetAccessor) || firstPropertyLikeDecl));
-                            }
-                            if (p.flags & 32768 /* GetAccessor */) {
-                                var isPrivate_1 = modifierFlags & 8 /* Private */;
-                                result.push(ts.setTextRange(ts.createGetAccessor(
-                                /*decorators*/ undefined, ts.createModifiersFromModifierFlags(flag), name, [], isPrivate_1 ? undefined : serializeTypeForDeclaration(context, getTypeOfSymbol(p), p, enclosingDeclaration, includePrivateSymbol, bundled), 
-                                /*body*/ undefined), ts.find(p.declarations, ts.isGetAccessor) || firstPropertyLikeDecl));
-                            }
-                            return result;
-                        }
-                        // This is an else/if as accessors and properties can't merge in TS, but might in JS
-                        // If this happens, we assume the accessor takes priority, as it imposes more constraints
-                        else if (p.flags & (4 /* Property */ | 3 /* Variable */)) {
-                            return ts.setTextRange(createProperty(
-                            /*decorators*/ undefined, ts.createModifiersFromModifierFlags((isReadonlySymbol(p) ? 64 /* Readonly */ : 0) | flag), name, p.flags & 16777216 /* Optional */ ? ts.createToken(57 /* QuestionToken */) : undefined, isPrivate ? undefined : serializeTypeForDeclaration(context, getTypeOfSymbol(p), p, enclosingDeclaration, includePrivateSymbol, bundled), 
-                            // TODO: https://github.com/microsoft/TypeScript/pull/32372#discussion_r328386357
-                            // interface members can't have initializers, however class members _can_
-                            /*initializer*/ undefined), ts.find(p.declarations, ts.or(ts.isPropertyDeclaration, ts.isVariableDeclaration)) || firstPropertyLikeDecl);
-                        }
-                        if (p.flags & (8192 /* Method */ | 16 /* Function */)) {
-                            var type = getTypeOfSymbol(p);
-                            var signatures = getSignaturesOfType(type, 0 /* Call */);
-                            if (flag & 8 /* Private */) {
-                                return ts.setTextRange(createProperty(
-                                /*decorators*/ undefined, ts.createModifiersFromModifierFlags((isReadonlySymbol(p) ? 64 /* Readonly */ : 0) | flag), name, p.flags & 16777216 /* Optional */ ? ts.createToken(57 /* QuestionToken */) : undefined, 
-                                /*type*/ undefined, 
-                                /*initializer*/ undefined), ts.find(p.declarations, ts.isFunctionLikeDeclaration) || signatures[0] && signatures[0].declaration || p.declarations[0]);
-                            }
-                            var results_1 = [];
-                            for (var _i = 0, signatures_3 = signatures; _i < signatures_3.length; _i++) {
-                                var sig = signatures_3[_i];
-                                // Each overload becomes a separate method declaration, in order
-                                var decl = signatureToSignatureDeclarationHelper(sig, methodKind, context);
-                                decl.name = name; // TODO: Clone
-                                if (flag) {
-                                    decl.modifiers = ts.createNodeArray(ts.createModifiersFromModifierFlags(flag));
-                                }
-                                if (p.flags & 16777216 /* Optional */) {
-                                    decl.questionToken = ts.createToken(57 /* QuestionToken */);
-                                }
-                                results_1.push(ts.setTextRange(decl, sig.declaration));
-                            }
-                            return results_1;
-                        }
-                        // The `Constructor`'s symbol isn't in the class's properties lists, obviously, since it's a signature on the static
-                        return ts.Debug.fail("Unhandled class member kind! " + (p.__debugFlags || p.flags));
-                    };
-                }
-                function serializePropertySymbolForInterface(p, baseType) {
-                    return serializePropertySymbolForInterfaceWorker(p, /*isStatic*/ false, baseType);
+            return diagnostic;
+        }
+        function createSymbol(flags, name, checkFlags) {
+            symbolCount++;
+            var symbol = (new Symbol(flags | 33554432 /* Transient */, name));
+            symbol.checkFlags = checkFlags || 0;
+            return symbol;
+        }
+        function getExcludedSymbolFlags(flags) {
+            var result = 0;
+            if (flags & 2 /* BlockScopedVariable */)
+                result |= 111551 /* BlockScopedVariableExcludes */;
+            if (flags & 1 /* FunctionScopedVariable */)
+                result |= 111550 /* FunctionScopedVariableExcludes */;
+            if (flags & 4 /* Property */)
+                result |= 0 /* PropertyExcludes */;
+            if (flags & 8 /* EnumMember */)
+                result |= 900095 /* EnumMemberExcludes */;
+            if (flags & 16 /* Function */)
+                result |= 110991 /* FunctionExcludes */;
+            if (flags & 32 /* Class */)
+                result |= 899503 /* ClassExcludes */;
+            if (flags & 64 /* Interface */)
+                result |= 788872 /* InterfaceExcludes */;
+            if (flags & 256 /* RegularEnum */)
+                result |= 899327 /* RegularEnumExcludes */;
+            if (flags & 128 /* ConstEnum */)
+                result |= 899967 /* ConstEnumExcludes */;
+            if (flags & 512 /* ValueModule */)
+                result |= 110735 /* ValueModuleExcludes */;
+            if (flags & 8192 /* Method */)
+                result |= 103359 /* MethodExcludes */;
+            if (flags & 32768 /* GetAccessor */)
+                result |= 46015 /* GetAccessorExcludes */;
+            if (flags & 65536 /* SetAccessor */)
+                result |= 78783 /* SetAccessorExcludes */;
+            if (flags & 262144 /* TypeParameter */)
+                result |= 526824 /* TypeParameterExcludes */;
+            if (flags & 524288 /* TypeAlias */)
+                result |= 788968 /* TypeAliasExcludes */;
+            if (flags & 2097152 /* Alias */)
+                result |= 2097152 /* AliasExcludes */;
+            return result;
+        }
+        function recordMergedSymbol(target, source) {
+            if (!source.mergeId) {
+                source.mergeId = nextMergeId;
+                nextMergeId++;
+            }
+            mergedSymbols[source.mergeId] = target;
+        }
+        function cloneSymbol(symbol) {
+            var result = createSymbol(symbol.flags, symbol.escapedName);
+            result.declarations = symbol.declarations ? symbol.declarations.slice() : [];
+            result.parent = symbol.parent;
+            if (symbol.valueDeclaration)
+                result.valueDeclaration = symbol.valueDeclaration;
+            if (symbol.constEnumOnlyModule)
+                result.constEnumOnlyModule = true;
+            if (symbol.members)
+                result.members = new ts.Map(symbol.members);
+            if (symbol.exports)
+                result.exports = new ts.Map(symbol.exports);
+            recordMergedSymbol(result, symbol);
+            return result;
+        }
+        /**
+         * Note: if target is transient, then it is mutable, and mergeSymbol with both mutate and return it.
+         * If target is not transient, mergeSymbol will produce a transient clone, mutate that and return it.
+         */
+        function mergeSymbol(target, source, unidirectional) {
+            if (unidirectional === void 0) { unidirectional = false; }
+            if (!(target.flags & getExcludedSymbolFlags(source.flags)) ||
+                (source.flags | target.flags) & 67108864 /* Assignment */) {
+                if (source === target) {
+                    // This can happen when an export assigned namespace exports something also erroneously exported at the top level
+                    // See `declarationFileNoCrashOnExtraExportModifier` for an example
+                    return target;
                 }
-                function serializeSignatures(kind, input, baseType, outputKind) {
-                    var signatures = getSignaturesOfType(input, kind);
-                    if (kind === 1 /* Construct */) {
-                        if (!baseType && ts.every(signatures, function (s) { return ts.length(s.parameters) === 0; })) {
-                            return []; // No base type, every constructor is empty - elide the extraneous `constructor()`
-                        }
-                        if (baseType) {
-                            // If there is a base type, if every signature in the class is identical to a signature in the baseType, elide all the declarations
-                            var baseSigs = getSignaturesOfType(baseType, 1 /* Construct */);
-                            if (!ts.length(baseSigs) && ts.every(signatures, function (s) { return ts.length(s.parameters) === 0; })) {
-                                return []; // Base had no explicit signatures, if all our signatures are also implicit, return an empty list
-                            }
-                            if (baseSigs.length === signatures.length) {
-                                var failed = false;
-                                for (var i = 0; i < baseSigs.length; i++) {
-                                    if (!compareSignaturesIdentical(signatures[i], baseSigs[i], /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ true, compareTypesIdentical)) {
-                                        failed = true;
-                                        break;
-                                    }
-                                }
-                                if (!failed) {
-                                    return []; // Every signature was identical - elide constructor list as it is inherited
-                                }
-                            }
-                        }
-                        var privateProtected = 0;
-                        for (var _i = 0, signatures_4 = signatures; _i < signatures_4.length; _i++) {
-                            var s = signatures_4[_i];
-                            if (s.declaration) {
-                                privateProtected |= ts.getSelectedModifierFlags(s.declaration, 8 /* Private */ | 16 /* Protected */);
-                            }
-                        }
-                        if (privateProtected) {
-                            return [ts.setTextRange(ts.createConstructor(
-                                /*decorators*/ undefined, ts.createModifiersFromModifierFlags(privateProtected), 
-                                /*parameters*/ [], 
-                                /*body*/ undefined), signatures[0].declaration)];
-                        }
-                    }
-                    var results = [];
-                    for (var _a = 0, signatures_5 = signatures; _a < signatures_5.length; _a++) {
-                        var sig = signatures_5[_a];
-                        // Each overload becomes a separate constructor declaration, in order
-                        var decl = signatureToSignatureDeclarationHelper(sig, outputKind, context);
-                        results.push(ts.setTextRange(decl, sig.declaration));
+                if (!(target.flags & 33554432 /* Transient */)) {
+                    var resolvedTarget = resolveSymbol(target);
+                    if (resolvedTarget === unknownSymbol) {
+                        return source;
                     }
-                    return results;
+                    target = cloneSymbol(resolvedTarget);
                 }
-                function serializeIndexSignatures(input, baseType) {
-                    var results = [];
-                    for (var _i = 0, _a = [0 /* String */, 1 /* Number */]; _i < _a.length; _i++) {
-                        var type = _a[_i];
-                        var info = getIndexInfoOfType(input, type);
-                        if (info) {
-                            if (baseType) {
-                                var baseInfo = getIndexInfoOfType(baseType, type);
-                                if (baseInfo) {
-                                    if (isTypeIdenticalTo(info.type, baseInfo.type)) {
-                                        continue; // elide identical index signatures
-                                    }
-                                }
-                            }
-                            results.push(indexInfoToIndexSignatureDeclarationHelper(info, type, context));
-                        }
-                    }
-                    return results;
+                // Javascript static-property-assignment declarations always merge, even though they are also values
+                if (source.flags & 512 /* ValueModule */ && target.flags & 512 /* ValueModule */ && target.constEnumOnlyModule && !source.constEnumOnlyModule) {
+                    // reset flag when merging instantiated module into value module that has only const enums
+                    target.constEnumOnlyModule = false;
                 }
-                function serializeBaseType(t, staticType, rootName) {
-                    var ref = trySerializeAsTypeReference(t);
-                    if (ref) {
-                        return ref;
-                    }
-                    var tempName = getUnusedName(rootName + "_base");
-                    var statement = ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([
-                        ts.createVariableDeclaration(tempName, typeToTypeNodeHelper(staticType, context))
-                    ], 2 /* Const */));
-                    addResult(statement, 0 /* None */);
-                    return ts.createExpressionWithTypeArguments(/*typeArgs*/ undefined, ts.createIdentifier(tempName));
+                target.flags |= source.flags;
+                if (source.valueDeclaration) {
+                    ts.setValueDeclaration(target, source.valueDeclaration);
                 }
-                function trySerializeAsTypeReference(t) {
-                    var typeArgs;
-                    var reference;
-                    // We don't use `isValueSymbolAccessible` below. since that considers alternative containers (like modules)
-                    // which we can't write out in a syntactically valid way as an expression
-                    if (t.target && getAccessibleSymbolChain(t.target.symbol, enclosingDeclaration, 111551 /* Value */, /*useOnlyExternalAliasing*/ false)) {
-                        typeArgs = ts.map(getTypeArguments(t), function (t) { return typeToTypeNodeHelper(t, context); });
-                        reference = symbolToExpression(t.target.symbol, context, 788968 /* Type */);
-                    }
-                    else if (t.symbol && getAccessibleSymbolChain(t.symbol, enclosingDeclaration, 111551 /* Value */, /*useOnlyExternalAliasing*/ false)) {
-                        reference = symbolToExpression(t.symbol, context, 788968 /* Type */);
-                    }
-                    if (reference) {
-                        return ts.createExpressionWithTypeArguments(typeArgs, reference);
-                    }
+                ts.addRange(target.declarations, source.declarations);
+                if (source.members) {
+                    if (!target.members)
+                        target.members = ts.createSymbolTable();
+                    mergeSymbolTable(target.members, source.members, unidirectional);
                 }
-                function getUnusedName(input, symbol) {
-                    if (symbol) {
-                        if (context.remappedSymbolNames.has("" + getSymbolId(symbol))) {
-                            return context.remappedSymbolNames.get("" + getSymbolId(symbol));
-                        }
-                    }
-                    if (symbol) {
-                        input = getNameCandidateWorker(symbol, input);
-                    }
-                    var i = 0;
-                    var original = input;
-                    while (context.usedSymbolNames.has(input)) {
-                        i++;
-                        input = original + "_" + i;
-                    }
-                    context.usedSymbolNames.set(input, true);
-                    if (symbol) {
-                        context.remappedSymbolNames.set("" + getSymbolId(symbol), input);
-                    }
-                    return input;
+                if (source.exports) {
+                    if (!target.exports)
+                        target.exports = ts.createSymbolTable();
+                    mergeSymbolTable(target.exports, source.exports, unidirectional);
                 }
-                function getNameCandidateWorker(symbol, localName) {
-                    if (localName === "default" /* Default */ || localName === "__class" /* Class */ || localName === "__function" /* Function */) {
-                        var flags = context.flags;
-                        context.flags |= 16777216 /* InInitialEntityName */;
-                        var nameCandidate = getNameOfSymbolAsWritten(symbol, context);
-                        context.flags = flags;
-                        localName = nameCandidate.length > 0 && ts.isSingleOrDoubleQuote(nameCandidate.charCodeAt(0)) ? ts.stripQuotes(nameCandidate) : nameCandidate;
-                    }
-                    if (localName === "default" /* Default */) {
-                        localName = "_default";
-                    }
-                    else if (localName === "export=" /* ExportEquals */) {
-                        localName = "_exports";
-                    }
-                    localName = ts.isIdentifierText(localName, languageVersion) && !ts.isStringANonContextualKeyword(localName) ? localName : "_" + localName.replace(/[^a-zA-Z0-9]/g, "_");
-                    return localName;
+                if (!unidirectional) {
+                    recordMergedSymbol(target, source);
                 }
-                function getInternalSymbolName(symbol, localName) {
-                    if (context.remappedSymbolNames.has("" + getSymbolId(symbol))) {
-                        return context.remappedSymbolNames.get("" + getSymbolId(symbol));
-                    }
-                    localName = getNameCandidateWorker(symbol, localName);
-                    // The result of this is going to be used as the symbol's name - lock it in, so `getUnusedName` will also pick it up
-                    context.remappedSymbolNames.set("" + getSymbolId(symbol), localName);
-                    return localName;
+            }
+            else if (target.flags & 1024 /* NamespaceModule */) {
+                // Do not report an error when merging `var globalThis` with the built-in `globalThis`,
+                // as we will already report a "Declaration name conflicts..." error, and this error
+                // won't make much sense.
+                if (target !== globalThisSymbol) {
+                    error(ts.getNameOfDeclaration(source.declarations[0]), ts.Diagnostics.Cannot_augment_module_0_with_value_exports_because_it_resolves_to_a_non_module_entity, symbolToString(target));
+                }
+            }
+            else { // error
+                var isEitherEnum = !!(target.flags & 384 /* Enum */ || source.flags & 384 /* Enum */);
+                var isEitherBlockScoped_1 = !!(target.flags & 2 /* BlockScopedVariable */ || source.flags & 2 /* BlockScopedVariable */);
+                var message = isEitherEnum
+                    ? ts.Diagnostics.Enum_declarations_can_only_merge_with_namespace_or_other_enum_declarations
+                    : isEitherBlockScoped_1
+                        ? ts.Diagnostics.Cannot_redeclare_block_scoped_variable_0
+                        : ts.Diagnostics.Duplicate_identifier_0;
+                var sourceSymbolFile = source.declarations && ts.getSourceFileOfNode(source.declarations[0]);
+                var targetSymbolFile = target.declarations && ts.getSourceFileOfNode(target.declarations[0]);
+                var symbolName_1 = symbolToString(source);
+                // Collect top-level duplicate identifier errors into one mapping, so we can then merge their diagnostics if there are a bunch
+                if (sourceSymbolFile && targetSymbolFile && amalgamatedDuplicates && !isEitherEnum && sourceSymbolFile !== targetSymbolFile) {
+                    var firstFile_1 = ts.comparePaths(sourceSymbolFile.path, targetSymbolFile.path) === -1 /* LessThan */ ? sourceSymbolFile : targetSymbolFile;
+                    var secondFile_1 = firstFile_1 === sourceSymbolFile ? targetSymbolFile : sourceSymbolFile;
+                    var filesDuplicates = ts.getOrUpdate(amalgamatedDuplicates, firstFile_1.path + "|" + secondFile_1.path, function () {
+                        return ({ firstFile: firstFile_1, secondFile: secondFile_1, conflictingSymbols: new ts.Map() });
+                    });
+                    var conflictingSymbolInfo = ts.getOrUpdate(filesDuplicates.conflictingSymbols, symbolName_1, function () {
+                        return ({ isBlockScoped: isEitherBlockScoped_1, firstFileLocations: [], secondFileLocations: [] });
+                    });
+                    addDuplicateLocations(conflictingSymbolInfo.firstFileLocations, source);
+                    addDuplicateLocations(conflictingSymbolInfo.secondFileLocations, target);
+                }
+                else {
+                    addDuplicateDeclarationErrorsForSymbols(source, message, symbolName_1, target);
+                    addDuplicateDeclarationErrorsForSymbols(target, message, symbolName_1, source);
+                }
+            }
+            return target;
+            function addDuplicateLocations(locs, symbol) {
+                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                    var decl = _a[_i];
+                    ts.pushIfUnique(locs, decl);
                 }
             }
         }
-        function typePredicateToString(typePredicate, enclosingDeclaration, flags, writer) {
-            if (flags === void 0) { flags = 16384 /* UseAliasDefinedOutsideCurrentScope */; }
-            return writer ? typePredicateToStringWorker(writer).getText() : ts.usingSingleLineStringWriter(typePredicateToStringWorker);
-            function typePredicateToStringWorker(writer) {
-                var predicate = ts.createTypePredicateNodeWithModifier(typePredicate.kind === 2 /* AssertsThis */ || typePredicate.kind === 3 /* AssertsIdentifier */ ? ts.createToken(124 /* AssertsKeyword */) : undefined, typePredicate.kind === 1 /* Identifier */ || typePredicate.kind === 3 /* AssertsIdentifier */ ? ts.createIdentifier(typePredicate.parameterName) : ts.createThisTypeNode(), typePredicate.type && nodeBuilder.typeToTypeNode(typePredicate.type, enclosingDeclaration, toNodeBuilderFlags(flags) | 70221824 /* IgnoreErrors */ | 512 /* WriteTypeParametersInQualifiedName */) // TODO: GH#18217
-                );
-                var printer = ts.createPrinter({ removeComments: true });
-                var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
-                printer.writeNode(4 /* Unspecified */, predicate, /*sourceFile*/ sourceFile, writer);
-                return writer;
+        function addDuplicateDeclarationErrorsForSymbols(target, message, symbolName, source) {
+            ts.forEach(target.declarations, function (node) {
+                addDuplicateDeclarationError(node, message, symbolName, source.declarations);
+            });
+        }
+        function addDuplicateDeclarationError(node, message, symbolName, relatedNodes) {
+            var errorNode = (ts.getExpandoInitializer(node, /*isPrototypeAssignment*/ false) ? ts.getNameOfExpando(node) : ts.getNameOfDeclaration(node)) || node;
+            var err = lookupOrIssueError(errorNode, message, symbolName);
+            var _loop_7 = function (relatedNode) {
+                var adjustedNode = (ts.getExpandoInitializer(relatedNode, /*isPrototypeAssignment*/ false) ? ts.getNameOfExpando(relatedNode) : ts.getNameOfDeclaration(relatedNode)) || relatedNode;
+                if (adjustedNode === errorNode)
+                    return "continue";
+                err.relatedInformation = err.relatedInformation || [];
+                var leadingMessage = ts.createDiagnosticForNode(adjustedNode, ts.Diagnostics._0_was_also_declared_here, symbolName);
+                var followOnMessage = ts.createDiagnosticForNode(adjustedNode, ts.Diagnostics.and_here);
+                if (ts.length(err.relatedInformation) >= 5 || ts.some(err.relatedInformation, function (r) { return ts.compareDiagnostics(r, followOnMessage) === 0 /* EqualTo */ || ts.compareDiagnostics(r, leadingMessage) === 0 /* EqualTo */; }))
+                    return "continue";
+                ts.addRelatedInfo(err, !ts.length(err.relatedInformation) ? leadingMessage : followOnMessage);
+            };
+            for (var _i = 0, _a = relatedNodes || ts.emptyArray; _i < _a.length; _i++) {
+                var relatedNode = _a[_i];
+                _loop_7(relatedNode);
             }
         }
-        function formatUnionTypes(types) {
-            var result = [];
-            var flags = 0;
-            for (var i = 0; i < types.length; i++) {
-                var t = types[i];
-                flags |= t.flags;
-                if (!(t.flags & 98304 /* Nullable */)) {
-                    if (t.flags & (512 /* BooleanLiteral */ | 1024 /* EnumLiteral */)) {
-                        var baseType = t.flags & 512 /* BooleanLiteral */ ? booleanType : getBaseTypeOfEnumLiteralType(t);
-                        if (baseType.flags & 1048576 /* Union */) {
-                            var count = baseType.types.length;
-                            if (i + count <= types.length && getRegularTypeOfLiteralType(types[i + count - 1]) === getRegularTypeOfLiteralType(baseType.types[count - 1])) {
-                                result.push(baseType);
-                                i += count - 1;
-                                continue;
+        function combineSymbolTables(first, second) {
+            if (!(first === null || first === void 0 ? void 0 : first.size))
+                return second;
+            if (!(second === null || second === void 0 ? void 0 : second.size))
+                return first;
+            var combined = ts.createSymbolTable();
+            mergeSymbolTable(combined, first);
+            mergeSymbolTable(combined, second);
+            return combined;
+        }
+        function mergeSymbolTable(target, source, unidirectional) {
+            if (unidirectional === void 0) { unidirectional = false; }
+            source.forEach(function (sourceSymbol, id) {
+                var targetSymbol = target.get(id);
+                target.set(id, targetSymbol ? mergeSymbol(targetSymbol, sourceSymbol, unidirectional) : sourceSymbol);
+            });
+        }
+        function mergeModuleAugmentation(moduleName) {
+            var _a, _b;
+            var moduleAugmentation = moduleName.parent;
+            if (moduleAugmentation.symbol.declarations[0] !== moduleAugmentation) {
+                // this is a combined symbol for multiple augmentations within the same file.
+                // its symbol already has accumulated information for all declarations
+                // so we need to add it just once - do the work only for first declaration
+                ts.Debug.assert(moduleAugmentation.symbol.declarations.length > 1);
+                return;
+            }
+            if (ts.isGlobalScopeAugmentation(moduleAugmentation)) {
+                mergeSymbolTable(globals, moduleAugmentation.symbol.exports);
+            }
+            else {
+                // find a module that about to be augmented
+                // do not validate names of augmentations that are defined in ambient context
+                var moduleNotFoundError = !(moduleName.parent.parent.flags & 8388608 /* Ambient */)
+                    ? ts.Diagnostics.Invalid_module_name_in_augmentation_module_0_cannot_be_found
+                    : undefined;
+                var mainModule_1 = resolveExternalModuleNameWorker(moduleName, moduleName, moduleNotFoundError, /*isForAugmentation*/ true);
+                if (!mainModule_1) {
+                    return;
+                }
+                // obtain item referenced by 'export='
+                mainModule_1 = resolveExternalModuleSymbol(mainModule_1);
+                if (mainModule_1.flags & 1920 /* Namespace */) {
+                    // If we're merging an augmentation to a pattern ambient module, we want to
+                    // perform the merge unidirectionally from the augmentation ('a.foo') to
+                    // the pattern ('*.foo'), so that 'getMergedSymbol()' on a.foo gives you
+                    // all the exports both from the pattern and from the augmentation, but
+                    // 'getMergedSymbol()' on *.foo only gives you exports from *.foo.
+                    if (ts.some(patternAmbientModules, function (module) { return mainModule_1 === module.symbol; })) {
+                        var merged = mergeSymbol(moduleAugmentation.symbol, mainModule_1, /*unidirectional*/ true);
+                        if (!patternAmbientModuleAugmentations) {
+                            patternAmbientModuleAugmentations = new ts.Map();
+                        }
+                        // moduleName will be a StringLiteral since this is not `declare global`.
+                        patternAmbientModuleAugmentations.set(moduleName.text, merged);
+                    }
+                    else {
+                        if (((_a = mainModule_1.exports) === null || _a === void 0 ? void 0 : _a.get("__export" /* ExportStar */)) && ((_b = moduleAugmentation.symbol.exports) === null || _b === void 0 ? void 0 : _b.size)) {
+                            // We may need to merge the module augmentation's exports into the target symbols of the resolved exports
+                            var resolvedExports = getResolvedMembersOrExportsOfSymbol(mainModule_1, "resolvedExports" /* resolvedExports */);
+                            for (var _i = 0, _c = ts.arrayFrom(moduleAugmentation.symbol.exports.entries()); _i < _c.length; _i++) {
+                                var _d = _c[_i], key = _d[0], value = _d[1];
+                                if (resolvedExports.has(key) && !mainModule_1.exports.has(key)) {
+                                    mergeSymbol(resolvedExports.get(key), value);
+                                }
                             }
                         }
+                        mergeSymbol(mainModule_1, moduleAugmentation.symbol);
                     }
-                    result.push(t);
+                }
+                else {
+                    // moduleName will be a StringLiteral since this is not `declare global`.
+                    error(moduleName, ts.Diagnostics.Cannot_augment_module_0_because_it_resolves_to_a_non_module_entity, moduleName.text);
                 }
             }
-            if (flags & 65536 /* Null */)
-                result.push(nullType);
-            if (flags & 32768 /* Undefined */)
-                result.push(undefinedType);
-            return result || types;
-        }
-        function visibilityToString(flags) {
-            if (flags === 8 /* Private */) {
-                return "private";
-            }
-            if (flags === 16 /* Protected */) {
-                return "protected";
-            }
-            return "public";
         }
-        function getTypeAliasForTypeLiteral(type) {
-            if (type.symbol && type.symbol.flags & 2048 /* TypeLiteral */) {
-                var node = ts.findAncestor(type.symbol.declarations[0].parent, function (n) { return n.kind !== 182 /* ParenthesizedType */; });
-                if (node.kind === 247 /* TypeAliasDeclaration */) {
-                    return getSymbolOfNode(node);
+        function addToSymbolTable(target, source, message) {
+            source.forEach(function (sourceSymbol, id) {
+                var targetSymbol = target.get(id);
+                if (targetSymbol) {
+                    // Error on redeclarations
+                    ts.forEach(targetSymbol.declarations, addDeclarationDiagnostic(ts.unescapeLeadingUnderscores(id), message));
                 }
+                else {
+                    target.set(id, sourceSymbol);
+                }
+            });
+            function addDeclarationDiagnostic(id, message) {
+                return function (declaration) { return diagnostics.add(ts.createDiagnosticForNode(declaration, message, id)); };
             }
-            return undefined;
         }
-        function isTopLevelInExternalModuleAugmentation(node) {
-            return node && node.parent &&
-                node.parent.kind === 250 /* ModuleBlock */ &&
-                ts.isExternalModuleAugmentation(node.parent.parent);
+        function getSymbolLinks(symbol) {
+            if (symbol.flags & 33554432 /* Transient */)
+                return symbol;
+            var id = getSymbolId(symbol);
+            return symbolLinks[id] || (symbolLinks[id] = new SymbolLinks());
         }
-        function isDefaultBindingContext(location) {
-            return location.kind === 290 /* SourceFile */ || ts.isAmbientModule(location);
+        function getNodeLinks(node) {
+            var nodeId = getNodeId(node);
+            return nodeLinks[nodeId] || (nodeLinks[nodeId] = new NodeLinks());
         }
-        function getNameOfSymbolFromNameType(symbol, context) {
-            var nameType = getSymbolLinks(symbol).nameType;
-            if (nameType) {
-                if (nameType.flags & 384 /* StringOrNumberLiteral */) {
-                    var name = "" + nameType.value;
-                    if (!ts.isIdentifierText(name, compilerOptions.target) && !isNumericLiteralName(name)) {
-                        return "\"" + ts.escapeString(name, 34 /* doubleQuote */) + "\"";
+        function isGlobalSourceFile(node) {
+            return node.kind === 297 /* SourceFile */ && !ts.isExternalOrCommonJsModule(node);
+        }
+        function getSymbol(symbols, name, meaning) {
+            if (meaning) {
+                var symbol = getMergedSymbol(symbols.get(name));
+                if (symbol) {
+                    ts.Debug.assert((ts.getCheckFlags(symbol) & 1 /* Instantiated */) === 0, "Should never get an instantiated symbol here.");
+                    if (symbol.flags & meaning) {
+                        return symbol;
                     }
-                    if (isNumericLiteralName(name) && ts.startsWith(name, "-")) {
-                        return "[" + name + "]";
+                    if (symbol.flags & 2097152 /* Alias */) {
+                        var target = resolveAlias(symbol);
+                        // Unknown symbol means an error occurred in alias resolution, treat it as positive answer to avoid cascading errors
+                        if (target === unknownSymbol || target.flags & meaning) {
+                            return symbol;
+                        }
                     }
-                    return name;
-                }
-                if (nameType.flags & 8192 /* UniqueESSymbol */) {
-                    return "[" + getNameOfSymbolAsWritten(nameType.symbol, context) + "]";
                 }
             }
+            // return undefined if we can't find a symbol.
         }
         /**
-         * Gets a human-readable name for a symbol.
-         * Should *not* be used for the right-hand side of a `.` -- use `symbolName(symbol)` for that instead.
-         *
-         * Unlike `symbolName(symbol)`, this will include quotes if the name is from a string literal.
-         * It will also use a representation of a number as written instead of a decimal form, e.g. `0o11` instead of `9`.
+         * Get symbols that represent parameter-property-declaration as parameter and as property declaration
+         * @param parameter a parameterDeclaration node
+         * @param parameterName a name of the parameter to get the symbols for.
+         * @return a tuple of two symbols
          */
-        function getNameOfSymbolAsWritten(symbol, context) {
-            if (context && symbol.escapedName === "default" /* Default */ && !(context.flags & 16384 /* UseAliasDefinedOutsideCurrentScope */) &&
-                // If it's not the first part of an entity name, it must print as `default`
-                (!(context.flags & 16777216 /* InInitialEntityName */) ||
-                    // if the symbol is synthesized, it will only be referenced externally it must print as `default`
-                    !symbol.declarations ||
-                    // if not in the same binding context (source file, module declaration), it must print as `default`
-                    (context.enclosingDeclaration && ts.findAncestor(symbol.declarations[0], isDefaultBindingContext) !== ts.findAncestor(context.enclosingDeclaration, isDefaultBindingContext)))) {
-                return "default";
+        function getSymbolsOfParameterPropertyDeclaration(parameter, parameterName) {
+            var constructorDeclaration = parameter.parent;
+            var classDeclaration = parameter.parent.parent;
+            var parameterSymbol = getSymbol(constructorDeclaration.locals, parameterName, 111551 /* Value */);
+            var propertySymbol = getSymbol(getMembersOfSymbol(classDeclaration.symbol), parameterName, 111551 /* Value */);
+            if (parameterSymbol && propertySymbol) {
+                return [parameterSymbol, propertySymbol];
             }
-            if (symbol.declarations && symbol.declarations.length) {
-                var declaration = ts.firstDefined(symbol.declarations, function (d) { return ts.getNameOfDeclaration(d) ? d : undefined; }); // Try using a declaration with a name, first
-                var name_2 = declaration && ts.getNameOfDeclaration(declaration);
-                if (declaration && name_2) {
-                    if (ts.isCallExpression(declaration) && ts.isBindableObjectDefinePropertyCall(declaration)) {
-                        return ts.symbolName(symbol);
-                    }
-                    if (ts.isComputedPropertyName(name_2) && !(ts.getCheckFlags(symbol) & 4096 /* Late */)) {
-                        var nameType = getSymbolLinks(symbol).nameType;
-                        if (nameType && nameType.flags & 384 /* StringOrNumberLiteral */) {
-                            // Computed property name isn't late bound, but has a well-known name type - use name type to generate a symbol name
-                            var result = getNameOfSymbolFromNameType(symbol, context);
-                            if (result !== undefined) {
-                                return result;
-                            }
-                        }
+            return ts.Debug.fail("There should exist two symbols, one as property declaration and one as parameter declaration");
+        }
+        function isBlockScopedNameDeclaredBeforeUse(declaration, usage) {
+            var declarationFile = ts.getSourceFileOfNode(declaration);
+            var useFile = ts.getSourceFileOfNode(usage);
+            var declContainer = ts.getEnclosingBlockScopeContainer(declaration);
+            if (declarationFile !== useFile) {
+                if ((moduleKind && (declarationFile.externalModuleIndicator || useFile.externalModuleIndicator)) ||
+                    (!ts.outFile(compilerOptions)) ||
+                    isInTypeQuery(usage) ||
+                    declaration.flags & 8388608 /* Ambient */) {
+                    // nodes are in different files and order cannot be determined
+                    return true;
+                }
+                // declaration is after usage
+                // can be legal if usage is deferred (i.e. inside function or in initializer of instance property)
+                if (isUsedInFunctionOrInstanceProperty(usage, declaration)) {
+                    return true;
+                }
+                var sourceFiles = host.getSourceFiles();
+                return sourceFiles.indexOf(declarationFile) <= sourceFiles.indexOf(useFile);
+            }
+            if (declaration.pos <= usage.pos && !(ts.isPropertyDeclaration(declaration) && ts.isThisProperty(usage.parent) && !declaration.initializer && !declaration.exclamationToken)) {
+                // declaration is before usage
+                if (declaration.kind === 198 /* BindingElement */) {
+                    // still might be illegal if declaration and usage are both binding elements (eg var [a = b, b = b] = [1, 2])
+                    var errorBindingElement = ts.getAncestor(usage, 198 /* BindingElement */);
+                    if (errorBindingElement) {
+                        return ts.findAncestor(errorBindingElement, ts.isBindingElement) !== ts.findAncestor(declaration, ts.isBindingElement) ||
+                            declaration.pos < errorBindingElement.pos;
                     }
-                    return ts.declarationNameToString(name_2);
+                    // or it might be illegal if usage happens before parent variable is declared (eg var [a] = a)
+                    return isBlockScopedNameDeclaredBeforeUse(ts.getAncestor(declaration, 249 /* VariableDeclaration */), usage);
                 }
-                if (!declaration) {
-                    declaration = symbol.declarations[0]; // Declaration may be nameless, but we'll try anyway
+                else if (declaration.kind === 249 /* VariableDeclaration */) {
+                    // still might be illegal if usage is in the initializer of the variable declaration (eg var a = a)
+                    return !isImmediatelyUsedInInitializerOfBlockScopedVariable(declaration, usage);
                 }
-                if (declaration.parent && declaration.parent.kind === 242 /* VariableDeclaration */) {
-                    return ts.declarationNameToString(declaration.parent.name);
+                else if (ts.isClassDeclaration(declaration)) {
+                    // still might be illegal if the usage is within a computed property name in the class (eg class A { static p = "a"; [A.p]() {} })
+                    return !ts.findAncestor(usage, function (n) { return ts.isComputedPropertyName(n) && n.parent.parent === declaration; });
                 }
-                switch (declaration.kind) {
-                    case 214 /* ClassExpression */:
-                    case 201 /* FunctionExpression */:
-                    case 202 /* ArrowFunction */:
-                        if (context && !context.encounteredError && !(context.flags & 131072 /* AllowAnonymousIdentifier */)) {
-                            context.encounteredError = true;
-                        }
-                        return declaration.kind === 214 /* ClassExpression */ ? "(Anonymous class)" : "(Anonymous function)";
+                else if (ts.isPropertyDeclaration(declaration)) {
+                    // still might be illegal if a self-referencing property initializer (eg private x = this.x)
+                    return !isPropertyImmediatelyReferencedWithinDeclaration(declaration, usage, /*stopAtAnyPropertyDeclaration*/ false);
+                }
+                else if (ts.isParameterPropertyDeclaration(declaration, declaration.parent)) {
+                    // foo = this.bar is illegal in esnext+useDefineForClassFields when bar is a parameter property
+                    return !(compilerOptions.target === 99 /* ESNext */ && !!compilerOptions.useDefineForClassFields
+                        && ts.getContainingClass(declaration) === ts.getContainingClass(usage)
+                        && isUsedInFunctionOrInstanceProperty(usage, declaration));
                 }
+                return true;
             }
-            var name = getNameOfSymbolFromNameType(symbol, context);
-            return name !== undefined ? name : ts.symbolName(symbol);
-        }
-        function isDeclarationVisible(node) {
-            if (node) {
-                var links = getNodeLinks(node);
-                if (links.isVisible === undefined) {
-                    links.isVisible = !!determineIfDeclarationIsVisible();
+            // declaration is after usage, but it can still be legal if usage is deferred:
+            // 1. inside an export specifier
+            // 2. inside a function
+            // 3. inside an instance property initializer, a reference to a non-instance property
+            //    (except when target: "esnext" and useDefineForClassFields: true and the reference is to a parameter property)
+            // 4. inside a static property initializer, a reference to a static method in the same class
+            // 5. inside a TS export= declaration (since we will move the export statement during emit to avoid TDZ)
+            // or if usage is in a type context:
+            // 1. inside a type query (typeof in type position)
+            // 2. inside a jsdoc comment
+            if (usage.parent.kind === 270 /* ExportSpecifier */ || (usage.parent.kind === 266 /* ExportAssignment */ && usage.parent.isExportEquals)) {
+                // export specifiers do not use the variable, they only make it available for use
+                return true;
+            }
+            // When resolving symbols for exports, the `usage` location passed in can be the export site directly
+            if (usage.kind === 266 /* ExportAssignment */ && usage.isExportEquals) {
+                return true;
+            }
+            if (!!(usage.flags & 4194304 /* JSDoc */) || isInTypeQuery(usage) || usageInTypeDeclaration()) {
+                return true;
+            }
+            if (isUsedInFunctionOrInstanceProperty(usage, declaration)) {
+                if (compilerOptions.target === 99 /* ESNext */ && !!compilerOptions.useDefineForClassFields
+                    && ts.getContainingClass(declaration)
+                    && (ts.isPropertyDeclaration(declaration) || ts.isParameterPropertyDeclaration(declaration, declaration.parent))) {
+                    return !isPropertyImmediatelyReferencedWithinDeclaration(declaration, usage, /*stopAtAnyPropertyDeclaration*/ true);
+                }
+                else {
+                    return true;
                 }
-                return links.isVisible;
             }
             return false;
-            function determineIfDeclarationIsVisible() {
-                switch (node.kind) {
-                    case 315 /* JSDocCallbackTag */:
-                    case 322 /* JSDocTypedefTag */:
-                    case 316 /* JSDocEnumTag */:
-                        // Top-level jsdoc type aliases are considered exported
-                        // First parent is comment node, second is hosting declaration or token; we only care about those tokens or declarations whose parent is a source file
-                        return !!(node.parent && node.parent.parent && node.parent.parent.parent && ts.isSourceFile(node.parent.parent.parent));
-                    case 191 /* BindingElement */:
-                        return isDeclarationVisible(node.parent.parent);
-                    case 242 /* VariableDeclaration */:
-                        if (ts.isBindingPattern(node.name) &&
-                            !node.name.elements.length) {
-                            // If the binding pattern is empty, this variable declaration is not visible
-                            return false;
-                        }
-                    // falls through
-                    case 249 /* ModuleDeclaration */:
-                    case 245 /* ClassDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 244 /* FunctionDeclaration */:
-                    case 248 /* EnumDeclaration */:
-                    case 253 /* ImportEqualsDeclaration */:
-                        // external module augmentation is always visible
-                        if (ts.isExternalModuleAugmentation(node)) {
+            function usageInTypeDeclaration() {
+                return !!ts.findAncestor(usage, function (node) { return ts.isInterfaceDeclaration(node) || ts.isTypeAliasDeclaration(node); });
+            }
+            function isImmediatelyUsedInInitializerOfBlockScopedVariable(declaration, usage) {
+                switch (declaration.parent.parent.kind) {
+                    case 232 /* VariableStatement */:
+                    case 237 /* ForStatement */:
+                    case 239 /* ForOfStatement */:
+                        // variable statement/for/for-of statement case,
+                        // use site should not be inside variable declaration (initializer of declaration or binding element)
+                        if (isSameScopeDescendentOf(usage, declaration, declContainer)) {
                             return true;
                         }
-                        var parent = getDeclarationContainer(node);
-                        // If the node is not exported or it is not ambient module element (except import declaration)
-                        if (!(ts.getCombinedModifierFlags(node) & 1 /* Export */) &&
-                            !(node.kind !== 253 /* ImportEqualsDeclaration */ && parent.kind !== 290 /* SourceFile */ && parent.flags & 8388608 /* Ambient */)) {
-                            return isGlobalSourceFile(parent);
-                        }
-                        // Exported members/ambient module elements (exception import declaration) are visible if parent is visible
-                        return isDeclarationVisible(parent);
-                    case 159 /* PropertyDeclaration */:
-                    case 158 /* PropertySignature */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                    case 161 /* MethodDeclaration */:
-                    case 160 /* MethodSignature */:
-                        if (ts.hasModifier(node, 8 /* Private */ | 16 /* Protected */)) {
-                            // Private/protected properties/methods are not visible
-                            return false;
-                        }
-                    // Public properties/methods are visible if its parents are visible, so:
-                    // falls through
-                    case 162 /* Constructor */:
-                    case 166 /* ConstructSignature */:
-                    case 165 /* CallSignature */:
-                    case 167 /* IndexSignature */:
-                    case 156 /* Parameter */:
-                    case 250 /* ModuleBlock */:
-                    case 170 /* FunctionType */:
-                    case 171 /* ConstructorType */:
-                    case 173 /* TypeLiteral */:
-                    case 169 /* TypeReference */:
-                    case 174 /* ArrayType */:
-                    case 175 /* TupleType */:
-                    case 178 /* UnionType */:
-                    case 179 /* IntersectionType */:
-                    case 182 /* ParenthesizedType */:
-                        return isDeclarationVisible(node.parent);
-                    // Default binding, import specifier and namespace import is visible
-                    // only on demand so by default it is not visible
-                    case 255 /* ImportClause */:
-                    case 256 /* NamespaceImport */:
-                    case 258 /* ImportSpecifier */:
-                        return false;
-                    // Type parameters are always visible
-                    case 155 /* TypeParameter */:
-                    // Source file and namespace export are always visible
-                    // falls through
-                    case 290 /* SourceFile */:
-                    case 252 /* NamespaceExportDeclaration */:
-                        return true;
-                    // Export assignments do not create name bindings outside the module
-                    case 259 /* ExportAssignment */:
-                        return false;
-                    default:
-                        return false;
+                        break;
                 }
+                // ForIn/ForOf case - use site should not be used in expression part
+                var grandparent = declaration.parent.parent;
+                return ts.isForInOrOfStatement(grandparent) && isSameScopeDescendentOf(usage, grandparent.expression, declContainer);
             }
-        }
-        function collectLinkedAliases(node, setVisibility) {
-            var exportSymbol;
-            if (node.parent && node.parent.kind === 259 /* ExportAssignment */) {
-                exportSymbol = resolveName(node, node.escapedText, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */, /*nameNotFoundMessage*/ undefined, node, /*isUse*/ false);
-            }
-            else if (node.parent.kind === 263 /* ExportSpecifier */) {
-                exportSymbol = getTargetOfExportSpecifier(node.parent, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */);
-            }
-            var result;
-            var visited;
-            if (exportSymbol) {
-                visited = ts.createMap();
-                visited.set("" + getSymbolId(exportSymbol), true);
-                buildVisibleNodeList(exportSymbol.declarations);
-            }
-            return result;
-            function buildVisibleNodeList(declarations) {
-                ts.forEach(declarations, function (declaration) {
-                    var resultNode = getAnyImportSyntax(declaration) || declaration;
-                    if (setVisibility) {
-                        getNodeLinks(declaration).isVisible = true;
+            function isUsedInFunctionOrInstanceProperty(usage, declaration) {
+                return !!ts.findAncestor(usage, function (current) {
+                    if (current === declContainer) {
+                        return "quit";
                     }
-                    else {
-                        result = result || [];
-                        ts.pushIfUnique(result, resultNode);
+                    if (ts.isFunctionLike(current)) {
+                        return true;
                     }
-                    if (ts.isInternalModuleImportEqualsDeclaration(declaration)) {
-                        // Add the referenced top container visible
-                        var internalModuleReference = declaration.moduleReference;
-                        var firstIdentifier = ts.getFirstIdentifier(internalModuleReference);
-                        var importSymbol = resolveName(declaration, firstIdentifier.escapedText, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, undefined, undefined, /*isUse*/ false);
-                        var id = importSymbol && "" + getSymbolId(importSymbol);
-                        if (importSymbol && !visited.has(id)) {
-                            visited.set(id, true);
-                            buildVisibleNodeList(importSymbol.declarations);
+                    var initializerOfProperty = current.parent &&
+                        current.parent.kind === 163 /* PropertyDeclaration */ &&
+                        current.parent.initializer === current;
+                    if (initializerOfProperty) {
+                        if (ts.hasSyntacticModifier(current.parent, 32 /* Static */)) {
+                            if (declaration.kind === 165 /* MethodDeclaration */) {
+                                return true;
+                            }
+                        }
+                        else {
+                            var isDeclarationInstanceProperty = declaration.kind === 163 /* PropertyDeclaration */ && !ts.hasSyntacticModifier(declaration, 32 /* Static */);
+                            if (!isDeclarationInstanceProperty || ts.getContainingClass(usage) !== ts.getContainingClass(declaration)) {
+                                return true;
+                            }
                         }
                     }
+                    return false;
                 });
             }
-        }
-        /**
-         * Push an entry on the type resolution stack. If an entry with the given target and the given property name
-         * is already on the stack, and no entries in between already have a type, then a circularity has occurred.
-         * In this case, the result values of the existing entry and all entries pushed after it are changed to false,
-         * and the value false is returned. Otherwise, the new entry is just pushed onto the stack, and true is returned.
-         * In order to see if the same query has already been done before, the target object and the propertyName both
-         * must match the one passed in.
-         *
-         * @param target The symbol, type, or signature whose type is being queried
-         * @param propertyName The property name that should be used to query the target for its type
-         */
-        function pushTypeResolution(target, propertyName) {
-            var resolutionCycleStartIndex = findResolutionCycleStartIndex(target, propertyName);
-            if (resolutionCycleStartIndex >= 0) {
-                // A cycle was found
-                var length_3 = resolutionTargets.length;
-                for (var i = resolutionCycleStartIndex; i < length_3; i++) {
-                    resolutionResults[i] = false;
+            /** stopAtAnyPropertyDeclaration is used for detecting ES-standard class field use-before-def errors */
+            function isPropertyImmediatelyReferencedWithinDeclaration(declaration, usage, stopAtAnyPropertyDeclaration) {
+                // always legal if usage is after declaration
+                if (usage.end > declaration.end) {
+                    return false;
                 }
-                return false;
+                // still might be legal if usage is deferred (e.g. x: any = () => this.x)
+                // otherwise illegal if immediately referenced within the declaration (e.g. x: any = this.x)
+                var ancestorChangingReferenceScope = ts.findAncestor(usage, function (node) {
+                    if (node === declaration) {
+                        return "quit";
+                    }
+                    switch (node.kind) {
+                        case 209 /* ArrowFunction */:
+                            return true;
+                        case 163 /* PropertyDeclaration */:
+                            // even when stopping at any property declaration, they need to come from the same class
+                            return stopAtAnyPropertyDeclaration &&
+                                (ts.isPropertyDeclaration(declaration) && node.parent === declaration.parent
+                                    || ts.isParameterPropertyDeclaration(declaration, declaration.parent) && node.parent === declaration.parent.parent)
+                                ? "quit" : true;
+                        case 230 /* Block */:
+                            switch (node.parent.kind) {
+                                case 167 /* GetAccessor */:
+                                case 165 /* MethodDeclaration */:
+                                case 168 /* SetAccessor */:
+                                    return true;
+                                default:
+                                    return false;
+                            }
+                        default:
+                            return false;
+                    }
+                });
+                return ancestorChangingReferenceScope === undefined;
             }
-            resolutionTargets.push(target);
-            resolutionResults.push(/*items*/ true);
-            resolutionPropertyNames.push(propertyName);
-            return true;
         }
-        function findResolutionCycleStartIndex(target, propertyName) {
-            for (var i = resolutionTargets.length - 1; i >= 0; i--) {
-                if (hasType(resolutionTargets[i], resolutionPropertyNames[i])) {
-                    return -1;
-                }
-                if (resolutionTargets[i] === target && resolutionPropertyNames[i] === propertyName) {
-                    return i;
+        function useOuterVariableScopeInParameter(result, location, lastLocation) {
+            var target = ts.getEmitScriptTarget(compilerOptions);
+            var functionLocation = location;
+            if (ts.isParameter(lastLocation) && functionLocation.body && result.valueDeclaration.pos >= functionLocation.body.pos && result.valueDeclaration.end <= functionLocation.body.end) {
+                // check for several cases where we introduce temporaries that require moving the name/initializer of the parameter to the body
+                // - static field in a class expression
+                // - optional chaining pre-es2020
+                // - nullish coalesce pre-es2020
+                // - spread assignment in binding pattern pre-es2017
+                if (target >= 2 /* ES2015 */) {
+                    var links = getNodeLinks(functionLocation);
+                    if (links.declarationRequiresScopeChange === undefined) {
+                        links.declarationRequiresScopeChange = ts.forEach(functionLocation.parameters, requiresScopeChange) || false;
+                    }
+                    return !links.declarationRequiresScopeChange;
                 }
             }
-            return -1;
-        }
-        function hasType(target, propertyName) {
-            switch (propertyName) {
-                case 0 /* Type */:
-                    return !!getSymbolLinks(target).type;
-                case 5 /* EnumTagType */:
-                    return !!(getNodeLinks(target).resolvedEnumType);
-                case 2 /* DeclaredType */:
-                    return !!getSymbolLinks(target).declaredType;
-                case 1 /* ResolvedBaseConstructorType */:
-                    return !!target.resolvedBaseConstructorType;
-                case 3 /* ResolvedReturnType */:
-                    return !!target.resolvedReturnType;
-                case 4 /* ImmediateBaseConstraint */:
-                    return !!target.immediateBaseConstraint;
-                case 6 /* ResolvedTypeArguments */:
-                    return !!target.resolvedTypeArguments;
+            return false;
+            function requiresScopeChange(node) {
+                return requiresScopeChangeWorker(node.name)
+                    || !!node.initializer && requiresScopeChangeWorker(node.initializer);
             }
-            return ts.Debug.assertNever(propertyName);
-        }
-        /**
-         * Pop an entry from the type resolution stack and return its associated result value. The result value will
-         * be true if no circularities were detected, or false if a circularity was found.
-         */
-        function popTypeResolution() {
-            resolutionTargets.pop();
-            resolutionPropertyNames.pop();
-            return resolutionResults.pop();
-        }
-        function getDeclarationContainer(node) {
-            return ts.findAncestor(ts.getRootDeclaration(node), function (node) {
+            function requiresScopeChangeWorker(node) {
                 switch (node.kind) {
-                    case 242 /* VariableDeclaration */:
-                    case 243 /* VariableDeclarationList */:
-                    case 258 /* ImportSpecifier */:
-                    case 257 /* NamedImports */:
-                    case 256 /* NamespaceImport */:
-                    case 255 /* ImportClause */:
+                    case 209 /* ArrowFunction */:
+                    case 208 /* FunctionExpression */:
+                    case 251 /* FunctionDeclaration */:
+                    case 166 /* Constructor */:
+                        // do not descend into these
                         return false;
+                    case 165 /* MethodDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 288 /* PropertyAssignment */:
+                        return requiresScopeChangeWorker(node.name);
+                    case 163 /* PropertyDeclaration */:
+                        // static properties in classes introduce temporary variables
+                        if (ts.hasStaticModifier(node)) {
+                            return target < 99 /* ESNext */ || !compilerOptions.useDefineForClassFields;
+                        }
+                        return requiresScopeChangeWorker(node.name);
                     default:
-                        return true;
+                        // null coalesce and optional chain pre-es2020 produce temporary variables
+                        if (ts.isNullishCoalesce(node) || ts.isOptionalChain(node)) {
+                            return target < 7 /* ES2020 */;
+                        }
+                        if (ts.isBindingElement(node) && node.dotDotDotToken && ts.isObjectBindingPattern(node.parent)) {
+                            return target < 4 /* ES2017 */;
+                        }
+                        if (ts.isTypeNode(node))
+                            return false;
+                        return ts.forEachChild(node, requiresScopeChangeWorker) || false;
                 }
-            }).parent;
-        }
-        function getTypeOfPrototypeProperty(prototype) {
-            // TypeScript 1.0 spec (April 2014): 8.4
-            // Every class automatically contains a static property member named 'prototype',
-            // the type of which is an instantiation of the class type with type Any supplied as a type argument for each type parameter.
-            // It is an error to explicitly declare a static property member with the name 'prototype'.
-            var classType = getDeclaredTypeOfSymbol(getParentOfSymbol(prototype));
-            return classType.typeParameters ? createTypeReference(classType, ts.map(classType.typeParameters, function (_) { return anyType; })) : classType;
-        }
-        // Return the type of the given property in the given type, or undefined if no such property exists
-        function getTypeOfPropertyOfType(type, name) {
-            var prop = getPropertyOfType(type, name);
-            return prop ? getTypeOfSymbol(prop) : undefined;
-        }
-        function getTypeOfPropertyOrIndexSignature(type, name) {
-            return getTypeOfPropertyOfType(type, name) || isNumericLiteralName(name) && getIndexTypeOfType(type, 1 /* Number */) || getIndexTypeOfType(type, 0 /* String */) || unknownType;
-        }
-        function isTypeAny(type) {
-            return type && (type.flags & 1 /* Any */) !== 0;
+            }
         }
-        // Return the type of a binding element parent. We check SymbolLinks first to see if a type has been
-        // assigned by contextual typing.
-        function getTypeForBindingElementParent(node) {
-            var symbol = getSymbolOfNode(node);
-            return symbol && getSymbolLinks(symbol).type || getTypeForVariableLikeDeclaration(node, /*includeOptionality*/ false);
+        /**
+         * Resolve a given name for a given meaning at a given location. An error is reported if the name was not found and
+         * the nameNotFoundMessage argument is not undefined. Returns the resolved symbol, or undefined if no symbol with
+         * the given name can be found.
+         *
+         * @param isUse If true, this will count towards --noUnusedLocals / --noUnusedParameters.
+         */
+        function resolveName(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, suggestedNameNotFoundMessage) {
+            if (excludeGlobals === void 0) { excludeGlobals = false; }
+            return resolveNameHelper(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, getSymbol, suggestedNameNotFoundMessage);
         }
-        function getRestType(source, properties, symbol) {
-            source = filterType(source, function (t) { return !(t.flags & 98304 /* Nullable */); });
-            if (source.flags & 131072 /* Never */) {
-                return emptyObjectType;
+        function resolveNameHelper(location, name, meaning, nameNotFoundMessage, nameArg, isUse, excludeGlobals, lookup, suggestedNameNotFoundMessage) {
+            var originalLocation = location; // needed for did-you-mean error reporting, which gathers candidates starting from the original location
+            var result;
+            var lastLocation;
+            var lastSelfReferenceLocation;
+            var propertyWithInvalidInitializer;
+            var associatedDeclarationForContainingInitializerOrBindingName;
+            var withinDeferredContext = false;
+            var errorLocation = location;
+            var grandparent;
+            var isInExternalModule = false;
+            loop: while (location) {
+                // Locals of a source file are not in scope (because they get merged into the global symbol table)
+                if (location.locals && !isGlobalSourceFile(location)) {
+                    if (result = lookup(location.locals, name, meaning)) {
+                        var useResult = true;
+                        if (ts.isFunctionLike(location) && lastLocation && lastLocation !== location.body) {
+                            // symbol lookup restrictions for function-like declarations
+                            // - Type parameters of a function are in scope in the entire function declaration, including the parameter
+                            //   list and return type. However, local types are only in scope in the function body.
+                            // - parameters are only in the scope of function body
+                            // This restriction does not apply to JSDoc comment types because they are parented
+                            // at a higher level than type parameters would normally be
+                            if (meaning & result.flags & 788968 /* Type */ && lastLocation.kind !== 311 /* JSDocComment */) {
+                                useResult = result.flags & 262144 /* TypeParameter */
+                                    // type parameters are visible in parameter list, return type and type parameter list
+                                    ? lastLocation === location.type ||
+                                        lastLocation.kind === 160 /* Parameter */ ||
+                                        lastLocation.kind === 159 /* TypeParameter */
+                                    // local types not visible outside the function body
+                                    : false;
+                            }
+                            if (meaning & result.flags & 3 /* Variable */) {
+                                // expression inside parameter will lookup as normal variable scope when targeting es2015+
+                                if (useOuterVariableScopeInParameter(result, location, lastLocation)) {
+                                    useResult = false;
+                                }
+                                else if (result.flags & 1 /* FunctionScopedVariable */) {
+                                    // parameters are visible only inside function body, parameter list and return type
+                                    // technically for parameter list case here we might mix parameters and variables declared in function,
+                                    // however it is detected separately when checking initializers of parameters
+                                    // to make sure that they reference no variables declared after them.
+                                    useResult =
+                                        lastLocation.kind === 160 /* Parameter */ ||
+                                            (lastLocation === location.type &&
+                                                !!ts.findAncestor(result.valueDeclaration, ts.isParameter));
+                                }
+                            }
+                        }
+                        else if (location.kind === 184 /* ConditionalType */) {
+                            // A type parameter declared using 'infer T' in a conditional type is visible only in
+                            // the true branch of the conditional type.
+                            useResult = lastLocation === location.trueType;
+                        }
+                        if (useResult) {
+                            break loop;
+                        }
+                        else {
+                            result = undefined;
+                        }
+                    }
+                }
+                withinDeferredContext = withinDeferredContext || getIsDeferredContext(location, lastLocation);
+                switch (location.kind) {
+                    case 297 /* SourceFile */:
+                        if (!ts.isExternalOrCommonJsModule(location))
+                            break;
+                        isInExternalModule = true;
+                    // falls through
+                    case 256 /* ModuleDeclaration */:
+                        var moduleExports = getSymbolOfNode(location).exports || emptySymbols;
+                        if (location.kind === 297 /* SourceFile */ || (ts.isModuleDeclaration(location) && location.flags & 8388608 /* Ambient */ && !ts.isGlobalScopeAugmentation(location))) {
+                            // It's an external module. First see if the module has an export default and if the local
+                            // name of that export default matches.
+                            if (result = moduleExports.get("default" /* Default */)) {
+                                var localSymbol = ts.getLocalSymbolForExportDefault(result);
+                                if (localSymbol && (result.flags & meaning) && localSymbol.escapedName === name) {
+                                    break loop;
+                                }
+                                result = undefined;
+                            }
+                            // Because of module/namespace merging, a module's exports are in scope,
+                            // yet we never want to treat an export specifier as putting a member in scope.
+                            // Therefore, if the name we find is purely an export specifier, it is not actually considered in scope.
+                            // Two things to note about this:
+                            //     1. We have to check this without calling getSymbol. The problem with calling getSymbol
+                            //        on an export specifier is that it might find the export specifier itself, and try to
+                            //        resolve it as an alias. This will cause the checker to consider the export specifier
+                            //        a circular alias reference when it might not be.
+                            //     2. We check === SymbolFlags.Alias in order to check that the symbol is *purely*
+                            //        an alias. If we used &, we'd be throwing out symbols that have non alias aspects,
+                            //        which is not the desired behavior.
+                            var moduleExport = moduleExports.get(name);
+                            if (moduleExport &&
+                                moduleExport.flags === 2097152 /* Alias */ &&
+                                (ts.getDeclarationOfKind(moduleExport, 270 /* ExportSpecifier */) || ts.getDeclarationOfKind(moduleExport, 269 /* NamespaceExport */))) {
+                                break;
+                            }
+                        }
+                        // ES6 exports are also visible locally (except for 'default'), but commonjs exports are not (except typedefs)
+                        if (name !== "default" /* Default */ && (result = lookup(moduleExports, name, meaning & 2623475 /* ModuleMember */))) {
+                            if (ts.isSourceFile(location) && location.commonJsModuleIndicator && !result.declarations.some(ts.isJSDocTypeAlias)) {
+                                result = undefined;
+                            }
+                            else {
+                                break loop;
+                            }
+                        }
+                        break;
+                    case 255 /* EnumDeclaration */:
+                        if (result = lookup(getSymbolOfNode(location).exports, name, meaning & 8 /* EnumMember */)) {
+                            break loop;
+                        }
+                        break;
+                    case 163 /* PropertyDeclaration */:
+                        // TypeScript 1.0 spec (April 2014): 8.4.1
+                        // Initializer expressions for instance member variables are evaluated in the scope
+                        // of the class constructor body but are not permitted to reference parameters or
+                        // local variables of the constructor. This effectively means that entities from outer scopes
+                        // by the same name as a constructor parameter or local variable are inaccessible
+                        // in initializer expressions for instance member variables.
+                        if (!ts.hasSyntacticModifier(location, 32 /* Static */)) {
+                            var ctor = findConstructorDeclaration(location.parent);
+                            if (ctor && ctor.locals) {
+                                if (lookup(ctor.locals, name, meaning & 111551 /* Value */)) {
+                                    // Remember the property node, it will be used later to report appropriate error
+                                    propertyWithInvalidInitializer = location;
+                                }
+                            }
+                        }
+                        break;
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 253 /* InterfaceDeclaration */:
+                        // The below is used to lookup type parameters within a class or interface, as they are added to the class/interface locals
+                        // These can never be latebound, so the symbol's raw members are sufficient. `getMembersOfNode` cannot be used, as it would
+                        // trigger resolving late-bound names, which we may already be in the process of doing while we're here!
+                        if (result = lookup(getSymbolOfNode(location).members || emptySymbols, name, meaning & 788968 /* Type */)) {
+                            if (!isTypeParameterSymbolDeclaredInContainer(result, location)) {
+                                // ignore type parameters not declared in this container
+                                result = undefined;
+                                break;
+                            }
+                            if (lastLocation && ts.hasSyntacticModifier(lastLocation, 32 /* Static */)) {
+                                // TypeScript 1.0 spec (April 2014): 3.4.1
+                                // The scope of a type parameter extends over the entire declaration with which the type
+                                // parameter list is associated, with the exception of static member declarations in classes.
+                                error(errorLocation, ts.Diagnostics.Static_members_cannot_reference_class_type_parameters);
+                                return undefined;
+                            }
+                            break loop;
+                        }
+                        if (location.kind === 221 /* ClassExpression */ && meaning & 32 /* Class */) {
+                            var className = location.name;
+                            if (className && name === className.escapedText) {
+                                result = location.symbol;
+                                break loop;
+                            }
+                        }
+                        break;
+                    case 223 /* ExpressionWithTypeArguments */:
+                        // The type parameters of a class are not in scope in the base class expression.
+                        if (lastLocation === location.expression && location.parent.token === 93 /* ExtendsKeyword */) {
+                            var container = location.parent.parent;
+                            if (ts.isClassLike(container) && (result = lookup(getSymbolOfNode(container).members, name, meaning & 788968 /* Type */))) {
+                                if (nameNotFoundMessage) {
+                                    error(errorLocation, ts.Diagnostics.Base_class_expressions_cannot_reference_class_type_parameters);
+                                }
+                                return undefined;
+                            }
+                        }
+                        break;
+                    // It is not legal to reference a class's own type parameters from a computed property name that
+                    // belongs to the class. For example:
+                    //
+                    //   function foo<T>() { return '' }
+                    //   class C<T> { // <-- Class's own type parameter T
+                    //       [foo<T>()]() { } // <-- Reference to T from class's own computed property
+                    //   }
+                    //
+                    case 158 /* ComputedPropertyName */:
+                        grandparent = location.parent.parent;
+                        if (ts.isClassLike(grandparent) || grandparent.kind === 253 /* InterfaceDeclaration */) {
+                            // A reference to this grandparent's type parameters would be an error
+                            if (result = lookup(getSymbolOfNode(grandparent).members, name, meaning & 788968 /* Type */)) {
+                                error(errorLocation, ts.Diagnostics.A_computed_property_name_cannot_reference_a_type_parameter_from_its_containing_type);
+                                return undefined;
+                            }
+                        }
+                        break;
+                    case 209 /* ArrowFunction */:
+                        // when targeting ES6 or higher there is no 'arguments' in an arrow function
+                        // for lower compile targets the resolved symbol is used to emit an error
+                        if (compilerOptions.target >= 2 /* ES2015 */) {
+                            break;
+                        }
+                    // falls through
+                    case 165 /* MethodDeclaration */:
+                    case 166 /* Constructor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 251 /* FunctionDeclaration */:
+                        if (meaning & 3 /* Variable */ && name === "arguments") {
+                            result = argumentsSymbol;
+                            break loop;
+                        }
+                        break;
+                    case 208 /* FunctionExpression */:
+                        if (meaning & 3 /* Variable */ && name === "arguments") {
+                            result = argumentsSymbol;
+                            break loop;
+                        }
+                        if (meaning & 16 /* Function */) {
+                            var functionName = location.name;
+                            if (functionName && name === functionName.escapedText) {
+                                result = location.symbol;
+                                break loop;
+                            }
+                        }
+                        break;
+                    case 161 /* Decorator */:
+                        // Decorators are resolved at the class declaration. Resolving at the parameter
+                        // or member would result in looking up locals in the method.
+                        //
+                        //   function y() {}
+                        //   class C {
+                        //       method(@y x, y) {} // <-- decorator y should be resolved at the class declaration, not the parameter.
+                        //   }
+                        //
+                        if (location.parent && location.parent.kind === 160 /* Parameter */) {
+                            location = location.parent;
+                        }
+                        //
+                        //   function y() {}
+                        //   class C {
+                        //       @y method(x, y) {} // <-- decorator y should be resolved at the class declaration, not the method.
+                        //   }
+                        //
+                        // class Decorators are resolved outside of the class to avoid referencing type parameters of that class.
+                        //
+                        //   type T = number;
+                        //   declare function y(x: T): any;
+                        //   @param(1 as T) // <-- T should resolve to the type alias outside of class C
+                        //   class C<T> {}
+                        if (location.parent && (ts.isClassElement(location.parent) || location.parent.kind === 252 /* ClassDeclaration */)) {
+                            location = location.parent;
+                        }
+                        break;
+                    case 331 /* JSDocTypedefTag */:
+                    case 324 /* JSDocCallbackTag */:
+                    case 325 /* JSDocEnumTag */:
+                        // js type aliases do not resolve names from their host, so skip past it
+                        location = ts.getJSDocHost(location);
+                        break;
+                    case 160 /* Parameter */:
+                        if (lastLocation && (lastLocation === location.initializer ||
+                            lastLocation === location.name && ts.isBindingPattern(lastLocation))) {
+                            if (!associatedDeclarationForContainingInitializerOrBindingName) {
+                                associatedDeclarationForContainingInitializerOrBindingName = location;
+                            }
+                        }
+                        break;
+                    case 198 /* BindingElement */:
+                        if (lastLocation && (lastLocation === location.initializer ||
+                            lastLocation === location.name && ts.isBindingPattern(lastLocation))) {
+                            if (ts.isParameterDeclaration(location) && !associatedDeclarationForContainingInitializerOrBindingName) {
+                                associatedDeclarationForContainingInitializerOrBindingName = location;
+                            }
+                        }
+                        break;
+                    case 185 /* InferType */:
+                        if (meaning & 262144 /* TypeParameter */) {
+                            var parameterName = location.typeParameter.name;
+                            if (parameterName && name === parameterName.escapedText) {
+                                result = location.typeParameter.symbol;
+                                break loop;
+                            }
+                        }
+                        break;
+                }
+                if (isSelfReferenceLocation(location)) {
+                    lastSelfReferenceLocation = location;
+                }
+                lastLocation = location;
+                location = location.parent;
             }
-            if (source.flags & 1048576 /* Union */) {
-                return mapType(source, function (t) { return getRestType(t, properties, symbol); });
+            // We just climbed up parents looking for the name, meaning that we started in a descendant node of `lastLocation`.
+            // If `result === lastSelfReferenceLocation.symbol`, that means that we are somewhere inside `lastSelfReferenceLocation` looking up a name, and resolving to `lastLocation` itself.
+            // That means that this is a self-reference of `lastLocation`, and shouldn't count this when considering whether `lastLocation` is used.
+            if (isUse && result && (!lastSelfReferenceLocation || result !== lastSelfReferenceLocation.symbol)) {
+                result.isReferenced |= meaning;
+            }
+            if (!result) {
+                if (lastLocation) {
+                    ts.Debug.assert(lastLocation.kind === 297 /* SourceFile */);
+                    if (lastLocation.commonJsModuleIndicator && name === "exports" && meaning & lastLocation.symbol.flags) {
+                        return lastLocation.symbol;
+                    }
+                }
+                if (!excludeGlobals) {
+                    result = lookup(globals, name, meaning);
+                }
             }
-            var omitKeyType = getUnionType(ts.map(properties, getLiteralTypeFromPropertyName));
-            if (isGenericObjectType(source) || isGenericIndexType(omitKeyType)) {
-                if (omitKeyType.flags & 131072 /* Never */) {
-                    return source;
+            if (!result) {
+                if (originalLocation && ts.isInJSFile(originalLocation) && originalLocation.parent) {
+                    if (ts.isRequireCall(originalLocation.parent, /*checkArgumentIsStringLiteralLike*/ false)) {
+                        return requireSymbol;
+                    }
                 }
-                var omitTypeAlias = getGlobalOmitSymbol();
-                if (!omitTypeAlias) {
-                    return errorType;
+            }
+            if (!result) {
+                if (nameNotFoundMessage) {
+                    if (!errorLocation ||
+                        !checkAndReportErrorForMissingPrefix(errorLocation, name, nameArg) && // TODO: GH#18217
+                            !checkAndReportErrorForExtendingInterface(errorLocation) &&
+                            !checkAndReportErrorForUsingTypeAsNamespace(errorLocation, name, meaning) &&
+                            !checkAndReportErrorForExportingPrimitiveType(errorLocation, name) &&
+                            !checkAndReportErrorForUsingTypeAsValue(errorLocation, name, meaning) &&
+                            !checkAndReportErrorForUsingNamespaceModuleAsValue(errorLocation, name, meaning) &&
+                            !checkAndReportErrorForUsingValueAsType(errorLocation, name, meaning)) {
+                        var suggestion = void 0;
+                        if (suggestedNameNotFoundMessage && suggestionCount < maximumSuggestionCount) {
+                            suggestion = getSuggestedSymbolForNonexistentSymbol(originalLocation, name, meaning);
+                            if (suggestion) {
+                                var suggestionName = symbolToString(suggestion);
+                                var diagnostic = error(errorLocation, suggestedNameNotFoundMessage, diagnosticName(nameArg), suggestionName);
+                                if (suggestion.valueDeclaration) {
+                                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(suggestion.valueDeclaration, ts.Diagnostics._0_is_declared_here, suggestionName));
+                                }
+                            }
+                        }
+                        if (!suggestion) {
+                            if (nameArg) {
+                                var lib = getSuggestedLibForNonExistentName(nameArg);
+                                if (lib) {
+                                    error(errorLocation, nameNotFoundMessage, diagnosticName(nameArg), lib);
+                                }
+                                else {
+                                    error(errorLocation, nameNotFoundMessage, diagnosticName(nameArg));
+                                }
+                            }
+                        }
+                        suggestionCount++;
+                    }
                 }
-                return getTypeAliasInstantiation(omitTypeAlias, [source, omitKeyType]);
+                return undefined;
             }
-            var members = ts.createSymbolTable();
-            for (var _i = 0, _a = getPropertiesOfType(source); _i < _a.length; _i++) {
-                var prop = _a[_i];
-                if (!isTypeAssignableTo(getLiteralTypeFromProperty(prop, 8576 /* StringOrNumberLiteralOrUnique */), omitKeyType)
-                    && !(ts.getDeclarationModifierFlagsFromSymbol(prop) & (8 /* Private */ | 16 /* Protected */))
-                    && isSpreadableProperty(prop)) {
-                    members.set(prop.escapedName, getSpreadSymbol(prop, /*readonly*/ false));
+            // Perform extra checks only if error reporting was requested
+            if (nameNotFoundMessage) {
+                if (propertyWithInvalidInitializer && !(compilerOptions.target === 99 /* ESNext */ && compilerOptions.useDefineForClassFields)) {
+                    // We have a match, but the reference occurred within a property initializer and the identifier also binds
+                    // to a local variable in the constructor where the code will be emitted. Note that this is actually allowed
+                    // with ESNext+useDefineForClassFields because the scope semantics are different.
+                    var propertyName = propertyWithInvalidInitializer.name;
+                    error(errorLocation, ts.Diagnostics.Initializer_of_instance_member_variable_0_cannot_reference_identifier_1_declared_in_the_constructor, ts.declarationNameToString(propertyName), diagnosticName(nameArg));
+                    return undefined;
+                }
+                // Only check for block-scoped variable if we have an error location and are looking for the
+                // name with variable meaning
+                //      For example,
+                //          declare module foo {
+                //              interface bar {}
+                //          }
+                //      const foo/*1*/: foo/*2*/.bar;
+                // The foo at /*1*/ and /*2*/ will share same symbol with two meanings:
+                // block-scoped variable and namespace module. However, only when we
+                // try to resolve name in /*1*/ which is used in variable position,
+                // we want to check for block-scoped
+                if (errorLocation &&
+                    (meaning & 2 /* BlockScopedVariable */ ||
+                        ((meaning & 32 /* Class */ || meaning & 384 /* Enum */) && (meaning & 111551 /* Value */) === 111551 /* Value */))) {
+                    var exportOrLocalSymbol = getExportSymbolOfValueSymbolIfExported(result);
+                    if (exportOrLocalSymbol.flags & 2 /* BlockScopedVariable */ || exportOrLocalSymbol.flags & 32 /* Class */ || exportOrLocalSymbol.flags & 384 /* Enum */) {
+                        checkResolvedBlockScopedVariable(exportOrLocalSymbol, errorLocation);
+                    }
+                }
+                // If we're in an external module, we can't reference value symbols created from UMD export declarations
+                if (result && isInExternalModule && (meaning & 111551 /* Value */) === 111551 /* Value */ && !(originalLocation.flags & 4194304 /* JSDoc */)) {
+                    var merged = getMergedSymbol(result);
+                    if (ts.length(merged.declarations) && ts.every(merged.declarations, function (d) { return ts.isNamespaceExportDeclaration(d) || ts.isSourceFile(d) && !!d.symbol.globalExports; })) {
+                        errorOrSuggestion(!compilerOptions.allowUmdGlobalAccess, errorLocation, ts.Diagnostics._0_refers_to_a_UMD_global_but_the_current_file_is_a_module_Consider_adding_an_import_instead, ts.unescapeLeadingUnderscores(name));
+                    }
+                }
+                // If we're in a parameter initializer or binding name, we can't reference the values of the parameter whose initializer we're within or parameters to the right
+                if (result && associatedDeclarationForContainingInitializerOrBindingName && !withinDeferredContext && (meaning & 111551 /* Value */) === 111551 /* Value */) {
+                    var candidate = getMergedSymbol(getLateBoundSymbol(result));
+                    var root = ts.getRootDeclaration(associatedDeclarationForContainingInitializerOrBindingName);
+                    // A parameter initializer or binding pattern initializer within a parameter cannot refer to itself
+                    if (candidate === getSymbolOfNode(associatedDeclarationForContainingInitializerOrBindingName)) {
+                        error(errorLocation, ts.Diagnostics.Parameter_0_cannot_reference_itself, ts.declarationNameToString(associatedDeclarationForContainingInitializerOrBindingName.name));
+                    }
+                    // And it cannot refer to any declarations which come after it
+                    else if (candidate.valueDeclaration && candidate.valueDeclaration.pos > associatedDeclarationForContainingInitializerOrBindingName.pos && root.parent.locals && lookup(root.parent.locals, candidate.escapedName, meaning) === candidate) {
+                        error(errorLocation, ts.Diagnostics.Parameter_0_cannot_reference_identifier_1_declared_after_it, ts.declarationNameToString(associatedDeclarationForContainingInitializerOrBindingName.name), ts.declarationNameToString(errorLocation));
+                    }
+                }
+                if (result && errorLocation && meaning & 111551 /* Value */ && result.flags & 2097152 /* Alias */) {
+                    checkSymbolUsageInExpressionContext(result, name, errorLocation);
                 }
             }
-            var stringIndexInfo = getIndexInfoOfType(source, 0 /* String */);
-            var numberIndexInfo = getIndexInfoOfType(source, 1 /* Number */);
-            var result = createAnonymousType(symbol, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
-            result.objectFlags |= 131072 /* ObjectRestType */;
             return result;
         }
-        // Determine the control flow type associated with a destructuring declaration or assignment. The following
-        // forms of destructuring are possible:
-        //   let { x } = obj;  // BindingElement
-        //   let [ x ] = obj;  // BindingElement
-        //   { x } = obj;      // ShorthandPropertyAssignment
-        //   { x: v } = obj;   // PropertyAssignment
-        //   [ x ] = obj;      // Expression
-        // We construct a synthetic element access expression corresponding to 'obj.x' such that the control
-        // flow analyzer doesn't have to handle all the different syntactic forms.
-        function getFlowTypeOfDestructuring(node, declaredType) {
-            var reference = getSyntheticElementAccess(node);
-            return reference ? getFlowTypeOfReference(reference, declaredType) : declaredType;
-        }
-        function getSyntheticElementAccess(node) {
-            var parentAccess = getParentElementAccess(node);
-            if (parentAccess && parentAccess.flowNode) {
-                var propName = getDestructuringPropertyName(node);
-                if (propName) {
-                    var result = ts.createNode(195 /* ElementAccessExpression */, node.pos, node.end);
-                    result.parent = node;
-                    result.expression = parentAccess;
-                    var literal = ts.createNode(10 /* StringLiteral */, node.pos, node.end);
-                    literal.parent = result;
-                    literal.text = propName;
-                    result.argumentExpression = literal;
-                    result.flowNode = parentAccess.flowNode;
-                    return result;
+        function checkSymbolUsageInExpressionContext(symbol, name, useSite) {
+            if (!ts.isValidTypeOnlyAliasUseSite(useSite)) {
+                var typeOnlyDeclaration = getTypeOnlyAliasDeclaration(symbol);
+                if (typeOnlyDeclaration) {
+                    var isExport = ts.typeOnlyDeclarationIsExport(typeOnlyDeclaration);
+                    var message = isExport
+                        ? ts.Diagnostics._0_cannot_be_used_as_a_value_because_it_was_exported_using_export_type
+                        : ts.Diagnostics._0_cannot_be_used_as_a_value_because_it_was_imported_using_import_type;
+                    var relatedMessage = isExport
+                        ? ts.Diagnostics._0_was_exported_here
+                        : ts.Diagnostics._0_was_imported_here;
+                    var unescapedName = ts.unescapeLeadingUnderscores(name);
+                    ts.addRelatedInfo(error(useSite, message, unescapedName), ts.createDiagnosticForNode(typeOnlyDeclaration, relatedMessage, unescapedName));
                 }
             }
         }
-        function getParentElementAccess(node) {
-            var ancestor = node.parent.parent;
-            switch (ancestor.kind) {
-                case 191 /* BindingElement */:
-                case 281 /* PropertyAssignment */:
-                    return getSyntheticElementAccess(ancestor);
-                case 192 /* ArrayLiteralExpression */:
-                    return getSyntheticElementAccess(node.parent);
-                case 242 /* VariableDeclaration */:
-                    return ancestor.initializer;
-                case 209 /* BinaryExpression */:
-                    return ancestor.right;
+        function getIsDeferredContext(location, lastLocation) {
+            if (location.kind !== 209 /* ArrowFunction */ && location.kind !== 208 /* FunctionExpression */) {
+                // initializers in instance property declaration of class like entities are executed in constructor and thus deferred
+                return ts.isTypeQueryNode(location) || ((ts.isFunctionLikeDeclaration(location) ||
+                    (location.kind === 163 /* PropertyDeclaration */ && !ts.hasSyntacticModifier(location, 32 /* Static */))) && (!lastLocation || lastLocation !== location.name)); // A name is evaluated within the enclosing scope - so it shouldn't count as deferred
             }
-        }
-        function getDestructuringPropertyName(node) {
-            var parent = node.parent;
-            if (node.kind === 191 /* BindingElement */ && parent.kind === 189 /* ObjectBindingPattern */) {
-                return getLiteralPropertyNameText(node.propertyName || node.name);
+            if (lastLocation && lastLocation === location.name) {
+                return false;
             }
-            if (node.kind === 281 /* PropertyAssignment */ || node.kind === 282 /* ShorthandPropertyAssignment */) {
-                return getLiteralPropertyNameText(node.name);
+            // generator functions and async functions are not inlined in control flow when immediately invoked
+            if (location.asteriskToken || ts.hasSyntacticModifier(location, 256 /* Async */)) {
+                return true;
             }
-            return "" + parent.elements.indexOf(node);
-        }
-        function getLiteralPropertyNameText(name) {
-            var type = getLiteralTypeFromPropertyName(name);
-            return type.flags & (128 /* StringLiteral */ | 256 /* NumberLiteral */) ? "" + type.value : undefined;
+            return !ts.getImmediatelyInvokedFunctionExpression(location);
         }
-        /** Return the inferred type for a binding element */
-        function getTypeForBindingElement(declaration) {
-            var pattern = declaration.parent;
-            var parentType = getTypeForBindingElementParent(pattern.parent);
-            // If no type or an any type was inferred for parent, infer that for the binding element
-            if (!parentType || isTypeAny(parentType)) {
-                return parentType;
+        function isSelfReferenceLocation(node) {
+            switch (node.kind) {
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 256 /* ModuleDeclaration */: // For `namespace N { N; }`
+                    return true;
+                default:
+                    return false;
             }
-            // Relax null check on ambient destructuring parameters, since the parameters have no implementation and are just documentation
-            if (strictNullChecks && declaration.flags & 8388608 /* Ambient */ && ts.isParameterDeclaration(declaration)) {
-                parentType = getNonNullableType(parentType);
+        }
+        function diagnosticName(nameArg) {
+            return ts.isString(nameArg) ? ts.unescapeLeadingUnderscores(nameArg) : ts.declarationNameToString(nameArg);
+        }
+        function isTypeParameterSymbolDeclaredInContainer(symbol, container) {
+            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                var decl = _a[_i];
+                if (decl.kind === 159 /* TypeParameter */) {
+                    var parent = ts.isJSDocTemplateTag(decl.parent) ? ts.getJSDocHost(decl.parent) : decl.parent;
+                    if (parent === container) {
+                        return !(ts.isJSDocTemplateTag(decl.parent) && ts.find(decl.parent.parent.tags, ts.isJSDocTypeAlias)); // TODO: GH#18217
+                    }
+                }
             }
-            // Filter `undefined` from the type we check against if the parent has an initializer and that initializer is not possibly `undefined`
-            else if (strictNullChecks && pattern.parent.initializer && !(getTypeFacts(getTypeOfInitializer(pattern.parent.initializer)) & 65536 /* EQUndefined */)) {
-                parentType = getTypeWithFacts(parentType, 524288 /* NEUndefined */);
+            return false;
+        }
+        function checkAndReportErrorForMissingPrefix(errorLocation, name, nameArg) {
+            if (!ts.isIdentifier(errorLocation) || errorLocation.escapedText !== name || isTypeReferenceIdentifier(errorLocation) || isInTypeQuery(errorLocation)) {
+                return false;
             }
-            var type;
-            if (pattern.kind === 189 /* ObjectBindingPattern */) {
-                if (declaration.dotDotDotToken) {
-                    parentType = getReducedType(parentType);
-                    if (parentType.flags & 2 /* Unknown */ || !isValidSpreadType(parentType)) {
-                        error(declaration, ts.Diagnostics.Rest_types_may_only_be_created_from_object_types);
-                        return errorType;
+            var container = ts.getThisContainer(errorLocation, /*includeArrowFunctions*/ false);
+            var location = container;
+            while (location) {
+                if (ts.isClassLike(location.parent)) {
+                    var classSymbol = getSymbolOfNode(location.parent);
+                    if (!classSymbol) {
+                        break;
                     }
-                    var literalMembers = [];
-                    for (var _i = 0, _a = pattern.elements; _i < _a.length; _i++) {
-                        var element = _a[_i];
-                        if (!element.dotDotDotToken) {
-                            literalMembers.push(element.propertyName || element.name);
+                    // Check to see if a static member exists.
+                    var constructorType = getTypeOfSymbol(classSymbol);
+                    if (getPropertyOfType(constructorType, name)) {
+                        error(errorLocation, ts.Diagnostics.Cannot_find_name_0_Did_you_mean_the_static_member_1_0, diagnosticName(nameArg), symbolToString(classSymbol));
+                        return true;
+                    }
+                    // No static member is present.
+                    // Check if we're in an instance method and look for a relevant instance member.
+                    if (location === container && !ts.hasSyntacticModifier(location, 32 /* Static */)) {
+                        var instanceType = getDeclaredTypeOfSymbol(classSymbol).thisType; // TODO: GH#18217
+                        if (getPropertyOfType(instanceType, name)) {
+                            error(errorLocation, ts.Diagnostics.Cannot_find_name_0_Did_you_mean_the_instance_member_this_0, diagnosticName(nameArg));
+                            return true;
                         }
                     }
-                    type = getRestType(parentType, literalMembers, declaration.symbol);
-                }
-                else {
-                    // Use explicitly specified property name ({ p: xxx } form), or otherwise the implied name ({ p } form)
-                    var name = declaration.propertyName || declaration.name;
-                    var indexType = getLiteralTypeFromPropertyName(name);
-                    var declaredType = getConstraintForLocation(getIndexedAccessType(parentType, indexType, name), declaration.name);
-                    type = getFlowTypeOfDestructuring(declaration, declaredType);
-                }
-            }
-            else {
-                // This elementType will be used if the specific property corresponding to this index is not
-                // present (aka the tuple element property). This call also checks that the parentType is in
-                // fact an iterable or array (depending on target language).
-                var elementType = checkIteratedTypeOrElementType(65 /* Destructuring */, parentType, undefinedType, pattern);
-                var index_1 = pattern.elements.indexOf(declaration);
-                if (declaration.dotDotDotToken) {
-                    // If the parent is a tuple type, the rest element has a tuple type of the
-                    // remaining tuple element types. Otherwise, the rest element has an array type with same
-                    // element type as the parent type.
-                    type = everyType(parentType, isTupleType) ?
-                        mapType(parentType, function (t) { return sliceTupleType(t, index_1); }) :
-                        createArrayType(elementType);
                 }
-                else if (isArrayLikeType(parentType)) {
-                    var indexType = getLiteralType(index_1);
-                    var accessFlags = hasDefaultValue(declaration) ? 8 /* NoTupleBoundsCheck */ : 0;
-                    var declaredType = getConstraintForLocation(getIndexedAccessTypeOrUndefined(parentType, indexType, declaration.name, accessFlags) || errorType, declaration.name);
-                    type = getFlowTypeOfDestructuring(declaration, declaredType);
-                }
-                else {
-                    type = elementType;
-                }
-            }
-            if (!declaration.initializer) {
-                return type;
-            }
-            if (ts.getEffectiveTypeAnnotationNode(ts.walkUpBindingElementsAndPatterns(declaration))) {
-                // In strict null checking mode, if a default value of a non-undefined type is specified, remove
-                // undefined from the final type.
-                return strictNullChecks && !(getFalsyFlags(checkDeclarationInitializer(declaration)) & 32768 /* Undefined */) ?
-                    getTypeWithFacts(type, 524288 /* NEUndefined */) :
-                    type;
+                location = location.parent;
             }
-            return widenTypeInferredFromInitializer(declaration, getUnionType([getTypeWithFacts(type, 524288 /* NEUndefined */), checkDeclarationInitializer(declaration)], 2 /* Subtype */));
+            return false;
         }
-        function getTypeForDeclarationFromJSDocComment(declaration) {
-            var jsdocType = ts.getJSDocType(declaration);
-            if (jsdocType) {
-                return getTypeFromTypeNode(jsdocType);
+        function checkAndReportErrorForExtendingInterface(errorLocation) {
+            var expression = getEntityNameForExtendingInterface(errorLocation);
+            if (expression && resolveEntityName(expression, 64 /* Interface */, /*ignoreErrors*/ true)) {
+                error(errorLocation, ts.Diagnostics.Cannot_extend_an_interface_0_Did_you_mean_implements, ts.getTextOfNode(expression));
+                return true;
             }
-            return undefined;
-        }
-        function isNullOrUndefined(node) {
-            var expr = ts.skipParentheses(node);
-            return expr.kind === 100 /* NullKeyword */ || expr.kind === 75 /* Identifier */ && getResolvedSymbol(expr) === undefinedSymbol;
-        }
-        function isEmptyArrayLiteral(node) {
-            var expr = ts.skipParentheses(node);
-            return expr.kind === 192 /* ArrayLiteralExpression */ && expr.elements.length === 0;
-        }
-        function addOptionality(type, optional) {
-            if (optional === void 0) { optional = true; }
-            return strictNullChecks && optional ? getOptionalType(type) : type;
+            return false;
         }
-        // Return the inferred type for a variable, parameter, or property declaration
-        function getTypeForVariableLikeDeclaration(declaration, includeOptionality) {
-            // A variable declared in a for..in statement is of type string, or of type keyof T when the
-            // right hand expression is of a type parameter type.
-            if (ts.isVariableDeclaration(declaration) && declaration.parent.parent.kind === 231 /* ForInStatement */) {
-                var indexType = getIndexType(getNonNullableTypeIfNeeded(checkExpression(declaration.parent.parent.expression)));
-                return indexType.flags & (262144 /* TypeParameter */ | 4194304 /* Index */) ? getExtractStringType(indexType) : stringType;
-            }
-            if (ts.isVariableDeclaration(declaration) && declaration.parent.parent.kind === 232 /* ForOfStatement */) {
-                // checkRightHandSideOfForOf will return undefined if the for-of expression type was
-                // missing properties/signatures required to get its iteratedType (like
-                // [Symbol.iterator] or next). This may be because we accessed properties from anyType,
-                // or it may have led to an error inside getElementTypeOfIterable.
-                var forOfStatement = declaration.parent.parent;
-                return checkRightHandSideOfForOf(forOfStatement) || anyType;
-            }
-            if (ts.isBindingPattern(declaration.parent)) {
-                return getTypeForBindingElement(declaration);
-            }
-            var isOptional = includeOptionality && (ts.isParameter(declaration) && isJSDocOptionalParameter(declaration)
-                || !ts.isBindingElement(declaration) && !ts.isVariableDeclaration(declaration) && !!declaration.questionToken);
-            // Use type from type annotation if one is present
-            var declaredType = tryGetTypeFromEffectiveTypeNode(declaration);
-            if (declaredType) {
-                return addOptionality(declaredType, isOptional);
-            }
-            if ((noImplicitAny || ts.isInJSFile(declaration)) &&
-                declaration.kind === 242 /* VariableDeclaration */ && !ts.isBindingPattern(declaration.name) &&
-                !(ts.getCombinedModifierFlags(declaration) & 1 /* Export */) && !(declaration.flags & 8388608 /* Ambient */)) {
-                // If --noImplicitAny is on or the declaration is in a Javascript file,
-                // use control flow tracked 'any' type for non-ambient, non-exported var or let variables with no
-                // initializer or a 'null' or 'undefined' initializer.
-                if (!(ts.getCombinedNodeFlags(declaration) & 2 /* Const */) && (!declaration.initializer || isNullOrUndefined(declaration.initializer))) {
-                    return autoType;
-                }
-                // Use control flow tracked 'any[]' type for non-ambient, non-exported variables with an empty array
-                // literal initializer.
-                if (declaration.initializer && isEmptyArrayLiteral(declaration.initializer)) {
-                    return autoArrayType;
-                }
+        /**
+         * Climbs up parents to an ExpressionWithTypeArguments, and returns its expression,
+         * but returns undefined if that expression is not an EntityNameExpression.
+         */
+        function getEntityNameForExtendingInterface(node) {
+            switch (node.kind) {
+                case 78 /* Identifier */:
+                case 201 /* PropertyAccessExpression */:
+                    return node.parent ? getEntityNameForExtendingInterface(node.parent) : undefined;
+                case 223 /* ExpressionWithTypeArguments */:
+                    if (ts.isEntityNameExpression(node.expression)) {
+                        return node.expression;
+                    }
+                // falls through
+                default:
+                    return undefined;
             }
-            if (declaration.kind === 156 /* Parameter */) {
-                var func = declaration.parent;
-                // For a parameter of a set accessor, use the type of the get accessor if one is present
-                if (func.kind === 164 /* SetAccessor */ && !hasNonBindableDynamicName(func)) {
-                    var getter = ts.getDeclarationOfKind(getSymbolOfNode(declaration.parent), 163 /* GetAccessor */);
-                    if (getter) {
-                        var getterSignature = getSignatureFromDeclaration(getter);
-                        var thisParameter = getAccessorThisParameter(func);
-                        if (thisParameter && declaration === thisParameter) {
-                            // Use the type from the *getter*
-                            ts.Debug.assert(!thisParameter.type);
-                            return getTypeOfSymbol(getterSignature.thisParameter);
+        }
+        function checkAndReportErrorForUsingTypeAsNamespace(errorLocation, name, meaning) {
+            var namespaceMeaning = 1920 /* Namespace */ | (ts.isInJSFile(errorLocation) ? 111551 /* Value */ : 0);
+            if (meaning === namespaceMeaning) {
+                var symbol = resolveSymbol(resolveName(errorLocation, name, 788968 /* Type */ & ~namespaceMeaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
+                var parent = errorLocation.parent;
+                if (symbol) {
+                    if (ts.isQualifiedName(parent)) {
+                        ts.Debug.assert(parent.left === errorLocation, "Should only be resolving left side of qualified name as a namespace");
+                        var propName = parent.right.escapedText;
+                        var propType = getPropertyOfType(getDeclaredTypeOfSymbol(symbol), propName);
+                        if (propType) {
+                            error(parent, ts.Diagnostics.Cannot_access_0_1_because_0_is_a_type_but_not_a_namespace_Did_you_mean_to_retrieve_the_type_of_the_property_1_in_0_with_0_1, ts.unescapeLeadingUnderscores(name), ts.unescapeLeadingUnderscores(propName));
+                            return true;
                         }
-                        return getReturnTypeOfSignature(getterSignature);
-                    }
-                }
-                if (ts.isInJSFile(declaration)) {
-                    var typeTag = ts.getJSDocType(func);
-                    if (typeTag && ts.isFunctionTypeNode(typeTag)) {
-                        return getTypeAtPosition(getSignatureFromDeclaration(typeTag), func.parameters.indexOf(declaration));
                     }
-                }
-                // Use contextual parameter type if one is available
-                var type = declaration.symbol.escapedName === "this" /* This */ ? getContextualThisParameterType(func) : getContextuallyTypedParameterType(declaration);
-                if (type) {
-                    return addOptionality(type, isOptional);
+                    error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_namespace_here, ts.unescapeLeadingUnderscores(name));
+                    return true;
                 }
             }
-            else if (ts.isInJSFile(declaration)) {
-                var containerObjectType = getJSContainerObjectType(declaration, getSymbolOfNode(declaration), ts.getDeclaredExpandoInitializer(declaration));
-                if (containerObjectType) {
-                    return containerObjectType;
+            return false;
+        }
+        function checkAndReportErrorForUsingValueAsType(errorLocation, name, meaning) {
+            if (meaning & (788968 /* Type */ & ~1920 /* Namespace */)) {
+                var symbol = resolveSymbol(resolveName(errorLocation, name, ~788968 /* Type */ & 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
+                if (symbol && !(symbol.flags & 1920 /* Namespace */)) {
+                    error(errorLocation, ts.Diagnostics._0_refers_to_a_value_but_is_being_used_as_a_type_here_Did_you_mean_typeof_0, ts.unescapeLeadingUnderscores(name));
+                    return true;
                 }
             }
-            // Use the type of the initializer expression if one is present and the declaration is
-            // not a parameter of a contextually typed function
-            if (declaration.initializer) {
-                var type = widenTypeInferredFromInitializer(declaration, checkDeclarationInitializer(declaration));
-                return addOptionality(type, isOptional);
-            }
-            if (ts.isJsxAttribute(declaration)) {
-                // if JSX attribute doesn't have initializer, by default the attribute will have boolean value of true.
-                // I.e <Elem attr /> is sugar for <Elem attr={true} />
-                return trueType;
-            }
-            // If the declaration specifies a binding pattern and is not a parameter of a contextually
-            // typed function, use the type implied by the binding pattern
-            if (ts.isBindingPattern(declaration.name)) {
-                return getTypeFromBindingPattern(declaration.name, /*includePatternInType*/ false, /*reportErrors*/ true);
-            }
-            // No type specified and nothing can be inferred
-            return undefined;
+            return false;
         }
-        function getWidenedTypeForAssignmentDeclaration(symbol, resolvedSymbol) {
-            // function/class/{} initializers are themselves containers, so they won't merge in the same way as other initializers
-            var container = ts.getAssignedExpandoInitializer(symbol.valueDeclaration);
-            if (container) {
-                var tag = ts.getJSDocTypeTag(container);
-                if (tag && tag.typeExpression) {
-                    return getTypeFromTypeNode(tag.typeExpression);
-                }
-                var containerObjectType = getJSContainerObjectType(symbol.valueDeclaration, symbol, container);
-                return containerObjectType || getWidenedLiteralType(checkExpressionCached(container));
+        function isPrimitiveTypeName(name) {
+            return name === "any" || name === "string" || name === "number" || name === "boolean" || name === "never" || name === "unknown";
+        }
+        function checkAndReportErrorForExportingPrimitiveType(errorLocation, name) {
+            if (isPrimitiveTypeName(name) && errorLocation.parent.kind === 270 /* ExportSpecifier */) {
+                error(errorLocation, ts.Diagnostics.Cannot_export_0_Only_local_declarations_can_be_exported_from_a_module, name);
+                return true;
             }
-            var definedInConstructor = false;
-            var definedInMethod = false;
-            var jsdocType;
-            var types;
-            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                var declaration = _a[_i];
-                var expression = (ts.isBinaryExpression(declaration) || ts.isCallExpression(declaration)) ? declaration :
-                    ts.isAccessExpression(declaration) ? ts.isBinaryExpression(declaration.parent) ? declaration.parent : declaration :
-                        undefined;
-                if (!expression) {
-                    continue; // Non-assignment declaration merged in (eg, an Identifier to mark the thing as a namespace) - skip over it and pull type info from elsewhere
+            return false;
+        }
+        function checkAndReportErrorForUsingTypeAsValue(errorLocation, name, meaning) {
+            if (meaning & (111551 /* Value */ & ~1024 /* NamespaceModule */)) {
+                if (isPrimitiveTypeName(name)) {
+                    error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here, ts.unescapeLeadingUnderscores(name));
+                    return true;
                 }
-                var kind = ts.isAccessExpression(expression)
-                    ? ts.getAssignmentDeclarationPropertyAccessKind(expression)
-                    : ts.getAssignmentDeclarationKind(expression);
-                if (kind === 4 /* ThisProperty */) {
-                    if (isDeclarationInConstructor(expression)) {
-                        definedInConstructor = true;
+                var symbol = resolveSymbol(resolveName(errorLocation, name, 788968 /* Type */ & ~111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
+                if (symbol && !(symbol.flags & 1024 /* NamespaceModule */)) {
+                    var rawName = ts.unescapeLeadingUnderscores(name);
+                    if (isES2015OrLaterConstructorName(name)) {
+                        error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_es2015_or_later, rawName);
                     }
-                    else {
-                        definedInMethod = true;
+                    else if (maybeMappedType(errorLocation, symbol)) {
+                        error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Did_you_mean_to_use_1_in_0, rawName, rawName === "K" ? "P" : "K");
                     }
-                }
-                if (!ts.isCallExpression(expression)) {
-                    jsdocType = getAnnotatedTypeForAssignmentDeclaration(jsdocType, expression, symbol, declaration);
-                }
-                if (!jsdocType) {
-                    (types || (types = [])).push((ts.isBinaryExpression(expression) || ts.isCallExpression(expression)) ? getInitializerTypeFromAssignmentDeclaration(symbol, resolvedSymbol, expression, kind) : neverType);
-                }
-            }
-            var type = jsdocType;
-            if (!type) {
-                if (!ts.length(types)) {
-                    return errorType; // No types from any declarations :(
-                }
-                var constructorTypes = definedInConstructor ? getConstructorDefinedThisAssignmentTypes(types, symbol.declarations) : undefined;
-                // use only the constructor types unless they were only assigned null | undefined (including widening variants)
-                if (definedInMethod) {
-                    var propType = getTypeOfAssignmentDeclarationPropertyOfBaseType(symbol);
-                    if (propType) {
-                        (constructorTypes || (constructorTypes = [])).push(propType);
-                        definedInConstructor = true;
+                    else {
+                        error(errorLocation, ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here, rawName);
                     }
+                    return true;
                 }
-                var sourceTypes = ts.some(constructorTypes, function (t) { return !!(t.flags & ~98304 /* Nullable */); }) ? constructorTypes : types; // TODO: GH#18217
-                type = getUnionType(sourceTypes, 2 /* Subtype */);
-            }
-            var widened = getWidenedType(addOptionality(type, definedInMethod && !definedInConstructor));
-            if (filterType(widened, function (t) { return !!(t.flags & ~98304 /* Nullable */); }) === neverType) {
-                reportImplicitAny(symbol.valueDeclaration, anyType);
-                return anyType;
             }
-            return widened;
+            return false;
         }
-        function getJSContainerObjectType(decl, symbol, init) {
-            if (!ts.isInJSFile(decl) || !init || !ts.isObjectLiteralExpression(init) || init.properties.length) {
-                return undefined;
-            }
-            var exports = ts.createSymbolTable();
-            while (ts.isBinaryExpression(decl) || ts.isPropertyAccessExpression(decl)) {
-                var s_2 = getSymbolOfNode(decl);
-                if (s_2 && ts.hasEntries(s_2.exports)) {
-                    mergeSymbolTable(exports, s_2.exports);
-                }
-                decl = ts.isBinaryExpression(decl) ? decl.parent : decl.parent.parent;
+        function maybeMappedType(node, symbol) {
+            var container = ts.findAncestor(node.parent, function (n) {
+                return ts.isComputedPropertyName(n) || ts.isPropertySignature(n) ? false : ts.isTypeLiteralNode(n) || "quit";
+            });
+            if (container && container.members.length === 1) {
+                var type = getDeclaredTypeOfSymbol(symbol);
+                return !!(type.flags & 1048576 /* Union */) && allTypesAssignableToKind(type, 384 /* StringOrNumberLiteral */, /*strict*/ true);
             }
-            var s = getSymbolOfNode(decl);
-            if (s && ts.hasEntries(s.exports)) {
-                mergeSymbolTable(exports, s.exports);
+            return false;
+        }
+        function isES2015OrLaterConstructorName(n) {
+            switch (n) {
+                case "Promise":
+                case "Symbol":
+                case "Map":
+                case "WeakMap":
+                case "Set":
+                case "WeakSet":
+                    return true;
             }
-            var type = createAnonymousType(symbol, exports, ts.emptyArray, ts.emptyArray, undefined, undefined);
-            type.objectFlags |= 16384 /* JSLiteral */;
-            return type;
+            return false;
         }
-        function getAnnotatedTypeForAssignmentDeclaration(declaredType, expression, symbol, declaration) {
-            var typeNode = ts.getEffectiveTypeAnnotationNode(expression.parent);
-            if (typeNode) {
-                var type = getWidenedType(getTypeFromTypeNode(typeNode));
-                if (!declaredType) {
-                    return type;
-                }
-                else if (declaredType !== errorType && type !== errorType && !isTypeIdenticalTo(declaredType, type)) {
-                    errorNextVariableOrPropertyDeclarationMustHaveSameType(/*firstDeclaration*/ undefined, declaredType, declaration, type);
+        function checkAndReportErrorForUsingNamespaceModuleAsValue(errorLocation, name, meaning) {
+            if (meaning & (111551 /* Value */ & ~1024 /* NamespaceModule */ & ~788968 /* Type */)) {
+                var symbol = resolveSymbol(resolveName(errorLocation, name, 1024 /* NamespaceModule */ & ~111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
+                if (symbol) {
+                    error(errorLocation, ts.Diagnostics.Cannot_use_namespace_0_as_a_value, ts.unescapeLeadingUnderscores(name));
+                    return true;
                 }
             }
-            if (symbol.parent) {
-                var typeNode_2 = ts.getEffectiveTypeAnnotationNode(symbol.parent.valueDeclaration);
-                if (typeNode_2) {
-                    return getTypeOfPropertyOfType(getTypeFromTypeNode(typeNode_2), symbol.escapedName);
+            else if (meaning & (788968 /* Type */ & ~1024 /* NamespaceModule */ & ~111551 /* Value */)) {
+                var symbol = resolveSymbol(resolveName(errorLocation, name, (512 /* ValueModule */ | 1024 /* NamespaceModule */) & ~788968 /* Type */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false));
+                if (symbol) {
+                    error(errorLocation, ts.Diagnostics.Cannot_use_namespace_0_as_a_type, ts.unescapeLeadingUnderscores(name));
+                    return true;
                 }
             }
-            return declaredType;
+            return false;
         }
-        /** If we don't have an explicit JSDoc type, get the type from the initializer. */
-        function getInitializerTypeFromAssignmentDeclaration(symbol, resolvedSymbol, expression, kind) {
-            if (ts.isCallExpression(expression)) {
-                if (resolvedSymbol) {
-                    return getTypeOfSymbol(resolvedSymbol); // This shouldn't happen except under some hopefully forbidden merges of export assignments and object define assignments
+        function checkResolvedBlockScopedVariable(result, errorLocation) {
+            ts.Debug.assert(!!(result.flags & 2 /* BlockScopedVariable */ || result.flags & 32 /* Class */ || result.flags & 384 /* Enum */));
+            if (result.flags & (16 /* Function */ | 1 /* FunctionScopedVariable */ | 67108864 /* Assignment */) && result.flags & 32 /* Class */) {
+                // constructor functions aren't block scoped
+                return;
+            }
+            // Block-scoped variables cannot be used before their definition
+            var declaration = ts.find(result.declarations, function (d) { return ts.isBlockOrCatchScoped(d) || ts.isClassLike(d) || (d.kind === 255 /* EnumDeclaration */); });
+            if (declaration === undefined)
+                return ts.Debug.fail("checkResolvedBlockScopedVariable could not find block-scoped declaration");
+            if (!(declaration.flags & 8388608 /* Ambient */) && !isBlockScopedNameDeclaredBeforeUse(declaration, errorLocation)) {
+                var diagnosticMessage = void 0;
+                var declarationName = ts.declarationNameToString(ts.getNameOfDeclaration(declaration));
+                if (result.flags & 2 /* BlockScopedVariable */) {
+                    diagnosticMessage = error(errorLocation, ts.Diagnostics.Block_scoped_variable_0_used_before_its_declaration, declarationName);
                 }
-                var objectLitType = checkExpressionCached(expression.arguments[2]);
-                var valueType = getTypeOfPropertyOfType(objectLitType, "value");
-                if (valueType) {
-                    return valueType;
+                else if (result.flags & 32 /* Class */) {
+                    diagnosticMessage = error(errorLocation, ts.Diagnostics.Class_0_used_before_its_declaration, declarationName);
                 }
-                var getFunc = getTypeOfPropertyOfType(objectLitType, "get");
-                if (getFunc) {
-                    var getSig = getSingleCallSignature(getFunc);
-                    if (getSig) {
-                        return getReturnTypeOfSignature(getSig);
-                    }
+                else if (result.flags & 256 /* RegularEnum */) {
+                    diagnosticMessage = error(errorLocation, ts.Diagnostics.Enum_0_used_before_its_declaration, declarationName);
                 }
-                var setFunc = getTypeOfPropertyOfType(objectLitType, "set");
-                if (setFunc) {
-                    var setSig = getSingleCallSignature(setFunc);
-                    if (setSig) {
-                        return getTypeOfFirstParameterOfSignature(setSig);
+                else {
+                    ts.Debug.assert(!!(result.flags & 128 /* ConstEnum */));
+                    if (compilerOptions.preserveConstEnums) {
+                        diagnosticMessage = error(errorLocation, ts.Diagnostics.Enum_0_used_before_its_declaration, declarationName);
                     }
                 }
-                return anyType;
-            }
-            if (containsSameNamedThisProperty(expression.left, expression.right)) {
-                return anyType;
-            }
-            var type = resolvedSymbol ? getTypeOfSymbol(resolvedSymbol) : getWidenedLiteralType(checkExpressionCached(expression.right));
-            if (type.flags & 524288 /* Object */ &&
-                kind === 2 /* ModuleExports */ &&
-                symbol.escapedName === "export=" /* ExportEquals */) {
-                var exportedType = resolveStructuredTypeMembers(type);
-                var members_4 = ts.createSymbolTable();
-                ts.copyEntries(exportedType.members, members_4);
-                if (resolvedSymbol && !resolvedSymbol.exports) {
-                    resolvedSymbol.exports = ts.createSymbolTable();
+                if (diagnosticMessage) {
+                    ts.addRelatedInfo(diagnosticMessage, ts.createDiagnosticForNode(declaration, ts.Diagnostics._0_is_declared_here, declarationName));
                 }
-                (resolvedSymbol || symbol).exports.forEach(function (s, name) {
-                    var _a;
-                    var exportedMember = members_4.get(name);
-                    if (exportedMember && exportedMember !== s) {
-                        if (s.flags & 111551 /* Value */) {
-                            // If the member has an additional value-like declaration, union the types from the two declarations,
-                            // but issue an error if they occurred in two different files. The purpose is to support a JS file with
-                            // a pattern like:
-                            //
-                            // module.exports = { a: true };
-                            // module.exports.a = 3;
-                            //
-                            // but we may have a JS file with `module.exports = { a: true }` along with a TypeScript module augmentation
-                            // declaring an `export const a: number`. In that case, we issue a duplicate identifier error, because
-                            // it's unclear what that's supposed to mean, so it's probably a mistake.
-                            if (ts.getSourceFileOfNode(s.valueDeclaration) !== ts.getSourceFileOfNode(exportedMember.valueDeclaration)) {
-                                var unescapedName = ts.unescapeLeadingUnderscores(s.escapedName);
-                                var exportedMemberName = ((_a = ts.tryCast(exportedMember.valueDeclaration, ts.isNamedDeclaration)) === null || _a === void 0 ? void 0 : _a.name) || exportedMember.valueDeclaration;
-                                ts.addRelatedInfo(error(s.valueDeclaration, ts.Diagnostics.Duplicate_identifier_0, unescapedName), ts.createDiagnosticForNode(exportedMemberName, ts.Diagnostics._0_was_also_declared_here, unescapedName));
-                                ts.addRelatedInfo(error(exportedMemberName, ts.Diagnostics.Duplicate_identifier_0, unescapedName), ts.createDiagnosticForNode(s.valueDeclaration, ts.Diagnostics._0_was_also_declared_here, unescapedName));
-                            }
-                            var union = createSymbol(s.flags | exportedMember.flags, name);
-                            union.type = getUnionType([getTypeOfSymbol(s), getTypeOfSymbol(exportedMember)]);
-                            union.valueDeclaration = exportedMember.valueDeclaration;
-                            union.declarations = ts.concatenate(exportedMember.declarations, s.declarations);
-                            members_4.set(name, union);
-                        }
-                        else {
-                            members_4.set(name, mergeSymbol(s, exportedMember));
-                        }
-                    }
-                    else {
-                        members_4.set(name, s);
-                    }
-                });
-                var result = createAnonymousType(exportedType.symbol, members_4, exportedType.callSignatures, exportedType.constructSignatures, exportedType.stringIndexInfo, exportedType.numberIndexInfo);
-                result.objectFlags |= (ts.getObjectFlags(type) & 16384 /* JSLiteral */); // Propagate JSLiteral flag
-                return result;
             }
-            if (isEmptyArrayLiteralType(type)) {
-                reportImplicitAny(expression, anyArrayType);
-                return anyArrayType;
+        }
+        /* Starting from 'initial' node walk up the parent chain until 'stopAt' node is reached.
+         * If at any point current node is equal to 'parent' node - return true.
+         * Return false if 'stopAt' node is reached or isFunctionLike(current) === true.
+         */
+        function isSameScopeDescendentOf(initial, parent, stopAt) {
+            return !!parent && !!ts.findAncestor(initial, function (n) { return n === stopAt || ts.isFunctionLike(n) ? "quit" : n === parent; });
+        }
+        function getAnyImportSyntax(node) {
+            switch (node.kind) {
+                case 260 /* ImportEqualsDeclaration */:
+                    return node;
+                case 262 /* ImportClause */:
+                    return node.parent;
+                case 263 /* NamespaceImport */:
+                    return node.parent.parent;
+                case 265 /* ImportSpecifier */:
+                    return node.parent.parent.parent;
+                default:
+                    return undefined;
             }
-            return type;
         }
-        function containsSameNamedThisProperty(thisProperty, expression) {
-            return ts.isPropertyAccessExpression(thisProperty)
-                && thisProperty.expression.kind === 104 /* ThisKeyword */
-                && ts.forEachChildRecursively(expression, function (n) { return isMatchingReference(thisProperty, n); });
+        function getDeclarationOfAliasSymbol(symbol) {
+            return ts.find(symbol.declarations, isAliasSymbolDeclaration);
         }
-        function isDeclarationInConstructor(expression) {
-            var thisContainer = ts.getThisContainer(expression, /*includeArrowFunctions*/ false);
-            // Properties defined in a constructor (or base constructor, or javascript constructor function) don't get undefined added.
-            // Function expressions that are assigned to the prototype count as methods.
-            return thisContainer.kind === 162 /* Constructor */ ||
-                thisContainer.kind === 244 /* FunctionDeclaration */ ||
-                (thisContainer.kind === 201 /* FunctionExpression */ && !ts.isPrototypePropertyAssignment(thisContainer.parent));
+        /**
+         * An alias symbol is created by one of the following declarations:
+         * import <symbol> = ...
+         * import <symbol> from ...
+         * import * as <symbol> from ...
+         * import { x as <symbol> } from ...
+         * export { x as <symbol> } from ...
+         * export * as ns <symbol> from ...
+         * export = <EntityNameExpression>
+         * export default <EntityNameExpression>
+         * module.exports = <EntityNameExpression>
+         * {<Identifier>}
+         * {name: <EntityNameExpression>}
+         */
+        function isAliasSymbolDeclaration(node) {
+            return node.kind === 260 /* ImportEqualsDeclaration */
+                || node.kind === 259 /* NamespaceExportDeclaration */
+                || node.kind === 262 /* ImportClause */ && !!node.name
+                || node.kind === 263 /* NamespaceImport */
+                || node.kind === 269 /* NamespaceExport */
+                || node.kind === 265 /* ImportSpecifier */
+                || node.kind === 270 /* ExportSpecifier */
+                || node.kind === 266 /* ExportAssignment */ && ts.exportAssignmentIsAlias(node)
+                || ts.isBinaryExpression(node) && ts.getAssignmentDeclarationKind(node) === 2 /* ModuleExports */ && ts.exportAssignmentIsAlias(node)
+                || ts.isAccessExpression(node)
+                    && ts.isBinaryExpression(node.parent)
+                    && node.parent.left === node
+                    && node.parent.operatorToken.kind === 62 /* EqualsToken */
+                    && isAliasableOrJsExpression(node.parent.right)
+                || node.kind === 289 /* ShorthandPropertyAssignment */
+                || node.kind === 288 /* PropertyAssignment */ && isAliasableOrJsExpression(node.initializer)
+                || ts.isRequireVariableDeclaration(node, /*requireStringLiteralLikeArgument*/ true);
         }
-        function getConstructorDefinedThisAssignmentTypes(types, declarations) {
-            ts.Debug.assert(types.length === declarations.length);
-            return types.filter(function (_, i) {
-                var declaration = declarations[i];
-                var expression = ts.isBinaryExpression(declaration) ? declaration :
-                    ts.isBinaryExpression(declaration.parent) ? declaration.parent : undefined;
-                return expression && isDeclarationInConstructor(expression);
-            });
+        function isAliasableOrJsExpression(e) {
+            return ts.isAliasableExpression(e) || ts.isFunctionExpression(e) && isJSConstructor(e);
         }
-        /** check for definition in base class if any declaration is in a class */
-        function getTypeOfAssignmentDeclarationPropertyOfBaseType(property) {
-            var parentDeclaration = ts.forEach(property.declarations, function (d) {
-                var parent = ts.getThisContainer(d, /*includeArrowFunctions*/ false).parent;
-                return ts.isClassLike(parent) && parent;
-            });
-            if (parentDeclaration) {
-                var classType = getDeclaredTypeOfSymbol(getSymbolOfNode(parentDeclaration));
-                var baseClassType = classType && getBaseTypes(classType)[0];
-                if (baseClassType) {
-                    return getTypeOfPropertyOfType(baseClassType, property.escapedName);
-                }
+        function getTargetOfImportEqualsDeclaration(node, dontResolveAlias) {
+            var commonJSPropertyAccess = getCommonJSPropertyAccess(node);
+            if (commonJSPropertyAccess) {
+                var name = ts.getLeftmostAccessExpression(commonJSPropertyAccess.expression).arguments[0];
+                return ts.isIdentifier(commonJSPropertyAccess.name)
+                    ? resolveSymbol(getPropertyOfType(resolveExternalModuleTypeByLiteral(name), commonJSPropertyAccess.name.escapedText))
+                    : undefined;
+            }
+            if (ts.isVariableDeclaration(node) || node.moduleReference.kind === 272 /* ExternalModuleReference */) {
+                var immediate = resolveExternalModuleName(node, ts.getExternalModuleRequireArgument(node) || ts.getExternalModuleImportEqualsDeclarationExpression(node));
+                var resolved_4 = resolveExternalModuleSymbol(immediate);
+                markSymbolOfAliasDeclarationIfTypeOnly(node, immediate, resolved_4, /*overwriteEmpty*/ false);
+                return resolved_4;
             }
+            var resolved = getSymbolOfPartOfRightHandSideOfImportEquals(node.moduleReference, dontResolveAlias);
+            checkAndReportErrorForResolvingImportAliasToTypeOnlySymbol(node, resolved);
+            return resolved;
         }
-        // Return the type implied by a binding pattern element. This is the type of the initializer of the element if
-        // one is present. Otherwise, if the element is itself a binding pattern, it is the type implied by the binding
-        // pattern. Otherwise, it is the type any.
-        function getTypeFromBindingElement(element, includePatternInType, reportErrors) {
-            if (element.initializer) {
-                // The type implied by a binding pattern is independent of context, so we check the initializer with no
-                // contextual type or, if the element itself is a binding pattern, with the type implied by that binding
-                // pattern.
-                var contextualType = ts.isBindingPattern(element.name) ? getTypeFromBindingPattern(element.name, /*includePatternInType*/ true, /*reportErrors*/ false) : unknownType;
-                return addOptionality(widenTypeInferredFromInitializer(element, checkDeclarationInitializer(element, contextualType)));
+        function checkAndReportErrorForResolvingImportAliasToTypeOnlySymbol(node, resolved) {
+            if (markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false)) {
+                var typeOnlyDeclaration = getTypeOnlyAliasDeclaration(getSymbolOfNode(node));
+                var isExport = ts.typeOnlyDeclarationIsExport(typeOnlyDeclaration);
+                var message = isExport
+                    ? ts.Diagnostics.An_import_alias_cannot_reference_a_declaration_that_was_exported_using_export_type
+                    : ts.Diagnostics.An_import_alias_cannot_reference_a_declaration_that_was_imported_using_import_type;
+                var relatedMessage = isExport
+                    ? ts.Diagnostics._0_was_exported_here
+                    : ts.Diagnostics._0_was_imported_here;
+                // Non-null assertion is safe because the optionality comes from ImportClause,
+                // but if an ImportClause was the typeOnlyDeclaration, it had to have a `name`.
+                var name = ts.unescapeLeadingUnderscores(typeOnlyDeclaration.name.escapedText);
+                ts.addRelatedInfo(error(node.moduleReference, message), ts.createDiagnosticForNode(typeOnlyDeclaration, relatedMessage, name));
             }
-            if (ts.isBindingPattern(element.name)) {
-                return getTypeFromBindingPattern(element.name, includePatternInType, reportErrors);
+        }
+        function resolveExportByName(moduleSymbol, name, sourceNode, dontResolveAlias) {
+            var exportValue = moduleSymbol.exports.get("export=" /* ExportEquals */);
+            var exportSymbol = exportValue ? getPropertyOfType(getTypeOfSymbol(exportValue), name) : moduleSymbol.exports.get(name);
+            var resolved = resolveSymbol(exportSymbol, dontResolveAlias);
+            markSymbolOfAliasDeclarationIfTypeOnly(sourceNode, exportSymbol, resolved, /*overwriteEmpty*/ false);
+            return resolved;
+        }
+        function isSyntacticDefault(node) {
+            return ((ts.isExportAssignment(node) && !node.isExportEquals) || ts.hasSyntacticModifier(node, 512 /* Default */) || ts.isExportSpecifier(node));
+        }
+        function canHaveSyntheticDefault(file, moduleSymbol, dontResolveAlias) {
+            if (!allowSyntheticDefaultImports) {
+                return false;
             }
-            if (reportErrors && !declarationBelongsToPrivateAmbientMember(element)) {
-                reportImplicitAny(element, anyType);
+            // Declaration files (and ambient modules)
+            if (!file || file.isDeclarationFile) {
+                // Definitely cannot have a synthetic default if they have a syntactic default member specified
+                var defaultExportSymbol = resolveExportByName(moduleSymbol, "default" /* Default */, /*sourceNode*/ undefined, /*dontResolveAlias*/ true); // Dont resolve alias because we want the immediately exported symbol's declaration
+                if (defaultExportSymbol && ts.some(defaultExportSymbol.declarations, isSyntacticDefault)) {
+                    return false;
+                }
+                // It _might_ still be incorrect to assume there is no __esModule marker on the import at runtime, even if there is no `default` member
+                // So we check a bit more,
+                if (resolveExportByName(moduleSymbol, ts.escapeLeadingUnderscores("__esModule"), /*sourceNode*/ undefined, dontResolveAlias)) {
+                    // If there is an `__esModule` specified in the declaration (meaning someone explicitly added it or wrote it in their code),
+                    // it definitely is a module and does not have a synthetic default
+                    return false;
+                }
+                // There are _many_ declaration files not written with esmodules in mind that still get compiled into a format with __esModule set
+                // Meaning there may be no default at runtime - however to be on the permissive side, we allow access to a synthetic default member
+                // as there is no marker to indicate if the accompanying JS has `__esModule` or not, or is even native esm
+                return true;
             }
-            // When we're including the pattern in the type (an indication we're obtaining a contextual type), we
-            // use the non-inferrable any type. Inference will never directly infer this type, but it is possible
-            // to infer a type that contains it, e.g. for a binding pattern like [foo] or { foo }. In such cases,
-            // widening of the binding pattern type substitutes a regular any for the non-inferrable any.
-            return includePatternInType ? nonInferrableAnyType : anyType;
+            // TypeScript files never have a synthetic default (as they are always emitted with an __esModule marker) _unless_ they contain an export= statement
+            if (!ts.isSourceFileJS(file)) {
+                return hasExportAssignmentSymbol(moduleSymbol);
+            }
+            // JS files have a synthetic default if they do not contain ES2015+ module syntax (export = is not valid in js) _and_ do not have an __esModule marker
+            return !file.externalModuleIndicator && !resolveExportByName(moduleSymbol, ts.escapeLeadingUnderscores("__esModule"), /*sourceNode*/ undefined, dontResolveAlias);
         }
-        // Return the type implied by an object binding pattern
-        function getTypeFromObjectBindingPattern(pattern, includePatternInType, reportErrors) {
-            var members = ts.createSymbolTable();
-            var stringIndexInfo;
-            var objectFlags = 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
-            ts.forEach(pattern.elements, function (e) {
-                var name = e.propertyName || e.name;
-                if (e.dotDotDotToken) {
-                    stringIndexInfo = createIndexInfo(anyType, /*isReadonly*/ false);
-                    return;
+        function getTargetOfImportClause(node, dontResolveAlias) {
+            var moduleSymbol = resolveExternalModuleName(node, node.parent.moduleSpecifier);
+            if (moduleSymbol) {
+                var exportDefaultSymbol = void 0;
+                if (ts.isShorthandAmbientModuleSymbol(moduleSymbol)) {
+                    exportDefaultSymbol = moduleSymbol;
                 }
-                var exprType = getLiteralTypeFromPropertyName(name);
-                if (!isTypeUsableAsPropertyName(exprType)) {
-                    // do not include computed properties in the implied type
-                    objectFlags |= 512 /* ObjectLiteralPatternWithComputedProperties */;
-                    return;
+                else {
+                    exportDefaultSymbol = resolveExportByName(moduleSymbol, "default" /* Default */, node, dontResolveAlias);
                 }
-                var text = getPropertyNameFromType(exprType);
-                var flags = 4 /* Property */ | (e.initializer ? 16777216 /* Optional */ : 0);
-                var symbol = createSymbol(flags, text);
-                symbol.type = getTypeFromBindingElement(e, includePatternInType, reportErrors);
-                symbol.bindingElement = e;
-                members.set(symbol.escapedName, symbol);
-            });
-            var result = createAnonymousType(undefined, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, undefined);
-            result.objectFlags |= objectFlags;
-            if (includePatternInType) {
-                result.pattern = pattern;
-                result.objectFlags |= 1048576 /* ContainsObjectOrArrayLiteral */;
+                var file = ts.find(moduleSymbol.declarations, ts.isSourceFile);
+                var hasSyntheticDefault = canHaveSyntheticDefault(file, moduleSymbol, dontResolveAlias);
+                if (!exportDefaultSymbol && !hasSyntheticDefault) {
+                    if (hasExportAssignmentSymbol(moduleSymbol)) {
+                        var compilerOptionName = moduleKind >= ts.ModuleKind.ES2015 ? "allowSyntheticDefaultImports" : "esModuleInterop";
+                        var exportEqualsSymbol = moduleSymbol.exports.get("export=" /* ExportEquals */);
+                        var exportAssignment = exportEqualsSymbol.valueDeclaration;
+                        var err = error(node.name, ts.Diagnostics.Module_0_can_only_be_default_imported_using_the_1_flag, symbolToString(moduleSymbol), compilerOptionName);
+                        ts.addRelatedInfo(err, ts.createDiagnosticForNode(exportAssignment, ts.Diagnostics.This_module_is_declared_with_using_export_and_can_only_be_used_with_a_default_import_when_using_the_0_flag, compilerOptionName));
+                    }
+                    else {
+                        reportNonDefaultExport(moduleSymbol, node);
+                    }
+                }
+                else if (hasSyntheticDefault) {
+                    // per emit behavior, a synthetic default overrides a "real" .default member if `__esModule` is not present
+                    var resolved = resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias) || resolveSymbol(moduleSymbol, dontResolveAlias);
+                    markSymbolOfAliasDeclarationIfTypeOnly(node, moduleSymbol, resolved, /*overwriteTypeOnly*/ false);
+                    return resolved;
+                }
+                markSymbolOfAliasDeclarationIfTypeOnly(node, exportDefaultSymbol, /*finalTarget*/ undefined, /*overwriteTypeOnly*/ false);
+                return exportDefaultSymbol;
             }
-            return result;
         }
-        // Return the type implied by an array binding pattern
-        function getTypeFromArrayBindingPattern(pattern, includePatternInType, reportErrors) {
-            var elements = pattern.elements;
-            var lastElement = ts.lastOrUndefined(elements);
-            var hasRestElement = !!(lastElement && lastElement.kind === 191 /* BindingElement */ && lastElement.dotDotDotToken);
-            if (elements.length === 0 || elements.length === 1 && hasRestElement) {
-                return languageVersion >= 2 /* ES2015 */ ? createIterableType(anyType) : anyArrayType;
+        function reportNonDefaultExport(moduleSymbol, node) {
+            var _a, _b;
+            if ((_a = moduleSymbol.exports) === null || _a === void 0 ? void 0 : _a.has(node.symbol.escapedName)) {
+                error(node.name, ts.Diagnostics.Module_0_has_no_default_export_Did_you_mean_to_use_import_1_from_0_instead, symbolToString(moduleSymbol), symbolToString(node.symbol));
             }
-            var elementTypes = ts.map(elements, function (e) { return ts.isOmittedExpression(e) ? anyType : getTypeFromBindingElement(e, includePatternInType, reportErrors); });
-            var minLength = ts.findLastIndex(elements, function (e) { return !ts.isOmittedExpression(e) && !hasDefaultValue(e); }, elements.length - (hasRestElement ? 2 : 1)) + 1;
-            var result = createTupleType(elementTypes, minLength, hasRestElement);
-            if (includePatternInType) {
-                result = cloneTypeReference(result);
-                result.pattern = pattern;
-                result.objectFlags |= 1048576 /* ContainsObjectOrArrayLiteral */;
+            else {
+                var diagnostic = error(node.name, ts.Diagnostics.Module_0_has_no_default_export, symbolToString(moduleSymbol));
+                var exportStar = (_b = moduleSymbol.exports) === null || _b === void 0 ? void 0 : _b.get("__export" /* ExportStar */);
+                if (exportStar) {
+                    var defaultExport = ts.find(exportStar.declarations, function (decl) {
+                        var _a, _b;
+                        return !!(ts.isExportDeclaration(decl) && decl.moduleSpecifier && ((_b = (_a = resolveExternalModuleName(decl, decl.moduleSpecifier)) === null || _a === void 0 ? void 0 : _a.exports) === null || _b === void 0 ? void 0 : _b.has("default" /* Default */)));
+                    });
+                    if (defaultExport) {
+                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(defaultExport, ts.Diagnostics.export_Asterisk_does_not_re_export_a_default));
+                    }
+                }
             }
-            return result;
         }
-        // Return the type implied by a binding pattern. This is the type implied purely by the binding pattern itself
-        // and without regard to its context (i.e. without regard any type annotation or initializer associated with the
-        // declaration in which the binding pattern is contained). For example, the implied type of [x, y] is [any, any]
-        // and the implied type of { x, y: z = 1 } is { x: any; y: number; }. The type implied by a binding pattern is
-        // used as the contextual type of an initializer associated with the binding pattern. Also, for a destructuring
-        // parameter with no type annotation or initializer, the type implied by the binding pattern becomes the type of
-        // the parameter.
-        function getTypeFromBindingPattern(pattern, includePatternInType, reportErrors) {
-            if (includePatternInType === void 0) { includePatternInType = false; }
-            if (reportErrors === void 0) { reportErrors = false; }
-            return pattern.kind === 189 /* ObjectBindingPattern */
-                ? getTypeFromObjectBindingPattern(pattern, includePatternInType, reportErrors)
-                : getTypeFromArrayBindingPattern(pattern, includePatternInType, reportErrors);
+        function getTargetOfNamespaceImport(node, dontResolveAlias) {
+            var moduleSpecifier = node.parent.parent.moduleSpecifier;
+            var immediate = resolveExternalModuleName(node, moduleSpecifier);
+            var resolved = resolveESModuleSymbol(immediate, moduleSpecifier, dontResolveAlias, /*suppressUsageError*/ false);
+            markSymbolOfAliasDeclarationIfTypeOnly(node, immediate, resolved, /*overwriteEmpty*/ false);
+            return resolved;
         }
-        // Return the type associated with a variable, parameter, or property declaration. In the simple case this is the type
-        // specified in a type annotation or inferred from an initializer. However, in the case of a destructuring declaration it
-        // is a bit more involved. For example:
+        function getTargetOfNamespaceExport(node, dontResolveAlias) {
+            var moduleSpecifier = node.parent.moduleSpecifier;
+            var immediate = moduleSpecifier && resolveExternalModuleName(node, moduleSpecifier);
+            var resolved = moduleSpecifier && resolveESModuleSymbol(immediate, moduleSpecifier, dontResolveAlias, /*suppressUsageError*/ false);
+            markSymbolOfAliasDeclarationIfTypeOnly(node, immediate, resolved, /*overwriteEmpty*/ false);
+            return resolved;
+        }
+        // This function creates a synthetic symbol that combines the value side of one symbol with the
+        // type/namespace side of another symbol. Consider this example:
         //
-        //   var [x, s = ""] = [1, "one"];
+        //   declare module graphics {
+        //       interface Point {
+        //           x: number;
+        //           y: number;
+        //       }
+        //   }
+        //   declare var graphics: {
+        //       Point: new (x: number, y: number) => graphics.Point;
+        //   }
+        //   declare module "graphics" {
+        //       export = graphics;
+        //   }
         //
-        // Here, the array literal [1, "one"] is contextually typed by the type [any, string], which is the implied type of the
-        // binding pattern [x, s = ""]. Because the contextual type is a tuple type, the resulting type of [1, "one"] is the
-        // tuple type [number, string]. Thus, the type inferred for 'x' is number and the type inferred for 's' is string.
-        function getWidenedTypeForVariableLikeDeclaration(declaration, reportErrors) {
-            return widenTypeForVariableLikeDeclaration(getTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ true), declaration, reportErrors);
-        }
-        function widenTypeForVariableLikeDeclaration(type, declaration, reportErrors) {
-            if (type) {
-                if (reportErrors) {
-                    reportErrorsFromWidening(declaration, type);
-                }
-                // always widen a 'unique symbol' type if the type was created for a different declaration.
-                if (type.flags & 8192 /* UniqueESSymbol */ && (ts.isBindingElement(declaration) || !declaration.type) && type.symbol !== getSymbolOfNode(declaration)) {
-                    type = esSymbolType;
-                }
-                return getWidenedType(type);
+        // An 'import { Point } from "graphics"' needs to create a symbol that combines the value side 'Point'
+        // property with the type/namespace side interface 'Point'.
+        function combineValueAndTypeSymbols(valueSymbol, typeSymbol) {
+            if (valueSymbol === unknownSymbol && typeSymbol === unknownSymbol) {
+                return unknownSymbol;
             }
-            // Rest parameters default to type any[], other parameters default to type any
-            type = ts.isParameter(declaration) && declaration.dotDotDotToken ? anyArrayType : anyType;
-            // Report implicit any errors unless this is a private property within an ambient declaration
-            if (reportErrors) {
-                if (!declarationBelongsToPrivateAmbientMember(declaration)) {
-                    reportImplicitAny(declaration, type);
-                }
+            if (valueSymbol.flags & (788968 /* Type */ | 1920 /* Namespace */)) {
+                return valueSymbol;
             }
-            return type;
-        }
-        function declarationBelongsToPrivateAmbientMember(declaration) {
-            var root = ts.getRootDeclaration(declaration);
-            var memberDeclaration = root.kind === 156 /* Parameter */ ? root.parent : root;
-            return isPrivateWithinAmbient(memberDeclaration);
+            var result = createSymbol(valueSymbol.flags | typeSymbol.flags, valueSymbol.escapedName);
+            result.declarations = ts.deduplicate(ts.concatenate(valueSymbol.declarations, typeSymbol.declarations), ts.equateValues);
+            result.parent = valueSymbol.parent || typeSymbol.parent;
+            if (valueSymbol.valueDeclaration)
+                result.valueDeclaration = valueSymbol.valueDeclaration;
+            if (typeSymbol.members)
+                result.members = new ts.Map(typeSymbol.members);
+            if (valueSymbol.exports)
+                result.exports = new ts.Map(valueSymbol.exports);
+            return result;
         }
-        function tryGetTypeFromEffectiveTypeNode(declaration) {
-            var typeNode = ts.getEffectiveTypeAnnotationNode(declaration);
-            if (typeNode) {
-                return getTypeFromTypeNode(typeNode);
+        function getExportOfModule(symbol, name, specifier, dontResolveAlias) {
+            if (symbol.flags & 1536 /* Module */) {
+                var exportSymbol = getExportsOfSymbol(symbol).get(name.escapedText);
+                var resolved = resolveSymbol(exportSymbol, dontResolveAlias);
+                markSymbolOfAliasDeclarationIfTypeOnly(specifier, exportSymbol, resolved, /*overwriteEmpty*/ false);
+                return resolved;
             }
         }
-        function getTypeOfVariableOrParameterOrProperty(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.type) {
-                var type = getTypeOfVariableOrParameterOrPropertyWorker(symbol);
-                // For a contextually typed parameter it is possible that a type has already
-                // been assigned (in assignTypeToParameterAndFixTypeParameters), and we want
-                // to preserve this type.
-                if (!links.type) {
-                    links.type = type;
+        function getPropertyOfVariable(symbol, name) {
+            if (symbol.flags & 3 /* Variable */) {
+                var typeAnnotation = symbol.valueDeclaration.type;
+                if (typeAnnotation) {
+                    return resolveSymbol(getPropertyOfType(getTypeFromTypeNode(typeAnnotation), name));
                 }
             }
-            return links.type;
         }
-        function getTypeOfVariableOrParameterOrPropertyWorker(symbol) {
-            // Handle prototype property
-            if (symbol.flags & 4194304 /* Prototype */) {
-                return getTypeOfPrototypeProperty(symbol);
-            }
-            // CommonsJS require and module both have type any.
-            if (symbol === requireSymbol) {
-                return anyType;
-            }
-            if (symbol.flags & 134217728 /* ModuleExports */) {
-                var fileSymbol = getSymbolOfNode(ts.getSourceFileOfNode(symbol.valueDeclaration));
-                var members = ts.createSymbolTable();
-                members.set("exports", fileSymbol);
-                return createAnonymousType(symbol, members, ts.emptyArray, ts.emptyArray, undefined, undefined);
-            }
-            // Handle catch clause variables
-            var declaration = symbol.valueDeclaration;
-            if (ts.isCatchClauseVariableDeclarationOrBindingElement(declaration)) {
-                return anyType;
-            }
-            // Handle export default expressions
-            if (ts.isSourceFile(declaration) && ts.isJsonSourceFile(declaration)) {
-                if (!declaration.statements.length) {
-                    return emptyObjectType;
-                }
-                return getWidenedType(getWidenedLiteralType(checkExpression(declaration.statements[0].expression)));
-            }
-            // Handle variable, parameter or property
-            if (!pushTypeResolution(symbol, 0 /* Type */)) {
-                // Symbol is property of some kind that is merged with something - should use `getTypeOfFuncClassEnumModule` and not `getTypeOfVariableOrParameterOrProperty`
-                if (symbol.flags & 512 /* ValueModule */ && !(symbol.flags & 67108864 /* Assignment */)) {
-                    return getTypeOfFuncClassEnumModule(symbol);
-                }
-                return reportCircularityError(symbol);
-            }
-            var type;
-            if (declaration.kind === 259 /* ExportAssignment */) {
-                type = widenTypeForVariableLikeDeclaration(checkExpressionCached(declaration.expression), declaration);
+        function getExternalModuleMember(node, specifier, dontResolveAlias) {
+            var _a;
+            if (dontResolveAlias === void 0) { dontResolveAlias = false; }
+            var moduleSpecifier = ts.getExternalModuleRequireArgument(node) || node.moduleSpecifier;
+            var moduleSymbol = resolveExternalModuleName(node, moduleSpecifier); // TODO: GH#18217
+            var name = !ts.isPropertyAccessExpression(specifier) && specifier.propertyName || specifier.name;
+            if (!ts.isIdentifier(name)) {
+                return undefined;
             }
-            else if (ts.isBinaryExpression(declaration) ||
-                (ts.isInJSFile(declaration) &&
-                    (ts.isCallExpression(declaration) || (ts.isPropertyAccessExpression(declaration) || ts.isBindableStaticElementAccessExpression(declaration)) && ts.isBinaryExpression(declaration.parent)))) {
-                type = getWidenedTypeForAssignmentDeclaration(symbol);
+            var suppressInteropError = name.escapedText === "default" /* Default */ && !!(compilerOptions.allowSyntheticDefaultImports || compilerOptions.esModuleInterop);
+            var targetSymbol = resolveESModuleSymbol(moduleSymbol, moduleSpecifier, dontResolveAlias, suppressInteropError);
+            if (targetSymbol) {
+                if (name.escapedText) {
+                    if (ts.isShorthandAmbientModuleSymbol(moduleSymbol)) {
+                        return moduleSymbol;
+                    }
+                    var symbolFromVariable = void 0;
+                    // First check if module was specified with "export=". If so, get the member from the resolved type
+                    if (moduleSymbol && moduleSymbol.exports && moduleSymbol.exports.get("export=" /* ExportEquals */)) {
+                        symbolFromVariable = getPropertyOfType(getTypeOfSymbol(targetSymbol), name.escapedText);
+                    }
+                    else {
+                        symbolFromVariable = getPropertyOfVariable(targetSymbol, name.escapedText);
+                    }
+                    // if symbolFromVariable is export - get its final target
+                    symbolFromVariable = resolveSymbol(symbolFromVariable, dontResolveAlias);
+                    var symbolFromModule = getExportOfModule(targetSymbol, name, specifier, dontResolveAlias);
+                    if (symbolFromModule === undefined && name.escapedText === "default" /* Default */) {
+                        var file = ts.find(moduleSymbol.declarations, ts.isSourceFile);
+                        if (canHaveSyntheticDefault(file, moduleSymbol, dontResolveAlias)) {
+                            symbolFromModule = resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias) || resolveSymbol(moduleSymbol, dontResolveAlias);
+                        }
+                    }
+                    var symbol = symbolFromModule && symbolFromVariable && symbolFromModule !== symbolFromVariable ?
+                        combineValueAndTypeSymbols(symbolFromVariable, symbolFromModule) :
+                        symbolFromModule || symbolFromVariable;
+                    if (!symbol) {
+                        var moduleName = getFullyQualifiedName(moduleSymbol, node);
+                        var declarationName = ts.declarationNameToString(name);
+                        var suggestion = getSuggestedSymbolForNonexistentModule(name, targetSymbol);
+                        if (suggestion !== undefined) {
+                            var suggestionName = symbolToString(suggestion);
+                            var diagnostic = error(name, ts.Diagnostics._0_has_no_exported_member_named_1_Did_you_mean_2, moduleName, declarationName, suggestionName);
+                            if (suggestion.valueDeclaration) {
+                                ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(suggestion.valueDeclaration, ts.Diagnostics._0_is_declared_here, suggestionName));
+                            }
+                        }
+                        else {
+                            if ((_a = moduleSymbol.exports) === null || _a === void 0 ? void 0 : _a.has("default" /* Default */)) {
+                                error(name, ts.Diagnostics.Module_0_has_no_exported_member_1_Did_you_mean_to_use_import_1_from_0_instead, moduleName, declarationName);
+                            }
+                            else {
+                                reportNonExportedMember(node, name, declarationName, moduleSymbol, moduleName);
+                            }
+                        }
+                    }
+                    return symbol;
+                }
             }
-            else if (ts.isJSDocPropertyLikeTag(declaration)
-                || ts.isPropertyAccessExpression(declaration)
-                || ts.isElementAccessExpression(declaration)
-                || ts.isIdentifier(declaration)
-                || ts.isStringLiteralLike(declaration)
-                || ts.isNumericLiteral(declaration)
-                || ts.isClassDeclaration(declaration)
-                || ts.isFunctionDeclaration(declaration)
-                || (ts.isMethodDeclaration(declaration) && !ts.isObjectLiteralMethod(declaration))
-                || ts.isMethodSignature(declaration)
-                || ts.isSourceFile(declaration)) {
-                // Symbol is property of some kind that is merged with something - should use `getTypeOfFuncClassEnumModule` and not `getTypeOfVariableOrParameterOrProperty`
-                if (symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 384 /* Enum */ | 512 /* ValueModule */)) {
-                    return getTypeOfFuncClassEnumModule(symbol);
+        }
+        function reportNonExportedMember(node, name, declarationName, moduleSymbol, moduleName) {
+            var _a;
+            var localSymbol = (_a = moduleSymbol.valueDeclaration.locals) === null || _a === void 0 ? void 0 : _a.get(name.escapedText);
+            var exports = moduleSymbol.exports;
+            if (localSymbol) {
+                var exportedEqualsSymbol = exports === null || exports === void 0 ? void 0 : exports.get("export=" /* ExportEquals */);
+                if (exportedEqualsSymbol) {
+                    getSymbolIfSameReference(exportedEqualsSymbol, localSymbol) ? reportInvalidImportEqualsExportMember(node, name, declarationName, moduleName) :
+                        error(name, ts.Diagnostics.Module_0_has_no_exported_member_1, moduleName, declarationName);
+                }
+                else {
+                    var exportedSymbol = exports ? ts.find(symbolsToArray(exports), function (symbol) { return !!getSymbolIfSameReference(symbol, localSymbol); }) : undefined;
+                    var diagnostic = exportedSymbol ? error(name, ts.Diagnostics.Module_0_declares_1_locally_but_it_is_exported_as_2, moduleName, declarationName, symbolToString(exportedSymbol)) :
+                        error(name, ts.Diagnostics.Module_0_declares_1_locally_but_it_is_not_exported, moduleName, declarationName);
+                    ts.addRelatedInfo.apply(void 0, __spreadArrays([diagnostic], ts.map(localSymbol.declarations, function (decl, index) {
+                        return ts.createDiagnosticForNode(decl, index === 0 ? ts.Diagnostics._0_is_declared_here : ts.Diagnostics.and_here, declarationName);
+                    })));
                 }
-                type = ts.isBinaryExpression(declaration.parent) ?
-                    getWidenedTypeForAssignmentDeclaration(symbol) :
-                    tryGetTypeFromEffectiveTypeNode(declaration) || anyType;
             }
-            else if (ts.isPropertyAssignment(declaration)) {
-                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkPropertyAssignment(declaration);
+            else {
+                error(name, ts.Diagnostics.Module_0_has_no_exported_member_1, moduleName, declarationName);
             }
-            else if (ts.isJsxAttribute(declaration)) {
-                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkJsxAttribute(declaration);
+        }
+        function reportInvalidImportEqualsExportMember(node, name, declarationName, moduleName) {
+            if (moduleKind >= ts.ModuleKind.ES2015) {
+                var message = compilerOptions.esModuleInterop ? ts.Diagnostics._0_can_only_be_imported_by_using_a_default_import :
+                    ts.Diagnostics._0_can_only_be_imported_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import;
+                error(name, message, declarationName);
             }
-            else if (ts.isShorthandPropertyAssignment(declaration)) {
-                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkExpressionForMutableLocation(declaration.name, 0 /* Normal */);
+            else {
+                if (ts.isInJSFile(node)) {
+                    var message = compilerOptions.esModuleInterop ? ts.Diagnostics._0_can_only_be_imported_by_using_a_require_call_or_by_using_a_default_import :
+                        ts.Diagnostics._0_can_only_be_imported_by_using_a_require_call_or_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import;
+                    error(name, message, declarationName);
+                }
+                else {
+                    var message = compilerOptions.esModuleInterop ? ts.Diagnostics._0_can_only_be_imported_by_using_import_1_require_2_or_a_default_import :
+                        ts.Diagnostics._0_can_only_be_imported_by_using_import_1_require_2_or_by_turning_on_the_esModuleInterop_flag_and_using_a_default_import;
+                    error(name, message, declarationName, declarationName, moduleName);
+                }
             }
-            else if (ts.isObjectLiteralMethod(declaration)) {
-                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkObjectLiteralMethod(declaration, 0 /* Normal */);
+        }
+        function getTargetOfImportSpecifier(node, dontResolveAlias) {
+            var root = ts.isBindingElement(node) ? ts.getRootDeclaration(node) : node.parent.parent.parent;
+            var commonJSPropertyAccess = getCommonJSPropertyAccess(root);
+            var resolved = getExternalModuleMember(root, commonJSPropertyAccess || node, dontResolveAlias);
+            var name = node.propertyName || node.name;
+            if (commonJSPropertyAccess && resolved && ts.isIdentifier(name)) {
+                return resolveSymbol(getPropertyOfType(getTypeOfSymbol(resolved), name.escapedText), dontResolveAlias);
             }
-            else if (ts.isParameter(declaration)
-                || ts.isPropertyDeclaration(declaration)
-                || ts.isPropertySignature(declaration)
-                || ts.isVariableDeclaration(declaration)
-                || ts.isBindingElement(declaration)) {
-                type = getWidenedTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ true);
+            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
+            return resolved;
+        }
+        function getCommonJSPropertyAccess(node) {
+            if (ts.isVariableDeclaration(node) && node.initializer && ts.isPropertyAccessExpression(node.initializer)) {
+                return node.initializer;
             }
-            // getTypeOfSymbol dispatches some JS merges incorrectly because their symbol flags are not mutually exclusive.
-            // Re-dispatch based on valueDeclaration.kind instead.
-            else if (ts.isEnumDeclaration(declaration)) {
-                type = getTypeOfFuncClassEnumModule(symbol);
+        }
+        function getTargetOfNamespaceExportDeclaration(node, dontResolveAlias) {
+            var resolved = resolveExternalModuleSymbol(node.parent.symbol, dontResolveAlias);
+            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
+            return resolved;
+        }
+        function getTargetOfExportSpecifier(node, meaning, dontResolveAlias) {
+            var resolved = node.parent.parent.moduleSpecifier ?
+                getExternalModuleMember(node.parent.parent, node, dontResolveAlias) :
+                resolveEntityName(node.propertyName || node.name, meaning, /*ignoreErrors*/ false, dontResolveAlias);
+            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
+            return resolved;
+        }
+        function getTargetOfExportAssignment(node, dontResolveAlias) {
+            var expression = ts.isExportAssignment(node) ? node.expression : node.right;
+            var resolved = getTargetOfAliasLikeExpression(expression, dontResolveAlias);
+            markSymbolOfAliasDeclarationIfTypeOnly(node, /*immediateTarget*/ undefined, resolved, /*overwriteEmpty*/ false);
+            return resolved;
+        }
+        function getTargetOfAliasLikeExpression(expression, dontResolveAlias) {
+            if (ts.isClassExpression(expression)) {
+                return checkExpressionCached(expression).symbol;
             }
-            else if (ts.isEnumMember(declaration)) {
-                type = getTypeOfEnumMember(symbol);
+            if (!ts.isEntityName(expression) && !ts.isEntityNameExpression(expression)) {
+                return undefined;
             }
-            else if (ts.isAccessor(declaration)) {
-                type = resolveTypeOfAccessors(symbol);
+            var aliasLike = resolveEntityName(expression, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, /*ignoreErrors*/ true, dontResolveAlias);
+            if (aliasLike) {
+                return aliasLike;
             }
-            else {
-                return ts.Debug.fail("Unhandled declaration kind! " + ts.Debug.formatSyntaxKind(declaration.kind) + " for " + ts.Debug.formatSymbol(symbol));
+            checkExpressionCached(expression);
+            return getNodeLinks(expression).resolvedSymbol;
+        }
+        function getTargetOfPropertyAssignment(node, dontRecursivelyResolve) {
+            var expression = node.initializer;
+            return getTargetOfAliasLikeExpression(expression, dontRecursivelyResolve);
+        }
+        function getTargetOfAccessExpression(node, dontRecursivelyResolve) {
+            if (!(ts.isBinaryExpression(node.parent) && node.parent.left === node && node.parent.operatorToken.kind === 62 /* EqualsToken */)) {
+                return undefined;
             }
-            if (!popTypeResolution()) {
-                // Symbol is property of some kind that is merged with something - should use `getTypeOfFuncClassEnumModule` and not `getTypeOfVariableOrParameterOrProperty`
-                if (symbol.flags & 512 /* ValueModule */ && !(symbol.flags & 67108864 /* Assignment */)) {
-                    return getTypeOfFuncClassEnumModule(symbol);
-                }
-                return reportCircularityError(symbol);
+            return getTargetOfAliasLikeExpression(node.parent.right, dontRecursivelyResolve);
+        }
+        function getTargetOfAliasDeclaration(node, dontRecursivelyResolve) {
+            if (dontRecursivelyResolve === void 0) { dontRecursivelyResolve = false; }
+            switch (node.kind) {
+                case 260 /* ImportEqualsDeclaration */:
+                case 249 /* VariableDeclaration */:
+                    return getTargetOfImportEqualsDeclaration(node, dontRecursivelyResolve);
+                case 262 /* ImportClause */:
+                    return getTargetOfImportClause(node, dontRecursivelyResolve);
+                case 263 /* NamespaceImport */:
+                    return getTargetOfNamespaceImport(node, dontRecursivelyResolve);
+                case 269 /* NamespaceExport */:
+                    return getTargetOfNamespaceExport(node, dontRecursivelyResolve);
+                case 265 /* ImportSpecifier */:
+                case 198 /* BindingElement */:
+                    return getTargetOfImportSpecifier(node, dontRecursivelyResolve);
+                case 270 /* ExportSpecifier */:
+                    return getTargetOfExportSpecifier(node, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, dontRecursivelyResolve);
+                case 266 /* ExportAssignment */:
+                case 216 /* BinaryExpression */:
+                    return getTargetOfExportAssignment(node, dontRecursivelyResolve);
+                case 259 /* NamespaceExportDeclaration */:
+                    return getTargetOfNamespaceExportDeclaration(node, dontRecursivelyResolve);
+                case 289 /* ShorthandPropertyAssignment */:
+                    return resolveEntityName(node.name, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, /*ignoreErrors*/ true, dontRecursivelyResolve);
+                case 288 /* PropertyAssignment */:
+                    return getTargetOfPropertyAssignment(node, dontRecursivelyResolve);
+                case 202 /* ElementAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
+                    return getTargetOfAccessExpression(node, dontRecursivelyResolve);
+                default:
+                    return ts.Debug.fail();
             }
-            return type;
         }
-        function getAnnotatedAccessorTypeNode(accessor) {
-            if (accessor) {
-                if (accessor.kind === 163 /* GetAccessor */) {
-                    var getterTypeAnnotation = ts.getEffectiveReturnTypeNode(accessor);
-                    return getterTypeAnnotation;
+        /**
+         * Indicates that a symbol is an alias that does not merge with a local declaration.
+         * OR Is a JSContainer which may merge an alias with a local declaration
+         */
+        function isNonLocalAlias(symbol, excludes) {
+            if (excludes === void 0) { excludes = 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */; }
+            if (!symbol)
+                return false;
+            return (symbol.flags & (2097152 /* Alias */ | excludes)) === 2097152 /* Alias */ || !!(symbol.flags & 2097152 /* Alias */ && symbol.flags & 67108864 /* Assignment */);
+        }
+        function resolveSymbol(symbol, dontResolveAlias) {
+            return !dontResolveAlias && isNonLocalAlias(symbol) ? resolveAlias(symbol) : symbol;
+        }
+        function resolveAlias(symbol) {
+            ts.Debug.assert((symbol.flags & 2097152 /* Alias */) !== 0, "Should only get Alias here.");
+            var links = getSymbolLinks(symbol);
+            if (!links.target) {
+                links.target = resolvingSymbol;
+                var node = getDeclarationOfAliasSymbol(symbol);
+                if (!node)
+                    return ts.Debug.fail();
+                var target = getTargetOfAliasDeclaration(node);
+                if (links.target === resolvingSymbol) {
+                    links.target = target || unknownSymbol;
                 }
                 else {
-                    var setterTypeAnnotation = ts.getEffectiveSetAccessorTypeAnnotationNode(accessor);
-                    return setterTypeAnnotation;
+                    error(node, ts.Diagnostics.Circular_definition_of_import_alias_0, symbolToString(symbol));
                 }
             }
-            return undefined;
+            else if (links.target === resolvingSymbol) {
+                links.target = unknownSymbol;
+            }
+            return links.target;
         }
-        function getAnnotatedAccessorType(accessor) {
-            var node = getAnnotatedAccessorTypeNode(accessor);
-            return node && getTypeFromTypeNode(node);
+        function tryResolveAlias(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (links.target !== resolvingSymbol) {
+                return resolveAlias(symbol);
+            }
+            return undefined;
         }
-        function getAnnotatedAccessorThisParameter(accessor) {
-            var parameter = getAccessorThisParameter(accessor);
-            return parameter && parameter.symbol;
+        /**
+         * Marks a symbol as type-only if its declaration is syntactically type-only.
+         * If it is not itself marked type-only, but resolves to a type-only alias
+         * somewhere in its resolution chain, save a reference to the type-only alias declaration
+         * so the alias _not_ marked type-only can be identified as _transitively_ type-only.
+         *
+         * This function is called on each alias declaration that could be type-only or resolve to
+         * another type-only alias during `resolveAlias`, so that later, when an alias is used in a
+         * JS-emitting expression, we can quickly determine if that symbol is effectively type-only
+         * and issue an error if so.
+         *
+         * @param aliasDeclaration The alias declaration not marked as type-only
+         * has already been marked as not resolving to a type-only alias. Used when recursively resolving qualified
+         * names of import aliases, e.g. `import C = a.b.C`. If namespace `a` is not found to be type-only, the
+         * import declaration will initially be marked as not resolving to a type-only symbol. But, namespace `b`
+         * must still be checked for a type-only marker, overwriting the previous negative result if found.
+         * @param immediateTarget The symbol to which the alias declaration immediately resolves
+         * @param finalTarget The symbol to which the alias declaration ultimately resolves
+         * @param overwriteEmpty Checks `resolvesToSymbol` for type-only declarations even if `aliasDeclaration`
+         */
+        function markSymbolOfAliasDeclarationIfTypeOnly(aliasDeclaration, immediateTarget, finalTarget, overwriteEmpty) {
+            if (!aliasDeclaration || ts.isPropertyAccessExpression(aliasDeclaration))
+                return false;
+            // If the declaration itself is type-only, mark it and return.
+            // No need to check what it resolves to.
+            var sourceSymbol = getSymbolOfNode(aliasDeclaration);
+            if (ts.isTypeOnlyImportOrExportDeclaration(aliasDeclaration)) {
+                var links_1 = getSymbolLinks(sourceSymbol);
+                links_1.typeOnlyDeclaration = aliasDeclaration;
+                return true;
+            }
+            var links = getSymbolLinks(sourceSymbol);
+            return markSymbolOfAliasDeclarationIfTypeOnlyWorker(links, immediateTarget, overwriteEmpty)
+                || markSymbolOfAliasDeclarationIfTypeOnlyWorker(links, finalTarget, overwriteEmpty);
         }
-        function getThisTypeOfDeclaration(declaration) {
-            return getThisTypeOfSignature(getSignatureFromDeclaration(declaration));
+        function markSymbolOfAliasDeclarationIfTypeOnlyWorker(aliasDeclarationLinks, target, overwriteEmpty) {
+            var _a, _b, _c;
+            if (target && (aliasDeclarationLinks.typeOnlyDeclaration === undefined || overwriteEmpty && aliasDeclarationLinks.typeOnlyDeclaration === false)) {
+                var exportSymbol = (_b = (_a = target.exports) === null || _a === void 0 ? void 0 : _a.get("export=" /* ExportEquals */)) !== null && _b !== void 0 ? _b : target;
+                var typeOnly = exportSymbol.declarations && ts.find(exportSymbol.declarations, ts.isTypeOnlyImportOrExportDeclaration);
+                aliasDeclarationLinks.typeOnlyDeclaration = (_c = typeOnly !== null && typeOnly !== void 0 ? typeOnly : getSymbolLinks(exportSymbol).typeOnlyDeclaration) !== null && _c !== void 0 ? _c : false;
+            }
+            return !!aliasDeclarationLinks.typeOnlyDeclaration;
         }
-        function getTypeOfAccessors(symbol) {
+        /** Indicates that a symbol directly or indirectly resolves to a type-only import or export. */
+        function getTypeOnlyAliasDeclaration(symbol) {
+            if (!(symbol.flags & 2097152 /* Alias */)) {
+                return undefined;
+            }
             var links = getSymbolLinks(symbol);
-            return links.type || (links.type = getTypeOfAccessorsWorker(symbol));
+            return links.typeOnlyDeclaration || undefined;
         }
-        function getTypeOfAccessorsWorker(symbol) {
-            if (!pushTypeResolution(symbol, 0 /* Type */)) {
-                return errorType;
-            }
-            var type = resolveTypeOfAccessors(symbol);
-            if (!popTypeResolution()) {
-                type = anyType;
-                if (noImplicitAny) {
-                    var getter = ts.getDeclarationOfKind(symbol, 163 /* GetAccessor */);
-                    error(getter, ts.Diagnostics._0_implicitly_has_return_type_any_because_it_does_not_have_a_return_type_annotation_and_is_referenced_directly_or_indirectly_in_one_of_its_return_expressions, symbolToString(symbol));
+        function markExportAsReferenced(node) {
+            var symbol = getSymbolOfNode(node);
+            var target = resolveAlias(symbol);
+            if (target) {
+                var markAlias = target === unknownSymbol ||
+                    ((target.flags & 111551 /* Value */) && !isConstEnumOrConstEnumOnlyModule(target) && !getTypeOnlyAliasDeclaration(symbol));
+                if (markAlias) {
+                    markAliasSymbolAsReferenced(symbol);
                 }
             }
-            return type;
         }
-        function resolveTypeOfAccessors(symbol) {
-            var getter = ts.getDeclarationOfKind(symbol, 163 /* GetAccessor */);
-            var setter = ts.getDeclarationOfKind(symbol, 164 /* SetAccessor */);
-            if (getter && ts.isInJSFile(getter)) {
-                var jsDocType = getTypeForDeclarationFromJSDocComment(getter);
-                if (jsDocType) {
-                    return jsDocType;
-                }
-            }
-            // First try to see if the user specified a return type on the get-accessor.
-            var getterReturnType = getAnnotatedAccessorType(getter);
-            if (getterReturnType) {
-                return getterReturnType;
-            }
-            else {
-                // If the user didn't specify a return type, try to use the set-accessor's parameter type.
-                var setterParameterType = getAnnotatedAccessorType(setter);
-                if (setterParameterType) {
-                    return setterParameterType;
-                }
-                else {
-                    // If there are no specified types, try to infer it from the body of the get accessor if it exists.
-                    if (getter && getter.body) {
-                        return getReturnTypeFromBody(getter);
-                    }
-                    // Otherwise, fall back to 'any'.
-                    else {
-                        if (setter) {
-                            if (!isPrivateWithinAmbient(setter)) {
-                                errorOrSuggestion(noImplicitAny, setter, ts.Diagnostics.Property_0_implicitly_has_type_any_because_its_set_accessor_lacks_a_parameter_type_annotation, symbolToString(symbol));
-                            }
-                        }
-                        else {
-                            ts.Debug.assert(!!getter, "there must exist a getter as we are current checking either setter or getter in this function");
-                            if (!isPrivateWithinAmbient(getter)) {
-                                errorOrSuggestion(noImplicitAny, getter, ts.Diagnostics.Property_0_implicitly_has_type_any_because_its_get_accessor_lacks_a_return_type_annotation, symbolToString(symbol));
-                            }
-                        }
-                        return anyType;
+        // When an alias symbol is referenced, we need to mark the entity it references as referenced and in turn repeat that until
+        // we reach a non-alias or an exported entity (which is always considered referenced). We do this by checking the target of
+        // the alias as an expression (which recursively takes us back here if the target references another alias).
+        function markAliasSymbolAsReferenced(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.referenced) {
+                links.referenced = true;
+                var node = getDeclarationOfAliasSymbol(symbol);
+                if (!node)
+                    return ts.Debug.fail();
+                // We defer checking of the reference of an `import =` until the import itself is referenced,
+                // This way a chain of imports can be elided if ultimately the final input is only used in a type
+                // position.
+                if (ts.isInternalModuleImportEqualsDeclaration(node)) {
+                    var target = resolveSymbol(symbol);
+                    if (target === unknownSymbol || target.flags & 111551 /* Value */) {
+                        // import foo = <symbol>
+                        checkExpressionCached(node.moduleReference);
                     }
                 }
             }
         }
-        function getBaseTypeVariableOfClass(symbol) {
-            var baseConstructorType = getBaseConstructorTypeOfClass(getDeclaredTypeOfClassOrInterface(symbol));
-            return baseConstructorType.flags & 8650752 /* TypeVariable */ ? baseConstructorType :
-                baseConstructorType.flags & 2097152 /* Intersection */ ? ts.find(baseConstructorType.types, function (t) { return !!(t.flags & 8650752 /* TypeVariable */); }) :
-                    undefined;
-        }
-        function getTypeOfFuncClassEnumModule(symbol) {
+        // Aliases that resolve to const enums are not marked as referenced because they are not emitted,
+        // but their usage in value positions must be tracked to determine if the import can be type-only.
+        function markConstEnumAliasAsReferenced(symbol) {
             var links = getSymbolLinks(symbol);
-            var originalLinks = links;
-            if (!links.type) {
-                var jsDeclaration = symbol.valueDeclaration && ts.getDeclarationOfExpando(symbol.valueDeclaration);
-                if (jsDeclaration) {
-                    var merged = mergeJSSymbols(symbol, getSymbolOfNode(jsDeclaration));
-                    if (merged) {
-                        // note:we overwrite links because we just cloned the symbol
-                        symbol = links = merged;
-                    }
-                }
-                originalLinks.type = links.type = getTypeOfFuncClassEnumModuleWorker(symbol);
+            if (!links.constEnumReferenced) {
+                links.constEnumReferenced = true;
             }
-            return links.type;
         }
-        function getTypeOfFuncClassEnumModuleWorker(symbol) {
-            var declaration = symbol.valueDeclaration;
-            if (symbol.flags & 1536 /* Module */ && ts.isShorthandAmbientModuleSymbol(symbol)) {
-                return anyType;
+        // This function is only for imports with entity names
+        function getSymbolOfPartOfRightHandSideOfImportEquals(entityName, dontResolveAlias) {
+            // There are three things we might try to look for. In the following examples,
+            // the search term is enclosed in |...|:
+            //
+            //     import a = |b|; // Namespace
+            //     import a = |b.c|; // Value, type, namespace
+            //     import a = |b.c|.d; // Namespace
+            if (entityName.kind === 78 /* Identifier */ && ts.isRightSideOfQualifiedNameOrPropertyAccess(entityName)) {
+                entityName = entityName.parent;
             }
-            else if (declaration && (declaration.kind === 209 /* BinaryExpression */ ||
-                ts.isAccessExpression(declaration) &&
-                    declaration.parent.kind === 209 /* BinaryExpression */)) {
-                return getWidenedTypeForAssignmentDeclaration(symbol);
+            // Check for case 1 and 3 in the above example
+            if (entityName.kind === 78 /* Identifier */ || entityName.parent.kind === 157 /* QualifiedName */) {
+                return resolveEntityName(entityName, 1920 /* Namespace */, /*ignoreErrors*/ false, dontResolveAlias);
             }
-            else if (symbol.flags & 512 /* ValueModule */ && declaration && ts.isSourceFile(declaration) && declaration.commonJsModuleIndicator) {
-                var resolvedModule = resolveExternalModuleSymbol(symbol);
-                if (resolvedModule !== symbol) {
-                    if (!pushTypeResolution(symbol, 0 /* Type */)) {
-                        return errorType;
+            else {
+                // Case 2 in above example
+                // entityName.kind could be a QualifiedName or a Missing identifier
+                ts.Debug.assert(entityName.parent.kind === 260 /* ImportEqualsDeclaration */);
+                return resolveEntityName(entityName, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, /*ignoreErrors*/ false, dontResolveAlias);
+            }
+        }
+        function getFullyQualifiedName(symbol, containingLocation) {
+            return symbol.parent ? getFullyQualifiedName(symbol.parent, containingLocation) + "." + symbolToString(symbol) : symbolToString(symbol, containingLocation, /*meaning*/ undefined, 16 /* DoNotIncludeSymbolChain */ | 4 /* AllowAnyNodeKind */);
+        }
+        /**
+         * Resolves a qualified name and any involved aliases.
+         */
+        function resolveEntityName(name, meaning, ignoreErrors, dontResolveAlias, location) {
+            if (ts.nodeIsMissing(name)) {
+                return undefined;
+            }
+            var namespaceMeaning = 1920 /* Namespace */ | (ts.isInJSFile(name) ? meaning & 111551 /* Value */ : 0);
+            var symbol;
+            if (name.kind === 78 /* Identifier */) {
+                var message = meaning === namespaceMeaning || ts.nodeIsSynthesized(name) ? ts.Diagnostics.Cannot_find_namespace_0 : getCannotFindNameDiagnosticForName(ts.getFirstIdentifier(name));
+                var symbolFromJSPrototype = ts.isInJSFile(name) && !ts.nodeIsSynthesized(name) ? resolveEntityNameFromAssignmentDeclaration(name, meaning) : undefined;
+                symbol = getMergedSymbol(resolveName(location || name, name.escapedText, meaning, ignoreErrors || symbolFromJSPrototype ? undefined : message, name, /*isUse*/ true));
+                if (!symbol) {
+                    return getMergedSymbol(symbolFromJSPrototype);
+                }
+            }
+            else if (name.kind === 157 /* QualifiedName */ || name.kind === 201 /* PropertyAccessExpression */) {
+                var left = name.kind === 157 /* QualifiedName */ ? name.left : name.expression;
+                var right = name.kind === 157 /* QualifiedName */ ? name.right : name.name;
+                var namespace = resolveEntityName(left, namespaceMeaning, ignoreErrors, /*dontResolveAlias*/ false, location);
+                if (!namespace || ts.nodeIsMissing(right)) {
+                    return undefined;
+                }
+                else if (namespace === unknownSymbol) {
+                    return namespace;
+                }
+                if (ts.isInJSFile(name)) {
+                    if (namespace.valueDeclaration &&
+                        ts.isVariableDeclaration(namespace.valueDeclaration) &&
+                        namespace.valueDeclaration.initializer &&
+                        isCommonJsRequire(namespace.valueDeclaration.initializer)) {
+                        var moduleName = namespace.valueDeclaration.initializer.arguments[0];
+                        var moduleSym = resolveExternalModuleName(moduleName, moduleName);
+                        if (moduleSym) {
+                            var resolvedModuleSymbol = resolveExternalModuleSymbol(moduleSym);
+                            if (resolvedModuleSymbol) {
+                                namespace = resolvedModuleSymbol;
+                            }
+                        }
                     }
-                    var exportEquals = getMergedSymbol(symbol.exports.get("export=" /* ExportEquals */));
-                    var type_1 = getWidenedTypeForAssignmentDeclaration(exportEquals, exportEquals === resolvedModule ? undefined : resolvedModule);
-                    if (!popTypeResolution()) {
-                        return reportCircularityError(symbol);
+                }
+                symbol = getMergedSymbol(getSymbol(getExportsOfSymbol(namespace), right.escapedText, meaning));
+                if (!symbol) {
+                    if (!ignoreErrors) {
+                        var namespaceName = getFullyQualifiedName(namespace);
+                        var declarationName = ts.declarationNameToString(right);
+                        var suggestion = getSuggestedSymbolForNonexistentModule(right, namespace);
+                        suggestion ?
+                            error(right, ts.Diagnostics._0_has_no_exported_member_named_1_Did_you_mean_2, namespaceName, declarationName, symbolToString(suggestion)) :
+                            error(right, ts.Diagnostics.Namespace_0_has_no_exported_member_1, namespaceName, declarationName);
                     }
-                    return type_1;
+                    return undefined;
                 }
             }
-            var type = createObjectType(16 /* Anonymous */, symbol);
-            if (symbol.flags & 32 /* Class */) {
-                var baseTypeVariable = getBaseTypeVariableOfClass(symbol);
-                return baseTypeVariable ? getIntersectionType([type, baseTypeVariable]) : type;
-            }
             else {
-                return strictNullChecks && symbol.flags & 16777216 /* Optional */ ? getOptionalType(type) : type;
+                throw ts.Debug.assertNever(name, "Unknown entity name kind.");
             }
-        }
-        function getTypeOfEnumMember(symbol) {
-            var links = getSymbolLinks(symbol);
-            return links.type || (links.type = getDeclaredTypeOfEnumMember(symbol));
-        }
-        function getTypeOfAlias(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.type) {
-                var targetSymbol = resolveAlias(symbol);
-                // It only makes sense to get the type of a value symbol. If the result of resolving
-                // the alias is not a value, then it has no type. To get the type associated with a
-                // type symbol, call getDeclaredTypeOfSymbol.
-                // This check is important because without it, a call to getTypeOfSymbol could end
-                // up recursively calling getTypeOfAlias, causing a stack overflow.
-                links.type = targetSymbol.flags & 111551 /* Value */
-                    ? getTypeOfSymbol(targetSymbol)
-                    : errorType;
+            ts.Debug.assert((ts.getCheckFlags(symbol) & 1 /* Instantiated */) === 0, "Should never get an instantiated symbol here.");
+            if (!ts.nodeIsSynthesized(name) && ts.isEntityName(name) && (symbol.flags & 2097152 /* Alias */ || name.parent.kind === 266 /* ExportAssignment */)) {
+                markSymbolOfAliasDeclarationIfTypeOnly(ts.getAliasDeclarationFromName(name), symbol, /*finalTarget*/ undefined, /*overwriteEmpty*/ true);
             }
-            return links.type;
+            return (symbol.flags & meaning) || dontResolveAlias ? symbol : resolveAlias(symbol);
         }
-        function getTypeOfInstantiatedSymbol(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.type) {
-                if (!pushTypeResolution(symbol, 0 /* Type */)) {
-                    return links.type = errorType;
-                }
-                var type = instantiateType(getTypeOfSymbol(links.target), links.mapper);
-                if (!popTypeResolution()) {
-                    type = reportCircularityError(symbol);
+        /**
+         * 1. For prototype-property methods like `A.prototype.m = function () ...`, try to resolve names in the scope of `A` too.
+         * Note that prototype-property assignment to locations outside the current file (eg globals) doesn't work, so
+         * name resolution won't work either.
+         * 2. For property assignments like `{ x: function f () { } }`, try to resolve names in the scope of `f` too.
+         */
+        function resolveEntityNameFromAssignmentDeclaration(name, meaning) {
+            if (isJSDocTypeReference(name.parent)) {
+                var secondaryLocation = getAssignmentDeclarationLocation(name.parent);
+                if (secondaryLocation) {
+                    return resolveName(secondaryLocation, name.escapedText, meaning, /*nameNotFoundMessage*/ undefined, name, /*isUse*/ true);
                 }
-                links.type = type;
             }
-            return links.type;
         }
-        function reportCircularityError(symbol) {
-            var declaration = symbol.valueDeclaration;
-            // Check if variable has type annotation that circularly references the variable itself
-            if (ts.getEffectiveTypeAnnotationNode(declaration)) {
-                error(symbol.valueDeclaration, ts.Diagnostics._0_is_referenced_directly_or_indirectly_in_its_own_type_annotation, symbolToString(symbol));
-                return errorType;
+        function getAssignmentDeclarationLocation(node) {
+            var typeAlias = ts.findAncestor(node, function (node) { return !(ts.isJSDocNode(node) || node.flags & 4194304 /* JSDoc */) ? "quit" : ts.isJSDocTypeAlias(node); });
+            if (typeAlias) {
+                return;
             }
-            // Check if variable has initializer that circularly references the variable itself
-            if (noImplicitAny && (declaration.kind !== 156 /* Parameter */ || declaration.initializer)) {
-                error(symbol.valueDeclaration, ts.Diagnostics._0_implicitly_has_type_any_because_it_does_not_have_a_type_annotation_and_is_referenced_directly_or_indirectly_in_its_own_initializer, symbolToString(symbol));
+            var host = ts.getJSDocHost(node);
+            if (ts.isExpressionStatement(host) &&
+                ts.isBinaryExpression(host.expression) &&
+                ts.getAssignmentDeclarationKind(host.expression) === 3 /* PrototypeProperty */) {
+                // X.prototype.m = /** @param {K} p */ function () { } <-- look for K on X's declaration
+                var symbol = getSymbolOfNode(host.expression.left);
+                if (symbol) {
+                    return getDeclarationOfJSPrototypeContainer(symbol);
+                }
             }
-            // Circularities could also result from parameters in function expressions that end up
-            // having themselves as contextual types following type argument inference. In those cases
-            // we have already reported an implicit any error so we don't report anything here.
-            return anyType;
-        }
-        function getTypeOfSymbolWithDeferredType(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.type) {
-                ts.Debug.assertIsDefined(links.deferralParent);
-                ts.Debug.assertIsDefined(links.deferralConstituents);
-                links.type = links.deferralParent.flags & 1048576 /* Union */ ? getUnionType(links.deferralConstituents) : getIntersectionType(links.deferralConstituents);
+            if ((ts.isObjectLiteralMethod(host) || ts.isPropertyAssignment(host)) &&
+                ts.isBinaryExpression(host.parent.parent) &&
+                ts.getAssignmentDeclarationKind(host.parent.parent) === 6 /* Prototype */) {
+                // X.prototype = { /** @param {K} p */m() { } } <-- look for K on X's declaration
+                var symbol = getSymbolOfNode(host.parent.parent.left);
+                if (symbol) {
+                    return getDeclarationOfJSPrototypeContainer(symbol);
+                }
+            }
+            var sig = ts.getEffectiveJSDocHost(node);
+            if (sig && ts.isFunctionLike(sig)) {
+                var symbol = getSymbolOfNode(sig);
+                return symbol && symbol.valueDeclaration;
             }
-            return links.type;
         }
-        function getTypeOfSymbol(symbol) {
-            var checkFlags = ts.getCheckFlags(symbol);
-            if (checkFlags & 65536 /* DeferredType */) {
-                return getTypeOfSymbolWithDeferredType(symbol);
+        function getDeclarationOfJSPrototypeContainer(symbol) {
+            var decl = symbol.parent.valueDeclaration;
+            if (!decl) {
+                return undefined;
             }
-            if (checkFlags & 1 /* Instantiated */) {
-                return getTypeOfInstantiatedSymbol(symbol);
+            var initializer = ts.isAssignmentDeclaration(decl) ? ts.getAssignedExpandoInitializer(decl) :
+                ts.hasOnlyExpressionInitializer(decl) ? ts.getDeclaredExpandoInitializer(decl) :
+                    undefined;
+            return initializer || decl;
+        }
+        /**
+         * Get the real symbol of a declaration with an expando initializer.
+         *
+         * Normally, declarations have an associated symbol, but when a declaration has an expando
+         * initializer, the expando's symbol is the one that has all the members merged into it.
+         */
+        function getExpandoSymbol(symbol) {
+            var decl = symbol.valueDeclaration;
+            if (!decl || !ts.isInJSFile(decl) || symbol.flags & 524288 /* TypeAlias */ || ts.getExpandoInitializer(decl, /*isPrototypeAssignment*/ false)) {
+                return undefined;
             }
-            if (checkFlags & 262144 /* Mapped */) {
-                return getTypeOfMappedSymbol(symbol);
+            var init = ts.isVariableDeclaration(decl) ? ts.getDeclaredExpandoInitializer(decl) : ts.getAssignedExpandoInitializer(decl);
+            if (init) {
+                var initSymbol = getSymbolOfNode(init);
+                if (initSymbol) {
+                    return mergeJSSymbols(initSymbol, symbol);
+                }
             }
-            if (checkFlags & 8192 /* ReverseMapped */) {
-                return getTypeOfReverseMappedSymbol(symbol);
+        }
+        function resolveExternalModuleName(location, moduleReferenceExpression, ignoreErrors) {
+            var isClassic = ts.getEmitModuleResolutionKind(compilerOptions) === ts.ModuleResolutionKind.Classic;
+            var errorMessage = isClassic ?
+                ts.Diagnostics.Cannot_find_module_0_Did_you_mean_to_set_the_moduleResolution_option_to_node_or_to_add_aliases_to_the_paths_option
+                : ts.Diagnostics.Cannot_find_module_0_or_its_corresponding_type_declarations;
+            return resolveExternalModuleNameWorker(location, moduleReferenceExpression, ignoreErrors ? undefined : errorMessage);
+        }
+        function resolveExternalModuleNameWorker(location, moduleReferenceExpression, moduleNotFoundError, isForAugmentation) {
+            if (isForAugmentation === void 0) { isForAugmentation = false; }
+            return ts.isStringLiteralLike(moduleReferenceExpression)
+                ? resolveExternalModule(location, moduleReferenceExpression.text, moduleNotFoundError, moduleReferenceExpression, isForAugmentation)
+                : undefined;
+        }
+        function resolveExternalModule(location, moduleReference, moduleNotFoundError, errorNode, isForAugmentation) {
+            if (isForAugmentation === void 0) { isForAugmentation = false; }
+            if (ts.startsWith(moduleReference, "@types/")) {
+                var diag = ts.Diagnostics.Cannot_import_type_declaration_files_Consider_importing_0_instead_of_1;
+                var withoutAtTypePrefix = ts.removePrefix(moduleReference, "@types/");
+                error(errorNode, diag, withoutAtTypePrefix, moduleReference);
             }
-            if (symbol.flags & (3 /* Variable */ | 4 /* Property */)) {
-                return getTypeOfVariableOrParameterOrProperty(symbol);
+            var ambientModule = tryFindAmbientModule(moduleReference, /*withAugmentations*/ true);
+            if (ambientModule) {
+                return ambientModule;
             }
-            if (symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 384 /* Enum */ | 512 /* ValueModule */)) {
-                return getTypeOfFuncClassEnumModule(symbol);
+            var currentSourceFile = ts.getSourceFileOfNode(location);
+            var resolvedModule = ts.getResolvedModule(currentSourceFile, moduleReference); // TODO: GH#18217
+            var resolutionDiagnostic = resolvedModule && ts.getResolutionDiagnostic(compilerOptions, resolvedModule);
+            var sourceFile = resolvedModule && !resolutionDiagnostic && host.getSourceFile(resolvedModule.resolvedFileName);
+            if (sourceFile) {
+                if (sourceFile.symbol) {
+                    if (resolvedModule.isExternalLibraryImport && !ts.resolutionExtensionIsTSOrJson(resolvedModule.extension)) {
+                        errorOnImplicitAnyModule(/*isError*/ false, errorNode, resolvedModule, moduleReference);
+                    }
+                    // merged symbol is module declaration symbol combined with all augmentations
+                    return getMergedSymbol(sourceFile.symbol);
+                }
+                if (moduleNotFoundError) {
+                    // report errors only if it was requested
+                    error(errorNode, ts.Diagnostics.File_0_is_not_a_module, sourceFile.fileName);
+                }
+                return undefined;
             }
-            if (symbol.flags & 8 /* EnumMember */) {
-                return getTypeOfEnumMember(symbol);
+            if (patternAmbientModules) {
+                var pattern = ts.findBestPatternMatch(patternAmbientModules, function (_) { return _.pattern; }, moduleReference);
+                if (pattern) {
+                    // If the module reference matched a pattern ambient module ('*.foo') but there's also a
+                    // module augmentation by the specific name requested ('a.foo'), we store the merged symbol
+                    // by the augmentation name ('a.foo'), because asking for *.foo should not give you exports
+                    // from a.foo.
+                    var augmentation = patternAmbientModuleAugmentations && patternAmbientModuleAugmentations.get(moduleReference);
+                    if (augmentation) {
+                        return getMergedSymbol(augmentation);
+                    }
+                    return getMergedSymbol(pattern.symbol);
+                }
             }
-            if (symbol.flags & 98304 /* Accessor */) {
-                return getTypeOfAccessors(symbol);
+            // May be an untyped module. If so, ignore resolutionDiagnostic.
+            if (resolvedModule && !ts.resolutionExtensionIsTSOrJson(resolvedModule.extension) && resolutionDiagnostic === undefined || resolutionDiagnostic === ts.Diagnostics.Could_not_find_a_declaration_file_for_module_0_1_implicitly_has_an_any_type) {
+                if (isForAugmentation) {
+                    var diag = ts.Diagnostics.Invalid_module_name_in_augmentation_Module_0_resolves_to_an_untyped_module_at_1_which_cannot_be_augmented;
+                    error(errorNode, diag, moduleReference, resolvedModule.resolvedFileName);
+                }
+                else {
+                    errorOnImplicitAnyModule(/*isError*/ noImplicitAny && !!moduleNotFoundError, errorNode, resolvedModule, moduleReference);
+                }
+                // Failed imports and untyped modules are both treated in an untyped manner; only difference is whether we give a diagnostic first.
+                return undefined;
             }
-            if (symbol.flags & 2097152 /* Alias */) {
-                return getTypeOfAlias(symbol);
+            if (moduleNotFoundError) {
+                // See if this was possibly a projectReference redirect
+                if (resolvedModule) {
+                    var redirect = host.getProjectReferenceRedirect(resolvedModule.resolvedFileName);
+                    if (redirect) {
+                        error(errorNode, ts.Diagnostics.Output_file_0_has_not_been_built_from_source_file_1, redirect, resolvedModule.resolvedFileName);
+                        return undefined;
+                    }
+                }
+                if (resolutionDiagnostic) {
+                    error(errorNode, resolutionDiagnostic, moduleReference, resolvedModule.resolvedFileName);
+                }
+                else {
+                    var tsExtension = ts.tryExtractTSExtension(moduleReference);
+                    if (tsExtension) {
+                        var diag = ts.Diagnostics.An_import_path_cannot_end_with_a_0_extension_Consider_importing_1_instead;
+                        error(errorNode, diag, tsExtension, ts.removeExtension(moduleReference, tsExtension));
+                    }
+                    else if (!compilerOptions.resolveJsonModule &&
+                        ts.fileExtensionIs(moduleReference, ".json" /* Json */) &&
+                        ts.getEmitModuleResolutionKind(compilerOptions) === ts.ModuleResolutionKind.NodeJs &&
+                        ts.hasJsonModuleEmitEnabled(compilerOptions)) {
+                        error(errorNode, ts.Diagnostics.Cannot_find_module_0_Consider_using_resolveJsonModule_to_import_module_with_json_extension, moduleReference);
+                    }
+                    else {
+                        error(errorNode, moduleNotFoundError, moduleReference);
+                    }
+                }
             }
-            return errorType;
+            return undefined;
         }
-        function isReferenceToType(type, target) {
-            return type !== undefined
-                && target !== undefined
-                && (ts.getObjectFlags(type) & 4 /* Reference */) !== 0
-                && type.target === target;
+        function errorOnImplicitAnyModule(isError, errorNode, _a, moduleReference) {
+            var packageId = _a.packageId, resolvedFileName = _a.resolvedFileName;
+            var errorInfo = !ts.isExternalModuleNameRelative(moduleReference) && packageId
+                ? typesPackageExists(packageId.name)
+                    ? ts.chainDiagnosticMessages(
+                    /*details*/ undefined, ts.Diagnostics.If_the_0_package_actually_exposes_this_module_consider_sending_a_pull_request_to_amend_https_Colon_Slash_Slashgithub_com_SlashDefinitelyTyped_SlashDefinitelyTyped_Slashtree_Slashmaster_Slashtypes_Slash_1, packageId.name, ts.mangleScopedPackageName(packageId.name))
+                    : ts.chainDiagnosticMessages(
+                    /*details*/ undefined, ts.Diagnostics.Try_npm_i_save_dev_types_Slash_1_if_it_exists_or_add_a_new_declaration_d_ts_file_containing_declare_module_0, moduleReference, ts.mangleScopedPackageName(packageId.name))
+                : undefined;
+            errorOrSuggestion(isError, errorNode, ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Could_not_find_a_declaration_file_for_module_0_1_implicitly_has_an_any_type, moduleReference, resolvedFileName));
         }
-        function getTargetType(type) {
-            return ts.getObjectFlags(type) & 4 /* Reference */ ? type.target : type;
+        function typesPackageExists(packageName) {
+            return getPackagesSet().has(ts.getTypesPackageName(packageName));
         }
-        // TODO: GH#18217 If `checkBase` is undefined, we should not call this because this will always return false.
-        function hasBaseType(type, checkBase) {
-            return check(type);
-            function check(type) {
-                if (ts.getObjectFlags(type) & (3 /* ClassOrInterface */ | 4 /* Reference */)) {
-                    var target = getTargetType(type);
-                    return target === checkBase || ts.some(getBaseTypes(target), check);
-                }
-                else if (type.flags & 2097152 /* Intersection */) {
-                    return ts.some(type.types, check);
-                }
-                return false;
+        function resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias) {
+            if (moduleSymbol === null || moduleSymbol === void 0 ? void 0 : moduleSymbol.exports) {
+                var exportEquals = resolveSymbol(moduleSymbol.exports.get("export=" /* ExportEquals */), dontResolveAlias);
+                var exported = getCommonJsExportEquals(getMergedSymbol(exportEquals), getMergedSymbol(moduleSymbol));
+                return getMergedSymbol(exported) || moduleSymbol;
             }
+            return undefined;
         }
-        // Appends the type parameters given by a list of declarations to a set of type parameters and returns the resulting set.
-        // The function allocates a new array if the input type parameter set is undefined, but otherwise it modifies the set
-        // in-place and returns the same array.
-        function appendTypeParameters(typeParameters, declarations) {
-            for (var _i = 0, declarations_2 = declarations; _i < declarations_2.length; _i++) {
-                var declaration = declarations_2[_i];
-                typeParameters = ts.appendIfUnique(typeParameters, getDeclaredTypeOfTypeParameter(getSymbolOfNode(declaration)));
+        function getCommonJsExportEquals(exported, moduleSymbol) {
+            if (!exported || exported === unknownSymbol || exported === moduleSymbol || moduleSymbol.exports.size === 1 || exported.flags & 2097152 /* Alias */) {
+                return exported;
             }
-            return typeParameters;
+            var links = getSymbolLinks(exported);
+            if (links.cjsExportMerged) {
+                return links.cjsExportMerged;
+            }
+            var merged = exported.flags & 33554432 /* Transient */ ? exported : cloneSymbol(exported);
+            merged.flags = merged.flags | 512 /* ValueModule */;
+            if (merged.exports === undefined) {
+                merged.exports = ts.createSymbolTable();
+            }
+            moduleSymbol.exports.forEach(function (s, name) {
+                if (name === "export=" /* ExportEquals */)
+                    return;
+                merged.exports.set(name, merged.exports.has(name) ? mergeSymbol(merged.exports.get(name), s) : s);
+            });
+            getSymbolLinks(merged).cjsExportMerged = merged;
+            return links.cjsExportMerged = merged;
         }
-        // Return the outer type parameters of a node or undefined if the node has no outer type parameters.
-        function getOuterTypeParameters(node, includeThisTypes) {
-            while (true) {
-                node = node.parent; // TODO: GH#18217 Use SourceFile kind check instead
-                if (node && ts.isBinaryExpression(node)) {
-                    // prototype assignments get the outer type parameters of their constructor function
-                    var assignmentKind = ts.getAssignmentDeclarationKind(node);
-                    if (assignmentKind === 6 /* Prototype */ || assignmentKind === 3 /* PrototypeProperty */) {
-                        var symbol = getSymbolOfNode(node.left);
-                        if (symbol && symbol.parent && !ts.findAncestor(symbol.parent.valueDeclaration, function (d) { return node === d; })) {
-                            node = symbol.parent.valueDeclaration;
-                        }
-                    }
-                }
-                if (!node) {
-                    return undefined;
+        // An external module with an 'export =' declaration may be referenced as an ES6 module provided the 'export ='
+        // references a symbol that is at least declared as a module or a variable. The target of the 'export =' may
+        // combine other declarations with the module or variable (e.g. a class/module, function/module, interface/variable).
+        function resolveESModuleSymbol(moduleSymbol, referencingLocation, dontResolveAlias, suppressInteropError) {
+            var symbol = resolveExternalModuleSymbol(moduleSymbol, dontResolveAlias);
+            if (!dontResolveAlias && symbol) {
+                if (!suppressInteropError && !(symbol.flags & (1536 /* Module */ | 3 /* Variable */)) && !ts.getDeclarationOfKind(symbol, 297 /* SourceFile */)) {
+                    var compilerOptionName = moduleKind >= ts.ModuleKind.ES2015
+                        ? "allowSyntheticDefaultImports"
+                        : "esModuleInterop";
+                    error(referencingLocation, ts.Diagnostics.This_module_can_only_be_referenced_with_ECMAScript_imports_Slashexports_by_turning_on_the_0_flag_and_referencing_its_default_export, compilerOptionName);
+                    return symbol;
                 }
-                switch (node.kind) {
-                    case 225 /* VariableStatement */:
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 165 /* CallSignature */:
-                    case 166 /* ConstructSignature */:
-                    case 160 /* MethodSignature */:
-                    case 170 /* FunctionType */:
-                    case 171 /* ConstructorType */:
-                    case 300 /* JSDocFunctionType */:
-                    case 244 /* FunctionDeclaration */:
-                    case 161 /* MethodDeclaration */:
-                    case 201 /* FunctionExpression */:
-                    case 202 /* ArrowFunction */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 321 /* JSDocTemplateTag */:
-                    case 322 /* JSDocTypedefTag */:
-                    case 316 /* JSDocEnumTag */:
-                    case 315 /* JSDocCallbackTag */:
-                    case 186 /* MappedType */:
-                    case 180 /* ConditionalType */:
-                        var outerTypeParameters = getOuterTypeParameters(node, includeThisTypes);
-                        if (node.kind === 186 /* MappedType */) {
-                            return ts.append(outerTypeParameters, getDeclaredTypeOfTypeParameter(getSymbolOfNode(node.typeParameter)));
-                        }
-                        else if (node.kind === 180 /* ConditionalType */) {
-                            return ts.concatenate(outerTypeParameters, getInferTypeParameters(node));
+                if (compilerOptions.esModuleInterop) {
+                    var referenceParent = referencingLocation.parent;
+                    if ((ts.isImportDeclaration(referenceParent) && ts.getNamespaceDeclarationNode(referenceParent)) ||
+                        ts.isImportCall(referenceParent)) {
+                        var type = getTypeOfSymbol(symbol);
+                        var sigs = getSignaturesOfStructuredType(type, 0 /* Call */);
+                        if (!sigs || !sigs.length) {
+                            sigs = getSignaturesOfStructuredType(type, 1 /* Construct */);
                         }
-                        else if (node.kind === 225 /* VariableStatement */ && !ts.isInJSFile(node)) {
-                            break;
+                        if (sigs && sigs.length) {
+                            var moduleType = getTypeWithSyntheticDefaultImportType(type, symbol, moduleSymbol);
+                            // Create a new symbol which has the module's type less the call and construct signatures
+                            var result = createSymbol(symbol.flags, symbol.escapedName);
+                            result.declarations = symbol.declarations ? symbol.declarations.slice() : [];
+                            result.parent = symbol.parent;
+                            result.target = symbol;
+                            result.originatingImport = referenceParent;
+                            if (symbol.valueDeclaration)
+                                result.valueDeclaration = symbol.valueDeclaration;
+                            if (symbol.constEnumOnlyModule)
+                                result.constEnumOnlyModule = true;
+                            if (symbol.members)
+                                result.members = new ts.Map(symbol.members);
+                            if (symbol.exports)
+                                result.exports = new ts.Map(symbol.exports);
+                            var resolvedModuleType = resolveStructuredTypeMembers(moduleType); // Should already be resolved from the signature checks above
+                            result.type = createAnonymousType(result, resolvedModuleType.members, ts.emptyArray, ts.emptyArray, resolvedModuleType.stringIndexInfo, resolvedModuleType.numberIndexInfo);
+                            return result;
                         }
-                        var outerAndOwnTypeParameters = appendTypeParameters(outerTypeParameters, ts.getEffectiveTypeParameterDeclarations(node));
-                        var thisType = includeThisTypes &&
-                            (node.kind === 245 /* ClassDeclaration */ || node.kind === 214 /* ClassExpression */ || node.kind === 246 /* InterfaceDeclaration */ || isJSConstructor(node)) &&
-                            getDeclaredTypeOfClassOrInterface(getSymbolOfNode(node)).thisType;
-                        return thisType ? ts.append(outerAndOwnTypeParameters, thisType) : outerAndOwnTypeParameters;
+                    }
                 }
             }
+            return symbol;
         }
-        // The outer type parameters are those defined by enclosing generic classes, methods, or functions.
-        function getOuterTypeParametersOfClassOrInterface(symbol) {
-            var declaration = symbol.flags & 32 /* Class */ ? symbol.valueDeclaration : ts.getDeclarationOfKind(symbol, 246 /* InterfaceDeclaration */);
-            ts.Debug.assert(!!declaration, "Class was missing valueDeclaration -OR- non-class had no interface declarations");
-            return getOuterTypeParameters(declaration);
+        function hasExportAssignmentSymbol(moduleSymbol) {
+            return moduleSymbol.exports.get("export=" /* ExportEquals */) !== undefined;
         }
-        // The local type parameters are the combined set of type parameters from all declarations of the class,
-        // interface, or type alias.
-        function getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol) {
-            var result;
-            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                var node = _a[_i];
-                if (node.kind === 246 /* InterfaceDeclaration */ ||
-                    node.kind === 245 /* ClassDeclaration */ ||
-                    node.kind === 214 /* ClassExpression */ ||
-                    isJSConstructor(node) ||
-                    ts.isTypeAlias(node)) {
-                    var declaration = node;
-                    result = appendTypeParameters(result, ts.getEffectiveTypeParameterDeclarations(declaration));
-                }
-            }
-            return result;
+        function getExportsOfModuleAsArray(moduleSymbol) {
+            return symbolsToArray(getExportsOfModule(moduleSymbol));
         }
-        // The full set of type parameters for a generic class or interface type consists of its outer type parameters plus
-        // its locally declared type parameters.
-        function getTypeParametersOfClassOrInterface(symbol) {
-            return ts.concatenate(getOuterTypeParametersOfClassOrInterface(symbol), getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol));
+        function getExportsAndPropertiesOfModule(moduleSymbol) {
+            var exports = getExportsOfModuleAsArray(moduleSymbol);
+            var exportEquals = resolveExternalModuleSymbol(moduleSymbol);
+            if (exportEquals !== moduleSymbol) {
+                ts.addRange(exports, getPropertiesOfType(getTypeOfSymbol(exportEquals)));
+            }
+            return exports;
         }
-        // A type is a mixin constructor if it has a single construct signature taking no type parameters and a single
-        // rest parameter of type any[].
-        function isMixinConstructorType(type) {
-            var signatures = getSignaturesOfType(type, 1 /* Construct */);
-            if (signatures.length === 1) {
-                var s = signatures[0];
-                return !s.typeParameters && s.parameters.length === 1 && signatureHasRestParameter(s) && getElementTypeOfArrayType(getTypeOfParameter(s.parameters[0])) === anyType;
+        function tryGetMemberInModuleExports(memberName, moduleSymbol) {
+            var symbolTable = getExportsOfModule(moduleSymbol);
+            if (symbolTable) {
+                return symbolTable.get(memberName);
             }
-            return false;
         }
-        function isConstructorType(type) {
-            if (getSignaturesOfType(type, 1 /* Construct */).length > 0) {
-                return true;
+        function tryGetMemberInModuleExportsAndProperties(memberName, moduleSymbol) {
+            var symbol = tryGetMemberInModuleExports(memberName, moduleSymbol);
+            if (symbol) {
+                return symbol;
             }
-            if (type.flags & 8650752 /* TypeVariable */) {
-                var constraint = getBaseConstraintOfType(type);
-                return !!constraint && isMixinConstructorType(constraint);
+            var exportEquals = resolveExternalModuleSymbol(moduleSymbol);
+            if (exportEquals === moduleSymbol) {
+                return undefined;
             }
-            return false;
-        }
-        function getBaseTypeNodeOfClass(type) {
-            return ts.getEffectiveBaseTypeNode(type.symbol.valueDeclaration);
+            var type = getTypeOfSymbol(exportEquals);
+            return type.flags & 131068 /* Primitive */ ||
+                ts.getObjectFlags(type) & 1 /* Class */ ||
+                isArrayOrTupleLikeType(type)
+                ? undefined
+                : getPropertyOfType(type, memberName);
         }
-        function getConstructorsForTypeArguments(type, typeArgumentNodes, location) {
-            var typeArgCount = ts.length(typeArgumentNodes);
-            var isJavascript = ts.isInJSFile(location);
-            return ts.filter(getSignaturesOfType(type, 1 /* Construct */), function (sig) { return (isJavascript || typeArgCount >= getMinTypeArgumentCount(sig.typeParameters)) && typeArgCount <= ts.length(sig.typeParameters); });
+        function getExportsOfSymbol(symbol) {
+            return symbol.flags & 6256 /* LateBindingContainer */ ? getResolvedMembersOrExportsOfSymbol(symbol, "resolvedExports" /* resolvedExports */) :
+                symbol.flags & 1536 /* Module */ ? getExportsOfModule(symbol) :
+                    symbol.exports || emptySymbols;
         }
-        function getInstantiatedConstructorsForTypeArguments(type, typeArgumentNodes, location) {
-            var signatures = getConstructorsForTypeArguments(type, typeArgumentNodes, location);
-            var typeArguments = ts.map(typeArgumentNodes, getTypeFromTypeNode);
-            return ts.sameMap(signatures, function (sig) { return ts.some(sig.typeParameters) ? getSignatureInstantiation(sig, typeArguments, ts.isInJSFile(location)) : sig; });
+        function getExportsOfModule(moduleSymbol) {
+            var links = getSymbolLinks(moduleSymbol);
+            return links.resolvedExports || (links.resolvedExports = getExportsOfModuleWorker(moduleSymbol));
         }
         /**
-         * The base constructor of a class can resolve to
-         * * undefinedType if the class has no extends clause,
-         * * unknownType if an error occurred during resolution of the extends expression,
-         * * nullType if the extends expression is the null value,
-         * * anyType if the extends expression has type any, or
-         * * an object type with at least one construct signature.
+         * Extends one symbol table with another while collecting information on name collisions for error message generation into the `lookupTable` argument
+         * Not passing `lookupTable` and `exportNode` disables this collection, and just extends the tables
          */
-        function getBaseConstructorTypeOfClass(type) {
-            if (!type.resolvedBaseConstructorType) {
-                var decl = type.symbol.valueDeclaration;
-                var extended = ts.getEffectiveBaseTypeNode(decl);
-                var baseTypeNode = getBaseTypeNodeOfClass(type);
-                if (!baseTypeNode) {
-                    return type.resolvedBaseConstructorType = undefinedType;
-                }
-                if (!pushTypeResolution(type, 1 /* ResolvedBaseConstructorType */)) {
-                    return errorType;
-                }
-                var baseConstructorType = checkExpression(baseTypeNode.expression);
-                if (extended && baseTypeNode !== extended) {
-                    ts.Debug.assert(!extended.typeArguments); // Because this is in a JS file, and baseTypeNode is in an @extends tag
-                    checkExpression(extended.expression);
+        function extendExportSymbols(target, source, lookupTable, exportNode) {
+            if (!source)
+                return;
+            source.forEach(function (sourceSymbol, id) {
+                if (id === "default" /* Default */)
+                    return;
+                var targetSymbol = target.get(id);
+                if (!targetSymbol) {
+                    target.set(id, sourceSymbol);
+                    if (lookupTable && exportNode) {
+                        lookupTable.set(id, {
+                            specifierText: ts.getTextOfNode(exportNode.moduleSpecifier)
+                        });
+                    }
                 }
-                if (baseConstructorType.flags & (524288 /* Object */ | 2097152 /* Intersection */)) {
-                    // Resolving the members of a class requires us to resolve the base class of that class.
-                    // We force resolution here such that we catch circularities now.
-                    resolveStructuredTypeMembers(baseConstructorType);
+                else if (lookupTable && exportNode && targetSymbol && resolveSymbol(targetSymbol) !== resolveSymbol(sourceSymbol)) {
+                    var collisionTracker = lookupTable.get(id);
+                    if (!collisionTracker.exportsWithDuplicate) {
+                        collisionTracker.exportsWithDuplicate = [exportNode];
+                    }
+                    else {
+                        collisionTracker.exportsWithDuplicate.push(exportNode);
+                    }
                 }
-                if (!popTypeResolution()) {
-                    error(type.symbol.valueDeclaration, ts.Diagnostics._0_is_referenced_directly_or_indirectly_in_its_own_base_expression, symbolToString(type.symbol));
-                    return type.resolvedBaseConstructorType = errorType;
+            });
+        }
+        function getExportsOfModuleWorker(moduleSymbol) {
+            var visitedSymbols = [];
+            // A module defined by an 'export=' consists of one export that needs to be resolved
+            moduleSymbol = resolveExternalModuleSymbol(moduleSymbol);
+            return visit(moduleSymbol) || emptySymbols;
+            // The ES6 spec permits export * declarations in a module to circularly reference the module itself. For example,
+            // module 'a' can 'export * from "b"' and 'b' can 'export * from "a"' without error.
+            function visit(symbol) {
+                if (!(symbol && symbol.exports && ts.pushIfUnique(visitedSymbols, symbol))) {
+                    return;
                 }
-                if (!(baseConstructorType.flags & 1 /* Any */) && baseConstructorType !== nullWideningType && !isConstructorType(baseConstructorType)) {
-                    var err = error(baseTypeNode.expression, ts.Diagnostics.Type_0_is_not_a_constructor_function_type, typeToString(baseConstructorType));
-                    if (baseConstructorType.flags & 262144 /* TypeParameter */) {
-                        var constraint = getConstraintFromTypeParameter(baseConstructorType);
-                        var ctorReturn = unknownType;
-                        if (constraint) {
-                            var ctorSig = getSignaturesOfType(constraint, 1 /* Construct */);
-                            if (ctorSig[0]) {
-                                ctorReturn = getReturnTypeOfSignature(ctorSig[0]);
-                            }
-                        }
-                        ts.addRelatedInfo(err, ts.createDiagnosticForNode(baseConstructorType.symbol.declarations[0], ts.Diagnostics.Did_you_mean_for_0_to_be_constrained_to_type_new_args_Colon_any_1, symbolToString(baseConstructorType.symbol), typeToString(ctorReturn)));
+                var symbols = new ts.Map(symbol.exports);
+                // All export * declarations are collected in an __export symbol by the binder
+                var exportStars = symbol.exports.get("__export" /* ExportStar */);
+                if (exportStars) {
+                    var nestedSymbols = ts.createSymbolTable();
+                    var lookupTable_1 = new ts.Map();
+                    for (var _i = 0, _a = exportStars.declarations; _i < _a.length; _i++) {
+                        var node = _a[_i];
+                        var resolvedModule = resolveExternalModuleName(node, node.moduleSpecifier);
+                        var exportedSymbols = visit(resolvedModule);
+                        extendExportSymbols(nestedSymbols, exportedSymbols, lookupTable_1, node);
                     }
-                    return type.resolvedBaseConstructorType = errorType;
+                    lookupTable_1.forEach(function (_a, id) {
+                        var exportsWithDuplicate = _a.exportsWithDuplicate;
+                        // It's not an error if the file with multiple `export *`s with duplicate names exports a member with that name itself
+                        if (id === "export=" || !(exportsWithDuplicate && exportsWithDuplicate.length) || symbols.has(id)) {
+                            return;
+                        }
+                        for (var _i = 0, exportsWithDuplicate_1 = exportsWithDuplicate; _i < exportsWithDuplicate_1.length; _i++) {
+                            var node = exportsWithDuplicate_1[_i];
+                            diagnostics.add(ts.createDiagnosticForNode(node, ts.Diagnostics.Module_0_has_already_exported_a_member_named_1_Consider_explicitly_re_exporting_to_resolve_the_ambiguity, lookupTable_1.get(id).specifierText, ts.unescapeLeadingUnderscores(id)));
+                        }
+                    });
+                    extendExportSymbols(symbols, nestedSymbols);
                 }
-                type.resolvedBaseConstructorType = baseConstructorType;
+                return symbols;
             }
-            return type.resolvedBaseConstructorType;
         }
-        function getImplementsTypes(type) {
-            var resolvedImplementsTypes = ts.emptyArray;
-            for (var _i = 0, _a = type.symbol.declarations; _i < _a.length; _i++) {
-                var declaration = _a[_i];
-                var implementsTypeNodes = ts.getEffectiveImplementsTypeNodes(declaration);
-                if (!implementsTypeNodes)
+        function getMergedSymbol(symbol) {
+            var merged;
+            return symbol && symbol.mergeId && (merged = mergedSymbols[symbol.mergeId]) ? merged : symbol;
+        }
+        function getSymbolOfNode(node) {
+            return getMergedSymbol(node.symbol && getLateBoundSymbol(node.symbol));
+        }
+        function getParentOfSymbol(symbol) {
+            return getMergedSymbol(symbol.parent && getLateBoundSymbol(symbol.parent));
+        }
+        function getAlternativeContainingModules(symbol, enclosingDeclaration) {
+            var containingFile = ts.getSourceFileOfNode(enclosingDeclaration);
+            var id = getNodeId(containingFile);
+            var links = getSymbolLinks(symbol);
+            var results;
+            if (links.extendedContainersByFile && (results = links.extendedContainersByFile.get(id))) {
+                return results;
+            }
+            if (containingFile && containingFile.imports) {
+                // Try to make an import using an import already in the enclosing file, if possible
+                for (var _i = 0, _a = containingFile.imports; _i < _a.length; _i++) {
+                    var importRef = _a[_i];
+                    if (ts.nodeIsSynthesized(importRef))
+                        continue; // Synthetic names can't be resolved by `resolveExternalModuleName` - they'll cause a debug assert if they error
+                    var resolvedModule = resolveExternalModuleName(enclosingDeclaration, importRef, /*ignoreErrors*/ true);
+                    if (!resolvedModule)
+                        continue;
+                    var ref = getAliasForSymbolInContainer(resolvedModule, symbol);
+                    if (!ref)
+                        continue;
+                    results = ts.append(results, resolvedModule);
+                }
+                if (ts.length(results)) {
+                    (links.extendedContainersByFile || (links.extendedContainersByFile = new ts.Map())).set(id, results);
+                    return results;
+                }
+            }
+            if (links.extendedContainers) {
+                return links.extendedContainers;
+            }
+            // No results from files already being imported by this file - expand search (expensive, but not location-specific, so cached)
+            var otherFiles = host.getSourceFiles();
+            for (var _b = 0, otherFiles_1 = otherFiles; _b < otherFiles_1.length; _b++) {
+                var file = otherFiles_1[_b];
+                if (!ts.isExternalModule(file))
                     continue;
-                for (var _b = 0, implementsTypeNodes_1 = implementsTypeNodes; _b < implementsTypeNodes_1.length; _b++) {
-                    var node = implementsTypeNodes_1[_b];
-                    var implementsType = getTypeFromTypeNode(node);
-                    if (implementsType !== errorType) {
-                        if (resolvedImplementsTypes === ts.emptyArray) {
-                            resolvedImplementsTypes = [implementsType];
-                        }
-                        else {
-                            resolvedImplementsTypes.push(implementsType);
-                        }
-                    }
+                var sym = getSymbolOfNode(file);
+                var ref = getAliasForSymbolInContainer(sym, symbol);
+                if (!ref)
+                    continue;
+                results = ts.append(results, sym);
+            }
+            return links.extendedContainers = results || ts.emptyArray;
+        }
+        /**
+         * Attempts to find the symbol corresponding to the container a symbol is in - usually this
+         * is just its' `.parent`, but for locals, this value is `undefined`
+         */
+        function getContainersOfSymbol(symbol, enclosingDeclaration, meaning) {
+            var container = getParentOfSymbol(symbol);
+            // Type parameters end up in the `members` lists but are not externally visible
+            if (container && !(symbol.flags & 262144 /* TypeParameter */)) {
+                var additionalContainers = ts.mapDefined(container.declarations, fileSymbolIfFileSymbolExportEqualsContainer);
+                var reexportContainers = enclosingDeclaration && getAlternativeContainingModules(symbol, enclosingDeclaration);
+                var objectLiteralContainer = getVariableDeclarationOfObjectLiteral(container, meaning);
+                if (enclosingDeclaration && getAccessibleSymbolChain(container, enclosingDeclaration, 1920 /* Namespace */, /*externalOnly*/ false)) {
+                    return ts.append(ts.concatenate(ts.concatenate([container], additionalContainers), reexportContainers), objectLiteralContainer); // This order expresses a preference for the real container if it is in scope
                 }
+                var res = ts.append(ts.append(additionalContainers, container), objectLiteralContainer);
+                return ts.concatenate(res, reexportContainers);
             }
-            return resolvedImplementsTypes;
-        }
-        function getBaseTypes(type) {
-            if (!type.resolvedBaseTypes) {
-                if (type.objectFlags & 8 /* Tuple */) {
-                    type.resolvedBaseTypes = [createArrayType(getUnionType(type.typeParameters || ts.emptyArray), type.readonly)];
+            var candidates = ts.mapDefined(symbol.declarations, function (d) {
+                if (!ts.isAmbientModule(d) && d.parent && hasNonGlobalAugmentationExternalModuleSymbol(d.parent)) {
+                    return getSymbolOfNode(d.parent);
                 }
-                else if (type.symbol.flags & (32 /* Class */ | 64 /* Interface */)) {
-                    if (type.symbol.flags & 32 /* Class */) {
-                        resolveBaseTypesOfClass(type);
-                    }
-                    if (type.symbol.flags & 64 /* Interface */) {
-                        resolveBaseTypesOfInterface(type);
+                if (ts.isClassExpression(d) && ts.isBinaryExpression(d.parent) && d.parent.operatorToken.kind === 62 /* EqualsToken */ && ts.isAccessExpression(d.parent.left) && ts.isEntityNameExpression(d.parent.left.expression)) {
+                    if (ts.isModuleExportsAccessExpression(d.parent.left) || ts.isExportsIdentifier(d.parent.left.expression)) {
+                        return getSymbolOfNode(ts.getSourceFileOfNode(d));
                     }
+                    checkExpressionCached(d.parent.left.expression);
+                    return getNodeLinks(d.parent.left.expression).resolvedSymbol;
                 }
-                else {
-                    ts.Debug.fail("type must be class or interface");
-                }
-            }
-            return type.resolvedBaseTypes;
-        }
-        function resolveBaseTypesOfClass(type) {
-            type.resolvedBaseTypes = ts.resolvingEmptyArray;
-            var baseConstructorType = getApparentType(getBaseConstructorTypeOfClass(type));
-            if (!(baseConstructorType.flags & (524288 /* Object */ | 2097152 /* Intersection */ | 1 /* Any */))) {
-                return type.resolvedBaseTypes = ts.emptyArray;
-            }
-            var baseTypeNode = getBaseTypeNodeOfClass(type);
-            var baseType;
-            var originalBaseType = baseConstructorType.symbol ? getDeclaredTypeOfSymbol(baseConstructorType.symbol) : undefined;
-            if (baseConstructorType.symbol && baseConstructorType.symbol.flags & 32 /* Class */ &&
-                areAllOuterTypeParametersApplied(originalBaseType)) {
-                // When base constructor type is a class with no captured type arguments we know that the constructors all have the same type parameters as the
-                // class and all return the instance type of the class. There is no need for further checks and we can apply the
-                // type arguments in the same manner as a type reference to get the same error reporting experience.
-                baseType = getTypeFromClassOrInterfaceReference(baseTypeNode, baseConstructorType.symbol);
+            });
+            if (!ts.length(candidates)) {
+                return undefined;
             }
-            else if (baseConstructorType.flags & 1 /* Any */) {
-                baseType = baseConstructorType;
+            return ts.mapDefined(candidates, function (candidate) { return getAliasForSymbolInContainer(candidate, symbol) ? candidate : undefined; });
+            function fileSymbolIfFileSymbolExportEqualsContainer(d) {
+                return container && getFileSymbolIfFileSymbolExportEqualsContainer(d, container);
             }
-            else {
-                // The class derives from a "class-like" constructor function, check that we have at least one construct signature
-                // with a matching number of type parameters and use the return type of the first instantiated signature. Elsewhere
-                // we check that all instantiated signatures return the same type.
-                var constructors = getInstantiatedConstructorsForTypeArguments(baseConstructorType, baseTypeNode.typeArguments, baseTypeNode);
-                if (!constructors.length) {
-                    error(baseTypeNode.expression, ts.Diagnostics.No_base_constructor_has_the_specified_number_of_type_arguments);
-                    return type.resolvedBaseTypes = ts.emptyArray;
+        }
+        function getVariableDeclarationOfObjectLiteral(symbol, meaning) {
+            // If we're trying to reference some object literal in, eg `var a = { x: 1 }`, the symbol for the literal, `__object`, is distinct
+            // from the symbol of the declaration it is being assigned to. Since we can use the declaration to refer to the literal, however,
+            // we'd like to make that connection here - potentially causing us to paint the declaration's visibility, and therefore the literal.
+            var firstDecl = !!ts.length(symbol.declarations) && ts.first(symbol.declarations);
+            if (meaning & 111551 /* Value */ && firstDecl && firstDecl.parent && ts.isVariableDeclaration(firstDecl.parent)) {
+                if (ts.isObjectLiteralExpression(firstDecl) && firstDecl === firstDecl.parent.initializer || ts.isTypeLiteralNode(firstDecl) && firstDecl === firstDecl.parent.type) {
+                    return getSymbolOfNode(firstDecl.parent);
                 }
-                baseType = getReturnTypeOfSignature(constructors[0]);
-            }
-            if (baseType === errorType) {
-                return type.resolvedBaseTypes = ts.emptyArray;
-            }
-            var reducedBaseType = getReducedType(baseType);
-            if (!isValidBaseType(reducedBaseType)) {
-                var elaboration = elaborateNeverIntersection(/*errorInfo*/ undefined, baseType);
-                var diagnostic = ts.chainDiagnosticMessages(elaboration, ts.Diagnostics.Base_constructor_return_type_0_is_not_an_object_type_or_intersection_of_object_types_with_statically_known_members, typeToString(reducedBaseType));
-                diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(baseTypeNode.expression, diagnostic));
-                return type.resolvedBaseTypes = ts.emptyArray;
-            }
-            if (type === reducedBaseType || hasBaseType(reducedBaseType, type)) {
-                error(type.symbol.valueDeclaration, ts.Diagnostics.Type_0_recursively_references_itself_as_a_base_type, typeToString(type, /*enclosingDeclaration*/ undefined, 2 /* WriteArrayAsGenericType */));
-                return type.resolvedBaseTypes = ts.emptyArray;
             }
-            if (type.resolvedBaseTypes === ts.resolvingEmptyArray) {
-                // Circular reference, likely through instantiation of default parameters
-                // (otherwise there'd be an error from hasBaseType) - this is fine, but `.members` should be reset
-                // as `getIndexedAccessType` via `instantiateType` via `getTypeFromClassOrInterfaceReference` forces a
-                // partial instantiation of the members without the base types fully resolved
-                type.members = undefined;
-            }
-            return type.resolvedBaseTypes = [reducedBaseType];
         }
-        function areAllOuterTypeParametersApplied(type) {
-            // An unapplied type parameter has its symbol still the same as the matching argument symbol.
-            // Since parameters are applied outer-to-inner, only the last outer parameter needs to be checked.
-            var outerTypeParameters = type.outerTypeParameters;
-            if (outerTypeParameters) {
-                var last_1 = outerTypeParameters.length - 1;
-                var typeArguments = getTypeArguments(type);
-                return outerTypeParameters[last_1].symbol !== typeArguments[last_1].symbol;
-            }
-            return true;
+        function getFileSymbolIfFileSymbolExportEqualsContainer(d, container) {
+            var fileSymbol = getExternalModuleContainer(d);
+            var exported = fileSymbol && fileSymbol.exports && fileSymbol.exports.get("export=" /* ExportEquals */);
+            return exported && getSymbolIfSameReference(exported, container) ? fileSymbol : undefined;
         }
-        // A valid base type is `any`, an object type or intersection of object types.
-        function isValidBaseType(type) {
-            if (type.flags & 262144 /* TypeParameter */) {
-                var constraint = getBaseConstraintOfType(type);
-                if (constraint) {
-                    return isValidBaseType(constraint);
-                }
+        function getAliasForSymbolInContainer(container, symbol) {
+            if (container === getParentOfSymbol(symbol)) {
+                // fast path, `symbol` is either already the alias or isn't aliased
+                return symbol;
             }
-            // TODO: Given that we allow type parmeters here now, is this `!isGenericMappedType(type)` check really needed?
-            // There's no reason a `T` should be allowed while a `Readonly<T>` should not.
-            return !!(type.flags & (524288 /* Object */ | 67108864 /* NonPrimitive */ | 1 /* Any */) && !isGenericMappedType(type) ||
-                type.flags & 2097152 /* Intersection */ && ts.every(type.types, isValidBaseType));
-        }
-        function resolveBaseTypesOfInterface(type) {
-            type.resolvedBaseTypes = type.resolvedBaseTypes || ts.emptyArray;
-            for (var _i = 0, _a = type.symbol.declarations; _i < _a.length; _i++) {
-                var declaration = _a[_i];
-                if (declaration.kind === 246 /* InterfaceDeclaration */ && ts.getInterfaceBaseTypeNodes(declaration)) {
-                    for (var _b = 0, _c = ts.getInterfaceBaseTypeNodes(declaration); _b < _c.length; _b++) {
-                        var node = _c[_b];
-                        var baseType = getReducedType(getTypeFromTypeNode(node));
-                        if (baseType !== errorType) {
-                            if (isValidBaseType(baseType)) {
-                                if (type !== baseType && !hasBaseType(baseType, type)) {
-                                    if (type.resolvedBaseTypes === ts.emptyArray) {
-                                        type.resolvedBaseTypes = [baseType];
-                                    }
-                                    else {
-                                        type.resolvedBaseTypes.push(baseType);
-                                    }
-                                }
-                                else {
-                                    error(declaration, ts.Diagnostics.Type_0_recursively_references_itself_as_a_base_type, typeToString(type, /*enclosingDeclaration*/ undefined, 2 /* WriteArrayAsGenericType */));
-                                }
-                            }
-                            else {
-                                error(node, ts.Diagnostics.An_interface_can_only_extend_an_object_type_or_intersection_of_object_types_with_statically_known_members);
-                            }
-                        }
-                    }
-                }
+            // Check if container is a thing with an `export=` which points directly at `symbol`, and if so, return
+            // the container itself as the alias for the symbol
+            var exportEquals = container.exports && container.exports.get("export=" /* ExportEquals */);
+            if (exportEquals && getSymbolIfSameReference(exportEquals, symbol)) {
+                return container;
+            }
+            var exports = getExportsOfSymbol(container);
+            var quick = exports.get(symbol.escapedName);
+            if (quick && getSymbolIfSameReference(quick, symbol)) {
+                return quick;
             }
+            return ts.forEachEntry(exports, function (exported) {
+                if (getSymbolIfSameReference(exported, symbol)) {
+                    return exported;
+                }
+            });
         }
         /**
-         * Returns true if the interface given by the symbol is free of "this" references.
-         *
-         * Specifically, the result is true if the interface itself contains no references
-         * to "this" in its body, if all base types are interfaces,
-         * and if none of the base interfaces have a "this" type.
+         * Checks if two symbols, through aliasing and/or merging, refer to the same thing
          */
-        function isThislessInterface(symbol) {
-            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                var declaration = _a[_i];
-                if (declaration.kind === 246 /* InterfaceDeclaration */) {
-                    if (declaration.flags & 128 /* ContainsThis */) {
-                        return false;
-                    }
-                    var baseTypeNodes = ts.getInterfaceBaseTypeNodes(declaration);
-                    if (baseTypeNodes) {
-                        for (var _b = 0, baseTypeNodes_1 = baseTypeNodes; _b < baseTypeNodes_1.length; _b++) {
-                            var node = baseTypeNodes_1[_b];
-                            if (ts.isEntityNameExpression(node.expression)) {
-                                var baseSymbol = resolveEntityName(node.expression, 788968 /* Type */, /*ignoreErrors*/ true);
-                                if (!baseSymbol || !(baseSymbol.flags & 64 /* Interface */) || getDeclaredTypeOfClassOrInterface(baseSymbol).thisType) {
-                                    return false;
-                                }
-                            }
-                        }
-                    }
-                }
+        function getSymbolIfSameReference(s1, s2) {
+            if (getMergedSymbol(resolveSymbol(getMergedSymbol(s1))) === getMergedSymbol(resolveSymbol(getMergedSymbol(s2)))) {
+                return s1;
             }
-            return true;
         }
-        function getDeclaredTypeOfClassOrInterface(symbol) {
-            var links = getSymbolLinks(symbol);
-            var originalLinks = links;
-            if (!links.declaredType) {
-                var kind = symbol.flags & 32 /* Class */ ? 1 /* Class */ : 2 /* Interface */;
-                var merged = mergeJSSymbols(symbol, getAssignedClassSymbol(symbol.valueDeclaration));
-                if (merged) {
-                    // note:we overwrite links because we just cloned the symbol
-                    symbol = links = merged;
-                }
-                var type = originalLinks.declaredType = links.declaredType = createObjectType(kind, symbol);
-                var outerTypeParameters = getOuterTypeParametersOfClassOrInterface(symbol);
-                var localTypeParameters = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
-                // A class or interface is generic if it has type parameters or a "this" type. We always give classes a "this" type
-                // because it is not feasible to analyze all members to determine if the "this" type escapes the class (in particular,
-                // property types inferred from initializers and method return types inferred from return statements are very hard
-                // to exhaustively analyze). We give interfaces a "this" type if we can't definitely determine that they are free of
-                // "this" references.
-                if (outerTypeParameters || localTypeParameters || kind === 1 /* Class */ || !isThislessInterface(symbol)) {
-                    type.objectFlags |= 4 /* Reference */;
-                    type.typeParameters = ts.concatenate(outerTypeParameters, localTypeParameters);
-                    type.outerTypeParameters = outerTypeParameters;
-                    type.localTypeParameters = localTypeParameters;
-                    type.instantiations = ts.createMap();
-                    type.instantiations.set(getTypeListId(type.typeParameters), type);
-                    type.target = type;
-                    type.resolvedTypeArguments = type.typeParameters;
-                    type.thisType = createTypeParameter(symbol);
-                    type.thisType.isThisType = true;
-                    type.thisType.constraint = type;
+        function getExportSymbolOfValueSymbolIfExported(symbol) {
+            return getMergedSymbol(symbol && (symbol.flags & 1048576 /* ExportValue */) !== 0 ? symbol.exportSymbol : symbol);
+        }
+        function symbolIsValue(symbol) {
+            return !!(symbol.flags & 111551 /* Value */ || symbol.flags & 2097152 /* Alias */ && resolveAlias(symbol).flags & 111551 /* Value */ && !getTypeOnlyAliasDeclaration(symbol));
+        }
+        function findConstructorDeclaration(node) {
+            var members = node.members;
+            for (var _i = 0, members_3 = members; _i < members_3.length; _i++) {
+                var member = members_3[_i];
+                if (member.kind === 166 /* Constructor */ && ts.nodeIsPresent(member.body)) {
+                    return member;
                 }
             }
-            return links.declaredType;
         }
-        function getDeclaredTypeOfTypeAlias(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.declaredType) {
-                // Note that we use the links object as the target here because the symbol object is used as the unique
-                // identity for resolution of the 'type' property in SymbolLinks.
-                if (!pushTypeResolution(symbol, 2 /* DeclaredType */)) {
-                    return errorType;
+        function createType(flags) {
+            var result = new Type(checker, flags);
+            typeCount++;
+            result.id = typeCount;
+            typeCatalog.push(result);
+            return result;
+        }
+        function createIntrinsicType(kind, intrinsicName, objectFlags) {
+            if (objectFlags === void 0) { objectFlags = 0; }
+            var type = createType(kind);
+            type.intrinsicName = intrinsicName;
+            type.objectFlags = objectFlags;
+            return type;
+        }
+        function createBooleanType(trueFalseTypes) {
+            var type = getUnionType(trueFalseTypes);
+            type.flags |= 16 /* Boolean */;
+            type.intrinsicName = "boolean";
+            return type;
+        }
+        function createObjectType(objectFlags, symbol) {
+            var type = createType(524288 /* Object */);
+            type.objectFlags = objectFlags;
+            type.symbol = symbol;
+            type.members = undefined;
+            type.properties = undefined;
+            type.callSignatures = undefined;
+            type.constructSignatures = undefined;
+            type.stringIndexInfo = undefined;
+            type.numberIndexInfo = undefined;
+            return type;
+        }
+        function createTypeofType() {
+            return getUnionType(ts.arrayFrom(typeofEQFacts.keys(), getLiteralType));
+        }
+        function createTypeParameter(symbol) {
+            var type = createType(262144 /* TypeParameter */);
+            if (symbol)
+                type.symbol = symbol;
+            return type;
+        }
+        // A reserved member name starts with two underscores, but the third character cannot be an underscore,
+        // @, or #. A third underscore indicates an escaped form of an identifier that started
+        // with at least two underscores. The @ character indicates that the name is denoted by a well known ES
+        // Symbol instance and the # character indicates that the name is a PrivateIdentifier.
+        function isReservedMemberName(name) {
+            return name.charCodeAt(0) === 95 /* _ */ &&
+                name.charCodeAt(1) === 95 /* _ */ &&
+                name.charCodeAt(2) !== 95 /* _ */ &&
+                name.charCodeAt(2) !== 64 /* at */ &&
+                name.charCodeAt(2) !== 35 /* hash */;
+        }
+        function getNamedMembers(members) {
+            var result;
+            members.forEach(function (symbol, id) {
+                if (!isReservedMemberName(id) && symbolIsValue(symbol)) {
+                    (result || (result = [])).push(symbol);
                 }
-                var declaration = ts.Debug.checkDefined(ts.find(symbol.declarations, ts.isTypeAlias), "Type alias symbol with no valid declaration found");
-                var typeNode = ts.isJSDocTypeAlias(declaration) ? declaration.typeExpression : declaration.type;
-                // If typeNode is missing, we will error in checkJSDocTypedefTag.
-                var type = typeNode ? getTypeFromTypeNode(typeNode) : errorType;
-                if (popTypeResolution()) {
-                    var typeParameters = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
-                    if (typeParameters) {
-                        // Initialize the instantiation cache for generic type aliases. The declared type corresponds to
-                        // an instantiation of the type alias with the type parameters supplied as type arguments.
-                        links.typeParameters = typeParameters;
-                        links.instantiations = ts.createMap();
-                        links.instantiations.set(getTypeListId(typeParameters), type);
+            });
+            return result || ts.emptyArray;
+        }
+        function setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo) {
+            type.members = members;
+            type.properties = members === emptySymbols ? ts.emptyArray : getNamedMembers(members);
+            type.callSignatures = callSignatures;
+            type.constructSignatures = constructSignatures;
+            type.stringIndexInfo = stringIndexInfo;
+            type.numberIndexInfo = numberIndexInfo;
+            return type;
+        }
+        function createAnonymousType(symbol, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo) {
+            return setStructuredTypeMembers(createObjectType(16 /* Anonymous */, symbol), members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
+        }
+        function forEachSymbolTableInScope(enclosingDeclaration, callback) {
+            var result;
+            var _loop_8 = function (location) {
+                // Locals of a source file are not in scope (because they get merged into the global symbol table)
+                if (location.locals && !isGlobalSourceFile(location)) {
+                    if (result = callback(location.locals)) {
+                        return { value: result };
                     }
                 }
-                else {
-                    type = errorType;
-                    error(ts.isNamedDeclaration(declaration) ? declaration.name : declaration || declaration, ts.Diagnostics.Type_alias_0_circularly_references_itself, symbolToString(symbol));
+                switch (location.kind) {
+                    case 297 /* SourceFile */:
+                        if (!ts.isExternalOrCommonJsModule(location)) {
+                            break;
+                        }
+                    // falls through
+                    case 256 /* ModuleDeclaration */:
+                        var sym = getSymbolOfNode(location);
+                        // `sym` may not have exports if this module declaration is backed by the symbol for a `const` that's being rewritten
+                        // into a namespace - in such cases, it's best to just let the namespace appear empty (the const members couldn't have referred
+                        // to one another anyway)
+                        if (result = callback((sym === null || sym === void 0 ? void 0 : sym.exports) || emptySymbols)) {
+                            return { value: result };
+                        }
+                        break;
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 253 /* InterfaceDeclaration */:
+                        // Type parameters are bound into `members` lists so they can merge across declarations
+                        // This is troublesome, since in all other respects, they behave like locals :cries:
+                        // TODO: the below is shared with similar code in `resolveName` - in fact, rephrasing all this symbol
+                        // lookup logic in terms of `resolveName` would be nice
+                        // The below is used to lookup type parameters within a class or interface, as they are added to the class/interface locals
+                        // These can never be latebound, so the symbol's raw members are sufficient. `getMembersOfNode` cannot be used, as it would
+                        // trigger resolving late-bound names, which we may already be in the process of doing while we're here!
+                        var table_1;
+                        // TODO: Should this filtered table be cached in some way?
+                        (getSymbolOfNode(location).members || emptySymbols).forEach(function (memberSymbol, key) {
+                            if (memberSymbol.flags & (788968 /* Type */ & ~67108864 /* Assignment */)) {
+                                (table_1 || (table_1 = ts.createSymbolTable())).set(key, memberSymbol);
+                            }
+                        });
+                        if (table_1 && (result = callback(table_1))) {
+                            return { value: result };
+                        }
+                        break;
                 }
-                links.declaredType = type;
+            };
+            for (var location = enclosingDeclaration; location; location = location.parent) {
+                var state_2 = _loop_8(location);
+                if (typeof state_2 === "object")
+                    return state_2.value;
             }
-            return links.declaredType;
+            return callback(globals);
         }
-        function isStringConcatExpression(expr) {
-            if (ts.isStringLiteralLike(expr)) {
-                return true;
+        function getQualifiedLeftMeaning(rightMeaning) {
+            // If we are looking in value space, the parent meaning is value, other wise it is namespace
+            return rightMeaning === 111551 /* Value */ ? 111551 /* Value */ : 1920 /* Namespace */;
+        }
+        function getAccessibleSymbolChain(symbol, enclosingDeclaration, meaning, useOnlyExternalAliasing, visitedSymbolTablesMap) {
+            if (visitedSymbolTablesMap === void 0) { visitedSymbolTablesMap = new ts.Map(); }
+            if (!(symbol && !isPropertyOrMethodDeclarationSymbol(symbol))) {
+                return undefined;
             }
-            else if (expr.kind === 209 /* BinaryExpression */) {
-                return isStringConcatExpression(expr.left) && isStringConcatExpression(expr.right);
+            var id = getSymbolId(symbol);
+            var visitedSymbolTables = visitedSymbolTablesMap.get(id);
+            if (!visitedSymbolTables) {
+                visitedSymbolTablesMap.set(id, visitedSymbolTables = []);
             }
-            return false;
-        }
-        function isLiteralEnumMember(member) {
-            var expr = member.initializer;
-            if (!expr) {
-                return !(member.flags & 8388608 /* Ambient */);
+            return forEachSymbolTableInScope(enclosingDeclaration, getAccessibleSymbolChainFromSymbolTable);
+            /**
+             * @param {ignoreQualification} boolean Set when a symbol is being looked for through the exports of another symbol (meaning we have a route to qualify it already)
+             */
+            function getAccessibleSymbolChainFromSymbolTable(symbols, ignoreQualification) {
+                if (!ts.pushIfUnique(visitedSymbolTables, symbols)) {
+                    return undefined;
+                }
+                var result = trySymbolTable(symbols, ignoreQualification);
+                visitedSymbolTables.pop();
+                return result;
             }
-            switch (expr.kind) {
-                case 10 /* StringLiteral */:
-                case 8 /* NumericLiteral */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                    return true;
-                case 207 /* PrefixUnaryExpression */:
-                    return expr.operator === 40 /* MinusToken */ &&
-                        expr.operand.kind === 8 /* NumericLiteral */;
-                case 75 /* Identifier */:
-                    return ts.nodeIsMissing(expr) || !!getSymbolOfNode(member.parent).exports.get(expr.escapedText);
-                case 209 /* BinaryExpression */:
-                    return isStringConcatExpression(expr);
-                default:
-                    return false;
+            function canQualifySymbol(symbolFromSymbolTable, meaning) {
+                // If the symbol is equivalent and doesn't need further qualification, this symbol is accessible
+                return !needsQualification(symbolFromSymbolTable, enclosingDeclaration, meaning) ||
+                    // If symbol needs qualification, make sure that parent is accessible, if it is then this symbol is accessible too
+                    !!getAccessibleSymbolChain(symbolFromSymbolTable.parent, enclosingDeclaration, getQualifiedLeftMeaning(meaning), useOnlyExternalAliasing, visitedSymbolTablesMap);
             }
-        }
-        function getEnumKind(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (links.enumKind !== undefined) {
-                return links.enumKind;
+            function isAccessible(symbolFromSymbolTable, resolvedAliasSymbol, ignoreQualification) {
+                return (symbol === (resolvedAliasSymbol || symbolFromSymbolTable) || getMergedSymbol(symbol) === getMergedSymbol(resolvedAliasSymbol || symbolFromSymbolTable)) &&
+                    // if the symbolFromSymbolTable is not external module (it could be if it was determined as ambient external module and would be in globals table)
+                    // and if symbolFromSymbolTable or alias resolution matches the symbol,
+                    // check the symbol can be qualified, it is only then this symbol is accessible
+                    !ts.some(symbolFromSymbolTable.declarations, hasNonGlobalAugmentationExternalModuleSymbol) &&
+                    (ignoreQualification || canQualifySymbol(getMergedSymbol(symbolFromSymbolTable), meaning));
             }
-            var hasNonLiteralMember = false;
-            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                var declaration = _a[_i];
-                if (declaration.kind === 248 /* EnumDeclaration */) {
-                    for (var _b = 0, _c = declaration.members; _b < _c.length; _b++) {
-                        var member = _c[_b];
-                        if (member.initializer && ts.isStringLiteralLike(member.initializer)) {
-                            return links.enumKind = 1 /* Literal */;
+            function trySymbolTable(symbols, ignoreQualification) {
+                // If symbol is directly available by its name in the symbol table
+                if (isAccessible(symbols.get(symbol.escapedName), /*resolvedAliasSymbol*/ undefined, ignoreQualification)) {
+                    return [symbol];
+                }
+                // Check if symbol is any of the aliases in scope
+                var result = ts.forEachEntry(symbols, function (symbolFromSymbolTable) {
+                    if (symbolFromSymbolTable.flags & 2097152 /* Alias */
+                        && symbolFromSymbolTable.escapedName !== "export=" /* ExportEquals */
+                        && symbolFromSymbolTable.escapedName !== "default" /* Default */
+                        && !(ts.isUMDExportSymbol(symbolFromSymbolTable) && enclosingDeclaration && ts.isExternalModule(ts.getSourceFileOfNode(enclosingDeclaration)))
+                        // If `!useOnlyExternalAliasing`, we can use any type of alias to get the name
+                        && (!useOnlyExternalAliasing || ts.some(symbolFromSymbolTable.declarations, ts.isExternalModuleImportEqualsDeclaration))
+                        // While exports are generally considered to be in scope, export-specifier declared symbols are _not_
+                        // See similar comment in `resolveName` for details
+                        && (ignoreQualification || !ts.getDeclarationOfKind(symbolFromSymbolTable, 270 /* ExportSpecifier */))) {
+                        var resolvedImportedSymbol = resolveAlias(symbolFromSymbolTable);
+                        var candidate = getCandidateListForSymbol(symbolFromSymbolTable, resolvedImportedSymbol, ignoreQualification);
+                        if (candidate) {
+                            return candidate;
                         }
-                        if (!isLiteralEnumMember(member)) {
-                            hasNonLiteralMember = true;
+                    }
+                    if (symbolFromSymbolTable.escapedName === symbol.escapedName && symbolFromSymbolTable.exportSymbol) {
+                        if (isAccessible(getMergedSymbol(symbolFromSymbolTable.exportSymbol), /*aliasSymbol*/ undefined, ignoreQualification)) {
+                            return [symbol];
                         }
                     }
+                });
+                // If there's no result and we're looking at the global symbol table, treat `globalThis` like an alias and try to lookup thru that
+                return result || (symbols === globals ? getCandidateListForSymbol(globalThisSymbol, globalThisSymbol, ignoreQualification) : undefined);
+            }
+            function getCandidateListForSymbol(symbolFromSymbolTable, resolvedImportedSymbol, ignoreQualification) {
+                if (isAccessible(symbolFromSymbolTable, resolvedImportedSymbol, ignoreQualification)) {
+                    return [symbolFromSymbolTable];
+                }
+                // Look in the exported members, if we can find accessibleSymbolChain, symbol is accessible using this chain
+                // but only if the symbolFromSymbolTable can be qualified
+                var candidateTable = getExportsOfSymbol(resolvedImportedSymbol);
+                var accessibleSymbolsFromExports = candidateTable && getAccessibleSymbolChainFromSymbolTable(candidateTable, /*ignoreQualification*/ true);
+                if (accessibleSymbolsFromExports && canQualifySymbol(symbolFromSymbolTable, getQualifiedLeftMeaning(meaning))) {
+                    return [symbolFromSymbolTable].concat(accessibleSymbolsFromExports);
                 }
             }
-            return links.enumKind = hasNonLiteralMember ? 0 /* Numeric */ : 1 /* Literal */;
         }
-        function getBaseTypeOfEnumLiteralType(type) {
-            return type.flags & 1024 /* EnumLiteral */ && !(type.flags & 1048576 /* Union */) ? getDeclaredTypeOfSymbol(getParentOfSymbol(type.symbol)) : type;
+        function needsQualification(symbol, enclosingDeclaration, meaning) {
+            var qualify = false;
+            forEachSymbolTableInScope(enclosingDeclaration, function (symbolTable) {
+                // If symbol of this name is not available in the symbol table we are ok
+                var symbolFromSymbolTable = getMergedSymbol(symbolTable.get(symbol.escapedName));
+                if (!symbolFromSymbolTable) {
+                    // Continue to the next symbol table
+                    return false;
+                }
+                // If the symbol with this name is present it should refer to the symbol
+                if (symbolFromSymbolTable === symbol) {
+                    // No need to qualify
+                    return true;
+                }
+                // Qualify if the symbol from symbol table has same meaning as expected
+                symbolFromSymbolTable = (symbolFromSymbolTable.flags & 2097152 /* Alias */ && !ts.getDeclarationOfKind(symbolFromSymbolTable, 270 /* ExportSpecifier */)) ? resolveAlias(symbolFromSymbolTable) : symbolFromSymbolTable;
+                if (symbolFromSymbolTable.flags & meaning) {
+                    qualify = true;
+                    return true;
+                }
+                // Continue to the next symbol table
+                return false;
+            });
+            return qualify;
         }
-        function getDeclaredTypeOfEnum(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (links.declaredType) {
-                return links.declaredType;
-            }
-            if (getEnumKind(symbol) === 1 /* Literal */) {
-                enumCount++;
-                var memberTypeList = [];
+        function isPropertyOrMethodDeclarationSymbol(symbol) {
+            if (symbol.declarations && symbol.declarations.length) {
                 for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
                     var declaration = _a[_i];
-                    if (declaration.kind === 248 /* EnumDeclaration */) {
-                        for (var _b = 0, _c = declaration.members; _b < _c.length; _b++) {
-                            var member = _c[_b];
-                            var value = getEnumMemberValue(member);
-                            var memberType = getFreshTypeOfLiteralType(getLiteralType(value !== undefined ? value : 0, enumCount, getSymbolOfNode(member)));
-                            getSymbolLinks(getSymbolOfNode(member)).declaredType = memberType;
-                            memberTypeList.push(getRegularTypeOfLiteralType(memberType));
-                        }
-                    }
-                }
-                if (memberTypeList.length) {
-                    var enumType_1 = getUnionType(memberTypeList, 1 /* Literal */, symbol, /*aliasTypeArguments*/ undefined);
-                    if (enumType_1.flags & 1048576 /* Union */) {
-                        enumType_1.flags |= 1024 /* EnumLiteral */;
-                        enumType_1.symbol = symbol;
+                    switch (declaration.kind) {
+                        case 163 /* PropertyDeclaration */:
+                        case 165 /* MethodDeclaration */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
+                            continue;
+                        default:
+                            return false;
                     }
-                    return links.declaredType = enumType_1;
-                }
-            }
-            var enumType = createType(32 /* Enum */);
-            enumType.symbol = symbol;
-            return links.declaredType = enumType;
-        }
-        function getDeclaredTypeOfEnumMember(symbol) {
-            var links = getSymbolLinks(symbol);
-            if (!links.declaredType) {
-                var enumType = getDeclaredTypeOfEnum(getParentOfSymbol(symbol));
-                if (!links.declaredType) {
-                    links.declaredType = enumType;
                 }
+                return true;
             }
-            return links.declaredType;
+            return false;
         }
-        function getDeclaredTypeOfTypeParameter(symbol) {
-            var links = getSymbolLinks(symbol);
-            return links.declaredType || (links.declaredType = createTypeParameter(symbol));
+        function isTypeSymbolAccessible(typeSymbol, enclosingDeclaration) {
+            var access = isSymbolAccessibleWorker(typeSymbol, enclosingDeclaration, 788968 /* Type */, /*shouldComputeAliasesToMakeVisible*/ false, /*allowModules*/ true);
+            return access.accessibility === 0 /* Accessible */;
         }
-        function getDeclaredTypeOfAlias(symbol) {
-            var links = getSymbolLinks(symbol);
-            return links.declaredType || (links.declaredType = getDeclaredTypeOfSymbol(resolveAlias(symbol)));
+        function isValueSymbolAccessible(typeSymbol, enclosingDeclaration) {
+            var access = isSymbolAccessibleWorker(typeSymbol, enclosingDeclaration, 111551 /* Value */, /*shouldComputeAliasesToMakeVisible*/ false, /*allowModules*/ true);
+            return access.accessibility === 0 /* Accessible */;
         }
-        function getDeclaredTypeOfSymbol(symbol) {
-            return tryGetDeclaredTypeOfSymbol(symbol) || errorType;
+        function isSymbolAccessibleByFlags(typeSymbol, enclosingDeclaration, flags) {
+            var access = isSymbolAccessibleWorker(typeSymbol, enclosingDeclaration, flags, /*shouldComputeAliasesToMakeVisible*/ false, /*allowModules*/ false);
+            return access.accessibility === 0 /* Accessible */;
         }
-        function tryGetDeclaredTypeOfSymbol(symbol) {
-            if (symbol.flags & (32 /* Class */ | 64 /* Interface */)) {
-                return getDeclaredTypeOfClassOrInterface(symbol);
-            }
-            if (symbol.flags & 524288 /* TypeAlias */) {
-                return getDeclaredTypeOfTypeAlias(symbol);
-            }
-            if (symbol.flags & 262144 /* TypeParameter */) {
-                return getDeclaredTypeOfTypeParameter(symbol);
-            }
-            if (symbol.flags & 384 /* Enum */) {
-                return getDeclaredTypeOfEnum(symbol);
+        function isAnySymbolAccessible(symbols, enclosingDeclaration, initialSymbol, meaning, shouldComputeAliasesToMakeVisible, allowModules) {
+            if (!ts.length(symbols))
+                return;
+            var hadAccessibleChain;
+            var earlyModuleBail = false;
+            for (var _i = 0, _a = symbols; _i < _a.length; _i++) {
+                var symbol = _a[_i];
+                // Symbol is accessible if it by itself is accessible
+                var accessibleSymbolChain = getAccessibleSymbolChain(symbol, enclosingDeclaration, meaning, /*useOnlyExternalAliasing*/ false);
+                if (accessibleSymbolChain) {
+                    hadAccessibleChain = symbol;
+                    var hasAccessibleDeclarations = hasVisibleDeclarations(accessibleSymbolChain[0], shouldComputeAliasesToMakeVisible);
+                    if (hasAccessibleDeclarations) {
+                        return hasAccessibleDeclarations;
+                    }
+                }
+                else if (allowModules) {
+                    if (ts.some(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
+                        if (shouldComputeAliasesToMakeVisible) {
+                            earlyModuleBail = true;
+                            // Generally speaking, we want to use the aliases that already exist to refer to a module, if present
+                            // In order to do so, we need to find those aliases in order to retain them in declaration emit; so
+                            // if we are in declaration emit, we cannot use the fast path for module visibility until we've exhausted
+                            // all other visibility options (in order to capture the possible aliases used to reference the module)
+                            continue;
+                        }
+                        // Any meaning of a module symbol is always accessible via an `import` type
+                        return {
+                            accessibility: 0 /* Accessible */
+                        };
+                    }
+                }
+                // If we haven't got the accessible symbol, it doesn't mean the symbol is actually inaccessible.
+                // It could be a qualified symbol and hence verify the path
+                // e.g.:
+                // module m {
+                //     export class c {
+                //     }
+                // }
+                // const x: typeof m.c
+                // In the above example when we start with checking if typeof m.c symbol is accessible,
+                // we are going to see if c can be accessed in scope directly.
+                // But it can't, hence the accessible is going to be undefined, but that doesn't mean m.c is inaccessible
+                // It is accessible if the parent m is accessible because then m.c can be accessed through qualification
+                var containers = getContainersOfSymbol(symbol, enclosingDeclaration, meaning);
+                var parentResult = isAnySymbolAccessible(containers, enclosingDeclaration, initialSymbol, initialSymbol === symbol ? getQualifiedLeftMeaning(meaning) : meaning, shouldComputeAliasesToMakeVisible, allowModules);
+                if (parentResult) {
+                    return parentResult;
+                }
             }
-            if (symbol.flags & 8 /* EnumMember */) {
-                return getDeclaredTypeOfEnumMember(symbol);
+            if (earlyModuleBail) {
+                return {
+                    accessibility: 0 /* Accessible */
+                };
             }
-            if (symbol.flags & 2097152 /* Alias */) {
-                return getDeclaredTypeOfAlias(symbol);
+            if (hadAccessibleChain) {
+                return {
+                    accessibility: 1 /* NotAccessible */,
+                    errorSymbolName: symbolToString(initialSymbol, enclosingDeclaration, meaning),
+                    errorModuleName: hadAccessibleChain !== initialSymbol ? symbolToString(hadAccessibleChain, enclosingDeclaration, 1920 /* Namespace */) : undefined,
+                };
             }
-            return undefined;
         }
         /**
-         * A type is free of this references if it's the any, string, number, boolean, symbol, or void keyword, a string
-         * literal type, an array with an element type that is free of this references, or a type reference that is
-         * free of this references.
+         * Check if the given symbol in given enclosing declaration is accessible and mark all associated alias to be visible if requested
+         *
+         * @param symbol a Symbol to check if accessible
+         * @param enclosingDeclaration a Node containing reference to the symbol
+         * @param meaning a SymbolFlags to check if such meaning of the symbol is accessible
+         * @param shouldComputeAliasToMakeVisible a boolean value to indicate whether to return aliases to be mark visible in case the symbol is accessible
          */
-        function isThislessType(node) {
-            switch (node.kind) {
-                case 125 /* AnyKeyword */:
-                case 148 /* UnknownKeyword */:
-                case 143 /* StringKeyword */:
-                case 140 /* NumberKeyword */:
-                case 151 /* BigIntKeyword */:
-                case 128 /* BooleanKeyword */:
-                case 144 /* SymbolKeyword */:
-                case 141 /* ObjectKeyword */:
-                case 110 /* VoidKeyword */:
-                case 146 /* UndefinedKeyword */:
-                case 100 /* NullKeyword */:
-                case 137 /* NeverKeyword */:
-                case 187 /* LiteralType */:
-                    return true;
-                case 174 /* ArrayType */:
-                    return isThislessType(node.elementType);
-                case 169 /* TypeReference */:
-                    return !node.typeArguments || node.typeArguments.every(isThislessType);
+        function isSymbolAccessible(symbol, enclosingDeclaration, meaning, shouldComputeAliasesToMakeVisible) {
+            return isSymbolAccessibleWorker(symbol, enclosingDeclaration, meaning, shouldComputeAliasesToMakeVisible, /*allowModules*/ true);
+        }
+        function isSymbolAccessibleWorker(symbol, enclosingDeclaration, meaning, shouldComputeAliasesToMakeVisible, allowModules) {
+            if (symbol && enclosingDeclaration) {
+                var result = isAnySymbolAccessible([symbol], enclosingDeclaration, symbol, meaning, shouldComputeAliasesToMakeVisible, allowModules);
+                if (result) {
+                    return result;
+                }
+                // This could be a symbol that is not exported in the external module
+                // or it could be a symbol from different external module that is not aliased and hence cannot be named
+                var symbolExternalModule = ts.forEach(symbol.declarations, getExternalModuleContainer);
+                if (symbolExternalModule) {
+                    var enclosingExternalModule = getExternalModuleContainer(enclosingDeclaration);
+                    if (symbolExternalModule !== enclosingExternalModule) {
+                        // name from different external module that is not visible
+                        return {
+                            accessibility: 2 /* CannotBeNamed */,
+                            errorSymbolName: symbolToString(symbol, enclosingDeclaration, meaning),
+                            errorModuleName: symbolToString(symbolExternalModule)
+                        };
+                    }
+                }
+                // Just a local name that is not accessible
+                return {
+                    accessibility: 1 /* NotAccessible */,
+                    errorSymbolName: symbolToString(symbol, enclosingDeclaration, meaning),
+                };
             }
-            return false;
+            return { accessibility: 0 /* Accessible */ };
         }
-        /** A type parameter is thisless if its constraint is thisless, or if it has no constraint. */
-        function isThislessTypeParameter(node) {
-            var constraint = ts.getEffectiveConstraintOfTypeParameter(node);
-            return !constraint || isThislessType(constraint);
+        function getExternalModuleContainer(declaration) {
+            var node = ts.findAncestor(declaration, hasExternalModuleSymbol);
+            return node && getSymbolOfNode(node);
         }
-        /**
-         * A variable-like declaration is free of this references if it has a type annotation
-         * that is thisless, or if it has no type annotation and no initializer (and is thus of type any).
-         */
-        function isThislessVariableLikeDeclaration(node) {
-            var typeNode = ts.getEffectiveTypeAnnotationNode(node);
-            return typeNode ? isThislessType(typeNode) : !ts.hasInitializer(node);
+        function hasExternalModuleSymbol(declaration) {
+            return ts.isAmbientModule(declaration) || (declaration.kind === 297 /* SourceFile */ && ts.isExternalOrCommonJsModule(declaration));
         }
-        /**
-         * A function-like declaration is considered free of `this` references if it has a return type
-         * annotation that is free of this references and if each parameter is thisless and if
-         * each type parameter (if present) is thisless.
-         */
-        function isThislessFunctionLikeDeclaration(node) {
-            var returnType = ts.getEffectiveReturnTypeNode(node);
-            var typeParameters = ts.getEffectiveTypeParameterDeclarations(node);
-            return (node.kind === 162 /* Constructor */ || (!!returnType && isThislessType(returnType))) &&
-                node.parameters.every(isThislessVariableLikeDeclaration) &&
-                typeParameters.every(isThislessTypeParameter);
+        function hasNonGlobalAugmentationExternalModuleSymbol(declaration) {
+            return ts.isModuleWithStringLiteralName(declaration) || (declaration.kind === 297 /* SourceFile */ && ts.isExternalOrCommonJsModule(declaration));
         }
-        /**
-         * Returns true if the class or interface member given by the symbol is free of "this" references. The
-         * function may return false for symbols that are actually free of "this" references because it is not
-         * feasible to perform a complete analysis in all cases. In particular, property members with types
-         * inferred from their initializers and function members with inferred return types are conservatively
-         * assumed not to be free of "this" references.
-         */
-        function isThisless(symbol) {
-            if (symbol.declarations && symbol.declarations.length === 1) {
-                var declaration = symbol.declarations[0];
-                if (declaration) {
-                    switch (declaration.kind) {
-                        case 159 /* PropertyDeclaration */:
-                        case 158 /* PropertySignature */:
-                            return isThislessVariableLikeDeclaration(declaration);
-                        case 161 /* MethodDeclaration */:
-                        case 160 /* MethodSignature */:
-                        case 162 /* Constructor */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
-                            return isThislessFunctionLikeDeclaration(declaration);
+        function hasVisibleDeclarations(symbol, shouldComputeAliasToMakeVisible) {
+            var aliasesToMakeVisible;
+            if (!ts.every(ts.filter(symbol.declarations, function (d) { return d.kind !== 78 /* Identifier */; }), getIsDeclarationVisible)) {
+                return undefined;
+            }
+            return { accessibility: 0 /* Accessible */, aliasesToMakeVisible: aliasesToMakeVisible };
+            function getIsDeclarationVisible(declaration) {
+                var _a, _b;
+                if (!isDeclarationVisible(declaration)) {
+                    // Mark the unexported alias as visible if its parent is visible
+                    // because these kind of aliases can be used to name types in declaration file
+                    var anyImportSyntax = getAnyImportSyntax(declaration);
+                    if (anyImportSyntax &&
+                        !ts.hasSyntacticModifier(anyImportSyntax, 1 /* Export */) && // import clause without export
+                        isDeclarationVisible(anyImportSyntax.parent)) {
+                        return addVisibleAlias(declaration, anyImportSyntax);
+                    }
+                    else if (ts.isVariableDeclaration(declaration) && ts.isVariableStatement(declaration.parent.parent) &&
+                        !ts.hasSyntacticModifier(declaration.parent.parent, 1 /* Export */) && // unexported variable statement
+                        isDeclarationVisible(declaration.parent.parent.parent)) {
+                        return addVisibleAlias(declaration, declaration.parent.parent);
+                    }
+                    else if (ts.isLateVisibilityPaintedStatement(declaration) // unexported top-level statement
+                        && !ts.hasSyntacticModifier(declaration, 1 /* Export */)
+                        && isDeclarationVisible(declaration.parent)) {
+                        return addVisibleAlias(declaration, declaration);
                     }
+                    else if (symbol.flags & 2097152 /* Alias */ && ts.isBindingElement(declaration) && ts.isInJSFile(declaration) && ((_a = declaration.parent) === null || _a === void 0 ? void 0 : _a.parent) // exported import-like top-level JS require statement
+                        && ts.isVariableDeclaration(declaration.parent.parent)
+                        && ((_b = declaration.parent.parent.parent) === null || _b === void 0 ? void 0 : _b.parent) && ts.isVariableStatement(declaration.parent.parent.parent.parent)
+                        && !ts.hasSyntacticModifier(declaration.parent.parent.parent.parent, 1 /* Export */)
+                        && declaration.parent.parent.parent.parent.parent // check if the thing containing the variable statement is visible (ie, the file)
+                        && isDeclarationVisible(declaration.parent.parent.parent.parent.parent)) {
+                        return addVisibleAlias(declaration, declaration.parent.parent.parent.parent);
+                    }
+                    // Declaration is not visible
+                    return false;
                 }
+                return true;
             }
-            return false;
-        }
-        // The mappingThisOnly flag indicates that the only type parameter being mapped is "this". When the flag is true,
-        // we check symbols to see if we can quickly conclude they are free of "this" references, thus needing no instantiation.
-        function createInstantiatedSymbolTable(symbols, mapper, mappingThisOnly) {
-            var result = ts.createSymbolTable();
-            for (var _i = 0, symbols_2 = symbols; _i < symbols_2.length; _i++) {
-                var symbol = symbols_2[_i];
-                result.set(symbol.escapedName, mappingThisOnly && isThisless(symbol) ? symbol : instantiateSymbol(symbol, mapper));
-            }
-            return result;
-        }
-        function addInheritedMembers(symbols, baseSymbols) {
-            for (var _i = 0, baseSymbols_1 = baseSymbols; _i < baseSymbols_1.length; _i++) {
-                var s = baseSymbols_1[_i];
-                if (!symbols.has(s.escapedName) && !isStaticPrivateIdentifierProperty(s)) {
-                    symbols.set(s.escapedName, s);
+            function addVisibleAlias(declaration, aliasingStatement) {
+                // In function "buildTypeDisplay" where we decide whether to write type-alias or serialize types,
+                // we want to just check if type- alias is accessible or not but we don't care about emitting those alias at that time
+                // since we will do the emitting later in trackSymbol.
+                if (shouldComputeAliasToMakeVisible) {
+                    getNodeLinks(declaration).isVisible = true;
+                    aliasesToMakeVisible = ts.appendIfUnique(aliasesToMakeVisible, aliasingStatement);
                 }
+                return true;
             }
         }
-        function isStaticPrivateIdentifierProperty(s) {
-            return !!s.valueDeclaration && ts.isPrivateIdentifierPropertyDeclaration(s.valueDeclaration) && ts.hasModifier(s.valueDeclaration, 32 /* Static */);
-        }
-        function resolveDeclaredMembers(type) {
-            if (!type.declaredProperties) {
-                var symbol = type.symbol;
-                var members = getMembersOfSymbol(symbol);
-                type.declaredProperties = getNamedMembers(members);
-                // Start with signatures at empty array in case of recursive types
-                type.declaredCallSignatures = ts.emptyArray;
-                type.declaredConstructSignatures = ts.emptyArray;
-                type.declaredCallSignatures = getSignaturesOfSymbol(members.get("__call" /* Call */));
-                type.declaredConstructSignatures = getSignaturesOfSymbol(members.get("__new" /* New */));
-                type.declaredStringIndexInfo = getIndexInfoOfSymbol(symbol, 0 /* String */);
-                type.declaredNumberIndexInfo = getIndexInfoOfSymbol(symbol, 1 /* Number */);
+        function isEntityNameVisible(entityName, enclosingDeclaration) {
+            // get symbol of the first identifier of the entityName
+            var meaning;
+            if (entityName.parent.kind === 176 /* TypeQuery */ ||
+                ts.isExpressionWithTypeArgumentsInClassExtendsClause(entityName.parent) ||
+                entityName.parent.kind === 158 /* ComputedPropertyName */) {
+                // Typeof value
+                meaning = 111551 /* Value */ | 1048576 /* ExportValue */;
             }
-            return type;
+            else if (entityName.kind === 157 /* QualifiedName */ || entityName.kind === 201 /* PropertyAccessExpression */ ||
+                entityName.parent.kind === 260 /* ImportEqualsDeclaration */) {
+                // Left identifier from type reference or TypeAlias
+                // Entity name of the import declaration
+                meaning = 1920 /* Namespace */;
+            }
+            else {
+                // Type Reference or TypeAlias entity = Identifier
+                meaning = 788968 /* Type */;
+            }
+            var firstIdentifier = ts.getFirstIdentifier(entityName);
+            var symbol = resolveName(enclosingDeclaration, firstIdentifier.escapedText, meaning, /*nodeNotFoundErrorMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
+            if (symbol && symbol.flags & 262144 /* TypeParameter */ && meaning & 788968 /* Type */) {
+                return { accessibility: 0 /* Accessible */ };
+            }
+            // Verify if the symbol is accessible
+            return (symbol && hasVisibleDeclarations(symbol, /*shouldComputeAliasToMakeVisible*/ true)) || {
+                accessibility: 1 /* NotAccessible */,
+                errorSymbolName: ts.getTextOfNode(firstIdentifier),
+                errorNode: firstIdentifier
+            };
         }
-        /**
-         * Indicates whether a type can be used as a property name.
-         */
-        function isTypeUsableAsPropertyName(type) {
-            return !!(type.flags & 8576 /* StringOrNumberLiteralOrUnique */);
+        function symbolToString(symbol, enclosingDeclaration, meaning, flags, writer) {
+            if (flags === void 0) { flags = 4 /* AllowAnyNodeKind */; }
+            var nodeFlags = 70221824 /* IgnoreErrors */;
+            if (flags & 2 /* UseOnlyExternalAliasing */) {
+                nodeFlags |= 128 /* UseOnlyExternalAliasing */;
+            }
+            if (flags & 1 /* WriteTypeParametersOrArguments */) {
+                nodeFlags |= 512 /* WriteTypeParametersInQualifiedName */;
+            }
+            if (flags & 8 /* UseAliasDefinedOutsideCurrentScope */) {
+                nodeFlags |= 16384 /* UseAliasDefinedOutsideCurrentScope */;
+            }
+            if (flags & 16 /* DoNotIncludeSymbolChain */) {
+                nodeFlags |= 134217728 /* DoNotIncludeSymbolChain */;
+            }
+            var builder = flags & 4 /* AllowAnyNodeKind */ ? nodeBuilder.symbolToExpression : nodeBuilder.symbolToEntityName;
+            return writer ? symbolToStringWorker(writer).getText() : ts.usingSingleLineStringWriter(symbolToStringWorker);
+            function symbolToStringWorker(writer) {
+                var entity = builder(symbol, meaning, enclosingDeclaration, nodeFlags); // TODO: GH#18217
+                // add neverAsciiEscape for GH#39027
+                var printer = (enclosingDeclaration === null || enclosingDeclaration === void 0 ? void 0 : enclosingDeclaration.kind) === 297 /* SourceFile */ ? ts.createPrinter({ removeComments: true, neverAsciiEscape: true }) : ts.createPrinter({ removeComments: true });
+                var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
+                printer.writeNode(4 /* Unspecified */, entity, /*sourceFile*/ sourceFile, writer);
+                return writer;
+            }
         }
-        /**
-         * Indicates whether a declaration name is definitely late-bindable.
-         * A declaration name is only late-bindable if:
-         * - It is a `ComputedPropertyName`.
-         * - Its expression is an `Identifier` or either a `PropertyAccessExpression` an
-         * `ElementAccessExpression` consisting only of these same three types of nodes.
-         * - The type of its expression is a string or numeric literal type, or is a `unique symbol` type.
-         */
-        function isLateBindableName(node) {
-            if (!ts.isComputedPropertyName(node) && !ts.isElementAccessExpression(node)) {
-                return false;
+        function signatureToString(signature, enclosingDeclaration, flags, kind, writer) {
+            if (flags === void 0) { flags = 0 /* None */; }
+            return writer ? signatureToStringWorker(writer).getText() : ts.usingSingleLineStringWriter(signatureToStringWorker);
+            function signatureToStringWorker(writer) {
+                var sigOutput;
+                if (flags & 262144 /* WriteArrowStyleSignature */) {
+                    sigOutput = kind === 1 /* Construct */ ? 175 /* ConstructorType */ : 174 /* FunctionType */;
+                }
+                else {
+                    sigOutput = kind === 1 /* Construct */ ? 170 /* ConstructSignature */ : 169 /* CallSignature */;
+                }
+                var sig = nodeBuilder.signatureToSignatureDeclaration(signature, sigOutput, enclosingDeclaration, toNodeBuilderFlags(flags) | 70221824 /* IgnoreErrors */ | 512 /* WriteTypeParametersInQualifiedName */);
+                var printer = ts.createPrinter({ removeComments: true, omitTrailingSemicolon: true });
+                var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
+                printer.writeNode(4 /* Unspecified */, sig, /*sourceFile*/ sourceFile, ts.getTrailingSemicolonDeferringWriter(writer)); // TODO: GH#18217
+                return writer;
             }
-            var expr = ts.isComputedPropertyName(node) ? node.expression : node.argumentExpression;
-            return ts.isEntityNameExpression(expr)
-                && isTypeUsableAsPropertyName(ts.isComputedPropertyName(node) ? checkComputedPropertyName(node) : checkExpressionCached(expr));
         }
-        function isLateBoundName(name) {
-            return name.charCodeAt(0) === 95 /* _ */ &&
-                name.charCodeAt(1) === 95 /* _ */ &&
-                name.charCodeAt(2) === 64 /* at */;
+        function typeToString(type, enclosingDeclaration, flags, writer) {
+            if (flags === void 0) { flags = 1048576 /* AllowUniqueESSymbolType */ | 16384 /* UseAliasDefinedOutsideCurrentScope */; }
+            if (writer === void 0) { writer = ts.createTextWriter(""); }
+            var noTruncation = compilerOptions.noErrorTruncation || flags & 1 /* NoTruncation */;
+            var typeNode = nodeBuilder.typeToTypeNode(type, enclosingDeclaration, toNodeBuilderFlags(flags) | 70221824 /* IgnoreErrors */ | (noTruncation ? 1 /* NoTruncation */ : 0), writer);
+            if (typeNode === undefined)
+                return ts.Debug.fail("should always get typenode");
+            var options = { removeComments: true };
+            var printer = ts.createPrinter(options);
+            var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
+            printer.writeNode(4 /* Unspecified */, typeNode, /*sourceFile*/ sourceFile, writer);
+            var result = writer.getText();
+            var maxLength = noTruncation ? ts.noTruncationMaximumTruncationLength * 2 : ts.defaultMaximumTruncationLength * 2;
+            if (maxLength && result && result.length >= maxLength) {
+                return result.substr(0, maxLength - "...".length) + "...";
+            }
+            return result;
         }
-        /**
-         * Indicates whether a declaration has a late-bindable dynamic name.
-         */
-        function hasLateBindableName(node) {
-            var name = ts.getNameOfDeclaration(node);
-            return !!name && isLateBindableName(name);
+        function getTypeNamesForErrorDisplay(left, right) {
+            var leftStr = symbolValueDeclarationIsContextSensitive(left.symbol) ? typeToString(left, left.symbol.valueDeclaration) : typeToString(left);
+            var rightStr = symbolValueDeclarationIsContextSensitive(right.symbol) ? typeToString(right, right.symbol.valueDeclaration) : typeToString(right);
+            if (leftStr === rightStr) {
+                leftStr = getTypeNameForErrorDisplay(left);
+                rightStr = getTypeNameForErrorDisplay(right);
+            }
+            return [leftStr, rightStr];
         }
-        /**
-         * Indicates whether a declaration has a dynamic name that cannot be late-bound.
-         */
-        function hasNonBindableDynamicName(node) {
-            return ts.hasDynamicName(node) && !hasLateBindableName(node);
+        function getTypeNameForErrorDisplay(type) {
+            return typeToString(type, /*enclosingDeclaration*/ undefined, 64 /* UseFullyQualifiedType */);
         }
-        /**
-         * Indicates whether a declaration name is a dynamic name that cannot be late-bound.
-         */
-        function isNonBindableDynamicName(node) {
-            return ts.isDynamicName(node) && !isLateBindableName(node);
+        function symbolValueDeclarationIsContextSensitive(symbol) {
+            return symbol && symbol.valueDeclaration && ts.isExpression(symbol.valueDeclaration) && !isContextSensitive(symbol.valueDeclaration);
         }
-        /**
-         * Gets the symbolic name for a member from its type.
-         */
-        function getPropertyNameFromType(type) {
-            if (type.flags & 8192 /* UniqueESSymbol */) {
-                return type.escapedName;
-            }
-            if (type.flags & (128 /* StringLiteral */ | 256 /* NumberLiteral */)) {
-                return ts.escapeLeadingUnderscores("" + type.value);
-            }
-            return ts.Debug.fail();
+        function toNodeBuilderFlags(flags) {
+            if (flags === void 0) { flags = 0 /* None */; }
+            return flags & 814775659 /* NodeBuilderFlagsMask */;
         }
-        /**
-         * Adds a declaration to a late-bound dynamic member. This performs the same function for
-         * late-bound members that `addDeclarationToSymbol` in binder.ts performs for early-bound
-         * members.
-         */
-        function addDeclarationToLateBoundSymbol(symbol, member, symbolFlags) {
-            ts.Debug.assert(!!(ts.getCheckFlags(symbol) & 4096 /* Late */), "Expected a late-bound symbol.");
-            symbol.flags |= symbolFlags;
-            getSymbolLinks(member.symbol).lateSymbol = symbol;
-            if (!symbol.declarations) {
-                symbol.declarations = [member];
-            }
-            else {
-                symbol.declarations.push(member);
-            }
-            if (symbolFlags & 111551 /* Value */) {
-                if (!symbol.valueDeclaration || symbol.valueDeclaration.kind !== member.kind) {
-                    symbol.valueDeclaration = member;
-                }
-            }
+        function isClassInstanceSide(type) {
+            return !!type.symbol && !!(type.symbol.flags & 32 /* Class */) && (type === getDeclaredTypeOfClassOrInterface(type.symbol) || !!(ts.getObjectFlags(type) & 1073741824 /* IsClassInstanceClone */));
         }
-        /**
-         * Performs late-binding of a dynamic member. This performs the same function for
-         * late-bound members that `declareSymbol` in binder.ts performs for early-bound
-         * members.
-         *
-         * If a symbol is a dynamic name from a computed property, we perform an additional "late"
-         * binding phase to attempt to resolve the name for the symbol from the type of the computed
-         * property's expression. If the type of the expression is a string-literal, numeric-literal,
-         * or unique symbol type, we can use that type as the name of the symbol.
-         *
-         * For example, given:
-         *
-         *   const x = Symbol();
-         *
-         *   interface I {
-         *     [x]: number;
-         *   }
-         *
-         * The binder gives the property `[x]: number` a special symbol with the name "__computed".
-         * In the late-binding phase we can type-check the expression `x` and see that it has a
-         * unique symbol type which we can then use as the name of the member. This allows users
-         * to define custom symbols that can be used in the members of an object type.
-         *
-         * @param parent The containing symbol for the member.
-         * @param earlySymbols The early-bound symbols of the parent.
-         * @param lateSymbols The late-bound symbols of the parent.
-         * @param decl The member to bind.
-         */
-        function lateBindMember(parent, earlySymbols, lateSymbols, decl) {
-            ts.Debug.assert(!!decl.symbol, "The member is expected to have a symbol.");
-            var links = getNodeLinks(decl);
-            if (!links.resolvedSymbol) {
-                // In the event we attempt to resolve the late-bound name of this member recursively,
-                // fall back to the early-bound name of this member.
-                links.resolvedSymbol = decl.symbol;
-                var declName = ts.isBinaryExpression(decl) ? decl.left : decl.name;
-                var type = ts.isElementAccessExpression(declName) ? checkExpressionCached(declName.argumentExpression) : checkComputedPropertyName(declName);
-                if (isTypeUsableAsPropertyName(type)) {
-                    var memberName = getPropertyNameFromType(type);
-                    var symbolFlags = decl.symbol.flags;
-                    // Get or add a late-bound symbol for the member. This allows us to merge late-bound accessor declarations.
-                    var lateSymbol = lateSymbols.get(memberName);
-                    if (!lateSymbol)
-                        lateSymbols.set(memberName, lateSymbol = createSymbol(0 /* None */, memberName, 4096 /* Late */));
-                    // Report an error if a late-bound member has the same name as an early-bound member,
-                    // or if we have another early-bound symbol declaration with the same name and
-                    // conflicting flags.
-                    var earlySymbol = earlySymbols && earlySymbols.get(memberName);
-                    if (lateSymbol.flags & getExcludedSymbolFlags(symbolFlags) || earlySymbol) {
-                        // If we have an existing early-bound member, combine its declarations so that we can
-                        // report an error at each declaration.
-                        var declarations = earlySymbol ? ts.concatenate(earlySymbol.declarations, lateSymbol.declarations) : lateSymbol.declarations;
-                        var name_3 = !(type.flags & 8192 /* UniqueESSymbol */) && ts.unescapeLeadingUnderscores(memberName) || ts.declarationNameToString(declName);
-                        ts.forEach(declarations, function (declaration) { return error(ts.getNameOfDeclaration(declaration) || declaration, ts.Diagnostics.Property_0_was_also_declared_here, name_3); });
-                        error(declName || decl, ts.Diagnostics.Duplicate_property_0, name_3);
-                        lateSymbol = createSymbol(0 /* None */, memberName, 4096 /* Late */);
+        function createNodeBuilder() {
+            return {
+                typeToTypeNode: function (type, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return typeToTypeNodeHelper(type, context); });
+                },
+                indexInfoToIndexSignatureDeclaration: function (indexInfo, kind, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return indexInfoToIndexSignatureDeclarationHelper(indexInfo, kind, context, /*typeNode*/ undefined); });
+                },
+                signatureToSignatureDeclaration: function (signature, kind, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return signatureToSignatureDeclarationHelper(signature, kind, context); });
+                },
+                symbolToEntityName: function (symbol, meaning, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolToName(symbol, context, meaning, /*expectsIdentifier*/ false); });
+                },
+                symbolToExpression: function (symbol, meaning, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolToExpression(symbol, context, meaning); });
+                },
+                symbolToTypeParameterDeclarations: function (symbol, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return typeParametersToTypeParameterDeclarations(symbol, context); });
+                },
+                symbolToParameterDeclaration: function (symbol, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolToParameterDeclaration(symbol, context); });
+                },
+                typeParameterToDeclaration: function (parameter, enclosingDeclaration, flags, tracker) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return typeParameterToDeclaration(parameter, context); });
+                },
+                symbolTableToDeclarationStatements: function (symbolTable, enclosingDeclaration, flags, tracker, bundled) {
+                    return withContext(enclosingDeclaration, flags, tracker, function (context) { return symbolTableToDeclarationStatements(symbolTable, context, bundled); });
+                },
+            };
+            function withContext(enclosingDeclaration, flags, tracker, cb) {
+                var _a, _b;
+                ts.Debug.assert(enclosingDeclaration === undefined || (enclosingDeclaration.flags & 8 /* Synthesized */) === 0);
+                var context = {
+                    enclosingDeclaration: enclosingDeclaration,
+                    flags: flags || 0 /* None */,
+                    // If no full tracker is provided, fake up a dummy one with a basic limited-functionality moduleResolverHost
+                    tracker: tracker && tracker.trackSymbol ? tracker : { trackSymbol: ts.noop, moduleResolverHost: flags & 134217728 /* DoNotIncludeSymbolChain */ ? {
+                            getCommonSourceDirectory: !!host.getCommonSourceDirectory ? function () { return host.getCommonSourceDirectory(); } : function () { return ""; },
+                            getSourceFiles: function () { return host.getSourceFiles(); },
+                            getCurrentDirectory: function () { return host.getCurrentDirectory(); },
+                            getSymlinkCache: ts.maybeBind(host, host.getSymlinkCache),
+                            useCaseSensitiveFileNames: ts.maybeBind(host, host.useCaseSensitiveFileNames),
+                            redirectTargetsMap: host.redirectTargetsMap,
+                            getProjectReferenceRedirect: function (fileName) { return host.getProjectReferenceRedirect(fileName); },
+                            isSourceOfProjectReferenceRedirect: function (fileName) { return host.isSourceOfProjectReferenceRedirect(fileName); },
+                            fileExists: function (fileName) { return host.fileExists(fileName); },
+                        } : undefined },
+                    encounteredError: false,
+                    visitedTypes: undefined,
+                    symbolDepth: undefined,
+                    inferTypeParameters: undefined,
+                    approximateLength: 0
+                };
+                var resultingNode = cb(context);
+                if (context.truncating && context.flags & 1 /* NoTruncation */) {
+                    (_b = (_a = context.tracker) === null || _a === void 0 ? void 0 : _a.reportTruncationError) === null || _b === void 0 ? void 0 : _b.call(_a);
+                }
+                return context.encounteredError ? undefined : resultingNode;
+            }
+            function checkTruncationLength(context) {
+                if (context.truncating)
+                    return context.truncating;
+                return context.truncating = context.approximateLength > ((context.flags & 1 /* NoTruncation */) ? ts.noTruncationMaximumTruncationLength : ts.defaultMaximumTruncationLength);
+            }
+            function typeToTypeNodeHelper(type, context) {
+                if (cancellationToken && cancellationToken.throwIfCancellationRequested) {
+                    cancellationToken.throwIfCancellationRequested();
+                }
+                var inTypeAlias = context.flags & 8388608 /* InTypeAlias */;
+                context.flags &= ~8388608 /* InTypeAlias */;
+                if (!type) {
+                    if (!(context.flags & 262144 /* AllowEmptyUnionOrIntersection */)) {
+                        context.encounteredError = true;
+                        return undefined; // TODO: GH#18217
                     }
-                    lateSymbol.nameType = type;
-                    addDeclarationToLateBoundSymbol(lateSymbol, decl, symbolFlags);
-                    if (lateSymbol.parent) {
-                        ts.Debug.assert(lateSymbol.parent === parent, "Existing symbol parent should match new one");
+                    context.approximateLength += 3;
+                    return ts.factory.createKeywordTypeNode(128 /* AnyKeyword */);
+                }
+                if (!(context.flags & 536870912 /* NoTypeReduction */)) {
+                    type = getReducedType(type);
+                }
+                if (type.flags & 1 /* Any */) {
+                    context.approximateLength += 3;
+                    return ts.factory.createKeywordTypeNode(type === intrinsicMarkerType ? 136 /* IntrinsicKeyword */ : 128 /* AnyKeyword */);
+                }
+                if (type.flags & 2 /* Unknown */) {
+                    return ts.factory.createKeywordTypeNode(152 /* UnknownKeyword */);
+                }
+                if (type.flags & 4 /* String */) {
+                    context.approximateLength += 6;
+                    return ts.factory.createKeywordTypeNode(147 /* StringKeyword */);
+                }
+                if (type.flags & 8 /* Number */) {
+                    context.approximateLength += 6;
+                    return ts.factory.createKeywordTypeNode(144 /* NumberKeyword */);
+                }
+                if (type.flags & 64 /* BigInt */) {
+                    context.approximateLength += 6;
+                    return ts.factory.createKeywordTypeNode(155 /* BigIntKeyword */);
+                }
+                if (type.flags & 16 /* Boolean */) {
+                    context.approximateLength += 7;
+                    return ts.factory.createKeywordTypeNode(131 /* BooleanKeyword */);
+                }
+                if (type.flags & 1024 /* EnumLiteral */ && !(type.flags & 1048576 /* Union */)) {
+                    var parentSymbol = getParentOfSymbol(type.symbol);
+                    var parentName = symbolToTypeNode(parentSymbol, context, 788968 /* Type */);
+                    if (getDeclaredTypeOfSymbol(parentSymbol) === type) {
+                        return parentName;
+                    }
+                    var memberName = ts.symbolName(type.symbol);
+                    if (ts.isIdentifierText(memberName, 0 /* ES3 */)) {
+                        return appendReferenceToType(parentName, ts.factory.createTypeReferenceNode(memberName, /*typeArguments*/ undefined));
+                    }
+                    if (ts.isImportTypeNode(parentName)) {
+                        parentName.isTypeOf = true; // mutably update, node is freshly manufactured anyhow
+                        return ts.factory.createIndexedAccessTypeNode(parentName, ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral(memberName)));
+                    }
+                    else if (ts.isTypeReferenceNode(parentName)) {
+                        return ts.factory.createIndexedAccessTypeNode(ts.factory.createTypeQueryNode(parentName.typeName), ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral(memberName)));
                     }
                     else {
-                        lateSymbol.parent = parent;
+                        return ts.Debug.fail("Unhandled type node kind returned from `symbolToTypeNode`.");
                     }
-                    return links.resolvedSymbol = lateSymbol;
                 }
-            }
-            return links.resolvedSymbol;
-        }
-        function getResolvedMembersOrExportsOfSymbol(symbol, resolutionKind) {
-            var links = getSymbolLinks(symbol);
-            if (!links[resolutionKind]) {
-                var isStatic = resolutionKind === "resolvedExports" /* resolvedExports */;
-                var earlySymbols = !isStatic ? symbol.members :
-                    symbol.flags & 1536 /* Module */ ? getExportsOfModuleWorker(symbol) :
-                        symbol.exports;
-                // In the event we recursively resolve the members/exports of the symbol, we
-                // set the initial value of resolvedMembers/resolvedExports to the early-bound
-                // members/exports of the symbol.
-                links[resolutionKind] = earlySymbols || emptySymbols;
-                // fill in any as-yet-unresolved late-bound members.
-                var lateSymbols = ts.createSymbolTable();
-                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                    var decl = _a[_i];
-                    var members = ts.getMembersOfDeclaration(decl);
-                    if (members) {
-                        for (var _b = 0, members_5 = members; _b < members_5.length; _b++) {
-                            var member = members_5[_b];
-                            if (isStatic === ts.hasStaticModifier(member) && hasLateBindableName(member)) {
-                                lateBindMember(symbol, earlySymbols, lateSymbols, member);
-                            }
+                if (type.flags & 1056 /* EnumLike */) {
+                    return symbolToTypeNode(type.symbol, context, 788968 /* Type */);
+                }
+                if (type.flags & 128 /* StringLiteral */) {
+                    context.approximateLength += (type.value.length + 2);
+                    return ts.factory.createLiteralTypeNode(ts.setEmitFlags(ts.factory.createStringLiteral(type.value, !!(context.flags & 268435456 /* UseSingleQuotesForStringLiteralType */)), 16777216 /* NoAsciiEscaping */));
+                }
+                if (type.flags & 256 /* NumberLiteral */) {
+                    var value = type.value;
+                    context.approximateLength += ("" + value).length;
+                    return ts.factory.createLiteralTypeNode(value < 0 ? ts.factory.createPrefixUnaryExpression(40 /* MinusToken */, ts.factory.createNumericLiteral(-value)) : ts.factory.createNumericLiteral(value));
+                }
+                if (type.flags & 2048 /* BigIntLiteral */) {
+                    context.approximateLength += (ts.pseudoBigIntToString(type.value).length) + 1;
+                    return ts.factory.createLiteralTypeNode((ts.factory.createBigIntLiteral(type.value)));
+                }
+                if (type.flags & 512 /* BooleanLiteral */) {
+                    context.approximateLength += type.intrinsicName.length;
+                    return ts.factory.createLiteralTypeNode(type.intrinsicName === "true" ? ts.factory.createTrue() : ts.factory.createFalse());
+                }
+                if (type.flags & 8192 /* UniqueESSymbol */) {
+                    if (!(context.flags & 1048576 /* AllowUniqueESSymbolType */)) {
+                        if (isValueSymbolAccessible(type.symbol, context.enclosingDeclaration)) {
+                            context.approximateLength += 6;
+                            return symbolToTypeNode(type.symbol, context, 111551 /* Value */);
+                        }
+                        if (context.tracker.reportInaccessibleUniqueSymbolError) {
+                            context.tracker.reportInaccessibleUniqueSymbolError();
                         }
                     }
+                    context.approximateLength += 13;
+                    return ts.factory.createTypeOperatorNode(151 /* UniqueKeyword */, ts.factory.createKeywordTypeNode(148 /* SymbolKeyword */));
                 }
-                var assignments = symbol.assignmentDeclarationMembers;
-                if (assignments) {
-                    var decls = ts.arrayFrom(assignments.values());
-                    for (var _c = 0, decls_1 = decls; _c < decls_1.length; _c++) {
-                        var member = decls_1[_c];
-                        var assignmentKind = ts.getAssignmentDeclarationKind(member);
-                        var isInstanceMember = assignmentKind === 3 /* PrototypeProperty */
-                            || assignmentKind === 4 /* ThisProperty */
-                            || assignmentKind === 9 /* ObjectDefinePrototypeProperty */
-                            || assignmentKind === 6 /* Prototype */; // A straight `Prototype` assignment probably can never have a computed name
-                        if (isStatic === !isInstanceMember && hasLateBindableName(member)) {
-                            lateBindMember(symbol, earlySymbols, lateSymbols, member);
+                if (type.flags & 16384 /* Void */) {
+                    context.approximateLength += 4;
+                    return ts.factory.createKeywordTypeNode(113 /* VoidKeyword */);
+                }
+                if (type.flags & 32768 /* Undefined */) {
+                    context.approximateLength += 9;
+                    return ts.factory.createKeywordTypeNode(150 /* UndefinedKeyword */);
+                }
+                if (type.flags & 65536 /* Null */) {
+                    context.approximateLength += 4;
+                    return ts.factory.createLiteralTypeNode(ts.factory.createNull());
+                }
+                if (type.flags & 131072 /* Never */) {
+                    context.approximateLength += 5;
+                    return ts.factory.createKeywordTypeNode(141 /* NeverKeyword */);
+                }
+                if (type.flags & 4096 /* ESSymbol */) {
+                    context.approximateLength += 6;
+                    return ts.factory.createKeywordTypeNode(148 /* SymbolKeyword */);
+                }
+                if (type.flags & 67108864 /* NonPrimitive */) {
+                    context.approximateLength += 6;
+                    return ts.factory.createKeywordTypeNode(145 /* ObjectKeyword */);
+                }
+                if (isThisTypeParameter(type)) {
+                    if (context.flags & 4194304 /* InObjectTypeLiteral */) {
+                        if (!context.encounteredError && !(context.flags & 32768 /* AllowThisInObjectLiteral */)) {
+                            context.encounteredError = true;
+                        }
+                        if (context.tracker.reportInaccessibleThisError) {
+                            context.tracker.reportInaccessibleThisError();
                         }
                     }
+                    context.approximateLength += 4;
+                    return ts.factory.createThisTypeNode();
                 }
-                links[resolutionKind] = combineSymbolTables(earlySymbols, lateSymbols) || emptySymbols;
-            }
-            return links[resolutionKind];
-        }
-        /**
-         * Gets a SymbolTable containing both the early- and late-bound members of a symbol.
-         *
-         * For a description of late-binding, see `lateBindMember`.
-         */
-        function getMembersOfSymbol(symbol) {
-            return symbol.flags & 6256 /* LateBindingContainer */
-                ? getResolvedMembersOrExportsOfSymbol(symbol, "resolvedMembers" /* resolvedMembers */)
-                : symbol.members || emptySymbols;
-        }
-        /**
-         * If a symbol is the dynamic name of the member of an object type, get the late-bound
-         * symbol of the member.
-         *
-         * For a description of late-binding, see `lateBindMember`.
-         */
-        function getLateBoundSymbol(symbol) {
-            if (symbol.flags & 106500 /* ClassMember */ && symbol.escapedName === "__computed" /* Computed */) {
-                var links = getSymbolLinks(symbol);
-                if (!links.lateSymbol && ts.some(symbol.declarations, hasLateBindableName)) {
-                    // force late binding of members/exports. This will set the late-bound symbol
-                    var parent = getMergedSymbol(symbol.parent);
-                    if (ts.some(symbol.declarations, ts.hasStaticModifier)) {
-                        getExportsOfSymbol(parent);
+                if (!inTypeAlias && type.aliasSymbol && (context.flags & 16384 /* UseAliasDefinedOutsideCurrentScope */ || isTypeSymbolAccessible(type.aliasSymbol, context.enclosingDeclaration))) {
+                    var typeArgumentNodes = mapToTypeNodes(type.aliasTypeArguments, context);
+                    if (isReservedMemberName(type.aliasSymbol.escapedName) && !(type.aliasSymbol.flags & 32 /* Class */))
+                        return ts.factory.createTypeReferenceNode(ts.factory.createIdentifier(""), typeArgumentNodes);
+                    return symbolToTypeNode(type.aliasSymbol, context, 788968 /* Type */, typeArgumentNodes);
+                }
+                var objectFlags = ts.getObjectFlags(type);
+                if (objectFlags & 4 /* Reference */) {
+                    ts.Debug.assert(!!(type.flags & 524288 /* Object */));
+                    return type.node ? visitAndTransformType(type, typeReferenceToTypeNode) : typeReferenceToTypeNode(type);
+                }
+                if (type.flags & 262144 /* TypeParameter */ || objectFlags & 3 /* ClassOrInterface */) {
+                    if (type.flags & 262144 /* TypeParameter */ && ts.contains(context.inferTypeParameters, type)) {
+                        context.approximateLength += (ts.symbolName(type.symbol).length + 6);
+                        return ts.factory.createInferTypeNode(typeParameterToDeclarationWithConstraint(type, context, /*constraintNode*/ undefined));
+                    }
+                    if (context.flags & 4 /* GenerateNamesForShadowedTypeParams */ &&
+                        type.flags & 262144 /* TypeParameter */ &&
+                        !isTypeSymbolAccessible(type.symbol, context.enclosingDeclaration)) {
+                        var name = typeParameterToName(type, context);
+                        context.approximateLength += ts.idText(name).length;
+                        return ts.factory.createTypeReferenceNode(ts.factory.createIdentifier(ts.idText(name)), /*typeArguments*/ undefined);
+                    }
+                    // Ignore constraint/default when creating a usage (as opposed to declaration) of a type parameter.
+                    return type.symbol
+                        ? symbolToTypeNode(type.symbol, context, 788968 /* Type */)
+                        : ts.factory.createTypeReferenceNode(ts.factory.createIdentifier("?"), /*typeArguments*/ undefined);
+                }
+                if (type.flags & (1048576 /* Union */ | 2097152 /* Intersection */)) {
+                    var types = type.flags & 1048576 /* Union */ ? formatUnionTypes(type.types) : type.types;
+                    if (ts.length(types) === 1) {
+                        return typeToTypeNodeHelper(types[0], context);
+                    }
+                    var typeNodes = mapToTypeNodes(types, context, /*isBareList*/ true);
+                    if (typeNodes && typeNodes.length > 0) {
+                        var unionOrIntersectionTypeNode = type.flags & 1048576 /* Union */ ? ts.factory.createUnionTypeNode(typeNodes) : ts.factory.createIntersectionTypeNode(typeNodes);
+                        return unionOrIntersectionTypeNode;
                     }
                     else {
-                        getMembersOfSymbol(parent);
+                        if (!context.encounteredError && !(context.flags & 262144 /* AllowEmptyUnionOrIntersection */)) {
+                            context.encounteredError = true;
+                        }
+                        return undefined; // TODO: GH#18217
                     }
                 }
-                return links.lateSymbol || (links.lateSymbol = symbol);
-            }
-            return symbol;
-        }
-        function getTypeWithThisArgument(type, thisArgument, needApparentType) {
-            if (ts.getObjectFlags(type) & 4 /* Reference */) {
-                var target = type.target;
-                var typeArguments = getTypeArguments(type);
-                if (ts.length(target.typeParameters) === ts.length(typeArguments)) {
-                    var ref = createTypeReference(target, ts.concatenate(typeArguments, [thisArgument || target.thisType]));
-                    return needApparentType ? getApparentType(ref) : ref;
+                if (objectFlags & (16 /* Anonymous */ | 32 /* Mapped */)) {
+                    ts.Debug.assert(!!(type.flags & 524288 /* Object */));
+                    // The type is an object literal type.
+                    return createAnonymousTypeNode(type);
                 }
-            }
-            else if (type.flags & 2097152 /* Intersection */) {
-                return getIntersectionType(ts.map(type.types, function (t) { return getTypeWithThisArgument(t, thisArgument, needApparentType); }));
-            }
-            return needApparentType ? getApparentType(type) : type;
-        }
-        function resolveObjectTypeMembers(type, source, typeParameters, typeArguments) {
-            var mapper;
-            var members;
-            var callSignatures;
-            var constructSignatures;
-            var stringIndexInfo;
-            var numberIndexInfo;
-            if (ts.rangeEquals(typeParameters, typeArguments, 0, typeParameters.length)) {
-                members = source.symbol ? getMembersOfSymbol(source.symbol) : ts.createSymbolTable(source.declaredProperties);
-                callSignatures = source.declaredCallSignatures;
-                constructSignatures = source.declaredConstructSignatures;
-                stringIndexInfo = source.declaredStringIndexInfo;
-                numberIndexInfo = source.declaredNumberIndexInfo;
-            }
-            else {
-                mapper = createTypeMapper(typeParameters, typeArguments);
-                members = createInstantiatedSymbolTable(source.declaredProperties, mapper, /*mappingThisOnly*/ typeParameters.length === 1);
-                callSignatures = instantiateSignatures(source.declaredCallSignatures, mapper);
-                constructSignatures = instantiateSignatures(source.declaredConstructSignatures, mapper);
-                stringIndexInfo = instantiateIndexInfo(source.declaredStringIndexInfo, mapper);
-                numberIndexInfo = instantiateIndexInfo(source.declaredNumberIndexInfo, mapper);
-            }
-            var baseTypes = getBaseTypes(source);
-            if (baseTypes.length) {
-                if (source.symbol && members === getMembersOfSymbol(source.symbol)) {
-                    members = ts.createSymbolTable(source.declaredProperties);
+                if (type.flags & 4194304 /* Index */) {
+                    var indexedType = type.type;
+                    context.approximateLength += 6;
+                    var indexTypeNode = typeToTypeNodeHelper(indexedType, context);
+                    return ts.factory.createTypeOperatorNode(138 /* KeyOfKeyword */, indexTypeNode);
                 }
-                setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
-                var thisArgument = ts.lastOrUndefined(typeArguments);
-                for (var _i = 0, baseTypes_1 = baseTypes; _i < baseTypes_1.length; _i++) {
-                    var baseType = baseTypes_1[_i];
-                    var instantiatedBaseType = thisArgument ? getTypeWithThisArgument(instantiateType(baseType, mapper), thisArgument) : baseType;
-                    addInheritedMembers(members, getPropertiesOfType(instantiatedBaseType));
-                    callSignatures = ts.concatenate(callSignatures, getSignaturesOfType(instantiatedBaseType, 0 /* Call */));
-                    constructSignatures = ts.concatenate(constructSignatures, getSignaturesOfType(instantiatedBaseType, 1 /* Construct */));
-                    if (!stringIndexInfo) {
-                        stringIndexInfo = instantiatedBaseType === anyType ?
-                            createIndexInfo(anyType, /*isReadonly*/ false) :
-                            getIndexInfoOfType(instantiatedBaseType, 0 /* String */);
-                    }
-                    numberIndexInfo = numberIndexInfo || getIndexInfoOfType(instantiatedBaseType, 1 /* Number */);
+                if (type.flags & 134217728 /* TemplateLiteral */) {
+                    var texts_1 = type.texts;
+                    var types_1 = type.types;
+                    var templateHead = ts.factory.createTemplateHead(texts_1[0]);
+                    var templateSpans = ts.factory.createNodeArray(ts.map(types_1, function (t, i) { return ts.factory.createTemplateLiteralTypeSpan(typeToTypeNodeHelper(t, context), (i < types_1.length - 1 ? ts.factory.createTemplateMiddle : ts.factory.createTemplateTail)(texts_1[i + 1])); }));
+                    context.approximateLength += 2;
+                    return ts.factory.createTemplateLiteralType(templateHead, templateSpans);
                 }
-            }
-            setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
-        }
-        function resolveClassOrInterfaceMembers(type) {
-            resolveObjectTypeMembers(type, resolveDeclaredMembers(type), ts.emptyArray, ts.emptyArray);
-        }
-        function resolveTypeReferenceMembers(type) {
-            var source = resolveDeclaredMembers(type.target);
-            var typeParameters = ts.concatenate(source.typeParameters, [source.thisType]);
-            var typeArguments = getTypeArguments(type);
-            var paddedTypeArguments = typeArguments.length === typeParameters.length ? typeArguments : ts.concatenate(typeArguments, [type]);
-            resolveObjectTypeMembers(type, source, typeParameters, paddedTypeArguments);
-        }
-        function createSignature(declaration, typeParameters, thisParameter, parameters, resolvedReturnType, resolvedTypePredicate, minArgumentCount, flags) {
-            var sig = new Signature(checker, flags);
-            sig.declaration = declaration;
-            sig.typeParameters = typeParameters;
-            sig.parameters = parameters;
-            sig.thisParameter = thisParameter;
-            sig.resolvedReturnType = resolvedReturnType;
-            sig.resolvedTypePredicate = resolvedTypePredicate;
-            sig.minArgumentCount = minArgumentCount;
-            sig.target = undefined;
-            sig.mapper = undefined;
-            sig.unionSignatures = undefined;
-            return sig;
-        }
-        function cloneSignature(sig) {
-            var result = createSignature(sig.declaration, sig.typeParameters, sig.thisParameter, sig.parameters, /*resolvedReturnType*/ undefined, 
-            /*resolvedTypePredicate*/ undefined, sig.minArgumentCount, sig.flags & 3 /* PropagatingFlags */);
-            result.target = sig.target;
-            result.mapper = sig.mapper;
-            result.unionSignatures = sig.unionSignatures;
-            return result;
-        }
-        function createUnionSignature(signature, unionSignatures) {
-            var result = cloneSignature(signature);
-            result.unionSignatures = unionSignatures;
-            result.target = undefined;
-            result.mapper = undefined;
-            return result;
-        }
-        function getOptionalCallSignature(signature, callChainFlags) {
-            if ((signature.flags & 12 /* CallChainFlags */) === callChainFlags) {
-                return signature;
-            }
-            if (!signature.optionalCallSignatureCache) {
-                signature.optionalCallSignatureCache = {};
-            }
-            var key = callChainFlags === 4 /* IsInnerCallChain */ ? "inner" : "outer";
-            return signature.optionalCallSignatureCache[key]
-                || (signature.optionalCallSignatureCache[key] = createOptionalCallSignature(signature, callChainFlags));
-        }
-        function createOptionalCallSignature(signature, callChainFlags) {
-            ts.Debug.assert(callChainFlags === 4 /* IsInnerCallChain */ || callChainFlags === 8 /* IsOuterCallChain */, "An optional call signature can either be for an inner call chain or an outer call chain, but not both.");
-            var result = cloneSignature(signature);
-            result.flags |= callChainFlags;
-            return result;
-        }
-        function getExpandedParameters(sig) {
-            if (signatureHasRestParameter(sig)) {
-                var restIndex_1 = sig.parameters.length - 1;
-                var restParameter = sig.parameters[restIndex_1];
-                var restType = getTypeOfSymbol(restParameter);
-                if (isTupleType(restType)) {
-                    var elementTypes = getTypeArguments(restType);
-                    var minLength_1 = restType.target.minLength;
-                    var tupleRestIndex_1 = restType.target.hasRestElement ? elementTypes.length - 1 : -1;
-                    var restParams = ts.map(elementTypes, function (t, i) {
-                        var name = getParameterNameAtPosition(sig, restIndex_1 + i);
-                        var checkFlags = i === tupleRestIndex_1 ? 32768 /* RestParameter */ :
-                            i >= minLength_1 ? 16384 /* OptionalParameter */ : 0;
-                        var symbol = createSymbol(1 /* FunctionScopedVariable */, name, checkFlags);
-                        symbol.type = i === tupleRestIndex_1 ? createArrayType(t) : t;
-                        return symbol;
-                    });
-                    return ts.concatenate(sig.parameters.slice(0, restIndex_1), restParams);
+                if (type.flags & 268435456 /* StringMapping */) {
+                    var typeNode = typeToTypeNodeHelper(type.type, context);
+                    return symbolToTypeNode(type.symbol, context, 788968 /* Type */, [typeNode]);
                 }
-            }
-            return sig.parameters;
-        }
-        function getDefaultConstructSignatures(classType) {
-            var baseConstructorType = getBaseConstructorTypeOfClass(classType);
-            var baseSignatures = getSignaturesOfType(baseConstructorType, 1 /* Construct */);
-            if (baseSignatures.length === 0) {
-                return [createSignature(undefined, classType.localTypeParameters, undefined, ts.emptyArray, classType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */)];
-            }
-            var baseTypeNode = getBaseTypeNodeOfClass(classType);
-            var isJavaScript = ts.isInJSFile(baseTypeNode);
-            var typeArguments = typeArgumentsFromTypeReferenceNode(baseTypeNode);
-            var typeArgCount = ts.length(typeArguments);
-            var result = [];
-            for (var _i = 0, baseSignatures_1 = baseSignatures; _i < baseSignatures_1.length; _i++) {
-                var baseSig = baseSignatures_1[_i];
-                var minTypeArgumentCount = getMinTypeArgumentCount(baseSig.typeParameters);
-                var typeParamCount = ts.length(baseSig.typeParameters);
-                if (isJavaScript || typeArgCount >= minTypeArgumentCount && typeArgCount <= typeParamCount) {
-                    var sig = typeParamCount ? createSignatureInstantiation(baseSig, fillMissingTypeArguments(typeArguments, baseSig.typeParameters, minTypeArgumentCount, isJavaScript)) : cloneSignature(baseSig);
-                    sig.typeParameters = classType.localTypeParameters;
-                    sig.resolvedReturnType = classType;
-                    result.push(sig);
+                if (type.flags & 8388608 /* IndexedAccess */) {
+                    var objectTypeNode = typeToTypeNodeHelper(type.objectType, context);
+                    var indexTypeNode = typeToTypeNodeHelper(type.indexType, context);
+                    context.approximateLength += 2;
+                    return ts.factory.createIndexedAccessTypeNode(objectTypeNode, indexTypeNode);
                 }
-            }
-            return result;
-        }
-        function findMatchingSignature(signatureList, signature, partialMatch, ignoreThisTypes, ignoreReturnTypes) {
-            for (var _i = 0, signatureList_1 = signatureList; _i < signatureList_1.length; _i++) {
-                var s = signatureList_1[_i];
-                if (compareSignaturesIdentical(s, signature, partialMatch, ignoreThisTypes, ignoreReturnTypes, partialMatch ? compareTypesSubtypeOf : compareTypesIdentical)) {
-                    return s;
+                if (type.flags & 16777216 /* Conditional */) {
+                    var checkTypeNode = typeToTypeNodeHelper(type.checkType, context);
+                    var saveInferTypeParameters = context.inferTypeParameters;
+                    context.inferTypeParameters = type.root.inferTypeParameters;
+                    var extendsTypeNode = typeToTypeNodeHelper(type.extendsType, context);
+                    context.inferTypeParameters = saveInferTypeParameters;
+                    var trueTypeNode = typeToTypeNodeOrCircularityElision(getTrueTypeFromConditionalType(type));
+                    var falseTypeNode = typeToTypeNodeOrCircularityElision(getFalseTypeFromConditionalType(type));
+                    context.approximateLength += 15;
+                    return ts.factory.createConditionalTypeNode(checkTypeNode, extendsTypeNode, trueTypeNode, falseTypeNode);
                 }
-            }
-        }
-        function findMatchingSignatures(signatureLists, signature, listIndex) {
-            if (signature.typeParameters) {
-                // We require an exact match for generic signatures, so we only return signatures from the first
-                // signature list and only if they have exact matches in the other signature lists.
-                if (listIndex > 0) {
-                    return undefined;
+                if (type.flags & 33554432 /* Substitution */) {
+                    return typeToTypeNodeHelper(type.baseType, context);
                 }
-                for (var i = 1; i < signatureLists.length; i++) {
-                    if (!findMatchingSignature(signatureLists[i], signature, /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ false)) {
-                        return undefined;
+                return ts.Debug.fail("Should be unreachable.");
+                function typeToTypeNodeOrCircularityElision(type) {
+                    var _a, _b, _c;
+                    if (type.flags & 1048576 /* Union */) {
+                        if ((_a = context.visitedTypes) === null || _a === void 0 ? void 0 : _a.has(getTypeId(type))) {
+                            if (!(context.flags & 131072 /* AllowAnonymousIdentifier */)) {
+                                context.encounteredError = true;
+                                (_c = (_b = context.tracker) === null || _b === void 0 ? void 0 : _b.reportCyclicStructureError) === null || _c === void 0 ? void 0 : _c.call(_b);
+                            }
+                            return createElidedInformationPlaceholder(context);
+                        }
+                        return visitAndTransformType(type, function (type) { return typeToTypeNodeHelper(type, context); });
                     }
+                    return typeToTypeNodeHelper(type, context);
                 }
-                return [signature];
-            }
-            var result;
-            for (var i = 0; i < signatureLists.length; i++) {
-                // Allow matching non-generic signatures to have excess parameters and different return types.
-                // Prefer matching this types if possible.
-                var match = i === listIndex ? signature : findMatchingSignature(signatureLists[i], signature, /*partialMatch*/ true, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ true);
-                if (!match) {
-                    return undefined;
+                function createMappedTypeNodeFromType(type) {
+                    ts.Debug.assert(!!(type.flags & 524288 /* Object */));
+                    var readonlyToken = type.declaration.readonlyToken ? ts.factory.createToken(type.declaration.readonlyToken.kind) : undefined;
+                    var questionToken = type.declaration.questionToken ? ts.factory.createToken(type.declaration.questionToken.kind) : undefined;
+                    var appropriateConstraintTypeNode;
+                    if (isMappedTypeWithKeyofConstraintDeclaration(type)) {
+                        // We have a { [P in keyof T]: X }
+                        // We do this to ensure we retain the toplevel keyof-ness of the type which may be lost due to keyof distribution during `getConstraintTypeFromMappedType`
+                        appropriateConstraintTypeNode = ts.factory.createTypeOperatorNode(138 /* KeyOfKeyword */, typeToTypeNodeHelper(getModifiersTypeFromMappedType(type), context));
+                    }
+                    else {
+                        appropriateConstraintTypeNode = typeToTypeNodeHelper(getConstraintTypeFromMappedType(type), context);
+                    }
+                    var typeParameterNode = typeParameterToDeclarationWithConstraint(getTypeParameterFromMappedType(type), context, appropriateConstraintTypeNode);
+                    var nameTypeNode = type.declaration.nameType ? typeToTypeNodeHelper(getNameTypeFromMappedType(type), context) : undefined;
+                    var templateTypeNode = typeToTypeNodeHelper(getTemplateTypeFromMappedType(type), context);
+                    var mappedTypeNode = ts.factory.createMappedTypeNode(readonlyToken, typeParameterNode, nameTypeNode, questionToken, templateTypeNode);
+                    context.approximateLength += 10;
+                    return ts.setEmitFlags(mappedTypeNode, 1 /* SingleLine */);
                 }
-                result = ts.appendIfUnique(result, match);
-            }
-            return result;
-        }
-        // The signatures of a union type are those signatures that are present in each of the constituent types.
-        // Generic signatures must match exactly, but non-generic signatures are allowed to have extra optional
-        // parameters and may differ in return types. When signatures differ in return types, the resulting return
-        // type is the union of the constituent return types.
-        function getUnionSignatures(signatureLists) {
-            var result;
-            var indexWithLengthOverOne;
-            for (var i = 0; i < signatureLists.length; i++) {
-                if (signatureLists[i].length === 0)
-                    return ts.emptyArray;
-                if (signatureLists[i].length > 1) {
-                    indexWithLengthOverOne = indexWithLengthOverOne === undefined ? i : -1; // -1 is a signal there are multiple overload sets
+                function createAnonymousTypeNode(type) {
+                    var _a;
+                    var typeId = type.id;
+                    var symbol = type.symbol;
+                    if (symbol) {
+                        var isInstanceType = isClassInstanceSide(type) ? 788968 /* Type */ : 111551 /* Value */;
+                        if (isJSConstructor(symbol.valueDeclaration)) {
+                            // Instance and static types share the same symbol; only add 'typeof' for the static side.
+                            return symbolToTypeNode(symbol, context, isInstanceType);
+                        }
+                        // Always use 'typeof T' for type of class, enum, and module objects
+                        else if (symbol.flags & 32 /* Class */ && !getBaseTypeVariableOfClass(symbol) && !(symbol.valueDeclaration.kind === 221 /* ClassExpression */ && context.flags & 2048 /* WriteClassExpressionAsTypeLiteral */) ||
+                            symbol.flags & (384 /* Enum */ | 512 /* ValueModule */) ||
+                            shouldWriteTypeOfFunctionSymbol()) {
+                            return symbolToTypeNode(symbol, context, isInstanceType);
+                        }
+                        else if ((_a = context.visitedTypes) === null || _a === void 0 ? void 0 : _a.has(typeId)) {
+                            // If type is an anonymous type literal in a type alias declaration, use type alias name
+                            var typeAlias = getTypeAliasForTypeLiteral(type);
+                            if (typeAlias) {
+                                // The specified symbol flags need to be reinterpreted as type flags
+                                return symbolToTypeNode(typeAlias, context, 788968 /* Type */);
+                            }
+                            else {
+                                return createElidedInformationPlaceholder(context);
+                            }
+                        }
+                        else {
+                            return visitAndTransformType(type, createTypeNodeFromObjectType);
+                        }
+                    }
+                    else {
+                        // Anonymous types without a symbol are never circular.
+                        return createTypeNodeFromObjectType(type);
+                    }
+                    function shouldWriteTypeOfFunctionSymbol() {
+                        var _a;
+                        var isStaticMethodSymbol = !!(symbol.flags & 8192 /* Method */) && // typeof static method
+                            ts.some(symbol.declarations, function (declaration) { return ts.hasSyntacticModifier(declaration, 32 /* Static */); });
+                        var isNonLocalFunctionSymbol = !!(symbol.flags & 16 /* Function */) &&
+                            (symbol.parent || // is exported function symbol
+                                ts.forEach(symbol.declarations, function (declaration) {
+                                    return declaration.parent.kind === 297 /* SourceFile */ || declaration.parent.kind === 257 /* ModuleBlock */;
+                                }));
+                        if (isStaticMethodSymbol || isNonLocalFunctionSymbol) {
+                            // typeof is allowed only for static/non local functions
+                            return (!!(context.flags & 4096 /* UseTypeOfFunction */) || ((_a = context.visitedTypes) === null || _a === void 0 ? void 0 : _a.has(typeId))) && // it is type of the symbol uses itself recursively
+                                (!(context.flags & 8 /* UseStructuralFallback */) || isValueSymbolAccessible(symbol, context.enclosingDeclaration)); // And the build is going to succeed without visibility error or there is no structural fallback allowed
+                        }
+                    }
                 }
-                for (var _i = 0, _a = signatureLists[i]; _i < _a.length; _i++) {
-                    var signature = _a[_i];
-                    // Only process signatures with parameter lists that aren't already in the result list
-                    if (!result || !findMatchingSignature(result, signature, /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ true)) {
-                        var unionSignatures = findMatchingSignatures(signatureLists, signature, i);
-                        if (unionSignatures) {
-                            var s = signature;
-                            // Union the result types when more than one signature matches
-                            if (unionSignatures.length > 1) {
-                                var thisParameter = signature.thisParameter;
-                                var firstThisParameterOfUnionSignatures = ts.forEach(unionSignatures, function (sig) { return sig.thisParameter; });
-                                if (firstThisParameterOfUnionSignatures) {
-                                    var thisType = getIntersectionType(ts.mapDefined(unionSignatures, function (sig) { return sig.thisParameter && getTypeOfSymbol(sig.thisParameter); }));
-                                    thisParameter = createSymbolWithType(firstThisParameterOfUnionSignatures, thisType);
+                function visitAndTransformType(type, transform) {
+                    var typeId = type.id;
+                    var isConstructorObject = ts.getObjectFlags(type) & 16 /* Anonymous */ && type.symbol && type.symbol.flags & 32 /* Class */;
+                    var id = ts.getObjectFlags(type) & 4 /* Reference */ && type.node ? "N" + getNodeId(type.node) :
+                        type.symbol ? (isConstructorObject ? "+" : "") + getSymbolId(type.symbol) :
+                            undefined;
+                    // Since instantiations of the same anonymous type have the same symbol, tracking symbols instead
+                    // of types allows us to catch circular references to instantiations of the same anonymous type
+                    if (!context.visitedTypes) {
+                        context.visitedTypes = new ts.Set();
+                    }
+                    if (id && !context.symbolDepth) {
+                        context.symbolDepth = new ts.Map();
+                    }
+                    var depth;
+                    if (id) {
+                        depth = context.symbolDepth.get(id) || 0;
+                        if (depth > 10) {
+                            return createElidedInformationPlaceholder(context);
+                        }
+                        context.symbolDepth.set(id, depth + 1);
+                    }
+                    context.visitedTypes.add(typeId);
+                    var result = transform(type);
+                    context.visitedTypes.delete(typeId);
+                    if (id) {
+                        context.symbolDepth.set(id, depth);
+                    }
+                    return result;
+                }
+                function createTypeNodeFromObjectType(type) {
+                    if (isGenericMappedType(type) || type.containsError) {
+                        return createMappedTypeNodeFromType(type);
+                    }
+                    var resolved = resolveStructuredTypeMembers(type);
+                    if (!resolved.properties.length && !resolved.stringIndexInfo && !resolved.numberIndexInfo) {
+                        if (!resolved.callSignatures.length && !resolved.constructSignatures.length) {
+                            context.approximateLength += 2;
+                            return ts.setEmitFlags(ts.factory.createTypeLiteralNode(/*members*/ undefined), 1 /* SingleLine */);
+                        }
+                        if (resolved.callSignatures.length === 1 && !resolved.constructSignatures.length) {
+                            var signature = resolved.callSignatures[0];
+                            var signatureNode = signatureToSignatureDeclarationHelper(signature, 174 /* FunctionType */, context);
+                            return signatureNode;
+                        }
+                        if (resolved.constructSignatures.length === 1 && !resolved.callSignatures.length) {
+                            var signature = resolved.constructSignatures[0];
+                            var signatureNode = signatureToSignatureDeclarationHelper(signature, 175 /* ConstructorType */, context);
+                            return signatureNode;
+                        }
+                    }
+                    var savedFlags = context.flags;
+                    context.flags |= 4194304 /* InObjectTypeLiteral */;
+                    var members = createTypeNodesFromResolvedType(resolved);
+                    context.flags = savedFlags;
+                    var typeLiteralNode = ts.factory.createTypeLiteralNode(members);
+                    context.approximateLength += 2;
+                    return ts.setEmitFlags(typeLiteralNode, (context.flags & 1024 /* MultilineObjectLiterals */) ? 0 : 1 /* SingleLine */);
+                }
+                function typeReferenceToTypeNode(type) {
+                    var typeArguments = getTypeArguments(type);
+                    if (type.target === globalArrayType || type.target === globalReadonlyArrayType) {
+                        if (context.flags & 2 /* WriteArrayAsGenericType */) {
+                            var typeArgumentNode = typeToTypeNodeHelper(typeArguments[0], context);
+                            return ts.factory.createTypeReferenceNode(type.target === globalArrayType ? "Array" : "ReadonlyArray", [typeArgumentNode]);
+                        }
+                        var elementType = typeToTypeNodeHelper(typeArguments[0], context);
+                        var arrayType = ts.factory.createArrayTypeNode(elementType);
+                        return type.target === globalArrayType ? arrayType : ts.factory.createTypeOperatorNode(142 /* ReadonlyKeyword */, arrayType);
+                    }
+                    else if (type.target.objectFlags & 8 /* Tuple */) {
+                        if (typeArguments.length > 0) {
+                            var arity = getTypeReferenceArity(type);
+                            var tupleConstituentNodes = mapToTypeNodes(typeArguments.slice(0, arity), context);
+                            if (tupleConstituentNodes) {
+                                if (type.target.labeledElementDeclarations) {
+                                    for (var i = 0; i < tupleConstituentNodes.length; i++) {
+                                        var flags = type.target.elementFlags[i];
+                                        tupleConstituentNodes[i] = ts.factory.createNamedTupleMember(flags & 12 /* Variable */ ? ts.factory.createToken(25 /* DotDotDotToken */) : undefined, ts.factory.createIdentifier(ts.unescapeLeadingUnderscores(getTupleElementLabel(type.target.labeledElementDeclarations[i]))), flags & 2 /* Optional */ ? ts.factory.createToken(57 /* QuestionToken */) : undefined, flags & 4 /* Rest */ ? ts.factory.createArrayTypeNode(tupleConstituentNodes[i]) :
+                                            tupleConstituentNodes[i]);
+                                    }
                                 }
-                                s = createUnionSignature(signature, unionSignatures);
-                                s.thisParameter = thisParameter;
+                                else {
+                                    for (var i = 0; i < Math.min(arity, tupleConstituentNodes.length); i++) {
+                                        var flags = type.target.elementFlags[i];
+                                        tupleConstituentNodes[i] =
+                                            flags & 12 /* Variable */ ? ts.factory.createRestTypeNode(flags & 4 /* Rest */ ? ts.factory.createArrayTypeNode(tupleConstituentNodes[i]) : tupleConstituentNodes[i]) :
+                                                flags & 2 /* Optional */ ? ts.factory.createOptionalTypeNode(tupleConstituentNodes[i]) :
+                                                    tupleConstituentNodes[i];
+                                    }
+                                }
+                                var tupleTypeNode = ts.setEmitFlags(ts.factory.createTupleTypeNode(tupleConstituentNodes), 1 /* SingleLine */);
+                                return type.target.readonly ? ts.factory.createTypeOperatorNode(142 /* ReadonlyKeyword */, tupleTypeNode) : tupleTypeNode;
                             }
-                            (result || (result = [])).push(s);
                         }
+                        if (context.encounteredError || (context.flags & 524288 /* AllowEmptyTuple */)) {
+                            var tupleTypeNode = ts.setEmitFlags(ts.factory.createTupleTypeNode([]), 1 /* SingleLine */);
+                            return type.target.readonly ? ts.factory.createTypeOperatorNode(142 /* ReadonlyKeyword */, tupleTypeNode) : tupleTypeNode;
+                        }
+                        context.encounteredError = true;
+                        return undefined; // TODO: GH#18217
+                    }
+                    else if (context.flags & 2048 /* WriteClassExpressionAsTypeLiteral */ &&
+                        type.symbol.valueDeclaration &&
+                        ts.isClassLike(type.symbol.valueDeclaration) &&
+                        !isValueSymbolAccessible(type.symbol, context.enclosingDeclaration)) {
+                        return createAnonymousTypeNode(type);
+                    }
+                    else {
+                        var outerTypeParameters = type.target.outerTypeParameters;
+                        var i = 0;
+                        var resultType = void 0;
+                        if (outerTypeParameters) {
+                            var length_2 = outerTypeParameters.length;
+                            while (i < length_2) {
+                                // Find group of type arguments for type parameters with the same declaring container.
+                                var start = i;
+                                var parent = getParentSymbolOfTypeParameter(outerTypeParameters[i]);
+                                do {
+                                    i++;
+                                } while (i < length_2 && getParentSymbolOfTypeParameter(outerTypeParameters[i]) === parent);
+                                // When type parameters are their own type arguments for the whole group (i.e. we have
+                                // the default outer type arguments), we don't show the group.
+                                if (!ts.rangeEquals(outerTypeParameters, typeArguments, start, i)) {
+                                    var typeArgumentSlice = mapToTypeNodes(typeArguments.slice(start, i), context);
+                                    var flags_3 = context.flags;
+                                    context.flags |= 16 /* ForbidIndexedAccessSymbolReferences */;
+                                    var ref = symbolToTypeNode(parent, context, 788968 /* Type */, typeArgumentSlice);
+                                    context.flags = flags_3;
+                                    resultType = !resultType ? ref : appendReferenceToType(resultType, ref);
+                                }
+                            }
+                        }
+                        var typeArgumentNodes = void 0;
+                        if (typeArguments.length > 0) {
+                            var typeParameterCount = (type.target.typeParameters || ts.emptyArray).length;
+                            typeArgumentNodes = mapToTypeNodes(typeArguments.slice(i, typeParameterCount), context);
+                        }
+                        var flags = context.flags;
+                        context.flags |= 16 /* ForbidIndexedAccessSymbolReferences */;
+                        var finalRef = symbolToTypeNode(type.symbol, context, 788968 /* Type */, typeArgumentNodes);
+                        context.flags = flags;
+                        return !resultType ? finalRef : appendReferenceToType(resultType, finalRef);
                     }
                 }
-            }
-            if (!ts.length(result) && indexWithLengthOverOne !== -1) {
-                // No sufficiently similar signature existed to subsume all the other signatures in the union - time to see if we can make a single
-                // signature that handles all over them. We only do this when there are overloads in only one constituent.
-                // (Overloads are conditional in nature and having overloads in multiple constituents would necessitate making a power set of
-                // signatures from the type, whose ordering would be non-obvious)
-                var masterList = signatureLists[indexWithLengthOverOne !== undefined ? indexWithLengthOverOne : 0];
-                var results = masterList.slice();
-                var _loop_9 = function (signatures) {
-                    if (signatures !== masterList) {
-                        var signature_1 = signatures[0];
-                        ts.Debug.assert(!!signature_1, "getUnionSignatures bails early on empty signature lists and should not have empty lists on second pass");
-                        results = signature_1.typeParameters && ts.some(results, function (s) { return !!s.typeParameters; }) ? undefined : ts.map(results, function (sig) { return combineSignaturesOfUnionMembers(sig, signature_1); });
-                        if (!results) {
-                            return "break";
+                function appendReferenceToType(root, ref) {
+                    if (ts.isImportTypeNode(root)) {
+                        // first shift type arguments
+                        var typeArguments = root.typeArguments;
+                        var qualifier = root.qualifier;
+                        if (qualifier) {
+                            if (ts.isIdentifier(qualifier)) {
+                                qualifier = ts.factory.updateIdentifier(qualifier, typeArguments);
+                            }
+                            else {
+                                qualifier = ts.factory.updateQualifiedName(qualifier, qualifier.left, ts.factory.updateIdentifier(qualifier.right, typeArguments));
+                            }
+                        }
+                        typeArguments = ref.typeArguments;
+                        // then move qualifiers
+                        var ids = getAccessStack(ref);
+                        for (var _i = 0, ids_1 = ids; _i < ids_1.length; _i++) {
+                            var id = ids_1[_i];
+                            qualifier = qualifier ? ts.factory.createQualifiedName(qualifier, id) : id;
                         }
+                        return ts.factory.updateImportTypeNode(root, root.argument, qualifier, typeArguments, root.isTypeOf);
+                    }
+                    else {
+                        // first shift type arguments
+                        var typeArguments = root.typeArguments;
+                        var typeName = root.typeName;
+                        if (ts.isIdentifier(typeName)) {
+                            typeName = ts.factory.updateIdentifier(typeName, typeArguments);
+                        }
+                        else {
+                            typeName = ts.factory.updateQualifiedName(typeName, typeName.left, ts.factory.updateIdentifier(typeName.right, typeArguments));
+                        }
+                        typeArguments = ref.typeArguments;
+                        // then move qualifiers
+                        var ids = getAccessStack(ref);
+                        for (var _a = 0, ids_2 = ids; _a < ids_2.length; _a++) {
+                            var id = ids_2[_a];
+                            typeName = ts.factory.createQualifiedName(typeName, id);
+                        }
+                        return ts.factory.updateTypeReferenceNode(root, typeName, typeArguments);
                     }
-                };
-                for (var _b = 0, signatureLists_1 = signatureLists; _b < signatureLists_1.length; _b++) {
-                    var signatures = signatureLists_1[_b];
-                    var state_3 = _loop_9(signatures);
-                    if (state_3 === "break")
-                        break;
                 }
-                result = results;
-            }
-            return result || ts.emptyArray;
-        }
-        function combineUnionThisParam(left, right) {
-            if (!left || !right) {
-                return left || right;
-            }
-            // A signature `this` type might be a read or a write position... It's very possible that it should be invariant
-            // and we should refuse to merge signatures if there are `this` types and they do not match. However, so as to be
-            // permissive when calling, for now, we'll intersect the `this` types just like we do for param types in union signatures.
-            var thisType = getIntersectionType([getTypeOfSymbol(left), getTypeOfSymbol(right)]);
-            return createSymbolWithType(left, thisType);
-        }
-        function combineUnionParameters(left, right) {
-            var leftCount = getParameterCount(left);
-            var rightCount = getParameterCount(right);
-            var longest = leftCount >= rightCount ? left : right;
-            var shorter = longest === left ? right : left;
-            var longestCount = longest === left ? leftCount : rightCount;
-            var eitherHasEffectiveRest = (hasEffectiveRestParameter(left) || hasEffectiveRestParameter(right));
-            var needsExtraRestElement = eitherHasEffectiveRest && !hasEffectiveRestParameter(longest);
-            var params = new Array(longestCount + (needsExtraRestElement ? 1 : 0));
-            for (var i = 0; i < longestCount; i++) {
-                var longestParamType = tryGetTypeAtPosition(longest, i);
-                var shorterParamType = tryGetTypeAtPosition(shorter, i) || unknownType;
-                var unionParamType = getIntersectionType([longestParamType, shorterParamType]);
-                var isRestParam = eitherHasEffectiveRest && !needsExtraRestElement && i === (longestCount - 1);
-                var isOptional = i >= getMinArgumentCount(longest) && i >= getMinArgumentCount(shorter);
-                var leftName = i >= leftCount ? undefined : getParameterNameAtPosition(left, i);
-                var rightName = i >= rightCount ? undefined : getParameterNameAtPosition(right, i);
-                var paramName = leftName === rightName ? leftName :
-                    !leftName ? rightName :
-                        !rightName ? leftName :
-                            undefined;
-                var paramSymbol = createSymbol(1 /* FunctionScopedVariable */ | (isOptional && !isRestParam ? 16777216 /* Optional */ : 0), paramName || "arg" + i);
-                paramSymbol.type = isRestParam ? createArrayType(unionParamType) : unionParamType;
-                params[i] = paramSymbol;
-            }
-            if (needsExtraRestElement) {
-                var restParamSymbol = createSymbol(1 /* FunctionScopedVariable */, "args");
-                restParamSymbol.type = createArrayType(getTypeAtPosition(shorter, longestCount));
-                params[longestCount] = restParamSymbol;
-            }
-            return params;
-        }
-        function combineSignaturesOfUnionMembers(left, right) {
-            var declaration = left.declaration;
-            var params = combineUnionParameters(left, right);
-            var thisParam = combineUnionThisParam(left.thisParameter, right.thisParameter);
-            var minArgCount = Math.max(left.minArgumentCount, right.minArgumentCount);
-            var result = createSignature(declaration, left.typeParameters || right.typeParameters, thisParam, params, 
-            /*resolvedReturnType*/ undefined, 
-            /*resolvedTypePredicate*/ undefined, minArgCount, (left.flags | right.flags) & 3 /* PropagatingFlags */);
-            result.unionSignatures = ts.concatenate(left.unionSignatures || [left], [right]);
-            return result;
-        }
-        function getUnionIndexInfo(types, kind) {
-            var indexTypes = [];
-            var isAnyReadonly = false;
-            for (var _i = 0, types_3 = types; _i < types_3.length; _i++) {
-                var type = types_3[_i];
-                var indexInfo = getIndexInfoOfType(getApparentType(type), kind);
-                if (!indexInfo) {
-                    return undefined;
+                function getAccessStack(ref) {
+                    var state = ref.typeName;
+                    var ids = [];
+                    while (!ts.isIdentifier(state)) {
+                        ids.unshift(state.right);
+                        state = state.left;
+                    }
+                    ids.unshift(state);
+                    return ids;
+                }
+                function createTypeNodesFromResolvedType(resolvedType) {
+                    if (checkTruncationLength(context)) {
+                        return [ts.factory.createPropertySignature(/*modifiers*/ undefined, "...", /*questionToken*/ undefined, /*type*/ undefined)];
+                    }
+                    var typeElements = [];
+                    for (var _i = 0, _a = resolvedType.callSignatures; _i < _a.length; _i++) {
+                        var signature = _a[_i];
+                        typeElements.push(signatureToSignatureDeclarationHelper(signature, 169 /* CallSignature */, context));
+                    }
+                    for (var _b = 0, _c = resolvedType.constructSignatures; _b < _c.length; _b++) {
+                        var signature = _c[_b];
+                        typeElements.push(signatureToSignatureDeclarationHelper(signature, 170 /* ConstructSignature */, context));
+                    }
+                    if (resolvedType.stringIndexInfo) {
+                        var indexSignature = void 0;
+                        if (resolvedType.objectFlags & 2048 /* ReverseMapped */) {
+                            indexSignature = indexInfoToIndexSignatureDeclarationHelper(createIndexInfo(anyType, resolvedType.stringIndexInfo.isReadonly, resolvedType.stringIndexInfo.declaration), 0 /* String */, context, createElidedInformationPlaceholder(context));
+                        }
+                        else {
+                            indexSignature = indexInfoToIndexSignatureDeclarationHelper(resolvedType.stringIndexInfo, 0 /* String */, context, /*typeNode*/ undefined);
+                        }
+                        typeElements.push(indexSignature);
+                    }
+                    if (resolvedType.numberIndexInfo) {
+                        typeElements.push(indexInfoToIndexSignatureDeclarationHelper(resolvedType.numberIndexInfo, 1 /* Number */, context, /*typeNode*/ undefined));
+                    }
+                    var properties = resolvedType.properties;
+                    if (!properties) {
+                        return typeElements;
+                    }
+                    var i = 0;
+                    for (var _d = 0, properties_1 = properties; _d < properties_1.length; _d++) {
+                        var propertySymbol = properties_1[_d];
+                        i++;
+                        if (context.flags & 2048 /* WriteClassExpressionAsTypeLiteral */) {
+                            if (propertySymbol.flags & 4194304 /* Prototype */) {
+                                continue;
+                            }
+                            if (ts.getDeclarationModifierFlagsFromSymbol(propertySymbol) & (8 /* Private */ | 16 /* Protected */) && context.tracker.reportPrivateInBaseOfClassExpression) {
+                                context.tracker.reportPrivateInBaseOfClassExpression(ts.unescapeLeadingUnderscores(propertySymbol.escapedName));
+                            }
+                        }
+                        if (checkTruncationLength(context) && (i + 2 < properties.length - 1)) {
+                            typeElements.push(ts.factory.createPropertySignature(/*modifiers*/ undefined, "... " + (properties.length - i) + " more ...", /*questionToken*/ undefined, /*type*/ undefined));
+                            addPropertyToElementList(properties[properties.length - 1], context, typeElements);
+                            break;
+                        }
+                        addPropertyToElementList(propertySymbol, context, typeElements);
+                    }
+                    return typeElements.length ? typeElements : undefined;
                 }
-                indexTypes.push(indexInfo.type);
-                isAnyReadonly = isAnyReadonly || indexInfo.isReadonly;
             }
-            return createIndexInfo(getUnionType(indexTypes, 2 /* Subtype */), isAnyReadonly);
-        }
-        function resolveUnionTypeMembers(type) {
-            // The members and properties collections are empty for union types. To get all properties of a union
-            // type use getPropertiesOfType (only the language service uses this).
-            var callSignatures = getUnionSignatures(ts.map(type.types, function (t) { return t === globalFunctionType ? [unknownSignature] : getSignaturesOfType(t, 0 /* Call */); }));
-            var constructSignatures = getUnionSignatures(ts.map(type.types, function (t) { return getSignaturesOfType(t, 1 /* Construct */); }));
-            var stringIndexInfo = getUnionIndexInfo(type.types, 0 /* String */);
-            var numberIndexInfo = getUnionIndexInfo(type.types, 1 /* Number */);
-            setStructuredTypeMembers(type, emptySymbols, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
-        }
-        function intersectTypes(type1, type2) {
-            return !type1 ? type2 : !type2 ? type1 : getIntersectionType([type1, type2]);
-        }
-        function intersectIndexInfos(info1, info2) {
-            return !info1 ? info2 : !info2 ? info1 : createIndexInfo(getIntersectionType([info1.type, info2.type]), info1.isReadonly && info2.isReadonly);
-        }
-        function unionSpreadIndexInfos(info1, info2) {
-            return info1 && info2 && createIndexInfo(getUnionType([info1.type, info2.type]), info1.isReadonly || info2.isReadonly);
-        }
-        function findMixins(types) {
-            var constructorTypeCount = ts.countWhere(types, function (t) { return getSignaturesOfType(t, 1 /* Construct */).length > 0; });
-            var mixinFlags = ts.map(types, isMixinConstructorType);
-            if (constructorTypeCount > 0 && constructorTypeCount === ts.countWhere(mixinFlags, function (b) { return b; })) {
-                var firstMixinIndex = mixinFlags.indexOf(/*searchElement*/ true);
-                mixinFlags[firstMixinIndex] = false;
+            function createElidedInformationPlaceholder(context) {
+                context.approximateLength += 3;
+                if (!(context.flags & 1 /* NoTruncation */)) {
+                    return ts.factory.createTypeReferenceNode(ts.factory.createIdentifier("..."), /*typeArguments*/ undefined);
+                }
+                return ts.factory.createKeywordTypeNode(128 /* AnyKeyword */);
             }
-            return mixinFlags;
-        }
-        function includeMixinType(type, types, mixinFlags, index) {
-            var mixedTypes = [];
-            for (var i = 0; i < types.length; i++) {
-                if (i === index) {
-                    mixedTypes.push(type);
+            function addPropertyToElementList(propertySymbol, context, typeElements) {
+                var propertyIsReverseMapped = !!(ts.getCheckFlags(propertySymbol) & 8192 /* ReverseMapped */);
+                var propertyType = propertyIsReverseMapped && context.flags & 33554432 /* InReverseMappedType */ ?
+                    anyType : getTypeOfSymbol(propertySymbol);
+                var saveEnclosingDeclaration = context.enclosingDeclaration;
+                context.enclosingDeclaration = undefined;
+                if (context.tracker.trackSymbol && ts.getCheckFlags(propertySymbol) & 4096 /* Late */) {
+                    var decl = ts.first(propertySymbol.declarations);
+                    if (hasLateBindableName(decl)) {
+                        if (ts.isBinaryExpression(decl)) {
+                            var name = ts.getNameOfDeclaration(decl);
+                            if (name && ts.isElementAccessExpression(name) && ts.isPropertyAccessEntityNameExpression(name.argumentExpression)) {
+                                trackComputedName(name.argumentExpression, saveEnclosingDeclaration, context);
+                            }
+                        }
+                        else {
+                            trackComputedName(decl.name.expression, saveEnclosingDeclaration, context);
+                        }
+                    }
                 }
-                else if (mixinFlags[i]) {
-                    mixedTypes.push(getReturnTypeOfSignature(getSignaturesOfType(types[i], 1 /* Construct */)[0]));
+                context.enclosingDeclaration = saveEnclosingDeclaration;
+                var propertyName = getPropertyNameNodeForSymbol(propertySymbol, context);
+                context.approximateLength += (ts.symbolName(propertySymbol).length + 1);
+                var optionalToken = propertySymbol.flags & 16777216 /* Optional */ ? ts.factory.createToken(57 /* QuestionToken */) : undefined;
+                if (propertySymbol.flags & (16 /* Function */ | 8192 /* Method */) && !getPropertiesOfObjectType(propertyType).length && !isReadonlySymbol(propertySymbol)) {
+                    var signatures = getSignaturesOfType(filterType(propertyType, function (t) { return !(t.flags & 32768 /* Undefined */); }), 0 /* Call */);
+                    for (var _i = 0, signatures_1 = signatures; _i < signatures_1.length; _i++) {
+                        var signature = signatures_1[_i];
+                        var methodDeclaration = signatureToSignatureDeclarationHelper(signature, 164 /* MethodSignature */, context, { name: propertyName, questionToken: optionalToken });
+                        typeElements.push(preserveCommentsOn(methodDeclaration));
+                    }
                 }
-            }
-            return getIntersectionType(mixedTypes);
-        }
-        function resolveIntersectionTypeMembers(type) {
-            // The members and properties collections are empty for intersection types. To get all properties of an
-            // intersection type use getPropertiesOfType (only the language service uses this).
-            var callSignatures;
-            var constructSignatures;
-            var stringIndexInfo;
-            var numberIndexInfo;
-            var types = type.types;
-            var mixinFlags = findMixins(types);
-            var mixinCount = ts.countWhere(mixinFlags, function (b) { return b; });
-            var _loop_10 = function (i) {
-                var t = type.types[i];
-                // When an intersection type contains mixin constructor types, the construct signatures from
-                // those types are discarded and their return types are mixed into the return types of all
-                // other construct signatures in the intersection type. For example, the intersection type
-                // '{ new(...args: any[]) => A } & { new(s: string) => B }' has a single construct signature
-                // 'new(s: string) => A & B'.
-                if (!mixinFlags[i]) {
-                    var signatures = getSignaturesOfType(t, 1 /* Construct */);
-                    if (signatures.length && mixinCount > 0) {
-                        signatures = ts.map(signatures, function (s) {
-                            var clone = cloneSignature(s);
-                            clone.resolvedReturnType = includeMixinType(getReturnTypeOfSignature(s), types, mixinFlags, i);
-                            return clone;
-                        });
+                else {
+                    var savedFlags = context.flags;
+                    context.flags |= propertyIsReverseMapped ? 33554432 /* InReverseMappedType */ : 0;
+                    var propertyTypeNode = void 0;
+                    if (propertyIsReverseMapped && !!(savedFlags & 33554432 /* InReverseMappedType */)) {
+                        propertyTypeNode = createElidedInformationPlaceholder(context);
                     }
-                    constructSignatures = appendSignatures(constructSignatures, signatures);
+                    else {
+                        propertyTypeNode = propertyType ? serializeTypeForDeclaration(context, propertyType, propertySymbol, saveEnclosingDeclaration) : ts.factory.createKeywordTypeNode(128 /* AnyKeyword */);
+                    }
+                    context.flags = savedFlags;
+                    var modifiers = isReadonlySymbol(propertySymbol) ? [ts.factory.createToken(142 /* ReadonlyKeyword */)] : undefined;
+                    if (modifiers) {
+                        context.approximateLength += 9;
+                    }
+                    var propertySignature = ts.factory.createPropertySignature(modifiers, propertyName, optionalToken, propertyTypeNode);
+                    typeElements.push(preserveCommentsOn(propertySignature));
                 }
-                callSignatures = appendSignatures(callSignatures, getSignaturesOfType(t, 0 /* Call */));
-                stringIndexInfo = intersectIndexInfos(stringIndexInfo, getIndexInfoOfType(t, 0 /* String */));
-                numberIndexInfo = intersectIndexInfos(numberIndexInfo, getIndexInfoOfType(t, 1 /* Number */));
-            };
-            for (var i = 0; i < types.length; i++) {
-                _loop_10(i);
-            }
-            setStructuredTypeMembers(type, emptySymbols, callSignatures || ts.emptyArray, constructSignatures || ts.emptyArray, stringIndexInfo, numberIndexInfo);
-        }
-        function appendSignatures(signatures, newSignatures) {
-            var _loop_11 = function (sig) {
-                if (!signatures || ts.every(signatures, function (s) { return !compareSignaturesIdentical(s, sig, /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ false, compareTypesIdentical); })) {
-                    signatures = ts.append(signatures, sig);
+                function preserveCommentsOn(node) {
+                    if (ts.some(propertySymbol.declarations, function (d) { return d.kind === 333 /* JSDocPropertyTag */; })) {
+                        var d = ts.find(propertySymbol.declarations, function (d) { return d.kind === 333 /* JSDocPropertyTag */; });
+                        var commentText = d.comment;
+                        if (commentText) {
+                            ts.setSyntheticLeadingComments(node, [{ kind: 3 /* MultiLineCommentTrivia */, text: "*\n * " + commentText.replace(/\n/g, "\n * ") + "\n ", pos: -1, end: -1, hasTrailingNewLine: true }]);
+                        }
+                    }
+                    else if (propertySymbol.valueDeclaration) {
+                        // Copy comments to node for declaration emit
+                        ts.setCommentRange(node, propertySymbol.valueDeclaration);
+                    }
+                    return node;
                 }
-            };
-            for (var _i = 0, newSignatures_1 = newSignatures; _i < newSignatures_1.length; _i++) {
-                var sig = newSignatures_1[_i];
-                _loop_11(sig);
-            }
-            return signatures;
-        }
-        /**
-         * Converts an AnonymousType to a ResolvedType.
-         */
-        function resolveAnonymousTypeMembers(type) {
-            var symbol = getMergedSymbol(type.symbol);
-            if (type.target) {
-                setStructuredTypeMembers(type, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-                var members = createInstantiatedSymbolTable(getPropertiesOfObjectType(type.target), type.mapper, /*mappingThisOnly*/ false);
-                var callSignatures = instantiateSignatures(getSignaturesOfType(type.target, 0 /* Call */), type.mapper);
-                var constructSignatures = instantiateSignatures(getSignaturesOfType(type.target, 1 /* Construct */), type.mapper);
-                var stringIndexInfo = instantiateIndexInfo(getIndexInfoOfType(type.target, 0 /* String */), type.mapper);
-                var numberIndexInfo = instantiateIndexInfo(getIndexInfoOfType(type.target, 1 /* Number */), type.mapper);
-                setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
-            }
-            else if (symbol.flags & 2048 /* TypeLiteral */) {
-                setStructuredTypeMembers(type, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-                var members = getMembersOfSymbol(symbol);
-                var callSignatures = getSignaturesOfSymbol(members.get("__call" /* Call */));
-                var constructSignatures = getSignaturesOfSymbol(members.get("__new" /* New */));
-                var stringIndexInfo = getIndexInfoOfSymbol(symbol, 0 /* String */);
-                var numberIndexInfo = getIndexInfoOfSymbol(symbol, 1 /* Number */);
-                setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
             }
-            else {
-                // Combinations of function, class, enum and module
-                var members = emptySymbols;
-                var stringIndexInfo = void 0;
-                if (symbol.exports) {
-                    members = getExportsOfSymbol(symbol);
-                    if (symbol === globalThisSymbol) {
-                        var varsOnly_1 = ts.createMap();
-                        members.forEach(function (p) {
-                            if (!(p.flags & 418 /* BlockScoped */)) {
-                                varsOnly_1.set(p.escapedName, p);
+            function mapToTypeNodes(types, context, isBareList) {
+                if (ts.some(types)) {
+                    if (checkTruncationLength(context)) {
+                        if (!isBareList) {
+                            return [ts.factory.createTypeReferenceNode("...", /*typeArguments*/ undefined)];
+                        }
+                        else if (types.length > 2) {
+                            return [
+                                typeToTypeNodeHelper(types[0], context),
+                                ts.factory.createTypeReferenceNode("... " + (types.length - 2) + " more ...", /*typeArguments*/ undefined),
+                                typeToTypeNodeHelper(types[types.length - 1], context)
+                            ];
+                        }
+                    }
+                    var mayHaveNameCollisions = !(context.flags & 64 /* UseFullyQualifiedType */);
+                    /** Map from type reference identifier text to [type, index in `result` where the type node is] */
+                    var seenNames = mayHaveNameCollisions ? ts.createUnderscoreEscapedMultiMap() : undefined;
+                    var result_4 = [];
+                    var i = 0;
+                    for (var _i = 0, types_2 = types; _i < types_2.length; _i++) {
+                        var type = types_2[_i];
+                        i++;
+                        if (checkTruncationLength(context) && (i + 2 < types.length - 1)) {
+                            result_4.push(ts.factory.createTypeReferenceNode("... " + (types.length - i) + " more ...", /*typeArguments*/ undefined));
+                            var typeNode_1 = typeToTypeNodeHelper(types[types.length - 1], context);
+                            if (typeNode_1) {
+                                result_4.push(typeNode_1);
+                            }
+                            break;
+                        }
+                        context.approximateLength += 2; // Account for whitespace + separator
+                        var typeNode = typeToTypeNodeHelper(type, context);
+                        if (typeNode) {
+                            result_4.push(typeNode);
+                            if (seenNames && ts.isIdentifierTypeReference(typeNode)) {
+                                seenNames.add(typeNode.typeName.escapedText, [type, result_4.length - 1]);
+                            }
+                        }
+                    }
+                    if (seenNames) {
+                        // To avoid printing types like `[Foo, Foo]` or `Bar & Bar` where
+                        // occurrences of the same name actually come from different
+                        // namespaces, go through the single-identifier type reference nodes
+                        // we just generated, and see if any names were generated more than
+                        // once while referring to different types. If so, regenerate the
+                        // type node for each entry by that name with the
+                        // `UseFullyQualifiedType` flag enabled.
+                        var saveContextFlags = context.flags;
+                        context.flags |= 64 /* UseFullyQualifiedType */;
+                        seenNames.forEach(function (types) {
+                            if (!ts.arrayIsHomogeneous(types, function (_a, _b) {
+                                var a = _a[0];
+                                var b = _b[0];
+                                return typesAreSameReference(a, b);
+                            })) {
+                                for (var _i = 0, types_3 = types; _i < types_3.length; _i++) {
+                                    var _a = types_3[_i], type = _a[0], resultIndex = _a[1];
+                                    result_4[resultIndex] = typeToTypeNodeHelper(type, context);
+                                }
                             }
                         });
-                        members = varsOnly_1;
+                        context.flags = saveContextFlags;
                     }
+                    return result_4;
                 }
-                setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, undefined, undefined);
-                if (symbol.flags & 32 /* Class */) {
-                    var classType = getDeclaredTypeOfClassOrInterface(symbol);
-                    var baseConstructorType = getBaseConstructorTypeOfClass(classType);
-                    if (baseConstructorType.flags & (524288 /* Object */ | 2097152 /* Intersection */ | 8650752 /* TypeVariable */)) {
-                        members = ts.createSymbolTable(getNamedMembers(members));
-                        addInheritedMembers(members, getPropertiesOfType(baseConstructorType));
-                    }
-                    else if (baseConstructorType === anyType) {
-                        stringIndexInfo = createIndexInfo(anyType, /*isReadonly*/ false);
-                    }
+            }
+            function typesAreSameReference(a, b) {
+                return a === b
+                    || !!a.symbol && a.symbol === b.symbol
+                    || !!a.aliasSymbol && a.aliasSymbol === b.aliasSymbol;
+            }
+            function indexInfoToIndexSignatureDeclarationHelper(indexInfo, kind, context, typeNode) {
+                var name = ts.getNameFromIndexInfo(indexInfo) || "x";
+                var indexerTypeNode = ts.factory.createKeywordTypeNode(kind === 0 /* String */ ? 147 /* StringKeyword */ : 144 /* NumberKeyword */);
+                var indexingParameter = ts.factory.createParameterDeclaration(
+                /*decorators*/ undefined, 
+                /*modifiers*/ undefined, 
+                /*dotDotDotToken*/ undefined, name, 
+                /*questionToken*/ undefined, indexerTypeNode, 
+                /*initializer*/ undefined);
+                if (!typeNode) {
+                    typeNode = typeToTypeNodeHelper(indexInfo.type || anyType, context);
                 }
-                var numberIndexInfo = symbol.flags & 384 /* Enum */ && (getDeclaredTypeOfSymbol(symbol).flags & 32 /* Enum */ ||
-                    ts.some(type.properties, function (prop) { return !!(getTypeOfSymbol(prop).flags & 296 /* NumberLike */); })) ? enumNumberIndexInfo : undefined;
-                setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
-                // We resolve the members before computing the signatures because a signature may use
-                // typeof with a qualified name expression that circularly references the type we are
-                // in the process of resolving (see issue #6072). The temporarily empty signature list
-                // will never be observed because a qualified name can't reference signatures.
-                if (symbol.flags & (16 /* Function */ | 8192 /* Method */)) {
-                    type.callSignatures = getSignaturesOfSymbol(symbol);
+                if (!indexInfo.type && !(context.flags & 2097152 /* AllowEmptyIndexInfoType */)) {
+                    context.encounteredError = true;
+                }
+                context.approximateLength += (name.length + 4);
+                return ts.factory.createIndexSignature(
+                /*decorators*/ undefined, indexInfo.isReadonly ? [ts.factory.createToken(142 /* ReadonlyKeyword */)] : undefined, [indexingParameter], typeNode);
+            }
+            function signatureToSignatureDeclarationHelper(signature, kind, context, options) {
+                var _a, _b, _c, _d;
+                var suppressAny = context.flags & 256 /* SuppressAnyReturnType */;
+                if (suppressAny)
+                    context.flags &= ~256 /* SuppressAnyReturnType */; // suppress only toplevel `any`s
+                var typeParameters;
+                var typeArguments;
+                if (context.flags & 32 /* WriteTypeArgumentsOfSignature */ && signature.target && signature.mapper && signature.target.typeParameters) {
+                    typeArguments = signature.target.typeParameters.map(function (parameter) { return typeToTypeNodeHelper(instantiateType(parameter, signature.mapper), context); });
+                }
+                else {
+                    typeParameters = signature.typeParameters && signature.typeParameters.map(function (parameter) { return typeParameterToDeclaration(parameter, context); });
                 }
-                // And likewise for construct signatures for classes
-                if (symbol.flags & 32 /* Class */) {
-                    var classType_1 = getDeclaredTypeOfClassOrInterface(symbol);
-                    var constructSignatures = symbol.members ? getSignaturesOfSymbol(symbol.members.get("__constructor" /* Constructor */)) : ts.emptyArray;
-                    if (symbol.flags & 16 /* Function */) {
-                        constructSignatures = ts.addRange(constructSignatures.slice(), ts.mapDefined(type.callSignatures, function (sig) { return isJSConstructor(sig.declaration) ?
-                            createSignature(sig.declaration, sig.typeParameters, sig.thisParameter, sig.parameters, classType_1, /*resolvedTypePredicate*/ undefined, sig.minArgumentCount, sig.flags & 3 /* PropagatingFlags */) :
-                            undefined; }));
+                var expandedParams = getExpandedParameters(signature, /*skipUnionExpanding*/ true)[0];
+                // If the expanded parameter list had a variadic in a non-trailing position, don't expand it
+                var parameters = (ts.some(expandedParams, function (p) { return p !== expandedParams[expandedParams.length - 1] && !!(ts.getCheckFlags(p) & 32768 /* RestParameter */); }) ? signature.parameters : expandedParams).map(function (parameter) { return symbolToParameterDeclaration(parameter, context, kind === 166 /* Constructor */, options === null || options === void 0 ? void 0 : options.privateSymbolVisitor, options === null || options === void 0 ? void 0 : options.bundledImports); });
+                if (signature.thisParameter) {
+                    var thisParameter = symbolToParameterDeclaration(signature.thisParameter, context);
+                    parameters.unshift(thisParameter);
+                }
+                var returnTypeNode;
+                var typePredicate = getTypePredicateOfSignature(signature);
+                if (typePredicate) {
+                    var assertsModifier = typePredicate.kind === 2 /* AssertsThis */ || typePredicate.kind === 3 /* AssertsIdentifier */ ?
+                        ts.factory.createToken(127 /* AssertsKeyword */) :
+                        undefined;
+                    var parameterName = typePredicate.kind === 1 /* Identifier */ || typePredicate.kind === 3 /* AssertsIdentifier */ ?
+                        ts.setEmitFlags(ts.factory.createIdentifier(typePredicate.parameterName), 16777216 /* NoAsciiEscaping */) :
+                        ts.factory.createThisTypeNode();
+                    var typeNode = typePredicate.type && typeToTypeNodeHelper(typePredicate.type, context);
+                    returnTypeNode = ts.factory.createTypePredicateNode(assertsModifier, parameterName, typeNode);
+                }
+                else {
+                    var returnType = getReturnTypeOfSignature(signature);
+                    if (returnType && !(suppressAny && isTypeAny(returnType))) {
+                        returnTypeNode = serializeReturnTypeForSignature(context, returnType, signature, options === null || options === void 0 ? void 0 : options.privateSymbolVisitor, options === null || options === void 0 ? void 0 : options.bundledImports);
                     }
-                    if (!constructSignatures.length) {
-                        constructSignatures = getDefaultConstructSignatures(classType_1);
+                    else if (!suppressAny) {
+                        returnTypeNode = ts.factory.createKeywordTypeNode(128 /* AnyKeyword */);
                     }
-                    type.constructSignatures = constructSignatures;
                 }
+                context.approximateLength += 3; // Usually a signature contributes a few more characters than this, but 3 is the minimum
+                var node = kind === 169 /* CallSignature */ ? ts.factory.createCallSignature(typeParameters, parameters, returnTypeNode) :
+                    kind === 170 /* ConstructSignature */ ? ts.factory.createConstructSignature(typeParameters, parameters, returnTypeNode) :
+                        kind === 164 /* MethodSignature */ ? ts.factory.createMethodSignature(options === null || options === void 0 ? void 0 : options.modifiers, (_a = options === null || options === void 0 ? void 0 : options.name) !== null && _a !== void 0 ? _a : ts.factory.createIdentifier(""), options === null || options === void 0 ? void 0 : options.questionToken, typeParameters, parameters, returnTypeNode) :
+                            kind === 165 /* MethodDeclaration */ ? ts.factory.createMethodDeclaration(/*decorators*/ undefined, options === null || options === void 0 ? void 0 : options.modifiers, /*asteriskToken*/ undefined, (_b = options === null || options === void 0 ? void 0 : options.name) !== null && _b !== void 0 ? _b : ts.factory.createIdentifier(""), /*questionToken*/ undefined, typeParameters, parameters, returnTypeNode, /*body*/ undefined) :
+                                kind === 166 /* Constructor */ ? ts.factory.createConstructorDeclaration(/*decorators*/ undefined, options === null || options === void 0 ? void 0 : options.modifiers, parameters, /*body*/ undefined) :
+                                    kind === 167 /* GetAccessor */ ? ts.factory.createGetAccessorDeclaration(/*decorators*/ undefined, options === null || options === void 0 ? void 0 : options.modifiers, (_c = options === null || options === void 0 ? void 0 : options.name) !== null && _c !== void 0 ? _c : ts.factory.createIdentifier(""), parameters, returnTypeNode, /*body*/ undefined) :
+                                        kind === 168 /* SetAccessor */ ? ts.factory.createSetAccessorDeclaration(/*decorators*/ undefined, options === null || options === void 0 ? void 0 : options.modifiers, (_d = options === null || options === void 0 ? void 0 : options.name) !== null && _d !== void 0 ? _d : ts.factory.createIdentifier(""), parameters, /*body*/ undefined) :
+                                            kind === 171 /* IndexSignature */ ? ts.factory.createIndexSignature(/*decorators*/ undefined, options === null || options === void 0 ? void 0 : options.modifiers, parameters, returnTypeNode) :
+                                                kind === 308 /* JSDocFunctionType */ ? ts.factory.createJSDocFunctionType(parameters, returnTypeNode) :
+                                                    kind === 174 /* FunctionType */ ? ts.factory.createFunctionTypeNode(typeParameters, parameters, returnTypeNode !== null && returnTypeNode !== void 0 ? returnTypeNode : ts.factory.createTypeReferenceNode(ts.factory.createIdentifier(""))) :
+                                                        kind === 175 /* ConstructorType */ ? ts.factory.createConstructorTypeNode(typeParameters, parameters, returnTypeNode !== null && returnTypeNode !== void 0 ? returnTypeNode : ts.factory.createTypeReferenceNode(ts.factory.createIdentifier(""))) :
+                                                            kind === 251 /* FunctionDeclaration */ ? ts.factory.createFunctionDeclaration(/*decorators*/ undefined, options === null || options === void 0 ? void 0 : options.modifiers, /*asteriskToken*/ undefined, (options === null || options === void 0 ? void 0 : options.name) ? ts.cast(options.name, ts.isIdentifier) : ts.factory.createIdentifier(""), typeParameters, parameters, returnTypeNode, /*body*/ undefined) :
+                                                                kind === 208 /* FunctionExpression */ ? ts.factory.createFunctionExpression(options === null || options === void 0 ? void 0 : options.modifiers, /*asteriskToken*/ undefined, (options === null || options === void 0 ? void 0 : options.name) ? ts.cast(options.name, ts.isIdentifier) : ts.factory.createIdentifier(""), typeParameters, parameters, returnTypeNode, ts.factory.createBlock([])) :
+                                                                    kind === 209 /* ArrowFunction */ ? ts.factory.createArrowFunction(options === null || options === void 0 ? void 0 : options.modifiers, typeParameters, parameters, returnTypeNode, /*equalsGreaterThanToken*/ undefined, ts.factory.createBlock([])) :
+                                                                        ts.Debug.assertNever(kind);
+                if (typeArguments) {
+                    node.typeArguments = ts.factory.createNodeArray(typeArguments);
+                }
+                return node;
             }
-        }
-        function resolveReverseMappedTypeMembers(type) {
-            var indexInfo = getIndexInfoOfType(type.source, 0 /* String */);
-            var modifiers = getMappedTypeModifiers(type.mappedType);
-            var readonlyMask = modifiers & 1 /* IncludeReadonly */ ? false : true;
-            var optionalMask = modifiers & 4 /* IncludeOptional */ ? 0 : 16777216 /* Optional */;
-            var stringIndexInfo = indexInfo && createIndexInfo(inferReverseMappedType(indexInfo.type, type.mappedType, type.constraintType), readonlyMask && indexInfo.isReadonly);
-            var members = ts.createSymbolTable();
-            for (var _i = 0, _a = getPropertiesOfType(type.source); _i < _a.length; _i++) {
-                var prop = _a[_i];
-                var checkFlags = 8192 /* ReverseMapped */ | (readonlyMask && isReadonlySymbol(prop) ? 8 /* Readonly */ : 0);
-                var inferredProp = createSymbol(4 /* Property */ | prop.flags & optionalMask, prop.escapedName, checkFlags);
-                inferredProp.declarations = prop.declarations;
-                inferredProp.nameType = getSymbolLinks(prop).nameType;
-                inferredProp.propertyType = getTypeOfSymbol(prop);
-                inferredProp.mappedType = type.mappedType;
-                inferredProp.constraintType = type.constraintType;
-                members.set(prop.escapedName, inferredProp);
-            }
-            setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, undefined);
-        }
-        // Return the lower bound of the key type in a mapped type. Intuitively, the lower
-        // bound includes those keys that are known to always be present, for example because
-        // because of constraints on type parameters (e.g. 'keyof T' for a constrained T).
-        function getLowerBoundOfKeyType(type) {
-            if (type.flags & (1 /* Any */ | 131068 /* Primitive */)) {
-                return type;
+            function typeParameterToDeclarationWithConstraint(type, context, constraintNode) {
+                var savedContextFlags = context.flags;
+                context.flags &= ~512 /* WriteTypeParametersInQualifiedName */; // Avoids potential infinite loop when building for a claimspace with a generic
+                var name = typeParameterToName(type, context);
+                var defaultParameter = getDefaultFromTypeParameter(type);
+                var defaultParameterNode = defaultParameter && typeToTypeNodeHelper(defaultParameter, context);
+                context.flags = savedContextFlags;
+                return ts.factory.createTypeParameterDeclaration(name, constraintNode, defaultParameterNode);
             }
-            if (type.flags & 4194304 /* Index */) {
-                return getIndexType(getApparentType(type.type));
+            function typeParameterToDeclaration(type, context, constraint) {
+                if (constraint === void 0) { constraint = getConstraintOfTypeParameter(type); }
+                var constraintNode = constraint && typeToTypeNodeHelper(constraint, context);
+                return typeParameterToDeclarationWithConstraint(type, context, constraintNode);
             }
-            if (type.flags & 16777216 /* Conditional */) {
-                if (type.root.isDistributive) {
-                    var checkType = type.checkType;
-                    var constraint = getLowerBoundOfKeyType(checkType);
-                    if (constraint !== checkType) {
-                        return getConditionalTypeInstantiation(type, prependTypeMapping(type.root.checkType, constraint, type.mapper));
+            function symbolToParameterDeclaration(parameterSymbol, context, preserveModifierFlags, privateSymbolVisitor, bundledImports) {
+                var parameterDeclaration = ts.getDeclarationOfKind(parameterSymbol, 160 /* Parameter */);
+                if (!parameterDeclaration && !ts.isTransientSymbol(parameterSymbol)) {
+                    parameterDeclaration = ts.getDeclarationOfKind(parameterSymbol, 326 /* JSDocParameterTag */);
+                }
+                var parameterType = getTypeOfSymbol(parameterSymbol);
+                if (parameterDeclaration && isRequiredInitializedParameter(parameterDeclaration)) {
+                    parameterType = getOptionalType(parameterType);
+                }
+                if ((context.flags & 1073741824 /* NoUndefinedOptionalParameterType */) && parameterDeclaration && !ts.isJSDocParameterTag(parameterDeclaration) && isOptionalUninitializedParameter(parameterDeclaration)) {
+                    parameterType = getTypeWithFacts(parameterType, 524288 /* NEUndefined */);
+                }
+                var parameterTypeNode = serializeTypeForDeclaration(context, parameterType, parameterSymbol, context.enclosingDeclaration, privateSymbolVisitor, bundledImports);
+                var modifiers = !(context.flags & 8192 /* OmitParameterModifiers */) && preserveModifierFlags && parameterDeclaration && parameterDeclaration.modifiers ? parameterDeclaration.modifiers.map(ts.factory.cloneNode) : undefined;
+                var isRest = parameterDeclaration && ts.isRestParameter(parameterDeclaration) || ts.getCheckFlags(parameterSymbol) & 32768 /* RestParameter */;
+                var dotDotDotToken = isRest ? ts.factory.createToken(25 /* DotDotDotToken */) : undefined;
+                var name = parameterDeclaration ? parameterDeclaration.name ?
+                    parameterDeclaration.name.kind === 78 /* Identifier */ ? ts.setEmitFlags(ts.factory.cloneNode(parameterDeclaration.name), 16777216 /* NoAsciiEscaping */) :
+                        parameterDeclaration.name.kind === 157 /* QualifiedName */ ? ts.setEmitFlags(ts.factory.cloneNode(parameterDeclaration.name.right), 16777216 /* NoAsciiEscaping */) :
+                            cloneBindingName(parameterDeclaration.name) :
+                    ts.symbolName(parameterSymbol) :
+                    ts.symbolName(parameterSymbol);
+                var isOptional = parameterDeclaration && isOptionalParameter(parameterDeclaration) || ts.getCheckFlags(parameterSymbol) & 16384 /* OptionalParameter */;
+                var questionToken = isOptional ? ts.factory.createToken(57 /* QuestionToken */) : undefined;
+                var parameterNode = ts.factory.createParameterDeclaration(
+                /*decorators*/ undefined, modifiers, dotDotDotToken, name, questionToken, parameterTypeNode, 
+                /*initializer*/ undefined);
+                context.approximateLength += ts.symbolName(parameterSymbol).length + 3;
+                return parameterNode;
+                function cloneBindingName(node) {
+                    return elideInitializerAndSetEmitFlags(node);
+                    function elideInitializerAndSetEmitFlags(node) {
+                        if (context.tracker.trackSymbol && ts.isComputedPropertyName(node) && isLateBindableName(node)) {
+                            trackComputedName(node.expression, context.enclosingDeclaration, context);
+                        }
+                        var visited = ts.visitEachChild(node, elideInitializerAndSetEmitFlags, ts.nullTransformationContext, /*nodesVisitor*/ undefined, elideInitializerAndSetEmitFlags);
+                        if (ts.isBindingElement(visited)) {
+                            visited = ts.factory.updateBindingElement(visited, visited.dotDotDotToken, visited.propertyName, visited.name, 
+                            /*initializer*/ undefined);
+                        }
+                        if (!ts.nodeIsSynthesized(visited)) {
+                            visited = ts.factory.cloneNode(visited);
+                        }
+                        return ts.setEmitFlags(visited, 1 /* SingleLine */ | 16777216 /* NoAsciiEscaping */);
                     }
                 }
-                return type;
             }
-            if (type.flags & 1048576 /* Union */) {
-                return getUnionType(ts.sameMap(type.types, getLowerBoundOfKeyType));
+            function trackComputedName(accessExpression, enclosingDeclaration, context) {
+                if (!context.tracker.trackSymbol)
+                    return;
+                // get symbol of the first identifier of the entityName
+                var firstIdentifier = ts.getFirstIdentifier(accessExpression);
+                var name = resolveName(firstIdentifier, firstIdentifier.escapedText, 111551 /* Value */ | 1048576 /* ExportValue */, /*nodeNotFoundErrorMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true);
+                if (name) {
+                    context.tracker.trackSymbol(name, enclosingDeclaration, 111551 /* Value */);
+                }
             }
-            if (type.flags & 2097152 /* Intersection */) {
-                return getIntersectionType(ts.sameMap(type.types, getLowerBoundOfKeyType));
+            function lookupSymbolChain(symbol, context, meaning, yieldModuleSymbol) {
+                context.tracker.trackSymbol(symbol, context.enclosingDeclaration, meaning); // TODO: GH#18217
+                return lookupSymbolChainWorker(symbol, context, meaning, yieldModuleSymbol);
             }
-            return neverType;
-        }
-        /** Resolve the members of a mapped type { [P in K]: T } */
-        function resolveMappedTypeMembers(type) {
-            var members = ts.createSymbolTable();
-            var stringIndexInfo;
-            var numberIndexInfo;
-            // Resolve upfront such that recursive references see an empty object type.
-            setStructuredTypeMembers(type, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
-            // In { [P in K]: T }, we refer to P as the type parameter type, K as the constraint type,
-            // and T as the template type.
-            var typeParameter = getTypeParameterFromMappedType(type);
-            var constraintType = getConstraintTypeFromMappedType(type);
-            var templateType = getTemplateTypeFromMappedType(type.target || type);
-            var modifiersType = getApparentType(getModifiersTypeFromMappedType(type)); // The 'T' in 'keyof T'
-            var templateModifiers = getMappedTypeModifiers(type);
-            var include = keyofStringsOnly ? 128 /* StringLiteral */ : 8576 /* StringOrNumberLiteralOrUnique */;
-            if (isMappedTypeWithKeyofConstraintDeclaration(type)) {
-                // We have a { [P in keyof T]: X }
-                for (var _i = 0, _a = getPropertiesOfType(modifiersType); _i < _a.length; _i++) {
-                    var prop = _a[_i];
-                    addMemberForKeyType(getLiteralTypeFromProperty(prop, include));
+            function lookupSymbolChainWorker(symbol, context, meaning, yieldModuleSymbol) {
+                // Try to get qualified name if the symbol is not a type parameter and there is an enclosing declaration.
+                var chain;
+                var isTypeParameter = symbol.flags & 262144 /* TypeParameter */;
+                if (!isTypeParameter && (context.enclosingDeclaration || context.flags & 64 /* UseFullyQualifiedType */) && !(context.flags & 134217728 /* DoNotIncludeSymbolChain */)) {
+                    chain = ts.Debug.checkDefined(getSymbolChain(symbol, meaning, /*endOfChain*/ true));
+                    ts.Debug.assert(chain && chain.length > 0);
                 }
-                if (modifiersType.flags & 1 /* Any */ || getIndexInfoOfType(modifiersType, 0 /* String */)) {
-                    addMemberForKeyType(stringType);
+                else {
+                    chain = [symbol];
                 }
-                if (!keyofStringsOnly && getIndexInfoOfType(modifiersType, 1 /* Number */)) {
-                    addMemberForKeyType(numberType);
+                return chain;
+                /** @param endOfChain Set to false for recursive calls; non-recursive calls should always output something. */
+                function getSymbolChain(symbol, meaning, endOfChain) {
+                    var accessibleSymbolChain = getAccessibleSymbolChain(symbol, context.enclosingDeclaration, meaning, !!(context.flags & 128 /* UseOnlyExternalAliasing */));
+                    var parentSpecifiers;
+                    if (!accessibleSymbolChain ||
+                        needsQualification(accessibleSymbolChain[0], context.enclosingDeclaration, accessibleSymbolChain.length === 1 ? meaning : getQualifiedLeftMeaning(meaning))) {
+                        // Go up and add our parent.
+                        var parents_1 = getContainersOfSymbol(accessibleSymbolChain ? accessibleSymbolChain[0] : symbol, context.enclosingDeclaration, meaning);
+                        if (ts.length(parents_1)) {
+                            parentSpecifiers = parents_1.map(function (symbol) {
+                                return ts.some(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)
+                                    ? getSpecifierForModuleSymbol(symbol, context)
+                                    : undefined;
+                            });
+                            var indices = parents_1.map(function (_, i) { return i; });
+                            indices.sort(sortByBestName);
+                            var sortedParents = indices.map(function (i) { return parents_1[i]; });
+                            for (var _i = 0, sortedParents_1 = sortedParents; _i < sortedParents_1.length; _i++) {
+                                var parent = sortedParents_1[_i];
+                                var parentChain = getSymbolChain(parent, getQualifiedLeftMeaning(meaning), /*endOfChain*/ false);
+                                if (parentChain) {
+                                    if (parent.exports && parent.exports.get("export=" /* ExportEquals */) &&
+                                        getSymbolIfSameReference(parent.exports.get("export=" /* ExportEquals */), symbol)) {
+                                        // parentChain root _is_ symbol - symbol is a module export=, so it kinda looks like it's own parent
+                                        // No need to lookup an alias for the symbol in itself
+                                        accessibleSymbolChain = parentChain;
+                                        break;
+                                    }
+                                    accessibleSymbolChain = parentChain.concat(accessibleSymbolChain || [getAliasForSymbolInContainer(parent, symbol) || symbol]);
+                                    break;
+                                }
+                            }
+                        }
+                    }
+                    if (accessibleSymbolChain) {
+                        return accessibleSymbolChain;
+                    }
+                    if (
+                    // If this is the last part of outputting the symbol, always output. The cases apply only to parent symbols.
+                    endOfChain ||
+                        // If a parent symbol is an anonymous type, don't write it.
+                        !(symbol.flags & (2048 /* TypeLiteral */ | 4096 /* ObjectLiteral */))) {
+                        // If a parent symbol is an external module, don't write it. (We prefer just `x` vs `"foo/bar".x`.)
+                        if (!endOfChain && !yieldModuleSymbol && !!ts.forEach(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
+                            return;
+                        }
+                        return [symbol];
+                    }
+                    function sortByBestName(a, b) {
+                        var specifierA = parentSpecifiers[a];
+                        var specifierB = parentSpecifiers[b];
+                        if (specifierA && specifierB) {
+                            var isBRelative = ts.pathIsRelative(specifierB);
+                            if (ts.pathIsRelative(specifierA) === isBRelative) {
+                                // Both relative or both non-relative, sort by number of parts
+                                return ts.moduleSpecifiers.countPathComponents(specifierA) - ts.moduleSpecifiers.countPathComponents(specifierB);
+                            }
+                            if (isBRelative) {
+                                // A is non-relative, B is relative: prefer A
+                                return -1;
+                            }
+                            // A is relative, B is non-relative: prefer B
+                            return 1;
+                        }
+                        return 0;
+                    }
                 }
             }
-            else {
-                forEachType(getLowerBoundOfKeyType(constraintType), addMemberForKeyType);
+            function typeParametersToTypeParameterDeclarations(symbol, context) {
+                var typeParameterNodes;
+                var targetSymbol = getTargetSymbol(symbol);
+                if (targetSymbol.flags & (32 /* Class */ | 64 /* Interface */ | 524288 /* TypeAlias */)) {
+                    typeParameterNodes = ts.factory.createNodeArray(ts.map(getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol), function (tp) { return typeParameterToDeclaration(tp, context); }));
+                }
+                return typeParameterNodes;
             }
-            setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
-            function addMemberForKeyType(t) {
-                // Create a mapper from T to the current iteration type constituent. Then, if the
-                // mapped type is itself an instantiated type, combine the iteration mapper with the
-                // instantiation mapper.
-                var templateMapper = appendTypeMapping(type.mapper, typeParameter, t);
-                // If the current iteration type constituent is a string literal type, create a property.
-                // Otherwise, for type string create a string index signature.
-                if (isTypeUsableAsPropertyName(t)) {
-                    var propName = getPropertyNameFromType(t);
-                    var modifiersProp = getPropertyOfType(modifiersType, propName);
-                    var isOptional = !!(templateModifiers & 4 /* IncludeOptional */ ||
-                        !(templateModifiers & 8 /* ExcludeOptional */) && modifiersProp && modifiersProp.flags & 16777216 /* Optional */);
-                    var isReadonly = !!(templateModifiers & 1 /* IncludeReadonly */ ||
-                        !(templateModifiers & 2 /* ExcludeReadonly */) && modifiersProp && isReadonlySymbol(modifiersProp));
-                    var stripOptional = strictNullChecks && !isOptional && modifiersProp && modifiersProp.flags & 16777216 /* Optional */;
-                    var prop = createSymbol(4 /* Property */ | (isOptional ? 16777216 /* Optional */ : 0), propName, 262144 /* Mapped */ | (isReadonly ? 8 /* Readonly */ : 0) | (stripOptional ? 524288 /* StripOptional */ : 0));
-                    prop.mappedType = type;
-                    prop.mapper = templateMapper;
-                    if (modifiersProp) {
-                        prop.syntheticOrigin = modifiersProp;
-                        prop.declarations = modifiersProp.declarations;
-                    }
-                    prop.nameType = t;
-                    members.set(propName, prop);
-                }
-                else if (t.flags & (1 /* Any */ | 4 /* String */ | 8 /* Number */ | 32 /* Enum */)) {
-                    var propType = instantiateType(templateType, templateMapper);
-                    if (t.flags & (1 /* Any */ | 4 /* String */)) {
-                        stringIndexInfo = createIndexInfo(propType, !!(templateModifiers & 1 /* IncludeReadonly */));
+            function lookupTypeParameterNodes(chain, index, context) {
+                var _a;
+                ts.Debug.assert(chain && 0 <= index && index < chain.length);
+                var symbol = chain[index];
+                var symbolId = getSymbolId(symbol);
+                if ((_a = context.typeParameterSymbolList) === null || _a === void 0 ? void 0 : _a.has(symbolId)) {
+                    return undefined;
+                }
+                (context.typeParameterSymbolList || (context.typeParameterSymbolList = new ts.Set())).add(symbolId);
+                var typeParameterNodes;
+                if (context.flags & 512 /* WriteTypeParametersInQualifiedName */ && index < (chain.length - 1)) {
+                    var parentSymbol = symbol;
+                    var nextSymbol_1 = chain[index + 1];
+                    if (ts.getCheckFlags(nextSymbol_1) & 1 /* Instantiated */) {
+                        var params = getTypeParametersOfClassOrInterface(parentSymbol.flags & 2097152 /* Alias */ ? resolveAlias(parentSymbol) : parentSymbol);
+                        typeParameterNodes = mapToTypeNodes(ts.map(params, function (t) { return getMappedType(t, nextSymbol_1.mapper); }), context);
                     }
                     else {
-                        numberIndexInfo = createIndexInfo(numberIndexInfo ? getUnionType([numberIndexInfo.type, propType]) : propType, !!(templateModifiers & 1 /* IncludeReadonly */));
+                        typeParameterNodes = typeParametersToTypeParameterDeclarations(symbol, context);
                     }
                 }
+                return typeParameterNodes;
             }
-        }
-        function getTypeOfMappedSymbol(symbol) {
-            if (!symbol.type) {
-                if (!pushTypeResolution(symbol, 0 /* Type */)) {
-                    return errorType;
-                }
-                var templateType = getTemplateTypeFromMappedType(symbol.mappedType.target || symbol.mappedType);
-                var propType = instantiateType(templateType, symbol.mapper);
-                // When creating an optional property in strictNullChecks mode, if 'undefined' isn't assignable to the
-                // type, we include 'undefined' in the type. Similarly, when creating a non-optional property in strictNullChecks
-                // mode, if the underlying property is optional we remove 'undefined' from the type.
-                var type = strictNullChecks && symbol.flags & 16777216 /* Optional */ && !maybeTypeOfKind(propType, 32768 /* Undefined */ | 16384 /* Void */) ? getOptionalType(propType) :
-                    symbol.checkFlags & 524288 /* StripOptional */ ? getTypeWithFacts(propType, 524288 /* NEUndefined */) :
-                        propType;
-                if (!popTypeResolution()) {
-                    error(currentNode, ts.Diagnostics.Type_of_property_0_circularly_references_itself_in_mapped_type_1, symbolToString(symbol), typeToString(symbol.mappedType));
-                    type = errorType;
+            /**
+             * Given A[B][C][D], finds A[B]
+             */
+            function getTopmostIndexedAccessType(top) {
+                if (ts.isIndexedAccessTypeNode(top.objectType)) {
+                    return getTopmostIndexedAccessType(top.objectType);
                 }
-                symbol.type = type;
-                symbol.mapper = undefined;
+                return top;
             }
-            return symbol.type;
-        }
-        function getTypeParameterFromMappedType(type) {
-            return type.typeParameter ||
-                (type.typeParameter = getDeclaredTypeOfTypeParameter(getSymbolOfNode(type.declaration.typeParameter)));
-        }
-        function getConstraintTypeFromMappedType(type) {
-            return type.constraintType ||
-                (type.constraintType = getConstraintOfTypeParameter(getTypeParameterFromMappedType(type)) || errorType);
-        }
-        function getTemplateTypeFromMappedType(type) {
-            return type.templateType ||
-                (type.templateType = type.declaration.type ?
-                    instantiateType(addOptionality(getTypeFromTypeNode(type.declaration.type), !!(getMappedTypeModifiers(type) & 4 /* IncludeOptional */)), type.mapper) :
-                    errorType);
-        }
-        function getConstraintDeclarationForMappedType(type) {
-            return ts.getEffectiveConstraintOfTypeParameter(type.declaration.typeParameter);
-        }
-        function isMappedTypeWithKeyofConstraintDeclaration(type) {
-            var constraintDeclaration = getConstraintDeclarationForMappedType(type); // TODO: GH#18217
-            return constraintDeclaration.kind === 184 /* TypeOperator */ &&
-                constraintDeclaration.operator === 134 /* KeyOfKeyword */;
-        }
-        function getModifiersTypeFromMappedType(type) {
-            if (!type.modifiersType) {
-                if (isMappedTypeWithKeyofConstraintDeclaration(type)) {
-                    // If the constraint declaration is a 'keyof T' node, the modifiers type is T. We check
-                    // AST nodes here because, when T is a non-generic type, the logic below eagerly resolves
-                    // 'keyof T' to a literal union type and we can't recover T from that type.
-                    type.modifiersType = instantiateType(getTypeFromTypeNode(getConstraintDeclarationForMappedType(type).type), type.mapper);
+            function getSpecifierForModuleSymbol(symbol, context) {
+                var _a;
+                var file = ts.getDeclarationOfKind(symbol, 297 /* SourceFile */);
+                if (!file) {
+                    var equivalentFileSymbol = ts.firstDefined(symbol.declarations, function (d) { return getFileSymbolIfFileSymbolExportEqualsContainer(d, symbol); });
+                    if (equivalentFileSymbol) {
+                        file = ts.getDeclarationOfKind(equivalentFileSymbol, 297 /* SourceFile */);
+                    }
                 }
-                else {
-                    // Otherwise, get the declared constraint type, and if the constraint type is a type parameter,
-                    // get the constraint of that type parameter. If the resulting type is an indexed type 'keyof T',
-                    // the modifiers type is T. Otherwise, the modifiers type is unknown.
-                    var declaredType = getTypeFromMappedTypeNode(type.declaration);
-                    var constraint = getConstraintTypeFromMappedType(declaredType);
-                    var extendedConstraint = constraint && constraint.flags & 262144 /* TypeParameter */ ? getConstraintOfTypeParameter(constraint) : constraint;
-                    type.modifiersType = extendedConstraint && extendedConstraint.flags & 4194304 /* Index */ ? instantiateType(extendedConstraint.type, type.mapper) : unknownType;
+                if (file && file.moduleName !== undefined) {
+                    // Use the amd name if it is available
+                    return file.moduleName;
                 }
-            }
-            return type.modifiersType;
-        }
-        function getMappedTypeModifiers(type) {
-            var declaration = type.declaration;
-            return (declaration.readonlyToken ? declaration.readonlyToken.kind === 40 /* MinusToken */ ? 2 /* ExcludeReadonly */ : 1 /* IncludeReadonly */ : 0) |
-                (declaration.questionToken ? declaration.questionToken.kind === 40 /* MinusToken */ ? 8 /* ExcludeOptional */ : 4 /* IncludeOptional */ : 0);
-        }
-        function getMappedTypeOptionality(type) {
-            var modifiers = getMappedTypeModifiers(type);
-            return modifiers & 8 /* ExcludeOptional */ ? -1 : modifiers & 4 /* IncludeOptional */ ? 1 : 0;
-        }
-        function getCombinedMappedTypeOptionality(type) {
-            var optionality = getMappedTypeOptionality(type);
-            var modifiersType = getModifiersTypeFromMappedType(type);
-            return optionality || (isGenericMappedType(modifiersType) ? getMappedTypeOptionality(modifiersType) : 0);
-        }
-        function isPartialMappedType(type) {
-            return !!(ts.getObjectFlags(type) & 32 /* Mapped */ && getMappedTypeModifiers(type) & 4 /* IncludeOptional */);
-        }
-        function isGenericMappedType(type) {
-            return !!(ts.getObjectFlags(type) & 32 /* Mapped */) && isGenericIndexType(getConstraintTypeFromMappedType(type));
-        }
-        function resolveStructuredTypeMembers(type) {
-            if (!type.members) {
-                if (type.flags & 524288 /* Object */) {
-                    if (type.objectFlags & 4 /* Reference */) {
-                        resolveTypeReferenceMembers(type);
+                if (!file) {
+                    if (context.tracker.trackReferencedAmbientModule) {
+                        var ambientDecls = ts.filter(symbol.declarations, ts.isAmbientModule);
+                        if (ts.length(ambientDecls)) {
+                            for (var _i = 0, ambientDecls_1 = ambientDecls; _i < ambientDecls_1.length; _i++) {
+                                var decl = ambientDecls_1[_i];
+                                context.tracker.trackReferencedAmbientModule(decl, symbol);
+                            }
+                        }
                     }
-                    else if (type.objectFlags & 3 /* ClassOrInterface */) {
-                        resolveClassOrInterfaceMembers(type);
+                    if (ambientModuleSymbolRegex.test(symbol.escapedName)) {
+                        return symbol.escapedName.substring(1, symbol.escapedName.length - 1);
                     }
-                    else if (type.objectFlags & 2048 /* ReverseMapped */) {
-                        resolveReverseMappedTypeMembers(type);
+                }
+                if (!context.enclosingDeclaration || !context.tracker.moduleResolverHost) {
+                    // If there's no context declaration, we can't lookup a non-ambient specifier, so we just use the symbol name
+                    if (ambientModuleSymbolRegex.test(symbol.escapedName)) {
+                        return symbol.escapedName.substring(1, symbol.escapedName.length - 1);
                     }
-                    else if (type.objectFlags & 16 /* Anonymous */) {
-                        resolveAnonymousTypeMembers(type);
+                    return ts.getSourceFileOfNode(ts.getNonAugmentationDeclaration(symbol)).fileName; // A resolver may not be provided for baselines and errors - in those cases we use the fileName in full
+                }
+                var contextFile = ts.getSourceFileOfNode(ts.getOriginalNode(context.enclosingDeclaration));
+                var links = getSymbolLinks(symbol);
+                var specifier = links.specifierCache && links.specifierCache.get(contextFile.path);
+                if (!specifier) {
+                    var isBundle_1 = !!ts.outFile(compilerOptions);
+                    // For declaration bundles, we need to generate absolute paths relative to the common source dir for imports,
+                    // just like how the declaration emitter does for the ambient module declarations - we can easily accomplish this
+                    // using the `baseUrl` compiler option (which we would otherwise never use in declaration emit) and a non-relative
+                    // specifier preference
+                    var moduleResolverHost = context.tracker.moduleResolverHost;
+                    var specifierCompilerOptions = isBundle_1 ? __assign(__assign({}, compilerOptions), { baseUrl: moduleResolverHost.getCommonSourceDirectory() }) : compilerOptions;
+                    specifier = ts.first(ts.moduleSpecifiers.getModuleSpecifiers(symbol, specifierCompilerOptions, contextFile, moduleResolverHost, { importModuleSpecifierPreference: isBundle_1 ? "non-relative" : "relative", importModuleSpecifierEnding: isBundle_1 ? "minimal" : undefined }));
+                    (_a = links.specifierCache) !== null && _a !== void 0 ? _a : (links.specifierCache = new ts.Map());
+                    links.specifierCache.set(contextFile.path, specifier);
+                }
+                return specifier;
+            }
+            function symbolToTypeNode(symbol, context, meaning, overrideTypeArguments) {
+                var chain = lookupSymbolChain(symbol, context, meaning, !(context.flags & 16384 /* UseAliasDefinedOutsideCurrentScope */)); // If we're using aliases outside the current scope, dont bother with the module
+                var isTypeOf = meaning === 111551 /* Value */;
+                if (ts.some(chain[0].declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
+                    // module is root, must use `ImportTypeNode`
+                    var nonRootParts = chain.length > 1 ? createAccessFromSymbolChain(chain, chain.length - 1, 1) : undefined;
+                    var typeParameterNodes = overrideTypeArguments || lookupTypeParameterNodes(chain, 0, context);
+                    var specifier = getSpecifierForModuleSymbol(chain[0], context);
+                    if (!(context.flags & 67108864 /* AllowNodeModulesRelativePaths */) && ts.getEmitModuleResolutionKind(compilerOptions) === ts.ModuleResolutionKind.NodeJs && specifier.indexOf("/node_modules/") >= 0) {
+                        // If ultimately we can only name the symbol with a reference that dives into a `node_modules` folder, we should error
+                        // since declaration files with these kinds of references are liable to fail when published :(
+                        context.encounteredError = true;
+                        if (context.tracker.reportLikelyUnsafeImportRequiredError) {
+                            context.tracker.reportLikelyUnsafeImportRequiredError(specifier);
+                        }
                     }
-                    else if (type.objectFlags & 32 /* Mapped */) {
-                        resolveMappedTypeMembers(type);
+                    var lit = ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral(specifier));
+                    if (context.tracker.trackExternalModuleSymbolOfImportTypeNode)
+                        context.tracker.trackExternalModuleSymbolOfImportTypeNode(chain[0]);
+                    context.approximateLength += specifier.length + 10; // specifier + import("")
+                    if (!nonRootParts || ts.isEntityName(nonRootParts)) {
+                        if (nonRootParts) {
+                            var lastId = ts.isIdentifier(nonRootParts) ? nonRootParts : nonRootParts.right;
+                            lastId.typeArguments = undefined;
+                        }
+                        return ts.factory.createImportTypeNode(lit, nonRootParts, typeParameterNodes, isTypeOf);
+                    }
+                    else {
+                        var splitNode = getTopmostIndexedAccessType(nonRootParts);
+                        var qualifier = splitNode.objectType.typeName;
+                        return ts.factory.createIndexedAccessTypeNode(ts.factory.createImportTypeNode(lit, qualifier, typeParameterNodes, isTypeOf), splitNode.indexType);
                     }
                 }
-                else if (type.flags & 1048576 /* Union */) {
-                    resolveUnionTypeMembers(type);
+                var entityName = createAccessFromSymbolChain(chain, chain.length - 1, 0);
+                if (ts.isIndexedAccessTypeNode(entityName)) {
+                    return entityName; // Indexed accesses can never be `typeof`
                 }
-                else if (type.flags & 2097152 /* Intersection */) {
-                    resolveIntersectionTypeMembers(type);
+                if (isTypeOf) {
+                    return ts.factory.createTypeQueryNode(entityName);
                 }
-            }
-            return type;
-        }
-        /** Return properties of an object type or an empty array for other types */
-        function getPropertiesOfObjectType(type) {
-            if (type.flags & 524288 /* Object */) {
-                return resolveStructuredTypeMembers(type).properties;
-            }
-            return ts.emptyArray;
-        }
-        /** If the given type is an object type and that type has a property by the given name,
-         * return the symbol for that property. Otherwise return undefined.
-         */
-        function getPropertyOfObjectType(type, name) {
-            if (type.flags & 524288 /* Object */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                var symbol = resolved.members.get(name);
-                if (symbol && symbolIsValue(symbol)) {
-                    return symbol;
+                else {
+                    var lastId = ts.isIdentifier(entityName) ? entityName : entityName.right;
+                    var lastTypeArgs = lastId.typeArguments;
+                    lastId.typeArguments = undefined;
+                    return ts.factory.createTypeReferenceNode(entityName, lastTypeArgs);
                 }
-            }
-        }
-        function getPropertiesOfUnionOrIntersectionType(type) {
-            if (!type.resolvedProperties) {
-                var members = ts.createSymbolTable();
-                for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
-                    var current = _a[_i];
-                    for (var _b = 0, _c = getPropertiesOfType(current); _b < _c.length; _b++) {
-                        var prop = _c[_b];
-                        if (!members.has(prop.escapedName)) {
-                            var combinedProp = getPropertyOfUnionOrIntersectionType(type, prop.escapedName);
-                            if (combinedProp) {
-                                members.set(prop.escapedName, combinedProp);
-                            }
+                function createAccessFromSymbolChain(chain, index, stopper) {
+                    var typeParameterNodes = index === (chain.length - 1) ? overrideTypeArguments : lookupTypeParameterNodes(chain, index, context);
+                    var symbol = chain[index];
+                    var parent = chain[index - 1];
+                    var symbolName;
+                    if (index === 0) {
+                        context.flags |= 16777216 /* InInitialEntityName */;
+                        symbolName = getNameOfSymbolAsWritten(symbol, context);
+                        context.approximateLength += (symbolName ? symbolName.length : 0) + 1;
+                        context.flags ^= 16777216 /* InInitialEntityName */;
+                    }
+                    else {
+                        if (parent && getExportsOfSymbol(parent)) {
+                            var exports_1 = getExportsOfSymbol(parent);
+                            ts.forEachEntry(exports_1, function (ex, name) {
+                                if (getSymbolIfSameReference(ex, symbol) && !isLateBoundName(name) && name !== "export=" /* ExportEquals */) {
+                                    symbolName = ts.unescapeLeadingUnderscores(name);
+                                    return true;
+                                }
+                            });
                         }
                     }
-                    // The properties of a union type are those that are present in all constituent types, so
-                    // we only need to check the properties of the first type without index signature
-                    if (type.flags & 1048576 /* Union */ && !getIndexInfoOfType(current, 0 /* String */) && !getIndexInfoOfType(current, 1 /* Number */)) {
-                        break;
+                    if (!symbolName) {
+                        symbolName = getNameOfSymbolAsWritten(symbol, context);
                     }
-                }
-                type.resolvedProperties = getNamedMembers(members);
-            }
-            return type.resolvedProperties;
-        }
-        function getPropertiesOfType(type) {
-            type = getReducedApparentType(type);
-            return type.flags & 3145728 /* UnionOrIntersection */ ?
-                getPropertiesOfUnionOrIntersectionType(type) :
-                getPropertiesOfObjectType(type);
-        }
-        function isTypeInvalidDueToUnionDiscriminant(contextualType, obj) {
-            var list = obj.properties;
-            return list.some(function (property) {
-                var nameType = property.name && getLiteralTypeFromPropertyName(property.name);
-                var name = nameType && isTypeUsableAsPropertyName(nameType) ? getPropertyNameFromType(nameType) : undefined;
-                var expected = name === undefined ? undefined : getTypeOfPropertyOfType(contextualType, name);
-                return !!expected && isLiteralType(expected) && !isTypeAssignableTo(getTypeOfNode(property), expected);
-            });
-        }
-        function getAllPossiblePropertiesOfTypes(types) {
-            var unionType = getUnionType(types);
-            if (!(unionType.flags & 1048576 /* Union */)) {
-                return getAugmentedPropertiesOfType(unionType);
-            }
-            var props = ts.createSymbolTable();
-            for (var _i = 0, types_4 = types; _i < types_4.length; _i++) {
-                var memberType = types_4[_i];
-                for (var _a = 0, _b = getAugmentedPropertiesOfType(memberType); _a < _b.length; _a++) {
-                    var escapedName = _b[_a].escapedName;
-                    if (!props.has(escapedName)) {
-                        var prop = createUnionOrIntersectionProperty(unionType, escapedName);
-                        // May be undefined if the property is private
-                        if (prop)
-                            props.set(escapedName, prop);
+                    context.approximateLength += symbolName.length + 1;
+                    if (!(context.flags & 16 /* ForbidIndexedAccessSymbolReferences */) && parent &&
+                        getMembersOfSymbol(parent) && getMembersOfSymbol(parent).get(symbol.escapedName) &&
+                        getSymbolIfSameReference(getMembersOfSymbol(parent).get(symbol.escapedName), symbol)) {
+                        // Should use an indexed access
+                        var LHS = createAccessFromSymbolChain(chain, index - 1, stopper);
+                        if (ts.isIndexedAccessTypeNode(LHS)) {
+                            return ts.factory.createIndexedAccessTypeNode(LHS, ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral(symbolName)));
+                        }
+                        else {
+                            return ts.factory.createIndexedAccessTypeNode(ts.factory.createTypeReferenceNode(LHS, typeParameterNodes), ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral(symbolName)));
+                        }
+                    }
+                    var identifier = ts.setEmitFlags(ts.factory.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
+                    identifier.symbol = symbol;
+                    if (index > stopper) {
+                        var LHS = createAccessFromSymbolChain(chain, index - 1, stopper);
+                        if (!ts.isEntityName(LHS)) {
+                            return ts.Debug.fail("Impossible construct - an export of an indexed access cannot be reachable");
+                        }
+                        return ts.factory.createQualifiedName(LHS, identifier);
                     }
+                    return identifier;
                 }
             }
-            return ts.arrayFrom(props.values());
-        }
-        function getConstraintOfType(type) {
-            return type.flags & 262144 /* TypeParameter */ ? getConstraintOfTypeParameter(type) :
-                type.flags & 8388608 /* IndexedAccess */ ? getConstraintOfIndexedAccess(type) :
-                    type.flags & 16777216 /* Conditional */ ? getConstraintOfConditionalType(type) :
-                        getBaseConstraintOfType(type);
-        }
-        function getConstraintOfTypeParameter(typeParameter) {
-            return hasNonCircularBaseConstraint(typeParameter) ? getConstraintFromTypeParameter(typeParameter) : undefined;
-        }
-        function getConstraintOfIndexedAccess(type) {
-            return hasNonCircularBaseConstraint(type) ? getConstraintFromIndexedAccess(type) : undefined;
-        }
-        function getSimplifiedTypeOrConstraint(type) {
-            var simplified = getSimplifiedType(type, /*writing*/ false);
-            return simplified !== type ? simplified : getConstraintOfType(type);
-        }
-        function getConstraintFromIndexedAccess(type) {
-            var indexConstraint = getSimplifiedTypeOrConstraint(type.indexType);
-            if (indexConstraint && indexConstraint !== type.indexType) {
-                var indexedAccess = getIndexedAccessTypeOrUndefined(type.objectType, indexConstraint);
-                if (indexedAccess) {
-                    return indexedAccess;
+            function typeParameterShadowsNameInScope(escapedName, context, type) {
+                var result = resolveName(context.enclosingDeclaration, escapedName, 788968 /* Type */, /*nameNotFoundArg*/ undefined, escapedName, /*isUse*/ false);
+                if (result) {
+                    if (result.flags & 262144 /* TypeParameter */ && result === type.symbol) {
+                        return false;
+                    }
+                    return true;
                 }
+                return false;
             }
-            var objectConstraint = getSimplifiedTypeOrConstraint(type.objectType);
-            if (objectConstraint && objectConstraint !== type.objectType) {
-                return getIndexedAccessTypeOrUndefined(objectConstraint, type.indexType);
-            }
-            return undefined;
-        }
-        function getDefaultConstraintOfConditionalType(type) {
-            if (!type.resolvedDefaultConstraint) {
-                // An `any` branch of a conditional type would normally be viral - specifically, without special handling here,
-                // a conditional type with a single branch of type `any` would be assignable to anything, since it's constraint would simplify to
-                // just `any`. This result is _usually_ unwanted - so instead here we elide an `any` branch from the constraint type,
-                // in effect treating `any` like `never` rather than `unknown` in this location.
-                var trueConstraint = getInferredTrueTypeFromConditionalType(type);
-                var falseConstraint = getFalseTypeFromConditionalType(type);
-                type.resolvedDefaultConstraint = isTypeAny(trueConstraint) ? falseConstraint : isTypeAny(falseConstraint) ? trueConstraint : getUnionType([trueConstraint, falseConstraint]);
-            }
-            return type.resolvedDefaultConstraint;
-        }
-        function getConstraintOfDistributiveConditionalType(type) {
-            // Check if we have a conditional type of the form 'T extends U ? X : Y', where T is a constrained
-            // type parameter. If so, create an instantiation of the conditional type where T is replaced
-            // with its constraint. We do this because if the constraint is a union type it will be distributed
-            // over the conditional type and possibly reduced. For example, 'T extends undefined ? never : T'
-            // removes 'undefined' from T.
-            // We skip returning a distributive constraint for a restrictive instantiation of a conditional type
-            // as the constraint for all type params (check type included) have been replace with `unknown`, which
-            // is going to produce even more false positive/negative results than the distribute constraint already does.
-            // Please note: the distributive constraint is a kludge for emulating what a negated type could to do filter
-            // a union - once negated types exist and are applied to the conditional false branch, this "constraint"
-            // likely doesn't need to exist.
-            if (type.root.isDistributive && type.restrictiveInstantiation !== type) {
-                var simplified = getSimplifiedType(type.checkType, /*writing*/ false);
-                var constraint = simplified === type.checkType ? getConstraintOfType(simplified) : simplified;
-                if (constraint && constraint !== type.checkType) {
-                    var instantiated = getConditionalTypeInstantiation(type, prependTypeMapping(type.root.checkType, constraint, type.mapper));
-                    if (!(instantiated.flags & 131072 /* Never */)) {
-                        return instantiated;
+            function typeParameterToName(type, context) {
+                var _a;
+                if (context.flags & 4 /* GenerateNamesForShadowedTypeParams */ && context.typeParameterNames) {
+                    var cached = context.typeParameterNames.get(getTypeId(type));
+                    if (cached) {
+                        return cached;
                     }
                 }
-            }
-            return undefined;
-        }
-        function getConstraintFromConditionalType(type) {
-            return getConstraintOfDistributiveConditionalType(type) || getDefaultConstraintOfConditionalType(type);
-        }
-        function getConstraintOfConditionalType(type) {
-            return hasNonCircularBaseConstraint(type) ? getConstraintFromConditionalType(type) : undefined;
-        }
-        function getEffectiveConstraintOfIntersection(types, targetIsUnion) {
-            var constraints;
-            var hasDisjointDomainType = false;
-            for (var _i = 0, types_5 = types; _i < types_5.length; _i++) {
-                var t = types_5[_i];
-                if (t.flags & 63176704 /* Instantiable */) {
-                    // We keep following constraints as long as we have an instantiable type that is known
-                    // not to be circular or infinite (hence we stop on index access types).
-                    var constraint = getConstraintOfType(t);
-                    while (constraint && constraint.flags & (262144 /* TypeParameter */ | 4194304 /* Index */ | 16777216 /* Conditional */)) {
-                        constraint = getConstraintOfType(constraint);
+                var result = symbolToName(type.symbol, context, 788968 /* Type */, /*expectsIdentifier*/ true);
+                if (!(result.kind & 78 /* Identifier */)) {
+                    return ts.factory.createIdentifier("(Missing type parameter)");
+                }
+                if (context.flags & 4 /* GenerateNamesForShadowedTypeParams */) {
+                    var rawtext = result.escapedText;
+                    var i = 0;
+                    var text = rawtext;
+                    while (((_a = context.typeParameterNamesByText) === null || _a === void 0 ? void 0 : _a.has(text)) || typeParameterShadowsNameInScope(text, context, type)) {
+                        i++;
+                        text = rawtext + "_" + i;
                     }
-                    if (constraint) {
-                        constraints = ts.append(constraints, constraint);
-                        if (targetIsUnion) {
-                            constraints = ts.append(constraints, t);
-                        }
+                    if (text !== rawtext) {
+                        result = ts.factory.createIdentifier(text, result.typeArguments);
                     }
+                    (context.typeParameterNames || (context.typeParameterNames = new ts.Map())).set(getTypeId(type), result);
+                    (context.typeParameterNamesByText || (context.typeParameterNamesByText = new ts.Set())).add(result.escapedText);
                 }
-                else if (t.flags & 67238908 /* DisjointDomains */) {
-                    hasDisjointDomainType = true;
-                }
+                return result;
             }
-            // If the target is a union type or if we are intersecting with types belonging to one of the
-            // disjoint domains, we may end up producing a constraint that hasn't been examined before.
-            if (constraints && (targetIsUnion || hasDisjointDomainType)) {
-                if (hasDisjointDomainType) {
-                    // We add any types belong to one of the disjoint domains because they might cause the final
-                    // intersection operation to reduce the union constraints.
-                    for (var _a = 0, types_6 = types; _a < types_6.length; _a++) {
-                        var t = types_6[_a];
-                        if (t.flags & 67238908 /* DisjointDomains */) {
-                            constraints = ts.append(constraints, t);
-                        }
+            function symbolToName(symbol, context, meaning, expectsIdentifier) {
+                var chain = lookupSymbolChain(symbol, context, meaning);
+                if (expectsIdentifier && chain.length !== 1
+                    && !context.encounteredError
+                    && !(context.flags & 65536 /* AllowQualifedNameInPlaceOfIdentifier */)) {
+                    context.encounteredError = true;
+                }
+                return createEntityNameFromSymbolChain(chain, chain.length - 1);
+                function createEntityNameFromSymbolChain(chain, index) {
+                    var typeParameterNodes = lookupTypeParameterNodes(chain, index, context);
+                    var symbol = chain[index];
+                    if (index === 0) {
+                        context.flags |= 16777216 /* InInitialEntityName */;
+                    }
+                    var symbolName = getNameOfSymbolAsWritten(symbol, context);
+                    if (index === 0) {
+                        context.flags ^= 16777216 /* InInitialEntityName */;
                     }
+                    var identifier = ts.setEmitFlags(ts.factory.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
+                    identifier.symbol = symbol;
+                    return index > 0 ? ts.factory.createQualifiedName(createEntityNameFromSymbolChain(chain, index - 1), identifier) : identifier;
                 }
-                return getIntersectionType(constraints);
-            }
-            return undefined;
-        }
-        function getBaseConstraintOfType(type) {
-            if (type.flags & (58982400 /* InstantiableNonPrimitive */ | 3145728 /* UnionOrIntersection */)) {
-                var constraint = getResolvedBaseConstraint(type);
-                return constraint !== noConstraintType && constraint !== circularConstraintType ? constraint : undefined;
             }
-            return type.flags & 4194304 /* Index */ ? keyofConstraintType : undefined;
-        }
-        /**
-         * This is similar to `getBaseConstraintOfType` except it returns the input type if there's no base constraint, instead of `undefined`
-         * It also doesn't map indexes to `string`, as where this is used this would be unneeded (and likely undesirable)
-         */
-        function getBaseConstraintOrType(type) {
-            return getBaseConstraintOfType(type) || type;
-        }
-        function hasNonCircularBaseConstraint(type) {
-            return getResolvedBaseConstraint(type) !== circularConstraintType;
-        }
-        /**
-         * Return the resolved base constraint of a type variable. The noConstraintType singleton is returned if the
-         * type variable has no constraint, and the circularConstraintType singleton is returned if the constraint
-         * circularly references the type variable.
-         */
-        function getResolvedBaseConstraint(type) {
-            var nonTerminating = false;
-            return type.resolvedBaseConstraint ||
-                (type.resolvedBaseConstraint = getTypeWithThisArgument(getImmediateBaseConstraint(type), type));
-            function getImmediateBaseConstraint(t) {
-                if (!t.immediateBaseConstraint) {
-                    if (!pushTypeResolution(t, 4 /* ImmediateBaseConstraint */)) {
-                        return circularConstraintType;
+            function symbolToExpression(symbol, context, meaning) {
+                var chain = lookupSymbolChain(symbol, context, meaning);
+                return createExpressionFromSymbolChain(chain, chain.length - 1);
+                function createExpressionFromSymbolChain(chain, index) {
+                    var typeParameterNodes = lookupTypeParameterNodes(chain, index, context);
+                    var symbol = chain[index];
+                    if (index === 0) {
+                        context.flags |= 16777216 /* InInitialEntityName */;
                     }
-                    if (constraintDepth >= 50) {
-                        // We have reached 50 recursive invocations of getImmediateBaseConstraint and there is a
-                        // very high likelihood we're dealing with an infinite generic type that perpetually generates
-                        // new type identities as we descend into it. We stop the recursion here and mark this type
-                        // and the outer types as having circular constraints.
-                        error(currentNode, ts.Diagnostics.Type_instantiation_is_excessively_deep_and_possibly_infinite);
-                        nonTerminating = true;
-                        return t.immediateBaseConstraint = noConstraintType;
+                    var symbolName = getNameOfSymbolAsWritten(symbol, context);
+                    if (index === 0) {
+                        context.flags ^= 16777216 /* InInitialEntityName */;
                     }
-                    constraintDepth++;
-                    var result = computeBaseConstraint(getSimplifiedType(t, /*writing*/ false));
-                    constraintDepth--;
-                    if (!popTypeResolution()) {
-                        if (t.flags & 262144 /* TypeParameter */) {
-                            var errorNode = getConstraintDeclaration(t);
-                            if (errorNode) {
-                                var diagnostic = error(errorNode, ts.Diagnostics.Type_parameter_0_has_a_circular_constraint, typeToString(t));
-                                if (currentNode && !ts.isNodeDescendantOf(errorNode, currentNode) && !ts.isNodeDescendantOf(currentNode, errorNode)) {
-                                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(currentNode, ts.Diagnostics.Circularity_originates_in_type_at_this_location));
-                                }
-                            }
-                        }
-                        result = circularConstraintType;
+                    var firstChar = symbolName.charCodeAt(0);
+                    if (ts.isSingleOrDoubleQuote(firstChar) && ts.some(symbol.declarations, hasNonGlobalAugmentationExternalModuleSymbol)) {
+                        return ts.factory.createStringLiteral(getSpecifierForModuleSymbol(symbol, context));
                     }
-                    if (nonTerminating) {
-                        result = circularConstraintType;
+                    var canUsePropertyAccess = firstChar === 35 /* hash */ ?
+                        symbolName.length > 1 && ts.isIdentifierStart(symbolName.charCodeAt(1), languageVersion) :
+                        ts.isIdentifierStart(firstChar, languageVersion);
+                    if (index === 0 || canUsePropertyAccess) {
+                        var identifier = ts.setEmitFlags(ts.factory.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
+                        identifier.symbol = symbol;
+                        return index > 0 ? ts.factory.createPropertyAccessExpression(createExpressionFromSymbolChain(chain, index - 1), identifier) : identifier;
+                    }
+                    else {
+                        if (firstChar === 91 /* openBracket */) {
+                            symbolName = symbolName.substring(1, symbolName.length - 1);
+                            firstChar = symbolName.charCodeAt(0);
+                        }
+                        var expression = void 0;
+                        if (ts.isSingleOrDoubleQuote(firstChar)) {
+                            expression = ts.factory.createStringLiteral(symbolName
+                                .substring(1, symbolName.length - 1)
+                                .replace(/\\./g, function (s) { return s.substring(1); }), firstChar === 39 /* singleQuote */);
+                        }
+                        else if (("" + +symbolName) === symbolName) {
+                            expression = ts.factory.createNumericLiteral(+symbolName);
+                        }
+                        if (!expression) {
+                            expression = ts.setEmitFlags(ts.factory.createIdentifier(symbolName, typeParameterNodes), 16777216 /* NoAsciiEscaping */);
+                            expression.symbol = symbol;
+                        }
+                        return ts.factory.createElementAccessExpression(createExpressionFromSymbolChain(chain, index - 1), expression);
                     }
-                    t.immediateBaseConstraint = result || noConstraintType;
                 }
-                return t.immediateBaseConstraint;
             }
-            function getBaseConstraint(t) {
-                var c = getImmediateBaseConstraint(t);
-                return c !== noConstraintType && c !== circularConstraintType ? c : undefined;
+            function isStringNamed(d) {
+                var name = ts.getNameOfDeclaration(d);
+                return !!name && ts.isStringLiteral(name);
             }
-            function computeBaseConstraint(t) {
-                if (t.flags & 262144 /* TypeParameter */) {
-                    var constraint = getConstraintFromTypeParameter(t);
-                    return t.isThisType || !constraint ?
-                        constraint :
-                        getBaseConstraint(constraint);
+            function isSingleQuotedStringNamed(d) {
+                var name = ts.getNameOfDeclaration(d);
+                return !!(name && ts.isStringLiteral(name) && (name.singleQuote || !ts.nodeIsSynthesized(name) && ts.startsWith(ts.getTextOfNode(name, /*includeTrivia*/ false), "'")));
+            }
+            function getPropertyNameNodeForSymbol(symbol, context) {
+                var singleQuote = !!ts.length(symbol.declarations) && ts.every(symbol.declarations, isSingleQuotedStringNamed);
+                var fromNameType = getPropertyNameNodeForSymbolFromNameType(symbol, context, singleQuote);
+                if (fromNameType) {
+                    return fromNameType;
                 }
-                if (t.flags & 3145728 /* UnionOrIntersection */) {
-                    var types = t.types;
-                    var baseTypes = [];
-                    for (var _i = 0, types_7 = types; _i < types_7.length; _i++) {
-                        var type_2 = types_7[_i];
-                        var baseType = getBaseConstraint(type_2);
-                        if (baseType) {
-                            baseTypes.push(baseType);
+                if (ts.isKnownSymbol(symbol)) {
+                    return ts.factory.createComputedPropertyName(ts.factory.createPropertyAccessExpression(ts.factory.createIdentifier("Symbol"), symbol.escapedName.substr(3)));
+                }
+                var rawName = ts.unescapeLeadingUnderscores(symbol.escapedName);
+                var stringNamed = !!ts.length(symbol.declarations) && ts.every(symbol.declarations, isStringNamed);
+                return createPropertyNameNodeForIdentifierOrLiteral(rawName, stringNamed, singleQuote);
+            }
+            // See getNameForSymbolFromNameType for a stringy equivalent
+            function getPropertyNameNodeForSymbolFromNameType(symbol, context, singleQuote) {
+                var nameType = getSymbolLinks(symbol).nameType;
+                if (nameType) {
+                    if (nameType.flags & 384 /* StringOrNumberLiteral */) {
+                        var name = "" + nameType.value;
+                        if (!ts.isIdentifierText(name, compilerOptions.target) && !isNumericLiteralName(name)) {
+                            return ts.factory.createStringLiteral(name, !!singleQuote);
                         }
+                        if (isNumericLiteralName(name) && ts.startsWith(name, "-")) {
+                            return ts.factory.createComputedPropertyName(ts.factory.createNumericLiteral(+name));
+                        }
+                        return createPropertyNameNodeForIdentifierOrLiteral(name);
+                    }
+                    if (nameType.flags & 8192 /* UniqueESSymbol */) {
+                        return ts.factory.createComputedPropertyName(symbolToExpression(nameType.symbol, context, 111551 /* Value */));
                     }
-                    return t.flags & 1048576 /* Union */ && baseTypes.length === types.length ? getUnionType(baseTypes) :
-                        t.flags & 2097152 /* Intersection */ && baseTypes.length ? getIntersectionType(baseTypes) :
-                            undefined;
-                }
-                if (t.flags & 4194304 /* Index */) {
-                    return keyofConstraintType;
                 }
-                if (t.flags & 8388608 /* IndexedAccess */) {
-                    var baseObjectType = getBaseConstraint(t.objectType);
-                    var baseIndexType = getBaseConstraint(t.indexType);
-                    var baseIndexedAccess = baseObjectType && baseIndexType && getIndexedAccessTypeOrUndefined(baseObjectType, baseIndexType);
-                    return baseIndexedAccess && getBaseConstraint(baseIndexedAccess);
+            }
+            function createPropertyNameNodeForIdentifierOrLiteral(name, stringNamed, singleQuote) {
+                return ts.isIdentifierText(name, compilerOptions.target) ? ts.factory.createIdentifier(name) :
+                    !stringNamed && isNumericLiteralName(name) && +name >= 0 ? ts.factory.createNumericLiteral(+name) :
+                        ts.factory.createStringLiteral(name, !!singleQuote);
+            }
+            function cloneNodeBuilderContext(context) {
+                var initial = __assign({}, context);
+                // Make type parameters created within this context not consume the name outside this context
+                // The symbol serializer ends up creating many sibling scopes that all need "separate" contexts when
+                // it comes to naming things - within a normal `typeToTypeNode` call, the node builder only ever descends
+                // through the type tree, so the only cases where we could have used distinct sibling scopes was when there
+                // were multiple generic overloads with similar generated type parameter names
+                // The effect:
+                // When we write out
+                // export const x: <T>(x: T) => T
+                // export const y: <T>(x: T) => T
+                // we write it out like that, rather than as
+                // export const x: <T>(x: T) => T
+                // export const y: <T_1>(x: T_1) => T_1
+                if (initial.typeParameterNames) {
+                    initial.typeParameterNames = new ts.Map(initial.typeParameterNames);
                 }
-                if (t.flags & 16777216 /* Conditional */) {
-                    var constraint = getConstraintFromConditionalType(t);
-                    constraintDepth++; // Penalize repeating conditional types (this captures the recursion within getConstraintFromConditionalType and carries it forward)
-                    var result = constraint && getBaseConstraint(constraint);
-                    constraintDepth--;
-                    return result;
+                if (initial.typeParameterNamesByText) {
+                    initial.typeParameterNamesByText = new ts.Set(initial.typeParameterNamesByText);
                 }
-                if (t.flags & 33554432 /* Substitution */) {
-                    return getBaseConstraint(t.substitute);
+                if (initial.typeParameterSymbolList) {
+                    initial.typeParameterSymbolList = new ts.Set(initial.typeParameterSymbolList);
                 }
-                return t;
+                return initial;
             }
-        }
-        function getApparentTypeOfIntersectionType(type) {
-            return type.resolvedApparentType || (type.resolvedApparentType = getTypeWithThisArgument(type, type, /*apparentType*/ true));
-        }
-        function getResolvedTypeParameterDefault(typeParameter) {
-            if (!typeParameter.default) {
-                if (typeParameter.target) {
-                    var targetDefault = getResolvedTypeParameterDefault(typeParameter.target);
-                    typeParameter.default = targetDefault ? instantiateType(targetDefault, typeParameter.mapper) : noConstraintType;
-                }
-                else {
-                    // To block recursion, set the initial value to the resolvingDefaultType.
-                    typeParameter.default = resolvingDefaultType;
-                    var defaultDeclaration = typeParameter.symbol && ts.forEach(typeParameter.symbol.declarations, function (decl) { return ts.isTypeParameterDeclaration(decl) && decl.default; });
-                    var defaultType = defaultDeclaration ? getTypeFromTypeNode(defaultDeclaration) : noConstraintType;
-                    if (typeParameter.default === resolvingDefaultType) {
-                        // If we have not been called recursively, set the correct default type.
-                        typeParameter.default = defaultType;
+            function getDeclarationWithTypeAnnotation(symbol, enclosingDeclaration) {
+                return symbol.declarations && ts.find(symbol.declarations, function (s) { return !!ts.getEffectiveTypeAnnotationNode(s) && (!enclosingDeclaration || !!ts.findAncestor(s, function (n) { return n === enclosingDeclaration; })); });
+            }
+            function existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(existing, type) {
+                return !(ts.getObjectFlags(type) & 4 /* Reference */) || !ts.isTypeReferenceNode(existing) || ts.length(existing.typeArguments) >= getMinTypeArgumentCount(type.target.typeParameters);
+            }
+            /**
+             * Unlike `typeToTypeNodeHelper`, this handles setting up the `AllowUniqueESSymbolType` flag
+             * so a `unique symbol` is returned when appropriate for the input symbol, rather than `typeof sym`
+             */
+            function serializeTypeForDeclaration(context, type, symbol, enclosingDeclaration, includePrivateSymbol, bundled) {
+                if (type !== errorType && enclosingDeclaration) {
+                    var declWithExistingAnnotation = getDeclarationWithTypeAnnotation(symbol, enclosingDeclaration);
+                    if (declWithExistingAnnotation && !ts.isFunctionLikeDeclaration(declWithExistingAnnotation)) {
+                        // try to reuse the existing annotation
+                        var existing = ts.getEffectiveTypeAnnotationNode(declWithExistingAnnotation);
+                        if (getTypeFromTypeNode(existing) === type && existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(existing, type)) {
+                            var result_5 = serializeExistingTypeNode(context, existing, includePrivateSymbol, bundled);
+                            if (result_5) {
+                                return result_5;
+                            }
+                        }
                     }
                 }
+                var oldFlags = context.flags;
+                if (type.flags & 8192 /* UniqueESSymbol */ &&
+                    type.symbol === symbol && (!context.enclosingDeclaration || ts.some(symbol.declarations, function (d) { return ts.getSourceFileOfNode(d) === ts.getSourceFileOfNode(context.enclosingDeclaration); }))) {
+                    context.flags |= 1048576 /* AllowUniqueESSymbolType */;
+                }
+                var result = typeToTypeNodeHelper(type, context);
+                context.flags = oldFlags;
+                return result;
             }
-            else if (typeParameter.default === resolvingDefaultType) {
-                // If we are called recursively for this type parameter, mark the default as circular.
-                typeParameter.default = circularConstraintType;
-            }
-            return typeParameter.default;
-        }
-        /**
-         * Gets the default type for a type parameter.
-         *
-         * If the type parameter is the result of an instantiation, this gets the instantiated
-         * default type of its target. If the type parameter has no default type or the default is
-         * circular, `undefined` is returned.
-         */
-        function getDefaultFromTypeParameter(typeParameter) {
-            var defaultType = getResolvedTypeParameterDefault(typeParameter);
-            return defaultType !== noConstraintType && defaultType !== circularConstraintType ? defaultType : undefined;
-        }
-        function hasNonCircularTypeParameterDefault(typeParameter) {
-            return getResolvedTypeParameterDefault(typeParameter) !== circularConstraintType;
-        }
-        /**
-         * Indicates whether the declaration of a typeParameter has a default type.
-         */
-        function hasTypeParameterDefault(typeParameter) {
-            return !!(typeParameter.symbol && ts.forEach(typeParameter.symbol.declarations, function (decl) { return ts.isTypeParameterDeclaration(decl) && decl.default; }));
-        }
-        function getApparentTypeOfMappedType(type) {
-            return type.resolvedApparentType || (type.resolvedApparentType = getResolvedApparentTypeOfMappedType(type));
-        }
-        function getResolvedApparentTypeOfMappedType(type) {
-            var typeVariable = getHomomorphicTypeVariable(type);
-            if (typeVariable) {
-                var constraint = getConstraintOfTypeParameter(typeVariable);
-                if (constraint && (isArrayType(constraint) || isTupleType(constraint))) {
-                    return instantiateType(type, prependTypeMapping(typeVariable, constraint, type.mapper));
+            function serializeReturnTypeForSignature(context, type, signature, includePrivateSymbol, bundled) {
+                if (type !== errorType && context.enclosingDeclaration) {
+                    var annotation = signature.declaration && ts.getEffectiveReturnTypeNode(signature.declaration);
+                    if (!!ts.findAncestor(annotation, function (n) { return n === context.enclosingDeclaration; }) && annotation && instantiateType(getTypeFromTypeNode(annotation), signature.mapper) === type && existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(annotation, type)) {
+                        var result = serializeExistingTypeNode(context, annotation, includePrivateSymbol, bundled);
+                        if (result) {
+                            return result;
+                        }
+                    }
                 }
+                return typeToTypeNodeHelper(type, context);
             }
-            return type;
-        }
-        /**
-         * For a type parameter, return the base constraint of the type parameter. For the string, number,
-         * boolean, and symbol primitive types, return the corresponding object types. Otherwise return the
-         * type itself.
-         */
-        function getApparentType(type) {
-            var t = type.flags & 63176704 /* Instantiable */ ? getBaseConstraintOfType(type) || unknownType : type;
-            return ts.getObjectFlags(t) & 32 /* Mapped */ ? getApparentTypeOfMappedType(t) :
-                t.flags & 2097152 /* Intersection */ ? getApparentTypeOfIntersectionType(t) :
-                    t.flags & 132 /* StringLike */ ? globalStringType :
-                        t.flags & 296 /* NumberLike */ ? globalNumberType :
-                            t.flags & 2112 /* BigIntLike */ ? getGlobalBigIntType(/*reportErrors*/ languageVersion >= 7 /* ES2020 */) :
-                                t.flags & 528 /* BooleanLike */ ? globalBooleanType :
-                                    t.flags & 12288 /* ESSymbolLike */ ? getGlobalESSymbolType(/*reportErrors*/ languageVersion >= 2 /* ES2015 */) :
-                                        t.flags & 67108864 /* NonPrimitive */ ? emptyObjectType :
-                                            t.flags & 4194304 /* Index */ ? keyofConstraintType :
-                                                t.flags & 2 /* Unknown */ && !strictNullChecks ? emptyObjectType :
-                                                    t;
-        }
-        function getReducedApparentType(type) {
-            // Since getApparentType may return a non-reduced union or intersection type, we need to perform
-            // type reduction both before and after obtaining the apparent type. For example, given a type parameter
-            // 'T extends A | B', the type 'T & X' becomes 'A & X | B & X' after obtaining the apparent type, and
-            // that type may need futher reduction to remove empty intersections.
-            return getReducedType(getApparentType(getReducedType(type)));
-        }
-        function createUnionOrIntersectionProperty(containingType, name) {
-            var singleProp;
-            var propSet;
-            var indexTypes;
-            var isUnion = containingType.flags & 1048576 /* Union */;
-            // Flags we want to propagate to the result if they exist in all source symbols
-            var optionalFlag = isUnion ? 0 /* None */ : 16777216 /* Optional */;
-            var syntheticFlag = 4 /* SyntheticMethod */;
-            var checkFlags = 0;
-            for (var _i = 0, _a = containingType.types; _i < _a.length; _i++) {
-                var current = _a[_i];
-                var type = getApparentType(current);
-                if (!(type === errorType || type.flags & 131072 /* Never */)) {
-                    var prop = getPropertyOfType(type, name);
-                    var modifiers = prop ? ts.getDeclarationModifierFlagsFromSymbol(prop) : 0;
-                    if (prop) {
-                        if (isUnion) {
-                            optionalFlag |= (prop.flags & 16777216 /* Optional */);
+            function serializeExistingTypeNode(context, existing, includePrivateSymbol, bundled) {
+                if (cancellationToken && cancellationToken.throwIfCancellationRequested) {
+                    cancellationToken.throwIfCancellationRequested();
+                }
+                var hadError = false;
+                var file = ts.getSourceFileOfNode(existing);
+                var transformed = ts.visitNode(existing, visitExistingNodeTreeSymbols);
+                if (hadError) {
+                    return undefined;
+                }
+                return transformed === existing ? ts.setTextRange(ts.factory.cloneNode(existing), existing) : transformed;
+                function visitExistingNodeTreeSymbols(node) {
+                    var _a, _b;
+                    // We don't _actually_ support jsdoc namepath types, emit `any` instead
+                    if (ts.isJSDocAllType(node) || node.kind === 310 /* JSDocNamepathType */) {
+                        return ts.factory.createKeywordTypeNode(128 /* AnyKeyword */);
+                    }
+                    if (ts.isJSDocUnknownType(node)) {
+                        return ts.factory.createKeywordTypeNode(152 /* UnknownKeyword */);
+                    }
+                    if (ts.isJSDocNullableType(node)) {
+                        return ts.factory.createUnionTypeNode([ts.visitNode(node.type, visitExistingNodeTreeSymbols), ts.factory.createLiteralTypeNode(ts.factory.createNull())]);
+                    }
+                    if (ts.isJSDocOptionalType(node)) {
+                        return ts.factory.createUnionTypeNode([ts.visitNode(node.type, visitExistingNodeTreeSymbols), ts.factory.createKeywordTypeNode(150 /* UndefinedKeyword */)]);
+                    }
+                    if (ts.isJSDocNonNullableType(node)) {
+                        return ts.visitNode(node.type, visitExistingNodeTreeSymbols);
+                    }
+                    if (ts.isJSDocVariadicType(node)) {
+                        return ts.factory.createArrayTypeNode(ts.visitNode(node.type, visitExistingNodeTreeSymbols));
+                    }
+                    if (ts.isJSDocTypeLiteral(node)) {
+                        return ts.factory.createTypeLiteralNode(ts.map(node.jsDocPropertyTags, function (t) {
+                            var name = ts.isIdentifier(t.name) ? t.name : t.name.right;
+                            var typeViaParent = getTypeOfPropertyOfType(getTypeFromTypeNode(node), name.escapedText);
+                            var overrideTypeNode = typeViaParent && t.typeExpression && getTypeFromTypeNode(t.typeExpression.type) !== typeViaParent ? typeToTypeNodeHelper(typeViaParent, context) : undefined;
+                            return ts.factory.createPropertySignature(
+                            /*modifiers*/ undefined, name, t.typeExpression && ts.isJSDocOptionalType(t.typeExpression.type) ? ts.factory.createToken(57 /* QuestionToken */) : undefined, overrideTypeNode || (t.typeExpression && ts.visitNode(t.typeExpression.type, visitExistingNodeTreeSymbols)) || ts.factory.createKeywordTypeNode(128 /* AnyKeyword */));
+                        }));
+                    }
+                    if (ts.isTypeReferenceNode(node) && ts.isIdentifier(node.typeName) && node.typeName.escapedText === "") {
+                        return ts.setOriginalNode(ts.factory.createKeywordTypeNode(128 /* AnyKeyword */), node);
+                    }
+                    if ((ts.isExpressionWithTypeArguments(node) || ts.isTypeReferenceNode(node)) && ts.isJSDocIndexSignature(node)) {
+                        return ts.factory.createTypeLiteralNode([ts.factory.createIndexSignature(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, [ts.factory.createParameterDeclaration(
+                                /*decorators*/ undefined, 
+                                /*modifiers*/ undefined, 
+                                /*dotdotdotToken*/ undefined, "x", 
+                                /*questionToken*/ undefined, ts.visitNode(node.typeArguments[0], visitExistingNodeTreeSymbols))], ts.visitNode(node.typeArguments[1], visitExistingNodeTreeSymbols))]);
+                    }
+                    if (ts.isJSDocFunctionType(node)) {
+                        if (ts.isJSDocConstructSignature(node)) {
+                            var newTypeNode_1;
+                            return ts.factory.createConstructorTypeNode(ts.visitNodes(node.typeParameters, visitExistingNodeTreeSymbols), ts.mapDefined(node.parameters, function (p, i) { return p.name && ts.isIdentifier(p.name) && p.name.escapedText === "new" ? (newTypeNode_1 = p.type, undefined) : ts.factory.createParameterDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, getEffectiveDotDotDotForParameter(p), getNameForJSDocFunctionParameter(p, i), p.questionToken, ts.visitNode(p.type, visitExistingNodeTreeSymbols), 
+                            /*initializer*/ undefined); }), ts.visitNode(newTypeNode_1 || node.type, visitExistingNodeTreeSymbols) || ts.factory.createKeywordTypeNode(128 /* AnyKeyword */));
                         }
                         else {
-                            optionalFlag &= prop.flags;
+                            return ts.factory.createFunctionTypeNode(ts.visitNodes(node.typeParameters, visitExistingNodeTreeSymbols), ts.map(node.parameters, function (p, i) { return ts.factory.createParameterDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, getEffectiveDotDotDotForParameter(p), getNameForJSDocFunctionParameter(p, i), p.questionToken, ts.visitNode(p.type, visitExistingNodeTreeSymbols), 
+                            /*initializer*/ undefined); }), ts.visitNode(node.type, visitExistingNodeTreeSymbols) || ts.factory.createKeywordTypeNode(128 /* AnyKeyword */));
                         }
-                        if (!singleProp) {
-                            singleProp = prop;
+                    }
+                    if (ts.isTypeReferenceNode(node) && ts.isInJSDoc(node) && (!existingTypeNodeIsNotReferenceOrIsReferenceWithCompatibleTypeArgumentCount(node, getTypeFromTypeNode(node)) || getIntendedTypeFromJSDocTypeReference(node) || unknownSymbol === resolveTypeReferenceName(getTypeReferenceName(node), 788968 /* Type */, /*ignoreErrors*/ true))) {
+                        return ts.setOriginalNode(typeToTypeNodeHelper(getTypeFromTypeNode(node), context), node);
+                    }
+                    if (ts.isLiteralImportTypeNode(node)) {
+                        return ts.factory.updateImportTypeNode(node, ts.factory.updateLiteralTypeNode(node.argument, rewriteModuleSpecifier(node, node.argument.literal)), node.qualifier, ts.visitNodes(node.typeArguments, visitExistingNodeTreeSymbols, ts.isTypeNode), node.isTypeOf);
+                    }
+                    if (ts.isEntityName(node) || ts.isEntityNameExpression(node)) {
+                        var leftmost = ts.getFirstIdentifier(node);
+                        if (ts.isInJSFile(node) && (ts.isExportsIdentifier(leftmost) || ts.isModuleExportsAccessExpression(leftmost.parent) || (ts.isQualifiedName(leftmost.parent) && ts.isModuleIdentifier(leftmost.parent.left) && ts.isExportsIdentifier(leftmost.parent.right)))) {
+                            hadError = true;
+                            return node;
                         }
-                        else if (prop !== singleProp) {
-                            if (!propSet) {
-                                propSet = ts.createMap();
-                                propSet.set("" + getSymbolId(singleProp), singleProp);
+                        var sym = resolveEntityName(leftmost, 67108863 /* All */, /*ignoreErrors*/ true, /*dontResolveALias*/ true);
+                        if (sym) {
+                            if (isSymbolAccessible(sym, context.enclosingDeclaration, 67108863 /* All */, /*shouldComputeAliasesToMakeVisible*/ false).accessibility !== 0 /* Accessible */) {
+                                hadError = true;
                             }
-                            var id = "" + getSymbolId(prop);
-                            if (!propSet.has(id)) {
-                                propSet.set(id, prop);
+                            else {
+                                (_b = (_a = context.tracker) === null || _a === void 0 ? void 0 : _a.trackSymbol) === null || _b === void 0 ? void 0 : _b.call(_a, sym, context.enclosingDeclaration, 67108863 /* All */);
+                                includePrivateSymbol === null || includePrivateSymbol === void 0 ? void 0 : includePrivateSymbol(sym);
+                            }
+                            if (ts.isIdentifier(node)) {
+                                var name = sym.flags & 262144 /* TypeParameter */ ? typeParameterToName(getDeclaredTypeOfSymbol(sym), context) : ts.factory.cloneNode(node);
+                                name.symbol = sym; // for quickinfo, which uses identifier symbol information
+                                return ts.setEmitFlags(ts.setOriginalNode(name, node), 16777216 /* NoAsciiEscaping */);
                             }
-                        }
-                        checkFlags |= (isReadonlySymbol(prop) ? 8 /* Readonly */ : 0) |
-                            (!(modifiers & 24 /* NonPublicAccessibilityModifier */) ? 256 /* ContainsPublic */ : 0) |
-                            (modifiers & 16 /* Protected */ ? 512 /* ContainsProtected */ : 0) |
-                            (modifiers & 8 /* Private */ ? 1024 /* ContainsPrivate */ : 0) |
-                            (modifiers & 32 /* Static */ ? 2048 /* ContainsStatic */ : 0);
-                        if (!isPrototypeProperty(prop)) {
-                            syntheticFlag = 2 /* SyntheticProperty */;
                         }
                     }
-                    else if (isUnion) {
-                        var indexInfo = !isLateBoundName(name) && (isNumericLiteralName(name) && getIndexInfoOfType(type, 1 /* Number */) || getIndexInfoOfType(type, 0 /* String */));
-                        if (indexInfo) {
-                            checkFlags |= 32 /* WritePartial */ | (indexInfo.isReadonly ? 8 /* Readonly */ : 0);
-                            indexTypes = ts.append(indexTypes, isTupleType(type) ? getRestTypeOfTupleType(type) || undefinedType : indexInfo.type);
-                        }
-                        else if (isObjectLiteralType(type)) {
-                            checkFlags |= 32 /* WritePartial */;
-                            indexTypes = ts.append(indexTypes, undefinedType);
+                    if (file && ts.isTupleTypeNode(node) && (ts.getLineAndCharacterOfPosition(file, node.pos).line === ts.getLineAndCharacterOfPosition(file, node.end).line)) {
+                        ts.setEmitFlags(node, 1 /* SingleLine */);
+                    }
+                    return ts.visitEachChild(node, visitExistingNodeTreeSymbols, ts.nullTransformationContext);
+                    function getEffectiveDotDotDotForParameter(p) {
+                        return p.dotDotDotToken || (p.type && ts.isJSDocVariadicType(p.type) ? ts.factory.createToken(25 /* DotDotDotToken */) : undefined);
+                    }
+                    /** Note that `new:T` parameters are not handled, but should be before calling this function. */
+                    function getNameForJSDocFunctionParameter(p, index) {
+                        return p.name && ts.isIdentifier(p.name) && p.name.escapedText === "this" ? "this"
+                            : getEffectiveDotDotDotForParameter(p) ? "args"
+                                : "arg" + index;
+                    }
+                    function rewriteModuleSpecifier(parent, lit) {
+                        if (bundled) {
+                            if (context.tracker && context.tracker.moduleResolverHost) {
+                                var targetFile = getExternalModuleFileFromDeclaration(parent);
+                                if (targetFile) {
+                                    var getCanonicalFileName = ts.createGetCanonicalFileName(!!host.useCaseSensitiveFileNames);
+                                    var resolverHost = {
+                                        getCanonicalFileName: getCanonicalFileName,
+                                        getCurrentDirectory: function () { return context.tracker.moduleResolverHost.getCurrentDirectory(); },
+                                        getCommonSourceDirectory: function () { return context.tracker.moduleResolverHost.getCommonSourceDirectory(); }
+                                    };
+                                    var newName = ts.getResolvedExternalModuleName(resolverHost, targetFile);
+                                    return ts.factory.createStringLiteral(newName);
+                                }
+                            }
                         }
                         else {
-                            checkFlags |= 16 /* ReadPartial */;
+                            if (context.tracker && context.tracker.trackExternalModuleSymbolOfImportTypeNode) {
+                                var moduleSym = resolveExternalModuleNameWorker(lit, lit, /*moduleNotFoundError*/ undefined);
+                                if (moduleSym) {
+                                    context.tracker.trackExternalModuleSymbolOfImportTypeNode(moduleSym);
+                                }
+                            }
                         }
+                        return lit;
                     }
                 }
             }
-            if (!singleProp || isUnion && (propSet || checkFlags & 48 /* Partial */) && checkFlags & (1024 /* ContainsPrivate */ | 512 /* ContainsProtected */)) {
-                // No property was found, or, in a union, a property has a private or protected declaration in one
-                // constituent, but is missing or has a different declaration in another constituent.
-                return undefined;
-            }
-            if (!propSet && !(checkFlags & 16 /* ReadPartial */) && !indexTypes) {
-                return singleProp;
-            }
-            var props = propSet ? ts.arrayFrom(propSet.values()) : [singleProp];
-            var declarations;
-            var firstType;
-            var nameType;
-            var propTypes = [];
-            var firstValueDeclaration;
-            var hasNonUniformValueDeclaration = false;
-            for (var _b = 0, props_1 = props; _b < props_1.length; _b++) {
-                var prop = props_1[_b];
-                if (!firstValueDeclaration) {
-                    firstValueDeclaration = prop.valueDeclaration;
-                }
-                else if (prop.valueDeclaration && prop.valueDeclaration !== firstValueDeclaration) {
-                    hasNonUniformValueDeclaration = true;
-                }
-                declarations = ts.addRange(declarations, prop.declarations);
-                var type = getTypeOfSymbol(prop);
-                if (!firstType) {
-                    firstType = type;
-                    nameType = getSymbolLinks(prop).nameType;
-                }
-                else if (type !== firstType) {
-                    checkFlags |= 64 /* HasNonUniformType */;
-                }
-                if (isLiteralType(type)) {
-                    checkFlags |= 128 /* HasLiteralType */;
-                }
-                if (type.flags & 131072 /* Never */) {
-                    checkFlags |= 131072 /* HasNeverType */;
-                }
-                propTypes.push(type);
-            }
-            ts.addRange(propTypes, indexTypes);
-            var result = createSymbol(4 /* Property */ | optionalFlag, name, syntheticFlag | checkFlags);
-            result.containingType = containingType;
-            if (!hasNonUniformValueDeclaration && firstValueDeclaration) {
-                result.valueDeclaration = firstValueDeclaration;
-                // Inherit information about parent type.
-                if (firstValueDeclaration.symbol.parent) {
-                    result.parent = firstValueDeclaration.symbol.parent;
-                }
-            }
-            result.declarations = declarations;
-            result.nameType = nameType;
-            if (propTypes.length > 2) {
-                // When `propTypes` has the potential to explode in size when normalized, defer normalization until absolutely needed
-                result.checkFlags |= 65536 /* DeferredType */;
-                result.deferralParent = containingType;
-                result.deferralConstituents = propTypes;
-            }
-            else {
-                result.type = isUnion ? getUnionType(propTypes) : getIntersectionType(propTypes);
-            }
-            return result;
-        }
-        // Return the symbol for a given property in a union or intersection type, or undefined if the property
-        // does not exist in any constituent type. Note that the returned property may only be present in some
-        // constituents, in which case the isPartial flag is set when the containing type is union type. We need
-        // these partial properties when identifying discriminant properties, but otherwise they are filtered out
-        // and do not appear to be present in the union type.
-        function getUnionOrIntersectionProperty(type, name) {
-            var properties = type.propertyCache || (type.propertyCache = ts.createSymbolTable());
-            var property = properties.get(name);
-            if (!property) {
-                property = createUnionOrIntersectionProperty(type, name);
-                if (property) {
-                    properties.set(name, property);
+            function symbolTableToDeclarationStatements(symbolTable, context, bundled) {
+                var serializePropertySymbolForClass = makeSerializePropertySymbol(ts.factory.createPropertyDeclaration, 165 /* MethodDeclaration */, /*useAcessors*/ true);
+                var serializePropertySymbolForInterfaceWorker = makeSerializePropertySymbol(function (_decorators, mods, name, question, type) { return ts.factory.createPropertySignature(mods, name, question, type); }, 164 /* MethodSignature */, /*useAcessors*/ false);
+                // TODO: Use `setOriginalNode` on original declaration names where possible so these declarations see some kind of
+                // declaration mapping
+                // We save the enclosing declaration off here so it's not adjusted by well-meaning declaration
+                // emit codepaths which want to apply more specific contexts (so we can still refer to the root real declaration
+                // we're trying to emit from later on)
+                var enclosingDeclaration = context.enclosingDeclaration;
+                var results = [];
+                var visitedSymbols = new ts.Set();
+                var deferredPrivatesStack = [];
+                var oldcontext = context;
+                context = __assign(__assign({}, oldcontext), { usedSymbolNames: new ts.Set(oldcontext.usedSymbolNames), remappedSymbolNames: new ts.Map(), tracker: __assign(__assign({}, oldcontext.tracker), { trackSymbol: function (sym, decl, meaning) {
+                            var accessibleResult = isSymbolAccessible(sym, decl, meaning, /*computeALiases*/ false);
+                            if (accessibleResult.accessibility === 0 /* Accessible */) {
+                                // Lookup the root symbol of the chain of refs we'll use to access it and serialize it
+                                var chain = lookupSymbolChainWorker(sym, context, meaning);
+                                if (!(sym.flags & 4 /* Property */)) {
+                                    includePrivateSymbol(chain[0]);
+                                }
+                            }
+                            else if (oldcontext.tracker && oldcontext.tracker.trackSymbol) {
+                                oldcontext.tracker.trackSymbol(sym, decl, meaning);
+                            }
+                        } }) });
+                ts.forEachEntry(symbolTable, function (symbol, name) {
+                    var baseName = ts.unescapeLeadingUnderscores(name);
+                    void getInternalSymbolName(symbol, baseName); // Called to cache values into `usedSymbolNames` and `remappedSymbolNames`
+                });
+                var addingDeclare = !bundled;
+                var exportEquals = symbolTable.get("export=" /* ExportEquals */);
+                if (exportEquals && symbolTable.size > 1 && exportEquals.flags & 2097152 /* Alias */) {
+                    symbolTable = ts.createSymbolTable();
+                    // Remove extraneous elements from root symbol table (they'll be mixed back in when the target of the `export=` is looked up)
+                    symbolTable.set("export=" /* ExportEquals */, exportEquals);
                 }
-            }
-            return property;
-        }
-        function getPropertyOfUnionOrIntersectionType(type, name) {
-            var property = getUnionOrIntersectionProperty(type, name);
-            // We need to filter out partial properties in union types
-            return property && !(ts.getCheckFlags(property) & 16 /* ReadPartial */) ? property : undefined;
-        }
-        /**
-         * Return the reduced form of the given type. For a union type, it is a union of the normalized constituent types.
-         * For an intersection of types containing one or more mututally exclusive discriminant properties, it is 'never'.
-         * For all other types, it is simply the type itself. Discriminant properties are considered mutually exclusive when
-         * no constituent property has type 'never', but the intersection of the constituent property types is 'never'.
-         */
-        function getReducedType(type) {
-            if (type.flags & 1048576 /* Union */ && type.objectFlags & 268435456 /* ContainsIntersections */) {
-                return type.resolvedReducedType || (type.resolvedReducedType = getReducedUnionType(type));
-            }
-            else if (type.flags & 2097152 /* Intersection */) {
-                if (!(type.objectFlags & 268435456 /* IsNeverIntersectionComputed */)) {
-                    type.objectFlags |= 268435456 /* IsNeverIntersectionComputed */ |
-                        (ts.some(getPropertiesOfUnionOrIntersectionType(type), isNeverReducedProperty) ? 536870912 /* IsNeverIntersection */ : 0);
+                visitSymbolTable(symbolTable);
+                return mergeRedundantStatements(results);
+                function isIdentifierAndNotUndefined(node) {
+                    return !!node && node.kind === 78 /* Identifier */;
                 }
-                return type.objectFlags & 536870912 /* IsNeverIntersection */ ? neverType : type;
-            }
-            return type;
-        }
-        function getReducedUnionType(unionType) {
-            var reducedTypes = ts.sameMap(unionType.types, getReducedType);
-            if (reducedTypes === unionType.types) {
-                return unionType;
-            }
-            var reduced = getUnionType(reducedTypes);
-            if (reduced.flags & 1048576 /* Union */) {
-                reduced.resolvedReducedType = reduced;
-            }
-            return reduced;
-        }
-        function isNeverReducedProperty(prop) {
-            return isDiscriminantWithNeverType(prop) || isConflictingPrivateProperty(prop);
-        }
-        function isDiscriminantWithNeverType(prop) {
-            // Return true for a synthetic non-optional property with non-uniform types, where at least one is
-            // a literal type and none is never, that reduces to never.
-            return !(prop.flags & 16777216 /* Optional */) &&
-                (ts.getCheckFlags(prop) & (192 /* Discriminant */ | 131072 /* HasNeverType */)) === 192 /* Discriminant */ &&
-                !!(getTypeOfSymbol(prop).flags & 131072 /* Never */);
-        }
-        function isConflictingPrivateProperty(prop) {
-            // Return true for a synthetic property with multiple declarations, at least one of which is private.
-            return !prop.valueDeclaration && !!(ts.getCheckFlags(prop) & 1024 /* ContainsPrivate */);
-        }
-        function elaborateNeverIntersection(errorInfo, type) {
-            if (ts.getObjectFlags(type) & 536870912 /* IsNeverIntersection */) {
-                var neverProp = ts.find(getPropertiesOfUnionOrIntersectionType(type), isDiscriminantWithNeverType);
-                if (neverProp) {
-                    return ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.The_intersection_0_was_reduced_to_never_because_property_1_has_conflicting_types_in_some_constituents, typeToString(type, /*enclosingDeclaration*/ undefined, 536870912 /* NoTypeReduction */), symbolToString(neverProp));
+                function getNamesOfDeclaration(statement) {
+                    if (ts.isVariableStatement(statement)) {
+                        return ts.filter(ts.map(statement.declarationList.declarations, ts.getNameOfDeclaration), isIdentifierAndNotUndefined);
+                    }
+                    return ts.filter([ts.getNameOfDeclaration(statement)], isIdentifierAndNotUndefined);
                 }
-                var privateProp = ts.find(getPropertiesOfUnionOrIntersectionType(type), isConflictingPrivateProperty);
-                if (privateProp) {
-                    return ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.The_intersection_0_was_reduced_to_never_because_property_1_exists_in_multiple_constituents_and_is_private_in_some, typeToString(type, /*enclosingDeclaration*/ undefined, 536870912 /* NoTypeReduction */), symbolToString(privateProp));
+                function flattenExportAssignedNamespace(statements) {
+                    var exportAssignment = ts.find(statements, ts.isExportAssignment);
+                    var nsIndex = ts.findIndex(statements, ts.isModuleDeclaration);
+                    var ns = nsIndex !== -1 ? statements[nsIndex] : undefined;
+                    if (ns && exportAssignment && exportAssignment.isExportEquals &&
+                        ts.isIdentifier(exportAssignment.expression) && ts.isIdentifier(ns.name) && ts.idText(ns.name) === ts.idText(exportAssignment.expression) &&
+                        ns.body && ts.isModuleBlock(ns.body)) {
+                        // Pass 0: Correct situations where a module has both an `export = ns` and multiple top-level exports by stripping the export modifiers from
+                        //  the top-level exports and exporting them in the targeted ns, as can occur when a js file has both typedefs and `module.export` assignments
+                        var excessExports = ts.filter(statements, function (s) { return !!(ts.getEffectiveModifierFlags(s) & 1 /* Export */); });
+                        var name_2 = ns.name;
+                        var body = ns.body;
+                        if (ts.length(excessExports)) {
+                            ns = ts.factory.updateModuleDeclaration(ns, ns.decorators, ns.modifiers, ns.name, body = ts.factory.updateModuleBlock(body, ts.factory.createNodeArray(__spreadArrays(ns.body.statements, [ts.factory.createExportDeclaration(
+                                /*decorators*/ undefined, 
+                                /*modifiers*/ undefined, 
+                                /*isTypeOnly*/ false, ts.factory.createNamedExports(ts.map(ts.flatMap(excessExports, function (e) { return getNamesOfDeclaration(e); }), function (id) { return ts.factory.createExportSpecifier(/*alias*/ undefined, id); })), 
+                                /*moduleSpecifier*/ undefined)]))));
+                            statements = __spreadArrays(statements.slice(0, nsIndex), [ns], statements.slice(nsIndex + 1));
+                        }
+                        // Pass 1: Flatten `export namespace _exports {} export = _exports;` so long as the `export=` only points at a single namespace declaration
+                        if (!ts.find(statements, function (s) { return s !== ns && ts.nodeHasName(s, name_2); })) {
+                            results = [];
+                            // If the namespace contains no export assignments or declarations, and no declarations flagged with `export`, then _everything_ is exported -
+                            // to respect this as the top level, we need to add an `export` modifier to everything
+                            var mixinExportFlag_1 = !ts.some(body.statements, function (s) { return ts.hasSyntacticModifier(s, 1 /* Export */) || ts.isExportAssignment(s) || ts.isExportDeclaration(s); });
+                            ts.forEach(body.statements, function (s) {
+                                addResult(s, mixinExportFlag_1 ? 1 /* Export */ : 0 /* None */); // Recalculates the ambient (and export, if applicable from above) flag
+                            });
+                            statements = __spreadArrays(ts.filter(statements, function (s) { return s !== ns && s !== exportAssignment; }), results);
+                        }
+                    }
+                    return statements;
                 }
-            }
-            return errorInfo;
-        }
-        /**
-         * Return the symbol for the property with the given name in the given type. Creates synthetic union properties when
-         * necessary, maps primitive types and type parameters are to their apparent types, and augments with properties from
-         * Object and Function as appropriate.
-         *
-         * @param type a type to look up property from
-         * @param name a name of property to look up in a given type
-         */
-        function getPropertyOfType(type, name) {
-            type = getReducedApparentType(type);
-            if (type.flags & 524288 /* Object */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                var symbol = resolved.members.get(name);
-                if (symbol && symbolIsValue(symbol)) {
-                    return symbol;
+                function mergeExportDeclarations(statements) {
+                    // Pass 2: Combine all `export {}` declarations
+                    var exports = ts.filter(statements, function (d) { return ts.isExportDeclaration(d) && !d.moduleSpecifier && !!d.exportClause && ts.isNamedExports(d.exportClause); });
+                    if (ts.length(exports) > 1) {
+                        var nonExports = ts.filter(statements, function (d) { return !ts.isExportDeclaration(d) || !!d.moduleSpecifier || !d.exportClause; });
+                        statements = __spreadArrays(nonExports, [ts.factory.createExportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, 
+                            /*isTypeOnly*/ false, ts.factory.createNamedExports(ts.flatMap(exports, function (e) { return ts.cast(e.exportClause, ts.isNamedExports).elements; })), 
+                            /*moduleSpecifier*/ undefined)]);
+                    }
+                    // Pass 2b: Also combine all `export {} from "..."` declarations as needed
+                    var reexports = ts.filter(statements, function (d) { return ts.isExportDeclaration(d) && !!d.moduleSpecifier && !!d.exportClause && ts.isNamedExports(d.exportClause); });
+                    if (ts.length(reexports) > 1) {
+                        var groups = ts.group(reexports, function (decl) { return ts.isStringLiteral(decl.moduleSpecifier) ? ">" + decl.moduleSpecifier.text : ">"; });
+                        if (groups.length !== reexports.length) {
+                            var _loop_9 = function (group_1) {
+                                if (group_1.length > 1) {
+                                    // remove group members from statements and then merge group members and add back to statements
+                                    statements = __spreadArrays(ts.filter(statements, function (s) { return group_1.indexOf(s) === -1; }), [
+                                        ts.factory.createExportDeclaration(
+                                        /*decorators*/ undefined, 
+                                        /*modifiers*/ undefined, 
+                                        /*isTypeOnly*/ false, ts.factory.createNamedExports(ts.flatMap(group_1, function (e) { return ts.cast(e.exportClause, ts.isNamedExports).elements; })), group_1[0].moduleSpecifier)
+                                    ]);
+                                }
+                            };
+                            for (var _i = 0, groups_1 = groups; _i < groups_1.length; _i++) {
+                                var group_1 = groups_1[_i];
+                                _loop_9(group_1);
+                            }
+                        }
+                    }
+                    return statements;
                 }
-                var functionType = resolved === anyFunctionType ? globalFunctionType :
-                    resolved.callSignatures.length ? globalCallableFunctionType :
-                        resolved.constructSignatures.length ? globalNewableFunctionType :
-                            undefined;
-                if (functionType) {
-                    var symbol_1 = getPropertyOfObjectType(functionType, name);
-                    if (symbol_1) {
-                        return symbol_1;
+                function inlineExportModifiers(statements) {
+                    // Pass 3: Move all `export {}`'s to `export` modifiers where possible
+                    var index = ts.findIndex(statements, function (d) { return ts.isExportDeclaration(d) && !d.moduleSpecifier && !!d.exportClause && ts.isNamedExports(d.exportClause); });
+                    if (index >= 0) {
+                        var exportDecl = statements[index];
+                        var replacements = ts.mapDefined(exportDecl.exportClause.elements, function (e) {
+                            if (!e.propertyName) {
+                                // export {name} - look thru `statements` for `name`, and if all results can take an `export` modifier, do so and filter it
+                                var indices = ts.indicesOf(statements);
+                                var associatedIndices = ts.filter(indices, function (i) { return ts.nodeHasName(statements[i], e.name); });
+                                if (ts.length(associatedIndices) && ts.every(associatedIndices, function (i) { return canHaveExportModifier(statements[i]); })) {
+                                    for (var _i = 0, associatedIndices_1 = associatedIndices; _i < associatedIndices_1.length; _i++) {
+                                        var index_1 = associatedIndices_1[_i];
+                                        statements[index_1] = addExportModifier(statements[index_1]);
+                                    }
+                                    return undefined;
+                                }
+                            }
+                            return e;
+                        });
+                        if (!ts.length(replacements)) {
+                            // all clauses removed, remove the export declaration
+                            ts.orderedRemoveItemAt(statements, index);
+                        }
+                        else {
+                            // some items filtered, others not - update the export declaration
+                            statements[index] = ts.factory.updateExportDeclaration(exportDecl, exportDecl.decorators, exportDecl.modifiers, exportDecl.isTypeOnly, ts.factory.updateNamedExports(exportDecl.exportClause, replacements), exportDecl.moduleSpecifier);
+                        }
                     }
+                    return statements;
                 }
-                return getPropertyOfObjectType(globalObjectType, name);
-            }
-            if (type.flags & 3145728 /* UnionOrIntersection */) {
-                return getPropertyOfUnionOrIntersectionType(type, name);
-            }
-            return undefined;
-        }
-        function getSignaturesOfStructuredType(type, kind) {
-            if (type.flags & 3670016 /* StructuredType */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                return kind === 0 /* Call */ ? resolved.callSignatures : resolved.constructSignatures;
-            }
-            return ts.emptyArray;
-        }
-        /**
-         * Return the signatures of the given kind in the given type. Creates synthetic union signatures when necessary and
-         * maps primitive types and type parameters are to their apparent types.
-         */
-        function getSignaturesOfType(type, kind) {
-            return getSignaturesOfStructuredType(getReducedApparentType(type), kind);
-        }
-        function getIndexInfoOfStructuredType(type, kind) {
-            if (type.flags & 3670016 /* StructuredType */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                return kind === 0 /* String */ ? resolved.stringIndexInfo : resolved.numberIndexInfo;
-            }
-        }
-        function getIndexTypeOfStructuredType(type, kind) {
-            var info = getIndexInfoOfStructuredType(type, kind);
-            return info && info.type;
-        }
-        // Return the indexing info of the given kind in the given type. Creates synthetic union index types when necessary and
-        // maps primitive types and type parameters are to their apparent types.
-        function getIndexInfoOfType(type, kind) {
-            return getIndexInfoOfStructuredType(getReducedApparentType(type), kind);
-        }
-        // Return the index type of the given kind in the given type. Creates synthetic union index types when necessary and
-        // maps primitive types and type parameters are to their apparent types.
-        function getIndexTypeOfType(type, kind) {
-            return getIndexTypeOfStructuredType(getReducedApparentType(type), kind);
-        }
-        function getImplicitIndexTypeOfType(type, kind) {
-            if (isObjectTypeWithInferableIndex(type)) {
-                var propTypes = [];
-                for (var _i = 0, _a = getPropertiesOfType(type); _i < _a.length; _i++) {
-                    var prop = _a[_i];
-                    if (kind === 0 /* String */ || isNumericLiteralName(prop.escapedName)) {
-                        propTypes.push(getTypeOfSymbol(prop));
+                function mergeRedundantStatements(statements) {
+                    statements = flattenExportAssignedNamespace(statements);
+                    statements = mergeExportDeclarations(statements);
+                    statements = inlineExportModifiers(statements);
+                    // Not a cleanup, but as a final step: If there is a mix of `export` and non-`export` declarations, but no `export =` or `export {}` add a `export {};` so
+                    // declaration privacy is respected.
+                    if (enclosingDeclaration &&
+                        ((ts.isSourceFile(enclosingDeclaration) && ts.isExternalOrCommonJsModule(enclosingDeclaration)) || ts.isModuleDeclaration(enclosingDeclaration)) &&
+                        (!ts.some(statements, ts.isExternalModuleIndicator) || (!ts.hasScopeMarker(statements) && ts.some(statements, ts.needsScopeMarker)))) {
+                        statements.push(ts.createEmptyExports(ts.factory));
                     }
+                    return statements;
                 }
-                if (kind === 0 /* String */) {
-                    ts.append(propTypes, getIndexTypeOfType(type, 1 /* Number */));
+                function canHaveExportModifier(node) {
+                    return ts.isEnumDeclaration(node) ||
+                        ts.isVariableStatement(node) ||
+                        ts.isFunctionDeclaration(node) ||
+                        ts.isClassDeclaration(node) ||
+                        (ts.isModuleDeclaration(node) && !ts.isExternalModuleAugmentation(node) && !ts.isGlobalScopeAugmentation(node)) ||
+                        ts.isInterfaceDeclaration(node) ||
+                        isTypeDeclaration(node);
                 }
-                if (propTypes.length) {
-                    return getUnionType(propTypes);
+                function addExportModifier(node) {
+                    var flags = (ts.getEffectiveModifierFlags(node) | 1 /* Export */) & ~2 /* Ambient */;
+                    return ts.factory.updateModifiers(node, flags);
                 }
-            }
-            return undefined;
-        }
-        // Return list of type parameters with duplicates removed (duplicate identifier errors are generated in the actual
-        // type checking functions).
-        function getTypeParametersFromDeclaration(declaration) {
-            var result;
-            for (var _i = 0, _a = ts.getEffectiveTypeParameterDeclarations(declaration); _i < _a.length; _i++) {
-                var node = _a[_i];
-                result = ts.appendIfUnique(result, getDeclaredTypeOfTypeParameter(node.symbol));
-            }
-            return result;
-        }
-        function symbolsToArray(symbols) {
-            var result = [];
-            symbols.forEach(function (symbol, id) {
-                if (!isReservedMemberName(id)) {
-                    result.push(symbol);
+                function removeExportModifier(node) {
+                    var flags = ts.getEffectiveModifierFlags(node) & ~1 /* Export */;
+                    return ts.factory.updateModifiers(node, flags);
                 }
-            });
-            return result;
-        }
-        function isJSDocOptionalParameter(node) {
-            return ts.isInJSFile(node) && (
-            // node.type should only be a JSDocOptionalType when node is a parameter of a JSDocFunctionType
-            node.type && node.type.kind === 299 /* JSDocOptionalType */
-                || ts.getJSDocParameterTags(node).some(function (_a) {
-                    var isBracketed = _a.isBracketed, typeExpression = _a.typeExpression;
-                    return isBracketed || !!typeExpression && typeExpression.type.kind === 299 /* JSDocOptionalType */;
-                }));
-        }
-        function tryFindAmbientModule(moduleName, withAugmentations) {
-            if (ts.isExternalModuleNameRelative(moduleName)) {
-                return undefined;
-            }
-            var symbol = getSymbol(globals, '"' + moduleName + '"', 512 /* ValueModule */);
-            // merged symbol is module declaration symbol combined with all augmentations
-            return symbol && withAugmentations ? getMergedSymbol(symbol) : symbol;
-        }
-        function isOptionalParameter(node) {
-            if (ts.hasQuestionToken(node) || isOptionalJSDocParameterTag(node) || isJSDocOptionalParameter(node)) {
-                return true;
-            }
-            if (node.initializer) {
-                var signature = getSignatureFromDeclaration(node.parent);
-                var parameterIndex = node.parent.parameters.indexOf(node);
-                ts.Debug.assert(parameterIndex >= 0);
-                return parameterIndex >= getMinArgumentCount(signature, /*strongArityForUntypedJS*/ true);
-            }
-            var iife = ts.getImmediatelyInvokedFunctionExpression(node.parent);
-            if (iife) {
-                return !node.type &&
-                    !node.dotDotDotToken &&
-                    node.parent.parameters.indexOf(node) >= iife.arguments.length;
-            }
-            return false;
-        }
-        function isOptionalJSDocParameterTag(node) {
-            if (!ts.isJSDocParameterTag(node)) {
-                return false;
-            }
-            var isBracketed = node.isBracketed, typeExpression = node.typeExpression;
-            return isBracketed || !!typeExpression && typeExpression.type.kind === 299 /* JSDocOptionalType */;
-        }
-        function createTypePredicate(kind, parameterName, parameterIndex, type) {
-            return { kind: kind, parameterName: parameterName, parameterIndex: parameterIndex, type: type };
-        }
-        /**
-         * Gets the minimum number of type arguments needed to satisfy all non-optional type
-         * parameters.
-         */
-        function getMinTypeArgumentCount(typeParameters) {
-            var minTypeArgumentCount = 0;
-            if (typeParameters) {
-                for (var i = 0; i < typeParameters.length; i++) {
-                    if (!hasTypeParameterDefault(typeParameters[i])) {
-                        minTypeArgumentCount = i + 1;
+                function visitSymbolTable(symbolTable, suppressNewPrivateContext, propertyAsAlias) {
+                    if (!suppressNewPrivateContext) {
+                        deferredPrivatesStack.push(new ts.Map());
                     }
-                }
-            }
-            return minTypeArgumentCount;
-        }
-        function fillMissingTypeArguments(typeArguments, typeParameters, minTypeArgumentCount, isJavaScriptImplicitAny) {
-            var numTypeParameters = ts.length(typeParameters);
-            if (!numTypeParameters) {
-                return [];
-            }
-            var numTypeArguments = ts.length(typeArguments);
-            if (isJavaScriptImplicitAny || (numTypeArguments >= minTypeArgumentCount && numTypeArguments <= numTypeParameters)) {
-                var result = typeArguments ? typeArguments.slice() : [];
-                // Map invalid forward references in default types to the error type
-                for (var i = numTypeArguments; i < numTypeParameters; i++) {
-                    result[i] = errorType;
-                }
-                var baseDefaultType = getDefaultTypeArgumentType(isJavaScriptImplicitAny);
-                for (var i = numTypeArguments; i < numTypeParameters; i++) {
-                    var defaultType = getDefaultFromTypeParameter(typeParameters[i]);
-                    if (isJavaScriptImplicitAny && defaultType && (isTypeIdenticalTo(defaultType, unknownType) || isTypeIdenticalTo(defaultType, emptyObjectType))) {
-                        defaultType = anyType;
+                    symbolTable.forEach(function (symbol) {
+                        serializeSymbol(symbol, /*isPrivate*/ false, !!propertyAsAlias);
+                    });
+                    if (!suppressNewPrivateContext) {
+                        // deferredPrivates will be filled up by visiting the symbol table
+                        // And will continue to iterate as elements are added while visited `deferredPrivates`
+                        // (As that's how a map iterator is defined to work)
+                        deferredPrivatesStack[deferredPrivatesStack.length - 1].forEach(function (symbol) {
+                            serializeSymbol(symbol, /*isPrivate*/ true, !!propertyAsAlias);
+                        });
+                        deferredPrivatesStack.pop();
                     }
-                    result[i] = defaultType ? instantiateType(defaultType, createTypeMapper(typeParameters, result)) : baseDefaultType;
                 }
-                result.length = typeParameters.length;
-                return result;
-            }
-            return typeArguments && typeArguments.slice();
-        }
-        function getSignatureFromDeclaration(declaration) {
-            var links = getNodeLinks(declaration);
-            if (!links.resolvedSignature) {
-                var parameters = [];
-                var flags = 0 /* None */;
-                var minArgumentCount = 0;
-                var thisParameter = void 0;
-                var hasThisParameter = false;
-                var iife = ts.getImmediatelyInvokedFunctionExpression(declaration);
-                var isJSConstructSignature = ts.isJSDocConstructSignature(declaration);
-                var isUntypedSignatureInJSFile = !iife &&
-                    ts.isInJSFile(declaration) &&
-                    ts.isValueSignatureDeclaration(declaration) &&
-                    !ts.hasJSDocParameterTags(declaration) &&
-                    !ts.getJSDocType(declaration);
-                if (isUntypedSignatureInJSFile) {
-                    flags |= 16 /* IsUntypedSignatureInJSFile */;
+                function serializeSymbol(symbol, isPrivate, propertyAsAlias) {
+                    // cache visited list based on merged symbol, since we want to use the unmerged top-level symbol, but
+                    // still skip reserializing it if we encounter the merged product later on
+                    var visitedSym = getMergedSymbol(symbol);
+                    if (visitedSymbols.has(getSymbolId(visitedSym))) {
+                        return; // Already printed
+                    }
+                    visitedSymbols.add(getSymbolId(visitedSym));
+                    // Only actually serialize symbols within the correct enclosing declaration, otherwise do nothing with the out-of-context symbol
+                    var skipMembershipCheck = !isPrivate; // We only call this on exported symbols when we know they're in the correct scope
+                    if (skipMembershipCheck || (!!ts.length(symbol.declarations) && ts.some(symbol.declarations, function (d) { return !!ts.findAncestor(d, function (n) { return n === enclosingDeclaration; }); }))) {
+                        var oldContext = context;
+                        context = cloneNodeBuilderContext(context);
+                        var result = serializeSymbolWorker(symbol, isPrivate, propertyAsAlias);
+                        context = oldContext;
+                        return result;
+                    }
                 }
-                // If this is a JSDoc construct signature, then skip the first parameter in the
-                // parameter list.  The first parameter represents the return type of the construct
-                // signature.
-                for (var i = isJSConstructSignature ? 1 : 0; i < declaration.parameters.length; i++) {
-                    var param = declaration.parameters[i];
-                    var paramSymbol = param.symbol;
-                    var type = ts.isJSDocParameterTag(param) ? (param.typeExpression && param.typeExpression.type) : param.type;
-                    // Include parameter symbol instead of property symbol in the signature
-                    if (paramSymbol && !!(paramSymbol.flags & 4 /* Property */) && !ts.isBindingPattern(param.name)) {
-                        var resolvedSymbol = resolveName(param, paramSymbol.escapedName, 111551 /* Value */, undefined, undefined, /*isUse*/ false);
-                        paramSymbol = resolvedSymbol;
+                // Synthesize declarations for a symbol - might be an Interface, a Class, a Namespace, a Type, a Variable (const, let, or var), an Alias
+                // or a merge of some number of those.
+                // An interesting challenge is ensuring that when classes merge with namespaces and interfaces, is keeping
+                // each symbol in only one of the representations
+                // Also, synthesizing a default export of some kind
+                // If it's an alias: emit `export default ref`
+                // If it's a property: emit `export default _default` with a `_default` prop
+                // If it's a class/interface/function: emit a class/interface/function with a `default` modifier
+                // These forms can merge, eg (`export default 12; export default interface A {}`)
+                function serializeSymbolWorker(symbol, isPrivate, propertyAsAlias) {
+                    var symbolName = ts.unescapeLeadingUnderscores(symbol.escapedName);
+                    var isDefault = symbol.escapedName === "default" /* Default */;
+                    if (isPrivate && !(context.flags & 131072 /* AllowAnonymousIdentifier */) && ts.isStringANonContextualKeyword(symbolName) && !isDefault) {
+                        // Oh no. We cannot use this symbol's name as it's name... It's likely some jsdoc had an invalid name like `export` or `default` :(
+                        context.encounteredError = true;
+                        // TODO: Issue error via symbol tracker?
+                        return; // If we need to emit a private with a keyword name, we're done for, since something else will try to refer to it by that name
                     }
-                    if (i === 0 && paramSymbol.escapedName === "this" /* This */) {
-                        hasThisParameter = true;
-                        thisParameter = param.symbol;
+                    var needsPostExportDefault = isDefault && !!(symbol.flags & -113 /* ExportDoesNotSupportDefaultModifier */
+                        || (symbol.flags & 16 /* Function */ && ts.length(getPropertiesOfType(getTypeOfSymbol(symbol))))) && !(symbol.flags & 2097152 /* Alias */); // An alias symbol should preclude needing to make an alias ourselves
+                    var needsExportDeclaration = !needsPostExportDefault && !isPrivate && ts.isStringANonContextualKeyword(symbolName) && !isDefault;
+                    // `serializeVariableOrProperty` will handle adding the export declaration if it is run (since `getInternalSymbolName` will create the name mapping), so we need to ensuer we unset `needsExportDeclaration` if it is
+                    if (needsPostExportDefault || needsExportDeclaration) {
+                        isPrivate = true;
                     }
-                    else {
-                        parameters.push(paramSymbol);
+                    var modifierFlags = (!isPrivate ? 1 /* Export */ : 0) | (isDefault && !needsPostExportDefault ? 512 /* Default */ : 0);
+                    var isConstMergedWithNS = symbol.flags & 1536 /* Module */ &&
+                        symbol.flags & (2 /* BlockScopedVariable */ | 1 /* FunctionScopedVariable */ | 4 /* Property */) &&
+                        symbol.escapedName !== "export=" /* ExportEquals */;
+                    var isConstMergedWithNSPrintableAsSignatureMerge = isConstMergedWithNS && isTypeRepresentableAsFunctionNamespaceMerge(getTypeOfSymbol(symbol), symbol);
+                    if (symbol.flags & (16 /* Function */ | 8192 /* Method */) || isConstMergedWithNSPrintableAsSignatureMerge) {
+                        serializeAsFunctionNamespaceMerge(getTypeOfSymbol(symbol), symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
                     }
-                    if (type && type.kind === 187 /* LiteralType */) {
-                        flags |= 2 /* HasLiteralTypes */;
+                    if (symbol.flags & 524288 /* TypeAlias */) {
+                        serializeTypeAlias(symbol, symbolName, modifierFlags);
                     }
-                    // Record a new minimum argument count if this is not an optional parameter
-                    var isOptionalParameter_1 = isOptionalJSDocParameterTag(param) ||
-                        param.initializer || param.questionToken || param.dotDotDotToken ||
-                        iife && parameters.length > iife.arguments.length && !type ||
-                        isJSDocOptionalParameter(param);
-                    if (!isOptionalParameter_1) {
-                        minArgumentCount = parameters.length;
+                    // Need to skip over export= symbols below - json source files get a single `Property` flagged
+                    // symbol of name `export=` which needs to be handled like an alias. It's not great, but it is what it is.
+                    if (symbol.flags & (2 /* BlockScopedVariable */ | 1 /* FunctionScopedVariable */ | 4 /* Property */)
+                        && symbol.escapedName !== "export=" /* ExportEquals */
+                        && !(symbol.flags & 4194304 /* Prototype */)
+                        && !(symbol.flags & 32 /* Class */)
+                        && !isConstMergedWithNSPrintableAsSignatureMerge) {
+                        if (propertyAsAlias) {
+                            var createdExport = serializeMaybeAliasAssignment(symbol);
+                            if (createdExport) {
+                                needsExportDeclaration = false;
+                                needsPostExportDefault = false;
+                            }
+                        }
+                        else {
+                            var type = getTypeOfSymbol(symbol);
+                            var localName = getInternalSymbolName(symbol, symbolName);
+                            if (!(symbol.flags & 16 /* Function */) && isTypeRepresentableAsFunctionNamespaceMerge(type, symbol)) {
+                                // If the type looks like a function declaration + ns could represent it, and it's type is sourced locally, rewrite it into a function declaration + ns
+                                serializeAsFunctionNamespaceMerge(type, symbol, localName, modifierFlags);
+                            }
+                            else {
+                                // A Class + Property merge is made for a `module.exports.Member = class {}`, and it doesn't serialize well as either a class _or_ a property symbol - in fact, _it behaves like an alias!_
+                                // `var` is `FunctionScopedVariable`, `const` and `let` are `BlockScopedVariable`, and `module.exports.thing =` is `Property`
+                                var flags = !(symbol.flags & 2 /* BlockScopedVariable */) ? undefined
+                                    : isConstVariable(symbol) ? 2 /* Const */
+                                        : 1 /* Let */;
+                                var name = (needsPostExportDefault || !(symbol.flags & 4 /* Property */)) ? localName : getUnusedName(localName, symbol);
+                                var textRange = symbol.declarations && ts.find(symbol.declarations, function (d) { return ts.isVariableDeclaration(d); });
+                                if (textRange && ts.isVariableDeclarationList(textRange.parent) && textRange.parent.declarations.length === 1) {
+                                    textRange = textRange.parent.parent;
+                                }
+                                var propertyAccessRequire = ts.find(symbol.declarations, ts.isPropertyAccessExpression);
+                                if (propertyAccessRequire && ts.isBinaryExpression(propertyAccessRequire.parent) && ts.isIdentifier(propertyAccessRequire.parent.right)
+                                    && type.symbol && ts.isSourceFile(type.symbol.valueDeclaration)) {
+                                    var alias = localName === propertyAccessRequire.parent.right.escapedText ? undefined : propertyAccessRequire.parent.right;
+                                    addResult(ts.factory.createExportDeclaration(
+                                    /*decorators*/ undefined, 
+                                    /*modifiers*/ undefined, 
+                                    /*isTypeOnly*/ false, ts.factory.createNamedExports([ts.factory.createExportSpecifier(alias, localName)])), 0 /* None */);
+                                    context.tracker.trackSymbol(type.symbol, context.enclosingDeclaration, 111551 /* Value */);
+                                }
+                                else {
+                                    var statement = ts.setTextRange(ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList([
+                                        ts.factory.createVariableDeclaration(name, /*exclamationToken*/ undefined, serializeTypeForDeclaration(context, type, symbol, enclosingDeclaration, includePrivateSymbol, bundled))
+                                    ], flags)), textRange);
+                                    addResult(statement, name !== localName ? modifierFlags & ~1 /* Export */ : modifierFlags);
+                                    if (name !== localName && !isPrivate) {
+                                        // We rename the variable declaration we generate for Property symbols since they may have a name which
+                                        // conflicts with a local declaration. For example, given input:
+                                        // ```
+                                        // function g() {}
+                                        // module.exports.g = g
+                                        // ```
+                                        // In such a situation, we have a local variable named `g`, and a separate exported variable named `g`.
+                                        // Naively, we would emit
+                                        // ```
+                                        // function g() {}
+                                        // export const g: typeof g;
+                                        // ```
+                                        // That's obviously incorrect - the `g` in the type annotation needs to refer to the local `g`, but
+                                        // the export declaration shadows it.
+                                        // To work around that, we instead write
+                                        // ```
+                                        // function g() {}
+                                        // const g_1: typeof g;
+                                        // export { g_1 as g };
+                                        // ```
+                                        // To create an export named `g` that does _not_ shadow the local `g`
+                                        addResult(ts.factory.createExportDeclaration(
+                                        /*decorators*/ undefined, 
+                                        /*modifiers*/ undefined, 
+                                        /*isTypeOnly*/ false, ts.factory.createNamedExports([ts.factory.createExportSpecifier(name, localName)])), 0 /* None */);
+                                        needsExportDeclaration = false;
+                                        needsPostExportDefault = false;
+                                    }
+                                }
+                            }
+                        }
+                    }
+                    if (symbol.flags & 384 /* Enum */) {
+                        serializeEnum(symbol, symbolName, modifierFlags);
+                    }
+                    if (symbol.flags & 32 /* Class */) {
+                        if (symbol.flags & 4 /* Property */ && ts.isBinaryExpression(symbol.valueDeclaration.parent) && ts.isClassExpression(symbol.valueDeclaration.parent.right)) {
+                            // Looks like a `module.exports.Sub = class {}` - if we serialize `symbol` as a class, the result will have no members,
+                            // since the classiness is actually from the target of the effective alias the symbol is. yes. A BlockScopedVariable|Class|Property
+                            // _really_ acts like an Alias, and none of a BlockScopedVariable, Class, or Property. This is the travesty of JS binding today.
+                            serializeAsAlias(symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
+                        }
+                        else {
+                            serializeAsClass(symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
+                        }
+                    }
+                    if ((symbol.flags & (512 /* ValueModule */ | 1024 /* NamespaceModule */) && (!isConstMergedWithNS || isTypeOnlyNamespace(symbol))) || isConstMergedWithNSPrintableAsSignatureMerge) {
+                        serializeModule(symbol, symbolName, modifierFlags);
+                    }
+                    // The class meaning serialization should handle serializing all interface members
+                    if (symbol.flags & 64 /* Interface */ && !(symbol.flags & 32 /* Class */)) {
+                        serializeInterface(symbol, symbolName, modifierFlags);
+                    }
+                    if (symbol.flags & 2097152 /* Alias */) {
+                        serializeAsAlias(symbol, getInternalSymbolName(symbol, symbolName), modifierFlags);
+                    }
+                    if (symbol.flags & 4 /* Property */ && symbol.escapedName === "export=" /* ExportEquals */) {
+                        serializeMaybeAliasAssignment(symbol);
+                    }
+                    if (symbol.flags & 8388608 /* ExportStar */) {
+                        // synthesize export * from "moduleReference"
+                        // Straightforward - only one thing to do - make an export declaration
+                        for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                            var node = _a[_i];
+                            var resolvedModule = resolveExternalModuleName(node, node.moduleSpecifier);
+                            if (!resolvedModule)
+                                continue;
+                            addResult(ts.factory.createExportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*isTypeOnly*/ false, /*exportClause*/ undefined, ts.factory.createStringLiteral(getSpecifierForModuleSymbol(resolvedModule, context))), 0 /* None */);
+                        }
+                    }
+                    if (needsPostExportDefault) {
+                        addResult(ts.factory.createExportAssignment(/*decorators*/ undefined, /*modifiers*/ undefined, /*isExportAssignment*/ false, ts.factory.createIdentifier(getInternalSymbolName(symbol, symbolName))), 0 /* None */);
+                    }
+                    else if (needsExportDeclaration) {
+                        addResult(ts.factory.createExportDeclaration(
+                        /*decorators*/ undefined, 
+                        /*modifiers*/ undefined, 
+                        /*isTypeOnly*/ false, ts.factory.createNamedExports([ts.factory.createExportSpecifier(getInternalSymbolName(symbol, symbolName), symbolName)])), 0 /* None */);
                     }
                 }
-                // If only one accessor includes a this-type annotation, the other behaves as if it had the same type annotation
-                if ((declaration.kind === 163 /* GetAccessor */ || declaration.kind === 164 /* SetAccessor */) &&
-                    !hasNonBindableDynamicName(declaration) &&
-                    (!hasThisParameter || !thisParameter)) {
-                    var otherKind = declaration.kind === 163 /* GetAccessor */ ? 164 /* SetAccessor */ : 163 /* GetAccessor */;
-                    var other = ts.getDeclarationOfKind(getSymbolOfNode(declaration), otherKind);
-                    if (other) {
-                        thisParameter = getAnnotatedAccessorThisParameter(other);
+                function includePrivateSymbol(symbol) {
+                    if (ts.some(symbol.declarations, ts.isParameterDeclaration))
+                        return;
+                    ts.Debug.assertIsDefined(deferredPrivatesStack[deferredPrivatesStack.length - 1]);
+                    getUnusedName(ts.unescapeLeadingUnderscores(symbol.escapedName), symbol); // Call to cache unique name for symbol
+                    // Blanket moving (import) aliases into the root private context should work, since imports are not valid within namespaces
+                    // (so they must have been in the root to begin with if they were real imports) cjs `require` aliases (an upcoming feature)
+                    // will throw a wrench in this, since those may have been nested, but we'll need to synthesize them in the outer scope
+                    // anyway, as that's the only place the import they translate to is valid. In such a case, we might need to use a unique name
+                    // for the moved import; which hopefully the above `getUnusedName` call should produce.
+                    var isExternalImportAlias = !!(symbol.flags & 2097152 /* Alias */) && !ts.some(symbol.declarations, function (d) {
+                        return !!ts.findAncestor(d, ts.isExportDeclaration) ||
+                            ts.isNamespaceExport(d) ||
+                            (ts.isImportEqualsDeclaration(d) && !ts.isExternalModuleReference(d.moduleReference));
+                    });
+                    deferredPrivatesStack[isExternalImportAlias ? 0 : (deferredPrivatesStack.length - 1)].set(getSymbolId(symbol), symbol);
+                }
+                function isExportingScope(enclosingDeclaration) {
+                    return ((ts.isSourceFile(enclosingDeclaration) && (ts.isExternalOrCommonJsModule(enclosingDeclaration) || ts.isJsonSourceFile(enclosingDeclaration))) ||
+                        (ts.isAmbientModule(enclosingDeclaration) && !ts.isGlobalScopeAugmentation(enclosingDeclaration)));
+                }
+                // Prepends a `declare` and/or `export` modifier if the context requires it, and then adds `node` to `result` and returns `node`
+                function addResult(node, additionalModifierFlags) {
+                    if (ts.canHaveModifiers(node)) {
+                        var newModifierFlags = 0 /* None */;
+                        if (additionalModifierFlags & 1 /* Export */ &&
+                            context.enclosingDeclaration &&
+                            (isExportingScope(context.enclosingDeclaration) || ts.isModuleDeclaration(context.enclosingDeclaration)) &&
+                            canHaveExportModifier(node)) {
+                            // Classes, namespaces, variables, functions, interfaces, and types should all be `export`ed in a module context if not private
+                            newModifierFlags |= 1 /* Export */;
+                        }
+                        if (addingDeclare && !(newModifierFlags & 1 /* Export */) &&
+                            (!context.enclosingDeclaration || !(context.enclosingDeclaration.flags & 8388608 /* Ambient */)) &&
+                            (ts.isEnumDeclaration(node) || ts.isVariableStatement(node) || ts.isFunctionDeclaration(node) || ts.isClassDeclaration(node) || ts.isModuleDeclaration(node))) {
+                            // Classes, namespaces, variables, enums, and functions all need `declare` modifiers to be valid in a declaration file top-level scope
+                            newModifierFlags |= 2 /* Ambient */;
+                        }
+                        if ((additionalModifierFlags & 512 /* Default */) && (ts.isClassDeclaration(node) || ts.isInterfaceDeclaration(node) || ts.isFunctionDeclaration(node))) {
+                            newModifierFlags |= 512 /* Default */;
+                        }
+                        if (newModifierFlags) {
+                            node = ts.factory.updateModifiers(node, newModifierFlags | ts.getEffectiveModifierFlags(node));
+                        }
                     }
+                    results.push(node);
                 }
-                var classType = declaration.kind === 162 /* Constructor */ ?
-                    getDeclaredTypeOfClassOrInterface(getMergedSymbol(declaration.parent.symbol))
-                    : undefined;
-                var typeParameters = classType ? classType.localTypeParameters : getTypeParametersFromDeclaration(declaration);
-                if (ts.hasRestParameter(declaration) || ts.isInJSFile(declaration) && maybeAddJsSyntheticRestParameter(declaration, parameters)) {
-                    flags |= 1 /* HasRestParameter */;
+                function serializeTypeAlias(symbol, symbolName, modifierFlags) {
+                    var aliasType = getDeclaredTypeOfTypeAlias(symbol);
+                    var typeParams = getSymbolLinks(symbol).typeParameters;
+                    var typeParamDecls = ts.map(typeParams, function (p) { return typeParameterToDeclaration(p, context); });
+                    var jsdocAliasDecl = ts.find(symbol.declarations, ts.isJSDocTypeAlias);
+                    var commentText = jsdocAliasDecl ? jsdocAliasDecl.comment || jsdocAliasDecl.parent.comment : undefined;
+                    var oldFlags = context.flags;
+                    context.flags |= 8388608 /* InTypeAlias */;
+                    addResult(ts.setSyntheticLeadingComments(ts.factory.createTypeAliasDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, getInternalSymbolName(symbol, symbolName), typeParamDecls, typeToTypeNodeHelper(aliasType, context)), !commentText ? [] : [{ kind: 3 /* MultiLineCommentTrivia */, text: "*\n * " + commentText.replace(/\n/g, "\n * ") + "\n ", pos: -1, end: -1, hasTrailingNewLine: true }]), modifierFlags);
+                    context.flags = oldFlags;
                 }
-                links.resolvedSignature = createSignature(declaration, typeParameters, thisParameter, parameters, 
-                /*resolvedReturnType*/ undefined, /*resolvedTypePredicate*/ undefined, minArgumentCount, flags);
-            }
-            return links.resolvedSignature;
-        }
-        /**
-         * A JS function gets a synthetic rest parameter if it references `arguments` AND:
-         * 1. It has no parameters but at least one `@param` with a type that starts with `...`
-         * OR
-         * 2. It has at least one parameter, and the last parameter has a matching `@param` with a type that starts with `...`
-         */
-        function maybeAddJsSyntheticRestParameter(declaration, parameters) {
-            if (ts.isJSDocSignature(declaration) || !containsArgumentsReference(declaration)) {
-                return false;
-            }
-            var lastParam = ts.lastOrUndefined(declaration.parameters);
-            var lastParamTags = lastParam ? ts.getJSDocParameterTags(lastParam) : ts.getJSDocTags(declaration).filter(ts.isJSDocParameterTag);
-            var lastParamVariadicType = ts.firstDefined(lastParamTags, function (p) {
-                return p.typeExpression && ts.isJSDocVariadicType(p.typeExpression.type) ? p.typeExpression.type : undefined;
-            });
-            var syntheticArgsSymbol = createSymbol(3 /* Variable */, "args", 32768 /* RestParameter */);
-            syntheticArgsSymbol.type = lastParamVariadicType ? createArrayType(getTypeFromTypeNode(lastParamVariadicType.type)) : anyArrayType;
-            if (lastParamVariadicType) {
-                // Replace the last parameter with a rest parameter.
-                parameters.pop();
-            }
-            parameters.push(syntheticArgsSymbol);
-            return true;
-        }
-        function getSignatureOfTypeTag(node) {
-            // should be attached to a function declaration or expression
-            if (!(ts.isInJSFile(node) && ts.isFunctionLikeDeclaration(node)))
-                return undefined;
-            var typeTag = ts.getJSDocTypeTag(node);
-            var signature = typeTag && typeTag.typeExpression && getSingleCallSignature(getTypeFromTypeNode(typeTag.typeExpression));
-            return signature && getErasedSignature(signature);
-        }
-        function getReturnTypeOfTypeTag(node) {
-            var signature = getSignatureOfTypeTag(node);
-            return signature && getReturnTypeOfSignature(signature);
-        }
-        function containsArgumentsReference(declaration) {
-            var links = getNodeLinks(declaration);
-            if (links.containsArgumentsReference === undefined) {
-                if (links.flags & 8192 /* CaptureArguments */) {
-                    links.containsArgumentsReference = true;
+                function serializeInterface(symbol, symbolName, modifierFlags) {
+                    var interfaceType = getDeclaredTypeOfClassOrInterface(symbol);
+                    var localParams = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
+                    var typeParamDecls = ts.map(localParams, function (p) { return typeParameterToDeclaration(p, context); });
+                    var baseTypes = getBaseTypes(interfaceType);
+                    var baseType = ts.length(baseTypes) ? getIntersectionType(baseTypes) : undefined;
+                    var members = ts.flatMap(getPropertiesOfType(interfaceType), function (p) { return serializePropertySymbolForInterface(p, baseType); });
+                    var callSignatures = serializeSignatures(0 /* Call */, interfaceType, baseType, 169 /* CallSignature */);
+                    var constructSignatures = serializeSignatures(1 /* Construct */, interfaceType, baseType, 170 /* ConstructSignature */);
+                    var indexSignatures = serializeIndexSignatures(interfaceType, baseType);
+                    var heritageClauses = !ts.length(baseTypes) ? undefined : [ts.factory.createHeritageClause(93 /* ExtendsKeyword */, ts.mapDefined(baseTypes, function (b) { return trySerializeAsTypeReference(b, 111551 /* Value */); }))];
+                    addResult(ts.factory.createInterfaceDeclaration(
+                    /*decorators*/ undefined, 
+                    /*modifiers*/ undefined, getInternalSymbolName(symbol, symbolName), typeParamDecls, heritageClauses, __spreadArrays(indexSignatures, constructSignatures, callSignatures, members)), modifierFlags);
                 }
-                else {
-                    links.containsArgumentsReference = traverse(declaration.body);
+                function getNamespaceMembersForSerialization(symbol) {
+                    return !symbol.exports ? [] : ts.filter(ts.arrayFrom(symbol.exports.values()), isNamespaceMember);
                 }
-            }
-            return links.containsArgumentsReference;
-            function traverse(node) {
-                if (!node)
-                    return false;
-                switch (node.kind) {
-                    case 75 /* Identifier */:
-                        return node.escapedText === "arguments" && ts.isExpressionNode(node);
-                    case 159 /* PropertyDeclaration */:
-                    case 161 /* MethodDeclaration */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        return node.name.kind === 154 /* ComputedPropertyName */
-                            && traverse(node.name);
-                    default:
-                        return !ts.nodeStartsNewLexicalEnvironment(node) && !ts.isPartOfTypeNode(node) && !!ts.forEachChild(node, traverse);
+                function isTypeOnlyNamespace(symbol) {
+                    return ts.every(getNamespaceMembersForSerialization(symbol), function (m) { return !(resolveSymbol(m).flags & 111551 /* Value */); });
                 }
-            }
-        }
-        function getSignaturesOfSymbol(symbol) {
-            if (!symbol)
-                return ts.emptyArray;
-            var result = [];
-            for (var i = 0; i < symbol.declarations.length; i++) {
-                var decl = symbol.declarations[i];
-                if (!ts.isFunctionLike(decl))
-                    continue;
-                // Don't include signature if node is the implementation of an overloaded function. A node is considered
-                // an implementation node if it has a body and the previous node is of the same kind and immediately
-                // precedes the implementation node (i.e. has the same parent and ends where the implementation starts).
-                if (i > 0 && decl.body) {
-                    var previous = symbol.declarations[i - 1];
-                    if (decl.parent === previous.parent && decl.kind === previous.kind && decl.pos === previous.end) {
-                        continue;
+                function serializeModule(symbol, symbolName, modifierFlags) {
+                    var members = getNamespaceMembersForSerialization(symbol);
+                    // Split NS members up by declaration - members whose parent symbol is the ns symbol vs those whose is not (but were added in later via merging)
+                    var locationMap = ts.arrayToMultiMap(members, function (m) { return m.parent && m.parent === symbol ? "real" : "merged"; });
+                    var realMembers = locationMap.get("real") || ts.emptyArray;
+                    var mergedMembers = locationMap.get("merged") || ts.emptyArray;
+                    // TODO: `suppressNewPrivateContext` is questionable -we need to simply be emitting privates in whatever scope they were declared in, rather
+                    // than whatever scope we traverse to them in. That's a bit of a complex rewrite, since we're not _actually_ tracking privates at all in advance,
+                    // so we don't even have placeholders to fill in.
+                    if (ts.length(realMembers)) {
+                        var localName = getInternalSymbolName(symbol, symbolName);
+                        serializeAsNamespaceDeclaration(realMembers, localName, modifierFlags, !!(symbol.flags & (16 /* Function */ | 67108864 /* Assignment */)));
+                    }
+                    if (ts.length(mergedMembers)) {
+                        var containingFile_1 = ts.getSourceFileOfNode(context.enclosingDeclaration);
+                        var localName = getInternalSymbolName(symbol, symbolName);
+                        var nsBody = ts.factory.createModuleBlock([ts.factory.createExportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, 
+                            /*isTypeOnly*/ false, ts.factory.createNamedExports(ts.mapDefined(ts.filter(mergedMembers, function (n) { return n.escapedName !== "export=" /* ExportEquals */; }), function (s) {
+                                var _a, _b;
+                                var name = ts.unescapeLeadingUnderscores(s.escapedName);
+                                var localName = getInternalSymbolName(s, name);
+                                var aliasDecl = s.declarations && getDeclarationOfAliasSymbol(s);
+                                if (containingFile_1 && (aliasDecl ? containingFile_1 !== ts.getSourceFileOfNode(aliasDecl) : !ts.some(s.declarations, function (d) { return ts.getSourceFileOfNode(d) === containingFile_1; }))) {
+                                    (_b = (_a = context.tracker) === null || _a === void 0 ? void 0 : _a.reportNonlocalAugmentation) === null || _b === void 0 ? void 0 : _b.call(_a, containingFile_1, symbol, s);
+                                    return undefined;
+                                }
+                                var target = aliasDecl && getTargetOfAliasDeclaration(aliasDecl, /*dontRecursivelyResolve*/ true);
+                                includePrivateSymbol(target || s);
+                                var targetName = target ? getInternalSymbolName(target, ts.unescapeLeadingUnderscores(target.escapedName)) : localName;
+                                return ts.factory.createExportSpecifier(name === targetName ? undefined : targetName, name);
+                            })))]);
+                        addResult(ts.factory.createModuleDeclaration(
+                        /*decorators*/ undefined, 
+                        /*modifiers*/ undefined, ts.factory.createIdentifier(localName), nsBody, 16 /* Namespace */), 0 /* None */);
                     }
                 }
-                result.push(getSignatureFromDeclaration(decl));
-            }
-            return result;
-        }
-        function resolveExternalModuleTypeByLiteral(name) {
-            var moduleSym = resolveExternalModuleName(name, name);
-            if (moduleSym) {
-                var resolvedModuleSymbol = resolveExternalModuleSymbol(moduleSym);
-                if (resolvedModuleSymbol) {
-                    return getTypeOfSymbol(resolvedModuleSymbol);
-                }
-            }
-            return anyType;
-        }
-        function getThisTypeOfSignature(signature) {
-            if (signature.thisParameter) {
-                return getTypeOfSymbol(signature.thisParameter);
-            }
-        }
-        function getTypePredicateOfSignature(signature) {
-            if (!signature.resolvedTypePredicate) {
-                if (signature.target) {
-                    var targetTypePredicate = getTypePredicateOfSignature(signature.target);
-                    signature.resolvedTypePredicate = targetTypePredicate ? instantiateTypePredicate(targetTypePredicate, signature.mapper) : noTypePredicate;
+                function serializeEnum(symbol, symbolName, modifierFlags) {
+                    addResult(ts.factory.createEnumDeclaration(
+                    /*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags(isConstEnumSymbol(symbol) ? 2048 /* Const */ : 0), getInternalSymbolName(symbol, symbolName), ts.map(ts.filter(getPropertiesOfType(getTypeOfSymbol(symbol)), function (p) { return !!(p.flags & 8 /* EnumMember */); }), function (p) {
+                        // TODO: Handle computed names
+                        // I hate that to get the initialized value we need to walk back to the declarations here; but there's no
+                        // other way to get the possible const value of an enum member that I'm aware of, as the value is cached
+                        // _on the declaration_, not on the declaration's symbol...
+                        var initializedValue = p.declarations && p.declarations[0] && ts.isEnumMember(p.declarations[0]) ? getConstantValue(p.declarations[0]) : undefined;
+                        return ts.factory.createEnumMember(ts.unescapeLeadingUnderscores(p.escapedName), initializedValue === undefined ? undefined :
+                            typeof initializedValue === "string" ? ts.factory.createStringLiteral(initializedValue) :
+                                ts.factory.createNumericLiteral(initializedValue));
+                    })), modifierFlags);
                 }
-                else if (signature.unionSignatures) {
-                    signature.resolvedTypePredicate = getUnionTypePredicate(signature.unionSignatures) || noTypePredicate;
+                function serializeAsFunctionNamespaceMerge(type, symbol, localName, modifierFlags) {
+                    var signatures = getSignaturesOfType(type, 0 /* Call */);
+                    for (var _i = 0, signatures_2 = signatures; _i < signatures_2.length; _i++) {
+                        var sig = signatures_2[_i];
+                        // Each overload becomes a separate function declaration, in order
+                        var decl = signatureToSignatureDeclarationHelper(sig, 251 /* FunctionDeclaration */, context, { name: ts.factory.createIdentifier(localName), privateSymbolVisitor: includePrivateSymbol, bundledImports: bundled });
+                        // for expressions assigned to `var`s, use the `var` as the text range
+                        addResult(ts.setTextRange(decl, sig.declaration && ts.isVariableDeclaration(sig.declaration.parent) && sig.declaration.parent.parent || sig.declaration), modifierFlags);
+                    }
+                    // Module symbol emit will take care of module-y members, provided it has exports
+                    if (!(symbol.flags & (512 /* ValueModule */ | 1024 /* NamespaceModule */) && !!symbol.exports && !!symbol.exports.size)) {
+                        var props = ts.filter(getPropertiesOfType(type), isNamespaceMember);
+                        serializeAsNamespaceDeclaration(props, localName, modifierFlags, /*suppressNewPrivateContext*/ true);
+                    }
                 }
-                else {
-                    var type = signature.declaration && ts.getEffectiveReturnTypeNode(signature.declaration);
-                    var jsdocPredicate = void 0;
-                    if (!type && ts.isInJSFile(signature.declaration)) {
-                        var jsdocSignature = getSignatureOfTypeTag(signature.declaration);
-                        if (jsdocSignature && signature !== jsdocSignature) {
-                            jsdocPredicate = getTypePredicateOfSignature(jsdocSignature);
-                        }
+                function serializeAsNamespaceDeclaration(props, localName, modifierFlags, suppressNewPrivateContext) {
+                    if (ts.length(props)) {
+                        var localVsRemoteMap = ts.arrayToMultiMap(props, function (p) {
+                            return !ts.length(p.declarations) || ts.some(p.declarations, function (d) {
+                                return ts.getSourceFileOfNode(d) === ts.getSourceFileOfNode(context.enclosingDeclaration);
+                            }) ? "local" : "remote";
+                        });
+                        var localProps = localVsRemoteMap.get("local") || ts.emptyArray;
+                        // handle remote props first - we need to make an `import` declaration that points at the module containing each remote
+                        // prop in the outermost scope (TODO: a namespace within a namespace would need to be appropriately handled by this)
+                        // Example:
+                        // import Foo_1 = require("./exporter");
+                        // export namespace ns {
+                        //     import Foo = Foo_1.Foo;
+                        //     export { Foo };
+                        //     export const c: number;
+                        // }
+                        // This is needed because in JS, statements like `const x = require("./f")` support both type and value lookup, even if they're
+                        // normally just value lookup (so it functions kinda like an alias even when it's not an alias)
+                        // _Usually_, we'll simply print the top-level as an alias instead of a `var` in such situations, however is is theoretically
+                        // possible to encounter a situation where a type has members from both the current file and other files - in those situations,
+                        // emit akin to the above would be needed.
+                        // Add a namespace
+                        // Create namespace as non-synthetic so it is usable as an enclosing declaration
+                        var fakespace = ts.parseNodeFactory.createModuleDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.factory.createIdentifier(localName), ts.factory.createModuleBlock([]), 16 /* Namespace */);
+                        ts.setParent(fakespace, enclosingDeclaration);
+                        fakespace.locals = ts.createSymbolTable(props);
+                        fakespace.symbol = props[0].parent;
+                        var oldResults = results;
+                        results = [];
+                        var oldAddingDeclare = addingDeclare;
+                        addingDeclare = false;
+                        var subcontext = __assign(__assign({}, context), { enclosingDeclaration: fakespace });
+                        var oldContext = context;
+                        context = subcontext;
+                        // TODO: implement handling for the localVsRemoteMap.get("remote") - should be difficult to trigger (see comment above), as only interesting cross-file js merges should make this possible
+                        visitSymbolTable(ts.createSymbolTable(localProps), suppressNewPrivateContext, /*propertyAsAlias*/ true);
+                        context = oldContext;
+                        addingDeclare = oldAddingDeclare;
+                        var declarations = results;
+                        results = oldResults;
+                        // replace namespace with synthetic version
+                        var defaultReplaced = ts.map(declarations, function (d) { return ts.isExportAssignment(d) && !d.isExportEquals && ts.isIdentifier(d.expression) ? ts.factory.createExportDeclaration(
+                        /*decorators*/ undefined, 
+                        /*modifiers*/ undefined, 
+                        /*isTypeOnly*/ false, ts.factory.createNamedExports([ts.factory.createExportSpecifier(d.expression, ts.factory.createIdentifier("default" /* Default */))])) : d; });
+                        var exportModifierStripped = ts.every(defaultReplaced, function (d) { return ts.hasSyntacticModifier(d, 1 /* Export */); }) ? ts.map(defaultReplaced, removeExportModifier) : defaultReplaced;
+                        fakespace = ts.factory.updateModuleDeclaration(fakespace, fakespace.decorators, fakespace.modifiers, fakespace.name, ts.factory.createModuleBlock(exportModifierStripped));
+                        addResult(fakespace, modifierFlags); // namespaces can never be default exported
                     }
-                    signature.resolvedTypePredicate = type && ts.isTypePredicateNode(type) ?
-                        createTypePredicateFromTypePredicateNode(type, signature) :
-                        jsdocPredicate || noTypePredicate;
                 }
-                ts.Debug.assert(!!signature.resolvedTypePredicate);
-            }
-            return signature.resolvedTypePredicate === noTypePredicate ? undefined : signature.resolvedTypePredicate;
-        }
-        function createTypePredicateFromTypePredicateNode(node, signature) {
-            var parameterName = node.parameterName;
-            var type = node.type && getTypeFromTypeNode(node.type);
-            return parameterName.kind === 183 /* ThisType */ ?
-                createTypePredicate(node.assertsModifier ? 2 /* AssertsThis */ : 0 /* This */, /*parameterName*/ undefined, /*parameterIndex*/ undefined, type) :
-                createTypePredicate(node.assertsModifier ? 3 /* AssertsIdentifier */ : 1 /* Identifier */, parameterName.escapedText, ts.findIndex(signature.parameters, function (p) { return p.escapedName === parameterName.escapedText; }), type);
-        }
-        function getReturnTypeOfSignature(signature) {
-            if (!signature.resolvedReturnType) {
-                if (!pushTypeResolution(signature, 3 /* ResolvedReturnType */)) {
-                    return errorType;
+                function isNamespaceMember(p) {
+                    return !!(p.flags & (788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */)) ||
+                        !(p.flags & 4194304 /* Prototype */ || p.escapedName === "prototype" || p.valueDeclaration && ts.getEffectiveModifierFlags(p.valueDeclaration) & 32 /* Static */ && ts.isClassLike(p.valueDeclaration.parent));
                 }
-                var type = signature.target ? instantiateType(getReturnTypeOfSignature(signature.target), signature.mapper) :
-                    signature.unionSignatures ? getUnionType(ts.map(signature.unionSignatures, getReturnTypeOfSignature), 2 /* Subtype */) :
-                        getReturnTypeFromAnnotation(signature.declaration) ||
-                            (ts.nodeIsMissing(signature.declaration.body) ? anyType : getReturnTypeFromBody(signature.declaration));
-                if (signature.flags & 4 /* IsInnerCallChain */) {
-                    type = addOptionalTypeMarker(type);
+                function serializeAsClass(symbol, localName, modifierFlags) {
+                    var _a;
+                    var localParams = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
+                    var typeParamDecls = ts.map(localParams, function (p) { return typeParameterToDeclaration(p, context); });
+                    var classType = getDeclaredTypeOfClassOrInterface(symbol);
+                    var baseTypes = getBaseTypes(classType);
+                    var implementsExpressions = ts.mapDefined(getImplementsTypes(classType), serializeImplementedType);
+                    var staticType = getTypeOfSymbol(symbol);
+                    var isClass = !!((_a = staticType.symbol) === null || _a === void 0 ? void 0 : _a.valueDeclaration) && ts.isClassLike(staticType.symbol.valueDeclaration);
+                    var staticBaseType = isClass
+                        ? getBaseConstructorTypeOfClass(staticType)
+                        : anyType;
+                    var heritageClauses = __spreadArrays(!ts.length(baseTypes) ? [] : [ts.factory.createHeritageClause(93 /* ExtendsKeyword */, ts.map(baseTypes, function (b) { return serializeBaseType(b, staticBaseType, localName); }))], !ts.length(implementsExpressions) ? [] : [ts.factory.createHeritageClause(116 /* ImplementsKeyword */, implementsExpressions)]);
+                    var symbolProps = getNonInterhitedProperties(classType, baseTypes, getPropertiesOfType(classType));
+                    var publicSymbolProps = ts.filter(symbolProps, function (s) {
+                        // `valueDeclaration` could be undefined if inherited from
+                        // a union/intersection base type, but inherited properties
+                        // don't matter here.
+                        var valueDecl = s.valueDeclaration;
+                        return valueDecl && !(ts.isNamedDeclaration(valueDecl) && ts.isPrivateIdentifier(valueDecl.name));
+                    });
+                    var hasPrivateIdentifier = ts.some(symbolProps, function (s) {
+                        // `valueDeclaration` could be undefined if inherited from
+                        // a union/intersection base type, but inherited properties
+                        // don't matter here.
+                        var valueDecl = s.valueDeclaration;
+                        return valueDecl && ts.isNamedDeclaration(valueDecl) && ts.isPrivateIdentifier(valueDecl.name);
+                    });
+                    // Boil down all private properties into a single one.
+                    var privateProperties = hasPrivateIdentifier ?
+                        [ts.factory.createPropertyDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, ts.factory.createPrivateIdentifier("#private"), 
+                            /*questionOrExclamationToken*/ undefined, 
+                            /*type*/ undefined, 
+                            /*initializer*/ undefined)] :
+                        ts.emptyArray;
+                    var publicProperties = ts.flatMap(publicSymbolProps, function (p) { return serializePropertySymbolForClass(p, /*isStatic*/ false, baseTypes[0]); });
+                    // Consider static members empty if symbol also has function or module meaning - function namespacey emit will handle statics
+                    var staticMembers = ts.flatMap(ts.filter(getPropertiesOfType(staticType), function (p) { return !(p.flags & 4194304 /* Prototype */) && p.escapedName !== "prototype" && !isNamespaceMember(p); }), function (p) { return serializePropertySymbolForClass(p, /*isStatic*/ true, staticBaseType); });
+                    // When we encounter an `X.prototype.y` assignment in a JS file, we bind `X` as a class regardless as to whether
+                    // the value is ever initialized with a class or function-like value. For cases where `X` could never be
+                    // created via `new`, we will inject a `private constructor()` declaration to indicate it is not createable.
+                    var isNonConstructableClassLikeInJsFile = !isClass &&
+                        !!symbol.valueDeclaration &&
+                        ts.isInJSFile(symbol.valueDeclaration) &&
+                        !ts.some(getSignaturesOfType(staticType, 1 /* Construct */));
+                    var constructors = isNonConstructableClassLikeInJsFile ?
+                        [ts.factory.createConstructorDeclaration(/*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags(8 /* Private */), [], /*body*/ undefined)] :
+                        serializeSignatures(1 /* Construct */, staticType, baseTypes[0], 166 /* Constructor */);
+                    var indexSignatures = serializeIndexSignatures(classType, baseTypes[0]);
+                    addResult(ts.setTextRange(ts.factory.createClassDeclaration(
+                    /*decorators*/ undefined, 
+                    /*modifiers*/ undefined, localName, typeParamDecls, heritageClauses, __spreadArrays(indexSignatures, staticMembers, constructors, publicProperties, privateProperties)), symbol.declarations && ts.filter(symbol.declarations, function (d) { return ts.isClassDeclaration(d) || ts.isClassExpression(d); })[0]), modifierFlags);
+                }
+                function serializeAsAlias(symbol, localName, modifierFlags) {
+                    var _a, _b, _c, _d, _e;
+                    // synthesize an alias, eg `export { symbolName as Name }`
+                    // need to mark the alias `symbol` points at
+                    // as something we need to serialize as a private declaration as well
+                    var node = getDeclarationOfAliasSymbol(symbol);
+                    if (!node)
+                        return ts.Debug.fail();
+                    var target = getMergedSymbol(getTargetOfAliasDeclaration(node, /*dontRecursivelyResolve*/ true));
+                    if (!target) {
+                        return;
+                    }
+                    var verbatimTargetName = ts.unescapeLeadingUnderscores(target.escapedName);
+                    if (verbatimTargetName === "export=" /* ExportEquals */ && (compilerOptions.esModuleInterop || compilerOptions.allowSyntheticDefaultImports)) {
+                        // target refers to an `export=` symbol that was hoisted into a synthetic default - rename here to match
+                        verbatimTargetName = "default" /* Default */;
+                    }
+                    var targetName = getInternalSymbolName(target, verbatimTargetName);
+                    includePrivateSymbol(target); // the target may be within the same scope - attempt to serialize it first
+                    switch (node.kind) {
+                        case 198 /* BindingElement */:
+                            if (((_b = (_a = node.parent) === null || _a === void 0 ? void 0 : _a.parent) === null || _b === void 0 ? void 0 : _b.kind) === 249 /* VariableDeclaration */) {
+                                // const { SomeClass } = require('./lib');
+                                var specifier_1 = getSpecifierForModuleSymbol(target.parent || target, context); // './lib'
+                                var propertyName = node.propertyName;
+                                addResult(ts.factory.createImportDeclaration(
+                                /*decorators*/ undefined, 
+                                /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, /*name*/ undefined, ts.factory.createNamedImports([ts.factory.createImportSpecifier(propertyName && ts.isIdentifier(propertyName) ? ts.factory.createIdentifier(ts.idText(propertyName)) : undefined, ts.factory.createIdentifier(localName))])), ts.factory.createStringLiteral(specifier_1)), 0 /* None */);
+                                break;
+                            }
+                            // We don't know how to serialize this (nested?) binding element
+                            ts.Debug.failBadSyntaxKind(((_c = node.parent) === null || _c === void 0 ? void 0 : _c.parent) || node, "Unhandled binding element grandparent kind in declaration serialization");
+                            break;
+                        case 289 /* ShorthandPropertyAssignment */:
+                            if (((_e = (_d = node.parent) === null || _d === void 0 ? void 0 : _d.parent) === null || _e === void 0 ? void 0 : _e.kind) === 216 /* BinaryExpression */) {
+                                // module.exports = { SomeClass }
+                                serializeExportSpecifier(ts.unescapeLeadingUnderscores(symbol.escapedName), targetName);
+                            }
+                            break;
+                        case 249 /* VariableDeclaration */:
+                            // commonjs require: const x = require('y')
+                            if (ts.isPropertyAccessExpression(node.initializer)) {
+                                // const x = require('y').z
+                                var initializer = node.initializer; // require('y').z
+                                var uniqueName = ts.factory.createUniqueName(localName); // _x
+                                var specifier_2 = getSpecifierForModuleSymbol(target.parent || target, context); // 'y'
+                                // import _x = require('y');
+                                addResult(ts.factory.createImportEqualsDeclaration(
+                                /*decorators*/ undefined, 
+                                /*modifiers*/ undefined, uniqueName, ts.factory.createExternalModuleReference(ts.factory.createStringLiteral(specifier_2))), 0 /* None */);
+                                // import x = _x.z
+                                addResult(ts.factory.createImportEqualsDeclaration(
+                                /*decorators*/ undefined, 
+                                /*modifiers*/ undefined, ts.factory.createIdentifier(localName), ts.factory.createQualifiedName(uniqueName, initializer.name)), modifierFlags);
+                                break;
+                            }
+                        // else fall through and treat commonjs require just like import=
+                        case 260 /* ImportEqualsDeclaration */:
+                            // This _specifically_ only exists to handle json declarations - where we make aliases, but since
+                            // we emit no declarations for the json document, must not refer to it in the declarations
+                            if (target.escapedName === "export=" /* ExportEquals */ && ts.some(target.declarations, ts.isJsonSourceFile)) {
+                                serializeMaybeAliasAssignment(symbol);
+                                break;
+                            }
+                            // Could be a local `import localName = ns.member` or
+                            // an external `import localName = require("whatever")`
+                            var isLocalImport = !(target.flags & 512 /* ValueModule */) && !ts.isVariableDeclaration(node);
+                            addResult(ts.factory.createImportEqualsDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, ts.factory.createIdentifier(localName), isLocalImport
+                                ? symbolToName(target, context, 67108863 /* All */, /*expectsIdentifier*/ false)
+                                : ts.factory.createExternalModuleReference(ts.factory.createStringLiteral(getSpecifierForModuleSymbol(target, context)))), isLocalImport ? modifierFlags : 0 /* None */);
+                            break;
+                        case 259 /* NamespaceExportDeclaration */:
+                            // export as namespace foo
+                            // TODO: Not part of a file's local or export symbol tables
+                            // Is bound into file.symbol.globalExports instead, which we don't currently traverse
+                            addResult(ts.factory.createNamespaceExportDeclaration(ts.idText(node.name)), 0 /* None */);
+                            break;
+                        case 262 /* ImportClause */:
+                            addResult(ts.factory.createImportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, ts.factory.createIdentifier(localName), /*namedBindings*/ undefined), 
+                            // We use `target.parent || target` below as `target.parent` is unset when the target is a module which has been export assigned
+                            // And then made into a default by the `esModuleInterop` or `allowSyntheticDefaultImports` flag
+                            // In such cases, the `target` refers to the module itself already
+                            ts.factory.createStringLiteral(getSpecifierForModuleSymbol(target.parent || target, context))), 0 /* None */);
+                            break;
+                        case 263 /* NamespaceImport */:
+                            addResult(ts.factory.createImportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, /*importClause*/ undefined, ts.factory.createNamespaceImport(ts.factory.createIdentifier(localName))), ts.factory.createStringLiteral(getSpecifierForModuleSymbol(target, context))), 0 /* None */);
+                            break;
+                        case 269 /* NamespaceExport */:
+                            addResult(ts.factory.createExportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, 
+                            /*isTypeOnly*/ false, ts.factory.createNamespaceExport(ts.factory.createIdentifier(localName)), ts.factory.createStringLiteral(getSpecifierForModuleSymbol(target, context))), 0 /* None */);
+                            break;
+                        case 265 /* ImportSpecifier */:
+                            addResult(ts.factory.createImportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, ts.factory.createImportClause(
+                            /*isTypeOnly*/ false, 
+                            /*importClause*/ undefined, ts.factory.createNamedImports([
+                                ts.factory.createImportSpecifier(localName !== verbatimTargetName ? ts.factory.createIdentifier(verbatimTargetName) : undefined, ts.factory.createIdentifier(localName))
+                            ])), ts.factory.createStringLiteral(getSpecifierForModuleSymbol(target.parent || target, context))), 0 /* None */);
+                            break;
+                        case 270 /* ExportSpecifier */:
+                            // does not use localName because the symbol name in this case refers to the name in the exports table,
+                            // which we must exactly preserve
+                            var specifier = node.parent.parent.moduleSpecifier;
+                            // targetName is only used when the target is local, as otherwise the target is an alias that points at
+                            // another file
+                            serializeExportSpecifier(ts.unescapeLeadingUnderscores(symbol.escapedName), specifier ? verbatimTargetName : targetName, specifier && ts.isStringLiteralLike(specifier) ? ts.factory.createStringLiteral(specifier.text) : undefined);
+                            break;
+                        case 266 /* ExportAssignment */:
+                            serializeMaybeAliasAssignment(symbol);
+                            break;
+                        case 216 /* BinaryExpression */:
+                        case 201 /* PropertyAccessExpression */:
+                        case 202 /* ElementAccessExpression */:
+                            // Could be best encoded as though an export specifier or as though an export assignment
+                            // If name is default or export=, do an export assignment
+                            // Otherwise do an export specifier
+                            if (symbol.escapedName === "default" /* Default */ || symbol.escapedName === "export=" /* ExportEquals */) {
+                                serializeMaybeAliasAssignment(symbol);
+                            }
+                            else {
+                                serializeExportSpecifier(localName, targetName);
+                            }
+                            break;
+                        default:
+                            return ts.Debug.failBadSyntaxKind(node, "Unhandled alias declaration kind in symbol serializer!");
+                    }
                 }
-                else if (signature.flags & 8 /* IsOuterCallChain */) {
-                    type = getOptionalType(type);
+                function serializeExportSpecifier(localName, targetName, specifier) {
+                    addResult(ts.factory.createExportDeclaration(
+                    /*decorators*/ undefined, 
+                    /*modifiers*/ undefined, 
+                    /*isTypeOnly*/ false, ts.factory.createNamedExports([ts.factory.createExportSpecifier(localName !== targetName ? targetName : undefined, localName)]), specifier), 0 /* None */);
                 }
-                if (!popTypeResolution()) {
-                    if (signature.declaration) {
-                        var typeNode = ts.getEffectiveReturnTypeNode(signature.declaration);
-                        if (typeNode) {
-                            error(typeNode, ts.Diagnostics.Return_type_annotation_circularly_references_itself);
+                /**
+                 * Returns `true` if an export assignment or declaration was produced for the symbol
+                 */
+                function serializeMaybeAliasAssignment(symbol) {
+                    if (symbol.flags & 4194304 /* Prototype */) {
+                        return false;
+                    }
+                    var name = ts.unescapeLeadingUnderscores(symbol.escapedName);
+                    var isExportEquals = name === "export=" /* ExportEquals */;
+                    var isDefault = name === "default" /* Default */;
+                    var isExportAssignmentCompatibleSymbolName = isExportEquals || isDefault;
+                    // synthesize export = ref
+                    // ref should refer to either be a locally scoped symbol which we need to emit, or
+                    // a reference to another namespace/module which we may need to emit an `import` statement for
+                    var aliasDecl = symbol.declarations && getDeclarationOfAliasSymbol(symbol);
+                    // serialize what the alias points to, preserve the declaration's initializer
+                    var target = aliasDecl && getTargetOfAliasDeclaration(aliasDecl, /*dontRecursivelyResolve*/ true);
+                    // If the target resolves and resolves to a thing defined in this file, emit as an alias, otherwise emit as a const
+                    if (target && ts.length(target.declarations) && ts.some(target.declarations, function (d) { return ts.getSourceFileOfNode(d) === ts.getSourceFileOfNode(enclosingDeclaration); })) {
+                        // In case `target` refers to a namespace member, look at the declaration and serialize the leftmost symbol in it
+                        // eg, `namespace A { export class B {} }; exports = A.B;`
+                        // Technically, this is all that's required in the case where the assignment is an entity name expression
+                        var expr = aliasDecl && ((ts.isExportAssignment(aliasDecl) || ts.isBinaryExpression(aliasDecl)) ? ts.getExportAssignmentExpression(aliasDecl) : ts.getPropertyAssignmentAliasLikeExpression(aliasDecl));
+                        var first_1 = expr && ts.isEntityNameExpression(expr) ? getFirstNonModuleExportsIdentifier(expr) : undefined;
+                        var referenced = first_1 && resolveEntityName(first_1, 67108863 /* All */, /*ignoreErrors*/ true, /*dontResolveAlias*/ true, enclosingDeclaration);
+                        if (referenced || target) {
+                            includePrivateSymbol(referenced || target);
                         }
-                        else if (noImplicitAny) {
-                            var declaration = signature.declaration;
-                            var name = ts.getNameOfDeclaration(declaration);
-                            if (name) {
-                                error(name, ts.Diagnostics._0_implicitly_has_return_type_any_because_it_does_not_have_a_return_type_annotation_and_is_referenced_directly_or_indirectly_in_one_of_its_return_expressions, ts.declarationNameToString(name));
+                        // We disable the context's symbol tracker for the duration of this name serialization
+                        // as, by virtue of being here, the name is required to print something, and we don't want to
+                        // issue a visibility error on it. Only anonymous classes that an alias points at _would_ issue
+                        // a visibility error here (as they're not visible within any scope), but we want to hoist them
+                        // into the containing scope anyway, so we want to skip the visibility checks.
+                        var oldTrack = context.tracker.trackSymbol;
+                        context.tracker.trackSymbol = ts.noop;
+                        if (isExportAssignmentCompatibleSymbolName) {
+                            results.push(ts.factory.createExportAssignment(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, isExportEquals, symbolToExpression(target, context, 67108863 /* All */)));
+                        }
+                        else {
+                            if (first_1 === expr && first_1) {
+                                // serialize as `export {target as name}`
+                                serializeExportSpecifier(name, ts.idText(first_1));
+                            }
+                            else if (expr && ts.isClassExpression(expr)) {
+                                serializeExportSpecifier(name, getInternalSymbolName(target, ts.symbolName(target)));
                             }
                             else {
-                                error(declaration, ts.Diagnostics.Function_implicitly_has_return_type_any_because_it_does_not_have_a_return_type_annotation_and_is_referenced_directly_or_indirectly_in_one_of_its_return_expressions);
+                                // serialize as `import _Ref = t.arg.et; export { _Ref as name }`
+                                var varName = getUnusedName(name, symbol);
+                                addResult(ts.factory.createImportEqualsDeclaration(
+                                /*decorators*/ undefined, 
+                                /*modifiers*/ undefined, ts.factory.createIdentifier(varName), symbolToName(target, context, 67108863 /* All */, /*expectsIdentifier*/ false)), 0 /* None */);
+                                serializeExportSpecifier(name, varName);
                             }
                         }
+                        context.tracker.trackSymbol = oldTrack;
+                        return true;
+                    }
+                    else {
+                        // serialize as an anonymous property declaration
+                        var varName = getUnusedName(name, symbol);
+                        // We have to use `getWidenedType` here since the object within a json file is unwidened within the file
+                        // (Unwidened types can only exist in expression contexts and should never be serialized)
+                        var typeToSerialize = getWidenedType(getTypeOfSymbol(getMergedSymbol(symbol)));
+                        if (isTypeRepresentableAsFunctionNamespaceMerge(typeToSerialize, symbol)) {
+                            // If there are no index signatures and `typeToSerialize` is an object type, emit as a namespace instead of a const
+                            serializeAsFunctionNamespaceMerge(typeToSerialize, symbol, varName, isExportAssignmentCompatibleSymbolName ? 0 /* None */ : 1 /* Export */);
+                        }
+                        else {
+                            var statement = ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList([
+                                ts.factory.createVariableDeclaration(varName, /*exclamationToken*/ undefined, serializeTypeForDeclaration(context, typeToSerialize, symbol, enclosingDeclaration, includePrivateSymbol, bundled))
+                            ], 2 /* Const */));
+                            // Inlined JSON types exported with [module.]exports= will already emit an export=, so should use `declare`.
+                            // Otherwise, the type itself should be exported.
+                            addResult(statement, target && target.flags & 4 /* Property */ && target.escapedName === "export=" /* ExportEquals */ ? 2 /* Ambient */
+                                : name === varName ? 1 /* Export */
+                                    : 0 /* None */);
+                        }
+                        if (isExportAssignmentCompatibleSymbolName) {
+                            results.push(ts.factory.createExportAssignment(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, isExportEquals, ts.factory.createIdentifier(varName)));
+                            return true;
+                        }
+                        else if (name !== varName) {
+                            serializeExportSpecifier(name, varName);
+                            return true;
+                        }
+                        return false;
                     }
-                    type = anyType;
                 }
-                signature.resolvedReturnType = type;
-            }
-            return signature.resolvedReturnType;
-        }
-        function getReturnTypeFromAnnotation(declaration) {
-            if (declaration.kind === 162 /* Constructor */) {
-                return getDeclaredTypeOfClassOrInterface(getMergedSymbol(declaration.parent.symbol));
-            }
-            if (ts.isJSDocConstructSignature(declaration)) {
-                return getTypeFromTypeNode(declaration.parameters[0].type); // TODO: GH#18217
-            }
-            var typeNode = ts.getEffectiveReturnTypeNode(declaration);
-            if (typeNode) {
-                return getTypeFromTypeNode(typeNode);
-            }
-            if (declaration.kind === 163 /* GetAccessor */ && !hasNonBindableDynamicName(declaration)) {
-                var jsDocType = ts.isInJSFile(declaration) && getTypeForDeclarationFromJSDocComment(declaration);
-                if (jsDocType) {
-                    return jsDocType;
+                function isTypeRepresentableAsFunctionNamespaceMerge(typeToSerialize, hostSymbol) {
+                    // Only object types which are not constructable, or indexable, whose members all come from the
+                    // context source file, and whose property names are all valid identifiers and not late-bound, _and_
+                    // whose input is not type annotated (if the input symbol has an annotation we can reuse, we should prefer it)
+                    var ctxSrc = ts.getSourceFileOfNode(context.enclosingDeclaration);
+                    return ts.getObjectFlags(typeToSerialize) & (16 /* Anonymous */ | 32 /* Mapped */) &&
+                        !getIndexInfoOfType(typeToSerialize, 0 /* String */) &&
+                        !getIndexInfoOfType(typeToSerialize, 1 /* Number */) &&
+                        !isClassInstanceSide(typeToSerialize) && // While a class instance is potentially representable as a NS, prefer printing a reference to the instance type and serializing the class
+                        !!(ts.length(ts.filter(getPropertiesOfType(typeToSerialize), isNamespaceMember)) || ts.length(getSignaturesOfType(typeToSerialize, 0 /* Call */))) &&
+                        !ts.length(getSignaturesOfType(typeToSerialize, 1 /* Construct */)) && // TODO: could probably serialize as function + ns + class, now that that's OK
+                        !getDeclarationWithTypeAnnotation(hostSymbol, enclosingDeclaration) &&
+                        !(typeToSerialize.symbol && ts.some(typeToSerialize.symbol.declarations, function (d) { return ts.getSourceFileOfNode(d) !== ctxSrc; })) &&
+                        !ts.some(getPropertiesOfType(typeToSerialize), function (p) { return isLateBoundName(p.escapedName); }) &&
+                        !ts.some(getPropertiesOfType(typeToSerialize), function (p) { return ts.some(p.declarations, function (d) { return ts.getSourceFileOfNode(d) !== ctxSrc; }); }) &&
+                        ts.every(getPropertiesOfType(typeToSerialize), function (p) { return ts.isIdentifierText(ts.symbolName(p), languageVersion); });
                 }
-                var setter = ts.getDeclarationOfKind(getSymbolOfNode(declaration), 164 /* SetAccessor */);
-                var setterType = getAnnotatedAccessorType(setter);
-                if (setterType) {
-                    return setterType;
+                function makeSerializePropertySymbol(createProperty, methodKind, useAccessors) {
+                    return function serializePropertySymbol(p, isStatic, baseType) {
+                        var modifierFlags = ts.getDeclarationModifierFlagsFromSymbol(p);
+                        var isPrivate = !!(modifierFlags & 8 /* Private */);
+                        if (isStatic && (p.flags & (788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */))) {
+                            // Only value-only-meaning symbols can be correctly encoded as class statics, type/namespace/alias meaning symbols
+                            // need to be merged namespace members
+                            return [];
+                        }
+                        if (p.flags & 4194304 /* Prototype */ ||
+                            (baseType && getPropertyOfType(baseType, p.escapedName)
+                                && isReadonlySymbol(getPropertyOfType(baseType, p.escapedName)) === isReadonlySymbol(p)
+                                && (p.flags & 16777216 /* Optional */) === (getPropertyOfType(baseType, p.escapedName).flags & 16777216 /* Optional */)
+                                && isTypeIdenticalTo(getTypeOfSymbol(p), getTypeOfPropertyOfType(baseType, p.escapedName)))) {
+                            return [];
+                        }
+                        var flag = (modifierFlags & ~256 /* Async */) | (isStatic ? 32 /* Static */ : 0);
+                        var name = getPropertyNameNodeForSymbol(p, context);
+                        var firstPropertyLikeDecl = ts.find(p.declarations, ts.or(ts.isPropertyDeclaration, ts.isAccessor, ts.isVariableDeclaration, ts.isPropertySignature, ts.isBinaryExpression, ts.isPropertyAccessExpression));
+                        if (p.flags & 98304 /* Accessor */ && useAccessors) {
+                            var result = [];
+                            if (p.flags & 65536 /* SetAccessor */) {
+                                result.push(ts.setTextRange(ts.factory.createSetAccessorDeclaration(
+                                /*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags(flag), name, [ts.factory.createParameterDeclaration(
+                                    /*decorators*/ undefined, 
+                                    /*modifiers*/ undefined, 
+                                    /*dotDotDotToken*/ undefined, "arg", 
+                                    /*questionToken*/ undefined, isPrivate ? undefined : serializeTypeForDeclaration(context, getTypeOfSymbol(p), p, enclosingDeclaration, includePrivateSymbol, bundled))], 
+                                /*body*/ undefined), ts.find(p.declarations, ts.isSetAccessor) || firstPropertyLikeDecl));
+                            }
+                            if (p.flags & 32768 /* GetAccessor */) {
+                                var isPrivate_1 = modifierFlags & 8 /* Private */;
+                                result.push(ts.setTextRange(ts.factory.createGetAccessorDeclaration(
+                                /*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags(flag), name, [], isPrivate_1 ? undefined : serializeTypeForDeclaration(context, getTypeOfSymbol(p), p, enclosingDeclaration, includePrivateSymbol, bundled), 
+                                /*body*/ undefined), ts.find(p.declarations, ts.isGetAccessor) || firstPropertyLikeDecl));
+                            }
+                            return result;
+                        }
+                        // This is an else/if as accessors and properties can't merge in TS, but might in JS
+                        // If this happens, we assume the accessor takes priority, as it imposes more constraints
+                        else if (p.flags & (4 /* Property */ | 3 /* Variable */ | 98304 /* Accessor */)) {
+                            return ts.setTextRange(createProperty(
+                            /*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags((isReadonlySymbol(p) ? 64 /* Readonly */ : 0) | flag), name, p.flags & 16777216 /* Optional */ ? ts.factory.createToken(57 /* QuestionToken */) : undefined, isPrivate ? undefined : serializeTypeForDeclaration(context, getTypeOfSymbol(p), p, enclosingDeclaration, includePrivateSymbol, bundled), 
+                            // TODO: https://github.com/microsoft/TypeScript/pull/32372#discussion_r328386357
+                            // interface members can't have initializers, however class members _can_
+                            /*initializer*/ undefined), ts.find(p.declarations, ts.or(ts.isPropertyDeclaration, ts.isVariableDeclaration)) || firstPropertyLikeDecl);
+                        }
+                        if (p.flags & (8192 /* Method */ | 16 /* Function */)) {
+                            var type = getTypeOfSymbol(p);
+                            var signatures = getSignaturesOfType(type, 0 /* Call */);
+                            if (flag & 8 /* Private */) {
+                                return ts.setTextRange(createProperty(
+                                /*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags((isReadonlySymbol(p) ? 64 /* Readonly */ : 0) | flag), name, p.flags & 16777216 /* Optional */ ? ts.factory.createToken(57 /* QuestionToken */) : undefined, 
+                                /*type*/ undefined, 
+                                /*initializer*/ undefined), ts.find(p.declarations, ts.isFunctionLikeDeclaration) || signatures[0] && signatures[0].declaration || p.declarations[0]);
+                            }
+                            var results_1 = [];
+                            for (var _i = 0, signatures_3 = signatures; _i < signatures_3.length; _i++) {
+                                var sig = signatures_3[_i];
+                                // Each overload becomes a separate method declaration, in order
+                                var decl = signatureToSignatureDeclarationHelper(sig, methodKind, context, {
+                                    name: name,
+                                    questionToken: p.flags & 16777216 /* Optional */ ? ts.factory.createToken(57 /* QuestionToken */) : undefined,
+                                    modifiers: flag ? ts.factory.createModifiersFromModifierFlags(flag) : undefined
+                                });
+                                results_1.push(ts.setTextRange(decl, sig.declaration));
+                            }
+                            return results_1;
+                        }
+                        // The `Constructor`'s symbol isn't in the class's properties lists, obviously, since it's a signature on the static
+                        return ts.Debug.fail("Unhandled class member kind! " + (p.__debugFlags || p.flags));
+                    };
                 }
-            }
-            return getReturnTypeOfTypeTag(declaration);
-        }
-        function isResolvingReturnTypeOfSignature(signature) {
-            return !signature.resolvedReturnType && findResolutionCycleStartIndex(signature, 3 /* ResolvedReturnType */) >= 0;
-        }
-        function getRestTypeOfSignature(signature) {
-            return tryGetRestTypeOfSignature(signature) || anyType;
-        }
-        function tryGetRestTypeOfSignature(signature) {
-            if (signatureHasRestParameter(signature)) {
-                var sigRestType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
-                var restType = isTupleType(sigRestType) ? getRestTypeOfTupleType(sigRestType) : sigRestType;
-                return restType && getIndexTypeOfType(restType, 1 /* Number */);
-            }
-            return undefined;
-        }
-        function getSignatureInstantiation(signature, typeArguments, isJavascript, inferredTypeParameters) {
-            var instantiatedSignature = getSignatureInstantiationWithoutFillingInTypeArguments(signature, fillMissingTypeArguments(typeArguments, signature.typeParameters, getMinTypeArgumentCount(signature.typeParameters), isJavascript));
-            if (inferredTypeParameters) {
-                var returnSignature = getSingleCallOrConstructSignature(getReturnTypeOfSignature(instantiatedSignature));
-                if (returnSignature) {
-                    var newReturnSignature = cloneSignature(returnSignature);
-                    newReturnSignature.typeParameters = inferredTypeParameters;
-                    var newInstantiatedSignature = cloneSignature(instantiatedSignature);
-                    newInstantiatedSignature.resolvedReturnType = getOrCreateTypeFromSignature(newReturnSignature);
-                    return newInstantiatedSignature;
+                function serializePropertySymbolForInterface(p, baseType) {
+                    return serializePropertySymbolForInterfaceWorker(p, /*isStatic*/ false, baseType);
                 }
-            }
-            return instantiatedSignature;
-        }
-        function getSignatureInstantiationWithoutFillingInTypeArguments(signature, typeArguments) {
-            var instantiations = signature.instantiations || (signature.instantiations = ts.createMap());
-            var id = getTypeListId(typeArguments);
-            var instantiation = instantiations.get(id);
-            if (!instantiation) {
-                instantiations.set(id, instantiation = createSignatureInstantiation(signature, typeArguments));
-            }
-            return instantiation;
-        }
-        function createSignatureInstantiation(signature, typeArguments) {
-            return instantiateSignature(signature, createSignatureTypeMapper(signature, typeArguments), /*eraseTypeParameters*/ true);
-        }
-        function createSignatureTypeMapper(signature, typeArguments) {
-            return createTypeMapper(signature.typeParameters, typeArguments);
-        }
-        function getErasedSignature(signature) {
-            return signature.typeParameters ?
-                signature.erasedSignatureCache || (signature.erasedSignatureCache = createErasedSignature(signature)) :
-                signature;
-        }
-        function createErasedSignature(signature) {
-            // Create an instantiation of the signature where all type arguments are the any type.
-            return instantiateSignature(signature, createTypeEraser(signature.typeParameters), /*eraseTypeParameters*/ true);
-        }
-        function getCanonicalSignature(signature) {
-            return signature.typeParameters ?
-                signature.canonicalSignatureCache || (signature.canonicalSignatureCache = createCanonicalSignature(signature)) :
-                signature;
-        }
-        function createCanonicalSignature(signature) {
-            // Create an instantiation of the signature where each unconstrained type parameter is replaced with
-            // its original. When a generic class or interface is instantiated, each generic method in the class or
-            // interface is instantiated with a fresh set of cloned type parameters (which we need to handle scenarios
-            // where different generations of the same type parameter are in scope). This leads to a lot of new type
-            // identities, and potentially a lot of work comparing those identities, so here we create an instantiation
-            // that uses the original type identities for all unconstrained type parameters.
-            return getSignatureInstantiation(signature, ts.map(signature.typeParameters, function (tp) { return tp.target && !getConstraintOfTypeParameter(tp.target) ? tp.target : tp; }), ts.isInJSFile(signature.declaration));
-        }
-        function getBaseSignature(signature) {
-            var typeParameters = signature.typeParameters;
-            if (typeParameters) {
-                var typeEraser_1 = createTypeEraser(typeParameters);
-                var baseConstraints = ts.map(typeParameters, function (tp) { return instantiateType(getBaseConstraintOfType(tp), typeEraser_1) || unknownType; });
-                return instantiateSignature(signature, createTypeMapper(typeParameters, baseConstraints), /*eraseTypeParameters*/ true);
-            }
-            return signature;
-        }
-        function getOrCreateTypeFromSignature(signature) {
-            // There are two ways to declare a construct signature, one is by declaring a class constructor
-            // using the constructor keyword, and the other is declaring a bare construct signature in an
-            // object type literal or interface (using the new keyword). Each way of declaring a constructor
-            // will result in a different declaration kind.
-            if (!signature.isolatedSignatureType) {
-                var kind = signature.declaration ? signature.declaration.kind : 0 /* Unknown */;
-                var isConstructor = kind === 162 /* Constructor */ || kind === 166 /* ConstructSignature */ || kind === 171 /* ConstructorType */;
-                var type = createObjectType(16 /* Anonymous */);
-                type.members = emptySymbols;
-                type.properties = ts.emptyArray;
-                type.callSignatures = !isConstructor ? [signature] : ts.emptyArray;
-                type.constructSignatures = isConstructor ? [signature] : ts.emptyArray;
-                signature.isolatedSignatureType = type;
-            }
-            return signature.isolatedSignatureType;
-        }
-        function getIndexSymbol(symbol) {
-            return symbol.members.get("__index" /* Index */);
-        }
-        function getIndexDeclarationOfSymbol(symbol, kind) {
-            var syntaxKind = kind === 1 /* Number */ ? 140 /* NumberKeyword */ : 143 /* StringKeyword */;
-            var indexSymbol = getIndexSymbol(symbol);
-            if (indexSymbol) {
-                for (var _i = 0, _a = indexSymbol.declarations; _i < _a.length; _i++) {
-                    var decl = _a[_i];
-                    var node = ts.cast(decl, ts.isIndexSignatureDeclaration);
-                    if (node.parameters.length === 1) {
-                        var parameter = node.parameters[0];
-                        if (parameter.type && parameter.type.kind === syntaxKind) {
-                            return node;
+                function serializeSignatures(kind, input, baseType, outputKind) {
+                    var signatures = getSignaturesOfType(input, kind);
+                    if (kind === 1 /* Construct */) {
+                        if (!baseType && ts.every(signatures, function (s) { return ts.length(s.parameters) === 0; })) {
+                            return []; // No base type, every constructor is empty - elide the extraneous `constructor()`
+                        }
+                        if (baseType) {
+                            // If there is a base type, if every signature in the class is identical to a signature in the baseType, elide all the declarations
+                            var baseSigs = getSignaturesOfType(baseType, 1 /* Construct */);
+                            if (!ts.length(baseSigs) && ts.every(signatures, function (s) { return ts.length(s.parameters) === 0; })) {
+                                return []; // Base had no explicit signatures, if all our signatures are also implicit, return an empty list
+                            }
+                            if (baseSigs.length === signatures.length) {
+                                var failed = false;
+                                for (var i = 0; i < baseSigs.length; i++) {
+                                    if (!compareSignaturesIdentical(signatures[i], baseSigs[i], /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ true, compareTypesIdentical)) {
+                                        failed = true;
+                                        break;
+                                    }
+                                }
+                                if (!failed) {
+                                    return []; // Every signature was identical - elide constructor list as it is inherited
+                                }
+                            }
+                        }
+                        var privateProtected = 0;
+                        for (var _i = 0, signatures_4 = signatures; _i < signatures_4.length; _i++) {
+                            var s = signatures_4[_i];
+                            if (s.declaration) {
+                                privateProtected |= ts.getSelectedEffectiveModifierFlags(s.declaration, 8 /* Private */ | 16 /* Protected */);
+                            }
                         }
+                        if (privateProtected) {
+                            return [ts.setTextRange(ts.factory.createConstructorDeclaration(
+                                /*decorators*/ undefined, ts.factory.createModifiersFromModifierFlags(privateProtected), 
+                                /*parameters*/ [], 
+                                /*body*/ undefined), signatures[0].declaration)];
+                        }
+                    }
+                    var results = [];
+                    for (var _a = 0, signatures_5 = signatures; _a < signatures_5.length; _a++) {
+                        var sig = signatures_5[_a];
+                        // Each overload becomes a separate constructor declaration, in order
+                        var decl = signatureToSignatureDeclarationHelper(sig, outputKind, context);
+                        results.push(ts.setTextRange(decl, sig.declaration));
                     }
+                    return results;
                 }
-            }
-            return undefined;
-        }
-        function createIndexInfo(type, isReadonly, declaration) {
-            return { type: type, isReadonly: isReadonly, declaration: declaration };
-        }
-        function getIndexInfoOfSymbol(symbol, kind) {
-            var declaration = getIndexDeclarationOfSymbol(symbol, kind);
-            if (declaration) {
-                return createIndexInfo(declaration.type ? getTypeFromTypeNode(declaration.type) : anyType, ts.hasModifier(declaration, 64 /* Readonly */), declaration);
-            }
-            return undefined;
-        }
-        function getConstraintDeclaration(type) {
-            return ts.mapDefined(ts.filter(type.symbol && type.symbol.declarations, ts.isTypeParameterDeclaration), ts.getEffectiveConstraintOfTypeParameter)[0];
-        }
-        function getInferredTypeParameterConstraint(typeParameter) {
-            var inferences;
-            if (typeParameter.symbol) {
-                for (var _i = 0, _a = typeParameter.symbol.declarations; _i < _a.length; _i++) {
-                    var declaration = _a[_i];
-                    if (declaration.parent.kind === 181 /* InferType */) {
-                        // When an 'infer T' declaration is immediately contained in a type reference node
-                        // (such as 'Foo<infer T>'), T's constraint is inferred from the constraint of the
-                        // corresponding type parameter in 'Foo'. When multiple 'infer T' declarations are
-                        // present, we form an intersection of the inferred constraint types.
-                        var grandParent = declaration.parent.parent;
-                        if (grandParent.kind === 169 /* TypeReference */) {
-                            var typeReference = grandParent;
-                            var typeParameters = getTypeParametersForTypeReference(typeReference);
-                            if (typeParameters) {
-                                var index = typeReference.typeArguments.indexOf(declaration.parent);
-                                if (index < typeParameters.length) {
-                                    var declaredConstraint = getConstraintOfTypeParameter(typeParameters[index]);
-                                    if (declaredConstraint) {
-                                        // Type parameter constraints can reference other type parameters so
-                                        // constraints need to be instantiated. If instantiation produces the
-                                        // type parameter itself, we discard that inference. For example, in
-                                        //   type Foo<T extends string, U extends T> = [T, U];
-                                        //   type Bar<T> = T extends Foo<infer X, infer X> ? Foo<X, X> : T;
-                                        // the instantiated constraint for U is X, so we discard that inference.
-                                        var mapper = createTypeMapper(typeParameters, getEffectiveTypeArguments(typeReference, typeParameters));
-                                        var constraint = instantiateType(declaredConstraint, mapper);
-                                        if (constraint !== typeParameter) {
-                                            inferences = ts.append(inferences, constraint);
-                                        }
+                function serializeIndexSignatures(input, baseType) {
+                    var results = [];
+                    for (var _i = 0, _a = [0 /* String */, 1 /* Number */]; _i < _a.length; _i++) {
+                        var type = _a[_i];
+                        var info = getIndexInfoOfType(input, type);
+                        if (info) {
+                            if (baseType) {
+                                var baseInfo = getIndexInfoOfType(baseType, type);
+                                if (baseInfo) {
+                                    if (isTypeIdenticalTo(info.type, baseInfo.type)) {
+                                        continue; // elide identical index signatures
                                     }
                                 }
                             }
-                        }
-                        // When an 'infer T' declaration is immediately contained in a rest parameter
-                        // declaration, we infer an 'unknown[]' constraint.
-                        else if (grandParent.kind === 156 /* Parameter */ && grandParent.dotDotDotToken) {
-                            inferences = ts.append(inferences, createArrayType(unknownType));
+                            results.push(indexInfoToIndexSignatureDeclarationHelper(info, type, context, /*typeNode*/ undefined));
                         }
                     }
+                    return results;
                 }
-            }
-            return inferences && getIntersectionType(inferences);
-        }
-        /** This is a worker function. Use getConstraintOfTypeParameter which guards against circular constraints. */
-        function getConstraintFromTypeParameter(typeParameter) {
-            if (!typeParameter.constraint) {
-                if (typeParameter.target) {
-                    var targetConstraint = getConstraintOfTypeParameter(typeParameter.target);
-                    typeParameter.constraint = targetConstraint ? instantiateType(targetConstraint, typeParameter.mapper) : noConstraintType;
+                function serializeBaseType(t, staticType, rootName) {
+                    var ref = trySerializeAsTypeReference(t, 111551 /* Value */);
+                    if (ref) {
+                        return ref;
+                    }
+                    var tempName = getUnusedName(rootName + "_base");
+                    var statement = ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList([
+                        ts.factory.createVariableDeclaration(tempName, /*exclamationToken*/ undefined, typeToTypeNodeHelper(staticType, context))
+                    ], 2 /* Const */));
+                    addResult(statement, 0 /* None */);
+                    return ts.factory.createExpressionWithTypeArguments(ts.factory.createIdentifier(tempName), /*typeArgs*/ undefined);
                 }
-                else {
-                    var constraintDeclaration = getConstraintDeclaration(typeParameter);
-                    if (!constraintDeclaration) {
-                        typeParameter.constraint = getInferredTypeParameterConstraint(typeParameter) || noConstraintType;
+                function trySerializeAsTypeReference(t, flags) {
+                    var typeArgs;
+                    var reference;
+                    // We don't use `isValueSymbolAccessible` below. since that considers alternative containers (like modules)
+                    // which we can't write out in a syntactically valid way as an expression
+                    if (t.target && isSymbolAccessibleByFlags(t.target.symbol, enclosingDeclaration, flags)) {
+                        typeArgs = ts.map(getTypeArguments(t), function (t) { return typeToTypeNodeHelper(t, context); });
+                        reference = symbolToExpression(t.target.symbol, context, 788968 /* Type */);
                     }
-                    else {
-                        var type = getTypeFromTypeNode(constraintDeclaration);
-                        if (type.flags & 1 /* Any */ && type !== errorType) { // Allow errorType to propegate to keep downstream errors suppressed
-                            // use keyofConstraintType as the base constraint for mapped type key constraints (unknown isn;t assignable to that, but `any` was),
-                            // use unknown otherwise
-                            type = constraintDeclaration.parent.parent.kind === 186 /* MappedType */ ? keyofConstraintType : unknownType;
+                    else if (t.symbol && isSymbolAccessibleByFlags(t.symbol, enclosingDeclaration, flags)) {
+                        reference = symbolToExpression(t.symbol, context, 788968 /* Type */);
+                    }
+                    if (reference) {
+                        return ts.factory.createExpressionWithTypeArguments(reference, typeArgs);
+                    }
+                }
+                function serializeImplementedType(t) {
+                    var ref = trySerializeAsTypeReference(t, 788968 /* Type */);
+                    if (ref) {
+                        return ref;
+                    }
+                    if (t.symbol) {
+                        return ts.factory.createExpressionWithTypeArguments(symbolToExpression(t.symbol, context, 788968 /* Type */), /*typeArgs*/ undefined);
+                    }
+                }
+                function getUnusedName(input, symbol) {
+                    var _a, _b;
+                    var id = symbol ? getSymbolId(symbol) : undefined;
+                    if (id) {
+                        if (context.remappedSymbolNames.has(id)) {
+                            return context.remappedSymbolNames.get(id);
                         }
-                        typeParameter.constraint = type;
                     }
+                    if (symbol) {
+                        input = getNameCandidateWorker(symbol, input);
+                    }
+                    var i = 0;
+                    var original = input;
+                    while ((_a = context.usedSymbolNames) === null || _a === void 0 ? void 0 : _a.has(input)) {
+                        i++;
+                        input = original + "_" + i;
+                    }
+                    (_b = context.usedSymbolNames) === null || _b === void 0 ? void 0 : _b.add(input);
+                    if (id) {
+                        context.remappedSymbolNames.set(id, input);
+                    }
+                    return input;
                 }
-            }
-            return typeParameter.constraint === noConstraintType ? undefined : typeParameter.constraint;
-        }
-        function getParentSymbolOfTypeParameter(typeParameter) {
-            var tp = ts.getDeclarationOfKind(typeParameter.symbol, 155 /* TypeParameter */);
-            var host = ts.isJSDocTemplateTag(tp.parent) ? ts.getHostSignatureFromJSDoc(tp.parent) : tp.parent;
-            return host && getSymbolOfNode(host);
-        }
-        function getTypeListId(types) {
-            var result = "";
-            if (types) {
-                var length_4 = types.length;
-                var i = 0;
-                while (i < length_4) {
-                    var startId = types[i].id;
-                    var count = 1;
-                    while (i + count < length_4 && types[i + count].id === startId + count) {
-                        count++;
+                function getNameCandidateWorker(symbol, localName) {
+                    if (localName === "default" /* Default */ || localName === "__class" /* Class */ || localName === "__function" /* Function */) {
+                        var flags = context.flags;
+                        context.flags |= 16777216 /* InInitialEntityName */;
+                        var nameCandidate = getNameOfSymbolAsWritten(symbol, context);
+                        context.flags = flags;
+                        localName = nameCandidate.length > 0 && ts.isSingleOrDoubleQuote(nameCandidate.charCodeAt(0)) ? ts.stripQuotes(nameCandidate) : nameCandidate;
                     }
-                    if (result.length) {
-                        result += ",";
+                    if (localName === "default" /* Default */) {
+                        localName = "_default";
                     }
-                    result += startId;
-                    if (count > 1) {
-                        result += ":" + count;
+                    else if (localName === "export=" /* ExportEquals */) {
+                        localName = "_exports";
                     }
-                    i += count;
+                    localName = ts.isIdentifierText(localName, languageVersion) && !ts.isStringANonContextualKeyword(localName) ? localName : "_" + localName.replace(/[^a-zA-Z0-9]/g, "_");
+                    return localName;
                 }
-            }
-            return result;
-        }
-        // This function is used to propagate certain flags when creating new object type references and union types.
-        // It is only necessary to do so if a constituent type might be the undefined type, the null type, the type
-        // of an object literal or the anyFunctionType. This is because there are operations in the type checker
-        // that care about the presence of such types at arbitrary depth in a containing type.
-        function getPropagatingFlagsOfTypes(types, excludeKinds) {
-            var result = 0;
-            for (var _i = 0, types_8 = types; _i < types_8.length; _i++) {
-                var type = types_8[_i];
-                if (!(type.flags & excludeKinds)) {
-                    result |= ts.getObjectFlags(type);
+                function getInternalSymbolName(symbol, localName) {
+                    var id = getSymbolId(symbol);
+                    if (context.remappedSymbolNames.has(id)) {
+                        return context.remappedSymbolNames.get(id);
+                    }
+                    localName = getNameCandidateWorker(symbol, localName);
+                    // The result of this is going to be used as the symbol's name - lock it in, so `getUnusedName` will also pick it up
+                    context.remappedSymbolNames.set(id, localName);
+                    return localName;
                 }
             }
-            return result & 3670016 /* PropagatingFlags */;
-        }
-        function createTypeReference(target, typeArguments) {
-            var id = getTypeListId(typeArguments);
-            var type = target.instantiations.get(id);
-            if (!type) {
-                type = createObjectType(4 /* Reference */, target.symbol);
-                target.instantiations.set(id, type);
-                type.objectFlags |= typeArguments ? getPropagatingFlagsOfTypes(typeArguments, /*excludeKinds*/ 0) : 0;
-                type.target = target;
-                type.resolvedTypeArguments = typeArguments;
-            }
-            return type;
-        }
-        function cloneTypeReference(source) {
-            var type = createType(source.flags);
-            type.symbol = source.symbol;
-            type.objectFlags = source.objectFlags;
-            type.target = source.target;
-            type.resolvedTypeArguments = source.resolvedTypeArguments;
-            return type;
-        }
-        function createDeferredTypeReference(target, node, mapper) {
-            var aliasSymbol = getAliasSymbolForTypeNode(node);
-            var aliasTypeArguments = getTypeArgumentsForAliasSymbol(aliasSymbol);
-            var type = createObjectType(4 /* Reference */, target.symbol);
-            type.target = target;
-            type.node = node;
-            type.mapper = mapper;
-            type.aliasSymbol = aliasSymbol;
-            type.aliasTypeArguments = mapper ? instantiateTypes(aliasTypeArguments, mapper) : aliasTypeArguments;
-            return type;
         }
-        function getTypeArguments(type) {
-            var _a, _b;
-            if (!type.resolvedTypeArguments) {
-                if (!pushTypeResolution(type, 6 /* ResolvedTypeArguments */)) {
-                    return ((_a = type.target.localTypeParameters) === null || _a === void 0 ? void 0 : _a.map(function () { return errorType; })) || ts.emptyArray;
-                }
-                var node = type.node;
-                var typeArguments = !node ? ts.emptyArray :
-                    node.kind === 169 /* TypeReference */ ? ts.concatenate(type.target.outerTypeParameters, getEffectiveTypeArguments(node, type.target.localTypeParameters)) :
-                        node.kind === 174 /* ArrayType */ ? [getTypeFromTypeNode(node.elementType)] :
-                            ts.map(node.elementTypes, getTypeFromTypeNode);
-                if (popTypeResolution()) {
-                    type.resolvedTypeArguments = type.mapper ? instantiateTypes(typeArguments, type.mapper) : typeArguments;
-                }
-                else {
-                    type.resolvedTypeArguments = ((_b = type.target.localTypeParameters) === null || _b === void 0 ? void 0 : _b.map(function () { return errorType; })) || ts.emptyArray;
-                    error(type.node || currentNode, type.target.symbol
-                        ? ts.Diagnostics.Type_arguments_for_0_circularly_reference_themselves
-                        : ts.Diagnostics.Tuple_type_arguments_circularly_reference_themselves, type.target.symbol && symbolToString(type.target.symbol));
-                }
+        function typePredicateToString(typePredicate, enclosingDeclaration, flags, writer) {
+            if (flags === void 0) { flags = 16384 /* UseAliasDefinedOutsideCurrentScope */; }
+            return writer ? typePredicateToStringWorker(writer).getText() : ts.usingSingleLineStringWriter(typePredicateToStringWorker);
+            function typePredicateToStringWorker(writer) {
+                var predicate = ts.factory.createTypePredicateNode(typePredicate.kind === 2 /* AssertsThis */ || typePredicate.kind === 3 /* AssertsIdentifier */ ? ts.factory.createToken(127 /* AssertsKeyword */) : undefined, typePredicate.kind === 1 /* Identifier */ || typePredicate.kind === 3 /* AssertsIdentifier */ ? ts.factory.createIdentifier(typePredicate.parameterName) : ts.factory.createThisTypeNode(), typePredicate.type && nodeBuilder.typeToTypeNode(typePredicate.type, enclosingDeclaration, toNodeBuilderFlags(flags) | 70221824 /* IgnoreErrors */ | 512 /* WriteTypeParametersInQualifiedName */) // TODO: GH#18217
+                );
+                var printer = ts.createPrinter({ removeComments: true });
+                var sourceFile = enclosingDeclaration && ts.getSourceFileOfNode(enclosingDeclaration);
+                printer.writeNode(4 /* Unspecified */, predicate, /*sourceFile*/ sourceFile, writer);
+                return writer;
             }
-            return type.resolvedTypeArguments;
-        }
-        function getTypeReferenceArity(type) {
-            return ts.length(type.target.typeParameters);
         }
-        /**
-         * Get type from type-reference that reference to class or interface
-         */
-        function getTypeFromClassOrInterfaceReference(node, symbol) {
-            var type = getDeclaredTypeOfSymbol(getMergedSymbol(symbol));
-            var typeParameters = type.localTypeParameters;
-            if (typeParameters) {
-                var numTypeArguments = ts.length(node.typeArguments);
-                var minTypeArgumentCount = getMinTypeArgumentCount(typeParameters);
-                var isJs = ts.isInJSFile(node);
-                var isJsImplicitAny = !noImplicitAny && isJs;
-                if (!isJsImplicitAny && (numTypeArguments < minTypeArgumentCount || numTypeArguments > typeParameters.length)) {
-                    var missingAugmentsTag = isJs && ts.isExpressionWithTypeArguments(node) && !ts.isJSDocAugmentsTag(node.parent);
-                    var diag = minTypeArgumentCount === typeParameters.length ?
-                        missingAugmentsTag ?
-                            ts.Diagnostics.Expected_0_type_arguments_provide_these_with_an_extends_tag :
-                            ts.Diagnostics.Generic_type_0_requires_1_type_argument_s :
-                        missingAugmentsTag ?
-                            ts.Diagnostics.Expected_0_1_type_arguments_provide_these_with_an_extends_tag :
-                            ts.Diagnostics.Generic_type_0_requires_between_1_and_2_type_arguments;
-                    var typeStr = typeToString(type, /*enclosingDeclaration*/ undefined, 2 /* WriteArrayAsGenericType */);
-                    error(node, diag, typeStr, minTypeArgumentCount, typeParameters.length);
-                    if (!isJs) {
-                        // TODO: Adopt same permissive behavior in TS as in JS to reduce follow-on editing experience failures (requires editing fillMissingTypeArguments)
-                        return errorType;
+        function formatUnionTypes(types) {
+            var result = [];
+            var flags = 0;
+            for (var i = 0; i < types.length; i++) {
+                var t = types[i];
+                flags |= t.flags;
+                if (!(t.flags & 98304 /* Nullable */)) {
+                    if (t.flags & (512 /* BooleanLiteral */ | 1024 /* EnumLiteral */)) {
+                        var baseType = t.flags & 512 /* BooleanLiteral */ ? booleanType : getBaseTypeOfEnumLiteralType(t);
+                        if (baseType.flags & 1048576 /* Union */) {
+                            var count = baseType.types.length;
+                            if (i + count <= types.length && getRegularTypeOfLiteralType(types[i + count - 1]) === getRegularTypeOfLiteralType(baseType.types[count - 1])) {
+                                result.push(baseType);
+                                i += count - 1;
+                                continue;
+                            }
+                        }
                     }
+                    result.push(t);
                 }
-                if (node.kind === 169 /* TypeReference */ && isDeferredTypeReferenceNode(node, ts.length(node.typeArguments) !== typeParameters.length)) {
-                    return createDeferredTypeReference(type, node, /*mapper*/ undefined);
-                }
-                // In a type reference, the outer type parameters of the referenced class or interface are automatically
-                // supplied as type arguments and the type reference only specifies arguments for the local type parameters
-                // of the class or interface.
-                var typeArguments = ts.concatenate(type.outerTypeParameters, fillMissingTypeArguments(typeArgumentsFromTypeReferenceNode(node), typeParameters, minTypeArgumentCount, isJs));
-                return createTypeReference(type, typeArguments);
             }
-            return checkNoTypeArguments(node, symbol) ? type : errorType;
+            if (flags & 65536 /* Null */)
+                result.push(nullType);
+            if (flags & 32768 /* Undefined */)
+                result.push(undefinedType);
+            return result || types;
         }
-        function getTypeAliasInstantiation(symbol, typeArguments) {
-            var type = getDeclaredTypeOfSymbol(symbol);
-            var links = getSymbolLinks(symbol);
-            var typeParameters = links.typeParameters;
-            var id = getTypeListId(typeArguments);
-            var instantiation = links.instantiations.get(id);
-            if (!instantiation) {
-                links.instantiations.set(id, instantiation = instantiateType(type, createTypeMapper(typeParameters, fillMissingTypeArguments(typeArguments, typeParameters, getMinTypeArgumentCount(typeParameters), ts.isInJSFile(symbol.valueDeclaration)))));
+        function visibilityToString(flags) {
+            if (flags === 8 /* Private */) {
+                return "private";
             }
-            return instantiation;
-        }
-        /**
-         * Get type from reference to type alias. When a type alias is generic, the declared type of the type alias may include
-         * references to the type parameters of the alias. We replace those with the actual type arguments by instantiating the
-         * declared type. Instantiations are cached using the type identities of the type arguments as the key.
-         */
-        function getTypeFromTypeAliasReference(node, symbol) {
-            var type = getDeclaredTypeOfSymbol(symbol);
-            var typeParameters = getSymbolLinks(symbol).typeParameters;
-            if (typeParameters) {
-                var numTypeArguments = ts.length(node.typeArguments);
-                var minTypeArgumentCount = getMinTypeArgumentCount(typeParameters);
-                if (numTypeArguments < minTypeArgumentCount || numTypeArguments > typeParameters.length) {
-                    error(node, minTypeArgumentCount === typeParameters.length ?
-                        ts.Diagnostics.Generic_type_0_requires_1_type_argument_s :
-                        ts.Diagnostics.Generic_type_0_requires_between_1_and_2_type_arguments, symbolToString(symbol), minTypeArgumentCount, typeParameters.length);
-                    return errorType;
-                }
-                return getTypeAliasInstantiation(symbol, typeArgumentsFromTypeReferenceNode(node));
+            if (flags === 16 /* Protected */) {
+                return "protected";
             }
-            return checkNoTypeArguments(node, symbol) ? type : errorType;
+            return "public";
         }
-        function getTypeReferenceName(node) {
-            switch (node.kind) {
-                case 169 /* TypeReference */:
-                    return node.typeName;
-                case 216 /* ExpressionWithTypeArguments */:
-                    // We only support expressions that are simple qualified names. For other
-                    // expressions this produces undefined.
-                    var expr = node.expression;
-                    if (ts.isEntityNameExpression(expr)) {
-                        return expr;
-                    }
-                // fall through;
+        function getTypeAliasForTypeLiteral(type) {
+            if (type.symbol && type.symbol.flags & 2048 /* TypeLiteral */) {
+                var node = ts.walkUpParenthesizedTypes(type.symbol.declarations[0].parent);
+                if (node.kind === 254 /* TypeAliasDeclaration */) {
+                    return getSymbolOfNode(node);
+                }
             }
             return undefined;
         }
-        function resolveTypeReferenceName(typeReferenceName, meaning, ignoreErrors) {
-            if (!typeReferenceName) {
-                return unknownSymbol;
-            }
-            return resolveEntityName(typeReferenceName, meaning, ignoreErrors) || unknownSymbol;
+        function isTopLevelInExternalModuleAugmentation(node) {
+            return node && node.parent &&
+                node.parent.kind === 257 /* ModuleBlock */ &&
+                ts.isExternalModuleAugmentation(node.parent.parent);
         }
-        function getTypeReferenceType(node, symbol) {
-            if (symbol === unknownSymbol) {
-                return errorType;
-            }
-            symbol = getExpandoSymbol(symbol) || symbol;
-            if (symbol.flags & (32 /* Class */ | 64 /* Interface */)) {
-                return getTypeFromClassOrInterfaceReference(node, symbol);
-            }
-            if (symbol.flags & 524288 /* TypeAlias */) {
-                return getTypeFromTypeAliasReference(node, symbol);
-            }
-            // Get type from reference to named type that cannot be generic (enum or type parameter)
-            var res = tryGetDeclaredTypeOfSymbol(symbol);
-            if (res) {
-                return checkNoTypeArguments(node, symbol) ? getRegularTypeOfLiteralType(res) : errorType;
-            }
-            if (symbol.flags & 111551 /* Value */ && isJSDocTypeReference(node)) {
-                var jsdocType = getTypeFromJSDocValueReference(node, symbol);
-                if (jsdocType) {
-                    return jsdocType;
+        function isDefaultBindingContext(location) {
+            return location.kind === 297 /* SourceFile */ || ts.isAmbientModule(location);
+        }
+        function getNameOfSymbolFromNameType(symbol, context) {
+            var nameType = getSymbolLinks(symbol).nameType;
+            if (nameType) {
+                if (nameType.flags & 384 /* StringOrNumberLiteral */) {
+                    var name = "" + nameType.value;
+                    if (!ts.isIdentifierText(name, compilerOptions.target) && !isNumericLiteralName(name)) {
+                        return "\"" + ts.escapeString(name, 34 /* doubleQuote */) + "\"";
+                    }
+                    if (isNumericLiteralName(name) && ts.startsWith(name, "-")) {
+                        return "[" + name + "]";
+                    }
+                    return name;
                 }
-                else {
-                    // Resolve the type reference as a Type for the purpose of reporting errors.
-                    resolveTypeReferenceName(getTypeReferenceName(node), 788968 /* Type */);
-                    return getTypeOfSymbol(symbol);
+                if (nameType.flags & 8192 /* UniqueESSymbol */) {
+                    return "[" + getNameOfSymbolAsWritten(nameType.symbol, context) + "]";
                 }
             }
-            return errorType;
         }
         /**
-         * A JSdoc TypeReference may be to a value, but resolve it as a type anyway.
-         * Note: If the value is imported from commonjs, it should really be an alias,
-         * but this function's special-case code fakes alias resolution as well.
+         * Gets a human-readable name for a symbol.
+         * Should *not* be used for the right-hand side of a `.` -- use `symbolName(symbol)` for that instead.
+         *
+         * Unlike `symbolName(symbol)`, this will include quotes if the name is from a string literal.
+         * It will also use a representation of a number as written instead of a decimal form, e.g. `0o11` instead of `9`.
          */
-        function getTypeFromJSDocValueReference(node, symbol) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedJSDocType) {
-                var valueType = getTypeOfSymbol(symbol);
-                var typeType = valueType;
-                if (symbol.valueDeclaration) {
-                    var decl = ts.getRootDeclaration(symbol.valueDeclaration);
-                    var isRequireAlias = false;
-                    if (ts.isVariableDeclaration(decl) && decl.initializer) {
-                        var expr = decl.initializer;
-                        // skip past entity names, eg `require("x").a.b.c`
-                        while (ts.isPropertyAccessExpression(expr)) {
-                            expr = expr.expression;
-                        }
-                        isRequireAlias = ts.isCallExpression(expr) && ts.isRequireCall(expr, /*requireStringLiteralLikeArgument*/ true) && !!valueType.symbol;
+        function getNameOfSymbolAsWritten(symbol, context) {
+            if (context && symbol.escapedName === "default" /* Default */ && !(context.flags & 16384 /* UseAliasDefinedOutsideCurrentScope */) &&
+                // If it's not the first part of an entity name, it must print as `default`
+                (!(context.flags & 16777216 /* InInitialEntityName */) ||
+                    // if the symbol is synthesized, it will only be referenced externally it must print as `default`
+                    !symbol.declarations ||
+                    // if not in the same binding context (source file, module declaration), it must print as `default`
+                    (context.enclosingDeclaration && ts.findAncestor(symbol.declarations[0], isDefaultBindingContext) !== ts.findAncestor(context.enclosingDeclaration, isDefaultBindingContext)))) {
+                return "default";
+            }
+            if (symbol.declarations && symbol.declarations.length) {
+                var declaration = ts.firstDefined(symbol.declarations, function (d) { return ts.getNameOfDeclaration(d) ? d : undefined; }); // Try using a declaration with a name, first
+                var name_3 = declaration && ts.getNameOfDeclaration(declaration);
+                if (declaration && name_3) {
+                    if (ts.isCallExpression(declaration) && ts.isBindableObjectDefinePropertyCall(declaration)) {
+                        return ts.symbolName(symbol);
                     }
-                    var isImportTypeWithQualifier = node.kind === 188 /* ImportType */ && node.qualifier;
-                    // valueType might not have a symbol, eg, {import('./b').STRING_LITERAL}
-                    if (valueType.symbol && (isRequireAlias || isImportTypeWithQualifier)) {
-                        typeType = getTypeReferenceType(node, valueType.symbol);
+                    if (ts.isComputedPropertyName(name_3) && !(ts.getCheckFlags(symbol) & 4096 /* Late */)) {
+                        var nameType = getSymbolLinks(symbol).nameType;
+                        if (nameType && nameType.flags & 384 /* StringOrNumberLiteral */) {
+                            // Computed property name isn't late bound, but has a well-known name type - use name type to generate a symbol name
+                            var result = getNameOfSymbolFromNameType(symbol, context);
+                            if (result !== undefined) {
+                                return result;
+                            }
+                        }
                     }
+                    return ts.declarationNameToString(name_3);
                 }
-                links.resolvedJSDocType = typeType;
-            }
-            return links.resolvedJSDocType;
-        }
-        function getSubstitutionType(baseType, substitute) {
-            if (substitute.flags & 3 /* AnyOrUnknown */ || substitute === baseType) {
-                return baseType;
-            }
-            var id = getTypeId(baseType) + ">" + getTypeId(substitute);
-            var cached = substitutionTypes.get(id);
-            if (cached) {
-                return cached;
-            }
-            var result = createType(33554432 /* Substitution */);
-            result.baseType = baseType;
-            result.substitute = substitute;
-            substitutionTypes.set(id, result);
-            return result;
-        }
-        function isUnaryTupleTypeNode(node) {
-            return node.kind === 175 /* TupleType */ && node.elementTypes.length === 1;
-        }
-        function getImpliedConstraint(type, checkNode, extendsNode) {
-            return isUnaryTupleTypeNode(checkNode) && isUnaryTupleTypeNode(extendsNode) ? getImpliedConstraint(type, checkNode.elementTypes[0], extendsNode.elementTypes[0]) :
-                getActualTypeVariable(getTypeFromTypeNode(checkNode)) === type ? getTypeFromTypeNode(extendsNode) :
-                    undefined;
-        }
-        function getConditionalFlowTypeOfType(type, node) {
-            var constraints;
-            while (node && !ts.isStatement(node) && node.kind !== 303 /* JSDocComment */) {
-                var parent = node.parent;
-                if (parent.kind === 180 /* ConditionalType */ && node === parent.trueType) {
-                    var constraint = getImpliedConstraint(type, parent.checkType, parent.extendsType);
-                    if (constraint) {
-                        constraints = ts.append(constraints, constraint);
-                    }
+                if (!declaration) {
+                    declaration = symbol.declarations[0]; // Declaration may be nameless, but we'll try anyway
                 }
-                node = parent;
-            }
-            return constraints ? getSubstitutionType(type, getIntersectionType(ts.append(constraints, type))) : type;
-        }
-        function isJSDocTypeReference(node) {
-            return !!(node.flags & 4194304 /* JSDoc */) && (node.kind === 169 /* TypeReference */ || node.kind === 188 /* ImportType */);
-        }
-        function checkNoTypeArguments(node, symbol) {
-            if (node.typeArguments) {
-                error(node, ts.Diagnostics.Type_0_is_not_generic, symbol ? symbolToString(symbol) : node.typeName ? ts.declarationNameToString(node.typeName) : anon);
-                return false;
-            }
-            return true;
-        }
-        function getIntendedTypeFromJSDocTypeReference(node) {
-            if (ts.isIdentifier(node.typeName)) {
-                var typeArgs = node.typeArguments;
-                switch (node.typeName.escapedText) {
-                    case "String":
-                        checkNoTypeArguments(node);
-                        return stringType;
-                    case "Number":
-                        checkNoTypeArguments(node);
-                        return numberType;
-                    case "Boolean":
-                        checkNoTypeArguments(node);
-                        return booleanType;
-                    case "Void":
-                        checkNoTypeArguments(node);
-                        return voidType;
-                    case "Undefined":
-                        checkNoTypeArguments(node);
-                        return undefinedType;
-                    case "Null":
-                        checkNoTypeArguments(node);
-                        return nullType;
-                    case "Function":
-                    case "function":
-                        checkNoTypeArguments(node);
-                        return globalFunctionType;
-                    case "array":
-                        return (!typeArgs || !typeArgs.length) && !noImplicitAny ? anyArrayType : undefined;
-                    case "promise":
-                        return (!typeArgs || !typeArgs.length) && !noImplicitAny ? createPromiseType(anyType) : undefined;
-                    case "Object":
-                        if (typeArgs && typeArgs.length === 2) {
-                            if (ts.isJSDocIndexSignature(node)) {
-                                var indexed = getTypeFromTypeNode(typeArgs[0]);
-                                var target = getTypeFromTypeNode(typeArgs[1]);
-                                var index = createIndexInfo(target, /*isReadonly*/ false);
-                                return createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, indexed === stringType ? index : undefined, indexed === numberType ? index : undefined);
-                            }
-                            return anyType;
+                if (declaration.parent && declaration.parent.kind === 249 /* VariableDeclaration */) {
+                    return ts.declarationNameToString(declaration.parent.name);
+                }
+                switch (declaration.kind) {
+                    case 221 /* ClassExpression */:
+                    case 208 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
+                        if (context && !context.encounteredError && !(context.flags & 131072 /* AllowAnonymousIdentifier */)) {
+                            context.encounteredError = true;
                         }
-                        checkNoTypeArguments(node);
-                        return !noImplicitAny ? anyType : undefined;
+                        return declaration.kind === 221 /* ClassExpression */ ? "(Anonymous class)" : "(Anonymous function)";
                 }
             }
+            var name = getNameOfSymbolFromNameType(symbol, context);
+            return name !== undefined ? name : ts.symbolName(symbol);
         }
-        function getTypeFromJSDocNullableTypeNode(node) {
-            var type = getTypeFromTypeNode(node.type);
-            return strictNullChecks ? getNullableType(type, 65536 /* Null */) : type;
-        }
-        function getTypeFromTypeReference(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                // handle LS queries on the `const` in `x as const` by resolving to the type of `x`
-                if (ts.isConstTypeReference(node) && ts.isAssertionExpression(node.parent)) {
-                    links.resolvedSymbol = unknownSymbol;
-                    return links.resolvedType = checkExpressionCached(node.parent.expression);
+        function isDeclarationVisible(node) {
+            if (node) {
+                var links = getNodeLinks(node);
+                if (links.isVisible === undefined) {
+                    links.isVisible = !!determineIfDeclarationIsVisible();
                 }
-                var symbol = void 0;
-                var type = void 0;
-                var meaning = 788968 /* Type */;
-                if (isJSDocTypeReference(node)) {
-                    type = getIntendedTypeFromJSDocTypeReference(node);
-                    if (!type) {
-                        symbol = resolveTypeReferenceName(getTypeReferenceName(node), meaning, /*ignoreErrors*/ true);
-                        if (symbol === unknownSymbol) {
-                            symbol = resolveTypeReferenceName(getTypeReferenceName(node), meaning | 111551 /* Value */);
+                return links.isVisible;
+            }
+            return false;
+            function determineIfDeclarationIsVisible() {
+                switch (node.kind) {
+                    case 324 /* JSDocCallbackTag */:
+                    case 331 /* JSDocTypedefTag */:
+                    case 325 /* JSDocEnumTag */:
+                        // Top-level jsdoc type aliases are considered exported
+                        // First parent is comment node, second is hosting declaration or token; we only care about those tokens or declarations whose parent is a source file
+                        return !!(node.parent && node.parent.parent && node.parent.parent.parent && ts.isSourceFile(node.parent.parent.parent));
+                    case 198 /* BindingElement */:
+                        return isDeclarationVisible(node.parent.parent);
+                    case 249 /* VariableDeclaration */:
+                        if (ts.isBindingPattern(node.name) &&
+                            !node.name.elements.length) {
+                            // If the binding pattern is empty, this variable declaration is not visible
+                            return false;
                         }
-                        else {
-                            resolveTypeReferenceName(getTypeReferenceName(node), meaning); // Resolve again to mark errors, if any
+                    // falls through
+                    case 256 /* ModuleDeclaration */:
+                    case 252 /* ClassDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 251 /* FunctionDeclaration */:
+                    case 255 /* EnumDeclaration */:
+                    case 260 /* ImportEqualsDeclaration */:
+                        // external module augmentation is always visible
+                        if (ts.isExternalModuleAugmentation(node)) {
+                            return true;
                         }
-                        type = getTypeReferenceType(node, symbol);
-                    }
-                }
-                if (!type) {
-                    symbol = resolveTypeReferenceName(getTypeReferenceName(node), meaning);
-                    type = getTypeReferenceType(node, symbol);
+                        var parent = getDeclarationContainer(node);
+                        // If the node is not exported or it is not ambient module element (except import declaration)
+                        if (!(ts.getCombinedModifierFlags(node) & 1 /* Export */) &&
+                            !(node.kind !== 260 /* ImportEqualsDeclaration */ && parent.kind !== 297 /* SourceFile */ && parent.flags & 8388608 /* Ambient */)) {
+                            return isGlobalSourceFile(parent);
+                        }
+                        // Exported members/ambient module elements (exception import declaration) are visible if parent is visible
+                        return isDeclarationVisible(parent);
+                    case 163 /* PropertyDeclaration */:
+                    case 162 /* PropertySignature */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 165 /* MethodDeclaration */:
+                    case 164 /* MethodSignature */:
+                        if (ts.hasEffectiveModifier(node, 8 /* Private */ | 16 /* Protected */)) {
+                            // Private/protected properties/methods are not visible
+                            return false;
+                        }
+                    // Public properties/methods are visible if its parents are visible, so:
+                    // falls through
+                    case 166 /* Constructor */:
+                    case 170 /* ConstructSignature */:
+                    case 169 /* CallSignature */:
+                    case 171 /* IndexSignature */:
+                    case 160 /* Parameter */:
+                    case 257 /* ModuleBlock */:
+                    case 174 /* FunctionType */:
+                    case 175 /* ConstructorType */:
+                    case 177 /* TypeLiteral */:
+                    case 173 /* TypeReference */:
+                    case 178 /* ArrayType */:
+                    case 179 /* TupleType */:
+                    case 182 /* UnionType */:
+                    case 183 /* IntersectionType */:
+                    case 186 /* ParenthesizedType */:
+                    case 192 /* NamedTupleMember */:
+                        return isDeclarationVisible(node.parent);
+                    // Default binding, import specifier and namespace import is visible
+                    // only on demand so by default it is not visible
+                    case 262 /* ImportClause */:
+                    case 263 /* NamespaceImport */:
+                    case 265 /* ImportSpecifier */:
+                        return false;
+                    // Type parameters are always visible
+                    case 159 /* TypeParameter */:
+                    // Source file and namespace export are always visible
+                    // falls through
+                    case 297 /* SourceFile */:
+                    case 259 /* NamespaceExportDeclaration */:
+                        return true;
+                    // Export assignments do not create name bindings outside the module
+                    case 266 /* ExportAssignment */:
+                        return false;
+                    default:
+                        return false;
                 }
-                // Cache both the resolved symbol and the resolved type. The resolved symbol is needed when we check the
-                // type reference in checkTypeReferenceNode.
-                links.resolvedSymbol = symbol;
-                links.resolvedType = type;
-            }
-            return links.resolvedType;
-        }
-        function typeArgumentsFromTypeReferenceNode(node) {
-            return ts.map(node.typeArguments, getTypeFromTypeNode);
-        }
-        function getTypeFromTypeQueryNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                // TypeScript 1.0 spec (April 2014): 3.6.3
-                // The expression is processed as an identifier expression (section 4.3)
-                // or property access expression(section 4.10),
-                // the widened type(section 3.9) of which becomes the result.
-                links.resolvedType = getRegularTypeOfLiteralType(getWidenedType(checkExpression(node.exprName)));
             }
-            return links.resolvedType;
         }
-        function getTypeOfGlobalSymbol(symbol, arity) {
-            function getTypeDeclaration(symbol) {
-                var declarations = symbol.declarations;
-                for (var _i = 0, declarations_3 = declarations; _i < declarations_3.length; _i++) {
-                    var declaration = declarations_3[_i];
-                    switch (declaration.kind) {
-                        case 245 /* ClassDeclaration */:
-                        case 246 /* InterfaceDeclaration */:
-                        case 248 /* EnumDeclaration */:
-                            return declaration;
-                    }
-                }
-            }
-            if (!symbol) {
-                return arity ? emptyGenericType : emptyObjectType;
+        function collectLinkedAliases(node, setVisibility) {
+            var exportSymbol;
+            if (node.parent && node.parent.kind === 266 /* ExportAssignment */) {
+                exportSymbol = resolveName(node, node.escapedText, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */, /*nameNotFoundMessage*/ undefined, node, /*isUse*/ false);
             }
-            var type = getDeclaredTypeOfSymbol(symbol);
-            if (!(type.flags & 524288 /* Object */)) {
-                error(getTypeDeclaration(symbol), ts.Diagnostics.Global_type_0_must_be_a_class_or_interface_type, ts.symbolName(symbol));
-                return arity ? emptyGenericType : emptyObjectType;
+            else if (node.parent.kind === 270 /* ExportSpecifier */) {
+                exportSymbol = getTargetOfExportSpecifier(node.parent, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */);
             }
-            if (ts.length(type.typeParameters) !== arity) {
-                error(getTypeDeclaration(symbol), ts.Diagnostics.Global_type_0_must_have_1_type_parameter_s, ts.symbolName(symbol), arity);
-                return arity ? emptyGenericType : emptyObjectType;
+            var result;
+            var visited;
+            if (exportSymbol) {
+                visited = new ts.Set();
+                visited.add(getSymbolId(exportSymbol));
+                buildVisibleNodeList(exportSymbol.declarations);
+            }
+            return result;
+            function buildVisibleNodeList(declarations) {
+                ts.forEach(declarations, function (declaration) {
+                    var resultNode = getAnyImportSyntax(declaration) || declaration;
+                    if (setVisibility) {
+                        getNodeLinks(declaration).isVisible = true;
+                    }
+                    else {
+                        result = result || [];
+                        ts.pushIfUnique(result, resultNode);
+                    }
+                    if (ts.isInternalModuleImportEqualsDeclaration(declaration)) {
+                        // Add the referenced top container visible
+                        var internalModuleReference = declaration.moduleReference;
+                        var firstIdentifier = ts.getFirstIdentifier(internalModuleReference);
+                        var importSymbol = resolveName(declaration, firstIdentifier.escapedText, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */, undefined, undefined, /*isUse*/ false);
+                        if (importSymbol && visited) {
+                            if (ts.tryAddToSet(visited, getSymbolId(importSymbol))) {
+                                buildVisibleNodeList(importSymbol.declarations);
+                            }
+                        }
+                    }
+                });
             }
-            return type;
-        }
-        function getGlobalValueSymbol(name, reportErrors) {
-            return getGlobalSymbol(name, 111551 /* Value */, reportErrors ? ts.Diagnostics.Cannot_find_global_value_0 : undefined);
-        }
-        function getGlobalTypeSymbol(name, reportErrors) {
-            return getGlobalSymbol(name, 788968 /* Type */, reportErrors ? ts.Diagnostics.Cannot_find_global_type_0 : undefined);
-        }
-        function getGlobalSymbol(name, meaning, diagnostic) {
-            // Don't track references for global symbols anyway, so value if `isReference` is arbitrary
-            return resolveName(undefined, name, meaning, diagnostic, name, /*isUse*/ false);
-        }
-        function getGlobalType(name, arity, reportErrors) {
-            var symbol = getGlobalTypeSymbol(name, reportErrors);
-            return symbol || reportErrors ? getTypeOfGlobalSymbol(symbol, arity) : undefined;
-        }
-        function getGlobalTypedPropertyDescriptorType() {
-            return deferredGlobalTypedPropertyDescriptorType || (deferredGlobalTypedPropertyDescriptorType = getGlobalType("TypedPropertyDescriptor", /*arity*/ 1, /*reportErrors*/ true)) || emptyGenericType;
-        }
-        function getGlobalTemplateStringsArrayType() {
-            return deferredGlobalTemplateStringsArrayType || (deferredGlobalTemplateStringsArrayType = getGlobalType("TemplateStringsArray", /*arity*/ 0, /*reportErrors*/ true)) || emptyObjectType;
-        }
-        function getGlobalImportMetaType() {
-            return deferredGlobalImportMetaType || (deferredGlobalImportMetaType = getGlobalType("ImportMeta", /*arity*/ 0, /*reportErrors*/ true)) || emptyObjectType;
-        }
-        function getGlobalESSymbolConstructorSymbol(reportErrors) {
-            return deferredGlobalESSymbolConstructorSymbol || (deferredGlobalESSymbolConstructorSymbol = getGlobalValueSymbol("Symbol", reportErrors));
-        }
-        function getGlobalESSymbolType(reportErrors) {
-            return deferredGlobalESSymbolType || (deferredGlobalESSymbolType = getGlobalType("Symbol", /*arity*/ 0, reportErrors)) || emptyObjectType;
-        }
-        function getGlobalPromiseType(reportErrors) {
-            return deferredGlobalPromiseType || (deferredGlobalPromiseType = getGlobalType("Promise", /*arity*/ 1, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalPromiseLikeType(reportErrors) {
-            return deferredGlobalPromiseLikeType || (deferredGlobalPromiseLikeType = getGlobalType("PromiseLike", /*arity*/ 1, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalPromiseConstructorSymbol(reportErrors) {
-            return deferredGlobalPromiseConstructorSymbol || (deferredGlobalPromiseConstructorSymbol = getGlobalValueSymbol("Promise", reportErrors));
-        }
-        function getGlobalPromiseConstructorLikeType(reportErrors) {
-            return deferredGlobalPromiseConstructorLikeType || (deferredGlobalPromiseConstructorLikeType = getGlobalType("PromiseConstructorLike", /*arity*/ 0, reportErrors)) || emptyObjectType;
-        }
-        function getGlobalAsyncIterableType(reportErrors) {
-            return deferredGlobalAsyncIterableType || (deferredGlobalAsyncIterableType = getGlobalType("AsyncIterable", /*arity*/ 1, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalAsyncIteratorType(reportErrors) {
-            return deferredGlobalAsyncIteratorType || (deferredGlobalAsyncIteratorType = getGlobalType("AsyncIterator", /*arity*/ 3, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalAsyncIterableIteratorType(reportErrors) {
-            return deferredGlobalAsyncIterableIteratorType || (deferredGlobalAsyncIterableIteratorType = getGlobalType("AsyncIterableIterator", /*arity*/ 1, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalAsyncGeneratorType(reportErrors) {
-            return deferredGlobalAsyncGeneratorType || (deferredGlobalAsyncGeneratorType = getGlobalType("AsyncGenerator", /*arity*/ 3, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalIterableType(reportErrors) {
-            return deferredGlobalIterableType || (deferredGlobalIterableType = getGlobalType("Iterable", /*arity*/ 1, reportErrors)) || emptyGenericType;
-        }
-        function getGlobalIteratorType(reportErrors) {
-            return deferredGlobalIteratorType || (deferredGlobalIteratorType = getGlobalType("Iterator", /*arity*/ 3, reportErrors)) || emptyGenericType;
         }
-        function getGlobalIterableIteratorType(reportErrors) {
-            return deferredGlobalIterableIteratorType || (deferredGlobalIterableIteratorType = getGlobalType("IterableIterator", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        /**
+         * Push an entry on the type resolution stack. If an entry with the given target and the given property name
+         * is already on the stack, and no entries in between already have a type, then a circularity has occurred.
+         * In this case, the result values of the existing entry and all entries pushed after it are changed to false,
+         * and the value false is returned. Otherwise, the new entry is just pushed onto the stack, and true is returned.
+         * In order to see if the same query has already been done before, the target object and the propertyName both
+         * must match the one passed in.
+         *
+         * @param target The symbol, type, or signature whose type is being queried
+         * @param propertyName The property name that should be used to query the target for its type
+         */
+        function pushTypeResolution(target, propertyName) {
+            var resolutionCycleStartIndex = findResolutionCycleStartIndex(target, propertyName);
+            if (resolutionCycleStartIndex >= 0) {
+                // A cycle was found
+                var length_3 = resolutionTargets.length;
+                for (var i = resolutionCycleStartIndex; i < length_3; i++) {
+                    resolutionResults[i] = false;
+                }
+                return false;
+            }
+            resolutionTargets.push(target);
+            resolutionResults.push(/*items*/ true);
+            resolutionPropertyNames.push(propertyName);
+            return true;
         }
-        function getGlobalGeneratorType(reportErrors) {
-            return deferredGlobalGeneratorType || (deferredGlobalGeneratorType = getGlobalType("Generator", /*arity*/ 3, reportErrors)) || emptyGenericType;
+        function findResolutionCycleStartIndex(target, propertyName) {
+            for (var i = resolutionTargets.length - 1; i >= 0; i--) {
+                if (hasType(resolutionTargets[i], resolutionPropertyNames[i])) {
+                    return -1;
+                }
+                if (resolutionTargets[i] === target && resolutionPropertyNames[i] === propertyName) {
+                    return i;
+                }
+            }
+            return -1;
         }
-        function getGlobalIteratorYieldResultType(reportErrors) {
-            return deferredGlobalIteratorYieldResultType || (deferredGlobalIteratorYieldResultType = getGlobalType("IteratorYieldResult", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        function hasType(target, propertyName) {
+            switch (propertyName) {
+                case 0 /* Type */:
+                    return !!getSymbolLinks(target).type;
+                case 5 /* EnumTagType */:
+                    return !!(getNodeLinks(target).resolvedEnumType);
+                case 2 /* DeclaredType */:
+                    return !!getSymbolLinks(target).declaredType;
+                case 1 /* ResolvedBaseConstructorType */:
+                    return !!target.resolvedBaseConstructorType;
+                case 3 /* ResolvedReturnType */:
+                    return !!target.resolvedReturnType;
+                case 4 /* ImmediateBaseConstraint */:
+                    return !!target.immediateBaseConstraint;
+                case 6 /* ResolvedTypeArguments */:
+                    return !!target.resolvedTypeArguments;
+                case 7 /* ResolvedBaseTypes */:
+                    return !!target.baseTypesResolved;
+            }
+            return ts.Debug.assertNever(propertyName);
         }
-        function getGlobalIteratorReturnResultType(reportErrors) {
-            return deferredGlobalIteratorReturnResultType || (deferredGlobalIteratorReturnResultType = getGlobalType("IteratorReturnResult", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        /**
+         * Pop an entry from the type resolution stack and return its associated result value. The result value will
+         * be true if no circularities were detected, or false if a circularity was found.
+         */
+        function popTypeResolution() {
+            resolutionTargets.pop();
+            resolutionPropertyNames.pop();
+            return resolutionResults.pop();
         }
-        function getGlobalTypeOrUndefined(name, arity) {
-            if (arity === void 0) { arity = 0; }
-            var symbol = getGlobalSymbol(name, 788968 /* Type */, /*diagnostic*/ undefined);
-            return symbol && getTypeOfGlobalSymbol(symbol, arity);
+        function getDeclarationContainer(node) {
+            return ts.findAncestor(ts.getRootDeclaration(node), function (node) {
+                switch (node.kind) {
+                    case 249 /* VariableDeclaration */:
+                    case 250 /* VariableDeclarationList */:
+                    case 265 /* ImportSpecifier */:
+                    case 264 /* NamedImports */:
+                    case 263 /* NamespaceImport */:
+                    case 262 /* ImportClause */:
+                        return false;
+                    default:
+                        return true;
+                }
+            }).parent;
         }
-        function getGlobalExtractSymbol() {
-            return deferredGlobalExtractSymbol || (deferredGlobalExtractSymbol = getGlobalSymbol("Extract", 524288 /* TypeAlias */, ts.Diagnostics.Cannot_find_global_type_0)); // TODO: GH#18217
+        function getTypeOfPrototypeProperty(prototype) {
+            // TypeScript 1.0 spec (April 2014): 8.4
+            // Every class automatically contains a static property member named 'prototype',
+            // the type of which is an instantiation of the class type with type Any supplied as a type argument for each type parameter.
+            // It is an error to explicitly declare a static property member with the name 'prototype'.
+            var classType = getDeclaredTypeOfSymbol(getParentOfSymbol(prototype));
+            return classType.typeParameters ? createTypeReference(classType, ts.map(classType.typeParameters, function (_) { return anyType; })) : classType;
         }
-        function getGlobalOmitSymbol() {
-            return deferredGlobalOmitSymbol || (deferredGlobalOmitSymbol = getGlobalSymbol("Omit", 524288 /* TypeAlias */, ts.Diagnostics.Cannot_find_global_type_0)); // TODO: GH#18217
+        // Return the type of the given property in the given type, or undefined if no such property exists
+        function getTypeOfPropertyOfType(type, name) {
+            var prop = getPropertyOfType(type, name);
+            return prop ? getTypeOfSymbol(prop) : undefined;
         }
-        function getGlobalBigIntType(reportErrors) {
-            return deferredGlobalBigIntType || (deferredGlobalBigIntType = getGlobalType("BigInt", /*arity*/ 0, reportErrors)) || emptyObjectType;
+        function getTypeOfPropertyOrIndexSignature(type, name) {
+            return getTypeOfPropertyOfType(type, name) || isNumericLiteralName(name) && getIndexTypeOfType(type, 1 /* Number */) || getIndexTypeOfType(type, 0 /* String */) || unknownType;
         }
-        /**
-         * Instantiates a global type that is generic with some element type, and returns that instantiation.
-         */
-        function createTypeFromGenericGlobalType(genericGlobalType, typeArguments) {
-            return genericGlobalType !== emptyGenericType ? createTypeReference(genericGlobalType, typeArguments) : emptyObjectType;
+        function isTypeAny(type) {
+            return type && (type.flags & 1 /* Any */) !== 0;
         }
-        function createTypedPropertyDescriptorType(propertyType) {
-            return createTypeFromGenericGlobalType(getGlobalTypedPropertyDescriptorType(), [propertyType]);
+        // Return the type of a binding element parent. We check SymbolLinks first to see if a type has been
+        // assigned by contextual typing.
+        function getTypeForBindingElementParent(node) {
+            var symbol = getSymbolOfNode(node);
+            return symbol && getSymbolLinks(symbol).type || getTypeForVariableLikeDeclaration(node, /*includeOptionality*/ false);
         }
-        function createIterableType(iteratedType) {
-            return createTypeFromGenericGlobalType(getGlobalIterableType(/*reportErrors*/ true), [iteratedType]);
+        function getRestType(source, properties, symbol) {
+            source = filterType(source, function (t) { return !(t.flags & 98304 /* Nullable */); });
+            if (source.flags & 131072 /* Never */) {
+                return emptyObjectType;
+            }
+            if (source.flags & 1048576 /* Union */) {
+                return mapType(source, function (t) { return getRestType(t, properties, symbol); });
+            }
+            var omitKeyType = getUnionType(ts.map(properties, getLiteralTypeFromPropertyName));
+            if (isGenericObjectType(source) || isGenericIndexType(omitKeyType)) {
+                if (omitKeyType.flags & 131072 /* Never */) {
+                    return source;
+                }
+                var omitTypeAlias = getGlobalOmitSymbol();
+                if (!omitTypeAlias) {
+                    return errorType;
+                }
+                return getTypeAliasInstantiation(omitTypeAlias, [source, omitKeyType]);
+            }
+            var members = ts.createSymbolTable();
+            for (var _i = 0, _a = getPropertiesOfType(source); _i < _a.length; _i++) {
+                var prop = _a[_i];
+                if (!isTypeAssignableTo(getLiteralTypeFromProperty(prop, 8576 /* StringOrNumberLiteralOrUnique */), omitKeyType)
+                    && !(ts.getDeclarationModifierFlagsFromSymbol(prop) & (8 /* Private */ | 16 /* Protected */))
+                    && isSpreadableProperty(prop)) {
+                    members.set(prop.escapedName, getSpreadSymbol(prop, /*readonly*/ false));
+                }
+            }
+            var stringIndexInfo = getIndexInfoOfType(source, 0 /* String */);
+            var numberIndexInfo = getIndexInfoOfType(source, 1 /* Number */);
+            var result = createAnonymousType(symbol, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
+            result.objectFlags |= 131072 /* ObjectRestType */;
+            return result;
         }
-        function createArrayType(elementType, readonly) {
-            return createTypeFromGenericGlobalType(readonly ? globalReadonlyArrayType : globalArrayType, [elementType]);
+        // Determine the control flow type associated with a destructuring declaration or assignment. The following
+        // forms of destructuring are possible:
+        //   let { x } = obj;  // BindingElement
+        //   let [ x ] = obj;  // BindingElement
+        //   { x } = obj;      // ShorthandPropertyAssignment
+        //   { x: v } = obj;   // PropertyAssignment
+        //   [ x ] = obj;      // Expression
+        // We construct a synthetic element access expression corresponding to 'obj.x' such that the control
+        // flow analyzer doesn't have to handle all the different syntactic forms.
+        function getFlowTypeOfDestructuring(node, declaredType) {
+            var reference = getSyntheticElementAccess(node);
+            return reference ? getFlowTypeOfReference(reference, declaredType) : declaredType;
         }
-        function getArrayOrTupleTargetType(node) {
-            var readonly = isReadonlyTypeOperator(node.parent);
-            if (node.kind === 174 /* ArrayType */ || node.elementTypes.length === 1 && node.elementTypes[0].kind === 177 /* RestType */) {
-                return readonly ? globalReadonlyArrayType : globalArrayType;
+        function getSyntheticElementAccess(node) {
+            var parentAccess = getParentElementAccess(node);
+            if (parentAccess && parentAccess.flowNode) {
+                var propName = getDestructuringPropertyName(node);
+                if (propName) {
+                    var literal = ts.setTextRange(ts.parseNodeFactory.createStringLiteral(propName), node);
+                    var lhsExpr = ts.isLeftHandSideExpression(parentAccess) ? parentAccess : ts.parseNodeFactory.createParenthesizedExpression(parentAccess);
+                    var result = ts.setTextRange(ts.parseNodeFactory.createElementAccessExpression(lhsExpr, literal), node);
+                    ts.setParent(literal, result);
+                    ts.setParent(result, node);
+                    if (lhsExpr !== parentAccess) {
+                        ts.setParent(lhsExpr, result);
+                    }
+                    result.flowNode = parentAccess.flowNode;
+                    return result;
+                }
             }
-            var lastElement = ts.lastOrUndefined(node.elementTypes);
-            var restElement = lastElement && lastElement.kind === 177 /* RestType */ ? lastElement : undefined;
-            var minLength = ts.findLastIndex(node.elementTypes, function (n) { return n.kind !== 176 /* OptionalType */ && n !== restElement; }) + 1;
-            return getTupleTypeOfArity(node.elementTypes.length, minLength, !!restElement, readonly, /*associatedNames*/ undefined);
         }
-        // Return true if the given type reference node is directly aliased or if it needs to be deferred
-        // because it is possibly contained in a circular chain of eagerly resolved types.
-        function isDeferredTypeReferenceNode(node, hasDefaultTypeArguments) {
-            return !!getAliasSymbolForTypeNode(node) || isResolvedByTypeAlias(node) && (node.kind === 174 /* ArrayType */ ? mayResolveTypeAlias(node.elementType) :
-                node.kind === 175 /* TupleType */ ? ts.some(node.elementTypes, mayResolveTypeAlias) :
-                    hasDefaultTypeArguments || ts.some(node.typeArguments, mayResolveTypeAlias));
+        function getParentElementAccess(node) {
+            var ancestor = node.parent.parent;
+            switch (ancestor.kind) {
+                case 198 /* BindingElement */:
+                case 288 /* PropertyAssignment */:
+                    return getSyntheticElementAccess(ancestor);
+                case 199 /* ArrayLiteralExpression */:
+                    return getSyntheticElementAccess(node.parent);
+                case 249 /* VariableDeclaration */:
+                    return ancestor.initializer;
+                case 216 /* BinaryExpression */:
+                    return ancestor.right;
+            }
         }
-        // Return true when the given node is transitively contained in type constructs that eagerly
-        // resolve their constituent types. We include SyntaxKind.TypeReference because type arguments
-        // of type aliases are eagerly resolved.
-        function isResolvedByTypeAlias(node) {
+        function getDestructuringPropertyName(node) {
             var parent = node.parent;
-            switch (parent.kind) {
-                case 182 /* ParenthesizedType */:
-                case 169 /* TypeReference */:
-                case 178 /* UnionType */:
-                case 179 /* IntersectionType */:
-                case 185 /* IndexedAccessType */:
-                case 180 /* ConditionalType */:
-                case 184 /* TypeOperator */:
-                case 174 /* ArrayType */:
-                case 175 /* TupleType */:
-                    return isResolvedByTypeAlias(parent);
-                case 247 /* TypeAliasDeclaration */:
-                    return true;
+            if (node.kind === 198 /* BindingElement */ && parent.kind === 196 /* ObjectBindingPattern */) {
+                return getLiteralPropertyNameText(node.propertyName || node.name);
             }
-            return false;
-        }
-        // Return true if resolving the given node (i.e. getTypeFromTypeNode) possibly causes resolution
-        // of a type alias.
-        function mayResolveTypeAlias(node) {
-            switch (node.kind) {
-                case 169 /* TypeReference */:
-                    return isJSDocTypeReference(node) || !!(resolveTypeReferenceName(node.typeName, 788968 /* Type */).flags & 524288 /* TypeAlias */);
-                case 172 /* TypeQuery */:
-                    return true;
-                case 184 /* TypeOperator */:
-                    return node.operator !== 147 /* UniqueKeyword */ && mayResolveTypeAlias(node.type);
-                case 182 /* ParenthesizedType */:
-                case 176 /* OptionalType */:
-                case 299 /* JSDocOptionalType */:
-                case 297 /* JSDocNullableType */:
-                case 298 /* JSDocNonNullableType */:
-                case 294 /* JSDocTypeExpression */:
-                    return mayResolveTypeAlias(node.type);
-                case 177 /* RestType */:
-                    return node.type.kind !== 174 /* ArrayType */ || mayResolveTypeAlias(node.type.elementType);
-                case 178 /* UnionType */:
-                case 179 /* IntersectionType */:
-                    return ts.some(node.types, mayResolveTypeAlias);
-                case 185 /* IndexedAccessType */:
-                    return mayResolveTypeAlias(node.objectType) || mayResolveTypeAlias(node.indexType);
-                case 180 /* ConditionalType */:
-                    return mayResolveTypeAlias(node.checkType) || mayResolveTypeAlias(node.extendsType) ||
-                        mayResolveTypeAlias(node.trueType) || mayResolveTypeAlias(node.falseType);
+            if (node.kind === 288 /* PropertyAssignment */ || node.kind === 289 /* ShorthandPropertyAssignment */) {
+                return getLiteralPropertyNameText(node.name);
             }
-            return false;
+            return "" + parent.elements.indexOf(node);
         }
-        function getTypeFromArrayOrTupleTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                var target = getArrayOrTupleTargetType(node);
-                if (target === emptyGenericType) {
-                    links.resolvedType = emptyObjectType;
-                }
-                else if (isDeferredTypeReferenceNode(node)) {
-                    links.resolvedType = node.kind === 175 /* TupleType */ && node.elementTypes.length === 0 ? target :
-                        createDeferredTypeReference(target, node, /*mapper*/ undefined);
+        function getLiteralPropertyNameText(name) {
+            var type = getLiteralTypeFromPropertyName(name);
+            return type.flags & (128 /* StringLiteral */ | 256 /* NumberLiteral */) ? "" + type.value : undefined;
+        }
+        /** Return the inferred type for a binding element */
+        function getTypeForBindingElement(declaration) {
+            var pattern = declaration.parent;
+            var parentType = getTypeForBindingElementParent(pattern.parent);
+            // If no type or an any type was inferred for parent, infer that for the binding element
+            if (!parentType || isTypeAny(parentType)) {
+                return parentType;
+            }
+            // Relax null check on ambient destructuring parameters, since the parameters have no implementation and are just documentation
+            if (strictNullChecks && declaration.flags & 8388608 /* Ambient */ && ts.isParameterDeclaration(declaration)) {
+                parentType = getNonNullableType(parentType);
+            }
+            // Filter `undefined` from the type we check against if the parent has an initializer and that initializer is not possibly `undefined`
+            else if (strictNullChecks && pattern.parent.initializer && !(getTypeFacts(getTypeOfInitializer(pattern.parent.initializer)) & 65536 /* EQUndefined */)) {
+                parentType = getTypeWithFacts(parentType, 524288 /* NEUndefined */);
+            }
+            var type;
+            if (pattern.kind === 196 /* ObjectBindingPattern */) {
+                if (declaration.dotDotDotToken) {
+                    parentType = getReducedType(parentType);
+                    if (parentType.flags & 2 /* Unknown */ || !isValidSpreadType(parentType)) {
+                        error(declaration, ts.Diagnostics.Rest_types_may_only_be_created_from_object_types);
+                        return errorType;
+                    }
+                    var literalMembers = [];
+                    for (var _i = 0, _a = pattern.elements; _i < _a.length; _i++) {
+                        var element = _a[_i];
+                        if (!element.dotDotDotToken) {
+                            literalMembers.push(element.propertyName || element.name);
+                        }
+                    }
+                    type = getRestType(parentType, literalMembers, declaration.symbol);
                 }
                 else {
-                    var elementTypes = node.kind === 174 /* ArrayType */ ? [getTypeFromTypeNode(node.elementType)] : ts.map(node.elementTypes, getTypeFromTypeNode);
-                    links.resolvedType = createTypeReference(target, elementTypes);
+                    // Use explicitly specified property name ({ p: xxx } form), or otherwise the implied name ({ p } form)
+                    var name = declaration.propertyName || declaration.name;
+                    var indexType = getLiteralTypeFromPropertyName(name);
+                    var declaredType = getConstraintForLocation(getIndexedAccessType(parentType, indexType, /*noUncheckedIndexedAccessCandidate*/ undefined, name, /*aliasSymbol*/ undefined, /*aliasTypeArguments*/ undefined, 16 /* ExpressionPosition */), declaration.name);
+                    type = getFlowTypeOfDestructuring(declaration, declaredType);
                 }
             }
-            return links.resolvedType;
-        }
-        function isReadonlyTypeOperator(node) {
-            return ts.isTypeOperatorNode(node) && node.operator === 138 /* ReadonlyKeyword */;
-        }
-        // We represent tuple types as type references to synthesized generic interface types created by
-        // this function. The types are of the form:
-        //
-        //   interface Tuple<T0, T1, T2, ...> extends Array<T0 | T1 | T2 | ...> { 0: T0, 1: T1, 2: T2, ... }
-        //
-        // Note that the generic type created by this function has no symbol associated with it. The same
-        // is true for each of the synthesized type parameters.
-        function createTupleTypeOfArity(arity, minLength, hasRestElement, readonly, associatedNames) {
-            var typeParameters;
-            var properties = [];
-            var maxLength = hasRestElement ? arity - 1 : arity;
-            if (arity) {
-                typeParameters = new Array(arity);
-                for (var i = 0; i < arity; i++) {
-                    var typeParameter = typeParameters[i] = createTypeParameter();
-                    if (i < maxLength) {
-                        var property = createSymbol(4 /* Property */ | (i >= minLength ? 16777216 /* Optional */ : 0), "" + i, readonly ? 8 /* Readonly */ : 0);
-                        property.type = typeParameter;
-                        properties.push(property);
-                    }
+            else {
+                // This elementType will be used if the specific property corresponding to this index is not
+                // present (aka the tuple element property). This call also checks that the parentType is in
+                // fact an iterable or array (depending on target language).
+                var elementType = checkIteratedTypeOrElementType(65 /* Destructuring */ | (declaration.dotDotDotToken ? 0 : 128 /* PossiblyOutOfBounds */), parentType, undefinedType, pattern);
+                var index_2 = pattern.elements.indexOf(declaration);
+                if (declaration.dotDotDotToken) {
+                    // If the parent is a tuple type, the rest element has a tuple type of the
+                    // remaining tuple element types. Otherwise, the rest element has an array type with same
+                    // element type as the parent type.
+                    type = everyType(parentType, isTupleType) ?
+                        mapType(parentType, function (t) { return sliceTupleType(t, index_2); }) :
+                        createArrayType(elementType);
+                }
+                else if (isArrayLikeType(parentType)) {
+                    var indexType = getLiteralType(index_2);
+                    var accessFlags = hasDefaultValue(declaration) ? 8 /* NoTupleBoundsCheck */ : 0;
+                    var declaredType = getConstraintForLocation(getIndexedAccessTypeOrUndefined(parentType, indexType, /*noUncheckedIndexedAccessCandidate*/ undefined, declaration.name, accessFlags | 16 /* ExpressionPosition */) || errorType, declaration.name);
+                    type = getFlowTypeOfDestructuring(declaration, declaredType);
+                }
+                else {
+                    type = elementType;
                 }
             }
-            var literalTypes = [];
-            for (var i = minLength; i <= maxLength; i++)
-                literalTypes.push(getLiteralType(i));
-            var lengthSymbol = createSymbol(4 /* Property */, "length");
-            lengthSymbol.type = hasRestElement ? numberType : getUnionType(literalTypes);
-            properties.push(lengthSymbol);
-            var type = createObjectType(8 /* Tuple */ | 4 /* Reference */);
-            type.typeParameters = typeParameters;
-            type.outerTypeParameters = undefined;
-            type.localTypeParameters = typeParameters;
-            type.instantiations = ts.createMap();
-            type.instantiations.set(getTypeListId(type.typeParameters), type);
-            type.target = type;
-            type.resolvedTypeArguments = type.typeParameters;
-            type.thisType = createTypeParameter();
-            type.thisType.isThisType = true;
-            type.thisType.constraint = type;
-            type.declaredProperties = properties;
-            type.declaredCallSignatures = ts.emptyArray;
-            type.declaredConstructSignatures = ts.emptyArray;
-            type.declaredStringIndexInfo = undefined;
-            type.declaredNumberIndexInfo = undefined;
-            type.minLength = minLength;
-            type.hasRestElement = hasRestElement;
-            type.readonly = readonly;
-            type.associatedNames = associatedNames;
-            return type;
-        }
-        function getTupleTypeOfArity(arity, minLength, hasRestElement, readonly, associatedNames) {
-            var key = arity + (hasRestElement ? "+" : ",") + minLength + (readonly ? "R" : "") + (associatedNames && associatedNames.length ? "," + associatedNames.join(",") : "");
-            var type = tupleTypes.get(key);
-            if (!type) {
-                tupleTypes.set(key, type = createTupleTypeOfArity(arity, minLength, hasRestElement, readonly, associatedNames));
+            if (!declaration.initializer) {
+                return type;
             }
-            return type;
-        }
-        function createTupleType(elementTypes, minLength, hasRestElement, readonly, associatedNames) {
-            if (minLength === void 0) { minLength = elementTypes.length; }
-            if (hasRestElement === void 0) { hasRestElement = false; }
-            if (readonly === void 0) { readonly = false; }
-            var arity = elementTypes.length;
-            if (arity === 1 && hasRestElement) {
-                return createArrayType(elementTypes[0], readonly);
+            if (ts.getEffectiveTypeAnnotationNode(ts.walkUpBindingElementsAndPatterns(declaration))) {
+                // In strict null checking mode, if a default value of a non-undefined type is specified, remove
+                // undefined from the final type.
+                return strictNullChecks && !(getFalsyFlags(checkDeclarationInitializer(declaration)) & 32768 /* Undefined */) ?
+                    getTypeWithFacts(type, 524288 /* NEUndefined */) :
+                    type;
             }
-            var tupleType = getTupleTypeOfArity(arity, minLength, arity > 0 && hasRestElement, readonly, associatedNames);
-            return elementTypes.length ? createTypeReference(tupleType, elementTypes) : tupleType;
+            return widenTypeInferredFromInitializer(declaration, getUnionType([getTypeWithFacts(type, 524288 /* NEUndefined */), checkDeclarationInitializer(declaration)], 2 /* Subtype */));
         }
-        function sliceTupleType(type, index) {
-            var tuple = type.target;
-            if (tuple.hasRestElement) {
-                // don't slice off rest element
-                index = Math.min(index, getTypeReferenceArity(type) - 1);
+        function getTypeForDeclarationFromJSDocComment(declaration) {
+            var jsdocType = ts.getJSDocType(declaration);
+            if (jsdocType) {
+                return getTypeFromTypeNode(jsdocType);
             }
-            return createTupleType(getTypeArguments(type).slice(index), Math.max(0, tuple.minLength - index), tuple.hasRestElement, tuple.readonly, tuple.associatedNames && tuple.associatedNames.slice(index));
+            return undefined;
         }
-        function getTypeFromOptionalTypeNode(node) {
-            var type = getTypeFromTypeNode(node.type);
-            return strictNullChecks ? getOptionalType(type) : type;
+        function isNullOrUndefined(node) {
+            var expr = ts.skipParentheses(node);
+            return expr.kind === 103 /* NullKeyword */ || expr.kind === 78 /* Identifier */ && getResolvedSymbol(expr) === undefinedSymbol;
         }
-        function getTypeId(type) {
-            return type.id;
+        function isEmptyArrayLiteral(node) {
+            var expr = ts.skipParentheses(node);
+            return expr.kind === 199 /* ArrayLiteralExpression */ && expr.elements.length === 0;
         }
-        function containsType(types, type) {
-            return ts.binarySearch(types, type, getTypeId, ts.compareValues) >= 0;
+        function addOptionality(type, optional) {
+            if (optional === void 0) { optional = true; }
+            return strictNullChecks && optional ? getOptionalType(type) : type;
         }
-        function insertType(types, type) {
-            var index = ts.binarySearch(types, type, getTypeId, ts.compareValues);
-            if (index < 0) {
-                types.splice(~index, 0, type);
-                return true;
+        // Return the inferred type for a variable, parameter, or property declaration
+        function getTypeForVariableLikeDeclaration(declaration, includeOptionality) {
+            // A variable declared in a for..in statement is of type string, or of type keyof T when the
+            // right hand expression is of a type parameter type.
+            if (ts.isVariableDeclaration(declaration) && declaration.parent.parent.kind === 238 /* ForInStatement */) {
+                var indexType = getIndexType(getNonNullableTypeIfNeeded(checkExpression(declaration.parent.parent.expression)));
+                return indexType.flags & (262144 /* TypeParameter */ | 4194304 /* Index */) ? getExtractStringType(indexType) : stringType;
             }
-            return false;
-        }
-        function addTypeToUnion(typeSet, includes, type) {
-            var flags = type.flags;
-            if (flags & 1048576 /* Union */) {
-                return addTypesToUnion(typeSet, includes, type.types);
+            if (ts.isVariableDeclaration(declaration) && declaration.parent.parent.kind === 239 /* ForOfStatement */) {
+                // checkRightHandSideOfForOf will return undefined if the for-of expression type was
+                // missing properties/signatures required to get its iteratedType (like
+                // [Symbol.iterator] or next). This may be because we accessed properties from anyType,
+                // or it may have led to an error inside getElementTypeOfIterable.
+                var forOfStatement = declaration.parent.parent;
+                return checkRightHandSideOfForOf(forOfStatement) || anyType;
             }
-            // We ignore 'never' types in unions
-            if (!(flags & 131072 /* Never */)) {
-                includes |= flags & 71041023 /* IncludesMask */;
-                if (flags & 66846720 /* StructuredOrInstantiable */)
-                    includes |= 262144 /* IncludesStructuredOrInstantiable */;
-                if (type === wildcardType)
-                    includes |= 8388608 /* IncludesWildcard */;
-                if (!strictNullChecks && flags & 98304 /* Nullable */) {
-                    if (!(ts.getObjectFlags(type) & 524288 /* ContainsWideningType */))
-                        includes |= 4194304 /* IncludesNonWideningType */;
+            if (ts.isBindingPattern(declaration.parent)) {
+                return getTypeForBindingElement(declaration);
+            }
+            var isOptional = includeOptionality && (ts.isParameter(declaration) && isJSDocOptionalParameter(declaration)
+                || isOptionalJSDocPropertyLikeTag(declaration)
+                || !ts.isBindingElement(declaration) && !ts.isVariableDeclaration(declaration) && !!declaration.questionToken);
+            // Use type from type annotation if one is present
+            var declaredType = tryGetTypeFromEffectiveTypeNode(declaration);
+            if (declaredType) {
+                return addOptionality(declaredType, isOptional);
+            }
+            if ((noImplicitAny || ts.isInJSFile(declaration)) &&
+                ts.isVariableDeclaration(declaration) && !ts.isBindingPattern(declaration.name) &&
+                !(ts.getCombinedModifierFlags(declaration) & 1 /* Export */) && !(declaration.flags & 8388608 /* Ambient */)) {
+                // If --noImplicitAny is on or the declaration is in a Javascript file,
+                // use control flow tracked 'any' type for non-ambient, non-exported var or let variables with no
+                // initializer or a 'null' or 'undefined' initializer.
+                if (!(ts.getCombinedNodeFlags(declaration) & 2 /* Const */) && (!declaration.initializer || isNullOrUndefined(declaration.initializer))) {
+                    return autoType;
                 }
-                else {
-                    var len = typeSet.length;
-                    var index = len && type.id > typeSet[len - 1].id ? ~len : ts.binarySearch(typeSet, type, getTypeId, ts.compareValues);
-                    if (index < 0) {
-                        typeSet.splice(~index, 0, type);
+                // Use control flow tracked 'any[]' type for non-ambient, non-exported variables with an empty array
+                // literal initializer.
+                if (declaration.initializer && isEmptyArrayLiteral(declaration.initializer)) {
+                    return autoArrayType;
+                }
+            }
+            if (ts.isParameter(declaration)) {
+                var func = declaration.parent;
+                // For a parameter of a set accessor, use the type of the get accessor if one is present
+                if (func.kind === 168 /* SetAccessor */ && !hasNonBindableDynamicName(func)) {
+                    var getter = ts.getDeclarationOfKind(getSymbolOfNode(declaration.parent), 167 /* GetAccessor */);
+                    if (getter) {
+                        var getterSignature = getSignatureFromDeclaration(getter);
+                        var thisParameter = getAccessorThisParameter(func);
+                        if (thisParameter && declaration === thisParameter) {
+                            // Use the type from the *getter*
+                            ts.Debug.assert(!thisParameter.type);
+                            return getTypeOfSymbol(getterSignature.thisParameter);
+                        }
+                        return getReturnTypeOfSignature(getterSignature);
+                    }
+                }
+                if (ts.isInJSFile(declaration)) {
+                    var typeTag = ts.getJSDocType(func);
+                    if (typeTag && ts.isFunctionTypeNode(typeTag)) {
+                        var signature = getSignatureFromDeclaration(typeTag);
+                        var pos = func.parameters.indexOf(declaration);
+                        return declaration.dotDotDotToken ? getRestTypeAtPosition(signature, pos) : getTypeAtPosition(signature, pos);
                     }
                 }
+                // Use contextual parameter type if one is available
+                var type = declaration.symbol.escapedName === "this" /* This */ ? getContextualThisParameterType(func) : getContextuallyTypedParameterType(declaration);
+                if (type) {
+                    return addOptionality(type, isOptional);
+                }
             }
-            return includes;
-        }
-        // Add the given types to the given type set. Order is preserved, duplicates are removed,
-        // and nested types of the given kind are flattened into the set.
-        function addTypesToUnion(typeSet, includes, types) {
-            for (var _i = 0, types_9 = types; _i < types_9.length; _i++) {
-                var type = types_9[_i];
-                includes = addTypeToUnion(typeSet, includes, type);
+            // Use the type of the initializer expression if one is present and the declaration is
+            // not a parameter of a contextually typed function
+            if (ts.hasOnlyExpressionInitializer(declaration) && !!declaration.initializer) {
+                if (ts.isInJSFile(declaration) && !ts.isParameter(declaration)) {
+                    var containerObjectType = getJSContainerObjectType(declaration, getSymbolOfNode(declaration), ts.getDeclaredExpandoInitializer(declaration));
+                    if (containerObjectType) {
+                        return containerObjectType;
+                    }
+                }
+                var type = widenTypeInferredFromInitializer(declaration, checkDeclarationInitializer(declaration));
+                return addOptionality(type, isOptional);
             }
-            return includes;
+            if (ts.isPropertyDeclaration(declaration) && !ts.hasStaticModifier(declaration) && (noImplicitAny || ts.isInJSFile(declaration))) {
+                // We have a property declaration with no type annotation or initializer, in noImplicitAny mode or a .js file.
+                // Use control flow analysis of this.xxx assignments in the constructor to determine the type of the property.
+                var constructor = findConstructorDeclaration(declaration.parent);
+                var type = constructor ? getFlowTypeInConstructor(declaration.symbol, constructor) :
+                    ts.getEffectiveModifierFlags(declaration) & 2 /* Ambient */ ? getTypeOfPropertyInBaseClass(declaration.symbol) :
+                        undefined;
+                return type && addOptionality(type, isOptional);
+            }
+            if (ts.isJsxAttribute(declaration)) {
+                // if JSX attribute doesn't have initializer, by default the attribute will have boolean value of true.
+                // I.e <Elem attr /> is sugar for <Elem attr={true} />
+                return trueType;
+            }
+            // If the declaration specifies a binding pattern and is not a parameter of a contextually
+            // typed function, use the type implied by the binding pattern
+            if (ts.isBindingPattern(declaration.name)) {
+                return getTypeFromBindingPattern(declaration.name, /*includePatternInType*/ false, /*reportErrors*/ true);
+            }
+            // No type specified and nothing can be inferred
+            return undefined;
         }
-        function isSetOfLiteralsFromSameEnum(types) {
-            var first = types[0];
-            if (first.flags & 1024 /* EnumLiteral */) {
-                var firstEnum = getParentOfSymbol(first.symbol);
-                for (var i = 1; i < types.length; i++) {
-                    var other = types[i];
-                    if (!(other.flags & 1024 /* EnumLiteral */) || (firstEnum !== getParentOfSymbol(other.symbol))) {
-                        return false;
-                    }
+        function isConstructorDeclaredProperty(symbol) {
+            // A property is considered a constructor declared property when all declaration sites are this.xxx assignments,
+            // when no declaration sites have JSDoc type annotations, and when at least one declaration site is in the body of
+            // a class constructor.
+            if (symbol.valueDeclaration && ts.isBinaryExpression(symbol.valueDeclaration)) {
+                var links = getSymbolLinks(symbol);
+                if (links.isConstructorDeclaredProperty === undefined) {
+                    links.isConstructorDeclaredProperty = false;
+                    links.isConstructorDeclaredProperty = !!getDeclaringConstructor(symbol) && ts.every(symbol.declarations, function (declaration) {
+                        return ts.isBinaryExpression(declaration) &&
+                            isPossiblyAliasedThisProperty(declaration) &&
+                            (declaration.left.kind !== 202 /* ElementAccessExpression */ || ts.isStringOrNumericLiteralLike(declaration.left.argumentExpression)) &&
+                            !getAnnotatedTypeForAssignmentDeclaration(/*declaredType*/ undefined, declaration, symbol, declaration);
+                    });
                 }
-                return true;
+                return links.isConstructorDeclaredProperty;
             }
             return false;
         }
-        function removeSubtypes(types, primitivesOnly) {
-            var len = types.length;
-            if (len === 0 || isSetOfLiteralsFromSameEnum(types)) {
-                return true;
+        function isAutoTypedProperty(symbol) {
+            // A property is auto-typed when its declaration has no type annotation or initializer and we're in
+            // noImplicitAny mode or a .js file.
+            var declaration = symbol.valueDeclaration;
+            return declaration && ts.isPropertyDeclaration(declaration) && !ts.getEffectiveTypeAnnotationNode(declaration) &&
+                !declaration.initializer && (noImplicitAny || ts.isInJSFile(declaration));
+        }
+        function getDeclaringConstructor(symbol) {
+            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                var declaration = _a[_i];
+                var container = ts.getThisContainer(declaration, /*includeArrowFunctions*/ false);
+                if (container && (container.kind === 166 /* Constructor */ || isJSConstructor(container))) {
+                    return container;
+                }
             }
-            var i = len;
-            var count = 0;
-            while (i > 0) {
-                i--;
-                var source = types[i];
-                for (var _i = 0, types_10 = types; _i < types_10.length; _i++) {
-                    var target = types_10[_i];
-                    if (source !== target) {
-                        if (count === 100000) {
-                            // After 100000 subtype checks we estimate the remaining amount of work by assuming the
-                            // same ratio of checks per element. If the estimated number of remaining type checks is
-                            // greater than an upper limit we deem the union type too complex to represent. The
-                            // upper limit is 25M for unions of primitives only, and 1M otherwise. This for example
-                            // caps union types at 5000 unique literal types and 1000 unique object types.
-                            var estimatedCount = (count / (len - i)) * len;
-                            if (estimatedCount > (primitivesOnly ? 25000000 : 1000000)) {
-                                error(currentNode, ts.Diagnostics.Expression_produces_a_union_type_that_is_too_complex_to_represent);
-                                return false;
-                            }
+        }
+        function getFlowTypeInConstructor(symbol, constructor) {
+            var accessName = ts.startsWith(symbol.escapedName, "__#")
+                ? ts.factory.createPrivateIdentifier(symbol.escapedName.split("@")[1])
+                : ts.unescapeLeadingUnderscores(symbol.escapedName);
+            var reference = ts.factory.createPropertyAccessExpression(ts.factory.createThis(), accessName);
+            ts.setParent(reference.expression, reference);
+            ts.setParent(reference, constructor);
+            reference.flowNode = constructor.returnFlowNode;
+            var flowType = getFlowTypeOfProperty(reference, symbol);
+            if (noImplicitAny && (flowType === autoType || flowType === autoArrayType)) {
+                error(symbol.valueDeclaration, ts.Diagnostics.Member_0_implicitly_has_an_1_type, symbolToString(symbol), typeToString(flowType));
+            }
+            // We don't infer a type if assignments are only null or undefined.
+            return everyType(flowType, isNullableType) ? undefined : convertAutoToAny(flowType);
+        }
+        function getFlowTypeOfProperty(reference, prop) {
+            var initialType = prop && (!isAutoTypedProperty(prop) || ts.getEffectiveModifierFlags(prop.valueDeclaration) & 2 /* Ambient */) && getTypeOfPropertyInBaseClass(prop) || undefinedType;
+            return getFlowTypeOfReference(reference, autoType, initialType);
+        }
+        function getWidenedTypeForAssignmentDeclaration(symbol, resolvedSymbol) {
+            // function/class/{} initializers are themselves containers, so they won't merge in the same way as other initializers
+            var container = ts.getAssignedExpandoInitializer(symbol.valueDeclaration);
+            if (container) {
+                var tag = ts.getJSDocTypeTag(container);
+                if (tag && tag.typeExpression) {
+                    return getTypeFromTypeNode(tag.typeExpression);
+                }
+                var containerObjectType = getJSContainerObjectType(symbol.valueDeclaration, symbol, container);
+                return containerObjectType || getWidenedLiteralType(checkExpressionCached(container));
+            }
+            var type;
+            var definedInConstructor = false;
+            var definedInMethod = false;
+            // We use control flow analysis to determine the type of the property if the property qualifies as a constructor
+            // declared property and the resulting control flow type isn't just undefined or null.
+            if (isConstructorDeclaredProperty(symbol)) {
+                type = getFlowTypeInConstructor(symbol, getDeclaringConstructor(symbol));
+            }
+            if (!type) {
+                var jsdocType = void 0;
+                var types = void 0;
+                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                    var declaration = _a[_i];
+                    var expression = (ts.isBinaryExpression(declaration) || ts.isCallExpression(declaration)) ? declaration :
+                        ts.isAccessExpression(declaration) ? ts.isBinaryExpression(declaration.parent) ? declaration.parent : declaration :
+                            undefined;
+                    if (!expression) {
+                        continue; // Non-assignment declaration merged in (eg, an Identifier to mark the thing as a namespace) - skip over it and pull type info from elsewhere
+                    }
+                    var kind = ts.isAccessExpression(expression)
+                        ? ts.getAssignmentDeclarationPropertyAccessKind(expression)
+                        : ts.getAssignmentDeclarationKind(expression);
+                    if (kind === 4 /* ThisProperty */ || ts.isBinaryExpression(expression) && isPossiblyAliasedThisProperty(expression, kind)) {
+                        if (isDeclarationInConstructor(expression)) {
+                            definedInConstructor = true;
                         }
-                        count++;
-                        if (isTypeRelatedTo(source, target, strictSubtypeRelation) && (!(ts.getObjectFlags(getTargetType(source)) & 1 /* Class */) ||
-                            !(ts.getObjectFlags(getTargetType(target)) & 1 /* Class */) ||
-                            isTypeDerivedFrom(source, target))) {
-                            ts.orderedRemoveItemAt(types, i);
-                            break;
+                        else {
+                            definedInMethod = true;
                         }
                     }
+                    if (!ts.isCallExpression(expression)) {
+                        jsdocType = getAnnotatedTypeForAssignmentDeclaration(jsdocType, expression, symbol, declaration);
+                    }
+                    if (!jsdocType) {
+                        (types || (types = [])).push((ts.isBinaryExpression(expression) || ts.isCallExpression(expression)) ? getInitializerTypeFromAssignmentDeclaration(symbol, resolvedSymbol, expression, kind) : neverType);
+                    }
                 }
-            }
-            return true;
-        }
-        function removeRedundantLiteralTypes(types, includes) {
-            var i = types.length;
-            while (i > 0) {
-                i--;
-                var t = types[i];
-                var remove = t.flags & 128 /* StringLiteral */ && includes & 4 /* String */ ||
-                    t.flags & 256 /* NumberLiteral */ && includes & 8 /* Number */ ||
-                    t.flags & 2048 /* BigIntLiteral */ && includes & 64 /* BigInt */ ||
-                    t.flags & 8192 /* UniqueESSymbol */ && includes & 4096 /* ESSymbol */ ||
-                    isFreshLiteralType(t) && containsType(types, t.regularType);
-                if (remove) {
-                    ts.orderedRemoveItemAt(types, i);
+                type = jsdocType;
+                if (!type) {
+                    if (!ts.length(types)) {
+                        return errorType; // No types from any declarations :(
+                    }
+                    var constructorTypes = definedInConstructor ? getConstructorDefinedThisAssignmentTypes(types, symbol.declarations) : undefined;
+                    // use only the constructor types unless they were only assigned null | undefined (including widening variants)
+                    if (definedInMethod) {
+                        var propType = getTypeOfPropertyInBaseClass(symbol);
+                        if (propType) {
+                            (constructorTypes || (constructorTypes = [])).push(propType);
+                            definedInConstructor = true;
+                        }
+                    }
+                    var sourceTypes = ts.some(constructorTypes, function (t) { return !!(t.flags & ~98304 /* Nullable */); }) ? constructorTypes : types; // TODO: GH#18217
+                    type = getUnionType(sourceTypes, 2 /* Subtype */);
                 }
             }
+            var widened = getWidenedType(addOptionality(type, definedInMethod && !definedInConstructor));
+            if (filterType(widened, function (t) { return !!(t.flags & ~98304 /* Nullable */); }) === neverType) {
+                reportImplicitAny(symbol.valueDeclaration, anyType);
+                return anyType;
+            }
+            return widened;
         }
-        // We sort and deduplicate the constituent types based on object identity. If the subtypeReduction
-        // flag is specified we also reduce the constituent type set to only include types that aren't subtypes
-        // of other types. Subtype reduction is expensive for large union types and is possible only when union
-        // types are known not to circularly reference themselves (as is the case with union types created by
-        // expression constructs such as array literals and the || and ?: operators). Named types can
-        // circularly reference themselves and therefore cannot be subtype reduced during their declaration.
-        // For example, "type Item = string | (() => Item" is a named type that circularly references itself.
-        function getUnionType(types, unionReduction, aliasSymbol, aliasTypeArguments) {
-            if (unionReduction === void 0) { unionReduction = 1 /* Literal */; }
-            if (types.length === 0) {
-                return neverType;
+        function getJSContainerObjectType(decl, symbol, init) {
+            var _a, _b;
+            if (!ts.isInJSFile(decl) || !init || !ts.isObjectLiteralExpression(init) || init.properties.length) {
+                return undefined;
             }
-            if (types.length === 1) {
-                return types[0];
+            var exports = ts.createSymbolTable();
+            while (ts.isBinaryExpression(decl) || ts.isPropertyAccessExpression(decl)) {
+                var s_2 = getSymbolOfNode(decl);
+                if ((_a = s_2 === null || s_2 === void 0 ? void 0 : s_2.exports) === null || _a === void 0 ? void 0 : _a.size) {
+                    mergeSymbolTable(exports, s_2.exports);
+                }
+                decl = ts.isBinaryExpression(decl) ? decl.parent : decl.parent.parent;
             }
-            var typeSet = [];
-            var includes = addTypesToUnion(typeSet, 0, types);
-            if (unionReduction !== 0 /* None */) {
-                if (includes & 3 /* AnyOrUnknown */) {
-                    return includes & 1 /* Any */ ? includes & 8388608 /* IncludesWildcard */ ? wildcardType : anyType : unknownType;
+            var s = getSymbolOfNode(decl);
+            if ((_b = s === null || s === void 0 ? void 0 : s.exports) === null || _b === void 0 ? void 0 : _b.size) {
+                mergeSymbolTable(exports, s.exports);
+            }
+            var type = createAnonymousType(symbol, exports, ts.emptyArray, ts.emptyArray, undefined, undefined);
+            type.objectFlags |= 16384 /* JSLiteral */;
+            return type;
+        }
+        function getAnnotatedTypeForAssignmentDeclaration(declaredType, expression, symbol, declaration) {
+            var typeNode = ts.getEffectiveTypeAnnotationNode(expression.parent);
+            if (typeNode) {
+                var type = getWidenedType(getTypeFromTypeNode(typeNode));
+                if (!declaredType) {
+                    return type;
                 }
-                switch (unionReduction) {
-                    case 1 /* Literal */:
-                        if (includes & (2944 /* Literal */ | 8192 /* UniqueESSymbol */)) {
-                            removeRedundantLiteralTypes(typeSet, includes);
-                        }
-                        break;
-                    case 2 /* Subtype */:
-                        if (!removeSubtypes(typeSet, !(includes & 262144 /* IncludesStructuredOrInstantiable */))) {
-                            return errorType;
-                        }
-                        break;
+                else if (declaredType !== errorType && type !== errorType && !isTypeIdenticalTo(declaredType, type)) {
+                    errorNextVariableOrPropertyDeclarationMustHaveSameType(/*firstDeclaration*/ undefined, declaredType, declaration, type);
                 }
-                if (typeSet.length === 0) {
-                    return includes & 65536 /* Null */ ? includes & 4194304 /* IncludesNonWideningType */ ? nullType : nullWideningType :
-                        includes & 32768 /* Undefined */ ? includes & 4194304 /* IncludesNonWideningType */ ? undefinedType : undefinedWideningType :
-                            neverType;
+            }
+            if (symbol.parent) {
+                var typeNode_2 = ts.getEffectiveTypeAnnotationNode(symbol.parent.valueDeclaration);
+                if (typeNode_2) {
+                    return getTypeOfPropertyOfType(getTypeFromTypeNode(typeNode_2), symbol.escapedName);
                 }
             }
-            var objectFlags = (includes & 66994211 /* NotPrimitiveUnion */ ? 0 : 262144 /* PrimitiveUnion */) |
-                (includes & 2097152 /* Intersection */ ? 268435456 /* ContainsIntersections */ : 0);
-            return getUnionTypeFromSortedList(typeSet, objectFlags, aliasSymbol, aliasTypeArguments);
+            return declaredType;
         }
-        function getUnionTypePredicate(signatures) {
-            var first;
-            var types = [];
-            for (var _i = 0, signatures_6 = signatures; _i < signatures_6.length; _i++) {
-                var sig = signatures_6[_i];
-                var pred = getTypePredicateOfSignature(sig);
-                if (!pred || pred.kind === 2 /* AssertsThis */ || pred.kind === 3 /* AssertsIdentifier */) {
-                    continue;
+        /** If we don't have an explicit JSDoc type, get the type from the initializer. */
+        function getInitializerTypeFromAssignmentDeclaration(symbol, resolvedSymbol, expression, kind) {
+            if (ts.isCallExpression(expression)) {
+                if (resolvedSymbol) {
+                    return getTypeOfSymbol(resolvedSymbol); // This shouldn't happen except under some hopefully forbidden merges of export assignments and object define assignments
                 }
-                if (first) {
-                    if (!typePredicateKindsMatch(first, pred)) {
-                        // No common type predicate.
-                        return undefined;
+                var objectLitType = checkExpressionCached(expression.arguments[2]);
+                var valueType = getTypeOfPropertyOfType(objectLitType, "value");
+                if (valueType) {
+                    return valueType;
+                }
+                var getFunc = getTypeOfPropertyOfType(objectLitType, "get");
+                if (getFunc) {
+                    var getSig = getSingleCallSignature(getFunc);
+                    if (getSig) {
+                        return getReturnTypeOfSignature(getSig);
                     }
                 }
-                else {
-                    first = pred;
+                var setFunc = getTypeOfPropertyOfType(objectLitType, "set");
+                if (setFunc) {
+                    var setSig = getSingleCallSignature(setFunc);
+                    if (setSig) {
+                        return getTypeOfFirstParameterOfSignature(setSig);
+                    }
                 }
-                types.push(pred.type);
+                return anyType;
             }
-            if (!first) {
-                // No union signatures had a type predicate.
-                return undefined;
+            if (containsSameNamedThisProperty(expression.left, expression.right)) {
+                return anyType;
             }
-            var unionType = getUnionType(types);
-            return createTypePredicate(first.kind, first.parameterName, first.parameterIndex, unionType);
+            var type = resolvedSymbol ? getTypeOfSymbol(resolvedSymbol) : getWidenedLiteralType(checkExpressionCached(expression.right));
+            if (type.flags & 524288 /* Object */ &&
+                kind === 2 /* ModuleExports */ &&
+                symbol.escapedName === "export=" /* ExportEquals */) {
+                var exportedType = resolveStructuredTypeMembers(type);
+                var members_4 = ts.createSymbolTable();
+                ts.copyEntries(exportedType.members, members_4);
+                var initialSize = members_4.size;
+                if (resolvedSymbol && !resolvedSymbol.exports) {
+                    resolvedSymbol.exports = ts.createSymbolTable();
+                }
+                (resolvedSymbol || symbol).exports.forEach(function (s, name) {
+                    var _a;
+                    var exportedMember = members_4.get(name);
+                    if (exportedMember && exportedMember !== s) {
+                        if (s.flags & 111551 /* Value */ && exportedMember.flags & 111551 /* Value */) {
+                            // If the member has an additional value-like declaration, union the types from the two declarations,
+                            // but issue an error if they occurred in two different files. The purpose is to support a JS file with
+                            // a pattern like:
+                            //
+                            // module.exports = { a: true };
+                            // module.exports.a = 3;
+                            //
+                            // but we may have a JS file with `module.exports = { a: true }` along with a TypeScript module augmentation
+                            // declaring an `export const a: number`. In that case, we issue a duplicate identifier error, because
+                            // it's unclear what that's supposed to mean, so it's probably a mistake.
+                            if (ts.getSourceFileOfNode(s.valueDeclaration) !== ts.getSourceFileOfNode(exportedMember.valueDeclaration)) {
+                                var unescapedName = ts.unescapeLeadingUnderscores(s.escapedName);
+                                var exportedMemberName = ((_a = ts.tryCast(exportedMember.valueDeclaration, ts.isNamedDeclaration)) === null || _a === void 0 ? void 0 : _a.name) || exportedMember.valueDeclaration;
+                                ts.addRelatedInfo(error(s.valueDeclaration, ts.Diagnostics.Duplicate_identifier_0, unescapedName), ts.createDiagnosticForNode(exportedMemberName, ts.Diagnostics._0_was_also_declared_here, unescapedName));
+                                ts.addRelatedInfo(error(exportedMemberName, ts.Diagnostics.Duplicate_identifier_0, unescapedName), ts.createDiagnosticForNode(s.valueDeclaration, ts.Diagnostics._0_was_also_declared_here, unescapedName));
+                            }
+                            var union = createSymbol(s.flags | exportedMember.flags, name);
+                            union.type = getUnionType([getTypeOfSymbol(s), getTypeOfSymbol(exportedMember)]);
+                            union.valueDeclaration = exportedMember.valueDeclaration;
+                            union.declarations = ts.concatenate(exportedMember.declarations, s.declarations);
+                            members_4.set(name, union);
+                        }
+                        else {
+                            members_4.set(name, mergeSymbol(s, exportedMember));
+                        }
+                    }
+                    else {
+                        members_4.set(name, s);
+                    }
+                });
+                var result = createAnonymousType(initialSize !== members_4.size ? undefined : exportedType.symbol, // Only set the type's symbol if it looks to be the same as the original type
+                members_4, exportedType.callSignatures, exportedType.constructSignatures, exportedType.stringIndexInfo, exportedType.numberIndexInfo);
+                result.objectFlags |= (ts.getObjectFlags(type) & 16384 /* JSLiteral */); // Propagate JSLiteral flag
+                if (result.symbol && result.symbol.flags & 32 /* Class */ && type === getDeclaredTypeOfClassOrInterface(result.symbol)) {
+                    result.objectFlags |= 1073741824 /* IsClassInstanceClone */; // Propagate the knowledge that this type is equivalent to the symbol's class instance type
+                }
+                return result;
+            }
+            if (isEmptyArrayLiteralType(type)) {
+                reportImplicitAny(expression, anyArrayType);
+                return anyArrayType;
+            }
+            return type;
         }
-        function typePredicateKindsMatch(a, b) {
-            return a.kind === b.kind && a.parameterIndex === b.parameterIndex;
+        function containsSameNamedThisProperty(thisProperty, expression) {
+            return ts.isPropertyAccessExpression(thisProperty)
+                && thisProperty.expression.kind === 107 /* ThisKeyword */
+                && ts.forEachChildRecursively(expression, function (n) { return isMatchingReference(thisProperty, n); });
         }
-        // This function assumes the constituent type list is sorted and deduplicated.
-        function getUnionTypeFromSortedList(types, objectFlags, aliasSymbol, aliasTypeArguments) {
-            if (types.length === 0) {
-                return neverType;
+        function isDeclarationInConstructor(expression) {
+            var thisContainer = ts.getThisContainer(expression, /*includeArrowFunctions*/ false);
+            // Properties defined in a constructor (or base constructor, or javascript constructor function) don't get undefined added.
+            // Function expressions that are assigned to the prototype count as methods.
+            return thisContainer.kind === 166 /* Constructor */ ||
+                thisContainer.kind === 251 /* FunctionDeclaration */ ||
+                (thisContainer.kind === 208 /* FunctionExpression */ && !ts.isPrototypePropertyAssignment(thisContainer.parent));
+        }
+        function getConstructorDefinedThisAssignmentTypes(types, declarations) {
+            ts.Debug.assert(types.length === declarations.length);
+            return types.filter(function (_, i) {
+                var declaration = declarations[i];
+                var expression = ts.isBinaryExpression(declaration) ? declaration :
+                    ts.isBinaryExpression(declaration.parent) ? declaration.parent : undefined;
+                return expression && isDeclarationInConstructor(expression);
+            });
+        }
+        // Return the type implied by a binding pattern element. This is the type of the initializer of the element if
+        // one is present. Otherwise, if the element is itself a binding pattern, it is the type implied by the binding
+        // pattern. Otherwise, it is the type any.
+        function getTypeFromBindingElement(element, includePatternInType, reportErrors) {
+            if (element.initializer) {
+                // The type implied by a binding pattern is independent of context, so we check the initializer with no
+                // contextual type or, if the element itself is a binding pattern, with the type implied by that binding
+                // pattern.
+                var contextualType = ts.isBindingPattern(element.name) ? getTypeFromBindingPattern(element.name, /*includePatternInType*/ true, /*reportErrors*/ false) : unknownType;
+                return addOptionality(widenTypeInferredFromInitializer(element, checkDeclarationInitializer(element, contextualType)));
             }
-            if (types.length === 1) {
-                return types[0];
+            if (ts.isBindingPattern(element.name)) {
+                return getTypeFromBindingPattern(element.name, includePatternInType, reportErrors);
             }
-            var id = getTypeListId(types);
-            var type = unionTypes.get(id);
-            if (!type) {
-                type = createType(1048576 /* Union */);
-                unionTypes.set(id, type);
-                type.objectFlags = objectFlags | getPropagatingFlagsOfTypes(types, /*excludeKinds*/ 98304 /* Nullable */);
-                type.types = types;
-                /*
-                Note: This is the alias symbol (or lack thereof) that we see when we first encounter this union type.
-                For aliases of identical unions, eg `type T = A | B; type U = A | B`, the symbol of the first alias encountered is the aliasSymbol.
-                (In the language service, the order may depend on the order in which a user takes actions, such as hovering over symbols.)
-                It's important that we create equivalent union types only once, so that's an unfortunate side effect.
-                */
-                type.aliasSymbol = aliasSymbol;
-                type.aliasTypeArguments = aliasTypeArguments;
+            if (reportErrors && !declarationBelongsToPrivateAmbientMember(element)) {
+                reportImplicitAny(element, anyType);
             }
-            return type;
+            // When we're including the pattern in the type (an indication we're obtaining a contextual type), we
+            // use the non-inferrable any type. Inference will never directly infer this type, but it is possible
+            // to infer a type that contains it, e.g. for a binding pattern like [foo] or { foo }. In such cases,
+            // widening of the binding pattern type substitutes a regular any for the non-inferrable any.
+            return includePatternInType ? nonInferrableAnyType : anyType;
         }
-        function getTypeFromUnionTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                var aliasSymbol = getAliasSymbolForTypeNode(node);
-                links.resolvedType = getUnionType(ts.map(node.types, getTypeFromTypeNode), 1 /* Literal */, aliasSymbol, getTypeArgumentsForAliasSymbol(aliasSymbol));
+        // Return the type implied by an object binding pattern
+        function getTypeFromObjectBindingPattern(pattern, includePatternInType, reportErrors) {
+            var members = ts.createSymbolTable();
+            var stringIndexInfo;
+            var objectFlags = 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
+            ts.forEach(pattern.elements, function (e) {
+                var name = e.propertyName || e.name;
+                if (e.dotDotDotToken) {
+                    stringIndexInfo = createIndexInfo(anyType, /*isReadonly*/ false);
+                    return;
+                }
+                var exprType = getLiteralTypeFromPropertyName(name);
+                if (!isTypeUsableAsPropertyName(exprType)) {
+                    // do not include computed properties in the implied type
+                    objectFlags |= 512 /* ObjectLiteralPatternWithComputedProperties */;
+                    return;
+                }
+                var text = getPropertyNameFromType(exprType);
+                var flags = 4 /* Property */ | (e.initializer ? 16777216 /* Optional */ : 0);
+                var symbol = createSymbol(flags, text);
+                symbol.type = getTypeFromBindingElement(e, includePatternInType, reportErrors);
+                symbol.bindingElement = e;
+                members.set(symbol.escapedName, symbol);
+            });
+            var result = createAnonymousType(undefined, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, undefined);
+            result.objectFlags |= objectFlags;
+            if (includePatternInType) {
+                result.pattern = pattern;
+                result.objectFlags |= 1048576 /* ContainsObjectOrArrayLiteral */;
             }
-            return links.resolvedType;
+            return result;
         }
-        function addTypeToIntersection(typeSet, includes, type) {
-            var flags = type.flags;
-            if (flags & 2097152 /* Intersection */) {
-                return addTypesToIntersection(typeSet, includes, type.types);
+        // Return the type implied by an array binding pattern
+        function getTypeFromArrayBindingPattern(pattern, includePatternInType, reportErrors) {
+            var elements = pattern.elements;
+            var lastElement = ts.lastOrUndefined(elements);
+            var restElement = lastElement && lastElement.kind === 198 /* BindingElement */ && lastElement.dotDotDotToken ? lastElement : undefined;
+            if (elements.length === 0 || elements.length === 1 && restElement) {
+                return languageVersion >= 2 /* ES2015 */ ? createIterableType(anyType) : anyArrayType;
             }
-            if (isEmptyAnonymousObjectType(type)) {
-                if (!(includes & 16777216 /* IncludesEmptyObject */)) {
-                    includes |= 16777216 /* IncludesEmptyObject */;
-                    typeSet.set(type.id.toString(), type);
-                }
+            var elementTypes = ts.map(elements, function (e) { return ts.isOmittedExpression(e) ? anyType : getTypeFromBindingElement(e, includePatternInType, reportErrors); });
+            var minLength = ts.findLastIndex(elements, function (e) { return !(e === restElement || ts.isOmittedExpression(e) || hasDefaultValue(e)); }, elements.length - 1) + 1;
+            var elementFlags = ts.map(elements, function (e, i) { return e === restElement ? 4 /* Rest */ : i >= minLength ? 2 /* Optional */ : 1 /* Required */; });
+            var result = createTupleType(elementTypes, elementFlags);
+            if (includePatternInType) {
+                result = cloneTypeReference(result);
+                result.pattern = pattern;
+                result.objectFlags |= 1048576 /* ContainsObjectOrArrayLiteral */;
             }
-            else {
-                if (flags & 3 /* AnyOrUnknown */) {
-                    if (type === wildcardType)
-                        includes |= 8388608 /* IncludesWildcard */;
+            return result;
+        }
+        // Return the type implied by a binding pattern. This is the type implied purely by the binding pattern itself
+        // and without regard to its context (i.e. without regard any type annotation or initializer associated with the
+        // declaration in which the binding pattern is contained). For example, the implied type of [x, y] is [any, any]
+        // and the implied type of { x, y: z = 1 } is { x: any; y: number; }. The type implied by a binding pattern is
+        // used as the contextual type of an initializer associated with the binding pattern. Also, for a destructuring
+        // parameter with no type annotation or initializer, the type implied by the binding pattern becomes the type of
+        // the parameter.
+        function getTypeFromBindingPattern(pattern, includePatternInType, reportErrors) {
+            if (includePatternInType === void 0) { includePatternInType = false; }
+            if (reportErrors === void 0) { reportErrors = false; }
+            return pattern.kind === 196 /* ObjectBindingPattern */
+                ? getTypeFromObjectBindingPattern(pattern, includePatternInType, reportErrors)
+                : getTypeFromArrayBindingPattern(pattern, includePatternInType, reportErrors);
+        }
+        // Return the type associated with a variable, parameter, or property declaration. In the simple case this is the type
+        // specified in a type annotation or inferred from an initializer. However, in the case of a destructuring declaration it
+        // is a bit more involved. For example:
+        //
+        //   var [x, s = ""] = [1, "one"];
+        //
+        // Here, the array literal [1, "one"] is contextually typed by the type [any, string], which is the implied type of the
+        // binding pattern [x, s = ""]. Because the contextual type is a tuple type, the resulting type of [1, "one"] is the
+        // tuple type [number, string]. Thus, the type inferred for 'x' is number and the type inferred for 's' is string.
+        function getWidenedTypeForVariableLikeDeclaration(declaration, reportErrors) {
+            return widenTypeForVariableLikeDeclaration(getTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ true), declaration, reportErrors);
+        }
+        function widenTypeForVariableLikeDeclaration(type, declaration, reportErrors) {
+            if (type) {
+                if (reportErrors) {
+                    reportErrorsFromWidening(declaration, type);
                 }
-                else if ((strictNullChecks || !(flags & 98304 /* Nullable */)) && !typeSet.has(type.id.toString())) {
-                    if (type.flags & 109440 /* Unit */ && includes & 109440 /* Unit */) {
-                        // We have seen two distinct unit types which means we should reduce to an
-                        // empty intersection. Adding TypeFlags.NonPrimitive causes that to happen.
-                        includes |= 67108864 /* NonPrimitive */;
-                    }
-                    typeSet.set(type.id.toString(), type);
+                // always widen a 'unique symbol' type if the type was created for a different declaration.
+                if (type.flags & 8192 /* UniqueESSymbol */ && (ts.isBindingElement(declaration) || !declaration.type) && type.symbol !== getSymbolOfNode(declaration)) {
+                    type = esSymbolType;
                 }
-                includes |= flags & 71041023 /* IncludesMask */;
+                return getWidenedType(type);
             }
-            return includes;
+            // Rest parameters default to type any[], other parameters default to type any
+            type = ts.isParameter(declaration) && declaration.dotDotDotToken ? anyArrayType : anyType;
+            // Report implicit any errors unless this is a private property within an ambient declaration
+            if (reportErrors) {
+                if (!declarationBelongsToPrivateAmbientMember(declaration)) {
+                    reportImplicitAny(declaration, type);
+                }
+            }
+            return type;
         }
-        // Add the given types to the given type set. Order is preserved, freshness is removed from literal
-        // types, duplicates are removed, and nested types of the given kind are flattened into the set.
-        function addTypesToIntersection(typeSet, includes, types) {
-            for (var _i = 0, types_11 = types; _i < types_11.length; _i++) {
-                var type = types_11[_i];
-                includes = addTypeToIntersection(typeSet, includes, getRegularTypeOfLiteralType(type));
+        function declarationBelongsToPrivateAmbientMember(declaration) {
+            var root = ts.getRootDeclaration(declaration);
+            var memberDeclaration = root.kind === 160 /* Parameter */ ? root.parent : root;
+            return isPrivateWithinAmbient(memberDeclaration);
+        }
+        function tryGetTypeFromEffectiveTypeNode(declaration) {
+            var typeNode = ts.getEffectiveTypeAnnotationNode(declaration);
+            if (typeNode) {
+                return getTypeFromTypeNode(typeNode);
             }
-            return includes;
         }
-        function removeRedundantPrimitiveTypes(types, includes) {
-            var i = types.length;
-            while (i > 0) {
-                i--;
-                var t = types[i];
-                var remove = t.flags & 4 /* String */ && includes & 128 /* StringLiteral */ ||
-                    t.flags & 8 /* Number */ && includes & 256 /* NumberLiteral */ ||
-                    t.flags & 64 /* BigInt */ && includes & 2048 /* BigIntLiteral */ ||
-                    t.flags & 4096 /* ESSymbol */ && includes & 8192 /* UniqueESSymbol */;
-                if (remove) {
-                    ts.orderedRemoveItemAt(types, i);
+        function getTypeOfVariableOrParameterOrProperty(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.type) {
+                var type = getTypeOfVariableOrParameterOrPropertyWorker(symbol);
+                // For a contextually typed parameter it is possible that a type has already
+                // been assigned (in assignTypeToParameterAndFixTypeParameters), and we want
+                // to preserve this type.
+                if (!links.type) {
+                    links.type = type;
                 }
             }
+            return links.type;
         }
-        // Check that the given type has a match in every union. A given type is matched by
-        // an identical type, and a literal type is additionally matched by its corresponding
-        // primitive type.
-        function eachUnionContains(unionTypes, type) {
-            for (var _i = 0, unionTypes_1 = unionTypes; _i < unionTypes_1.length; _i++) {
-                var u = unionTypes_1[_i];
-                if (!containsType(u.types, type)) {
-                    var primitive = type.flags & 128 /* StringLiteral */ ? stringType :
-                        type.flags & 256 /* NumberLiteral */ ? numberType :
-                            type.flags & 2048 /* BigIntLiteral */ ? bigintType :
-                                type.flags & 8192 /* UniqueESSymbol */ ? esSymbolType :
-                                    undefined;
-                    if (!primitive || !containsType(u.types, primitive)) {
-                        return false;
-                    }
+        function getTypeOfVariableOrParameterOrPropertyWorker(symbol) {
+            // Handle prototype property
+            if (symbol.flags & 4194304 /* Prototype */) {
+                return getTypeOfPrototypeProperty(symbol);
+            }
+            // CommonsJS require and module both have type any.
+            if (symbol === requireSymbol) {
+                return anyType;
+            }
+            if (symbol.flags & 134217728 /* ModuleExports */) {
+                var fileSymbol = getSymbolOfNode(ts.getSourceFileOfNode(symbol.valueDeclaration));
+                var members = ts.createSymbolTable();
+                members.set("exports", fileSymbol);
+                return createAnonymousType(symbol, members, ts.emptyArray, ts.emptyArray, undefined, undefined);
+            }
+            // Handle catch clause variables
+            var declaration = symbol.valueDeclaration;
+            if (ts.isCatchClauseVariableDeclarationOrBindingElement(declaration)) {
+                var decl = declaration;
+                if (!decl.type)
+                    return anyType;
+                var type_1 = getTypeOfNode(decl.type);
+                // an errorType will make `checkTryStatement` issue an error
+                return isTypeAny(type_1) || type_1 === unknownType ? type_1 : errorType;
+            }
+            // Handle export default expressions
+            if (ts.isSourceFile(declaration) && ts.isJsonSourceFile(declaration)) {
+                if (!declaration.statements.length) {
+                    return emptyObjectType;
                 }
+                return getWidenedType(getWidenedLiteralType(checkExpression(declaration.statements[0].expression)));
             }
-            return true;
-        }
-        function extractIrreducible(types, flag) {
-            if (ts.every(types, function (t) { return !!(t.flags & 1048576 /* Union */) && ts.some(t.types, function (tt) { return !!(tt.flags & flag); }); })) {
-                for (var i = 0; i < types.length; i++) {
-                    types[i] = filterType(types[i], function (t) { return !(t.flags & flag); });
+            // Handle variable, parameter or property
+            if (!pushTypeResolution(symbol, 0 /* Type */)) {
+                // Symbol is property of some kind that is merged with something - should use `getTypeOfFuncClassEnumModule` and not `getTypeOfVariableOrParameterOrProperty`
+                if (symbol.flags & 512 /* ValueModule */ && !(symbol.flags & 67108864 /* Assignment */)) {
+                    return getTypeOfFuncClassEnumModule(symbol);
                 }
-                return true;
+                return reportCircularityError(symbol);
             }
-            return false;
-        }
-        // If the given list of types contains more than one union of primitive types, replace the
-        // first with a union containing an intersection of those primitive types, then remove the
-        // other unions and return true. Otherwise, do nothing and return false.
-        function intersectUnionsOfPrimitiveTypes(types) {
-            var unionTypes;
-            var index = ts.findIndex(types, function (t) { return !!(ts.getObjectFlags(t) & 262144 /* PrimitiveUnion */); });
-            if (index < 0) {
-                return false;
+            var type;
+            if (declaration.kind === 266 /* ExportAssignment */) {
+                type = widenTypeForVariableLikeDeclaration(checkExpressionCached(declaration.expression), declaration);
             }
-            var i = index + 1;
-            // Remove all but the first union of primitive types and collect them in
-            // the unionTypes array.
-            while (i < types.length) {
-                var t = types[i];
-                if (ts.getObjectFlags(t) & 262144 /* PrimitiveUnion */) {
-                    (unionTypes || (unionTypes = [types[index]])).push(t);
-                    ts.orderedRemoveItemAt(types, i);
-                }
-                else {
-                    i++;
+            else if (ts.isBinaryExpression(declaration) ||
+                (ts.isInJSFile(declaration) &&
+                    (ts.isCallExpression(declaration) || (ts.isPropertyAccessExpression(declaration) || ts.isBindableStaticElementAccessExpression(declaration)) && ts.isBinaryExpression(declaration.parent)))) {
+                type = getWidenedTypeForAssignmentDeclaration(symbol);
+            }
+            else if (ts.isPropertyAccessExpression(declaration)
+                || ts.isElementAccessExpression(declaration)
+                || ts.isIdentifier(declaration)
+                || ts.isStringLiteralLike(declaration)
+                || ts.isNumericLiteral(declaration)
+                || ts.isClassDeclaration(declaration)
+                || ts.isFunctionDeclaration(declaration)
+                || (ts.isMethodDeclaration(declaration) && !ts.isObjectLiteralMethod(declaration))
+                || ts.isMethodSignature(declaration)
+                || ts.isSourceFile(declaration)) {
+                // Symbol is property of some kind that is merged with something - should use `getTypeOfFuncClassEnumModule` and not `getTypeOfVariableOrParameterOrProperty`
+                if (symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 384 /* Enum */ | 512 /* ValueModule */)) {
+                    return getTypeOfFuncClassEnumModule(symbol);
                 }
+                type = ts.isBinaryExpression(declaration.parent) ?
+                    getWidenedTypeForAssignmentDeclaration(symbol) :
+                    tryGetTypeFromEffectiveTypeNode(declaration) || anyType;
             }
-            // Return false if there was only one union of primitive types
-            if (!unionTypes) {
-                return false;
+            else if (ts.isPropertyAssignment(declaration)) {
+                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkPropertyAssignment(declaration);
             }
-            // We have more than one union of primitive types, now intersect them. For each
-            // type in each union we check if the type is matched in every union and if so
-            // we include it in the result.
-            var checked = [];
-            var result = [];
-            for (var _i = 0, unionTypes_2 = unionTypes; _i < unionTypes_2.length; _i++) {
-                var u = unionTypes_2[_i];
-                for (var _a = 0, _b = u.types; _a < _b.length; _a++) {
-                    var t = _b[_a];
-                    if (insertType(checked, t)) {
-                        if (eachUnionContains(unionTypes, t)) {
-                            insertType(result, t);
-                        }
-                    }
-                }
+            else if (ts.isJsxAttribute(declaration)) {
+                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkJsxAttribute(declaration);
             }
-            // Finally replace the first union with the result
-            types[index] = getUnionTypeFromSortedList(result, 262144 /* PrimitiveUnion */);
-            return true;
-        }
-        function createIntersectionType(types, aliasSymbol, aliasTypeArguments) {
-            var result = createType(2097152 /* Intersection */);
-            result.objectFlags = getPropagatingFlagsOfTypes(types, /*excludeKinds*/ 98304 /* Nullable */);
-            result.types = types;
-            result.aliasSymbol = aliasSymbol; // See comment in `getUnionTypeFromSortedList`.
-            result.aliasTypeArguments = aliasTypeArguments;
-            return result;
-        }
-        // We normalize combinations of intersection and union types based on the distributive property of the '&'
-        // operator. Specifically, because X & (A | B) is equivalent to X & A | X & B, we can transform intersection
-        // types with union type constituents into equivalent union types with intersection type constituents and
-        // effectively ensure that union types are always at the top level in type representations.
-        //
-        // We do not perform structural deduplication on intersection types. Intersection types are created only by the &
-        // type operator and we can't reduce those because we want to support recursive intersection types. For example,
-        // a type alias of the form "type List<T> = T & { next: List<T> }" cannot be reduced during its declaration.
-        // Also, unlike union types, the order of the constituent types is preserved in order that overload resolution
-        // for intersections of types with signatures can be deterministic.
-        function getIntersectionType(types, aliasSymbol, aliasTypeArguments) {
-            var typeMembershipMap = ts.createMap();
-            var includes = addTypesToIntersection(typeMembershipMap, 0, types);
-            var typeSet = ts.arrayFrom(typeMembershipMap.values());
-            // An intersection type is considered empty if it contains
-            // the type never, or
-            // more than one unit type or,
-            // an object type and a nullable type (null or undefined), or
-            // a string-like type and a type known to be non-string-like, or
-            // a number-like type and a type known to be non-number-like, or
-            // a symbol-like type and a type known to be non-symbol-like, or
-            // a void-like type and a type known to be non-void-like, or
-            // a non-primitive type and a type known to be primitive.
-            if (includes & 131072 /* Never */ ||
-                strictNullChecks && includes & 98304 /* Nullable */ && includes & (524288 /* Object */ | 67108864 /* NonPrimitive */ | 16777216 /* IncludesEmptyObject */) ||
-                includes & 67108864 /* NonPrimitive */ && includes & (67238908 /* DisjointDomains */ & ~67108864 /* NonPrimitive */) ||
-                includes & 132 /* StringLike */ && includes & (67238908 /* DisjointDomains */ & ~132 /* StringLike */) ||
-                includes & 296 /* NumberLike */ && includes & (67238908 /* DisjointDomains */ & ~296 /* NumberLike */) ||
-                includes & 2112 /* BigIntLike */ && includes & (67238908 /* DisjointDomains */ & ~2112 /* BigIntLike */) ||
-                includes & 12288 /* ESSymbolLike */ && includes & (67238908 /* DisjointDomains */ & ~12288 /* ESSymbolLike */) ||
-                includes & 49152 /* VoidLike */ && includes & (67238908 /* DisjointDomains */ & ~49152 /* VoidLike */)) {
-                return neverType;
+            else if (ts.isShorthandPropertyAssignment(declaration)) {
+                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkExpressionForMutableLocation(declaration.name, 0 /* Normal */);
             }
-            if (includes & 1 /* Any */) {
-                return includes & 8388608 /* IncludesWildcard */ ? wildcardType : anyType;
+            else if (ts.isObjectLiteralMethod(declaration)) {
+                type = tryGetTypeFromEffectiveTypeNode(declaration) || checkObjectLiteralMethod(declaration, 0 /* Normal */);
             }
-            if (!strictNullChecks && includes & 98304 /* Nullable */) {
-                return includes & 32768 /* Undefined */ ? undefinedType : nullType;
+            else if (ts.isParameter(declaration)
+                || ts.isPropertyDeclaration(declaration)
+                || ts.isPropertySignature(declaration)
+                || ts.isVariableDeclaration(declaration)
+                || ts.isBindingElement(declaration)
+                || ts.isJSDocPropertyLikeTag(declaration)) {
+                type = getWidenedTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ true);
             }
-            if (includes & 4 /* String */ && includes & 128 /* StringLiteral */ ||
-                includes & 8 /* Number */ && includes & 256 /* NumberLiteral */ ||
-                includes & 64 /* BigInt */ && includes & 2048 /* BigIntLiteral */ ||
-                includes & 4096 /* ESSymbol */ && includes & 8192 /* UniqueESSymbol */) {
-                removeRedundantPrimitiveTypes(typeSet, includes);
+            // getTypeOfSymbol dispatches some JS merges incorrectly because their symbol flags are not mutually exclusive.
+            // Re-dispatch based on valueDeclaration.kind instead.
+            else if (ts.isEnumDeclaration(declaration)) {
+                type = getTypeOfFuncClassEnumModule(symbol);
             }
-            if (includes & 16777216 /* IncludesEmptyObject */ && includes & 524288 /* Object */) {
-                ts.orderedRemoveItemAt(typeSet, ts.findIndex(typeSet, isEmptyAnonymousObjectType));
+            else if (ts.isEnumMember(declaration)) {
+                type = getTypeOfEnumMember(symbol);
             }
-            if (typeSet.length === 0) {
-                return unknownType;
+            else if (ts.isAccessor(declaration)) {
+                type = resolveTypeOfAccessors(symbol);
             }
-            if (typeSet.length === 1) {
-                return typeSet[0];
+            else {
+                return ts.Debug.fail("Unhandled declaration kind! " + ts.Debug.formatSyntaxKind(declaration.kind) + " for " + ts.Debug.formatSymbol(symbol));
             }
-            var id = getTypeListId(typeSet);
-            var result = intersectionTypes.get(id);
-            if (!result) {
-                if (includes & 1048576 /* Union */) {
-                    if (intersectUnionsOfPrimitiveTypes(typeSet)) {
-                        // When the intersection creates a reduced set (which might mean that *all* union types have
-                        // disappeared), we restart the operation to get a new set of combined flags. Once we have
-                        // reduced we'll never reduce again, so this occurs at most once.
-                        result = getIntersectionType(typeSet, aliasSymbol, aliasTypeArguments);
-                    }
-                    else if (extractIrreducible(typeSet, 32768 /* Undefined */)) {
-                        result = getUnionType([getIntersectionType(typeSet), undefinedType], 1 /* Literal */, aliasSymbol, aliasTypeArguments);
-                    }
-                    else if (extractIrreducible(typeSet, 65536 /* Null */)) {
-                        result = getUnionType([getIntersectionType(typeSet), nullType], 1 /* Literal */, aliasSymbol, aliasTypeArguments);
-                    }
-                    else {
-                        // We are attempting to construct a type of the form X & (A | B) & Y. Transform this into a type of
-                        // the form X & A & Y | X & B & Y and recursively reduce until no union type constituents remain.
-                        // If the estimated size of the resulting union type exceeds 100000 constituents, report an error.
-                        var size = ts.reduceLeft(typeSet, function (n, t) { return n * (t.flags & 1048576 /* Union */ ? t.types.length : 1); }, 1);
-                        if (size >= 100000) {
-                            error(currentNode, ts.Diagnostics.Expression_produces_a_union_type_that_is_too_complex_to_represent);
-                            return errorType;
-                        }
-                        var unionIndex_1 = ts.findIndex(typeSet, function (t) { return (t.flags & 1048576 /* Union */) !== 0; });
-                        var unionType = typeSet[unionIndex_1];
-                        result = getUnionType(ts.map(unionType.types, function (t) { return getIntersectionType(ts.replaceElement(typeSet, unionIndex_1, t)); }), 1 /* Literal */, aliasSymbol, aliasTypeArguments);
-                    }
-                }
-                else {
-                    result = createIntersectionType(typeSet, aliasSymbol, aliasTypeArguments);
+            if (!popTypeResolution()) {
+                // Symbol is property of some kind that is merged with something - should use `getTypeOfFuncClassEnumModule` and not `getTypeOfVariableOrParameterOrProperty`
+                if (symbol.flags & 512 /* ValueModule */ && !(symbol.flags & 67108864 /* Assignment */)) {
+                    return getTypeOfFuncClassEnumModule(symbol);
                 }
-                intersectionTypes.set(id, result);
-            }
-            return result;
-        }
-        function getTypeFromIntersectionTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                var aliasSymbol = getAliasSymbolForTypeNode(node);
-                links.resolvedType = getIntersectionType(ts.map(node.types, getTypeFromTypeNode), aliasSymbol, getTypeArgumentsForAliasSymbol(aliasSymbol));
-            }
-            return links.resolvedType;
-        }
-        function createIndexType(type, stringsOnly) {
-            var result = createType(4194304 /* Index */);
-            result.type = type;
-            result.stringsOnly = stringsOnly;
-            return result;
-        }
-        function getIndexTypeForGenericType(type, stringsOnly) {
-            return stringsOnly ?
-                type.resolvedStringIndexType || (type.resolvedStringIndexType = createIndexType(type, /*stringsOnly*/ true)) :
-                type.resolvedIndexType || (type.resolvedIndexType = createIndexType(type, /*stringsOnly*/ false));
-        }
-        function getLiteralTypeFromPropertyName(name) {
-            if (ts.isPrivateIdentifier(name)) {
-                return neverType;
+                return reportCircularityError(symbol);
             }
-            return ts.isIdentifier(name) ? getLiteralType(ts.unescapeLeadingUnderscores(name.escapedText)) :
-                getRegularTypeOfLiteralType(ts.isComputedPropertyName(name) ? checkComputedPropertyName(name) : checkExpression(name));
-        }
-        function getBigIntLiteralType(node) {
-            return getLiteralType({
-                negative: false,
-                base10Value: ts.parsePseudoBigInt(node.text)
-            });
+            return type;
         }
-        function getLiteralTypeFromProperty(prop, include) {
-            if (!(ts.getDeclarationModifierFlagsFromSymbol(prop) & 24 /* NonPublicAccessibilityModifier */)) {
-                var type = getSymbolLinks(getLateBoundSymbol(prop)).nameType;
-                if (!type && !ts.isKnownSymbol(prop)) {
-                    if (prop.escapedName === "default" /* Default */) {
-                        type = getLiteralType("default");
-                    }
-                    else {
-                        var name = prop.valueDeclaration && ts.getNameOfDeclaration(prop.valueDeclaration);
-                        type = name && getLiteralTypeFromPropertyName(name) || getLiteralType(ts.symbolName(prop));
-                    }
+        function getAnnotatedAccessorTypeNode(accessor) {
+            if (accessor) {
+                if (accessor.kind === 167 /* GetAccessor */) {
+                    var getterTypeAnnotation = ts.getEffectiveReturnTypeNode(accessor);
+                    return getterTypeAnnotation;
                 }
-                if (type && type.flags & include) {
-                    return type;
+                else {
+                    var setterTypeAnnotation = ts.getEffectiveSetAccessorTypeAnnotationNode(accessor);
+                    return setterTypeAnnotation;
                 }
             }
-            return neverType;
-        }
-        function getLiteralTypeFromProperties(type, include) {
-            return getUnionType(ts.map(getPropertiesOfType(type), function (p) { return getLiteralTypeFromProperty(p, include); }));
+            return undefined;
         }
-        function getNonEnumNumberIndexInfo(type) {
-            var numberIndexInfo = getIndexInfoOfType(type, 1 /* Number */);
-            return numberIndexInfo !== enumNumberIndexInfo ? numberIndexInfo : undefined;
+        function getAnnotatedAccessorType(accessor) {
+            var node = getAnnotatedAccessorTypeNode(accessor);
+            return node && getTypeFromTypeNode(node);
         }
-        function getIndexType(type, stringsOnly, noIndexSignatures) {
-            if (stringsOnly === void 0) { stringsOnly = keyofStringsOnly; }
-            type = getReducedType(type);
-            return type.flags & 1048576 /* Union */ ? getIntersectionType(ts.map(type.types, function (t) { return getIndexType(t, stringsOnly, noIndexSignatures); })) :
-                type.flags & 2097152 /* Intersection */ ? getUnionType(ts.map(type.types, function (t) { return getIndexType(t, stringsOnly, noIndexSignatures); })) :
-                    maybeTypeOfKind(type, 58982400 /* InstantiableNonPrimitive */) ? getIndexTypeForGenericType(type, stringsOnly) :
-                        ts.getObjectFlags(type) & 32 /* Mapped */ ? filterType(getConstraintTypeFromMappedType(type), function (t) { return !(noIndexSignatures && t.flags & (1 /* Any */ | 4 /* String */)); }) :
-                            type === wildcardType ? wildcardType :
-                                type.flags & 2 /* Unknown */ ? neverType :
-                                    type.flags & (1 /* Any */ | 131072 /* Never */) ? keyofConstraintType :
-                                        stringsOnly ? !noIndexSignatures && getIndexInfoOfType(type, 0 /* String */) ? stringType : getLiteralTypeFromProperties(type, 128 /* StringLiteral */) :
-                                            !noIndexSignatures && getIndexInfoOfType(type, 0 /* String */) ? getUnionType([stringType, numberType, getLiteralTypeFromProperties(type, 8192 /* UniqueESSymbol */)]) :
-                                                getNonEnumNumberIndexInfo(type) ? getUnionType([numberType, getLiteralTypeFromProperties(type, 128 /* StringLiteral */ | 8192 /* UniqueESSymbol */)]) :
-                                                    getLiteralTypeFromProperties(type, 8576 /* StringOrNumberLiteralOrUnique */);
+        function getAnnotatedAccessorThisParameter(accessor) {
+            var parameter = getAccessorThisParameter(accessor);
+            return parameter && parameter.symbol;
         }
-        function getExtractStringType(type) {
-            if (keyofStringsOnly) {
-                return type;
-            }
-            var extractTypeAlias = getGlobalExtractSymbol();
-            return extractTypeAlias ? getTypeAliasInstantiation(extractTypeAlias, [type, stringType]) : stringType;
+        function getThisTypeOfDeclaration(declaration) {
+            return getThisTypeOfSignature(getSignatureFromDeclaration(declaration));
         }
-        function getIndexTypeOrString(type) {
-            var indexType = getExtractStringType(getIndexType(type));
-            return indexType.flags & 131072 /* Never */ ? stringType : indexType;
+        function getTypeOfAccessors(symbol) {
+            var links = getSymbolLinks(symbol);
+            return links.type || (links.type = getTypeOfAccessorsWorker(symbol));
         }
-        function getTypeFromTypeOperatorNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                switch (node.operator) {
-                    case 134 /* KeyOfKeyword */:
-                        links.resolvedType = getIndexType(getTypeFromTypeNode(node.type));
-                        break;
-                    case 147 /* UniqueKeyword */:
-                        links.resolvedType = node.type.kind === 144 /* SymbolKeyword */
-                            ? getESSymbolLikeTypeForNode(ts.walkUpParenthesizedTypes(node.parent))
-                            : errorType;
-                        break;
-                    case 138 /* ReadonlyKeyword */:
-                        links.resolvedType = getTypeFromTypeNode(node.type);
-                        break;
-                    default:
-                        throw ts.Debug.assertNever(node.operator);
+        function getTypeOfAccessorsWorker(symbol) {
+            if (!pushTypeResolution(symbol, 0 /* Type */)) {
+                return errorType;
+            }
+            var type = resolveTypeOfAccessors(symbol);
+            if (!popTypeResolution()) {
+                type = anyType;
+                if (noImplicitAny) {
+                    var getter = ts.getDeclarationOfKind(symbol, 167 /* GetAccessor */);
+                    error(getter, ts.Diagnostics._0_implicitly_has_return_type_any_because_it_does_not_have_a_return_type_annotation_and_is_referenced_directly_or_indirectly_in_one_of_its_return_expressions, symbolToString(symbol));
                 }
             }
-            return links.resolvedType;
-        }
-        function createIndexedAccessType(objectType, indexType) {
-            var type = createType(8388608 /* IndexedAccess */);
-            type.objectType = objectType;
-            type.indexType = indexType;
             return type;
         }
-        /**
-         * Returns if a type is or consists of a JSLiteral object type
-         * In addition to objects which are directly literals,
-         * * unions where every element is a jsliteral
-         * * intersections where at least one element is a jsliteral
-         * * and instantiable types constrained to a jsliteral
-         * Should all count as literals and not print errors on access or assignment of possibly existing properties.
-         * This mirrors the behavior of the index signature propagation, to which this behaves similarly (but doesn't affect assignability or inference).
-         */
-        function isJSLiteralType(type) {
-            if (noImplicitAny) {
-                return false; // Flag is meaningless under `noImplicitAny` mode
-            }
-            if (ts.getObjectFlags(type) & 16384 /* JSLiteral */) {
-                return true;
-            }
-            if (type.flags & 1048576 /* Union */) {
-                return ts.every(type.types, isJSLiteralType);
-            }
-            if (type.flags & 2097152 /* Intersection */) {
-                return ts.some(type.types, isJSLiteralType);
+        function resolveTypeOfAccessors(symbol) {
+            var getter = ts.getDeclarationOfKind(symbol, 167 /* GetAccessor */);
+            var setter = ts.getDeclarationOfKind(symbol, 168 /* SetAccessor */);
+            if (getter && ts.isInJSFile(getter)) {
+                var jsDocType = getTypeForDeclarationFromJSDocComment(getter);
+                if (jsDocType) {
+                    return jsDocType;
+                }
             }
-            if (type.flags & 63176704 /* Instantiable */) {
-                return isJSLiteralType(getResolvedBaseConstraint(type));
+            // First try to see if the user specified a return type on the get-accessor.
+            var getterReturnType = getAnnotatedAccessorType(getter);
+            if (getterReturnType) {
+                return getterReturnType;
             }
-            return false;
-        }
-        function getPropertyNameFromIndex(indexType, accessNode) {
-            var accessExpression = accessNode && accessNode.kind === 195 /* ElementAccessExpression */ ? accessNode : undefined;
-            return isTypeUsableAsPropertyName(indexType) ?
-                getPropertyNameFromType(indexType) :
-                accessExpression && checkThatExpressionIsProperSymbolReference(accessExpression.argumentExpression, indexType, /*reportError*/ false) ?
-                    ts.getPropertyNameForKnownSymbolName(ts.idText(accessExpression.argumentExpression.name)) :
-                    accessNode && ts.isPropertyName(accessNode) ?
-                        // late bound names are handled in the first branch, so here we only need to handle normal names
-                        ts.getPropertyNameForPropertyNameNode(accessNode) :
-                        undefined;
-        }
-        function getPropertyTypeForIndexType(originalObjectType, objectType, indexType, fullIndexType, suppressNoImplicitAnyError, accessNode, accessFlags) {
-            var accessExpression = accessNode && accessNode.kind === 195 /* ElementAccessExpression */ ? accessNode : undefined;
-            var propName = accessNode && ts.isPrivateIdentifier(accessNode) ? undefined : getPropertyNameFromIndex(indexType, accessNode);
-            if (propName !== undefined) {
-                var prop = getPropertyOfType(objectType, propName);
-                if (prop) {
-                    if (accessExpression) {
-                        markPropertyAsReferenced(prop, accessExpression, /*isThisAccess*/ accessExpression.expression.kind === 104 /* ThisKeyword */);
-                        if (isAssignmentToReadonlyEntity(accessExpression, prop, ts.getAssignmentTargetKind(accessExpression))) {
-                            error(accessExpression.argumentExpression, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_read_only_property, symbolToString(prop));
-                            return undefined;
-                        }
-                        if (accessFlags & 4 /* CacheSymbol */) {
-                            getNodeLinks(accessNode).resolvedSymbol = prop;
-                        }
-                    }
-                    var propType = getTypeOfSymbol(prop);
-                    return accessExpression && ts.getAssignmentTargetKind(accessExpression) !== 1 /* Definite */ ?
-                        getFlowTypeOfReference(accessExpression, propType) :
-                        propType;
+            else {
+                // If the user didn't specify a return type, try to use the set-accessor's parameter type.
+                var setterParameterType = getAnnotatedAccessorType(setter);
+                if (setterParameterType) {
+                    return setterParameterType;
                 }
-                if (everyType(objectType, isTupleType) && isNumericLiteralName(propName) && +propName >= 0) {
-                    if (accessNode && everyType(objectType, function (t) { return !t.target.hasRestElement; }) && !(accessFlags & 8 /* NoTupleBoundsCheck */)) {
-                        var indexNode = getIndexNodeForAccessExpression(accessNode);
-                        if (isTupleType(objectType)) {
-                            error(indexNode, ts.Diagnostics.Tuple_type_0_of_length_1_has_no_element_at_index_2, typeToString(objectType), getTypeReferenceArity(objectType), ts.unescapeLeadingUnderscores(propName));
+                else {
+                    // If there are no specified types, try to infer it from the body of the get accessor if it exists.
+                    if (getter && getter.body) {
+                        return getReturnTypeFromBody(getter);
+                    }
+                    // Otherwise, fall back to 'any'.
+                    else {
+                        if (setter) {
+                            if (!isPrivateWithinAmbient(setter)) {
+                                errorOrSuggestion(noImplicitAny, setter, ts.Diagnostics.Property_0_implicitly_has_type_any_because_its_set_accessor_lacks_a_parameter_type_annotation, symbolToString(symbol));
+                            }
                         }
                         else {
-                            error(indexNode, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.unescapeLeadingUnderscores(propName), typeToString(objectType));
+                            ts.Debug.assert(!!getter, "there must exist a getter as we are current checking either setter or getter in this function");
+                            if (!isPrivateWithinAmbient(getter)) {
+                                errorOrSuggestion(noImplicitAny, getter, ts.Diagnostics.Property_0_implicitly_has_type_any_because_its_get_accessor_lacks_a_return_type_annotation, symbolToString(symbol));
+                            }
                         }
+                        return anyType;
                     }
-                    errorIfWritingToReadonlyIndex(getIndexInfoOfType(objectType, 1 /* Number */));
-                    return mapType(objectType, function (t) { return getRestTypeOfTupleType(t) || undefinedType; });
                 }
             }
-            if (!(indexType.flags & 98304 /* Nullable */) && isTypeAssignableToKind(indexType, 132 /* StringLike */ | 296 /* NumberLike */ | 12288 /* ESSymbolLike */)) {
-                if (objectType.flags & (1 /* Any */ | 131072 /* Never */)) {
-                    return objectType;
-                }
-                var stringIndexInfo = getIndexInfoOfType(objectType, 0 /* String */);
-                var indexInfo = isTypeAssignableToKind(indexType, 296 /* NumberLike */) && getIndexInfoOfType(objectType, 1 /* Number */) || stringIndexInfo;
-                if (indexInfo) {
-                    if (accessFlags & 1 /* NoIndexSignatures */ && indexInfo === stringIndexInfo) {
-                        if (accessExpression) {
-                            error(accessExpression, ts.Diagnostics.Type_0_cannot_be_used_to_index_type_1, typeToString(indexType), typeToString(originalObjectType));
-                        }
-                        return undefined;
-                    }
-                    if (accessNode && !isTypeAssignableToKind(indexType, 4 /* String */ | 8 /* Number */)) {
-                        var indexNode = getIndexNodeForAccessExpression(accessNode);
-                        error(indexNode, ts.Diagnostics.Type_0_cannot_be_used_as_an_index_type, typeToString(indexType));
-                        return indexInfo.type;
-                    }
-                    errorIfWritingToReadonlyIndex(indexInfo);
-                    return indexInfo.type;
-                }
-                if (indexType.flags & 131072 /* Never */) {
-                    return neverType;
-                }
-                if (isJSLiteralType(objectType)) {
-                    return anyType;
-                }
-                if (accessExpression && !isConstEnumObjectType(objectType)) {
-                    if (objectType.symbol === globalThisSymbol && propName !== undefined && globalThisSymbol.exports.has(propName) && (globalThisSymbol.exports.get(propName).flags & 418 /* BlockScoped */)) {
-                        error(accessExpression, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.unescapeLeadingUnderscores(propName), typeToString(objectType));
-                    }
-                    else if (noImplicitAny && !compilerOptions.suppressImplicitAnyIndexErrors && !suppressNoImplicitAnyError) {
-                        if (propName !== undefined && typeHasStaticProperty(propName, objectType)) {
-                            error(accessExpression, ts.Diagnostics.Property_0_is_a_static_member_of_type_1, propName, typeToString(objectType));
-                        }
-                        else if (getIndexTypeOfType(objectType, 1 /* Number */)) {
-                            error(accessExpression.argumentExpression, ts.Diagnostics.Element_implicitly_has_an_any_type_because_index_expression_is_not_of_type_number);
-                        }
-                        else {
-                            var suggestion = void 0;
-                            if (propName !== undefined && (suggestion = getSuggestionForNonexistentProperty(propName, objectType))) {
-                                if (suggestion !== undefined) {
-                                    error(accessExpression.argumentExpression, ts.Diagnostics.Property_0_does_not_exist_on_type_1_Did_you_mean_2, propName, typeToString(objectType), suggestion);
-                                }
-                            }
-                            else {
-                                var suggestion_1 = getSuggestionForNonexistentIndexSignature(objectType, accessExpression, indexType);
-                                if (suggestion_1 !== undefined) {
-                                    error(accessExpression, ts.Diagnostics.Element_implicitly_has_an_any_type_because_type_0_has_no_index_signature_Did_you_mean_to_call_1, typeToString(objectType), suggestion_1);
-                                }
-                                else {
-                                    var errorInfo = void 0;
-                                    if (indexType.flags & 1024 /* EnumLiteral */) {
-                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, "[" + typeToString(indexType) + "]", typeToString(objectType));
-                                    }
-                                    else if (indexType.flags & 8192 /* UniqueESSymbol */) {
-                                        var symbolName_2 = getFullyQualifiedName(indexType.symbol, accessExpression);
-                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, "[" + symbolName_2 + "]", typeToString(objectType));
-                                    }
-                                    else if (indexType.flags & 128 /* StringLiteral */) {
-                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, indexType.value, typeToString(objectType));
-                                    }
-                                    else if (indexType.flags & 256 /* NumberLiteral */) {
-                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, indexType.value, typeToString(objectType));
-                                    }
-                                    else if (indexType.flags & (8 /* Number */ | 4 /* String */)) {
-                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.No_index_signature_with_a_parameter_of_type_0_was_found_on_type_1, typeToString(indexType), typeToString(objectType));
-                                    }
-                                    errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Element_implicitly_has_an_any_type_because_expression_of_type_0_can_t_be_used_to_index_type_1, typeToString(fullIndexType), typeToString(objectType));
-                                    diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(accessExpression, errorInfo));
-                                }
-                            }
-                        }
+        }
+        function getBaseTypeVariableOfClass(symbol) {
+            var baseConstructorType = getBaseConstructorTypeOfClass(getDeclaredTypeOfClassOrInterface(symbol));
+            return baseConstructorType.flags & 8650752 /* TypeVariable */ ? baseConstructorType :
+                baseConstructorType.flags & 2097152 /* Intersection */ ? ts.find(baseConstructorType.types, function (t) { return !!(t.flags & 8650752 /* TypeVariable */); }) :
+                    undefined;
+        }
+        function getTypeOfFuncClassEnumModule(symbol) {
+            var links = getSymbolLinks(symbol);
+            var originalLinks = links;
+            if (!links.type) {
+                var expando = symbol.valueDeclaration && getSymbolOfExpando(symbol.valueDeclaration, /*allowDeclaration*/ false);
+                if (expando) {
+                    var merged = mergeJSSymbols(symbol, expando);
+                    if (merged) {
+                        // note:we overwrite links because we just cloned the symbol
+                        symbol = links = merged;
                     }
-                    return undefined;
                 }
+                originalLinks.type = links.type = getTypeOfFuncClassEnumModuleWorker(symbol);
             }
-            if (isJSLiteralType(objectType)) {
+            return links.type;
+        }
+        function getTypeOfFuncClassEnumModuleWorker(symbol) {
+            var declaration = symbol.valueDeclaration;
+            if (symbol.flags & 1536 /* Module */ && ts.isShorthandAmbientModuleSymbol(symbol)) {
                 return anyType;
             }
-            if (accessNode) {
-                var indexNode = getIndexNodeForAccessExpression(accessNode);
-                if (indexType.flags & (128 /* StringLiteral */ | 256 /* NumberLiteral */)) {
-                    error(indexNode, ts.Diagnostics.Property_0_does_not_exist_on_type_1, "" + indexType.value, typeToString(objectType));
-                }
-                else if (indexType.flags & (4 /* String */ | 8 /* Number */)) {
-                    error(indexNode, ts.Diagnostics.Type_0_has_no_matching_index_signature_for_type_1, typeToString(objectType), typeToString(indexType));
-                }
-                else {
-                    error(indexNode, ts.Diagnostics.Type_0_cannot_be_used_as_an_index_type, typeToString(indexType));
+            else if (declaration && (declaration.kind === 216 /* BinaryExpression */ ||
+                ts.isAccessExpression(declaration) &&
+                    declaration.parent.kind === 216 /* BinaryExpression */)) {
+                return getWidenedTypeForAssignmentDeclaration(symbol);
+            }
+            else if (symbol.flags & 512 /* ValueModule */ && declaration && ts.isSourceFile(declaration) && declaration.commonJsModuleIndicator) {
+                var resolvedModule = resolveExternalModuleSymbol(symbol);
+                if (resolvedModule !== symbol) {
+                    if (!pushTypeResolution(symbol, 0 /* Type */)) {
+                        return errorType;
+                    }
+                    var exportEquals = getMergedSymbol(symbol.exports.get("export=" /* ExportEquals */));
+                    var type_2 = getWidenedTypeForAssignmentDeclaration(exportEquals, exportEquals === resolvedModule ? undefined : resolvedModule);
+                    if (!popTypeResolution()) {
+                        return reportCircularityError(symbol);
+                    }
+                    return type_2;
                 }
             }
-            if (isTypeAny(indexType)) {
-                return indexType;
+            var type = createObjectType(16 /* Anonymous */, symbol);
+            if (symbol.flags & 32 /* Class */) {
+                var baseTypeVariable = getBaseTypeVariableOfClass(symbol);
+                return baseTypeVariable ? getIntersectionType([type, baseTypeVariable]) : type;
             }
-            return undefined;
-            function errorIfWritingToReadonlyIndex(indexInfo) {
-                if (indexInfo && indexInfo.isReadonly && accessExpression && (ts.isAssignmentTarget(accessExpression) || ts.isDeleteTarget(accessExpression))) {
-                    error(accessExpression, ts.Diagnostics.Index_signature_in_type_0_only_permits_reading, typeToString(objectType));
-                }
+            else {
+                return strictNullChecks && symbol.flags & 16777216 /* Optional */ ? getOptionalType(type) : type;
             }
         }
-        function getIndexNodeForAccessExpression(accessNode) {
-            return accessNode.kind === 195 /* ElementAccessExpression */ ? accessNode.argumentExpression :
-                accessNode.kind === 185 /* IndexedAccessType */ ? accessNode.indexType :
-                    accessNode.kind === 154 /* ComputedPropertyName */ ? accessNode.expression :
-                        accessNode;
+        function getTypeOfEnumMember(symbol) {
+            var links = getSymbolLinks(symbol);
+            return links.type || (links.type = getDeclaredTypeOfEnumMember(symbol));
         }
-        function isGenericObjectType(type) {
-            if (type.flags & 3145728 /* UnionOrIntersection */) {
-                if (!(type.objectFlags & 4194304 /* IsGenericObjectTypeComputed */)) {
-                    type.objectFlags |= 4194304 /* IsGenericObjectTypeComputed */ |
-                        (ts.some(type.types, isGenericObjectType) ? 8388608 /* IsGenericObjectType */ : 0);
-                }
-                return !!(type.objectFlags & 8388608 /* IsGenericObjectType */);
+        function getTypeOfAlias(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.type) {
+                var targetSymbol = resolveAlias(symbol);
+                // It only makes sense to get the type of a value symbol. If the result of resolving
+                // the alias is not a value, then it has no type. To get the type associated with a
+                // type symbol, call getDeclaredTypeOfSymbol.
+                // This check is important because without it, a call to getTypeOfSymbol could end
+                // up recursively calling getTypeOfAlias, causing a stack overflow.
+                links.type = targetSymbol.flags & 111551 /* Value */
+                    ? getTypeOfSymbol(targetSymbol)
+                    : errorType;
             }
-            return !!(type.flags & 58982400 /* InstantiableNonPrimitive */) || isGenericMappedType(type);
+            return links.type;
         }
-        function isGenericIndexType(type) {
-            if (type.flags & 3145728 /* UnionOrIntersection */) {
-                if (!(type.objectFlags & 16777216 /* IsGenericIndexTypeComputed */)) {
-                    type.objectFlags |= 16777216 /* IsGenericIndexTypeComputed */ |
-                        (ts.some(type.types, isGenericIndexType) ? 33554432 /* IsGenericIndexType */ : 0);
+        function getTypeOfInstantiatedSymbol(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.type) {
+                if (!pushTypeResolution(symbol, 0 /* Type */)) {
+                    return links.type = errorType;
                 }
-                return !!(type.objectFlags & 33554432 /* IsGenericIndexType */);
+                var type = instantiateType(getTypeOfSymbol(links.target), links.mapper);
+                if (!popTypeResolution()) {
+                    type = reportCircularityError(symbol);
+                }
+                links.type = type;
             }
-            return !!(type.flags & (58982400 /* InstantiableNonPrimitive */ | 4194304 /* Index */));
-        }
-        function isThisTypeParameter(type) {
-            return !!(type.flags & 262144 /* TypeParameter */ && type.isThisType);
-        }
-        function getSimplifiedType(type, writing) {
-            return type.flags & 8388608 /* IndexedAccess */ ? getSimplifiedIndexedAccessType(type, writing) :
-                type.flags & 16777216 /* Conditional */ ? getSimplifiedConditionalType(type, writing) :
-                    type;
+            return links.type;
         }
-        function distributeIndexOverObjectType(objectType, indexType, writing) {
-            // (T | U)[K] -> T[K] | U[K] (reading)
-            // (T | U)[K] -> T[K] & U[K] (writing)
-            // (T & U)[K] -> T[K] & U[K]
-            if (objectType.flags & 3145728 /* UnionOrIntersection */) {
-                var types = ts.map(objectType.types, function (t) { return getSimplifiedType(getIndexedAccessType(t, indexType), writing); });
-                return objectType.flags & 2097152 /* Intersection */ || writing ? getIntersectionType(types) : getUnionType(types);
+        function reportCircularityError(symbol) {
+            var declaration = symbol.valueDeclaration;
+            // Check if variable has type annotation that circularly references the variable itself
+            if (ts.getEffectiveTypeAnnotationNode(declaration)) {
+                error(symbol.valueDeclaration, ts.Diagnostics._0_is_referenced_directly_or_indirectly_in_its_own_type_annotation, symbolToString(symbol));
+                return errorType;
             }
-        }
-        function distributeObjectOverIndexType(objectType, indexType, writing) {
-            // T[A | B] -> T[A] | T[B] (reading)
-            // T[A | B] -> T[A] & T[B] (writing)
-            if (indexType.flags & 1048576 /* Union */) {
-                var types = ts.map(indexType.types, function (t) { return getSimplifiedType(getIndexedAccessType(objectType, t), writing); });
-                return writing ? getIntersectionType(types) : getUnionType(types);
+            // Check if variable has initializer that circularly references the variable itself
+            if (noImplicitAny && (declaration.kind !== 160 /* Parameter */ || declaration.initializer)) {
+                error(symbol.valueDeclaration, ts.Diagnostics._0_implicitly_has_type_any_because_it_does_not_have_a_type_annotation_and_is_referenced_directly_or_indirectly_in_its_own_initializer, symbolToString(symbol));
             }
+            // Circularities could also result from parameters in function expressions that end up
+            // having themselves as contextual types following type argument inference. In those cases
+            // we have already reported an implicit any error so we don't report anything here.
+            return anyType;
         }
-        function unwrapSubstitution(type) {
-            if (type.flags & 33554432 /* Substitution */) {
-                return type.substitute;
+        function getTypeOfSymbolWithDeferredType(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.type) {
+                ts.Debug.assertIsDefined(links.deferralParent);
+                ts.Debug.assertIsDefined(links.deferralConstituents);
+                links.type = links.deferralParent.flags & 1048576 /* Union */ ? getUnionType(links.deferralConstituents) : getIntersectionType(links.deferralConstituents);
             }
-            return type;
+            return links.type;
         }
-        // Transform an indexed access to a simpler form, if possible. Return the simpler form, or return
-        // the type itself if no transformation is possible. The writing flag indicates that the type is
-        // the target of an assignment.
-        function getSimplifiedIndexedAccessType(type, writing) {
-            var cache = writing ? "simplifiedForWriting" : "simplifiedForReading";
-            if (type[cache]) {
-                return type[cache] === circularConstraintType ? type : type[cache];
-            }
-            type[cache] = circularConstraintType;
-            // We recursively simplify the object type as it may in turn be an indexed access type. For example, with
-            // '{ [P in T]: { [Q in U]: number } }[T][U]' we want to first simplify the inner indexed access type.
-            var objectType = unwrapSubstitution(getSimplifiedType(type.objectType, writing));
-            var indexType = getSimplifiedType(type.indexType, writing);
-            // T[A | B] -> T[A] | T[B] (reading)
-            // T[A | B] -> T[A] & T[B] (writing)
-            var distributedOverIndex = distributeObjectOverIndexType(objectType, indexType, writing);
-            if (distributedOverIndex) {
-                return type[cache] = distributedOverIndex;
+        function getTypeOfSymbol(symbol) {
+            var checkFlags = ts.getCheckFlags(symbol);
+            if (checkFlags & 65536 /* DeferredType */) {
+                return getTypeOfSymbolWithDeferredType(symbol);
             }
-            // Only do the inner distributions if the index can no longer be instantiated to cause index distribution again
-            if (!(indexType.flags & 63176704 /* Instantiable */)) {
-                // (T | U)[K] -> T[K] | U[K] (reading)
-                // (T | U)[K] -> T[K] & U[K] (writing)
-                // (T & U)[K] -> T[K] & U[K]
-                var distributedOverObject = distributeIndexOverObjectType(objectType, indexType, writing);
-                if (distributedOverObject) {
-                    return type[cache] = distributedOverObject;
-                }
+            if (checkFlags & 1 /* Instantiated */) {
+                return getTypeOfInstantiatedSymbol(symbol);
             }
-            // So ultimately (reading):
-            // ((A & B) | C)[K1 | K2] -> ((A & B) | C)[K1] | ((A & B) | C)[K2] -> (A & B)[K1] | C[K1] | (A & B)[K2] | C[K2] -> (A[K1] & B[K1]) | C[K1] | (A[K2] & B[K2]) | C[K2]
-            // If the object type is a mapped type { [P in K]: E }, where K is generic, instantiate E using a mapper
-            // that substitutes the index type for P. For example, for an index access { [P in K]: Box<T[P]> }[X], we
-            // construct the type Box<T[X]>.
-            if (isGenericMappedType(objectType)) {
-                return type[cache] = mapType(substituteIndexedMappedType(objectType, type.indexType), function (t) { return getSimplifiedType(t, writing); });
+            if (checkFlags & 262144 /* Mapped */) {
+                return getTypeOfMappedSymbol(symbol);
             }
-            return type[cache] = type;
-        }
-        function getSimplifiedConditionalType(type, writing) {
-            var checkType = type.checkType;
-            var extendsType = type.extendsType;
-            var trueType = getTrueTypeFromConditionalType(type);
-            var falseType = getFalseTypeFromConditionalType(type);
-            // Simplifications for types of the form `T extends U ? T : never` and `T extends U ? never : T`.
-            if (falseType.flags & 131072 /* Never */ && getActualTypeVariable(trueType) === getActualTypeVariable(checkType)) {
-                if (checkType.flags & 1 /* Any */ || isTypeAssignableTo(getRestrictiveInstantiation(checkType), getRestrictiveInstantiation(extendsType))) { // Always true
-                    return getSimplifiedType(trueType, writing);
-                }
-                else if (isIntersectionEmpty(checkType, extendsType)) { // Always false
-                    return neverType;
-                }
+            if (checkFlags & 8192 /* ReverseMapped */) {
+                return getTypeOfReverseMappedSymbol(symbol);
             }
-            else if (trueType.flags & 131072 /* Never */ && getActualTypeVariable(falseType) === getActualTypeVariable(checkType)) {
-                if (!(checkType.flags & 1 /* Any */) && isTypeAssignableTo(getRestrictiveInstantiation(checkType), getRestrictiveInstantiation(extendsType))) { // Always true
-                    return neverType;
-                }
-                else if (checkType.flags & 1 /* Any */ || isIntersectionEmpty(checkType, extendsType)) { // Always false
-                    return getSimplifiedType(falseType, writing);
-                }
+            if (symbol.flags & (3 /* Variable */ | 4 /* Property */)) {
+                return getTypeOfVariableOrParameterOrProperty(symbol);
             }
-            return type;
-        }
-        /**
-         * Invokes union simplification logic to determine if an intersection is considered empty as a union constituent
-         */
-        function isIntersectionEmpty(type1, type2) {
-            return !!(getUnionType([intersectTypes(type1, type2), neverType]).flags & 131072 /* Never */);
-        }
-        function substituteIndexedMappedType(objectType, index) {
-            var mapper = createTypeMapper([getTypeParameterFromMappedType(objectType)], [index]);
-            var templateMapper = combineTypeMappers(objectType.mapper, mapper);
-            return instantiateType(getTemplateTypeFromMappedType(objectType), templateMapper);
-        }
-        function getIndexedAccessType(objectType, indexType, accessNode) {
-            return getIndexedAccessTypeOrUndefined(objectType, indexType, accessNode, 0 /* None */) || (accessNode ? errorType : unknownType);
-        }
-        function getIndexedAccessTypeOrUndefined(objectType, indexType, accessNode, accessFlags) {
-            if (accessFlags === void 0) { accessFlags = 0 /* None */; }
-            if (objectType === wildcardType || indexType === wildcardType) {
-                return wildcardType;
+            if (symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 384 /* Enum */ | 512 /* ValueModule */)) {
+                return getTypeOfFuncClassEnumModule(symbol);
             }
-            // If the object type has a string index signature and no other members we know that the result will
-            // always be the type of that index signature and we can simplify accordingly.
-            if (isStringIndexSignatureOnlyType(objectType) && !(indexType.flags & 98304 /* Nullable */) && isTypeAssignableToKind(indexType, 4 /* String */ | 8 /* Number */)) {
-                indexType = stringType;
+            if (symbol.flags & 8 /* EnumMember */) {
+                return getTypeOfEnumMember(symbol);
             }
-            // If the index type is generic, or if the object type is generic and doesn't originate in an expression,
-            // we are performing a higher-order index access where we cannot meaningfully access the properties of the
-            // object type. Note that for a generic T and a non-generic K, we eagerly resolve T[K] if it originates in
-            // an expression. This is to preserve backwards compatibility. For example, an element access 'this["foo"]'
-            // has always been resolved eagerly using the constraint type of 'this' at the given location.
-            if (isGenericIndexType(indexType) || !(accessNode && accessNode.kind !== 185 /* IndexedAccessType */) && isGenericObjectType(objectType)) {
-                if (objectType.flags & 3 /* AnyOrUnknown */) {
-                    return objectType;
-                }
-                // Defer the operation by creating an indexed access type.
-                var id = objectType.id + "," + indexType.id;
-                var type = indexedAccessTypes.get(id);
-                if (!type) {
-                    indexedAccessTypes.set(id, type = createIndexedAccessType(objectType, indexType));
-                }
-                return type;
+            if (symbol.flags & 98304 /* Accessor */) {
+                return getTypeOfAccessors(symbol);
             }
-            // In the following we resolve T[K] to the type of the property in T selected by K.
-            // We treat boolean as different from other unions to improve errors;
-            // skipping straight to getPropertyTypeForIndexType gives errors with 'boolean' instead of 'true'.
-            var apparentObjectType = getReducedApparentType(objectType);
-            if (indexType.flags & 1048576 /* Union */ && !(indexType.flags & 16 /* Boolean */)) {
-                var propTypes = [];
-                var wasMissingProp = false;
-                for (var _i = 0, _a = indexType.types; _i < _a.length; _i++) {
-                    var t = _a[_i];
-                    var propType = getPropertyTypeForIndexType(objectType, apparentObjectType, t, indexType, wasMissingProp, accessNode, accessFlags);
-                    if (propType) {
-                        propTypes.push(propType);
-                    }
-                    else if (!accessNode) {
-                        // If there's no error node, we can immeditely stop, since error reporting is off
-                        return undefined;
-                    }
-                    else {
-                        // Otherwise we set a flag and return at the end of the loop so we still mark all errors
-                        wasMissingProp = true;
-                    }
-                }
-                if (wasMissingProp) {
-                    return undefined;
-                }
-                return accessFlags & 2 /* Writing */ ? getIntersectionType(propTypes) : getUnionType(propTypes);
+            if (symbol.flags & 2097152 /* Alias */) {
+                return getTypeOfAlias(symbol);
             }
-            return getPropertyTypeForIndexType(objectType, apparentObjectType, indexType, indexType, /* supressNoImplicitAnyError */ false, accessNode, accessFlags | 4 /* CacheSymbol */);
+            return errorType;
         }
-        function getTypeFromIndexedAccessTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                var objectType = getTypeFromTypeNode(node.objectType);
-                var indexType = getTypeFromTypeNode(node.indexType);
-                var resolved = getIndexedAccessType(objectType, indexType, node);
-                links.resolvedType = resolved.flags & 8388608 /* IndexedAccess */ &&
-                    resolved.objectType === objectType &&
-                    resolved.indexType === indexType ?
-                    getConditionalFlowTypeOfType(resolved, node) : resolved;
-            }
-            return links.resolvedType;
+        function isReferenceToType(type, target) {
+            return type !== undefined
+                && target !== undefined
+                && (ts.getObjectFlags(type) & 4 /* Reference */) !== 0
+                && type.target === target;
         }
-        function getTypeFromMappedTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                var type = createObjectType(32 /* Mapped */, node.symbol);
-                type.declaration = node;
-                type.aliasSymbol = getAliasSymbolForTypeNode(node);
-                type.aliasTypeArguments = getTypeArgumentsForAliasSymbol(type.aliasSymbol);
-                links.resolvedType = type;
-                // Eagerly resolve the constraint type which forces an error if the constraint type circularly
-                // references itself through one or more type aliases.
-                getConstraintTypeFromMappedType(type);
-            }
-            return links.resolvedType;
+        function getTargetType(type) {
+            return ts.getObjectFlags(type) & 4 /* Reference */ ? type.target : type;
         }
-        function getActualTypeVariable(type) {
-            if (type.flags & 33554432 /* Substitution */) {
-                return type.baseType;
+        // TODO: GH#18217 If `checkBase` is undefined, we should not call this because this will always return false.
+        function hasBaseType(type, checkBase) {
+            return check(type);
+            function check(type) {
+                if (ts.getObjectFlags(type) & (3 /* ClassOrInterface */ | 4 /* Reference */)) {
+                    var target = getTargetType(type);
+                    return target === checkBase || ts.some(getBaseTypes(target), check);
+                }
+                else if (type.flags & 2097152 /* Intersection */) {
+                    return ts.some(type.types, check);
+                }
+                return false;
             }
-            if (type.flags & 8388608 /* IndexedAccess */ && (type.objectType.flags & 33554432 /* Substitution */ ||
-                type.indexType.flags & 33554432 /* Substitution */)) {
-                return getIndexedAccessType(getActualTypeVariable(type.objectType), getActualTypeVariable(type.indexType));
+        }
+        // Appends the type parameters given by a list of declarations to a set of type parameters and returns the resulting set.
+        // The function allocates a new array if the input type parameter set is undefined, but otherwise it modifies the set
+        // in-place and returns the same array.
+        function appendTypeParameters(typeParameters, declarations) {
+            for (var _i = 0, declarations_2 = declarations; _i < declarations_2.length; _i++) {
+                var declaration = declarations_2[_i];
+                typeParameters = ts.appendIfUnique(typeParameters, getDeclaredTypeOfTypeParameter(getSymbolOfNode(declaration)));
             }
-            return type;
+            return typeParameters;
         }
-        function getConditionalType(root, mapper) {
-            var result;
-            var extraTypes;
-            var _loop_12 = function () {
-                var checkType = instantiateType(root.checkType, mapper);
-                var checkTypeInstantiable = isGenericObjectType(checkType) || isGenericIndexType(checkType);
-                var extendsType = instantiateType(root.extendsType, mapper);
-                if (checkType === wildcardType || extendsType === wildcardType) {
-                    return { value: wildcardType };
-                }
-                var combinedMapper = void 0;
-                if (root.inferTypeParameters) {
-                    var context = createInferenceContext(root.inferTypeParameters, /*signature*/ undefined, 0 /* None */);
-                    // We skip inference of the possible `infer` types unles the `extendsType` _is_ an infer type
-                    // if it was, it's trivial to say that extendsType = checkType, however such a pattern is used to
-                    // "reset" the type being build up during constraint calculation and avoid making an apparently "infinite" constraint
-                    // so in those cases we refain from performing inference and retain the uninfered type parameter
-                    if (!checkTypeInstantiable || !ts.some(root.inferTypeParameters, function (t) { return t === extendsType; })) {
-                        // We don't want inferences from constraints as they may cause us to eagerly resolve the
-                        // conditional type instead of deferring resolution. Also, we always want strict function
-                        // types rules (i.e. proper contravariance) for inferences.
-                        inferTypes(context.inferences, checkType, extendsType, 128 /* NoConstraints */ | 256 /* AlwaysStrict */);
+        // Return the outer type parameters of a node or undefined if the node has no outer type parameters.
+        function getOuterTypeParameters(node, includeThisTypes) {
+            while (true) {
+                node = node.parent; // TODO: GH#18217 Use SourceFile kind check instead
+                if (node && ts.isBinaryExpression(node)) {
+                    // prototype assignments get the outer type parameters of their constructor function
+                    var assignmentKind = ts.getAssignmentDeclarationKind(node);
+                    if (assignmentKind === 6 /* Prototype */ || assignmentKind === 3 /* PrototypeProperty */) {
+                        var symbol = getSymbolOfNode(node.left);
+                        if (symbol && symbol.parent && !ts.findAncestor(symbol.parent.valueDeclaration, function (d) { return node === d; })) {
+                            node = symbol.parent.valueDeclaration;
+                        }
                     }
-                    combinedMapper = mergeTypeMappers(mapper, context.mapper);
                 }
-                // Instantiate the extends type including inferences for 'infer T' type parameters
-                var inferredExtendsType = combinedMapper ? instantiateType(root.extendsType, combinedMapper) : extendsType;
-                // We attempt to resolve the conditional type only when the check and extends types are non-generic
-                if (!checkTypeInstantiable && !isGenericObjectType(inferredExtendsType) && !isGenericIndexType(inferredExtendsType)) {
-                    // Return falseType for a definitely false extends check. We check an instantiations of the two
-                    // types with type parameters mapped to the wildcard type, the most permissive instantiations
-                    // possible (the wildcard type is assignable to and from all types). If those are not related,
-                    // then no instantiations will be and we can just return the false branch type.
-                    if (!(inferredExtendsType.flags & 3 /* AnyOrUnknown */) && (checkType.flags & 1 /* Any */ || !isTypeAssignableTo(getPermissiveInstantiation(checkType), getPermissiveInstantiation(inferredExtendsType)))) {
-                        // Return union of trueType and falseType for 'any' since it matches anything
-                        if (checkType.flags & 1 /* Any */) {
-                            (extraTypes || (extraTypes = [])).push(instantiateTypeWithoutDepthIncrease(root.trueType, combinedMapper || mapper));
+                if (!node) {
+                    return undefined;
+                }
+                switch (node.kind) {
+                    case 232 /* VariableStatement */:
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 169 /* CallSignature */:
+                    case 170 /* ConstructSignature */:
+                    case 164 /* MethodSignature */:
+                    case 174 /* FunctionType */:
+                    case 175 /* ConstructorType */:
+                    case 308 /* JSDocFunctionType */:
+                    case 251 /* FunctionDeclaration */:
+                    case 165 /* MethodDeclaration */:
+                    case 208 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 330 /* JSDocTemplateTag */:
+                    case 331 /* JSDocTypedefTag */:
+                    case 325 /* JSDocEnumTag */:
+                    case 324 /* JSDocCallbackTag */:
+                    case 190 /* MappedType */:
+                    case 184 /* ConditionalType */:
+                        var outerTypeParameters = getOuterTypeParameters(node, includeThisTypes);
+                        if (node.kind === 190 /* MappedType */) {
+                            return ts.append(outerTypeParameters, getDeclaredTypeOfTypeParameter(getSymbolOfNode(node.typeParameter)));
                         }
-                        // If falseType is an immediately nested conditional type that isn't distributive or has an
-                        // identical checkType, switch to that type and loop.
-                        var falseType_1 = root.falseType;
-                        if (falseType_1.flags & 16777216 /* Conditional */) {
-                            var newRoot = falseType_1.root;
-                            if (newRoot.node.parent === root.node && (!newRoot.isDistributive || newRoot.checkType === root.checkType)) {
-                                root = newRoot;
-                                return "continue";
-                            }
+                        else if (node.kind === 184 /* ConditionalType */) {
+                            return ts.concatenate(outerTypeParameters, getInferTypeParameters(node));
                         }
-                        result = instantiateTypeWithoutDepthIncrease(falseType_1, mapper);
-                        return "break";
-                    }
-                    // Return trueType for a definitely true extends check. We check instantiations of the two
-                    // types with type parameters mapped to their restrictive form, i.e. a form of the type parameter
-                    // that has no constraint. This ensures that, for example, the type
-                    //   type Foo<T extends { x: any }> = T extends { x: string } ? string : number
-                    // doesn't immediately resolve to 'string' instead of being deferred.
-                    if (inferredExtendsType.flags & 3 /* AnyOrUnknown */ || isTypeAssignableTo(getRestrictiveInstantiation(checkType), getRestrictiveInstantiation(inferredExtendsType))) {
-                        result = instantiateTypeWithoutDepthIncrease(root.trueType, combinedMapper || mapper);
-                        return "break";
-                    }
+                        else if (node.kind === 232 /* VariableStatement */ && !ts.isInJSFile(node)) {
+                            break;
+                        }
+                        var outerAndOwnTypeParameters = appendTypeParameters(outerTypeParameters, ts.getEffectiveTypeParameterDeclarations(node));
+                        var thisType = includeThisTypes &&
+                            (node.kind === 252 /* ClassDeclaration */ || node.kind === 221 /* ClassExpression */ || node.kind === 253 /* InterfaceDeclaration */ || isJSConstructor(node)) &&
+                            getDeclaredTypeOfClassOrInterface(getSymbolOfNode(node)).thisType;
+                        return thisType ? ts.append(outerAndOwnTypeParameters, thisType) : outerAndOwnTypeParameters;
+                    case 326 /* JSDocParameterTag */:
+                        var paramSymbol = ts.getParameterSymbolFromJSDoc(node);
+                        if (paramSymbol) {
+                            node = paramSymbol.valueDeclaration;
+                        }
+                        break;
                 }
-                // Return a deferred type for a check that is neither definitely true nor definitely false
-                var erasedCheckType = getActualTypeVariable(checkType);
-                result = createType(16777216 /* Conditional */);
-                result.root = root;
-                result.checkType = erasedCheckType;
-                result.extendsType = extendsType;
-                result.mapper = mapper;
-                result.combinedMapper = combinedMapper;
-                result.aliasSymbol = root.aliasSymbol;
-                result.aliasTypeArguments = instantiateTypes(root.aliasTypeArguments, mapper); // TODO: GH#18217
-                return "break";
-            };
-            // We loop here for an immediately nested conditional type in the false position, effectively treating
-            // types of the form 'A extends B ? X : C extends D ? Y : E extends F ? Z : ...' as a single construct for
-            // purposes of resolution. This means such types aren't subject to the instatiation depth limiter.
-            while (true) {
-                var state_4 = _loop_12();
-                if (typeof state_4 === "object")
-                    return state_4.value;
-                if (state_4 === "break")
-                    break;
             }
-            return extraTypes ? getUnionType(ts.append(extraTypes, result)) : result;
         }
-        function getTrueTypeFromConditionalType(type) {
-            return type.resolvedTrueType || (type.resolvedTrueType = instantiateType(type.root.trueType, type.mapper));
-        }
-        function getFalseTypeFromConditionalType(type) {
-            return type.resolvedFalseType || (type.resolvedFalseType = instantiateType(type.root.falseType, type.mapper));
-        }
-        function getInferredTrueTypeFromConditionalType(type) {
-            return type.resolvedInferredTrueType || (type.resolvedInferredTrueType = type.combinedMapper ? instantiateType(type.root.trueType, type.combinedMapper) : getTrueTypeFromConditionalType(type));
+        // The outer type parameters are those defined by enclosing generic classes, methods, or functions.
+        function getOuterTypeParametersOfClassOrInterface(symbol) {
+            var declaration = symbol.flags & 32 /* Class */ ? symbol.valueDeclaration : ts.getDeclarationOfKind(symbol, 253 /* InterfaceDeclaration */);
+            ts.Debug.assert(!!declaration, "Class was missing valueDeclaration -OR- non-class had no interface declarations");
+            return getOuterTypeParameters(declaration);
         }
-        function getInferTypeParameters(node) {
+        // The local type parameters are the combined set of type parameters from all declarations of the class,
+        // interface, or type alias.
+        function getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol) {
             var result;
-            if (node.locals) {
-                node.locals.forEach(function (symbol) {
-                    if (symbol.flags & 262144 /* TypeParameter */) {
-                        result = ts.append(result, getDeclaredTypeOfSymbol(symbol));
-                    }
-                });
+            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                var node = _a[_i];
+                if (node.kind === 253 /* InterfaceDeclaration */ ||
+                    node.kind === 252 /* ClassDeclaration */ ||
+                    node.kind === 221 /* ClassExpression */ ||
+                    isJSConstructor(node) ||
+                    ts.isTypeAlias(node)) {
+                    var declaration = node;
+                    result = appendTypeParameters(result, ts.getEffectiveTypeParameterDeclarations(declaration));
+                }
             }
             return result;
         }
-        function getTypeFromConditionalTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                var checkType = getTypeFromTypeNode(node.checkType);
-                var aliasSymbol = getAliasSymbolForTypeNode(node);
-                var aliasTypeArguments = getTypeArgumentsForAliasSymbol(aliasSymbol);
-                var allOuterTypeParameters = getOuterTypeParameters(node, /*includeThisTypes*/ true);
-                var outerTypeParameters = aliasTypeArguments ? allOuterTypeParameters : ts.filter(allOuterTypeParameters, function (tp) { return isTypeParameterPossiblyReferenced(tp, node); });
-                var root = {
-                    node: node,
-                    checkType: checkType,
-                    extendsType: getTypeFromTypeNode(node.extendsType),
-                    trueType: getTypeFromTypeNode(node.trueType),
-                    falseType: getTypeFromTypeNode(node.falseType),
-                    isDistributive: !!(checkType.flags & 262144 /* TypeParameter */),
-                    inferTypeParameters: getInferTypeParameters(node),
-                    outerTypeParameters: outerTypeParameters,
-                    instantiations: undefined,
-                    aliasSymbol: aliasSymbol,
-                    aliasTypeArguments: aliasTypeArguments
-                };
-                links.resolvedType = getConditionalType(root, /*mapper*/ undefined);
-                if (outerTypeParameters) {
-                    root.instantiations = ts.createMap();
-                    root.instantiations.set(getTypeListId(outerTypeParameters), links.resolvedType);
-                }
-            }
-            return links.resolvedType;
+        // The full set of type parameters for a generic class or interface type consists of its outer type parameters plus
+        // its locally declared type parameters.
+        function getTypeParametersOfClassOrInterface(symbol) {
+            return ts.concatenate(getOuterTypeParametersOfClassOrInterface(symbol), getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol));
         }
-        function getTypeFromInferTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                links.resolvedType = getDeclaredTypeOfTypeParameter(getSymbolOfNode(node.typeParameter));
+        // A type is a mixin constructor if it has a single construct signature taking no type parameters and a single
+        // rest parameter of type any[].
+        function isMixinConstructorType(type) {
+            var signatures = getSignaturesOfType(type, 1 /* Construct */);
+            if (signatures.length === 1) {
+                var s = signatures[0];
+                return !s.typeParameters && s.parameters.length === 1 && signatureHasRestParameter(s) && getElementTypeOfArrayType(getTypeOfParameter(s.parameters[0])) === anyType;
             }
-            return links.resolvedType;
+            return false;
         }
-        function getIdentifierChain(node) {
-            if (ts.isIdentifier(node)) {
-                return [node];
+        function isConstructorType(type) {
+            if (getSignaturesOfType(type, 1 /* Construct */).length > 0) {
+                return true;
             }
-            else {
-                return ts.append(getIdentifierChain(node.left), node.right);
+            if (type.flags & 8650752 /* TypeVariable */) {
+                var constraint = getBaseConstraintOfType(type);
+                return !!constraint && isMixinConstructorType(constraint);
             }
+            return false;
         }
-        function getTypeFromImportTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                if (node.isTypeOf && node.typeArguments) { // Only the non-typeof form can make use of type arguments
-                    error(node, ts.Diagnostics.Type_arguments_cannot_be_used_here);
-                    links.resolvedSymbol = unknownSymbol;
-                    return links.resolvedType = errorType;
+        function getBaseTypeNodeOfClass(type) {
+            return ts.getEffectiveBaseTypeNode(type.symbol.valueDeclaration);
+        }
+        function getConstructorsForTypeArguments(type, typeArgumentNodes, location) {
+            var typeArgCount = ts.length(typeArgumentNodes);
+            var isJavascript = ts.isInJSFile(location);
+            return ts.filter(getSignaturesOfType(type, 1 /* Construct */), function (sig) { return (isJavascript || typeArgCount >= getMinTypeArgumentCount(sig.typeParameters)) && typeArgCount <= ts.length(sig.typeParameters); });
+        }
+        function getInstantiatedConstructorsForTypeArguments(type, typeArgumentNodes, location) {
+            var signatures = getConstructorsForTypeArguments(type, typeArgumentNodes, location);
+            var typeArguments = ts.map(typeArgumentNodes, getTypeFromTypeNode);
+            return ts.sameMap(signatures, function (sig) { return ts.some(sig.typeParameters) ? getSignatureInstantiation(sig, typeArguments, ts.isInJSFile(location)) : sig; });
+        }
+        /**
+         * The base constructor of a class can resolve to
+         * * undefinedType if the class has no extends clause,
+         * * unknownType if an error occurred during resolution of the extends expression,
+         * * nullType if the extends expression is the null value,
+         * * anyType if the extends expression has type any, or
+         * * an object type with at least one construct signature.
+         */
+        function getBaseConstructorTypeOfClass(type) {
+            if (!type.resolvedBaseConstructorType) {
+                var decl = type.symbol.valueDeclaration;
+                var extended = ts.getEffectiveBaseTypeNode(decl);
+                var baseTypeNode = getBaseTypeNodeOfClass(type);
+                if (!baseTypeNode) {
+                    return type.resolvedBaseConstructorType = undefinedType;
                 }
-                if (!ts.isLiteralImportTypeNode(node)) {
-                    error(node.argument, ts.Diagnostics.String_literal_expected);
-                    links.resolvedSymbol = unknownSymbol;
-                    return links.resolvedType = errorType;
+                if (!pushTypeResolution(type, 1 /* ResolvedBaseConstructorType */)) {
+                    return errorType;
                 }
-                var targetMeaning = node.isTypeOf ? 111551 /* Value */ : node.flags & 4194304 /* JSDoc */ ? 111551 /* Value */ | 788968 /* Type */ : 788968 /* Type */;
-                // TODO: Future work: support unions/generics/whatever via a deferred import-type
-                var innerModuleSymbol = resolveExternalModuleName(node, node.argument.literal);
-                if (!innerModuleSymbol) {
-                    links.resolvedSymbol = unknownSymbol;
-                    return links.resolvedType = errorType;
+                var baseConstructorType = checkExpression(baseTypeNode.expression);
+                if (extended && baseTypeNode !== extended) {
+                    ts.Debug.assert(!extended.typeArguments); // Because this is in a JS file, and baseTypeNode is in an @extends tag
+                    checkExpression(extended.expression);
                 }
-                var moduleSymbol = resolveExternalModuleSymbol(innerModuleSymbol, /*dontResolveAlias*/ false);
-                if (!ts.nodeIsMissing(node.qualifier)) {
-                    var nameStack = getIdentifierChain(node.qualifier);
-                    var currentNamespace = moduleSymbol;
-                    var current = void 0;
-                    while (current = nameStack.shift()) {
-                        var meaning = nameStack.length ? 1920 /* Namespace */ : targetMeaning;
-                        var next = getSymbol(getExportsOfSymbol(getMergedSymbol(resolveSymbol(currentNamespace))), current.escapedText, meaning);
-                        if (!next) {
-                            error(current, ts.Diagnostics.Namespace_0_has_no_exported_member_1, getFullyQualifiedName(currentNamespace), ts.declarationNameToString(current));
-                            return links.resolvedType = errorType;
+                if (baseConstructorType.flags & (524288 /* Object */ | 2097152 /* Intersection */)) {
+                    // Resolving the members of a class requires us to resolve the base class of that class.
+                    // We force resolution here such that we catch circularities now.
+                    resolveStructuredTypeMembers(baseConstructorType);
+                }
+                if (!popTypeResolution()) {
+                    error(type.symbol.valueDeclaration, ts.Diagnostics._0_is_referenced_directly_or_indirectly_in_its_own_base_expression, symbolToString(type.symbol));
+                    return type.resolvedBaseConstructorType = errorType;
+                }
+                if (!(baseConstructorType.flags & 1 /* Any */) && baseConstructorType !== nullWideningType && !isConstructorType(baseConstructorType)) {
+                    var err = error(baseTypeNode.expression, ts.Diagnostics.Type_0_is_not_a_constructor_function_type, typeToString(baseConstructorType));
+                    if (baseConstructorType.flags & 262144 /* TypeParameter */) {
+                        var constraint = getConstraintFromTypeParameter(baseConstructorType);
+                        var ctorReturn = unknownType;
+                        if (constraint) {
+                            var ctorSig = getSignaturesOfType(constraint, 1 /* Construct */);
+                            if (ctorSig[0]) {
+                                ctorReturn = getReturnTypeOfSignature(ctorSig[0]);
+                            }
+                        }
+                        ts.addRelatedInfo(err, ts.createDiagnosticForNode(baseConstructorType.symbol.declarations[0], ts.Diagnostics.Did_you_mean_for_0_to_be_constrained_to_type_new_args_Colon_any_1, symbolToString(baseConstructorType.symbol), typeToString(ctorReturn)));
+                    }
+                    return type.resolvedBaseConstructorType = errorType;
+                }
+                type.resolvedBaseConstructorType = baseConstructorType;
+            }
+            return type.resolvedBaseConstructorType;
+        }
+        function getImplementsTypes(type) {
+            var resolvedImplementsTypes = ts.emptyArray;
+            for (var _i = 0, _a = type.symbol.declarations; _i < _a.length; _i++) {
+                var declaration = _a[_i];
+                var implementsTypeNodes = ts.getEffectiveImplementsTypeNodes(declaration);
+                if (!implementsTypeNodes)
+                    continue;
+                for (var _b = 0, implementsTypeNodes_1 = implementsTypeNodes; _b < implementsTypeNodes_1.length; _b++) {
+                    var node = implementsTypeNodes_1[_b];
+                    var implementsType = getTypeFromTypeNode(node);
+                    if (implementsType !== errorType) {
+                        if (resolvedImplementsTypes === ts.emptyArray) {
+                            resolvedImplementsTypes = [implementsType];
+                        }
+                        else {
+                            resolvedImplementsTypes.push(implementsType);
                         }
-                        getNodeLinks(current).resolvedSymbol = next;
-                        getNodeLinks(current.parent).resolvedSymbol = next;
-                        currentNamespace = next;
                     }
-                    links.resolvedType = resolveImportSymbolType(node, links, currentNamespace, targetMeaning);
                 }
-                else {
-                    if (moduleSymbol.flags & targetMeaning) {
-                        links.resolvedType = resolveImportSymbolType(node, links, moduleSymbol, targetMeaning);
+            }
+            return resolvedImplementsTypes;
+        }
+        function reportCircularBaseType(node, type) {
+            error(node, ts.Diagnostics.Type_0_recursively_references_itself_as_a_base_type, typeToString(type, /*enclosingDeclaration*/ undefined, 2 /* WriteArrayAsGenericType */));
+        }
+        function getBaseTypes(type) {
+            if (!type.baseTypesResolved) {
+                if (pushTypeResolution(type, 7 /* ResolvedBaseTypes */)) {
+                    if (type.objectFlags & 8 /* Tuple */) {
+                        type.resolvedBaseTypes = [getTupleBaseType(type)];
+                    }
+                    else if (type.symbol.flags & (32 /* Class */ | 64 /* Interface */)) {
+                        if (type.symbol.flags & 32 /* Class */) {
+                            resolveBaseTypesOfClass(type);
+                        }
+                        if (type.symbol.flags & 64 /* Interface */) {
+                            resolveBaseTypesOfInterface(type);
+                        }
                     }
                     else {
-                        var errorMessage = targetMeaning === 111551 /* Value */
-                            ? ts.Diagnostics.Module_0_does_not_refer_to_a_value_but_is_used_as_a_value_here
-                            : ts.Diagnostics.Module_0_does_not_refer_to_a_type_but_is_used_as_a_type_here_Did_you_mean_typeof_import_0;
-                        error(node, errorMessage, node.argument.literal.text);
-                        links.resolvedSymbol = unknownSymbol;
-                        links.resolvedType = errorType;
+                        ts.Debug.fail("type must be class or interface");
+                    }
+                    if (!popTypeResolution()) {
+                        for (var _i = 0, _a = type.symbol.declarations; _i < _a.length; _i++) {
+                            var declaration = _a[_i];
+                            if (declaration.kind === 252 /* ClassDeclaration */ || declaration.kind === 253 /* InterfaceDeclaration */) {
+                                reportCircularBaseType(declaration, type);
+                            }
+                        }
                     }
                 }
+                type.baseTypesResolved = true;
             }
-            return links.resolvedType;
+            return type.resolvedBaseTypes;
         }
-        function resolveImportSymbolType(node, links, symbol, meaning) {
-            var resolvedSymbol = resolveSymbol(symbol);
-            links.resolvedSymbol = resolvedSymbol;
-            if (meaning === 111551 /* Value */) {
-                return getTypeOfSymbol(symbol); // intentionally doesn't use resolved symbol so type is cached as expected on the alias
+        function getTupleBaseType(type) {
+            var elementTypes = ts.sameMap(type.typeParameters, function (t, i) { return type.elementFlags[i] & 8 /* Variadic */ ? getIndexedAccessType(t, numberType) : t; });
+            return createArrayType(getUnionType(elementTypes || ts.emptyArray), type.readonly);
+        }
+        function resolveBaseTypesOfClass(type) {
+            type.resolvedBaseTypes = ts.resolvingEmptyArray;
+            var baseConstructorType = getApparentType(getBaseConstructorTypeOfClass(type));
+            if (!(baseConstructorType.flags & (524288 /* Object */ | 2097152 /* Intersection */ | 1 /* Any */))) {
+                return type.resolvedBaseTypes = ts.emptyArray;
             }
-            else {
-                return getTypeReferenceType(node, resolvedSymbol); // getTypeReferenceType doesn't handle aliases - it must get the resolved symbol
+            var baseTypeNode = getBaseTypeNodeOfClass(type);
+            var baseType;
+            var originalBaseType = baseConstructorType.symbol ? getDeclaredTypeOfSymbol(baseConstructorType.symbol) : undefined;
+            if (baseConstructorType.symbol && baseConstructorType.symbol.flags & 32 /* Class */ &&
+                areAllOuterTypeParametersApplied(originalBaseType)) {
+                // When base constructor type is a class with no captured type arguments we know that the constructors all have the same type parameters as the
+                // class and all return the instance type of the class. There is no need for further checks and we can apply the
+                // type arguments in the same manner as a type reference to get the same error reporting experience.
+                baseType = getTypeFromClassOrInterfaceReference(baseTypeNode, baseConstructorType.symbol);
             }
-        }
-        function getTypeFromTypeLiteralOrFunctionOrConstructorTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                // Deferred resolution of members is handled by resolveObjectTypeMembers
-                var aliasSymbol = getAliasSymbolForTypeNode(node);
-                if (getMembersOfSymbol(node.symbol).size === 0 && !aliasSymbol) {
-                    links.resolvedType = emptyTypeLiteralType;
-                }
-                else {
-                    var type = createObjectType(16 /* Anonymous */, node.symbol);
-                    type.aliasSymbol = aliasSymbol;
-                    type.aliasTypeArguments = getTypeArgumentsForAliasSymbol(aliasSymbol);
-                    if (ts.isJSDocTypeLiteral(node) && node.isArrayType) {
-                        type = createArrayType(type);
-                    }
-                    links.resolvedType = type;
+            else if (baseConstructorType.flags & 1 /* Any */) {
+                baseType = baseConstructorType;
+            }
+            else {
+                // The class derives from a "class-like" constructor function, check that we have at least one construct signature
+                // with a matching number of type parameters and use the return type of the first instantiated signature. Elsewhere
+                // we check that all instantiated signatures return the same type.
+                var constructors = getInstantiatedConstructorsForTypeArguments(baseConstructorType, baseTypeNode.typeArguments, baseTypeNode);
+                if (!constructors.length) {
+                    error(baseTypeNode.expression, ts.Diagnostics.No_base_constructor_has_the_specified_number_of_type_arguments);
+                    return type.resolvedBaseTypes = ts.emptyArray;
                 }
+                baseType = getReturnTypeOfSignature(constructors[0]);
             }
-            return links.resolvedType;
-        }
-        function getAliasSymbolForTypeNode(node) {
-            var host = node.parent;
-            while (ts.isParenthesizedTypeNode(host) || ts.isTypeOperatorNode(host) && host.operator === 138 /* ReadonlyKeyword */) {
-                host = host.parent;
+            if (baseType === errorType) {
+                return type.resolvedBaseTypes = ts.emptyArray;
             }
-            return ts.isTypeAlias(host) ? getSymbolOfNode(host) : undefined;
-        }
-        function getTypeArgumentsForAliasSymbol(symbol) {
-            return symbol ? getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol) : undefined;
-        }
-        function isNonGenericObjectType(type) {
-            return !!(type.flags & 524288 /* Object */) && !isGenericMappedType(type);
-        }
-        function isEmptyObjectTypeOrSpreadsIntoEmptyObject(type) {
-            return isEmptyObjectType(type) || !!(type.flags & (65536 /* Null */ | 32768 /* Undefined */ | 528 /* BooleanLike */ | 296 /* NumberLike */ | 2112 /* BigIntLike */ | 132 /* StringLike */ | 1056 /* EnumLike */ | 67108864 /* NonPrimitive */ | 4194304 /* Index */));
+            var reducedBaseType = getReducedType(baseType);
+            if (!isValidBaseType(reducedBaseType)) {
+                var elaboration = elaborateNeverIntersection(/*errorInfo*/ undefined, baseType);
+                var diagnostic = ts.chainDiagnosticMessages(elaboration, ts.Diagnostics.Base_constructor_return_type_0_is_not_an_object_type_or_intersection_of_object_types_with_statically_known_members, typeToString(reducedBaseType));
+                diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(baseTypeNode.expression, diagnostic));
+                return type.resolvedBaseTypes = ts.emptyArray;
+            }
+            if (type === reducedBaseType || hasBaseType(reducedBaseType, type)) {
+                error(type.symbol.valueDeclaration, ts.Diagnostics.Type_0_recursively_references_itself_as_a_base_type, typeToString(type, /*enclosingDeclaration*/ undefined, 2 /* WriteArrayAsGenericType */));
+                return type.resolvedBaseTypes = ts.emptyArray;
+            }
+            if (type.resolvedBaseTypes === ts.resolvingEmptyArray) {
+                // Circular reference, likely through instantiation of default parameters
+                // (otherwise there'd be an error from hasBaseType) - this is fine, but `.members` should be reset
+                // as `getIndexedAccessType` via `instantiateType` via `getTypeFromClassOrInterfaceReference` forces a
+                // partial instantiation of the members without the base types fully resolved
+                type.members = undefined;
+            }
+            return type.resolvedBaseTypes = [reducedBaseType];
         }
-        function isSinglePropertyAnonymousObjectType(type) {
-            return !!(type.flags & 524288 /* Object */) &&
-                !!(ts.getObjectFlags(type) & 16 /* Anonymous */) &&
-                (ts.length(getPropertiesOfType(type)) === 1 || ts.every(getPropertiesOfType(type), function (p) { return !!(p.flags & 16777216 /* Optional */); }));
+        function areAllOuterTypeParametersApplied(type) {
+            // An unapplied type parameter has its symbol still the same as the matching argument symbol.
+            // Since parameters are applied outer-to-inner, only the last outer parameter needs to be checked.
+            var outerTypeParameters = type.outerTypeParameters;
+            if (outerTypeParameters) {
+                var last_1 = outerTypeParameters.length - 1;
+                var typeArguments = getTypeArguments(type);
+                return outerTypeParameters[last_1].symbol !== typeArguments[last_1].symbol;
+            }
+            return true;
         }
-        function tryMergeUnionOfObjectTypeAndEmptyObject(type, readonly) {
-            if (type.types.length === 2) {
-                var firstType = type.types[0];
-                var secondType = type.types[1];
-                if (ts.every(type.types, isEmptyObjectTypeOrSpreadsIntoEmptyObject)) {
-                    return isEmptyObjectType(firstType) ? firstType : isEmptyObjectType(secondType) ? secondType : emptyObjectType;
-                }
-                if (isEmptyObjectTypeOrSpreadsIntoEmptyObject(firstType) && isSinglePropertyAnonymousObjectType(secondType)) {
-                    return getAnonymousPartialType(secondType);
-                }
-                if (isEmptyObjectTypeOrSpreadsIntoEmptyObject(secondType) && isSinglePropertyAnonymousObjectType(firstType)) {
-                    return getAnonymousPartialType(firstType);
+        // A valid base type is `any`, an object type or intersection of object types.
+        function isValidBaseType(type) {
+            if (type.flags & 262144 /* TypeParameter */) {
+                var constraint = getBaseConstraintOfType(type);
+                if (constraint) {
+                    return isValidBaseType(constraint);
                 }
             }
-            function getAnonymousPartialType(type) {
-                // gets the type as if it had been spread, but where everything in the spread is made optional
-                var members = ts.createSymbolTable();
-                for (var _i = 0, _a = getPropertiesOfType(type); _i < _a.length; _i++) {
-                    var prop = _a[_i];
-                    if (ts.getDeclarationModifierFlagsFromSymbol(prop) & (8 /* Private */ | 16 /* Protected */)) {
-                        // do nothing, skip privates
-                    }
-                    else if (isSpreadableProperty(prop)) {
-                        var isSetonlyAccessor = prop.flags & 65536 /* SetAccessor */ && !(prop.flags & 32768 /* GetAccessor */);
-                        var flags = 4 /* Property */ | 16777216 /* Optional */;
-                        var result = createSymbol(flags, prop.escapedName, readonly ? 8 /* Readonly */ : 0);
-                        result.type = isSetonlyAccessor ? undefinedType : getTypeOfSymbol(prop);
-                        result.declarations = prop.declarations;
-                        result.nameType = getSymbolLinks(prop).nameType;
-                        result.syntheticOrigin = prop;
-                        members.set(prop.escapedName, result);
+            // TODO: Given that we allow type parmeters here now, is this `!isGenericMappedType(type)` check really needed?
+            // There's no reason a `T` should be allowed while a `Readonly<T>` should not.
+            return !!(type.flags & (524288 /* Object */ | 67108864 /* NonPrimitive */ | 1 /* Any */) && !isGenericMappedType(type) ||
+                type.flags & 2097152 /* Intersection */ && ts.every(type.types, isValidBaseType));
+        }
+        function resolveBaseTypesOfInterface(type) {
+            type.resolvedBaseTypes = type.resolvedBaseTypes || ts.emptyArray;
+            for (var _i = 0, _a = type.symbol.declarations; _i < _a.length; _i++) {
+                var declaration = _a[_i];
+                if (declaration.kind === 253 /* InterfaceDeclaration */ && ts.getInterfaceBaseTypeNodes(declaration)) {
+                    for (var _b = 0, _c = ts.getInterfaceBaseTypeNodes(declaration); _b < _c.length; _b++) {
+                        var node = _c[_b];
+                        var baseType = getReducedType(getTypeFromTypeNode(node));
+                        if (baseType !== errorType) {
+                            if (isValidBaseType(baseType)) {
+                                if (type !== baseType && !hasBaseType(baseType, type)) {
+                                    if (type.resolvedBaseTypes === ts.emptyArray) {
+                                        type.resolvedBaseTypes = [baseType];
+                                    }
+                                    else {
+                                        type.resolvedBaseTypes.push(baseType);
+                                    }
+                                }
+                                else {
+                                    reportCircularBaseType(declaration, type);
+                                }
+                            }
+                            else {
+                                error(node, ts.Diagnostics.An_interface_can_only_extend_an_object_type_or_intersection_of_object_types_with_statically_known_members);
+                            }
+                        }
                     }
                 }
-                var spread = createAnonymousType(type.symbol, members, ts.emptyArray, ts.emptyArray, getIndexInfoOfType(type, 0 /* String */), getIndexInfoOfType(type, 1 /* Number */));
-                spread.objectFlags |= 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
-                return spread;
             }
         }
         /**
-         * Since the source of spread types are object literals, which are not binary,
-         * this function should be called in a left folding style, with left = previous result of getSpreadType
-         * and right = the new element to be spread.
+         * Returns true if the interface given by the symbol is free of "this" references.
+         *
+         * Specifically, the result is true if the interface itself contains no references
+         * to "this" in its body, if all base types are interfaces,
+         * and if none of the base interfaces have a "this" type.
          */
-        function getSpreadType(left, right, symbol, objectFlags, readonly) {
-            if (left.flags & 1 /* Any */ || right.flags & 1 /* Any */) {
-                return anyType;
-            }
-            if (left.flags & 2 /* Unknown */ || right.flags & 2 /* Unknown */) {
-                return unknownType;
-            }
-            if (left.flags & 131072 /* Never */) {
-                return right;
-            }
-            if (right.flags & 131072 /* Never */) {
-                return left;
-            }
-            if (left.flags & 1048576 /* Union */) {
-                var merged = tryMergeUnionOfObjectTypeAndEmptyObject(left, readonly);
-                if (merged) {
-                    return getSpreadType(merged, right, symbol, objectFlags, readonly);
+        function isThislessInterface(symbol) {
+            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                var declaration = _a[_i];
+                if (declaration.kind === 253 /* InterfaceDeclaration */) {
+                    if (declaration.flags & 128 /* ContainsThis */) {
+                        return false;
+                    }
+                    var baseTypeNodes = ts.getInterfaceBaseTypeNodes(declaration);
+                    if (baseTypeNodes) {
+                        for (var _b = 0, baseTypeNodes_1 = baseTypeNodes; _b < baseTypeNodes_1.length; _b++) {
+                            var node = baseTypeNodes_1[_b];
+                            if (ts.isEntityNameExpression(node.expression)) {
+                                var baseSymbol = resolveEntityName(node.expression, 788968 /* Type */, /*ignoreErrors*/ true);
+                                if (!baseSymbol || !(baseSymbol.flags & 64 /* Interface */) || getDeclaredTypeOfClassOrInterface(baseSymbol).thisType) {
+                                    return false;
+                                }
+                            }
+                        }
+                    }
                 }
-                return mapType(left, function (t) { return getSpreadType(t, right, symbol, objectFlags, readonly); });
             }
-            if (right.flags & 1048576 /* Union */) {
-                var merged = tryMergeUnionOfObjectTypeAndEmptyObject(right, readonly);
+            return true;
+        }
+        function getDeclaredTypeOfClassOrInterface(symbol) {
+            var links = getSymbolLinks(symbol);
+            var originalLinks = links;
+            if (!links.declaredType) {
+                var kind = symbol.flags & 32 /* Class */ ? 1 /* Class */ : 2 /* Interface */;
+                var merged = mergeJSSymbols(symbol, getAssignedClassSymbol(symbol.valueDeclaration));
                 if (merged) {
-                    return getSpreadType(left, merged, symbol, objectFlags, readonly);
+                    // note:we overwrite links because we just cloned the symbol
+                    symbol = links = merged;
+                }
+                var type = originalLinks.declaredType = links.declaredType = createObjectType(kind, symbol);
+                var outerTypeParameters = getOuterTypeParametersOfClassOrInterface(symbol);
+                var localTypeParameters = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
+                // A class or interface is generic if it has type parameters or a "this" type. We always give classes a "this" type
+                // because it is not feasible to analyze all members to determine if the "this" type escapes the class (in particular,
+                // property types inferred from initializers and method return types inferred from return statements are very hard
+                // to exhaustively analyze). We give interfaces a "this" type if we can't definitely determine that they are free of
+                // "this" references.
+                if (outerTypeParameters || localTypeParameters || kind === 1 /* Class */ || !isThislessInterface(symbol)) {
+                    type.objectFlags |= 4 /* Reference */;
+                    type.typeParameters = ts.concatenate(outerTypeParameters, localTypeParameters);
+                    type.outerTypeParameters = outerTypeParameters;
+                    type.localTypeParameters = localTypeParameters;
+                    type.instantiations = new ts.Map();
+                    type.instantiations.set(getTypeListId(type.typeParameters), type);
+                    type.target = type;
+                    type.resolvedTypeArguments = type.typeParameters;
+                    type.thisType = createTypeParameter(symbol);
+                    type.thisType.isThisType = true;
+                    type.thisType.constraint = type;
                 }
-                return mapType(right, function (t) { return getSpreadType(left, t, symbol, objectFlags, readonly); });
-            }
-            if (right.flags & (528 /* BooleanLike */ | 296 /* NumberLike */ | 2112 /* BigIntLike */ | 132 /* StringLike */ | 1056 /* EnumLike */ | 67108864 /* NonPrimitive */ | 4194304 /* Index */)) {
-                return left;
             }
-            if (isGenericObjectType(left) || isGenericObjectType(right)) {
-                if (isEmptyObjectType(left)) {
-                    return right;
+            return links.declaredType;
+        }
+        function getDeclaredTypeOfTypeAlias(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.declaredType) {
+                // Note that we use the links object as the target here because the symbol object is used as the unique
+                // identity for resolution of the 'type' property in SymbolLinks.
+                if (!pushTypeResolution(symbol, 2 /* DeclaredType */)) {
+                    return errorType;
                 }
-                // When the left type is an intersection, we may need to merge the last constituent of the
-                // intersection with the right type. For example when the left type is 'T & { a: string }'
-                // and the right type is '{ b: string }' we produce 'T & { a: string, b: string }'.
-                if (left.flags & 2097152 /* Intersection */) {
-                    var types = left.types;
-                    var lastLeft = types[types.length - 1];
-                    if (isNonGenericObjectType(lastLeft) && isNonGenericObjectType(right)) {
-                        return getIntersectionType(ts.concatenate(types.slice(0, types.length - 1), [getSpreadType(lastLeft, right, symbol, objectFlags, readonly)]));
+                var declaration = ts.Debug.checkDefined(ts.find(symbol.declarations, ts.isTypeAlias), "Type alias symbol with no valid declaration found");
+                var typeNode = ts.isJSDocTypeAlias(declaration) ? declaration.typeExpression : declaration.type;
+                // If typeNode is missing, we will error in checkJSDocTypedefTag.
+                var type = typeNode ? getTypeFromTypeNode(typeNode) : errorType;
+                if (popTypeResolution()) {
+                    var typeParameters = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol);
+                    if (typeParameters) {
+                        // Initialize the instantiation cache for generic type aliases. The declared type corresponds to
+                        // an instantiation of the type alias with the type parameters supplied as type arguments.
+                        links.typeParameters = typeParameters;
+                        links.instantiations = new ts.Map();
+                        links.instantiations.set(getTypeListId(typeParameters), type);
                     }
                 }
-                return getIntersectionType([left, right]);
+                else {
+                    type = errorType;
+                    error(ts.isNamedDeclaration(declaration) ? declaration.name : declaration || declaration, ts.Diagnostics.Type_alias_0_circularly_references_itself, symbolToString(symbol));
+                }
+                links.declaredType = type;
             }
-            var members = ts.createSymbolTable();
-            var skippedPrivateMembers = ts.createUnderscoreEscapedMap();
-            var stringIndexInfo;
-            var numberIndexInfo;
-            if (left === emptyObjectType) {
-                // for the first spread element, left === emptyObjectType, so take the right's string indexer
-                stringIndexInfo = getIndexInfoOfType(right, 0 /* String */);
-                numberIndexInfo = getIndexInfoOfType(right, 1 /* Number */);
+            return links.declaredType;
+        }
+        function isStringConcatExpression(expr) {
+            if (ts.isStringLiteralLike(expr)) {
+                return true;
             }
-            else {
-                stringIndexInfo = unionSpreadIndexInfos(getIndexInfoOfType(left, 0 /* String */), getIndexInfoOfType(right, 0 /* String */));
-                numberIndexInfo = unionSpreadIndexInfos(getIndexInfoOfType(left, 1 /* Number */), getIndexInfoOfType(right, 1 /* Number */));
+            else if (expr.kind === 216 /* BinaryExpression */) {
+                return isStringConcatExpression(expr.left) && isStringConcatExpression(expr.right);
             }
-            for (var _i = 0, _a = getPropertiesOfType(right); _i < _a.length; _i++) {
-                var rightProp = _a[_i];
-                if (ts.getDeclarationModifierFlagsFromSymbol(rightProp) & (8 /* Private */ | 16 /* Protected */)) {
-                    skippedPrivateMembers.set(rightProp.escapedName, true);
-                }
-                else if (isSpreadableProperty(rightProp)) {
-                    members.set(rightProp.escapedName, getSpreadSymbol(rightProp, readonly));
+            return false;
+        }
+        function isLiteralEnumMember(member) {
+            var expr = member.initializer;
+            if (!expr) {
+                return !(member.flags & 8388608 /* Ambient */);
+            }
+            switch (expr.kind) {
+                case 10 /* StringLiteral */:
+                case 8 /* NumericLiteral */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                    return true;
+                case 214 /* PrefixUnaryExpression */:
+                    return expr.operator === 40 /* MinusToken */ &&
+                        expr.operand.kind === 8 /* NumericLiteral */;
+                case 78 /* Identifier */:
+                    return ts.nodeIsMissing(expr) || !!getSymbolOfNode(member.parent).exports.get(expr.escapedText);
+                case 216 /* BinaryExpression */:
+                    return isStringConcatExpression(expr);
+                default:
+                    return false;
+            }
+        }
+        function getEnumKind(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (links.enumKind !== undefined) {
+                return links.enumKind;
+            }
+            var hasNonLiteralMember = false;
+            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                var declaration = _a[_i];
+                if (declaration.kind === 255 /* EnumDeclaration */) {
+                    for (var _b = 0, _c = declaration.members; _b < _c.length; _b++) {
+                        var member = _c[_b];
+                        if (member.initializer && ts.isStringLiteralLike(member.initializer)) {
+                            return links.enumKind = 1 /* Literal */;
+                        }
+                        if (!isLiteralEnumMember(member)) {
+                            hasNonLiteralMember = true;
+                        }
+                    }
                 }
             }
-            for (var _b = 0, _c = getPropertiesOfType(left); _b < _c.length; _b++) {
-                var leftProp = _c[_b];
-                if (skippedPrivateMembers.has(leftProp.escapedName) || !isSpreadableProperty(leftProp)) {
-                    continue;
+            return links.enumKind = hasNonLiteralMember ? 0 /* Numeric */ : 1 /* Literal */;
+        }
+        function getBaseTypeOfEnumLiteralType(type) {
+            return type.flags & 1024 /* EnumLiteral */ && !(type.flags & 1048576 /* Union */) ? getDeclaredTypeOfSymbol(getParentOfSymbol(type.symbol)) : type;
+        }
+        function getDeclaredTypeOfEnum(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (links.declaredType) {
+                return links.declaredType;
+            }
+            if (getEnumKind(symbol) === 1 /* Literal */) {
+                enumCount++;
+                var memberTypeList = [];
+                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                    var declaration = _a[_i];
+                    if (declaration.kind === 255 /* EnumDeclaration */) {
+                        for (var _b = 0, _c = declaration.members; _b < _c.length; _b++) {
+                            var member = _c[_b];
+                            var value = getEnumMemberValue(member);
+                            var memberType = getFreshTypeOfLiteralType(getLiteralType(value !== undefined ? value : 0, enumCount, getSymbolOfNode(member)));
+                            getSymbolLinks(getSymbolOfNode(member)).declaredType = memberType;
+                            memberTypeList.push(getRegularTypeOfLiteralType(memberType));
+                        }
+                    }
                 }
-                if (members.has(leftProp.escapedName)) {
-                    var rightProp = members.get(leftProp.escapedName);
-                    var rightType = getTypeOfSymbol(rightProp);
-                    if (rightProp.flags & 16777216 /* Optional */) {
-                        var declarations = ts.concatenate(leftProp.declarations, rightProp.declarations);
-                        var flags = 4 /* Property */ | (leftProp.flags & 16777216 /* Optional */);
-                        var result = createSymbol(flags, leftProp.escapedName);
-                        result.type = getUnionType([getTypeOfSymbol(leftProp), getTypeWithFacts(rightType, 524288 /* NEUndefined */)]);
-                        result.leftSpread = leftProp;
-                        result.rightSpread = rightProp;
-                        result.declarations = declarations;
-                        result.nameType = getSymbolLinks(leftProp).nameType;
-                        members.set(leftProp.escapedName, result);
+                if (memberTypeList.length) {
+                    var enumType_1 = getUnionType(memberTypeList, 1 /* Literal */, symbol, /*aliasTypeArguments*/ undefined);
+                    if (enumType_1.flags & 1048576 /* Union */) {
+                        enumType_1.flags |= 1024 /* EnumLiteral */;
+                        enumType_1.symbol = symbol;
                     }
+                    return links.declaredType = enumType_1;
                 }
-                else {
-                    members.set(leftProp.escapedName, getSpreadSymbol(leftProp, readonly));
+            }
+            var enumType = createType(32 /* Enum */);
+            enumType.symbol = symbol;
+            return links.declaredType = enumType;
+        }
+        function getDeclaredTypeOfEnumMember(symbol) {
+            var links = getSymbolLinks(symbol);
+            if (!links.declaredType) {
+                var enumType = getDeclaredTypeOfEnum(getParentOfSymbol(symbol));
+                if (!links.declaredType) {
+                    links.declaredType = enumType;
                 }
             }
-            var spread = createAnonymousType(symbol, members, ts.emptyArray, ts.emptyArray, getIndexInfoWithReadonly(stringIndexInfo, readonly), getIndexInfoWithReadonly(numberIndexInfo, readonly));
-            spread.objectFlags |= 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */ | 1024 /* ContainsSpread */ | objectFlags;
-            return spread;
+            return links.declaredType;
         }
-        /** We approximate own properties as non-methods plus methods that are inside the object literal */
-        function isSpreadableProperty(prop) {
-            return !ts.some(prop.declarations, ts.isPrivateIdentifierPropertyDeclaration) &&
-                (!(prop.flags & (8192 /* Method */ | 32768 /* GetAccessor */ | 65536 /* SetAccessor */)) ||
-                    !prop.declarations.some(function (decl) { return ts.isClassLike(decl.parent); }));
+        function getDeclaredTypeOfTypeParameter(symbol) {
+            var links = getSymbolLinks(symbol);
+            return links.declaredType || (links.declaredType = createTypeParameter(symbol));
         }
-        function getSpreadSymbol(prop, readonly) {
-            var isSetonlyAccessor = prop.flags & 65536 /* SetAccessor */ && !(prop.flags & 32768 /* GetAccessor */);
-            if (!isSetonlyAccessor && readonly === isReadonlySymbol(prop)) {
-                return prop;
+        function getDeclaredTypeOfAlias(symbol) {
+            var links = getSymbolLinks(symbol);
+            return links.declaredType || (links.declaredType = getDeclaredTypeOfSymbol(resolveAlias(symbol)));
+        }
+        function getDeclaredTypeOfSymbol(symbol) {
+            return tryGetDeclaredTypeOfSymbol(symbol) || errorType;
+        }
+        function tryGetDeclaredTypeOfSymbol(symbol) {
+            if (symbol.flags & (32 /* Class */ | 64 /* Interface */)) {
+                return getDeclaredTypeOfClassOrInterface(symbol);
             }
-            var flags = 4 /* Property */ | (prop.flags & 16777216 /* Optional */);
-            var result = createSymbol(flags, prop.escapedName, readonly ? 8 /* Readonly */ : 0);
-            result.type = isSetonlyAccessor ? undefinedType : getTypeOfSymbol(prop);
-            result.declarations = prop.declarations;
-            result.nameType = getSymbolLinks(prop).nameType;
-            result.syntheticOrigin = prop;
-            return result;
+            if (symbol.flags & 524288 /* TypeAlias */) {
+                return getDeclaredTypeOfTypeAlias(symbol);
+            }
+            if (symbol.flags & 262144 /* TypeParameter */) {
+                return getDeclaredTypeOfTypeParameter(symbol);
+            }
+            if (symbol.flags & 384 /* Enum */) {
+                return getDeclaredTypeOfEnum(symbol);
+            }
+            if (symbol.flags & 8 /* EnumMember */) {
+                return getDeclaredTypeOfEnumMember(symbol);
+            }
+            if (symbol.flags & 2097152 /* Alias */) {
+                return getDeclaredTypeOfAlias(symbol);
+            }
+            return undefined;
         }
-        function getIndexInfoWithReadonly(info, readonly) {
-            return info && info.isReadonly !== readonly ? createIndexInfo(info.type, readonly, info.declaration) : info;
+        /**
+         * A type is free of this references if it's the any, string, number, boolean, symbol, or void keyword, a string
+         * literal type, an array with an element type that is free of this references, or a type reference that is
+         * free of this references.
+         */
+        function isThislessType(node) {
+            switch (node.kind) {
+                case 128 /* AnyKeyword */:
+                case 152 /* UnknownKeyword */:
+                case 147 /* StringKeyword */:
+                case 144 /* NumberKeyword */:
+                case 155 /* BigIntKeyword */:
+                case 131 /* BooleanKeyword */:
+                case 148 /* SymbolKeyword */:
+                case 145 /* ObjectKeyword */:
+                case 113 /* VoidKeyword */:
+                case 150 /* UndefinedKeyword */:
+                case 141 /* NeverKeyword */:
+                case 191 /* LiteralType */:
+                    return true;
+                case 178 /* ArrayType */:
+                    return isThislessType(node.elementType);
+                case 173 /* TypeReference */:
+                    return !node.typeArguments || node.typeArguments.every(isThislessType);
+            }
+            return false;
         }
-        function createLiteralType(flags, value, symbol) {
-            var type = createType(flags);
-            type.symbol = symbol;
-            type.value = value;
-            return type;
+        /** A type parameter is thisless if its constraint is thisless, or if it has no constraint. */
+        function isThislessTypeParameter(node) {
+            var constraint = ts.getEffectiveConstraintOfTypeParameter(node);
+            return !constraint || isThislessType(constraint);
         }
-        function getFreshTypeOfLiteralType(type) {
-            if (type.flags & 2944 /* Literal */) {
-                if (!type.freshType) {
-                    var freshType = createLiteralType(type.flags, type.value, type.symbol);
-                    freshType.regularType = type;
-                    freshType.freshType = freshType;
-                    type.freshType = freshType;
+        /**
+         * A variable-like declaration is free of this references if it has a type annotation
+         * that is thisless, or if it has no type annotation and no initializer (and is thus of type any).
+         */
+        function isThislessVariableLikeDeclaration(node) {
+            var typeNode = ts.getEffectiveTypeAnnotationNode(node);
+            return typeNode ? isThislessType(typeNode) : !ts.hasInitializer(node);
+        }
+        /**
+         * A function-like declaration is considered free of `this` references if it has a return type
+         * annotation that is free of this references and if each parameter is thisless and if
+         * each type parameter (if present) is thisless.
+         */
+        function isThislessFunctionLikeDeclaration(node) {
+            var returnType = ts.getEffectiveReturnTypeNode(node);
+            var typeParameters = ts.getEffectiveTypeParameterDeclarations(node);
+            return (node.kind === 166 /* Constructor */ || (!!returnType && isThislessType(returnType))) &&
+                node.parameters.every(isThislessVariableLikeDeclaration) &&
+                typeParameters.every(isThislessTypeParameter);
+        }
+        /**
+         * Returns true if the class or interface member given by the symbol is free of "this" references. The
+         * function may return false for symbols that are actually free of "this" references because it is not
+         * feasible to perform a complete analysis in all cases. In particular, property members with types
+         * inferred from their initializers and function members with inferred return types are conservatively
+         * assumed not to be free of "this" references.
+         */
+        function isThisless(symbol) {
+            if (symbol.declarations && symbol.declarations.length === 1) {
+                var declaration = symbol.declarations[0];
+                if (declaration) {
+                    switch (declaration.kind) {
+                        case 163 /* PropertyDeclaration */:
+                        case 162 /* PropertySignature */:
+                            return isThislessVariableLikeDeclaration(declaration);
+                        case 165 /* MethodDeclaration */:
+                        case 164 /* MethodSignature */:
+                        case 166 /* Constructor */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
+                            return isThislessFunctionLikeDeclaration(declaration);
+                    }
                 }
-                return type.freshType;
             }
-            return type;
+            return false;
         }
-        function getRegularTypeOfLiteralType(type) {
-            return type.flags & 2944 /* Literal */ ? type.regularType :
-                type.flags & 1048576 /* Union */ ? (type.regularType || (type.regularType = getUnionType(ts.sameMap(type.types, getRegularTypeOfLiteralType)))) :
-                    type;
+        // The mappingThisOnly flag indicates that the only type parameter being mapped is "this". When the flag is true,
+        // we check symbols to see if we can quickly conclude they are free of "this" references, thus needing no instantiation.
+        function createInstantiatedSymbolTable(symbols, mapper, mappingThisOnly) {
+            var result = ts.createSymbolTable();
+            for (var _i = 0, symbols_2 = symbols; _i < symbols_2.length; _i++) {
+                var symbol = symbols_2[_i];
+                result.set(symbol.escapedName, mappingThisOnly && isThisless(symbol) ? symbol : instantiateSymbol(symbol, mapper));
+            }
+            return result;
         }
-        function isFreshLiteralType(type) {
-            return !!(type.flags & 2944 /* Literal */) && type.freshType === type;
+        function addInheritedMembers(symbols, baseSymbols) {
+            for (var _i = 0, baseSymbols_1 = baseSymbols; _i < baseSymbols_1.length; _i++) {
+                var s = baseSymbols_1[_i];
+                if (!symbols.has(s.escapedName) && !isStaticPrivateIdentifierProperty(s)) {
+                    symbols.set(s.escapedName, s);
+                }
+            }
         }
-        function getLiteralType(value, enumId, symbol) {
-            // We store all literal types in a single map with keys of the form '#NNN' and '@SSS',
-            // where NNN is the text representation of a numeric literal and SSS are the characters
-            // of a string literal. For literal enum members we use 'EEE#NNN' and 'EEE@SSS', where
-            // EEE is a unique id for the containing enum type.
-            var qualifier = typeof value === "number" ? "#" : typeof value === "string" ? "@" : "n";
-            var key = (enumId ? enumId : "") + qualifier + (typeof value === "object" ? ts.pseudoBigIntToString(value) : value);
-            var type = literalTypes.get(key);
-            if (!type) {
-                var flags = (typeof value === "number" ? 256 /* NumberLiteral */ :
-                    typeof value === "string" ? 128 /* StringLiteral */ : 2048 /* BigIntLiteral */) |
-                    (enumId ? 1024 /* EnumLiteral */ : 0);
-                literalTypes.set(key, type = createLiteralType(flags, value, symbol));
-                type.regularType = type;
+        function isStaticPrivateIdentifierProperty(s) {
+            return !!s.valueDeclaration && ts.isPrivateIdentifierPropertyDeclaration(s.valueDeclaration) && ts.hasSyntacticModifier(s.valueDeclaration, 32 /* Static */);
+        }
+        function resolveDeclaredMembers(type) {
+            if (!type.declaredProperties) {
+                var symbol = type.symbol;
+                var members = getMembersOfSymbol(symbol);
+                type.declaredProperties = getNamedMembers(members);
+                // Start with signatures at empty array in case of recursive types
+                type.declaredCallSignatures = ts.emptyArray;
+                type.declaredConstructSignatures = ts.emptyArray;
+                type.declaredCallSignatures = getSignaturesOfSymbol(members.get("__call" /* Call */));
+                type.declaredConstructSignatures = getSignaturesOfSymbol(members.get("__new" /* New */));
+                type.declaredStringIndexInfo = getIndexInfoOfSymbol(symbol, 0 /* String */);
+                type.declaredNumberIndexInfo = getIndexInfoOfSymbol(symbol, 1 /* Number */);
             }
             return type;
         }
-        function getTypeFromLiteralTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                links.resolvedType = getRegularTypeOfLiteralType(checkExpression(node.literal));
+        /**
+         * Indicates whether a type can be used as a property name.
+         */
+        function isTypeUsableAsPropertyName(type) {
+            return !!(type.flags & 8576 /* StringOrNumberLiteralOrUnique */);
+        }
+        /**
+         * Indicates whether a declaration name is definitely late-bindable.
+         * A declaration name is only late-bindable if:
+         * - It is a `ComputedPropertyName`.
+         * - Its expression is an `Identifier` or either a `PropertyAccessExpression` an
+         * `ElementAccessExpression` consisting only of these same three types of nodes.
+         * - The type of its expression is a string or numeric literal type, or is a `unique symbol` type.
+         */
+        function isLateBindableName(node) {
+            if (!ts.isComputedPropertyName(node) && !ts.isElementAccessExpression(node)) {
+                return false;
             }
-            return links.resolvedType;
+            var expr = ts.isComputedPropertyName(node) ? node.expression : node.argumentExpression;
+            return ts.isEntityNameExpression(expr)
+                && isTypeUsableAsPropertyName(ts.isComputedPropertyName(node) ? checkComputedPropertyName(node) : checkExpressionCached(expr));
         }
-        function createUniqueESSymbolType(symbol) {
-            var type = createType(8192 /* UniqueESSymbol */);
-            type.symbol = symbol;
-            type.escapedName = "__@" + type.symbol.escapedName + "@" + getSymbolId(type.symbol);
-            return type;
+        function isLateBoundName(name) {
+            return name.charCodeAt(0) === 95 /* _ */ &&
+                name.charCodeAt(1) === 95 /* _ */ &&
+                name.charCodeAt(2) === 64 /* at */;
         }
-        function getESSymbolLikeTypeForNode(node) {
-            if (ts.isValidESSymbolDeclaration(node)) {
-                var symbol = getSymbolOfNode(node);
-                var links = getSymbolLinks(symbol);
-                return links.uniqueESSymbolType || (links.uniqueESSymbolType = createUniqueESSymbolType(symbol));
-            }
-            return esSymbolType;
+        /**
+         * Indicates whether a declaration has a late-bindable dynamic name.
+         */
+        function hasLateBindableName(node) {
+            var name = ts.getNameOfDeclaration(node);
+            return !!name && isLateBindableName(name);
         }
-        function getThisType(node) {
-            var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
-            var parent = container && container.parent;
-            if (parent && (ts.isClassLike(parent) || parent.kind === 246 /* InterfaceDeclaration */)) {
-                if (!ts.hasModifier(container, 32 /* Static */) &&
-                    (!ts.isConstructorDeclaration(container) || ts.isNodeDescendantOf(node, container.body))) {
-                    return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(parent)).thisType;
-                }
+        /**
+         * Indicates whether a declaration has a dynamic name that cannot be late-bound.
+         */
+        function hasNonBindableDynamicName(node) {
+            return ts.hasDynamicName(node) && !hasLateBindableName(node);
+        }
+        /**
+         * Indicates whether a declaration name is a dynamic name that cannot be late-bound.
+         */
+        function isNonBindableDynamicName(node) {
+            return ts.isDynamicName(node) && !isLateBindableName(node);
+        }
+        /**
+         * Gets the symbolic name for a member from its type.
+         */
+        function getPropertyNameFromType(type) {
+            if (type.flags & 8192 /* UniqueESSymbol */) {
+                return type.escapedName;
             }
-            // inside x.prototype = { ... }
-            if (parent && ts.isObjectLiteralExpression(parent) && ts.isBinaryExpression(parent.parent) && ts.getAssignmentDeclarationKind(parent.parent) === 6 /* Prototype */) {
-                return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(parent.parent.left).parent).thisType;
+            if (type.flags & (128 /* StringLiteral */ | 256 /* NumberLiteral */)) {
+                return ts.escapeLeadingUnderscores("" + type.value);
             }
-            // /** @return {this} */
-            // x.prototype.m = function() { ... }
-            var host = node.flags & 4194304 /* JSDoc */ ? ts.getHostSignatureFromJSDoc(node) : undefined;
-            if (host && ts.isFunctionExpression(host) && ts.isBinaryExpression(host.parent) && ts.getAssignmentDeclarationKind(host.parent) === 3 /* PrototypeProperty */) {
-                return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(host.parent.left).parent).thisType;
+            return ts.Debug.fail();
+        }
+        /**
+         * Adds a declaration to a late-bound dynamic member. This performs the same function for
+         * late-bound members that `addDeclarationToSymbol` in binder.ts performs for early-bound
+         * members.
+         */
+        function addDeclarationToLateBoundSymbol(symbol, member, symbolFlags) {
+            ts.Debug.assert(!!(ts.getCheckFlags(symbol) & 4096 /* Late */), "Expected a late-bound symbol.");
+            symbol.flags |= symbolFlags;
+            getSymbolLinks(member.symbol).lateSymbol = symbol;
+            if (!symbol.declarations) {
+                symbol.declarations = [member];
             }
-            // inside constructor function C() { ... }
-            if (isJSConstructor(container) && ts.isNodeDescendantOf(node, container.body)) {
-                return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(container)).thisType;
+            else {
+                symbol.declarations.push(member);
             }
-            error(node, ts.Diagnostics.A_this_type_is_available_only_in_a_non_static_member_of_a_class_or_interface);
-            return errorType;
-        }
-        function getTypeFromThisTypeNode(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                links.resolvedType = getThisType(node);
+            if (symbolFlags & 111551 /* Value */) {
+                if (!symbol.valueDeclaration || symbol.valueDeclaration.kind !== member.kind) {
+                    symbol.valueDeclaration = member;
+                }
             }
-            return links.resolvedType;
         }
-        function getTypeFromTypeNode(node) {
-            return getConditionalFlowTypeOfType(getTypeFromTypeNodeWorker(node), node);
-        }
-        function getTypeFromTypeNodeWorker(node) {
-            switch (node.kind) {
-                case 125 /* AnyKeyword */:
-                case 295 /* JSDocAllType */:
-                case 296 /* JSDocUnknownType */:
-                    return anyType;
-                case 148 /* UnknownKeyword */:
-                    return unknownType;
-                case 143 /* StringKeyword */:
-                    return stringType;
-                case 140 /* NumberKeyword */:
-                    return numberType;
-                case 151 /* BigIntKeyword */:
-                    return bigintType;
-                case 128 /* BooleanKeyword */:
-                    return booleanType;
-                case 144 /* SymbolKeyword */:
-                    return esSymbolType;
-                case 110 /* VoidKeyword */:
-                    return voidType;
-                case 146 /* UndefinedKeyword */:
-                    return undefinedType;
-                case 100 /* NullKeyword */:
-                    return nullType;
-                case 137 /* NeverKeyword */:
-                    return neverType;
-                case 141 /* ObjectKeyword */:
-                    return node.flags & 131072 /* JavaScriptFile */ && !noImplicitAny ? anyType : nonPrimitiveType;
-                case 183 /* ThisType */:
-                case 104 /* ThisKeyword */:
-                    return getTypeFromThisTypeNode(node);
-                case 187 /* LiteralType */:
-                    return getTypeFromLiteralTypeNode(node);
-                case 169 /* TypeReference */:
-                    return getTypeFromTypeReference(node);
-                case 168 /* TypePredicate */:
-                    return node.assertsModifier ? voidType : booleanType;
-                case 216 /* ExpressionWithTypeArguments */:
-                    return getTypeFromTypeReference(node);
-                case 172 /* TypeQuery */:
-                    return getTypeFromTypeQueryNode(node);
-                case 174 /* ArrayType */:
-                case 175 /* TupleType */:
-                    return getTypeFromArrayOrTupleTypeNode(node);
-                case 176 /* OptionalType */:
-                    return getTypeFromOptionalTypeNode(node);
-                case 178 /* UnionType */:
-                    return getTypeFromUnionTypeNode(node);
-                case 179 /* IntersectionType */:
-                    return getTypeFromIntersectionTypeNode(node);
-                case 297 /* JSDocNullableType */:
-                    return getTypeFromJSDocNullableTypeNode(node);
-                case 299 /* JSDocOptionalType */:
-                    return addOptionality(getTypeFromTypeNode(node.type));
-                case 182 /* ParenthesizedType */:
-                case 298 /* JSDocNonNullableType */:
-                case 294 /* JSDocTypeExpression */:
-                    return getTypeFromTypeNode(node.type);
-                case 177 /* RestType */:
-                    return getElementTypeOfArrayType(getTypeFromTypeNode(node.type)) || errorType;
-                case 301 /* JSDocVariadicType */:
-                    return getTypeFromJSDocVariadicType(node);
-                case 170 /* FunctionType */:
-                case 171 /* ConstructorType */:
-                case 173 /* TypeLiteral */:
-                case 304 /* JSDocTypeLiteral */:
-                case 300 /* JSDocFunctionType */:
-                case 305 /* JSDocSignature */:
-                    return getTypeFromTypeLiteralOrFunctionOrConstructorTypeNode(node);
-                case 184 /* TypeOperator */:
-                    return getTypeFromTypeOperatorNode(node);
-                case 185 /* IndexedAccessType */:
-                    return getTypeFromIndexedAccessTypeNode(node);
-                case 186 /* MappedType */:
-                    return getTypeFromMappedTypeNode(node);
-                case 180 /* ConditionalType */:
-                    return getTypeFromConditionalTypeNode(node);
-                case 181 /* InferType */:
-                    return getTypeFromInferTypeNode(node);
-                case 188 /* ImportType */:
-                    return getTypeFromImportTypeNode(node);
-                // This function assumes that an identifier or qualified name is a type expression
-                // Callers should first ensure this by calling isTypeNode
-                case 75 /* Identifier */:
-                case 153 /* QualifiedName */:
-                    var symbol = getSymbolAtLocation(node);
-                    return symbol ? getDeclaredTypeOfSymbol(symbol) : errorType;
-                default:
-                    return errorType;
+        /**
+         * Performs late-binding of a dynamic member. This performs the same function for
+         * late-bound members that `declareSymbol` in binder.ts performs for early-bound
+         * members.
+         *
+         * If a symbol is a dynamic name from a computed property, we perform an additional "late"
+         * binding phase to attempt to resolve the name for the symbol from the type of the computed
+         * property's expression. If the type of the expression is a string-literal, numeric-literal,
+         * or unique symbol type, we can use that type as the name of the symbol.
+         *
+         * For example, given:
+         *
+         *   const x = Symbol();
+         *
+         *   interface I {
+         *     [x]: number;
+         *   }
+         *
+         * The binder gives the property `[x]: number` a special symbol with the name "__computed".
+         * In the late-binding phase we can type-check the expression `x` and see that it has a
+         * unique symbol type which we can then use as the name of the member. This allows users
+         * to define custom symbols that can be used in the members of an object type.
+         *
+         * @param parent The containing symbol for the member.
+         * @param earlySymbols The early-bound symbols of the parent.
+         * @param lateSymbols The late-bound symbols of the parent.
+         * @param decl The member to bind.
+         */
+        function lateBindMember(parent, earlySymbols, lateSymbols, decl) {
+            ts.Debug.assert(!!decl.symbol, "The member is expected to have a symbol.");
+            var links = getNodeLinks(decl);
+            if (!links.resolvedSymbol) {
+                // In the event we attempt to resolve the late-bound name of this member recursively,
+                // fall back to the early-bound name of this member.
+                links.resolvedSymbol = decl.symbol;
+                var declName = ts.isBinaryExpression(decl) ? decl.left : decl.name;
+                var type = ts.isElementAccessExpression(declName) ? checkExpressionCached(declName.argumentExpression) : checkComputedPropertyName(declName);
+                if (isTypeUsableAsPropertyName(type)) {
+                    var memberName = getPropertyNameFromType(type);
+                    var symbolFlags = decl.symbol.flags;
+                    // Get or add a late-bound symbol for the member. This allows us to merge late-bound accessor declarations.
+                    var lateSymbol = lateSymbols.get(memberName);
+                    if (!lateSymbol)
+                        lateSymbols.set(memberName, lateSymbol = createSymbol(0 /* None */, memberName, 4096 /* Late */));
+                    // Report an error if a late-bound member has the same name as an early-bound member,
+                    // or if we have another early-bound symbol declaration with the same name and
+                    // conflicting flags.
+                    var earlySymbol = earlySymbols && earlySymbols.get(memberName);
+                    if (lateSymbol.flags & getExcludedSymbolFlags(symbolFlags) || earlySymbol) {
+                        // If we have an existing early-bound member, combine its declarations so that we can
+                        // report an error at each declaration.
+                        var declarations = earlySymbol ? ts.concatenate(earlySymbol.declarations, lateSymbol.declarations) : lateSymbol.declarations;
+                        var name_4 = !(type.flags & 8192 /* UniqueESSymbol */) && ts.unescapeLeadingUnderscores(memberName) || ts.declarationNameToString(declName);
+                        ts.forEach(declarations, function (declaration) { return error(ts.getNameOfDeclaration(declaration) || declaration, ts.Diagnostics.Property_0_was_also_declared_here, name_4); });
+                        error(declName || decl, ts.Diagnostics.Duplicate_property_0, name_4);
+                        lateSymbol = createSymbol(0 /* None */, memberName, 4096 /* Late */);
+                    }
+                    lateSymbol.nameType = type;
+                    addDeclarationToLateBoundSymbol(lateSymbol, decl, symbolFlags);
+                    if (lateSymbol.parent) {
+                        ts.Debug.assert(lateSymbol.parent === parent, "Existing symbol parent should match new one");
+                    }
+                    else {
+                        lateSymbol.parent = parent;
+                    }
+                    return links.resolvedSymbol = lateSymbol;
+                }
             }
+            return links.resolvedSymbol;
         }
-        function instantiateList(items, mapper, instantiator) {
-            if (items && items.length) {
-                for (var i = 0; i < items.length; i++) {
-                    var item = items[i];
-                    var mapped = instantiator(item, mapper);
-                    if (item !== mapped) {
-                        var result = i === 0 ? [] : items.slice(0, i);
-                        result.push(mapped);
-                        for (i++; i < items.length; i++) {
-                            result.push(instantiator(items[i], mapper));
+        function getResolvedMembersOrExportsOfSymbol(symbol, resolutionKind) {
+            var links = getSymbolLinks(symbol);
+            if (!links[resolutionKind]) {
+                var isStatic = resolutionKind === "resolvedExports" /* resolvedExports */;
+                var earlySymbols = !isStatic ? symbol.members :
+                    symbol.flags & 1536 /* Module */ ? getExportsOfModuleWorker(symbol) :
+                        symbol.exports;
+                // In the event we recursively resolve the members/exports of the symbol, we
+                // set the initial value of resolvedMembers/resolvedExports to the early-bound
+                // members/exports of the symbol.
+                links[resolutionKind] = earlySymbols || emptySymbols;
+                // fill in any as-yet-unresolved late-bound members.
+                var lateSymbols = ts.createSymbolTable();
+                for (var _i = 0, _a = symbol.declarations || ts.emptyArray; _i < _a.length; _i++) {
+                    var decl = _a[_i];
+                    var members = ts.getMembersOfDeclaration(decl);
+                    if (members) {
+                        for (var _b = 0, members_5 = members; _b < members_5.length; _b++) {
+                            var member = members_5[_b];
+                            if (isStatic === ts.hasStaticModifier(member) && hasLateBindableName(member)) {
+                                lateBindMember(symbol, earlySymbols, lateSymbols, member);
+                            }
                         }
-                        return result;
                     }
                 }
-            }
-            return items;
-        }
-        function instantiateTypes(types, mapper) {
-            return instantiateList(types, mapper, instantiateType);
-        }
-        function instantiateSignatures(signatures, mapper) {
-            return instantiateList(signatures, mapper, instantiateSignature);
-        }
-        function createTypeMapper(sources, targets) {
-            return sources.length === 1 ? makeUnaryTypeMapper(sources[0], targets ? targets[0] : anyType) : makeArrayTypeMapper(sources, targets);
-        }
-        function getMappedType(type, mapper) {
-            switch (mapper.kind) {
-                case 0 /* Simple */:
-                    return type === mapper.source ? mapper.target : type;
-                case 1 /* Array */:
-                    var sources = mapper.sources;
-                    var targets = mapper.targets;
-                    for (var i = 0; i < sources.length; i++) {
-                        if (type === sources[i]) {
-                            return targets ? targets[i] : anyType;
+                var assignments = symbol.assignmentDeclarationMembers;
+                if (assignments) {
+                    var decls = ts.arrayFrom(assignments.values());
+                    for (var _c = 0, decls_1 = decls; _c < decls_1.length; _c++) {
+                        var member = decls_1[_c];
+                        var assignmentKind = ts.getAssignmentDeclarationKind(member);
+                        var isInstanceMember = assignmentKind === 3 /* PrototypeProperty */
+                            || ts.isBinaryExpression(member) && isPossiblyAliasedThisProperty(member, assignmentKind)
+                            || assignmentKind === 9 /* ObjectDefinePrototypeProperty */
+                            || assignmentKind === 6 /* Prototype */; // A straight `Prototype` assignment probably can never have a computed name
+                        if (isStatic === !isInstanceMember && hasLateBindableName(member)) {
+                            lateBindMember(symbol, earlySymbols, lateSymbols, member);
                         }
                     }
-                    return type;
-                case 2 /* Function */:
-                    return mapper.func(type);
-                case 3 /* Composite */:
-                case 4 /* Merged */:
-                    var t1 = getMappedType(type, mapper.mapper1);
-                    return t1 !== type && mapper.kind === 3 /* Composite */ ? instantiateType(t1, mapper.mapper2) : getMappedType(t1, mapper.mapper2);
+                }
+                links[resolutionKind] = combineSymbolTables(earlySymbols, lateSymbols) || emptySymbols;
             }
+            return links[resolutionKind];
         }
-        function makeUnaryTypeMapper(source, target) {
-            return { kind: 0 /* Simple */, source: source, target: target };
-        }
-        function makeArrayTypeMapper(sources, targets) {
-            return { kind: 1 /* Array */, sources: sources, targets: targets };
-        }
-        function makeFunctionTypeMapper(func) {
-            return { kind: 2 /* Function */, func: func };
-        }
-        function makeCompositeTypeMapper(kind, mapper1, mapper2) {
-            return { kind: kind, mapper1: mapper1, mapper2: mapper2 };
-        }
-        function createTypeEraser(sources) {
-            return createTypeMapper(sources, /*targets*/ undefined);
+        /**
+         * Gets a SymbolTable containing both the early- and late-bound members of a symbol.
+         *
+         * For a description of late-binding, see `lateBindMember`.
+         */
+        function getMembersOfSymbol(symbol) {
+            return symbol.flags & 6256 /* LateBindingContainer */
+                ? getResolvedMembersOrExportsOfSymbol(symbol, "resolvedMembers" /* resolvedMembers */)
+                : symbol.members || emptySymbols;
         }
         /**
-         * Maps forward-references to later types parameters to the empty object type.
-         * This is used during inference when instantiating type parameter defaults.
+         * If a symbol is the dynamic name of the member of an object type, get the late-bound
+         * symbol of the member.
+         *
+         * For a description of late-binding, see `lateBindMember`.
          */
-        function createBackreferenceMapper(context, index) {
-            return makeFunctionTypeMapper(function (t) { return ts.findIndex(context.inferences, function (info) { return info.typeParameter === t; }) >= index ? unknownType : t; });
+        function getLateBoundSymbol(symbol) {
+            if (symbol.flags & 106500 /* ClassMember */ && symbol.escapedName === "__computed" /* Computed */) {
+                var links = getSymbolLinks(symbol);
+                if (!links.lateSymbol && ts.some(symbol.declarations, hasLateBindableName)) {
+                    // force late binding of members/exports. This will set the late-bound symbol
+                    var parent = getMergedSymbol(symbol.parent);
+                    if (ts.some(symbol.declarations, ts.hasStaticModifier)) {
+                        getExportsOfSymbol(parent);
+                    }
+                    else {
+                        getMembersOfSymbol(parent);
+                    }
+                }
+                return links.lateSymbol || (links.lateSymbol = symbol);
+            }
+            return symbol;
         }
-        function combineTypeMappers(mapper1, mapper2) {
-            return mapper1 ? makeCompositeTypeMapper(3 /* Composite */, mapper1, mapper2) : mapper2;
+        function getTypeWithThisArgument(type, thisArgument, needApparentType) {
+            if (ts.getObjectFlags(type) & 4 /* Reference */) {
+                var target = type.target;
+                var typeArguments = getTypeArguments(type);
+                if (ts.length(target.typeParameters) === ts.length(typeArguments)) {
+                    var ref = createTypeReference(target, ts.concatenate(typeArguments, [thisArgument || target.thisType]));
+                    return needApparentType ? getApparentType(ref) : ref;
+                }
+            }
+            else if (type.flags & 2097152 /* Intersection */) {
+                return getIntersectionType(ts.map(type.types, function (t) { return getTypeWithThisArgument(t, thisArgument, needApparentType); }));
+            }
+            return needApparentType ? getApparentType(type) : type;
         }
-        function mergeTypeMappers(mapper1, mapper2) {
-            return mapper1 ? makeCompositeTypeMapper(4 /* Merged */, mapper1, mapper2) : mapper2;
+        function resolveObjectTypeMembers(type, source, typeParameters, typeArguments) {
+            var mapper;
+            var members;
+            var callSignatures;
+            var constructSignatures;
+            var stringIndexInfo;
+            var numberIndexInfo;
+            if (ts.rangeEquals(typeParameters, typeArguments, 0, typeParameters.length)) {
+                members = source.symbol ? getMembersOfSymbol(source.symbol) : ts.createSymbolTable(source.declaredProperties);
+                callSignatures = source.declaredCallSignatures;
+                constructSignatures = source.declaredConstructSignatures;
+                stringIndexInfo = source.declaredStringIndexInfo;
+                numberIndexInfo = source.declaredNumberIndexInfo;
+            }
+            else {
+                mapper = createTypeMapper(typeParameters, typeArguments);
+                members = createInstantiatedSymbolTable(source.declaredProperties, mapper, /*mappingThisOnly*/ typeParameters.length === 1);
+                callSignatures = instantiateSignatures(source.declaredCallSignatures, mapper);
+                constructSignatures = instantiateSignatures(source.declaredConstructSignatures, mapper);
+                stringIndexInfo = instantiateIndexInfo(source.declaredStringIndexInfo, mapper);
+                numberIndexInfo = instantiateIndexInfo(source.declaredNumberIndexInfo, mapper);
+            }
+            var baseTypes = getBaseTypes(source);
+            if (baseTypes.length) {
+                if (source.symbol && members === getMembersOfSymbol(source.symbol)) {
+                    members = ts.createSymbolTable(source.declaredProperties);
+                }
+                setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
+                var thisArgument = ts.lastOrUndefined(typeArguments);
+                for (var _i = 0, baseTypes_1 = baseTypes; _i < baseTypes_1.length; _i++) {
+                    var baseType = baseTypes_1[_i];
+                    var instantiatedBaseType = thisArgument ? getTypeWithThisArgument(instantiateType(baseType, mapper), thisArgument) : baseType;
+                    addInheritedMembers(members, getPropertiesOfType(instantiatedBaseType));
+                    callSignatures = ts.concatenate(callSignatures, getSignaturesOfType(instantiatedBaseType, 0 /* Call */));
+                    constructSignatures = ts.concatenate(constructSignatures, getSignaturesOfType(instantiatedBaseType, 1 /* Construct */));
+                    if (!stringIndexInfo) {
+                        stringIndexInfo = instantiatedBaseType === anyType ?
+                            createIndexInfo(anyType, /*isReadonly*/ false) :
+                            getIndexInfoOfType(instantiatedBaseType, 0 /* String */);
+                    }
+                    numberIndexInfo = numberIndexInfo || getIndexInfoOfType(instantiatedBaseType, 1 /* Number */);
+                }
+            }
+            setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
         }
-        function prependTypeMapping(source, target, mapper) {
-            return !mapper ? makeUnaryTypeMapper(source, target) : makeCompositeTypeMapper(4 /* Merged */, makeUnaryTypeMapper(source, target), mapper);
+        function resolveClassOrInterfaceMembers(type) {
+            resolveObjectTypeMembers(type, resolveDeclaredMembers(type), ts.emptyArray, ts.emptyArray);
         }
-        function appendTypeMapping(mapper, source, target) {
-            return !mapper ? makeUnaryTypeMapper(source, target) : makeCompositeTypeMapper(4 /* Merged */, mapper, makeUnaryTypeMapper(source, target));
+        function resolveTypeReferenceMembers(type) {
+            var source = resolveDeclaredMembers(type.target);
+            var typeParameters = ts.concatenate(source.typeParameters, [source.thisType]);
+            var typeArguments = getTypeArguments(type);
+            var paddedTypeArguments = typeArguments.length === typeParameters.length ? typeArguments : ts.concatenate(typeArguments, [type]);
+            resolveObjectTypeMembers(type, source, typeParameters, paddedTypeArguments);
         }
-        function getRestrictiveTypeParameter(tp) {
-            return tp.constraint === unknownType ? tp : tp.restrictiveInstantiation || (tp.restrictiveInstantiation = createTypeParameter(tp.symbol),
-                tp.restrictiveInstantiation.constraint = unknownType,
-                tp.restrictiveInstantiation);
+        function createSignature(declaration, typeParameters, thisParameter, parameters, resolvedReturnType, resolvedTypePredicate, minArgumentCount, flags) {
+            var sig = new Signature(checker, flags);
+            sig.declaration = declaration;
+            sig.typeParameters = typeParameters;
+            sig.parameters = parameters;
+            sig.thisParameter = thisParameter;
+            sig.resolvedReturnType = resolvedReturnType;
+            sig.resolvedTypePredicate = resolvedTypePredicate;
+            sig.minArgumentCount = minArgumentCount;
+            sig.resolvedMinArgumentCount = undefined;
+            sig.target = undefined;
+            sig.mapper = undefined;
+            sig.unionSignatures = undefined;
+            return sig;
         }
-        function cloneTypeParameter(typeParameter) {
-            var result = createTypeParameter(typeParameter.symbol);
-            result.target = typeParameter;
+        function cloneSignature(sig) {
+            var result = createSignature(sig.declaration, sig.typeParameters, sig.thisParameter, sig.parameters, /*resolvedReturnType*/ undefined, 
+            /*resolvedTypePredicate*/ undefined, sig.minArgumentCount, sig.flags & 19 /* PropagatingFlags */);
+            result.target = sig.target;
+            result.mapper = sig.mapper;
+            result.unionSignatures = sig.unionSignatures;
             return result;
         }
-        function instantiateTypePredicate(predicate, mapper) {
-            return createTypePredicate(predicate.kind, predicate.parameterName, predicate.parameterIndex, instantiateType(predicate.type, mapper));
+        function createUnionSignature(signature, unionSignatures) {
+            var result = cloneSignature(signature);
+            result.unionSignatures = unionSignatures;
+            result.target = undefined;
+            result.mapper = undefined;
+            return result;
         }
-        function instantiateSignature(signature, mapper, eraseTypeParameters) {
-            var freshTypeParameters;
-            if (signature.typeParameters && !eraseTypeParameters) {
-                // First create a fresh set of type parameters, then include a mapping from the old to the
-                // new type parameters in the mapper function. Finally store this mapper in the new type
-                // parameters such that we can use it when instantiating constraints.
-                freshTypeParameters = ts.map(signature.typeParameters, cloneTypeParameter);
-                mapper = combineTypeMappers(createTypeMapper(signature.typeParameters, freshTypeParameters), mapper);
-                for (var _i = 0, freshTypeParameters_1 = freshTypeParameters; _i < freshTypeParameters_1.length; _i++) {
-                    var tp = freshTypeParameters_1[_i];
-                    tp.mapper = mapper;
-                }
+        function getOptionalCallSignature(signature, callChainFlags) {
+            if ((signature.flags & 12 /* CallChainFlags */) === callChainFlags) {
+                return signature;
             }
-            // Don't compute resolvedReturnType and resolvedTypePredicate now,
-            // because using `mapper` now could trigger inferences to become fixed. (See `createInferenceContext`.)
-            // See GH#17600.
-            var result = createSignature(signature.declaration, freshTypeParameters, signature.thisParameter && instantiateSymbol(signature.thisParameter, mapper), instantiateList(signature.parameters, mapper, instantiateSymbol), 
-            /*resolvedReturnType*/ undefined, 
-            /*resolvedTypePredicate*/ undefined, signature.minArgumentCount, signature.flags & 3 /* PropagatingFlags */);
-            result.target = signature;
-            result.mapper = mapper;
+            if (!signature.optionalCallSignatureCache) {
+                signature.optionalCallSignatureCache = {};
+            }
+            var key = callChainFlags === 4 /* IsInnerCallChain */ ? "inner" : "outer";
+            return signature.optionalCallSignatureCache[key]
+                || (signature.optionalCallSignatureCache[key] = createOptionalCallSignature(signature, callChainFlags));
+        }
+        function createOptionalCallSignature(signature, callChainFlags) {
+            ts.Debug.assert(callChainFlags === 4 /* IsInnerCallChain */ || callChainFlags === 8 /* IsOuterCallChain */, "An optional call signature can either be for an inner call chain or an outer call chain, but not both.");
+            var result = cloneSignature(signature);
+            result.flags |= callChainFlags;
             return result;
         }
-        function instantiateSymbol(symbol, mapper) {
-            var links = getSymbolLinks(symbol);
-            if (links.type && !couldContainTypeVariables(links.type)) {
-                // If the type of the symbol is already resolved, and if that type could not possibly
-                // be affected by instantiation, simply return the symbol itself.
-                return symbol;
-            }
-            if (ts.getCheckFlags(symbol) & 1 /* Instantiated */) {
-                // If symbol being instantiated is itself a instantiation, fetch the original target and combine the
-                // type mappers. This ensures that original type identities are properly preserved and that aliases
-                // always reference a non-aliases.
-                symbol = links.target;
-                mapper = combineTypeMappers(links.mapper, mapper);
+        function getExpandedParameters(sig, skipUnionExpanding) {
+            if (signatureHasRestParameter(sig)) {
+                var restIndex_1 = sig.parameters.length - 1;
+                var restType = getTypeOfSymbol(sig.parameters[restIndex_1]);
+                if (isTupleType(restType)) {
+                    return [expandSignatureParametersWithTupleMembers(restType, restIndex_1)];
+                }
+                else if (!skipUnionExpanding && restType.flags & 1048576 /* Union */ && ts.every(restType.types, isTupleType)) {
+                    return ts.map(restType.types, function (t) { return expandSignatureParametersWithTupleMembers(t, restIndex_1); });
+                }
+            }
+            return [sig.parameters];
+            function expandSignatureParametersWithTupleMembers(restType, restIndex) {
+                var elementTypes = getTypeArguments(restType);
+                var associatedNames = restType.target.labeledElementDeclarations;
+                var restParams = ts.map(elementTypes, function (t, i) {
+                    // Lookup the label from the individual tuple passed in before falling back to the signature `rest` parameter name
+                    var tupleLabelName = !!associatedNames && getTupleElementLabel(associatedNames[i]);
+                    var name = tupleLabelName || getParameterNameAtPosition(sig, restIndex + i, restType);
+                    var flags = restType.target.elementFlags[i];
+                    var checkFlags = flags & 12 /* Variable */ ? 32768 /* RestParameter */ :
+                        flags & 2 /* Optional */ ? 16384 /* OptionalParameter */ : 0;
+                    var symbol = createSymbol(1 /* FunctionScopedVariable */, name, checkFlags);
+                    symbol.type = flags & 4 /* Rest */ ? createArrayType(t) : t;
+                    return symbol;
+                });
+                return ts.concatenate(sig.parameters.slice(0, restIndex), restParams);
             }
-            // Keep the flags from the symbol we're instantiating.  Mark that is instantiated, and
-            // also transient so that we can just store data on it directly.
-            var result = createSymbol(symbol.flags, symbol.escapedName, 1 /* Instantiated */ | ts.getCheckFlags(symbol) & (8 /* Readonly */ | 4096 /* Late */ | 16384 /* OptionalParameter */ | 32768 /* RestParameter */));
-            result.declarations = symbol.declarations;
-            result.parent = symbol.parent;
-            result.target = symbol;
-            result.mapper = mapper;
-            if (symbol.valueDeclaration) {
-                result.valueDeclaration = symbol.valueDeclaration;
+        }
+        function getDefaultConstructSignatures(classType) {
+            var baseConstructorType = getBaseConstructorTypeOfClass(classType);
+            var baseSignatures = getSignaturesOfType(baseConstructorType, 1 /* Construct */);
+            if (baseSignatures.length === 0) {
+                return [createSignature(undefined, classType.localTypeParameters, undefined, ts.emptyArray, classType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */)];
             }
-            if (links.nameType) {
-                result.nameType = links.nameType;
+            var baseTypeNode = getBaseTypeNodeOfClass(classType);
+            var isJavaScript = ts.isInJSFile(baseTypeNode);
+            var typeArguments = typeArgumentsFromTypeReferenceNode(baseTypeNode);
+            var typeArgCount = ts.length(typeArguments);
+            var result = [];
+            for (var _i = 0, baseSignatures_1 = baseSignatures; _i < baseSignatures_1.length; _i++) {
+                var baseSig = baseSignatures_1[_i];
+                var minTypeArgumentCount = getMinTypeArgumentCount(baseSig.typeParameters);
+                var typeParamCount = ts.length(baseSig.typeParameters);
+                if (isJavaScript || typeArgCount >= minTypeArgumentCount && typeArgCount <= typeParamCount) {
+                    var sig = typeParamCount ? createSignatureInstantiation(baseSig, fillMissingTypeArguments(typeArguments, baseSig.typeParameters, minTypeArgumentCount, isJavaScript)) : cloneSignature(baseSig);
+                    sig.typeParameters = classType.localTypeParameters;
+                    sig.resolvedReturnType = classType;
+                    result.push(sig);
+                }
             }
             return result;
         }
-        function getObjectTypeInstantiation(type, mapper) {
-            var target = type.objectFlags & 64 /* Instantiated */ ? type.target : type;
-            var node = type.objectFlags & 4 /* Reference */ ? type.node : type.symbol.declarations[0];
-            var links = getNodeLinks(node);
-            var typeParameters = links.outerTypeParameters;
-            if (!typeParameters) {
-                // The first time an anonymous type is instantiated we compute and store a list of the type
-                // parameters that are in scope (and therefore potentially referenced). For type literals that
-                // aren't the right hand side of a generic type alias declaration we optimize by reducing the
-                // set of type parameters to those that are possibly referenced in the literal.
-                var declaration_1 = node;
-                if (ts.isInJSFile(declaration_1)) {
-                    var paramTag = ts.findAncestor(declaration_1, ts.isJSDocParameterTag);
-                    if (paramTag) {
-                        var paramSymbol = ts.getParameterSymbolFromJSDoc(paramTag);
-                        if (paramSymbol) {
-                            declaration_1 = paramSymbol.valueDeclaration;
-                        }
-                    }
+        function findMatchingSignature(signatureList, signature, partialMatch, ignoreThisTypes, ignoreReturnTypes) {
+            for (var _i = 0, signatureList_1 = signatureList; _i < signatureList_1.length; _i++) {
+                var s = signatureList_1[_i];
+                if (compareSignaturesIdentical(s, signature, partialMatch, ignoreThisTypes, ignoreReturnTypes, partialMatch ? compareTypesSubtypeOf : compareTypesIdentical)) {
+                    return s;
                 }
-                var outerTypeParameters = getOuterTypeParameters(declaration_1, /*includeThisTypes*/ true);
-                if (isJSConstructor(declaration_1)) {
-                    var templateTagParameters = getTypeParametersFromDeclaration(declaration_1);
-                    outerTypeParameters = ts.addRange(outerTypeParameters, templateTagParameters);
+            }
+        }
+        function findMatchingSignatures(signatureLists, signature, listIndex) {
+            if (signature.typeParameters) {
+                // We require an exact match for generic signatures, so we only return signatures from the first
+                // signature list and only if they have exact matches in the other signature lists.
+                if (listIndex > 0) {
+                    return undefined;
                 }
-                typeParameters = outerTypeParameters || ts.emptyArray;
-                typeParameters = (target.objectFlags & 4 /* Reference */ || target.symbol.flags & 2048 /* TypeLiteral */) && !target.aliasTypeArguments ?
-                    ts.filter(typeParameters, function (tp) { return isTypeParameterPossiblyReferenced(tp, declaration_1); }) :
-                    typeParameters;
-                links.outerTypeParameters = typeParameters;
-                if (typeParameters.length) {
-                    links.instantiations = ts.createMap();
-                    links.instantiations.set(getTypeListId(typeParameters), target);
+                for (var i = 1; i < signatureLists.length; i++) {
+                    if (!findMatchingSignature(signatureLists[i], signature, /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ false)) {
+                        return undefined;
+                    }
                 }
+                return [signature];
             }
-            if (typeParameters.length) {
-                // We are instantiating an anonymous type that has one or more type parameters in scope. Apply the
-                // mapper to the type parameters to produce the effective list of type arguments, and compute the
-                // instantiation cache key from the type IDs of the type arguments.
-                var combinedMapper_1 = combineTypeMappers(type.mapper, mapper);
-                var typeArguments = ts.map(typeParameters, function (t) { return getMappedType(t, combinedMapper_1); });
-                var id = getTypeListId(typeArguments);
-                var result = links.instantiations.get(id);
-                if (!result) {
-                    var newMapper = createTypeMapper(typeParameters, typeArguments);
-                    result = target.objectFlags & 4 /* Reference */ ? createDeferredTypeReference(type.target, type.node, newMapper) :
-                        target.objectFlags & 32 /* Mapped */ ? instantiateMappedType(target, newMapper) :
-                            instantiateAnonymousType(target, newMapper);
-                    links.instantiations.set(id, result);
+            var result;
+            for (var i = 0; i < signatureLists.length; i++) {
+                // Allow matching non-generic signatures to have excess parameters and different return types.
+                // Prefer matching this types if possible.
+                var match = i === listIndex ? signature : findMatchingSignature(signatureLists[i], signature, /*partialMatch*/ true, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ true);
+                if (!match) {
+                    return undefined;
                 }
-                return result;
+                result = ts.appendIfUnique(result, match);
             }
-            return type;
-        }
-        function maybeTypeParameterReference(node) {
-            return !(node.kind === 153 /* QualifiedName */ ||
-                node.parent.kind === 169 /* TypeReference */ && node.parent.typeArguments && node === node.parent.typeName ||
-                node.parent.kind === 188 /* ImportType */ && node.parent.typeArguments && node === node.parent.qualifier);
+            return result;
         }
-        function isTypeParameterPossiblyReferenced(tp, node) {
-            // If the type parameter doesn't have exactly one declaration, if there are invening statement blocks
-            // between the node and the type parameter declaration, if the node contains actual references to the
-            // type parameter, or if the node contains type queries, we consider the type parameter possibly referenced.
-            if (tp.symbol && tp.symbol.declarations && tp.symbol.declarations.length === 1) {
-                var container = tp.symbol.declarations[0].parent;
-                for (var n = node; n !== container; n = n.parent) {
-                    if (!n || n.kind === 223 /* Block */ || n.kind === 180 /* ConditionalType */ && ts.forEachChild(n.extendsType, containsReference)) {
-                        return true;
+        // The signatures of a union type are those signatures that are present in each of the constituent types.
+        // Generic signatures must match exactly, but non-generic signatures are allowed to have extra optional
+        // parameters and may differ in return types. When signatures differ in return types, the resulting return
+        // type is the union of the constituent return types.
+        function getUnionSignatures(signatureLists) {
+            var result;
+            var indexWithLengthOverOne;
+            for (var i = 0; i < signatureLists.length; i++) {
+                if (signatureLists[i].length === 0)
+                    return ts.emptyArray;
+                if (signatureLists[i].length > 1) {
+                    indexWithLengthOverOne = indexWithLengthOverOne === undefined ? i : -1; // -1 is a signal there are multiple overload sets
+                }
+                for (var _i = 0, _a = signatureLists[i]; _i < _a.length; _i++) {
+                    var signature = _a[_i];
+                    // Only process signatures with parameter lists that aren't already in the result list
+                    if (!result || !findMatchingSignature(result, signature, /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ true)) {
+                        var unionSignatures = findMatchingSignatures(signatureLists, signature, i);
+                        if (unionSignatures) {
+                            var s = signature;
+                            // Union the result types when more than one signature matches
+                            if (unionSignatures.length > 1) {
+                                var thisParameter = signature.thisParameter;
+                                var firstThisParameterOfUnionSignatures = ts.forEach(unionSignatures, function (sig) { return sig.thisParameter; });
+                                if (firstThisParameterOfUnionSignatures) {
+                                    var thisType = getIntersectionType(ts.mapDefined(unionSignatures, function (sig) { return sig.thisParameter && getTypeOfSymbol(sig.thisParameter); }));
+                                    thisParameter = createSymbolWithType(firstThisParameterOfUnionSignatures, thisType);
+                                }
+                                s = createUnionSignature(signature, unionSignatures);
+                                s.thisParameter = thisParameter;
+                            }
+                            (result || (result = [])).push(s);
+                        }
                     }
                 }
-                return !!ts.forEachChild(node, containsReference);
             }
-            return true;
-            function containsReference(node) {
-                switch (node.kind) {
-                    case 183 /* ThisType */:
-                        return !!tp.isThisType;
-                    case 75 /* Identifier */:
-                        return !tp.isThisType && ts.isPartOfTypeNode(node) && maybeTypeParameterReference(node) &&
-                            getTypeFromTypeNodeWorker(node) === tp; // use worker because we're looking for === equality
-                    case 172 /* TypeQuery */:
-                        return true;
+            if (!ts.length(result) && indexWithLengthOverOne !== -1) {
+                // No sufficiently similar signature existed to subsume all the other signatures in the union - time to see if we can make a single
+                // signature that handles all over them. We only do this when there are overloads in only one constituent.
+                // (Overloads are conditional in nature and having overloads in multiple constituents would necessitate making a power set of
+                // signatures from the type, whose ordering would be non-obvious)
+                var masterList = signatureLists[indexWithLengthOverOne !== undefined ? indexWithLengthOverOne : 0];
+                var results = masterList.slice();
+                var _loop_10 = function (signatures) {
+                    if (signatures !== masterList) {
+                        var signature_1 = signatures[0];
+                        ts.Debug.assert(!!signature_1, "getUnionSignatures bails early on empty signature lists and should not have empty lists on second pass");
+                        results = signature_1.typeParameters && ts.some(results, function (s) { return !!s.typeParameters; }) ? undefined : ts.map(results, function (sig) { return combineSignaturesOfUnionMembers(sig, signature_1); });
+                        if (!results) {
+                            return "break";
+                        }
+                    }
+                };
+                for (var _b = 0, signatureLists_1 = signatureLists; _b < signatureLists_1.length; _b++) {
+                    var signatures = signatureLists_1[_b];
+                    var state_3 = _loop_10(signatures);
+                    if (state_3 === "break")
+                        break;
                 }
-                return !!ts.forEachChild(node, containsReference);
+                result = results;
             }
+            return result || ts.emptyArray;
         }
-        function getHomomorphicTypeVariable(type) {
-            var constraintType = getConstraintTypeFromMappedType(type);
-            if (constraintType.flags & 4194304 /* Index */) {
-                var typeVariable = getActualTypeVariable(constraintType.type);
-                if (typeVariable.flags & 262144 /* TypeParameter */) {
-                    return typeVariable;
-                }
+        function combineUnionThisParam(left, right) {
+            if (!left || !right) {
+                return left || right;
             }
-            return undefined;
+            // A signature `this` type might be a read or a write position... It's very possible that it should be invariant
+            // and we should refuse to merge signatures if there are `this` types and they do not match. However, so as to be
+            // permissive when calling, for now, we'll intersect the `this` types just like we do for param types in union signatures.
+            var thisType = getIntersectionType([getTypeOfSymbol(left), getTypeOfSymbol(right)]);
+            return createSymbolWithType(left, thisType);
         }
-        function instantiateMappedType(type, mapper) {
-            // For a homomorphic mapped type { [P in keyof T]: X }, where T is some type variable, the mapping
-            // operation depends on T as follows:
-            // * If T is a primitive type no mapping is performed and the result is simply T.
-            // * If T is a union type we distribute the mapped type over the union.
-            // * If T is an array we map to an array where the element type has been transformed.
-            // * If T is a tuple we map to a tuple where the element types have been transformed.
-            // * Otherwise we map to an object type where the type of each property has been transformed.
-            // For example, when T is instantiated to a union type A | B, we produce { [P in keyof A]: X } |
-            // { [P in keyof B]: X }, and when when T is instantiated to a union type A | undefined, we produce
-            // { [P in keyof A]: X } | undefined.
-            var typeVariable = getHomomorphicTypeVariable(type);
-            if (typeVariable) {
-                var mappedTypeVariable = instantiateType(typeVariable, mapper);
-                if (typeVariable !== mappedTypeVariable) {
-                    return mapType(getReducedType(mappedTypeVariable), function (t) {
-                        if (t.flags & (3 /* AnyOrUnknown */ | 58982400 /* InstantiableNonPrimitive */ | 524288 /* Object */ | 2097152 /* Intersection */) && t !== wildcardType && t !== errorType) {
-                            var replacementMapper = prependTypeMapping(typeVariable, t, mapper);
-                            return isArrayType(t) ? instantiateMappedArrayType(t, type, replacementMapper) :
-                                isTupleType(t) ? instantiateMappedTupleType(t, type, replacementMapper) :
-                                    instantiateAnonymousType(type, replacementMapper);
-                        }
-                        return t;
-                    });
+        function combineUnionParameters(left, right) {
+            var leftCount = getParameterCount(left);
+            var rightCount = getParameterCount(right);
+            var longest = leftCount >= rightCount ? left : right;
+            var shorter = longest === left ? right : left;
+            var longestCount = longest === left ? leftCount : rightCount;
+            var eitherHasEffectiveRest = (hasEffectiveRestParameter(left) || hasEffectiveRestParameter(right));
+            var needsExtraRestElement = eitherHasEffectiveRest && !hasEffectiveRestParameter(longest);
+            var params = new Array(longestCount + (needsExtraRestElement ? 1 : 0));
+            for (var i = 0; i < longestCount; i++) {
+                var longestParamType = tryGetTypeAtPosition(longest, i);
+                var shorterParamType = tryGetTypeAtPosition(shorter, i) || unknownType;
+                var unionParamType = getIntersectionType([longestParamType, shorterParamType]);
+                var isRestParam = eitherHasEffectiveRest && !needsExtraRestElement && i === (longestCount - 1);
+                var isOptional = i >= getMinArgumentCount(longest) && i >= getMinArgumentCount(shorter);
+                var leftName = i >= leftCount ? undefined : getParameterNameAtPosition(left, i);
+                var rightName = i >= rightCount ? undefined : getParameterNameAtPosition(right, i);
+                var paramName = leftName === rightName ? leftName :
+                    !leftName ? rightName :
+                        !rightName ? leftName :
+                            undefined;
+                var paramSymbol = createSymbol(1 /* FunctionScopedVariable */ | (isOptional && !isRestParam ? 16777216 /* Optional */ : 0), paramName || "arg" + i);
+                paramSymbol.type = isRestParam ? createArrayType(unionParamType) : unionParamType;
+                params[i] = paramSymbol;
+            }
+            if (needsExtraRestElement) {
+                var restParamSymbol = createSymbol(1 /* FunctionScopedVariable */, "args");
+                restParamSymbol.type = createArrayType(getTypeAtPosition(shorter, longestCount));
+                params[longestCount] = restParamSymbol;
+            }
+            return params;
+        }
+        function combineSignaturesOfUnionMembers(left, right) {
+            var declaration = left.declaration;
+            var params = combineUnionParameters(left, right);
+            var thisParam = combineUnionThisParam(left.thisParameter, right.thisParameter);
+            var minArgCount = Math.max(left.minArgumentCount, right.minArgumentCount);
+            var result = createSignature(declaration, left.typeParameters || right.typeParameters, thisParam, params, 
+            /*resolvedReturnType*/ undefined, 
+            /*resolvedTypePredicate*/ undefined, minArgCount, (left.flags | right.flags) & 19 /* PropagatingFlags */);
+            result.unionSignatures = ts.concatenate(left.unionSignatures || [left], [right]);
+            return result;
+        }
+        function getUnionIndexInfo(types, kind) {
+            var indexTypes = [];
+            var isAnyReadonly = false;
+            for (var _i = 0, types_4 = types; _i < types_4.length; _i++) {
+                var type = types_4[_i];
+                var indexInfo = getIndexInfoOfType(getApparentType(type), kind);
+                if (!indexInfo) {
+                    return undefined;
                 }
+                indexTypes.push(indexInfo.type);
+                isAnyReadonly = isAnyReadonly || indexInfo.isReadonly;
             }
-            return instantiateAnonymousType(type, mapper);
+            return createIndexInfo(getUnionType(indexTypes, 2 /* Subtype */), isAnyReadonly);
         }
-        function getModifiedReadonlyState(state, modifiers) {
-            return modifiers & 1 /* IncludeReadonly */ ? true : modifiers & 2 /* ExcludeReadonly */ ? false : state;
+        function resolveUnionTypeMembers(type) {
+            // The members and properties collections are empty for union types. To get all properties of a union
+            // type use getPropertiesOfType (only the language service uses this).
+            var callSignatures = getUnionSignatures(ts.map(type.types, function (t) { return t === globalFunctionType ? [unknownSignature] : getSignaturesOfType(t, 0 /* Call */); }));
+            var constructSignatures = getUnionSignatures(ts.map(type.types, function (t) { return getSignaturesOfType(t, 1 /* Construct */); }));
+            var stringIndexInfo = getUnionIndexInfo(type.types, 0 /* String */);
+            var numberIndexInfo = getUnionIndexInfo(type.types, 1 /* Number */);
+            setStructuredTypeMembers(type, emptySymbols, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
         }
-        function instantiateMappedArrayType(arrayType, mappedType, mapper) {
-            var elementType = instantiateMappedTypeTemplate(mappedType, numberType, /*isOptional*/ true, mapper);
-            return elementType === errorType ? errorType :
-                createArrayType(elementType, getModifiedReadonlyState(isReadonlyArrayType(arrayType), getMappedTypeModifiers(mappedType)));
+        function intersectTypes(type1, type2) {
+            return !type1 ? type2 : !type2 ? type1 : getIntersectionType([type1, type2]);
         }
-        function instantiateMappedTupleType(tupleType, mappedType, mapper) {
-            var minLength = tupleType.target.minLength;
-            var elementTypes = ts.map(getTypeArguments(tupleType), function (_, i) {
-                return instantiateMappedTypeTemplate(mappedType, getLiteralType("" + i), i >= minLength, mapper);
-            });
-            var modifiers = getMappedTypeModifiers(mappedType);
-            var newMinLength = modifiers & 4 /* IncludeOptional */ ? 0 :
-                modifiers & 8 /* ExcludeOptional */ ? getTypeReferenceArity(tupleType) - (tupleType.target.hasRestElement ? 1 : 0) :
-                    minLength;
-            var newReadonly = getModifiedReadonlyState(tupleType.target.readonly, modifiers);
-            return ts.contains(elementTypes, errorType) ? errorType :
-                createTupleType(elementTypes, newMinLength, tupleType.target.hasRestElement, newReadonly, tupleType.target.associatedNames);
+        function intersectIndexInfos(info1, info2) {
+            return !info1 ? info2 : !info2 ? info1 : createIndexInfo(getIntersectionType([info1.type, info2.type]), info1.isReadonly && info2.isReadonly);
         }
-        function instantiateMappedTypeTemplate(type, key, isOptional, mapper) {
-            var templateMapper = appendTypeMapping(mapper, getTypeParameterFromMappedType(type), key);
-            var propType = instantiateType(getTemplateTypeFromMappedType(type.target || type), templateMapper);
-            var modifiers = getMappedTypeModifiers(type);
-            return strictNullChecks && modifiers & 4 /* IncludeOptional */ && !maybeTypeOfKind(propType, 32768 /* Undefined */ | 16384 /* Void */) ? getOptionalType(propType) :
-                strictNullChecks && modifiers & 8 /* ExcludeOptional */ && isOptional ? getTypeWithFacts(propType, 524288 /* NEUndefined */) :
-                    propType;
+        function unionSpreadIndexInfos(info1, info2) {
+            return info1 && info2 && createIndexInfo(getUnionType([info1.type, info2.type]), info1.isReadonly || info2.isReadonly);
         }
-        function instantiateAnonymousType(type, mapper) {
-            var result = createObjectType(type.objectFlags | 64 /* Instantiated */, type.symbol);
-            if (type.objectFlags & 32 /* Mapped */) {
-                result.declaration = type.declaration;
-                // C.f. instantiateSignature
-                var origTypeParameter = getTypeParameterFromMappedType(type);
-                var freshTypeParameter = cloneTypeParameter(origTypeParameter);
-                result.typeParameter = freshTypeParameter;
-                mapper = combineTypeMappers(makeUnaryTypeMapper(origTypeParameter, freshTypeParameter), mapper);
-                freshTypeParameter.mapper = mapper;
+        function findMixins(types) {
+            var constructorTypeCount = ts.countWhere(types, function (t) { return getSignaturesOfType(t, 1 /* Construct */).length > 0; });
+            var mixinFlags = ts.map(types, isMixinConstructorType);
+            if (constructorTypeCount > 0 && constructorTypeCount === ts.countWhere(mixinFlags, function (b) { return b; })) {
+                var firstMixinIndex = mixinFlags.indexOf(/*searchElement*/ true);
+                mixinFlags[firstMixinIndex] = false;
             }
-            result.target = type;
-            result.mapper = mapper;
-            result.aliasSymbol = type.aliasSymbol;
-            result.aliasTypeArguments = instantiateTypes(type.aliasTypeArguments, mapper);
-            return result;
+            return mixinFlags;
         }
-        function getConditionalTypeInstantiation(type, mapper) {
-            var root = type.root;
-            if (root.outerTypeParameters) {
-                // We are instantiating a conditional type that has one or more type parameters in scope. Apply the
-                // mapper to the type parameters to produce the effective list of type arguments, and compute the
-                // instantiation cache key from the type IDs of the type arguments.
-                var typeArguments = ts.map(root.outerTypeParameters, function (t) { return getMappedType(t, mapper); });
-                var id = getTypeListId(typeArguments);
-                var result = root.instantiations.get(id);
-                if (!result) {
-                    var newMapper = createTypeMapper(root.outerTypeParameters, typeArguments);
-                    result = instantiateConditionalType(root, newMapper);
-                    root.instantiations.set(id, result);
+        function includeMixinType(type, types, mixinFlags, index) {
+            var mixedTypes = [];
+            for (var i = 0; i < types.length; i++) {
+                if (i === index) {
+                    mixedTypes.push(type);
+                }
+                else if (mixinFlags[i]) {
+                    mixedTypes.push(getReturnTypeOfSignature(getSignaturesOfType(types[i], 1 /* Construct */)[0]));
                 }
-                return result;
             }
-            return type;
+            return getIntersectionType(mixedTypes);
         }
-        function instantiateConditionalType(root, mapper) {
-            // Check if we have a conditional type where the check type is a naked type parameter. If so,
-            // the conditional type is distributive over union types and when T is instantiated to a union
-            // type A | B, we produce (A extends U ? X : Y) | (B extends U ? X : Y).
-            if (root.isDistributive) {
-                var checkType_1 = root.checkType;
-                var instantiatedType = getMappedType(checkType_1, mapper);
-                if (checkType_1 !== instantiatedType && instantiatedType.flags & (1048576 /* Union */ | 131072 /* Never */)) {
-                    return mapType(instantiatedType, function (t) { return getConditionalType(root, prependTypeMapping(checkType_1, t, mapper)); });
+        function resolveIntersectionTypeMembers(type) {
+            // The members and properties collections are empty for intersection types. To get all properties of an
+            // intersection type use getPropertiesOfType (only the language service uses this).
+            var callSignatures;
+            var constructSignatures;
+            var stringIndexInfo;
+            var numberIndexInfo;
+            var types = type.types;
+            var mixinFlags = findMixins(types);
+            var mixinCount = ts.countWhere(mixinFlags, function (b) { return b; });
+            var _loop_11 = function (i) {
+                var t = type.types[i];
+                // When an intersection type contains mixin constructor types, the construct signatures from
+                // those types are discarded and their return types are mixed into the return types of all
+                // other construct signatures in the intersection type. For example, the intersection type
+                // '{ new(...args: any[]) => A } & { new(s: string) => B }' has a single construct signature
+                // 'new(s: string) => A & B'.
+                if (!mixinFlags[i]) {
+                    var signatures = getSignaturesOfType(t, 1 /* Construct */);
+                    if (signatures.length && mixinCount > 0) {
+                        signatures = ts.map(signatures, function (s) {
+                            var clone = cloneSignature(s);
+                            clone.resolvedReturnType = includeMixinType(getReturnTypeOfSignature(s), types, mixinFlags, i);
+                            return clone;
+                        });
+                    }
+                    constructSignatures = appendSignatures(constructSignatures, signatures);
                 }
+                callSignatures = appendSignatures(callSignatures, getSignaturesOfType(t, 0 /* Call */));
+                stringIndexInfo = intersectIndexInfos(stringIndexInfo, getIndexInfoOfType(t, 0 /* String */));
+                numberIndexInfo = intersectIndexInfos(numberIndexInfo, getIndexInfoOfType(t, 1 /* Number */));
+            };
+            for (var i = 0; i < types.length; i++) {
+                _loop_11(i);
             }
-            return getConditionalType(root, mapper);
+            setStructuredTypeMembers(type, emptySymbols, callSignatures || ts.emptyArray, constructSignatures || ts.emptyArray, stringIndexInfo, numberIndexInfo);
         }
-        function instantiateType(type, mapper) {
-            if (!type || !mapper) {
-                return type;
-            }
-            if (instantiationDepth === 50 || instantiationCount >= 5000000) {
-                // We have reached 50 recursive type instantiations and there is a very high likelyhood we're dealing
-                // with a combination of infinite generic types that perpetually generate new type identities. We stop
-                // the recursion here by yielding the error type.
-                error(currentNode, ts.Diagnostics.Type_instantiation_is_excessively_deep_and_possibly_infinite);
-                return errorType;
+        function appendSignatures(signatures, newSignatures) {
+            var _loop_12 = function (sig) {
+                if (!signatures || ts.every(signatures, function (s) { return !compareSignaturesIdentical(s, sig, /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ false, compareTypesIdentical); })) {
+                    signatures = ts.append(signatures, sig);
+                }
+            };
+            for (var _i = 0, newSignatures_1 = newSignatures; _i < newSignatures_1.length; _i++) {
+                var sig = newSignatures_1[_i];
+                _loop_12(sig);
             }
-            totalInstantiationCount++;
-            instantiationCount++;
-            instantiationDepth++;
-            var result = instantiateTypeWorker(type, mapper);
-            instantiationDepth--;
-            return result;
+            return signatures;
         }
         /**
-         * This can be used to avoid the penalty on instantiation depth for types which result from immediate
-         * simplification. It essentially removes the depth increase done in `instantiateType`.
+         * Converts an AnonymousType to a ResolvedType.
          */
-        function instantiateTypeWithoutDepthIncrease(type, mapper) {
-            instantiationDepth--;
-            var result = instantiateType(type, mapper);
-            instantiationDepth++;
-            return result;
-        }
-        function instantiateTypeWorker(type, mapper) {
-            var flags = type.flags;
-            if (flags & 262144 /* TypeParameter */) {
-                return getMappedType(type, mapper);
+        function resolveAnonymousTypeMembers(type) {
+            var symbol = getMergedSymbol(type.symbol);
+            if (type.target) {
+                setStructuredTypeMembers(type, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+                var members = createInstantiatedSymbolTable(getPropertiesOfObjectType(type.target), type.mapper, /*mappingThisOnly*/ false);
+                var callSignatures = instantiateSignatures(getSignaturesOfType(type.target, 0 /* Call */), type.mapper);
+                var constructSignatures = instantiateSignatures(getSignaturesOfType(type.target, 1 /* Construct */), type.mapper);
+                var stringIndexInfo = instantiateIndexInfo(getIndexInfoOfType(type.target, 0 /* String */), type.mapper);
+                var numberIndexInfo = instantiateIndexInfo(getIndexInfoOfType(type.target, 1 /* Number */), type.mapper);
+                setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
             }
-            if (flags & 524288 /* Object */) {
-                var objectFlags = type.objectFlags;
-                if (objectFlags & 16 /* Anonymous */) {
-                    // If the anonymous type originates in a declaration of a function, method, class, or
-                    // interface, in an object type literal, or in an object literal expression, we may need
-                    // to instantiate the type because it might reference a type parameter.
-                    return couldContainTypeVariables(type) ?
-                        getObjectTypeInstantiation(type, mapper) : type;
-                }
-                if (objectFlags & 32 /* Mapped */) {
-                    return getObjectTypeInstantiation(type, mapper);
-                }
-                if (objectFlags & 4 /* Reference */) {
-                    if (type.node) {
-                        return getObjectTypeInstantiation(type, mapper);
+            else if (symbol.flags & 2048 /* TypeLiteral */) {
+                setStructuredTypeMembers(type, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+                var members = getMembersOfSymbol(symbol);
+                var callSignatures = getSignaturesOfSymbol(members.get("__call" /* Call */));
+                var constructSignatures = getSignaturesOfSymbol(members.get("__new" /* New */));
+                var stringIndexInfo = getIndexInfoOfSymbol(symbol, 0 /* String */);
+                var numberIndexInfo = getIndexInfoOfSymbol(symbol, 1 /* Number */);
+                setStructuredTypeMembers(type, members, callSignatures, constructSignatures, stringIndexInfo, numberIndexInfo);
+            }
+            else {
+                // Combinations of function, class, enum and module
+                var members = emptySymbols;
+                var stringIndexInfo = void 0;
+                if (symbol.exports) {
+                    members = getExportsOfSymbol(symbol);
+                    if (symbol === globalThisSymbol) {
+                        var varsOnly_1 = new ts.Map();
+                        members.forEach(function (p) {
+                            if (!(p.flags & 418 /* BlockScoped */)) {
+                                varsOnly_1.set(p.escapedName, p);
+                            }
+                        });
+                        members = varsOnly_1;
                     }
-                    var resolvedTypeArguments = type.resolvedTypeArguments;
-                    var newTypeArguments = instantiateTypes(resolvedTypeArguments, mapper);
-                    return newTypeArguments !== resolvedTypeArguments ? createTypeReference(type.target, newTypeArguments) : type;
                 }
-                return type;
-            }
-            if ((flags & 2097152 /* Intersection */) || (flags & 1048576 /* Union */ && !(flags & 131068 /* Primitive */))) {
-                if (!couldContainTypeVariables(type)) {
-                    return type;
+                setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, undefined, undefined);
+                if (symbol.flags & 32 /* Class */) {
+                    var classType = getDeclaredTypeOfClassOrInterface(symbol);
+                    var baseConstructorType = getBaseConstructorTypeOfClass(classType);
+                    if (baseConstructorType.flags & (524288 /* Object */ | 2097152 /* Intersection */ | 8650752 /* TypeVariable */)) {
+                        members = ts.createSymbolTable(getNamedMembers(members));
+                        addInheritedMembers(members, getPropertiesOfType(baseConstructorType));
+                    }
+                    else if (baseConstructorType === anyType) {
+                        stringIndexInfo = createIndexInfo(anyType, /*isReadonly*/ false);
+                    }
                 }
-                var types = type.types;
-                var newTypes = instantiateTypes(types, mapper);
-                return newTypes === types
-                    ? type
-                    : (flags & 2097152 /* Intersection */)
-                        ? getIntersectionType(newTypes, type.aliasSymbol, instantiateTypes(type.aliasTypeArguments, mapper))
-                        : getUnionType(newTypes, 1 /* Literal */, type.aliasSymbol, instantiateTypes(type.aliasTypeArguments, mapper));
-            }
-            if (flags & 4194304 /* Index */) {
-                return getIndexType(instantiateType(type.type, mapper));
-            }
-            if (flags & 8388608 /* IndexedAccess */) {
-                return getIndexedAccessType(instantiateType(type.objectType, mapper), instantiateType(type.indexType, mapper));
-            }
-            if (flags & 16777216 /* Conditional */) {
-                return getConditionalTypeInstantiation(type, combineTypeMappers(type.mapper, mapper));
-            }
-            if (flags & 33554432 /* Substitution */) {
-                var maybeVariable = instantiateType(type.baseType, mapper);
-                if (maybeVariable.flags & 8650752 /* TypeVariable */) {
-                    return getSubstitutionType(maybeVariable, instantiateType(type.substitute, mapper));
+                var numberIndexInfo = symbol.flags & 384 /* Enum */ && (getDeclaredTypeOfSymbol(symbol).flags & 32 /* Enum */ ||
+                    ts.some(type.properties, function (prop) { return !!(getTypeOfSymbol(prop).flags & 296 /* NumberLike */); })) ? enumNumberIndexInfo : undefined;
+                setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
+                // We resolve the members before computing the signatures because a signature may use
+                // typeof with a qualified name expression that circularly references the type we are
+                // in the process of resolving (see issue #6072). The temporarily empty signature list
+                // will never be observed because a qualified name can't reference signatures.
+                if (symbol.flags & (16 /* Function */ | 8192 /* Method */)) {
+                    type.callSignatures = getSignaturesOfSymbol(symbol);
                 }
-                else {
-                    var sub = instantiateType(type.substitute, mapper);
-                    if (sub.flags & 3 /* AnyOrUnknown */ || isTypeAssignableTo(getRestrictiveInstantiation(maybeVariable), getRestrictiveInstantiation(sub))) {
-                        return maybeVariable;
+                // And likewise for construct signatures for classes
+                if (symbol.flags & 32 /* Class */) {
+                    var classType_1 = getDeclaredTypeOfClassOrInterface(symbol);
+                    var constructSignatures = symbol.members ? getSignaturesOfSymbol(symbol.members.get("__constructor" /* Constructor */)) : ts.emptyArray;
+                    if (symbol.flags & 16 /* Function */) {
+                        constructSignatures = ts.addRange(constructSignatures.slice(), ts.mapDefined(type.callSignatures, function (sig) { return isJSConstructor(sig.declaration) ?
+                            createSignature(sig.declaration, sig.typeParameters, sig.thisParameter, sig.parameters, classType_1, /*resolvedTypePredicate*/ undefined, sig.minArgumentCount, sig.flags & 19 /* PropagatingFlags */) :
+                            undefined; }));
                     }
-                    return sub;
+                    if (!constructSignatures.length) {
+                        constructSignatures = getDefaultConstructSignatures(classType_1);
+                    }
+                    type.constructSignatures = constructSignatures;
                 }
             }
-            return type;
         }
-        function getPermissiveInstantiation(type) {
-            return type.flags & (131068 /* Primitive */ | 3 /* AnyOrUnknown */ | 131072 /* Never */) ? type :
-                type.permissiveInstantiation || (type.permissiveInstantiation = instantiateType(type, permissiveMapper));
+        function resolveReverseMappedTypeMembers(type) {
+            var indexInfo = getIndexInfoOfType(type.source, 0 /* String */);
+            var modifiers = getMappedTypeModifiers(type.mappedType);
+            var readonlyMask = modifiers & 1 /* IncludeReadonly */ ? false : true;
+            var optionalMask = modifiers & 4 /* IncludeOptional */ ? 0 : 16777216 /* Optional */;
+            var stringIndexInfo = indexInfo && createIndexInfo(inferReverseMappedType(indexInfo.type, type.mappedType, type.constraintType), readonlyMask && indexInfo.isReadonly);
+            var members = ts.createSymbolTable();
+            for (var _i = 0, _a = getPropertiesOfType(type.source); _i < _a.length; _i++) {
+                var prop = _a[_i];
+                var checkFlags = 8192 /* ReverseMapped */ | (readonlyMask && isReadonlySymbol(prop) ? 8 /* Readonly */ : 0);
+                var inferredProp = createSymbol(4 /* Property */ | prop.flags & optionalMask, prop.escapedName, checkFlags);
+                inferredProp.declarations = prop.declarations;
+                inferredProp.nameType = getSymbolLinks(prop).nameType;
+                inferredProp.propertyType = getTypeOfSymbol(prop);
+                inferredProp.mappedType = type.mappedType;
+                inferredProp.constraintType = type.constraintType;
+                members.set(prop.escapedName, inferredProp);
+            }
+            setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, undefined);
         }
-        function getRestrictiveInstantiation(type) {
-            if (type.flags & (131068 /* Primitive */ | 3 /* AnyOrUnknown */ | 131072 /* Never */)) {
+        // Return the lower bound of the key type in a mapped type. Intuitively, the lower
+        // bound includes those keys that are known to always be present, for example because
+        // because of constraints on type parameters (e.g. 'keyof T' for a constrained T).
+        function getLowerBoundOfKeyType(type) {
+            if (type.flags & 4194304 /* Index */) {
+                var t = getApparentType(type.type);
+                return isGenericTupleType(t) ? getKnownKeysOfTupleType(t) : getIndexType(t);
+            }
+            if (type.flags & 16777216 /* Conditional */) {
+                if (type.root.isDistributive) {
+                    var checkType = type.checkType;
+                    var constraint = getLowerBoundOfKeyType(checkType);
+                    if (constraint !== checkType) {
+                        return getConditionalTypeInstantiation(type, prependTypeMapping(type.root.checkType, constraint, type.mapper));
+                    }
+                }
                 return type;
             }
-            if (type.restrictiveInstantiation) {
-                return type.restrictiveInstantiation;
+            if (type.flags & 1048576 /* Union */) {
+                return getUnionType(ts.sameMap(type.types, getLowerBoundOfKeyType));
             }
-            type.restrictiveInstantiation = instantiateType(type, restrictiveMapper);
-            // We set the following so we don't attempt to set the restrictive instance of a restrictive instance
-            // which is redundant - we'll produce new type identities, but all type params have already been mapped.
-            // This also gives us a way to detect restrictive instances upon comparisons and _disable_ the "distributeive constraint"
-            // assignability check for them, which is distinctly unsafe, as once you have a restrctive instance, all the type parameters
-            // are constrained to `unknown` and produce tons of false positives/negatives!
-            type.restrictiveInstantiation.restrictiveInstantiation = type.restrictiveInstantiation;
-            return type.restrictiveInstantiation;
-        }
-        function instantiateIndexInfo(info, mapper) {
-            return info && createIndexInfo(instantiateType(info.type, mapper), info.isReadonly, info.declaration);
+            if (type.flags & 2097152 /* Intersection */) {
+                return getIntersectionType(ts.sameMap(type.types, getLowerBoundOfKeyType));
+            }
+            return type;
         }
-        // Returns true if the given expression contains (at any level of nesting) a function or arrow expression
-        // that is subject to contextual typing.
-        function isContextSensitive(node) {
-            ts.Debug.assert(node.kind !== 161 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
-            switch (node.kind) {
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 161 /* MethodDeclaration */:
-                case 244 /* FunctionDeclaration */: // Function declarations can have context when annotated with a jsdoc @type
-                    return isContextSensitiveFunctionLikeDeclaration(node);
-                case 193 /* ObjectLiteralExpression */:
-                    return ts.some(node.properties, isContextSensitive);
-                case 192 /* ArrayLiteralExpression */:
-                    return ts.some(node.elements, isContextSensitive);
-                case 210 /* ConditionalExpression */:
-                    return isContextSensitive(node.whenTrue) ||
-                        isContextSensitive(node.whenFalse);
-                case 209 /* BinaryExpression */:
-                    return (node.operatorToken.kind === 56 /* BarBarToken */ || node.operatorToken.kind === 60 /* QuestionQuestionToken */) &&
-                        (isContextSensitive(node.left) || isContextSensitive(node.right));
-                case 281 /* PropertyAssignment */:
-                    return isContextSensitive(node.initializer);
-                case 200 /* ParenthesizedExpression */:
-                    return isContextSensitive(node.expression);
-                case 274 /* JsxAttributes */:
-                    return ts.some(node.properties, isContextSensitive) || ts.isJsxOpeningElement(node.parent) && ts.some(node.parent.parent.children, isContextSensitive);
-                case 273 /* JsxAttribute */: {
-                    // If there is no initializer, JSX attribute has a boolean value of true which is not context sensitive.
-                    var initializer = node.initializer;
-                    return !!initializer && isContextSensitive(initializer);
+        /** Resolve the members of a mapped type { [P in K]: T } */
+        function resolveMappedTypeMembers(type) {
+            var members = ts.createSymbolTable();
+            var stringIndexInfo;
+            var numberIndexInfo;
+            // Resolve upfront such that recursive references see an empty object type.
+            setStructuredTypeMembers(type, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+            // In { [P in K]: T }, we refer to P as the type parameter type, K as the constraint type,
+            // and T as the template type.
+            var typeParameter = getTypeParameterFromMappedType(type);
+            var constraintType = getConstraintTypeFromMappedType(type);
+            var nameType = getNameTypeFromMappedType(type.target || type);
+            var templateType = getTemplateTypeFromMappedType(type.target || type);
+            var modifiersType = getApparentType(getModifiersTypeFromMappedType(type)); // The 'T' in 'keyof T'
+            var templateModifiers = getMappedTypeModifiers(type);
+            var include = keyofStringsOnly ? 128 /* StringLiteral */ : 8576 /* StringOrNumberLiteralOrUnique */;
+            if (isMappedTypeWithKeyofConstraintDeclaration(type)) {
+                // We have a { [P in keyof T]: X }
+                for (var _i = 0, _a = getPropertiesOfType(modifiersType); _i < _a.length; _i++) {
+                    var prop = _a[_i];
+                    addMemberForKeyType(getLiteralTypeFromProperty(prop, include));
                 }
-                case 276 /* JsxExpression */: {
-                    // It is possible to that node.expression is undefined (e.g <div x={} />)
-                    var expression = node.expression;
-                    return !!expression && isContextSensitive(expression);
+                if (modifiersType.flags & 1 /* Any */ || getIndexInfoOfType(modifiersType, 0 /* String */)) {
+                    addMemberForKeyType(stringType);
                 }
-            }
-            return false;
-        }
-        function isContextSensitiveFunctionLikeDeclaration(node) {
-            return (!ts.isFunctionDeclaration(node) || ts.isInJSFile(node) && !!getTypeForDeclarationFromJSDocComment(node)) &&
-                (hasContextSensitiveParameters(node) || hasContextSensitiveReturnExpression(node));
-        }
-        function hasContextSensitiveParameters(node) {
-            // Functions with type parameters are not context sensitive.
-            if (!node.typeParameters) {
-                // Functions with any parameters that lack type annotations are context sensitive.
-                if (ts.some(node.parameters, function (p) { return !ts.getEffectiveTypeAnnotationNode(p); })) {
-                    return true;
+                if (!keyofStringsOnly && getIndexInfoOfType(modifiersType, 1 /* Number */)) {
+                    addMemberForKeyType(numberType);
                 }
-                if (node.kind !== 202 /* ArrowFunction */) {
-                    // If the first parameter is not an explicit 'this' parameter, then the function has
-                    // an implicit 'this' parameter which is subject to contextual typing.
-                    var parameter = ts.firstOrUndefined(node.parameters);
-                    if (!(parameter && ts.parameterIsThisKeyword(parameter))) {
-                        return true;
+            }
+            else {
+                forEachType(getLowerBoundOfKeyType(constraintType), addMemberForKeyType);
+            }
+            setStructuredTypeMembers(type, members, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
+            function addMemberForKeyType(keyType) {
+                var propNameType = nameType ? instantiateType(nameType, appendTypeMapping(type.mapper, typeParameter, keyType)) : keyType;
+                forEachType(propNameType, function (t) { return addMemberForKeyTypeWorker(keyType, t); });
+            }
+            function addMemberForKeyTypeWorker(keyType, propNameType) {
+                // If the current iteration type constituent is a string literal type, create a property.
+                // Otherwise, for type string create a string index signature.
+                if (isTypeUsableAsPropertyName(propNameType)) {
+                    var propName = getPropertyNameFromType(propNameType);
+                    // String enum members from separate enums with identical values
+                    // are distinct types with the same property name. Make the resulting
+                    // property symbol's name type be the union of those enum member types.
+                    var existingProp = members.get(propName);
+                    if (existingProp) {
+                        existingProp.nameType = getUnionType([existingProp.nameType, propNameType]);
+                        existingProp.keyType = getUnionType([existingProp.keyType, keyType]);
+                    }
+                    else {
+                        var modifiersProp = isTypeUsableAsPropertyName(keyType) ? getPropertyOfType(modifiersType, getPropertyNameFromType(keyType)) : undefined;
+                        var isOptional = !!(templateModifiers & 4 /* IncludeOptional */ ||
+                            !(templateModifiers & 8 /* ExcludeOptional */) && modifiersProp && modifiersProp.flags & 16777216 /* Optional */);
+                        var isReadonly = !!(templateModifiers & 1 /* IncludeReadonly */ ||
+                            !(templateModifiers & 2 /* ExcludeReadonly */) && modifiersProp && isReadonlySymbol(modifiersProp));
+                        var stripOptional = strictNullChecks && !isOptional && modifiersProp && modifiersProp.flags & 16777216 /* Optional */;
+                        var prop = createSymbol(4 /* Property */ | (isOptional ? 16777216 /* Optional */ : 0), propName, 262144 /* Mapped */ | (isReadonly ? 8 /* Readonly */ : 0) | (stripOptional ? 524288 /* StripOptional */ : 0));
+                        prop.mappedType = type;
+                        prop.nameType = propNameType;
+                        prop.keyType = keyType;
+                        if (modifiersProp) {
+                            prop.syntheticOrigin = modifiersProp;
+                            prop.declarations = modifiersProp.declarations;
+                        }
+                        members.set(propName, prop);
+                    }
+                }
+                else if (propNameType.flags & (1 /* Any */ | 4 /* String */ | 8 /* Number */ | 32 /* Enum */)) {
+                    var propType = instantiateType(templateType, appendTypeMapping(type.mapper, typeParameter, keyType));
+                    if (propNameType.flags & (1 /* Any */ | 4 /* String */)) {
+                        stringIndexInfo = createIndexInfo(stringIndexInfo ? getUnionType([stringIndexInfo.type, propType]) : propType, !!(templateModifiers & 1 /* IncludeReadonly */));
+                    }
+                    else {
+                        numberIndexInfo = createIndexInfo(numberIndexInfo ? getUnionType([numberIndexInfo.type, propType]) : propType, !!(templateModifiers & 1 /* IncludeReadonly */));
                     }
                 }
             }
-            return false;
-        }
-        function hasContextSensitiveReturnExpression(node) {
-            // TODO(anhans): A block should be context-sensitive if it has a context-sensitive return value.
-            return !node.typeParameters && !ts.getEffectiveReturnTypeNode(node) && !!node.body && node.body.kind !== 223 /* Block */ && isContextSensitive(node.body);
-        }
-        function isContextSensitiveFunctionOrObjectLiteralMethod(func) {
-            return (ts.isInJSFile(func) && ts.isFunctionDeclaration(func) || isFunctionExpressionOrArrowFunction(func) || ts.isObjectLiteralMethod(func)) &&
-                isContextSensitiveFunctionLikeDeclaration(func);
         }
-        function getTypeWithoutSignatures(type) {
-            if (type.flags & 524288 /* Object */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                if (resolved.constructSignatures.length || resolved.callSignatures.length) {
-                    var result = createObjectType(16 /* Anonymous */, type.symbol);
-                    result.members = resolved.members;
-                    result.properties = resolved.properties;
-                    result.callSignatures = ts.emptyArray;
-                    result.constructSignatures = ts.emptyArray;
-                    return result;
+        function getTypeOfMappedSymbol(symbol) {
+            if (!symbol.type) {
+                var mappedType = symbol.mappedType;
+                if (!pushTypeResolution(symbol, 0 /* Type */)) {
+                    mappedType.containsError = true;
+                    return errorType;
                 }
+                var templateType = getTemplateTypeFromMappedType(mappedType.target || mappedType);
+                var mapper = appendTypeMapping(mappedType.mapper, getTypeParameterFromMappedType(mappedType), symbol.keyType);
+                var propType = instantiateType(templateType, mapper);
+                // When creating an optional property in strictNullChecks mode, if 'undefined' isn't assignable to the
+                // type, we include 'undefined' in the type. Similarly, when creating a non-optional property in strictNullChecks
+                // mode, if the underlying property is optional we remove 'undefined' from the type.
+                var type = strictNullChecks && symbol.flags & 16777216 /* Optional */ && !maybeTypeOfKind(propType, 32768 /* Undefined */ | 16384 /* Void */) ? getOptionalType(propType) :
+                    symbol.checkFlags & 524288 /* StripOptional */ ? getTypeWithFacts(propType, 524288 /* NEUndefined */) :
+                        propType;
+                if (!popTypeResolution()) {
+                    error(currentNode, ts.Diagnostics.Type_of_property_0_circularly_references_itself_in_mapped_type_1, symbolToString(symbol), typeToString(mappedType));
+                    type = errorType;
+                }
+                symbol.type = type;
             }
-            else if (type.flags & 2097152 /* Intersection */) {
-                return getIntersectionType(ts.map(type.types, getTypeWithoutSignatures));
-            }
-            return type;
-        }
-        // TYPE CHECKING
-        function isTypeIdenticalTo(source, target) {
-            return isTypeRelatedTo(source, target, identityRelation);
+            return symbol.type;
         }
-        function compareTypesIdentical(source, target) {
-            return isTypeRelatedTo(source, target, identityRelation) ? -1 /* True */ : 0 /* False */;
+        function getTypeParameterFromMappedType(type) {
+            return type.typeParameter ||
+                (type.typeParameter = getDeclaredTypeOfTypeParameter(getSymbolOfNode(type.declaration.typeParameter)));
         }
-        function compareTypesAssignable(source, target) {
-            return isTypeRelatedTo(source, target, assignableRelation) ? -1 /* True */ : 0 /* False */;
+        function getConstraintTypeFromMappedType(type) {
+            return type.constraintType ||
+                (type.constraintType = getConstraintOfTypeParameter(getTypeParameterFromMappedType(type)) || errorType);
         }
-        function compareTypesSubtypeOf(source, target) {
-            return isTypeRelatedTo(source, target, subtypeRelation) ? -1 /* True */ : 0 /* False */;
+        function getNameTypeFromMappedType(type) {
+            return type.declaration.nameType ?
+                type.nameType || (type.nameType = instantiateType(getTypeFromTypeNode(type.declaration.nameType), type.mapper)) :
+                undefined;
         }
-        function isTypeSubtypeOf(source, target) {
-            return isTypeRelatedTo(source, target, subtypeRelation);
+        function getTemplateTypeFromMappedType(type) {
+            return type.templateType ||
+                (type.templateType = type.declaration.type ?
+                    instantiateType(addOptionality(getTypeFromTypeNode(type.declaration.type), !!(getMappedTypeModifiers(type) & 4 /* IncludeOptional */)), type.mapper) :
+                    errorType);
         }
-        function isTypeAssignableTo(source, target) {
-            return isTypeRelatedTo(source, target, assignableRelation);
+        function getConstraintDeclarationForMappedType(type) {
+            return ts.getEffectiveConstraintOfTypeParameter(type.declaration.typeParameter);
         }
-        // An object type S is considered to be derived from an object type T if
-        // S is a union type and every constituent of S is derived from T,
-        // T is a union type and S is derived from at least one constituent of T, or
-        // S is a type variable with a base constraint that is derived from T,
-        // T is one of the global types Object and Function and S is a subtype of T, or
-        // T occurs directly or indirectly in an 'extends' clause of S.
-        // Note that this check ignores type parameters and only considers the
-        // inheritance hierarchy.
-        function isTypeDerivedFrom(source, target) {
-            return source.flags & 1048576 /* Union */ ? ts.every(source.types, function (t) { return isTypeDerivedFrom(t, target); }) :
-                target.flags & 1048576 /* Union */ ? ts.some(target.types, function (t) { return isTypeDerivedFrom(source, t); }) :
-                    source.flags & 58982400 /* InstantiableNonPrimitive */ ? isTypeDerivedFrom(getBaseConstraintOfType(source) || unknownType, target) :
-                        target === globalObjectType ? !!(source.flags & (524288 /* Object */ | 67108864 /* NonPrimitive */)) :
-                            target === globalFunctionType ? !!(source.flags & 524288 /* Object */) && isFunctionObjectType(source) :
-                                hasBaseType(source, getTargetType(target));
+        function isMappedTypeWithKeyofConstraintDeclaration(type) {
+            var constraintDeclaration = getConstraintDeclarationForMappedType(type); // TODO: GH#18217
+            return constraintDeclaration.kind === 188 /* TypeOperator */ &&
+                constraintDeclaration.operator === 138 /* KeyOfKeyword */;
         }
-        /**
-         * This is *not* a bi-directional relationship.
-         * If one needs to check both directions for comparability, use a second call to this function or 'checkTypeComparableTo'.
-         *
-         * A type S is comparable to a type T if some (but not necessarily all) of the possible values of S are also possible values of T.
-         * It is used to check following cases:
-         *   - the types of the left and right sides of equality/inequality operators (`===`, `!==`, `==`, `!=`).
-         *   - the types of `case` clause expressions and their respective `switch` expressions.
-         *   - the type of an expression in a type assertion with the type being asserted.
-         */
-        function isTypeComparableTo(source, target) {
-            return isTypeRelatedTo(source, target, comparableRelation);
+        function getModifiersTypeFromMappedType(type) {
+            if (!type.modifiersType) {
+                if (isMappedTypeWithKeyofConstraintDeclaration(type)) {
+                    // If the constraint declaration is a 'keyof T' node, the modifiers type is T. We check
+                    // AST nodes here because, when T is a non-generic type, the logic below eagerly resolves
+                    // 'keyof T' to a literal union type and we can't recover T from that type.
+                    type.modifiersType = instantiateType(getTypeFromTypeNode(getConstraintDeclarationForMappedType(type).type), type.mapper);
+                }
+                else {
+                    // Otherwise, get the declared constraint type, and if the constraint type is a type parameter,
+                    // get the constraint of that type parameter. If the resulting type is an indexed type 'keyof T',
+                    // the modifiers type is T. Otherwise, the modifiers type is unknown.
+                    var declaredType = getTypeFromMappedTypeNode(type.declaration);
+                    var constraint = getConstraintTypeFromMappedType(declaredType);
+                    var extendedConstraint = constraint && constraint.flags & 262144 /* TypeParameter */ ? getConstraintOfTypeParameter(constraint) : constraint;
+                    type.modifiersType = extendedConstraint && extendedConstraint.flags & 4194304 /* Index */ ? instantiateType(extendedConstraint.type, type.mapper) : unknownType;
+                }
+            }
+            return type.modifiersType;
         }
-        function areTypesComparable(type1, type2) {
-            return isTypeComparableTo(type1, type2) || isTypeComparableTo(type2, type1);
+        function getMappedTypeModifiers(type) {
+            var declaration = type.declaration;
+            return (declaration.readonlyToken ? declaration.readonlyToken.kind === 40 /* MinusToken */ ? 2 /* ExcludeReadonly */ : 1 /* IncludeReadonly */ : 0) |
+                (declaration.questionToken ? declaration.questionToken.kind === 40 /* MinusToken */ ? 8 /* ExcludeOptional */ : 4 /* IncludeOptional */ : 0);
         }
-        function checkTypeAssignableTo(source, target, errorNode, headMessage, containingMessageChain, errorOutputObject) {
-            return checkTypeRelatedTo(source, target, assignableRelation, errorNode, headMessage, containingMessageChain, errorOutputObject);
+        function getMappedTypeOptionality(type) {
+            var modifiers = getMappedTypeModifiers(type);
+            return modifiers & 8 /* ExcludeOptional */ ? -1 : modifiers & 4 /* IncludeOptional */ ? 1 : 0;
         }
-        /**
-         * Like `checkTypeAssignableTo`, but if it would issue an error, instead performs structural comparisons of the types using the given expression node to
-         * attempt to issue more specific errors on, for example, specific object literal properties or tuple members.
-         */
-        function checkTypeAssignableToAndOptionallyElaborate(source, target, errorNode, expr, headMessage, containingMessageChain) {
-            return checkTypeRelatedToAndOptionallyElaborate(source, target, assignableRelation, errorNode, expr, headMessage, containingMessageChain, /*errorOutputContainer*/ undefined);
+        function getCombinedMappedTypeOptionality(type) {
+            var optionality = getMappedTypeOptionality(type);
+            var modifiersType = getModifiersTypeFromMappedType(type);
+            return optionality || (isGenericMappedType(modifiersType) ? getMappedTypeOptionality(modifiersType) : 0);
         }
-        function checkTypeRelatedToAndOptionallyElaborate(source, target, relation, errorNode, expr, headMessage, containingMessageChain, errorOutputContainer) {
-            if (isTypeRelatedTo(source, target, relation))
-                return true;
-            if (!errorNode || !elaborateError(expr, source, target, relation, headMessage, containingMessageChain, errorOutputContainer)) {
-                return checkTypeRelatedTo(source, target, relation, errorNode, headMessage, containingMessageChain, errorOutputContainer);
-            }
-            return false;
+        function isPartialMappedType(type) {
+            return !!(ts.getObjectFlags(type) & 32 /* Mapped */ && getMappedTypeModifiers(type) & 4 /* IncludeOptional */);
         }
-        function isOrHasGenericConditional(type) {
-            return !!(type.flags & 16777216 /* Conditional */ || (type.flags & 2097152 /* Intersection */ && ts.some(type.types, isOrHasGenericConditional)));
+        function isGenericMappedType(type) {
+            return !!(ts.getObjectFlags(type) & 32 /* Mapped */) && isGenericIndexType(getConstraintTypeFromMappedType(type));
         }
-        function elaborateError(node, source, target, relation, headMessage, containingMessageChain, errorOutputContainer) {
-            if (!node || isOrHasGenericConditional(target))
-                return false;
-            if (!checkTypeRelatedTo(source, target, relation, /*errorNode*/ undefined)
-                && elaborateDidYouMeanToCallOrConstruct(node, source, target, relation, headMessage, containingMessageChain, errorOutputContainer)) {
-                return true;
-            }
-            switch (node.kind) {
-                case 276 /* JsxExpression */:
-                case 200 /* ParenthesizedExpression */:
-                    return elaborateError(node.expression, source, target, relation, headMessage, containingMessageChain, errorOutputContainer);
-                case 209 /* BinaryExpression */:
-                    switch (node.operatorToken.kind) {
-                        case 62 /* EqualsToken */:
-                        case 27 /* CommaToken */:
-                            return elaborateError(node.right, source, target, relation, headMessage, containingMessageChain, errorOutputContainer);
+        function resolveStructuredTypeMembers(type) {
+            if (!type.members) {
+                if (type.flags & 524288 /* Object */) {
+                    if (type.objectFlags & 4 /* Reference */) {
+                        resolveTypeReferenceMembers(type);
                     }
-                    break;
-                case 193 /* ObjectLiteralExpression */:
-                    return elaborateObjectLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer);
-                case 192 /* ArrayLiteralExpression */:
-                    return elaborateArrayLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer);
-                case 274 /* JsxAttributes */:
-                    return elaborateJsxComponents(node, source, target, relation, containingMessageChain, errorOutputContainer);
-                case 202 /* ArrowFunction */:
-                    return elaborateArrowFunction(node, source, target, relation, containingMessageChain, errorOutputContainer);
-            }
-            return false;
-        }
-        function elaborateDidYouMeanToCallOrConstruct(node, source, target, relation, headMessage, containingMessageChain, errorOutputContainer) {
-            var callSignatures = getSignaturesOfType(source, 0 /* Call */);
-            var constructSignatures = getSignaturesOfType(source, 1 /* Construct */);
-            for (var _i = 0, _a = [constructSignatures, callSignatures]; _i < _a.length; _i++) {
-                var signatures = _a[_i];
-                if (ts.some(signatures, function (s) {
-                    var returnType = getReturnTypeOfSignature(s);
-                    return !(returnType.flags & (1 /* Any */ | 131072 /* Never */)) && checkTypeRelatedTo(returnType, target, relation, /*errorNode*/ undefined);
-                })) {
-                    var resultObj = errorOutputContainer || {};
-                    checkTypeAssignableTo(source, target, node, headMessage, containingMessageChain, resultObj);
-                    var diagnostic = resultObj.errors[resultObj.errors.length - 1];
-                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(node, signatures === constructSignatures ? ts.Diagnostics.Did_you_mean_to_use_new_with_this_expression : ts.Diagnostics.Did_you_mean_to_call_this_expression));
-                    return true;
-                }
-            }
-            return false;
-        }
-        function elaborateArrowFunction(node, source, target, relation, containingMessageChain, errorOutputContainer) {
-            // Don't elaborate blocks
-            if (ts.isBlock(node.body)) {
-                return false;
-            }
-            // Or functions with annotated parameter types
-            if (ts.some(node.parameters, ts.hasType)) {
-                return false;
-            }
-            var sourceSig = getSingleCallSignature(source);
-            if (!sourceSig) {
-                return false;
-            }
-            var targetSignatures = getSignaturesOfType(target, 0 /* Call */);
-            if (!ts.length(targetSignatures)) {
-                return false;
-            }
-            var returnExpression = node.body;
-            var sourceReturn = getReturnTypeOfSignature(sourceSig);
-            var targetReturn = getUnionType(ts.map(targetSignatures, getReturnTypeOfSignature));
-            if (!checkTypeRelatedTo(sourceReturn, targetReturn, relation, /*errorNode*/ undefined)) {
-                var elaborated = returnExpression && elaborateError(returnExpression, sourceReturn, targetReturn, relation, /*headMessage*/ undefined, containingMessageChain, errorOutputContainer);
-                if (elaborated) {
-                    return elaborated;
-                }
-                var resultObj = errorOutputContainer || {};
-                checkTypeRelatedTo(sourceReturn, targetReturn, relation, returnExpression, /*message*/ undefined, containingMessageChain, resultObj);
-                if (resultObj.errors) {
-                    if (target.symbol && ts.length(target.symbol.declarations)) {
-                        ts.addRelatedInfo(resultObj.errors[resultObj.errors.length - 1], ts.createDiagnosticForNode(target.symbol.declarations[0], ts.Diagnostics.The_expected_type_comes_from_the_return_type_of_this_signature));
+                    else if (type.objectFlags & 3 /* ClassOrInterface */) {
+                        resolveClassOrInterfaceMembers(type);
                     }
-                    if ((ts.getFunctionFlags(node) & 2 /* Async */) === 0
-                        // exclude cases where source itself is promisy - this way we don't make a suggestion when relating
-                        // an IPromise and a Promise that are slightly different
-                        && !getTypeOfPropertyOfType(sourceReturn, "then")
-                        && checkTypeRelatedTo(createPromiseType(sourceReturn), targetReturn, relation, /*errorNode*/ undefined)) {
-                        ts.addRelatedInfo(resultObj.errors[resultObj.errors.length - 1], ts.createDiagnosticForNode(node, ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async));
+                    else if (type.objectFlags & 2048 /* ReverseMapped */) {
+                        resolveReverseMappedTypeMembers(type);
+                    }
+                    else if (type.objectFlags & 16 /* Anonymous */) {
+                        resolveAnonymousTypeMembers(type);
+                    }
+                    else if (type.objectFlags & 32 /* Mapped */) {
+                        resolveMappedTypeMembers(type);
                     }
-                    return true;
                 }
-            }
-            return false;
-        }
-        function getBestMatchIndexedAccessTypeOrUndefined(source, target, nameType) {
-            var idx = getIndexedAccessTypeOrUndefined(target, nameType);
-            if (idx) {
-                return idx;
-            }
-            if (target.flags & 1048576 /* Union */) {
-                var best = getBestMatchingType(source, target);
-                if (best) {
-                    return getIndexedAccessTypeOrUndefined(best, nameType);
+                else if (type.flags & 1048576 /* Union */) {
+                    resolveUnionTypeMembers(type);
+                }
+                else if (type.flags & 2097152 /* Intersection */) {
+                    resolveIntersectionTypeMembers(type);
                 }
             }
+            return type;
         }
-        function checkExpressionForMutableLocationWithContextualType(next, sourcePropType) {
-            next.contextualType = sourcePropType;
-            try {
-                return checkExpressionForMutableLocation(next, 1 /* Contextual */, sourcePropType);
-            }
-            finally {
-                next.contextualType = undefined;
+        /** Return properties of an object type or an empty array for other types */
+        function getPropertiesOfObjectType(type) {
+            if (type.flags & 524288 /* Object */) {
+                return resolveStructuredTypeMembers(type).properties;
             }
+            return ts.emptyArray;
         }
-        /**
-         * For every element returned from the iterator, checks that element to issue an error on a property of that element's type
-         * If that element would issue an error, we first attempt to dive into that element's inner expression and issue a more specific error by recuring into `elaborateError`
-         * Otherwise, we issue an error on _every_ element which fail the assignability check
+        /** If the given type is an object type and that type has a property by the given name,
+         * return the symbol for that property. Otherwise return undefined.
          */
-        function elaborateElementwise(iterator, source, target, relation, containingMessageChain, errorOutputContainer) {
-            // Assignability failure - check each prop individually, and if that fails, fall back on the bad error span
-            var reportedError = false;
-            for (var status = iterator.next(); !status.done; status = iterator.next()) {
-                var _a = status.value, prop = _a.errorNode, next = _a.innerExpression, nameType = _a.nameType, errorMessage = _a.errorMessage;
-                var targetPropType = getBestMatchIndexedAccessTypeOrUndefined(source, target, nameType);
-                if (!targetPropType || targetPropType.flags & 8388608 /* IndexedAccess */)
-                    continue; // Don't elaborate on indexes on generic variables
-                var sourcePropType = getIndexedAccessTypeOrUndefined(source, nameType);
-                if (sourcePropType && !checkTypeRelatedTo(sourcePropType, targetPropType, relation, /*errorNode*/ undefined)) {
-                    var elaborated = next && elaborateError(next, sourcePropType, targetPropType, relation, /*headMessage*/ undefined, containingMessageChain, errorOutputContainer);
-                    if (elaborated) {
-                        reportedError = true;
-                    }
-                    else {
-                        // Issue error on the prop itself, since the prop couldn't elaborate the error
-                        var resultObj = errorOutputContainer || {};
-                        // Use the expression type, if available
-                        var specificSource = next ? checkExpressionForMutableLocationWithContextualType(next, sourcePropType) : sourcePropType;
-                        var result = checkTypeRelatedTo(specificSource, targetPropType, relation, prop, errorMessage, containingMessageChain, resultObj);
-                        if (result && specificSource !== sourcePropType) {
-                            // If for whatever reason the expression type doesn't yield an error, make sure we still issue an error on the sourcePropType
-                            checkTypeRelatedTo(sourcePropType, targetPropType, relation, prop, errorMessage, containingMessageChain, resultObj);
-                        }
-                        if (resultObj.errors) {
-                            var reportedDiag = resultObj.errors[resultObj.errors.length - 1];
-                            var propertyName = isTypeUsableAsPropertyName(nameType) ? getPropertyNameFromType(nameType) : undefined;
-                            var targetProp = propertyName !== undefined ? getPropertyOfType(target, propertyName) : undefined;
-                            var issuedElaboration = false;
-                            if (!targetProp) {
-                                var indexInfo = isTypeAssignableToKind(nameType, 296 /* NumberLike */) && getIndexInfoOfType(target, 1 /* Number */) ||
-                                    getIndexInfoOfType(target, 0 /* String */) ||
-                                    undefined;
-                                if (indexInfo && indexInfo.declaration && !ts.getSourceFileOfNode(indexInfo.declaration).hasNoDefaultLib) {
-                                    issuedElaboration = true;
-                                    ts.addRelatedInfo(reportedDiag, ts.createDiagnosticForNode(indexInfo.declaration, ts.Diagnostics.The_expected_type_comes_from_this_index_signature));
-                                }
-                            }
-                            if (!issuedElaboration && (targetProp && ts.length(targetProp.declarations) || target.symbol && ts.length(target.symbol.declarations))) {
-                                var targetNode = targetProp && ts.length(targetProp.declarations) ? targetProp.declarations[0] : target.symbol.declarations[0];
-                                if (!ts.getSourceFileOfNode(targetNode).hasNoDefaultLib) {
-                                    ts.addRelatedInfo(reportedDiag, ts.createDiagnosticForNode(targetNode, ts.Diagnostics.The_expected_type_comes_from_property_0_which_is_declared_here_on_type_1, propertyName && !(nameType.flags & 8192 /* UniqueESSymbol */) ? ts.unescapeLeadingUnderscores(propertyName) : typeToString(nameType), typeToString(target)));
-                                }
+        function getPropertyOfObjectType(type, name) {
+            if (type.flags & 524288 /* Object */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                var symbol = resolved.members.get(name);
+                if (symbol && symbolIsValue(symbol)) {
+                    return symbol;
+                }
+            }
+        }
+        function getPropertiesOfUnionOrIntersectionType(type) {
+            if (!type.resolvedProperties) {
+                var members = ts.createSymbolTable();
+                for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
+                    var current = _a[_i];
+                    for (var _b = 0, _c = getPropertiesOfType(current); _b < _c.length; _b++) {
+                        var prop = _c[_b];
+                        if (!members.has(prop.escapedName)) {
+                            var combinedProp = getPropertyOfUnionOrIntersectionType(type, prop.escapedName);
+                            if (combinedProp) {
+                                members.set(prop.escapedName, combinedProp);
                             }
                         }
-                        reportedError = true;
+                    }
+                    // The properties of a union type are those that are present in all constituent types, so
+                    // we only need to check the properties of the first type without index signature
+                    if (type.flags & 1048576 /* Union */ && !getIndexInfoOfType(current, 0 /* String */) && !getIndexInfoOfType(current, 1 /* Number */)) {
+                        break;
                     }
                 }
+                type.resolvedProperties = getNamedMembers(members);
             }
-            return reportedError;
+            return type.resolvedProperties;
         }
-        function generateJsxAttributes(node) {
-            var _i, _a, prop;
-            return __generator(this, function (_b) {
-                switch (_b.label) {
-                    case 0:
-                        if (!ts.length(node.properties))
-                            return [2 /*return*/];
-                        _i = 0, _a = node.properties;
-                        _b.label = 1;
-                    case 1:
-                        if (!(_i < _a.length)) return [3 /*break*/, 4];
-                        prop = _a[_i];
-                        if (ts.isJsxSpreadAttribute(prop))
-                            return [3 /*break*/, 3];
-                        return [4 /*yield*/, { errorNode: prop.name, innerExpression: prop.initializer, nameType: getLiteralType(ts.idText(prop.name)) }];
-                    case 2:
-                        _b.sent();
-                        _b.label = 3;
-                    case 3:
-                        _i++;
-                        return [3 /*break*/, 1];
-                    case 4: return [2 /*return*/];
-                }
-            });
+        function getPropertiesOfType(type) {
+            type = getReducedApparentType(type);
+            return type.flags & 3145728 /* UnionOrIntersection */ ?
+                getPropertiesOfUnionOrIntersectionType(type) :
+                getPropertiesOfObjectType(type);
         }
-        function generateJsxChildren(node, getInvalidTextDiagnostic) {
-            var memberOffset, i, child, nameType, elem;
-            return __generator(this, function (_a) {
-                switch (_a.label) {
-                    case 0:
-                        if (!ts.length(node.children))
-                            return [2 /*return*/];
-                        memberOffset = 0;
-                        i = 0;
-                        _a.label = 1;
-                    case 1:
-                        if (!(i < node.children.length)) return [3 /*break*/, 5];
-                        child = node.children[i];
-                        nameType = getLiteralType(i - memberOffset);
-                        elem = getElaborationElementForJsxChild(child, nameType, getInvalidTextDiagnostic);
-                        if (!elem) return [3 /*break*/, 3];
-                        return [4 /*yield*/, elem];
-                    case 2:
-                        _a.sent();
-                        return [3 /*break*/, 4];
-                    case 3:
-                        memberOffset++;
-                        _a.label = 4;
-                    case 4:
-                        i++;
-                        return [3 /*break*/, 1];
-                    case 5: return [2 /*return*/];
-                }
+        function isTypeInvalidDueToUnionDiscriminant(contextualType, obj) {
+            var list = obj.properties;
+            return list.some(function (property) {
+                var nameType = property.name && getLiteralTypeFromPropertyName(property.name);
+                var name = nameType && isTypeUsableAsPropertyName(nameType) ? getPropertyNameFromType(nameType) : undefined;
+                var expected = name === undefined ? undefined : getTypeOfPropertyOfType(contextualType, name);
+                return !!expected && isLiteralType(expected) && !isTypeAssignableTo(getTypeOfNode(property), expected);
             });
         }
-        function getElaborationElementForJsxChild(child, nameType, getInvalidTextDiagnostic) {
-            switch (child.kind) {
-                case 276 /* JsxExpression */:
-                    // child is of the type of the expression
-                    return { errorNode: child, innerExpression: child.expression, nameType: nameType };
-                case 11 /* JsxText */:
-                    if (child.containsOnlyTriviaWhiteSpaces) {
-                        break; // Whitespace only jsx text isn't real jsx text
+        function getAllPossiblePropertiesOfTypes(types) {
+            var unionType = getUnionType(types);
+            if (!(unionType.flags & 1048576 /* Union */)) {
+                return getAugmentedPropertiesOfType(unionType);
+            }
+            var props = ts.createSymbolTable();
+            for (var _i = 0, types_5 = types; _i < types_5.length; _i++) {
+                var memberType = types_5[_i];
+                for (var _a = 0, _b = getAugmentedPropertiesOfType(memberType); _a < _b.length; _a++) {
+                    var escapedName = _b[_a].escapedName;
+                    if (!props.has(escapedName)) {
+                        var prop = createUnionOrIntersectionProperty(unionType, escapedName);
+                        // May be undefined if the property is private
+                        if (prop)
+                            props.set(escapedName, prop);
                     }
-                    // child is a string
-                    return { errorNode: child, innerExpression: undefined, nameType: nameType, errorMessage: getInvalidTextDiagnostic() };
-                case 266 /* JsxElement */:
-                case 267 /* JsxSelfClosingElement */:
-                case 270 /* JsxFragment */:
-                    // child is of type JSX.Element
-                    return { errorNode: child, innerExpression: child, nameType: nameType };
-                default:
-                    return ts.Debug.assertNever(child, "Found invalid jsx child");
+                }
             }
+            return ts.arrayFrom(props.values());
         }
-        function getSemanticJsxChildren(children) {
-            return ts.filter(children, function (i) { return !ts.isJsxText(i) || !i.containsOnlyTriviaWhiteSpaces; });
+        function getConstraintOfType(type) {
+            return type.flags & 262144 /* TypeParameter */ ? getConstraintOfTypeParameter(type) :
+                type.flags & 8388608 /* IndexedAccess */ ? getConstraintOfIndexedAccess(type) :
+                    type.flags & 16777216 /* Conditional */ ? getConstraintOfConditionalType(type) :
+                        getBaseConstraintOfType(type);
         }
-        function elaborateJsxComponents(node, source, target, relation, containingMessageChain, errorOutputContainer) {
-            var result = elaborateElementwise(generateJsxAttributes(node), source, target, relation, containingMessageChain, errorOutputContainer);
-            var invalidTextDiagnostic;
-            if (ts.isJsxOpeningElement(node.parent) && ts.isJsxElement(node.parent.parent)) {
-                var containingElement = node.parent.parent;
-                var childPropName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(node));
-                var childrenPropName = childPropName === undefined ? "children" : ts.unescapeLeadingUnderscores(childPropName);
-                var childrenNameType = getLiteralType(childrenPropName);
-                var childrenTargetType = getIndexedAccessType(target, childrenNameType);
-                var validChildren = getSemanticJsxChildren(containingElement.children);
-                if (!ts.length(validChildren)) {
-                    return result;
-                }
-                var moreThanOneRealChildren = ts.length(validChildren) > 1;
-                var arrayLikeTargetParts = filterType(childrenTargetType, isArrayOrTupleLikeType);
-                var nonArrayLikeTargetParts = filterType(childrenTargetType, function (t) { return !isArrayOrTupleLikeType(t); });
-                if (moreThanOneRealChildren) {
-                    if (arrayLikeTargetParts !== neverType) {
-                        var realSource = createTupleType(checkJsxChildren(containingElement, 0 /* Normal */));
-                        var children = generateJsxChildren(containingElement, getInvalidTextualChildDiagnostic);
-                        result = elaborateElementwise(children, realSource, arrayLikeTargetParts, relation, containingMessageChain, errorOutputContainer) || result;
-                    }
-                    else if (!isTypeRelatedTo(getIndexedAccessType(source, childrenNameType), childrenTargetType, relation)) {
-                        // arity mismatch
-                        result = true;
-                        var diag = error(containingElement.openingElement.tagName, ts.Diagnostics.This_JSX_tag_s_0_prop_expects_a_single_child_of_type_1_but_multiple_children_were_provided, childrenPropName, typeToString(childrenTargetType));
-                        if (errorOutputContainer && errorOutputContainer.skipLogging) {
-                            (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
-                        }
-                    }
-                }
-                else {
-                    if (nonArrayLikeTargetParts !== neverType) {
-                        var child = validChildren[0];
-                        var elem_1 = getElaborationElementForJsxChild(child, childrenNameType, getInvalidTextualChildDiagnostic);
-                        if (elem_1) {
-                            result = elaborateElementwise((function () { return __generator(this, function (_a) {
-                                switch (_a.label) {
-                                    case 0: return [4 /*yield*/, elem_1];
-                                    case 1:
-                                        _a.sent();
-                                        return [2 /*return*/];
-                                }
-                            }); })(), source, target, relation, containingMessageChain, errorOutputContainer) || result;
-                        }
-                    }
-                    else if (!isTypeRelatedTo(getIndexedAccessType(source, childrenNameType), childrenTargetType, relation)) {
-                        // arity mismatch
-                        result = true;
-                        var diag = error(containingElement.openingElement.tagName, ts.Diagnostics.This_JSX_tag_s_0_prop_expects_type_1_which_requires_multiple_children_but_only_a_single_child_was_provided, childrenPropName, typeToString(childrenTargetType));
-                        if (errorOutputContainer && errorOutputContainer.skipLogging) {
-                            (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
-                        }
-                    }
+        function getConstraintOfTypeParameter(typeParameter) {
+            return hasNonCircularBaseConstraint(typeParameter) ? getConstraintFromTypeParameter(typeParameter) : undefined;
+        }
+        function getConstraintOfIndexedAccess(type) {
+            return hasNonCircularBaseConstraint(type) ? getConstraintFromIndexedAccess(type) : undefined;
+        }
+        function getSimplifiedTypeOrConstraint(type) {
+            var simplified = getSimplifiedType(type, /*writing*/ false);
+            return simplified !== type ? simplified : getConstraintOfType(type);
+        }
+        function getConstraintFromIndexedAccess(type) {
+            var indexConstraint = getSimplifiedTypeOrConstraint(type.indexType);
+            if (indexConstraint && indexConstraint !== type.indexType) {
+                var indexedAccess = getIndexedAccessTypeOrUndefined(type.objectType, indexConstraint, type.noUncheckedIndexedAccessCandidate);
+                if (indexedAccess) {
+                    return indexedAccess;
                 }
             }
-            return result;
-            function getInvalidTextualChildDiagnostic() {
-                if (!invalidTextDiagnostic) {
-                    var tagNameText = ts.getTextOfNode(node.parent.tagName);
-                    var childPropName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(node));
-                    var childrenPropName = childPropName === undefined ? "children" : ts.unescapeLeadingUnderscores(childPropName);
-                    var childrenTargetType = getIndexedAccessType(target, getLiteralType(childrenPropName));
-                    var diagnostic = ts.Diagnostics._0_components_don_t_accept_text_as_child_elements_Text_in_JSX_has_the_type_string_but_the_expected_type_of_1_is_2;
-                    invalidTextDiagnostic = __assign(__assign({}, diagnostic), { key: "!!ALREADY FORMATTED!!", message: ts.formatMessage(/*_dummy*/ undefined, diagnostic, tagNameText, childrenPropName, typeToString(childrenTargetType)) });
-                }
-                return invalidTextDiagnostic;
+            var objectConstraint = getSimplifiedTypeOrConstraint(type.objectType);
+            if (objectConstraint && objectConstraint !== type.objectType) {
+                return getIndexedAccessTypeOrUndefined(objectConstraint, type.indexType, type.noUncheckedIndexedAccessCandidate);
             }
+            return undefined;
         }
-        function generateLimitedTupleElements(node, target) {
-            var len, i, elem, nameType;
-            return __generator(this, function (_a) {
-                switch (_a.label) {
-                    case 0:
-                        len = ts.length(node.elements);
-                        if (!len)
-                            return [2 /*return*/];
-                        i = 0;
-                        _a.label = 1;
-                    case 1:
-                        if (!(i < len)) return [3 /*break*/, 4];
-                        // Skip elements which do not exist in the target - a length error on the tuple overall is likely better than an error on a mismatched index signature
-                        if (isTupleLikeType(target) && !getPropertyOfType(target, ("" + i)))
-                            return [3 /*break*/, 3];
-                        elem = node.elements[i];
-                        if (ts.isOmittedExpression(elem))
-                            return [3 /*break*/, 3];
-                        nameType = getLiteralType(i);
-                        return [4 /*yield*/, { errorNode: elem, innerExpression: elem, nameType: nameType }];
-                    case 2:
-                        _a.sent();
-                        _a.label = 3;
-                    case 3:
-                        i++;
-                        return [3 /*break*/, 1];
-                    case 4: return [2 /*return*/];
-                }
-            });
-        }
-        function elaborateArrayLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer) {
-            if (target.flags & 131068 /* Primitive */)
-                return false;
-            if (isTupleLikeType(source)) {
-                return elaborateElementwise(generateLimitedTupleElements(node, target), source, target, relation, containingMessageChain, errorOutputContainer);
+        function getDefaultConstraintOfConditionalType(type) {
+            if (!type.resolvedDefaultConstraint) {
+                // An `any` branch of a conditional type would normally be viral - specifically, without special handling here,
+                // a conditional type with a single branch of type `any` would be assignable to anything, since it's constraint would simplify to
+                // just `any`. This result is _usually_ unwanted - so instead here we elide an `any` branch from the constraint type,
+                // in effect treating `any` like `never` rather than `unknown` in this location.
+                var trueConstraint = getInferredTrueTypeFromConditionalType(type);
+                var falseConstraint = getFalseTypeFromConditionalType(type);
+                type.resolvedDefaultConstraint = isTypeAny(trueConstraint) ? falseConstraint : isTypeAny(falseConstraint) ? trueConstraint : getUnionType([trueConstraint, falseConstraint]);
             }
-            // recreate a tuple from the elements, if possible
-            // Since we're re-doing the expression type, we need to reapply the contextual type
-            var oldContext = node.contextualType;
-            node.contextualType = target;
-            try {
-                var tupleizedType = checkArrayLiteral(node, 1 /* Contextual */, /*forceTuple*/ true);
-                node.contextualType = oldContext;
-                if (isTupleLikeType(tupleizedType)) {
-                    return elaborateElementwise(generateLimitedTupleElements(node, target), tupleizedType, target, relation, containingMessageChain, errorOutputContainer);
+            return type.resolvedDefaultConstraint;
+        }
+        function getConstraintOfDistributiveConditionalType(type) {
+            // Check if we have a conditional type of the form 'T extends U ? X : Y', where T is a constrained
+            // type parameter. If so, create an instantiation of the conditional type where T is replaced
+            // with its constraint. We do this because if the constraint is a union type it will be distributed
+            // over the conditional type and possibly reduced. For example, 'T extends undefined ? never : T'
+            // removes 'undefined' from T.
+            // We skip returning a distributive constraint for a restrictive instantiation of a conditional type
+            // as the constraint for all type params (check type included) have been replace with `unknown`, which
+            // is going to produce even more false positive/negative results than the distribute constraint already does.
+            // Please note: the distributive constraint is a kludge for emulating what a negated type could to do filter
+            // a union - once negated types exist and are applied to the conditional false branch, this "constraint"
+            // likely doesn't need to exist.
+            if (type.root.isDistributive && type.restrictiveInstantiation !== type) {
+                var simplified = getSimplifiedType(type.checkType, /*writing*/ false);
+                var constraint = simplified === type.checkType ? getConstraintOfType(simplified) : simplified;
+                if (constraint && constraint !== type.checkType) {
+                    var instantiated = getConditionalTypeInstantiation(type, prependTypeMapping(type.root.checkType, constraint, type.mapper));
+                    if (!(instantiated.flags & 131072 /* Never */)) {
+                        return instantiated;
+                    }
                 }
-                return false;
-            }
-            finally {
-                node.contextualType = oldContext;
             }
+            return undefined;
         }
-        function generateObjectLiteralElements(node) {
-            var _i, _a, prop, type, _b;
-            return __generator(this, function (_c) {
-                switch (_c.label) {
-                    case 0:
-                        if (!ts.length(node.properties))
-                            return [2 /*return*/];
-                        _i = 0, _a = node.properties;
-                        _c.label = 1;
-                    case 1:
-                        if (!(_i < _a.length)) return [3 /*break*/, 8];
-                        prop = _a[_i];
-                        if (ts.isSpreadAssignment(prop))
-                            return [3 /*break*/, 7];
-                        type = getLiteralTypeFromProperty(getSymbolOfNode(prop), 8576 /* StringOrNumberLiteralOrUnique */);
-                        if (!type || (type.flags & 131072 /* Never */)) {
-                            return [3 /*break*/, 7];
+        function getConstraintFromConditionalType(type) {
+            return getConstraintOfDistributiveConditionalType(type) || getDefaultConstraintOfConditionalType(type);
+        }
+        function getConstraintOfConditionalType(type) {
+            return hasNonCircularBaseConstraint(type) ? getConstraintFromConditionalType(type) : undefined;
+        }
+        function getEffectiveConstraintOfIntersection(types, targetIsUnion) {
+            var constraints;
+            var hasDisjointDomainType = false;
+            for (var _i = 0, types_6 = types; _i < types_6.length; _i++) {
+                var t = types_6[_i];
+                if (t.flags & 465829888 /* Instantiable */) {
+                    // We keep following constraints as long as we have an instantiable type that is known
+                    // not to be circular or infinite (hence we stop on index access types).
+                    var constraint = getConstraintOfType(t);
+                    while (constraint && constraint.flags & (262144 /* TypeParameter */ | 4194304 /* Index */ | 16777216 /* Conditional */)) {
+                        constraint = getConstraintOfType(constraint);
+                    }
+                    if (constraint) {
+                        constraints = ts.append(constraints, constraint);
+                        if (targetIsUnion) {
+                            constraints = ts.append(constraints, t);
                         }
-                        _b = prop.kind;
-                        switch (_b) {
-                            case 164 /* SetAccessor */: return [3 /*break*/, 2];
-                            case 163 /* GetAccessor */: return [3 /*break*/, 2];
-                            case 161 /* MethodDeclaration */: return [3 /*break*/, 2];
-                            case 282 /* ShorthandPropertyAssignment */: return [3 /*break*/, 2];
-                            case 281 /* PropertyAssignment */: return [3 /*break*/, 4];
+                    }
+                }
+                else if (t.flags & 469892092 /* DisjointDomains */) {
+                    hasDisjointDomainType = true;
+                }
+            }
+            // If the target is a union type or if we are intersecting with types belonging to one of the
+            // disjoint domains, we may end up producing a constraint that hasn't been examined before.
+            if (constraints && (targetIsUnion || hasDisjointDomainType)) {
+                if (hasDisjointDomainType) {
+                    // We add any types belong to one of the disjoint domains because they might cause the final
+                    // intersection operation to reduce the union constraints.
+                    for (var _a = 0, types_7 = types; _a < types_7.length; _a++) {
+                        var t = types_7[_a];
+                        if (t.flags & 469892092 /* DisjointDomains */) {
+                            constraints = ts.append(constraints, t);
                         }
-                        return [3 /*break*/, 6];
-                    case 2: return [4 /*yield*/, { errorNode: prop.name, innerExpression: undefined, nameType: type }];
-                    case 3:
-                        _c.sent();
-                        return [3 /*break*/, 7];
-                    case 4: return [4 /*yield*/, { errorNode: prop.name, innerExpression: prop.initializer, nameType: type, errorMessage: ts.isComputedNonLiteralName(prop.name) ? ts.Diagnostics.Type_of_computed_property_s_value_is_0_which_is_not_assignable_to_type_1 : undefined }];
-                    case 5:
-                        _c.sent();
-                        return [3 /*break*/, 7];
-                    case 6:
-                        ts.Debug.assertNever(prop);
-                        _c.label = 7;
-                    case 7:
-                        _i++;
-                        return [3 /*break*/, 1];
-                    case 8: return [2 /*return*/];
+                    }
                 }
-            });
+                return getIntersectionType(constraints);
+            }
+            return undefined;
         }
-        function elaborateObjectLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer) {
-            if (target.flags & 131068 /* Primitive */)
-                return false;
-            return elaborateElementwise(generateObjectLiteralElements(node), source, target, relation, containingMessageChain, errorOutputContainer);
+        function getBaseConstraintOfType(type) {
+            if (type.flags & (58982400 /* InstantiableNonPrimitive */ | 3145728 /* UnionOrIntersection */ | 134217728 /* TemplateLiteral */ | 268435456 /* StringMapping */)) {
+                var constraint = getResolvedBaseConstraint(type);
+                return constraint !== noConstraintType && constraint !== circularConstraintType ? constraint : undefined;
+            }
+            return type.flags & 4194304 /* Index */ ? keyofConstraintType : undefined;
         }
         /**
-         * This is *not* a bi-directional relationship.
-         * If one needs to check both directions for comparability, use a second call to this function or 'isTypeComparableTo'.
+         * This is similar to `getBaseConstraintOfType` except it returns the input type if there's no base constraint, instead of `undefined`
+         * It also doesn't map indexes to `string`, as where this is used this would be unneeded (and likely undesirable)
          */
-        function checkTypeComparableTo(source, target, errorNode, headMessage, containingMessageChain) {
-            return checkTypeRelatedTo(source, target, comparableRelation, errorNode, headMessage, containingMessageChain);
-        }
-        function isSignatureAssignableTo(source, target, ignoreReturnTypes) {
-            return compareSignaturesRelated(source, target, ignoreReturnTypes ? 4 /* IgnoreReturnTypes */ : 0, /*reportErrors*/ false, 
-            /*errorReporter*/ undefined, /*errorReporter*/ undefined, compareTypesAssignable, /*reportUnreliableMarkers*/ undefined) !== 0 /* False */;
+        function getBaseConstraintOrType(type) {
+            return getBaseConstraintOfType(type) || type;
         }
-        /**
-         * Returns true if `s` is `(...args: any[]) => any` or `(this: any, ...args: any[]) => any`
-         */
-        function isAnySignature(s) {
-            return !s.typeParameters && (!s.thisParameter || isTypeAny(getTypeOfParameter(s.thisParameter))) && s.parameters.length === 1 &&
-                signatureHasRestParameter(s) && (getTypeOfParameter(s.parameters[0]) === anyArrayType || isTypeAny(getTypeOfParameter(s.parameters[0]))) &&
-                isTypeAny(getReturnTypeOfSignature(s));
+        function hasNonCircularBaseConstraint(type) {
+            return getResolvedBaseConstraint(type) !== circularConstraintType;
         }
         /**
-         * See signatureRelatedTo, compareSignaturesIdentical
+         * Return the resolved base constraint of a type variable. The noConstraintType singleton is returned if the
+         * type variable has no constraint, and the circularConstraintType singleton is returned if the constraint
+         * circularly references the type variable.
          */
-        function compareSignaturesRelated(source, target, checkMode, reportErrors, errorReporter, incompatibleErrorReporter, compareTypes, reportUnreliableMarkers) {
-            // TODO (drosen): De-duplicate code between related functions.
-            if (source === target) {
-                return -1 /* True */;
-            }
-            if (isAnySignature(target)) {
-                return -1 /* True */;
-            }
-            var targetCount = getParameterCount(target);
-            var sourceHasMoreParameters = !hasEffectiveRestParameter(target) &&
-                (checkMode & 8 /* StrictArity */ ? hasEffectiveRestParameter(source) || getParameterCount(source) > targetCount : getMinArgumentCount(source) > targetCount);
-            if (sourceHasMoreParameters) {
-                return 0 /* False */;
-            }
-            if (source.typeParameters && source.typeParameters !== target.typeParameters) {
-                target = getCanonicalSignature(target);
-                source = instantiateSignatureInContextOf(source, target, /*inferenceContext*/ undefined, compareTypes);
-            }
-            var sourceCount = getParameterCount(source);
-            var sourceRestType = getNonArrayRestType(source);
-            var targetRestType = getNonArrayRestType(target);
-            if (sourceRestType || targetRestType) {
-                void instantiateType(sourceRestType || targetRestType, reportUnreliableMarkers);
-            }
-            if (sourceRestType && targetRestType && sourceCount !== targetCount) {
-                // We're not able to relate misaligned complex rest parameters
-                return 0 /* False */;
+        function getResolvedBaseConstraint(type) {
+            if (type.resolvedBaseConstraint) {
+                return type.resolvedBaseConstraint;
             }
-            var kind = target.declaration ? target.declaration.kind : 0 /* Unknown */;
-            var strictVariance = !(checkMode & 3 /* Callback */) && strictFunctionTypes && kind !== 161 /* MethodDeclaration */ &&
-                kind !== 160 /* MethodSignature */ && kind !== 162 /* Constructor */;
-            var result = -1 /* True */;
-            var sourceThisType = getThisTypeOfSignature(source);
-            if (sourceThisType && sourceThisType !== voidType) {
-                var targetThisType = getThisTypeOfSignature(target);
-                if (targetThisType) {
-                    // void sources are assignable to anything.
-                    var related = !strictVariance && compareTypes(sourceThisType, targetThisType, /*reportErrors*/ false)
-                        || compareTypes(targetThisType, sourceThisType, reportErrors);
-                    if (!related) {
-                        if (reportErrors) {
-                            errorReporter(ts.Diagnostics.The_this_types_of_each_signature_are_incompatible);
+            var nonTerminating = false;
+            var stack = [];
+            return type.resolvedBaseConstraint = getTypeWithThisArgument(getImmediateBaseConstraint(type), type);
+            function getImmediateBaseConstraint(t) {
+                if (!t.immediateBaseConstraint) {
+                    if (!pushTypeResolution(t, 4 /* ImmediateBaseConstraint */)) {
+                        return circularConstraintType;
+                    }
+                    if (constraintDepth >= 50) {
+                        // We have reached 50 recursive invocations of getImmediateBaseConstraint and there is a
+                        // very high likelihood we're dealing with an infinite generic type that perpetually generates
+                        // new type identities as we descend into it. We stop the recursion here and mark this type
+                        // and the outer types as having circular constraints.
+                        ts.tracing.instant("check" /* Check */, "getImmediateBaseConstraint_DepthLimit", { typeId: t.id, originalTypeId: type.id, depth: constraintDepth });
+                        error(currentNode, ts.Diagnostics.Type_instantiation_is_excessively_deep_and_possibly_infinite);
+                        nonTerminating = true;
+                        return t.immediateBaseConstraint = noConstraintType;
+                    }
+                    var result = void 0;
+                    if (!isDeeplyNestedType(t, stack, stack.length)) {
+                        stack.push(t);
+                        constraintDepth++;
+                        result = computeBaseConstraint(getSimplifiedType(t, /*writing*/ false));
+                        constraintDepth--;
+                        stack.pop();
+                    }
+                    if (!popTypeResolution()) {
+                        if (t.flags & 262144 /* TypeParameter */) {
+                            var errorNode = getConstraintDeclaration(t);
+                            if (errorNode) {
+                                var diagnostic = error(errorNode, ts.Diagnostics.Type_parameter_0_has_a_circular_constraint, typeToString(t));
+                                if (currentNode && !ts.isNodeDescendantOf(errorNode, currentNode) && !ts.isNodeDescendantOf(currentNode, errorNode)) {
+                                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(currentNode, ts.Diagnostics.Circularity_originates_in_type_at_this_location));
+                                }
+                            }
                         }
-                        return 0 /* False */;
+                        result = circularConstraintType;
                     }
-                    result &= related;
-                }
-            }
-            var paramCount = sourceRestType || targetRestType ? Math.min(sourceCount, targetCount) : Math.max(sourceCount, targetCount);
-            var restIndex = sourceRestType || targetRestType ? paramCount - 1 : -1;
-            for (var i = 0; i < paramCount; i++) {
-                var sourceType = i === restIndex ? getRestTypeAtPosition(source, i) : getTypeAtPosition(source, i);
-                var targetType = i === restIndex ? getRestTypeAtPosition(target, i) : getTypeAtPosition(target, i);
-                // In order to ensure that any generic type Foo<T> is at least co-variant with respect to T no matter
-                // how Foo uses T, we need to relate parameters bi-variantly (given that parameters are input positions,
-                // they naturally relate only contra-variantly). However, if the source and target parameters both have
-                // function types with a single call signature, we know we are relating two callback parameters. In
-                // that case it is sufficient to only relate the parameters of the signatures co-variantly because,
-                // similar to return values, callback parameters are output positions. This means that a Promise<T>,
-                // where T is used only in callback parameter positions, will be co-variant (as opposed to bi-variant)
-                // with respect to T.
-                var sourceSig = checkMode & 3 /* Callback */ ? undefined : getSingleCallSignature(getNonNullableType(sourceType));
-                var targetSig = checkMode & 3 /* Callback */ ? undefined : getSingleCallSignature(getNonNullableType(targetType));
-                var callbacks = sourceSig && targetSig && !getTypePredicateOfSignature(sourceSig) && !getTypePredicateOfSignature(targetSig) &&
-                    (getFalsyFlags(sourceType) & 98304 /* Nullable */) === (getFalsyFlags(targetType) & 98304 /* Nullable */);
-                var related = callbacks ?
-                    compareSignaturesRelated(targetSig, sourceSig, (checkMode & 8 /* StrictArity */) | (strictVariance ? 2 /* StrictCallback */ : 1 /* BivariantCallback */), reportErrors, errorReporter, incompatibleErrorReporter, compareTypes, reportUnreliableMarkers) :
-                    !(checkMode & 3 /* Callback */) && !strictVariance && compareTypes(sourceType, targetType, /*reportErrors*/ false) || compareTypes(targetType, sourceType, reportErrors);
-                // With strict arity, (x: number | undefined) => void is a subtype of (x?: number | undefined) => void
-                if (related && checkMode & 8 /* StrictArity */ && i >= getMinArgumentCount(source) && i < getMinArgumentCount(target) && compareTypes(sourceType, targetType, /*reportErrors*/ false)) {
-                    related = 0 /* False */;
-                }
-                if (!related) {
-                    if (reportErrors) {
-                        errorReporter(ts.Diagnostics.Types_of_parameters_0_and_1_are_incompatible, ts.unescapeLeadingUnderscores(getParameterNameAtPosition(source, i)), ts.unescapeLeadingUnderscores(getParameterNameAtPosition(target, i)));
+                    if (nonTerminating) {
+                        result = circularConstraintType;
                     }
-                    return 0 /* False */;
+                    t.immediateBaseConstraint = result || noConstraintType;
                 }
-                result &= related;
+                return t.immediateBaseConstraint;
             }
-            if (!(checkMode & 4 /* IgnoreReturnTypes */)) {
-                // If a signature resolution is already in-flight, skip issuing a circularity error
-                // here and just use the `any` type directly
-                var targetReturnType = isResolvingReturnTypeOfSignature(target) ? anyType
-                    : target.declaration && isJSConstructor(target.declaration) ? getDeclaredTypeOfClassOrInterface(getMergedSymbol(target.declaration.symbol))
-                        : getReturnTypeOfSignature(target);
-                if (targetReturnType === voidType) {
-                    return result;
+            function getBaseConstraint(t) {
+                var c = getImmediateBaseConstraint(t);
+                return c !== noConstraintType && c !== circularConstraintType ? c : undefined;
+            }
+            function computeBaseConstraint(t) {
+                if (t.flags & 262144 /* TypeParameter */) {
+                    var constraint = getConstraintFromTypeParameter(t);
+                    return t.isThisType || !constraint ?
+                        constraint :
+                        getBaseConstraint(constraint);
                 }
-                var sourceReturnType = isResolvingReturnTypeOfSignature(source) ? anyType
-                    : source.declaration && isJSConstructor(source.declaration) ? getDeclaredTypeOfClassOrInterface(getMergedSymbol(source.declaration.symbol))
-                        : getReturnTypeOfSignature(source);
-                // The following block preserves behavior forbidding boolean returning functions from being assignable to type guard returning functions
-                var targetTypePredicate = getTypePredicateOfSignature(target);
-                if (targetTypePredicate) {
-                    var sourceTypePredicate = getTypePredicateOfSignature(source);
-                    if (sourceTypePredicate) {
-                        result &= compareTypePredicateRelatedTo(sourceTypePredicate, targetTypePredicate, reportErrors, errorReporter, compareTypes);
-                    }
-                    else if (ts.isIdentifierTypePredicate(targetTypePredicate)) {
-                        if (reportErrors) {
-                            errorReporter(ts.Diagnostics.Signature_0_must_be_a_type_predicate, signatureToString(source));
+                if (t.flags & 3145728 /* UnionOrIntersection */) {
+                    var types = t.types;
+                    var baseTypes = [];
+                    for (var _i = 0, types_8 = types; _i < types_8.length; _i++) {
+                        var type_3 = types_8[_i];
+                        var baseType = getBaseConstraint(type_3);
+                        if (baseType) {
+                            baseTypes.push(baseType);
                         }
-                        return 0 /* False */;
                     }
+                    return t.flags & 1048576 /* Union */ && baseTypes.length === types.length ? getUnionType(baseTypes) :
+                        t.flags & 2097152 /* Intersection */ && baseTypes.length ? getIntersectionType(baseTypes) :
+                            undefined;
                 }
-                else {
-                    // When relating callback signatures, we still need to relate return types bi-variantly as otherwise
-                    // the containing type wouldn't be co-variant. For example, interface Foo<T> { add(cb: () => T): void }
-                    // wouldn't be co-variant for T without this rule.
-                    result &= checkMode & 1 /* BivariantCallback */ && compareTypes(targetReturnType, sourceReturnType, /*reportErrors*/ false) ||
-                        compareTypes(sourceReturnType, targetReturnType, reportErrors);
-                    if (!result && reportErrors && incompatibleErrorReporter) {
-                        incompatibleErrorReporter(sourceReturnType, targetReturnType);
-                    }
+                if (t.flags & 4194304 /* Index */) {
+                    return keyofConstraintType;
+                }
+                if (t.flags & 134217728 /* TemplateLiteral */) {
+                    var types = t.types;
+                    var constraints = ts.mapDefined(types, getBaseConstraint);
+                    return constraints.length === types.length ? getTemplateLiteralType(t.texts, constraints) : stringType;
+                }
+                if (t.flags & 268435456 /* StringMapping */) {
+                    var constraint = getBaseConstraint(t.type);
+                    return constraint ? getStringMappingType(t.symbol, constraint) : stringType;
+                }
+                if (t.flags & 8388608 /* IndexedAccess */) {
+                    var baseObjectType = getBaseConstraint(t.objectType);
+                    var baseIndexType = getBaseConstraint(t.indexType);
+                    var baseIndexedAccess = baseObjectType && baseIndexType && getIndexedAccessTypeOrUndefined(baseObjectType, baseIndexType, t.noUncheckedIndexedAccessCandidate);
+                    return baseIndexedAccess && getBaseConstraint(baseIndexedAccess);
+                }
+                if (t.flags & 16777216 /* Conditional */) {
+                    var constraint = getConstraintFromConditionalType(t);
+                    constraintDepth++; // Penalize repeating conditional types (this captures the recursion within getConstraintFromConditionalType and carries it forward)
+                    var result = constraint && getBaseConstraint(constraint);
+                    constraintDepth--;
+                    return result;
+                }
+                if (t.flags & 33554432 /* Substitution */) {
+                    return getBaseConstraint(t.substitute);
                 }
+                return t;
             }
-            return result;
         }
-        function compareTypePredicateRelatedTo(source, target, reportErrors, errorReporter, compareTypes) {
-            if (source.kind !== target.kind) {
-                if (reportErrors) {
-                    errorReporter(ts.Diagnostics.A_this_based_type_guard_is_not_compatible_with_a_parameter_based_type_guard);
-                    errorReporter(ts.Diagnostics.Type_predicate_0_is_not_assignable_to_1, typePredicateToString(source), typePredicateToString(target));
+        function getApparentTypeOfIntersectionType(type) {
+            return type.resolvedApparentType || (type.resolvedApparentType = getTypeWithThisArgument(type, type, /*apparentType*/ true));
+        }
+        function getResolvedTypeParameterDefault(typeParameter) {
+            if (!typeParameter.default) {
+                if (typeParameter.target) {
+                    var targetDefault = getResolvedTypeParameterDefault(typeParameter.target);
+                    typeParameter.default = targetDefault ? instantiateType(targetDefault, typeParameter.mapper) : noConstraintType;
                 }
-                return 0 /* False */;
-            }
-            if (source.kind === 1 /* Identifier */ || source.kind === 3 /* AssertsIdentifier */) {
-                if (source.parameterIndex !== target.parameterIndex) {
-                    if (reportErrors) {
-                        errorReporter(ts.Diagnostics.Parameter_0_is_not_in_the_same_position_as_parameter_1, source.parameterName, target.parameterName);
-                        errorReporter(ts.Diagnostics.Type_predicate_0_is_not_assignable_to_1, typePredicateToString(source), typePredicateToString(target));
+                else {
+                    // To block recursion, set the initial value to the resolvingDefaultType.
+                    typeParameter.default = resolvingDefaultType;
+                    var defaultDeclaration = typeParameter.symbol && ts.forEach(typeParameter.symbol.declarations, function (decl) { return ts.isTypeParameterDeclaration(decl) && decl.default; });
+                    var defaultType = defaultDeclaration ? getTypeFromTypeNode(defaultDeclaration) : noConstraintType;
+                    if (typeParameter.default === resolvingDefaultType) {
+                        // If we have not been called recursively, set the correct default type.
+                        typeParameter.default = defaultType;
                     }
-                    return 0 /* False */;
                 }
             }
-            var related = source.type === target.type ? -1 /* True */ :
-                source.type && target.type ? compareTypes(source.type, target.type, reportErrors) :
-                    0 /* False */;
-            if (related === 0 /* False */ && reportErrors) {
-                errorReporter(ts.Diagnostics.Type_predicate_0_is_not_assignable_to_1, typePredicateToString(source), typePredicateToString(target));
-            }
-            return related;
-        }
-        function isImplementationCompatibleWithOverload(implementation, overload) {
-            var erasedSource = getErasedSignature(implementation);
-            var erasedTarget = getErasedSignature(overload);
-            // First see if the return types are compatible in either direction.
-            var sourceReturnType = getReturnTypeOfSignature(erasedSource);
-            var targetReturnType = getReturnTypeOfSignature(erasedTarget);
-            if (targetReturnType === voidType
-                || isTypeRelatedTo(targetReturnType, sourceReturnType, assignableRelation)
-                || isTypeRelatedTo(sourceReturnType, targetReturnType, assignableRelation)) {
-                return isSignatureAssignableTo(erasedSource, erasedTarget, /*ignoreReturnTypes*/ true);
+            else if (typeParameter.default === resolvingDefaultType) {
+                // If we are called recursively for this type parameter, mark the default as circular.
+                typeParameter.default = circularConstraintType;
             }
-            return false;
+            return typeParameter.default;
         }
-        function isEmptyResolvedType(t) {
-            return t !== anyFunctionType &&
-                t.properties.length === 0 &&
-                t.callSignatures.length === 0 &&
-                t.constructSignatures.length === 0 &&
-                !t.stringIndexInfo &&
-                !t.numberIndexInfo;
+        /**
+         * Gets the default type for a type parameter.
+         *
+         * If the type parameter is the result of an instantiation, this gets the instantiated
+         * default type of its target. If the type parameter has no default type or the default is
+         * circular, `undefined` is returned.
+         */
+        function getDefaultFromTypeParameter(typeParameter) {
+            var defaultType = getResolvedTypeParameterDefault(typeParameter);
+            return defaultType !== noConstraintType && defaultType !== circularConstraintType ? defaultType : undefined;
         }
-        function isEmptyObjectType(type) {
-            return type.flags & 524288 /* Object */ ? !isGenericMappedType(type) && isEmptyResolvedType(resolveStructuredTypeMembers(type)) :
-                type.flags & 67108864 /* NonPrimitive */ ? true :
-                    type.flags & 1048576 /* Union */ ? ts.some(type.types, isEmptyObjectType) :
-                        type.flags & 2097152 /* Intersection */ ? ts.every(type.types, isEmptyObjectType) :
-                            false;
+        function hasNonCircularTypeParameterDefault(typeParameter) {
+            return getResolvedTypeParameterDefault(typeParameter) !== circularConstraintType;
         }
-        function isEmptyAnonymousObjectType(type) {
-            return !!(ts.getObjectFlags(type) & 16 /* Anonymous */) && isEmptyObjectType(type);
+        /**
+         * Indicates whether the declaration of a typeParameter has a default type.
+         */
+        function hasTypeParameterDefault(typeParameter) {
+            return !!(typeParameter.symbol && ts.forEach(typeParameter.symbol.declarations, function (decl) { return ts.isTypeParameterDeclaration(decl) && decl.default; }));
         }
-        function isStringIndexSignatureOnlyType(type) {
-            return type.flags & 524288 /* Object */ && !isGenericMappedType(type) && getPropertiesOfType(type).length === 0 && getIndexInfoOfType(type, 0 /* String */) && !getIndexInfoOfType(type, 1 /* Number */) ||
-                type.flags & 3145728 /* UnionOrIntersection */ && ts.every(type.types, isStringIndexSignatureOnlyType) ||
-                false;
+        function getApparentTypeOfMappedType(type) {
+            return type.resolvedApparentType || (type.resolvedApparentType = getResolvedApparentTypeOfMappedType(type));
         }
-        function isEnumTypeRelatedTo(sourceSymbol, targetSymbol, errorReporter) {
-            if (sourceSymbol === targetSymbol) {
-                return true;
-            }
-            var id = getSymbolId(sourceSymbol) + "," + getSymbolId(targetSymbol);
-            var entry = enumRelation.get(id);
-            if (entry !== undefined && !(!(entry & 4 /* Reported */) && entry & 2 /* Failed */ && errorReporter)) {
-                return !!(entry & 1 /* Succeeded */);
-            }
-            if (sourceSymbol.escapedName !== targetSymbol.escapedName || !(sourceSymbol.flags & 256 /* RegularEnum */) || !(targetSymbol.flags & 256 /* RegularEnum */)) {
-                enumRelation.set(id, 2 /* Failed */ | 4 /* Reported */);
-                return false;
+        function getResolvedApparentTypeOfMappedType(type) {
+            var typeVariable = getHomomorphicTypeVariable(type);
+            if (typeVariable && !type.declaration.nameType) {
+                var constraint = getConstraintOfTypeParameter(typeVariable);
+                if (constraint && (isArrayType(constraint) || isTupleType(constraint))) {
+                    return instantiateType(type, prependTypeMapping(typeVariable, constraint, type.mapper));
+                }
             }
-            var targetEnumType = getTypeOfSymbol(targetSymbol);
-            for (var _i = 0, _a = getPropertiesOfType(getTypeOfSymbol(sourceSymbol)); _i < _a.length; _i++) {
-                var property = _a[_i];
-                if (property.flags & 8 /* EnumMember */) {
-                    var targetProperty = getPropertyOfType(targetEnumType, property.escapedName);
-                    if (!targetProperty || !(targetProperty.flags & 8 /* EnumMember */)) {
-                        if (errorReporter) {
-                            errorReporter(ts.Diagnostics.Property_0_is_missing_in_type_1, ts.symbolName(property), typeToString(getDeclaredTypeOfSymbol(targetSymbol), /*enclosingDeclaration*/ undefined, 64 /* UseFullyQualifiedType */));
-                            enumRelation.set(id, 2 /* Failed */ | 4 /* Reported */);
+            return type;
+        }
+        /**
+         * For a type parameter, return the base constraint of the type parameter. For the string, number,
+         * boolean, and symbol primitive types, return the corresponding object types. Otherwise return the
+         * type itself.
+         */
+        function getApparentType(type) {
+            var t = type.flags & 465829888 /* Instantiable */ ? getBaseConstraintOfType(type) || unknownType : type;
+            return ts.getObjectFlags(t) & 32 /* Mapped */ ? getApparentTypeOfMappedType(t) :
+                t.flags & 2097152 /* Intersection */ ? getApparentTypeOfIntersectionType(t) :
+                    t.flags & 402653316 /* StringLike */ ? globalStringType :
+                        t.flags & 296 /* NumberLike */ ? globalNumberType :
+                            t.flags & 2112 /* BigIntLike */ ? getGlobalBigIntType(/*reportErrors*/ languageVersion >= 7 /* ES2020 */) :
+                                t.flags & 528 /* BooleanLike */ ? globalBooleanType :
+                                    t.flags & 12288 /* ESSymbolLike */ ? getGlobalESSymbolType(/*reportErrors*/ languageVersion >= 2 /* ES2015 */) :
+                                        t.flags & 67108864 /* NonPrimitive */ ? emptyObjectType :
+                                            t.flags & 4194304 /* Index */ ? keyofConstraintType :
+                                                t.flags & 2 /* Unknown */ && !strictNullChecks ? emptyObjectType :
+                                                    t;
+        }
+        function getReducedApparentType(type) {
+            // Since getApparentType may return a non-reduced union or intersection type, we need to perform
+            // type reduction both before and after obtaining the apparent type. For example, given a type parameter
+            // 'T extends A | B', the type 'T & X' becomes 'A & X | B & X' after obtaining the apparent type, and
+            // that type may need further reduction to remove empty intersections.
+            return getReducedType(getApparentType(getReducedType(type)));
+        }
+        function createUnionOrIntersectionProperty(containingType, name) {
+            var singleProp;
+            var propSet;
+            var indexTypes;
+            var isUnion = containingType.flags & 1048576 /* Union */;
+            // Flags we want to propagate to the result if they exist in all source symbols
+            var optionalFlag = isUnion ? 0 /* None */ : 16777216 /* Optional */;
+            var syntheticFlag = 4 /* SyntheticMethod */;
+            var checkFlags = 0;
+            for (var _i = 0, _a = containingType.types; _i < _a.length; _i++) {
+                var current = _a[_i];
+                var type = getApparentType(current);
+                if (!(type === errorType || type.flags & 131072 /* Never */)) {
+                    var prop = getPropertyOfType(type, name);
+                    var modifiers = prop ? ts.getDeclarationModifierFlagsFromSymbol(prop) : 0;
+                    if (prop) {
+                        if (isUnion) {
+                            optionalFlag |= (prop.flags & 16777216 /* Optional */);
                         }
                         else {
-                            enumRelation.set(id, 2 /* Failed */);
+                            optionalFlag &= prop.flags;
+                        }
+                        if (!singleProp) {
+                            singleProp = prop;
+                        }
+                        else if (prop !== singleProp) {
+                            if (!propSet) {
+                                propSet = new ts.Map();
+                                propSet.set(getSymbolId(singleProp), singleProp);
+                            }
+                            var id = getSymbolId(prop);
+                            if (!propSet.has(id)) {
+                                propSet.set(id, prop);
+                            }
+                        }
+                        checkFlags |= (isReadonlySymbol(prop) ? 8 /* Readonly */ : 0) |
+                            (!(modifiers & 24 /* NonPublicAccessibilityModifier */) ? 256 /* ContainsPublic */ : 0) |
+                            (modifiers & 16 /* Protected */ ? 512 /* ContainsProtected */ : 0) |
+                            (modifiers & 8 /* Private */ ? 1024 /* ContainsPrivate */ : 0) |
+                            (modifiers & 32 /* Static */ ? 2048 /* ContainsStatic */ : 0);
+                        if (!isPrototypeProperty(prop)) {
+                            syntheticFlag = 2 /* SyntheticProperty */;
+                        }
+                    }
+                    else if (isUnion) {
+                        var indexInfo = !isLateBoundName(name) && (isNumericLiteralName(name) && getIndexInfoOfType(type, 1 /* Number */) || getIndexInfoOfType(type, 0 /* String */));
+                        if (indexInfo) {
+                            checkFlags |= 32 /* WritePartial */ | (indexInfo.isReadonly ? 8 /* Readonly */ : 0);
+                            indexTypes = ts.append(indexTypes, isTupleType(type) ? getRestTypeOfTupleType(type) || undefinedType : indexInfo.type);
+                        }
+                        else if (isObjectLiteralType(type)) {
+                            checkFlags |= 32 /* WritePartial */;
+                            indexTypes = ts.append(indexTypes, undefinedType);
+                        }
+                        else {
+                            checkFlags |= 16 /* ReadPartial */;
                         }
-                        return false;
                     }
                 }
             }
-            enumRelation.set(id, 1 /* Succeeded */);
-            return true;
-        }
-        function isSimpleTypeRelatedTo(source, target, relation, errorReporter) {
-            var s = source.flags;
-            var t = target.flags;
-            if (t & 3 /* AnyOrUnknown */ || s & 131072 /* Never */ || source === wildcardType)
-                return true;
-            if (t & 131072 /* Never */)
-                return false;
-            if (s & 132 /* StringLike */ && t & 4 /* String */)
-                return true;
-            if (s & 128 /* StringLiteral */ && s & 1024 /* EnumLiteral */ &&
-                t & 128 /* StringLiteral */ && !(t & 1024 /* EnumLiteral */) &&
-                source.value === target.value)
-                return true;
-            if (s & 296 /* NumberLike */ && t & 8 /* Number */)
-                return true;
-            if (s & 256 /* NumberLiteral */ && s & 1024 /* EnumLiteral */ &&
-                t & 256 /* NumberLiteral */ && !(t & 1024 /* EnumLiteral */) &&
-                source.value === target.value)
-                return true;
-            if (s & 2112 /* BigIntLike */ && t & 64 /* BigInt */)
-                return true;
-            if (s & 528 /* BooleanLike */ && t & 16 /* Boolean */)
-                return true;
-            if (s & 12288 /* ESSymbolLike */ && t & 4096 /* ESSymbol */)
-                return true;
-            if (s & 32 /* Enum */ && t & 32 /* Enum */ && isEnumTypeRelatedTo(source.symbol, target.symbol, errorReporter))
-                return true;
-            if (s & 1024 /* EnumLiteral */ && t & 1024 /* EnumLiteral */) {
-                if (s & 1048576 /* Union */ && t & 1048576 /* Union */ && isEnumTypeRelatedTo(source.symbol, target.symbol, errorReporter))
-                    return true;
-                if (s & 2944 /* Literal */ && t & 2944 /* Literal */ &&
-                    source.value === target.value &&
-                    isEnumTypeRelatedTo(getParentOfSymbol(source.symbol), getParentOfSymbol(target.symbol), errorReporter))
-                    return true;
+            if (!singleProp || isUnion && (propSet || checkFlags & 48 /* Partial */) && checkFlags & (1024 /* ContainsPrivate */ | 512 /* ContainsProtected */)) {
+                // No property was found, or, in a union, a property has a private or protected declaration in one
+                // constituent, but is missing or has a different declaration in another constituent.
+                return undefined;
             }
-            if (s & 32768 /* Undefined */ && (!strictNullChecks || t & (32768 /* Undefined */ | 16384 /* Void */)))
-                return true;
-            if (s & 65536 /* Null */ && (!strictNullChecks || t & 65536 /* Null */))
-                return true;
-            if (s & 524288 /* Object */ && t & 67108864 /* NonPrimitive */)
-                return true;
-            if (relation === assignableRelation || relation === comparableRelation) {
-                if (s & 1 /* Any */)
-                    return true;
-                // Type number or any numeric literal type is assignable to any numeric enum type or any
-                // numeric enum literal type. This rule exists for backwards compatibility reasons because
-                // bit-flag enum types sometimes look like literal enum types with numeric literal values.
-                if (s & (8 /* Number */ | 256 /* NumberLiteral */) && !(s & 1024 /* EnumLiteral */) && (t & 32 /* Enum */ || t & 256 /* NumberLiteral */ && t & 1024 /* EnumLiteral */))
-                    return true;
+            if (!propSet && !(checkFlags & 16 /* ReadPartial */) && !indexTypes) {
+                return singleProp;
             }
-            return false;
-        }
-        function isTypeRelatedTo(source, target, relation) {
-            if (isFreshLiteralType(source)) {
-                source = source.regularType;
+            var props = propSet ? ts.arrayFrom(propSet.values()) : [singleProp];
+            var declarations;
+            var firstType;
+            var nameType;
+            var propTypes = [];
+            var firstValueDeclaration;
+            var hasNonUniformValueDeclaration = false;
+            for (var _b = 0, props_1 = props; _b < props_1.length; _b++) {
+                var prop = props_1[_b];
+                if (!firstValueDeclaration) {
+                    firstValueDeclaration = prop.valueDeclaration;
+                }
+                else if (prop.valueDeclaration && prop.valueDeclaration !== firstValueDeclaration) {
+                    hasNonUniformValueDeclaration = true;
+                }
+                declarations = ts.addRange(declarations, prop.declarations);
+                var type = getTypeOfSymbol(prop);
+                if (!firstType) {
+                    firstType = type;
+                    nameType = getSymbolLinks(prop).nameType;
+                }
+                else if (type !== firstType) {
+                    checkFlags |= 64 /* HasNonUniformType */;
+                }
+                if (isLiteralType(type)) {
+                    checkFlags |= 128 /* HasLiteralType */;
+                }
+                if (type.flags & 131072 /* Never */) {
+                    checkFlags |= 131072 /* HasNeverType */;
+                }
+                propTypes.push(type);
             }
-            if (isFreshLiteralType(target)) {
-                target = target.regularType;
+            ts.addRange(propTypes, indexTypes);
+            var result = createSymbol(4 /* Property */ | optionalFlag, name, syntheticFlag | checkFlags);
+            result.containingType = containingType;
+            if (!hasNonUniformValueDeclaration && firstValueDeclaration) {
+                result.valueDeclaration = firstValueDeclaration;
+                // Inherit information about parent type.
+                if (firstValueDeclaration.symbol.parent) {
+                    result.parent = firstValueDeclaration.symbol.parent;
+                }
             }
-            if (source === target) {
-                return true;
+            result.declarations = declarations;
+            result.nameType = nameType;
+            if (propTypes.length > 2) {
+                // When `propTypes` has the potential to explode in size when normalized, defer normalization until absolutely needed
+                result.checkFlags |= 65536 /* DeferredType */;
+                result.deferralParent = containingType;
+                result.deferralConstituents = propTypes;
             }
-            if (relation !== identityRelation) {
-                if (relation === comparableRelation && !(target.flags & 131072 /* Never */) && isSimpleTypeRelatedTo(target, source, relation) || isSimpleTypeRelatedTo(source, target, relation)) {
-                    return true;
+            else {
+                result.type = isUnion ? getUnionType(propTypes) : getIntersectionType(propTypes);
+            }
+            return result;
+        }
+        // Return the symbol for a given property in a union or intersection type, or undefined if the property
+        // does not exist in any constituent type. Note that the returned property may only be present in some
+        // constituents, in which case the isPartial flag is set when the containing type is union type. We need
+        // these partial properties when identifying discriminant properties, but otherwise they are filtered out
+        // and do not appear to be present in the union type.
+        function getUnionOrIntersectionProperty(type, name) {
+            var properties = type.propertyCache || (type.propertyCache = ts.createSymbolTable());
+            var property = properties.get(name);
+            if (!property) {
+                property = createUnionOrIntersectionProperty(type, name);
+                if (property) {
+                    properties.set(name, property);
                 }
             }
-            else {
-                if (!(source.flags & 3145728 /* UnionOrIntersection */) && !(target.flags & 3145728 /* UnionOrIntersection */) &&
-                    source.flags !== target.flags && !(source.flags & 66584576 /* Substructure */))
-                    return false;
+            return property;
+        }
+        function getPropertyOfUnionOrIntersectionType(type, name) {
+            var property = getUnionOrIntersectionProperty(type, name);
+            // We need to filter out partial properties in union types
+            return property && !(ts.getCheckFlags(property) & 16 /* ReadPartial */) ? property : undefined;
+        }
+        /**
+         * Return the reduced form of the given type. For a union type, it is a union of the normalized constituent types.
+         * For an intersection of types containing one or more mututally exclusive discriminant properties, it is 'never'.
+         * For all other types, it is simply the type itself. Discriminant properties are considered mutually exclusive when
+         * no constituent property has type 'never', but the intersection of the constituent property types is 'never'.
+         */
+        function getReducedType(type) {
+            if (type.flags & 1048576 /* Union */ && type.objectFlags & 268435456 /* ContainsIntersections */) {
+                return type.resolvedReducedType || (type.resolvedReducedType = getReducedUnionType(type));
             }
-            if (source.flags & 524288 /* Object */ && target.flags & 524288 /* Object */) {
-                var related = relation.get(getRelationKey(source, target, 0 /* None */, relation));
-                if (related !== undefined) {
-                    return !!(related & 1 /* Succeeded */);
+            else if (type.flags & 2097152 /* Intersection */) {
+                if (!(type.objectFlags & 268435456 /* IsNeverIntersectionComputed */)) {
+                    type.objectFlags |= 268435456 /* IsNeverIntersectionComputed */ |
+                        (ts.some(getPropertiesOfUnionOrIntersectionType(type), isNeverReducedProperty) ? 536870912 /* IsNeverIntersection */ : 0);
                 }
+                return type.objectFlags & 536870912 /* IsNeverIntersection */ ? neverType : type;
             }
-            if (source.flags & 66846720 /* StructuredOrInstantiable */ || target.flags & 66846720 /* StructuredOrInstantiable */) {
-                return checkTypeRelatedTo(source, target, relation, /*errorNode*/ undefined);
+            return type;
+        }
+        function getReducedUnionType(unionType) {
+            var reducedTypes = ts.sameMap(unionType.types, getReducedType);
+            if (reducedTypes === unionType.types) {
+                return unionType;
             }
-            return false;
+            var reduced = getUnionType(reducedTypes);
+            if (reduced.flags & 1048576 /* Union */) {
+                reduced.resolvedReducedType = reduced;
+            }
+            return reduced;
         }
-        function isIgnoredJsxProperty(source, sourceProp) {
-            return ts.getObjectFlags(source) & 4096 /* JsxAttributes */ && !isUnhyphenatedJsxName(sourceProp.escapedName);
+        function isNeverReducedProperty(prop) {
+            return isDiscriminantWithNeverType(prop) || isConflictingPrivateProperty(prop);
         }
-        function getNormalizedType(type, writing) {
-            while (true) {
-                var t = isFreshLiteralType(type) ? type.regularType :
-                    ts.getObjectFlags(type) & 4 /* Reference */ && type.node ? createTypeReference(type.target, getTypeArguments(type)) :
-                        type.flags & 3145728 /* UnionOrIntersection */ ? getReducedType(type) :
-                            type.flags & 33554432 /* Substitution */ ? writing ? type.baseType : type.substitute :
-                                type.flags & 25165824 /* Simplifiable */ ? getSimplifiedType(type, writing) :
-                                    type;
-                if (t === type)
-                    break;
-                type = t;
+        function isDiscriminantWithNeverType(prop) {
+            // Return true for a synthetic non-optional property with non-uniform types, where at least one is
+            // a literal type and none is never, that reduces to never.
+            return !(prop.flags & 16777216 /* Optional */) &&
+                (ts.getCheckFlags(prop) & (192 /* Discriminant */ | 131072 /* HasNeverType */)) === 192 /* Discriminant */ &&
+                !!(getTypeOfSymbol(prop).flags & 131072 /* Never */);
+        }
+        function isConflictingPrivateProperty(prop) {
+            // Return true for a synthetic property with multiple declarations, at least one of which is private.
+            return !prop.valueDeclaration && !!(ts.getCheckFlags(prop) & 1024 /* ContainsPrivate */);
+        }
+        function elaborateNeverIntersection(errorInfo, type) {
+            if (ts.getObjectFlags(type) & 536870912 /* IsNeverIntersection */) {
+                var neverProp = ts.find(getPropertiesOfUnionOrIntersectionType(type), isDiscriminantWithNeverType);
+                if (neverProp) {
+                    return ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.The_intersection_0_was_reduced_to_never_because_property_1_has_conflicting_types_in_some_constituents, typeToString(type, /*enclosingDeclaration*/ undefined, 536870912 /* NoTypeReduction */), symbolToString(neverProp));
+                }
+                var privateProp = ts.find(getPropertiesOfUnionOrIntersectionType(type), isConflictingPrivateProperty);
+                if (privateProp) {
+                    return ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.The_intersection_0_was_reduced_to_never_because_property_1_exists_in_multiple_constituents_and_is_private_in_some, typeToString(type, /*enclosingDeclaration*/ undefined, 536870912 /* NoTypeReduction */), symbolToString(privateProp));
+                }
             }
-            return type;
+            return errorInfo;
         }
         /**
-         * Checks if 'source' is related to 'target' (e.g.: is a assignable to).
-         * @param source The left-hand-side of the relation.
-         * @param target The right-hand-side of the relation.
-         * @param relation The relation considered. One of 'identityRelation', 'subtypeRelation', 'assignableRelation', or 'comparableRelation'.
-         * Used as both to determine which checks are performed and as a cache of previously computed results.
-         * @param errorNode The suggested node upon which all errors will be reported, if defined. This may or may not be the actual node used.
-         * @param headMessage If the error chain should be prepended by a head message, then headMessage will be used.
-         * @param containingMessageChain A chain of errors to prepend any new errors found.
-         * @param errorOutputContainer Return the diagnostic. Do not log if 'skipLogging' is truthy.
+         * Return the symbol for the property with the given name in the given type. Creates synthetic union properties when
+         * necessary, maps primitive types and type parameters are to their apparent types, and augments with properties from
+         * Object and Function as appropriate.
+         *
+         * @param type a type to look up property from
+         * @param name a name of property to look up in a given type
          */
-        function checkTypeRelatedTo(source, target, relation, errorNode, headMessage, containingMessageChain, errorOutputContainer) {
-            var errorInfo;
-            var relatedInfo;
-            var maybeKeys;
-            var sourceStack;
-            var targetStack;
-            var maybeCount = 0;
-            var depth = 0;
-            var expandingFlags = 0 /* None */;
-            var overflow = false;
-            var overrideNextErrorInfo = 0; // How many `reportRelationError` calls should be skipped in the elaboration pyramid
-            var lastSkippedInfo;
-            var incompatibleStack = [];
-            var inPropertyCheck = false;
-            ts.Debug.assert(relation !== identityRelation || !errorNode, "no error reporting in identity checking");
-            var result = isRelatedTo(source, target, /*reportErrors*/ !!errorNode, headMessage);
-            if (incompatibleStack.length) {
-                reportIncompatibleStack();
-            }
-            if (overflow) {
-                var diag = error(errorNode || currentNode, ts.Diagnostics.Excessive_stack_depth_comparing_types_0_and_1, typeToString(source), typeToString(target));
-                if (errorOutputContainer) {
-                    (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
+        function getPropertyOfType(type, name) {
+            type = getReducedApparentType(type);
+            if (type.flags & 524288 /* Object */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                var symbol = resolved.members.get(name);
+                if (symbol && symbolIsValue(symbol)) {
+                    return symbol;
                 }
-            }
-            else if (errorInfo) {
-                if (containingMessageChain) {
-                    var chain = containingMessageChain();
-                    if (chain) {
-                        ts.concatenateDiagnosticMessageChains(chain, errorInfo);
-                        errorInfo = chain;
+                var functionType = resolved === anyFunctionType ? globalFunctionType :
+                    resolved.callSignatures.length ? globalCallableFunctionType :
+                        resolved.constructSignatures.length ? globalNewableFunctionType :
+                            undefined;
+                if (functionType) {
+                    var symbol_1 = getPropertyOfObjectType(functionType, name);
+                    if (symbol_1) {
+                        return symbol_1;
                     }
                 }
-                var relatedInformation = void 0;
-                // Check if we should issue an extra diagnostic to produce a quickfix for a slightly incorrect import statement
-                if (headMessage && errorNode && !result && source.symbol) {
-                    var links = getSymbolLinks(source.symbol);
-                    if (links.originatingImport && !ts.isImportCall(links.originatingImport)) {
-                        var helpfulRetry = checkTypeRelatedTo(getTypeOfSymbol(links.target), target, relation, /*errorNode*/ undefined);
-                        if (helpfulRetry) {
-                            // Likely an incorrect import. Issue a helpful diagnostic to produce a quickfix to change the import
-                            var diag_1 = ts.createDiagnosticForNode(links.originatingImport, ts.Diagnostics.Type_originates_at_this_import_A_namespace_style_import_cannot_be_called_or_constructed_and_will_cause_a_failure_at_runtime_Consider_using_a_default_import_or_import_require_here_instead);
-                            relatedInformation = ts.append(relatedInformation, diag_1); // Cause the error to appear with the error that triggered it
-                        }
+                return getPropertyOfObjectType(globalObjectType, name);
+            }
+            if (type.flags & 3145728 /* UnionOrIntersection */) {
+                return getPropertyOfUnionOrIntersectionType(type, name);
+            }
+            return undefined;
+        }
+        function getSignaturesOfStructuredType(type, kind) {
+            if (type.flags & 3670016 /* StructuredType */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                return kind === 0 /* Call */ ? resolved.callSignatures : resolved.constructSignatures;
+            }
+            return ts.emptyArray;
+        }
+        /**
+         * Return the signatures of the given kind in the given type. Creates synthetic union signatures when necessary and
+         * maps primitive types and type parameters are to their apparent types.
+         */
+        function getSignaturesOfType(type, kind) {
+            return getSignaturesOfStructuredType(getReducedApparentType(type), kind);
+        }
+        function getIndexInfoOfStructuredType(type, kind) {
+            if (type.flags & 3670016 /* StructuredType */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                return kind === 0 /* String */ ? resolved.stringIndexInfo : resolved.numberIndexInfo;
+            }
+        }
+        function getIndexTypeOfStructuredType(type, kind) {
+            var info = getIndexInfoOfStructuredType(type, kind);
+            return info && info.type;
+        }
+        // Return the indexing info of the given kind in the given type. Creates synthetic union index types when necessary and
+        // maps primitive types and type parameters are to their apparent types.
+        function getIndexInfoOfType(type, kind) {
+            return getIndexInfoOfStructuredType(getReducedApparentType(type), kind);
+        }
+        // Return the index type of the given kind in the given type. Creates synthetic union index types when necessary and
+        // maps primitive types and type parameters are to their apparent types.
+        function getIndexTypeOfType(type, kind) {
+            return getIndexTypeOfStructuredType(getReducedApparentType(type), kind);
+        }
+        function getImplicitIndexTypeOfType(type, kind) {
+            if (isObjectTypeWithInferableIndex(type)) {
+                var propTypes = [];
+                for (var _i = 0, _a = getPropertiesOfType(type); _i < _a.length; _i++) {
+                    var prop = _a[_i];
+                    if (kind === 0 /* String */ || isNumericLiteralName(prop.escapedName)) {
+                        propTypes.push(getTypeOfSymbol(prop));
                     }
                 }
-                var diag = ts.createDiagnosticForNodeFromMessageChain(errorNode, errorInfo, relatedInformation);
-                if (relatedInfo) {
-                    ts.addRelatedInfo.apply(void 0, __spreadArrays([diag], relatedInfo));
+                if (kind === 0 /* String */) {
+                    ts.append(propTypes, getIndexTypeOfType(type, 1 /* Number */));
                 }
-                if (errorOutputContainer) {
-                    (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
+                if (propTypes.length) {
+                    return getUnionType(propTypes);
                 }
-                if (!errorOutputContainer || !errorOutputContainer.skipLogging) {
-                    diagnostics.add(diag);
+            }
+            return undefined;
+        }
+        // Return list of type parameters with duplicates removed (duplicate identifier errors are generated in the actual
+        // type checking functions).
+        function getTypeParametersFromDeclaration(declaration) {
+            var result;
+            for (var _i = 0, _a = ts.getEffectiveTypeParameterDeclarations(declaration); _i < _a.length; _i++) {
+                var node = _a[_i];
+                result = ts.appendIfUnique(result, getDeclaredTypeOfTypeParameter(node.symbol));
+            }
+            return result;
+        }
+        function symbolsToArray(symbols) {
+            var result = [];
+            symbols.forEach(function (symbol, id) {
+                if (!isReservedMemberName(id)) {
+                    result.push(symbol);
                 }
+            });
+            return result;
+        }
+        function isJSDocOptionalParameter(node) {
+            return ts.isInJSFile(node) && (
+            // node.type should only be a JSDocOptionalType when node is a parameter of a JSDocFunctionType
+            node.type && node.type.kind === 307 /* JSDocOptionalType */
+                || ts.getJSDocParameterTags(node).some(function (_a) {
+                    var isBracketed = _a.isBracketed, typeExpression = _a.typeExpression;
+                    return isBracketed || !!typeExpression && typeExpression.type.kind === 307 /* JSDocOptionalType */;
+                }));
+        }
+        function tryFindAmbientModule(moduleName, withAugmentations) {
+            if (ts.isExternalModuleNameRelative(moduleName)) {
+                return undefined;
             }
-            if (errorNode && errorOutputContainer && errorOutputContainer.skipLogging && result === 0 /* False */) {
-                ts.Debug.assert(!!errorOutputContainer.errors, "missed opportunity to interact with error.");
+            var symbol = getSymbol(globals, '"' + moduleName + '"', 512 /* ValueModule */);
+            // merged symbol is module declaration symbol combined with all augmentations
+            return symbol && withAugmentations ? getMergedSymbol(symbol) : symbol;
+        }
+        function isOptionalParameter(node) {
+            if (ts.hasQuestionToken(node) || isOptionalJSDocPropertyLikeTag(node) || isJSDocOptionalParameter(node)) {
+                return true;
             }
-            return result !== 0 /* False */;
-            function resetErrorInfo(saved) {
-                errorInfo = saved.errorInfo;
-                lastSkippedInfo = saved.lastSkippedInfo;
-                incompatibleStack = saved.incompatibleStack;
-                overrideNextErrorInfo = saved.overrideNextErrorInfo;
-                relatedInfo = saved.relatedInfo;
+            if (node.initializer) {
+                var signature = getSignatureFromDeclaration(node.parent);
+                var parameterIndex = node.parent.parameters.indexOf(node);
+                ts.Debug.assert(parameterIndex >= 0);
+                // Only consider syntactic or instantiated parameters as optional, not `void` parameters as this function is used
+                // in grammar checks and checking for `void` too early results in parameter types widening too early
+                // and causes some noImplicitAny errors to be lost.
+                return parameterIndex >= getMinArgumentCount(signature, 1 /* StrongArityForUntypedJS */ | 2 /* VoidIsNonOptional */);
             }
-            function captureErrorCalculationState() {
-                return {
-                    errorInfo: errorInfo,
-                    lastSkippedInfo: lastSkippedInfo,
-                    incompatibleStack: incompatibleStack.slice(),
-                    overrideNextErrorInfo: overrideNextErrorInfo,
-                    relatedInfo: !relatedInfo ? undefined : relatedInfo.slice()
-                };
+            var iife = ts.getImmediatelyInvokedFunctionExpression(node.parent);
+            if (iife) {
+                return !node.type &&
+                    !node.dotDotDotToken &&
+                    node.parent.parameters.indexOf(node) >= iife.arguments.length;
             }
-            function reportIncompatibleError(message, arg0, arg1, arg2, arg3) {
-                overrideNextErrorInfo++; // Suppress the next relation error
-                lastSkippedInfo = undefined; // Reset skipped info cache
-                incompatibleStack.push([message, arg0, arg1, arg2, arg3]);
+            return false;
+        }
+        function isOptionalJSDocPropertyLikeTag(node) {
+            if (!ts.isJSDocPropertyLikeTag(node)) {
+                return false;
             }
-            function reportIncompatibleStack() {
-                var stack = incompatibleStack;
-                incompatibleStack = [];
-                var info = lastSkippedInfo;
-                lastSkippedInfo = undefined;
-                if (stack.length === 1) {
-                    reportError.apply(void 0, stack[0]);
-                    if (info) {
-                        // Actually do the last relation error
-                        reportRelationError.apply(void 0, __spreadArrays([/*headMessage*/ undefined], info));
-                    }
-                    return;
-                }
-                // The first error will be the innermost, while the last will be the outermost - so by popping off the end,
-                // we can build from left to right
-                var path = "";
-                var secondaryRootErrors = [];
-                while (stack.length) {
-                    var _a = stack.pop(), msg = _a[0], args = _a.slice(1);
-                    switch (msg.code) {
-                        case ts.Diagnostics.Types_of_property_0_are_incompatible.code: {
-                            // Parenthesize a `new` if there is one
-                            if (path.indexOf("new ") === 0) {
-                                path = "(" + path + ")";
-                            }
-                            var str = "" + args[0];
-                            // If leading, just print back the arg (irrespective of if it's a valid identifier)
-                            if (path.length === 0) {
-                                path = "" + str;
-                            }
-                            // Otherwise write a dotted name if possible
-                            else if (ts.isIdentifierText(str, compilerOptions.target)) {
-                                path = path + "." + str;
-                            }
-                            // Failing that, check if the name is already a computed name
-                            else if (str[0] === "[" && str[str.length - 1] === "]") {
-                                path = "" + path + str;
-                            }
-                            // And finally write out a computed name as a last resort
-                            else {
-                                path = path + "[" + str + "]";
-                            }
-                            break;
-                        }
-                        case ts.Diagnostics.Call_signature_return_types_0_and_1_are_incompatible.code:
-                        case ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible.code:
-                        case ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code:
-                        case ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code: {
-                            if (path.length === 0) {
-                                // Don't flatten signature compatability errors at the start of a chain - instead prefer
-                                // to unify (the with no arguments bit is excessive for printback) and print them back
-                                var mappedMsg = msg;
-                                if (msg.code === ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code) {
-                                    mappedMsg = ts.Diagnostics.Call_signature_return_types_0_and_1_are_incompatible;
-                                }
-                                else if (msg.code === ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code) {
-                                    mappedMsg = ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible;
-                                }
-                                secondaryRootErrors.unshift([mappedMsg, args[0], args[1]]);
-                            }
-                            else {
-                                var prefix = (msg.code === ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible.code ||
-                                    msg.code === ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code)
-                                    ? "new "
-                                    : "";
-                                var params = (msg.code === ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code ||
-                                    msg.code === ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code)
-                                    ? ""
-                                    : "...";
-                                path = "" + prefix + path + "(" + params + ")";
-                            }
-                            break;
-                        }
-                        default:
-                            return ts.Debug.fail("Unhandled Diagnostic: " + msg.code);
+            var isBracketed = node.isBracketed, typeExpression = node.typeExpression;
+            return isBracketed || !!typeExpression && typeExpression.type.kind === 307 /* JSDocOptionalType */;
+        }
+        function createTypePredicate(kind, parameterName, parameterIndex, type) {
+            return { kind: kind, parameterName: parameterName, parameterIndex: parameterIndex, type: type };
+        }
+        /**
+         * Gets the minimum number of type arguments needed to satisfy all non-optional type
+         * parameters.
+         */
+        function getMinTypeArgumentCount(typeParameters) {
+            var minTypeArgumentCount = 0;
+            if (typeParameters) {
+                for (var i = 0; i < typeParameters.length; i++) {
+                    if (!hasTypeParameterDefault(typeParameters[i])) {
+                        minTypeArgumentCount = i + 1;
                     }
                 }
-                if (path) {
-                    reportError(path[path.length - 1] === ")"
-                        ? ts.Diagnostics.The_types_returned_by_0_are_incompatible_between_these_types
-                        : ts.Diagnostics.The_types_of_0_are_incompatible_between_these_types, path);
-                }
-                else {
-                    // Remove the innermost secondary error as it will duplicate the error already reported by `reportRelationError` on entry
-                    secondaryRootErrors.shift();
-                }
-                for (var _i = 0, secondaryRootErrors_1 = secondaryRootErrors; _i < secondaryRootErrors_1.length; _i++) {
-                    var _b = secondaryRootErrors_1[_i], msg = _b[0], args = _b.slice(1);
-                    var originalValue = msg.elidedInCompatabilityPyramid;
-                    msg.elidedInCompatabilityPyramid = false; // Teporarily override elision to ensure error is reported
-                    reportError.apply(void 0, __spreadArrays([msg], args));
-                    msg.elidedInCompatabilityPyramid = originalValue;
-                }
-                if (info) {
-                    // Actually do the last relation error
-                    reportRelationError.apply(void 0, __spreadArrays([/*headMessage*/ undefined], info));
-                }
             }
-            function reportError(message, arg0, arg1, arg2, arg3) {
-                ts.Debug.assert(!!errorNode);
-                if (incompatibleStack.length)
-                    reportIncompatibleStack();
-                if (message.elidedInCompatabilityPyramid)
-                    return;
-                errorInfo = ts.chainDiagnosticMessages(errorInfo, message, arg0, arg1, arg2, arg3);
+            return minTypeArgumentCount;
+        }
+        function fillMissingTypeArguments(typeArguments, typeParameters, minTypeArgumentCount, isJavaScriptImplicitAny) {
+            var numTypeParameters = ts.length(typeParameters);
+            if (!numTypeParameters) {
+                return [];
             }
-            function associateRelatedInfo(info) {
-                ts.Debug.assert(!!errorInfo);
-                if (!relatedInfo) {
-                    relatedInfo = [info];
+            var numTypeArguments = ts.length(typeArguments);
+            if (isJavaScriptImplicitAny || (numTypeArguments >= minTypeArgumentCount && numTypeArguments <= numTypeParameters)) {
+                var result = typeArguments ? typeArguments.slice() : [];
+                // Map invalid forward references in default types to the error type
+                for (var i = numTypeArguments; i < numTypeParameters; i++) {
+                    result[i] = errorType;
                 }
-                else {
-                    relatedInfo.push(info);
+                var baseDefaultType = getDefaultTypeArgumentType(isJavaScriptImplicitAny);
+                for (var i = numTypeArguments; i < numTypeParameters; i++) {
+                    var defaultType = getDefaultFromTypeParameter(typeParameters[i]);
+                    if (isJavaScriptImplicitAny && defaultType && (isTypeIdenticalTo(defaultType, unknownType) || isTypeIdenticalTo(defaultType, emptyObjectType))) {
+                        defaultType = anyType;
+                    }
+                    result[i] = defaultType ? instantiateType(defaultType, createTypeMapper(typeParameters, result)) : baseDefaultType;
                 }
+                result.length = typeParameters.length;
+                return result;
             }
-            function reportRelationError(message, source, target) {
-                if (incompatibleStack.length)
-                    reportIncompatibleStack();
-                var _a = getTypeNamesForErrorDisplay(source, target), sourceType = _a[0], targetType = _a[1];
-                if (target.flags & 262144 /* TypeParameter */) {
-                    var constraint = getBaseConstraintOfType(target);
-                    var constraintElab = constraint && isTypeAssignableTo(source, constraint);
-                    if (constraintElab) {
-                        reportError(ts.Diagnostics._0_is_assignable_to_the_constraint_of_type_1_but_1_could_be_instantiated_with_a_different_subtype_of_constraint_2, sourceType, targetType, typeToString(constraint));
-                    }
-                    else {
-                        reportError(ts.Diagnostics._0_could_be_instantiated_with_an_arbitrary_type_which_could_be_unrelated_to_1, targetType, sourceType);
-                    }
+            return typeArguments && typeArguments.slice();
+        }
+        function getSignatureFromDeclaration(declaration) {
+            var links = getNodeLinks(declaration);
+            if (!links.resolvedSignature) {
+                var parameters = [];
+                var flags = 0 /* None */;
+                var minArgumentCount = 0;
+                var thisParameter = void 0;
+                var hasThisParameter = false;
+                var iife = ts.getImmediatelyInvokedFunctionExpression(declaration);
+                var isJSConstructSignature = ts.isJSDocConstructSignature(declaration);
+                var isUntypedSignatureInJSFile = !iife &&
+                    ts.isInJSFile(declaration) &&
+                    ts.isValueSignatureDeclaration(declaration) &&
+                    !ts.hasJSDocParameterTags(declaration) &&
+                    !ts.getJSDocType(declaration);
+                if (isUntypedSignatureInJSFile) {
+                    flags |= 16 /* IsUntypedSignatureInJSFile */;
                 }
-                if (!message) {
-                    if (relation === comparableRelation) {
-                        message = ts.Diagnostics.Type_0_is_not_comparable_to_type_1;
+                // If this is a JSDoc construct signature, then skip the first parameter in the
+                // parameter list.  The first parameter represents the return type of the construct
+                // signature.
+                for (var i = isJSConstructSignature ? 1 : 0; i < declaration.parameters.length; i++) {
+                    var param = declaration.parameters[i];
+                    var paramSymbol = param.symbol;
+                    var type = ts.isJSDocParameterTag(param) ? (param.typeExpression && param.typeExpression.type) : param.type;
+                    // Include parameter symbol instead of property symbol in the signature
+                    if (paramSymbol && !!(paramSymbol.flags & 4 /* Property */) && !ts.isBindingPattern(param.name)) {
+                        var resolvedSymbol = resolveName(param, paramSymbol.escapedName, 111551 /* Value */, undefined, undefined, /*isUse*/ false);
+                        paramSymbol = resolvedSymbol;
                     }
-                    else if (sourceType === targetType) {
-                        message = ts.Diagnostics.Type_0_is_not_assignable_to_type_1_Two_different_types_with_this_name_exist_but_they_are_unrelated;
+                    if (i === 0 && paramSymbol.escapedName === "this" /* This */) {
+                        hasThisParameter = true;
+                        thisParameter = param.symbol;
                     }
                     else {
-                        message = ts.Diagnostics.Type_0_is_not_assignable_to_type_1;
+                        parameters.push(paramSymbol);
                     }
-                }
-                reportError(message, sourceType, targetType);
-            }
-            function tryElaborateErrorsForPrimitivesAndObjects(source, target) {
-                var sourceType = symbolValueDeclarationIsContextSensitive(source.symbol) ? typeToString(source, source.symbol.valueDeclaration) : typeToString(source);
-                var targetType = symbolValueDeclarationIsContextSensitive(target.symbol) ? typeToString(target, target.symbol.valueDeclaration) : typeToString(target);
-                if ((globalStringType === source && stringType === target) ||
-                    (globalNumberType === source && numberType === target) ||
-                    (globalBooleanType === source && booleanType === target) ||
-                    (getGlobalESSymbolType(/*reportErrors*/ false) === source && esSymbolType === target)) {
-                    reportError(ts.Diagnostics._0_is_a_primitive_but_1_is_a_wrapper_object_Prefer_using_0_when_possible, targetType, sourceType);
-                }
-            }
-            /**
-             * Try and elaborate array and tuple errors. Returns false
-             * if we have found an elaboration, or we should ignore
-             * any other elaborations when relating the `source` and
-             * `target` types.
-             */
-            function tryElaborateArrayLikeErrors(source, target, reportErrors) {
-                /**
-                 * The spec for elaboration is:
-                 * - If the source is a readonly tuple and the target is a mutable array or tuple, elaborate on mutability and skip property elaborations.
-                 * - If the source is a tuple then skip property elaborations if the target is an array or tuple.
-                 * - If the source is a readonly array and the target is a mutable array or tuple, elaborate on mutability and skip property elaborations.
-                 * - If the source an array then skip property elaborations if the target is a tuple.
-                 */
-                if (isTupleType(source)) {
-                    if (source.target.readonly && isMutableArrayOrTuple(target)) {
-                        if (reportErrors) {
-                            reportError(ts.Diagnostics.The_type_0_is_readonly_and_cannot_be_assigned_to_the_mutable_type_1, typeToString(source), typeToString(target));
-                        }
-                        return false;
+                    if (type && type.kind === 191 /* LiteralType */) {
+                        flags |= 2 /* HasLiteralTypes */;
+                    }
+                    // Record a new minimum argument count if this is not an optional parameter
+                    var isOptionalParameter_1 = isOptionalJSDocPropertyLikeTag(param) ||
+                        param.initializer || param.questionToken || param.dotDotDotToken ||
+                        iife && parameters.length > iife.arguments.length && !type ||
+                        isJSDocOptionalParameter(param);
+                    if (!isOptionalParameter_1) {
+                        minArgumentCount = parameters.length;
                     }
-                    return isTupleType(target) || isArrayType(target);
                 }
-                if (isReadonlyArrayType(source) && isMutableArrayOrTuple(target)) {
-                    if (reportErrors) {
-                        reportError(ts.Diagnostics.The_type_0_is_readonly_and_cannot_be_assigned_to_the_mutable_type_1, typeToString(source), typeToString(target));
+                // If only one accessor includes a this-type annotation, the other behaves as if it had the same type annotation
+                if ((declaration.kind === 167 /* GetAccessor */ || declaration.kind === 168 /* SetAccessor */) &&
+                    !hasNonBindableDynamicName(declaration) &&
+                    (!hasThisParameter || !thisParameter)) {
+                    var otherKind = declaration.kind === 167 /* GetAccessor */ ? 168 /* SetAccessor */ : 167 /* GetAccessor */;
+                    var other = ts.getDeclarationOfKind(getSymbolOfNode(declaration), otherKind);
+                    if (other) {
+                        thisParameter = getAnnotatedAccessorThisParameter(other);
                     }
-                    return false;
                 }
-                if (isTupleType(target)) {
-                    return isArrayType(source);
+                var classType = declaration.kind === 166 /* Constructor */ ?
+                    getDeclaredTypeOfClassOrInterface(getMergedSymbol(declaration.parent.symbol))
+                    : undefined;
+                var typeParameters = classType ? classType.localTypeParameters : getTypeParametersFromDeclaration(declaration);
+                if (ts.hasRestParameter(declaration) || ts.isInJSFile(declaration) && maybeAddJsSyntheticRestParameter(declaration, parameters)) {
+                    flags |= 1 /* HasRestParameter */;
                 }
-                return true;
+                links.resolvedSignature = createSignature(declaration, typeParameters, thisParameter, parameters, 
+                /*resolvedReturnType*/ undefined, /*resolvedTypePredicate*/ undefined, minArgumentCount, flags);
             }
-            /**
-             * Compare two types and return
-             * * Ternary.True if they are related with no assumptions,
-             * * Ternary.Maybe if they are related with assumptions of other relationships, or
-             * * Ternary.False if they are not related.
-             */
-            function isRelatedTo(originalSource, originalTarget, reportErrors, headMessage, intersectionState) {
-                if (reportErrors === void 0) { reportErrors = false; }
-                if (intersectionState === void 0) { intersectionState = 0 /* None */; }
-                // Before normalization: if `source` is type an object type, and `target` is primitive,
-                // skip all the checks we don't need and just return `isSimpleTypeRelatedTo` result
-                if (originalSource.flags & 524288 /* Object */ && originalTarget.flags & 131068 /* Primitive */) {
-                    if (isSimpleTypeRelatedTo(originalSource, originalTarget, relation, reportErrors ? reportError : undefined)) {
-                        return -1 /* True */;
-                    }
-                    reportErrorResults(originalSource, originalTarget, 0 /* False */, !!(ts.getObjectFlags(originalSource) & 4096 /* JsxAttributes */));
-                    return 0 /* False */;
+            return links.resolvedSignature;
+        }
+        /**
+         * A JS function gets a synthetic rest parameter if it references `arguments` AND:
+         * 1. It has no parameters but at least one `@param` with a type that starts with `...`
+         * OR
+         * 2. It has at least one parameter, and the last parameter has a matching `@param` with a type that starts with `...`
+         */
+        function maybeAddJsSyntheticRestParameter(declaration, parameters) {
+            if (ts.isJSDocSignature(declaration) || !containsArgumentsReference(declaration)) {
+                return false;
+            }
+            var lastParam = ts.lastOrUndefined(declaration.parameters);
+            var lastParamTags = lastParam ? ts.getJSDocParameterTags(lastParam) : ts.getJSDocTags(declaration).filter(ts.isJSDocParameterTag);
+            var lastParamVariadicType = ts.firstDefined(lastParamTags, function (p) {
+                return p.typeExpression && ts.isJSDocVariadicType(p.typeExpression.type) ? p.typeExpression.type : undefined;
+            });
+            var syntheticArgsSymbol = createSymbol(3 /* Variable */, "args", 32768 /* RestParameter */);
+            syntheticArgsSymbol.type = lastParamVariadicType ? createArrayType(getTypeFromTypeNode(lastParamVariadicType.type)) : anyArrayType;
+            if (lastParamVariadicType) {
+                // Replace the last parameter with a rest parameter.
+                parameters.pop();
+            }
+            parameters.push(syntheticArgsSymbol);
+            return true;
+        }
+        function getSignatureOfTypeTag(node) {
+            // should be attached to a function declaration or expression
+            if (!(ts.isInJSFile(node) && ts.isFunctionLikeDeclaration(node)))
+                return undefined;
+            var typeTag = ts.getJSDocTypeTag(node);
+            var signature = typeTag && typeTag.typeExpression && getSingleCallSignature(getTypeFromTypeNode(typeTag.typeExpression));
+            return signature && getErasedSignature(signature);
+        }
+        function getReturnTypeOfTypeTag(node) {
+            var signature = getSignatureOfTypeTag(node);
+            return signature && getReturnTypeOfSignature(signature);
+        }
+        function containsArgumentsReference(declaration) {
+            var links = getNodeLinks(declaration);
+            if (links.containsArgumentsReference === undefined) {
+                if (links.flags & 8192 /* CaptureArguments */) {
+                    links.containsArgumentsReference = true;
                 }
-                // Normalize the source and target types: Turn fresh literal types into regular literal types,
-                // turn deferred type references into regular type references, simplify indexed access and
-                // conditional types, and resolve substitution types to either the substitution (on the source
-                // side) or the type variable (on the target side).
-                var source = getNormalizedType(originalSource, /*writing*/ false);
-                var target = getNormalizedType(originalTarget, /*writing*/ true);
-                if (source === target)
-                    return -1 /* True */;
-                if (relation === identityRelation) {
-                    return isIdenticalTo(source, target);
+                else {
+                    links.containsArgumentsReference = traverse(declaration.body);
                 }
-                // We fastpath comparing a type parameter to exactly its constraint, as this is _super_ common,
-                // and otherwise, for type parameters in large unions, causes us to need to compare the union to itself,
-                // as we break down the _target_ union first, _then_ get the source constraint - so for every
-                // member of the target, we attempt to find a match in the source. This avoids that in cases where
-                // the target is exactly the constraint.
-                if (source.flags & 262144 /* TypeParameter */ && getConstraintOfType(source) === target) {
-                    return -1 /* True */;
+            }
+            return links.containsArgumentsReference;
+            function traverse(node) {
+                if (!node)
+                    return false;
+                switch (node.kind) {
+                    case 78 /* Identifier */:
+                        return node.escapedText === "arguments" && ts.isExpressionNode(node);
+                    case 163 /* PropertyDeclaration */:
+                    case 165 /* MethodDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        return node.name.kind === 158 /* ComputedPropertyName */
+                            && traverse(node.name);
+                    default:
+                        return !ts.nodeStartsNewLexicalEnvironment(node) && !ts.isPartOfTypeNode(node) && !!ts.forEachChild(node, traverse);
                 }
-                // Try to see if we're relating something like `Foo` -> `Bar | null | undefined`.
-                // If so, reporting the `null` and `undefined` in the type is hardly useful.
-                // First, see if we're even relating an object type to a union.
-                // Then see if the target is stripped down to a single non-union type.
-                // Note
-                //  * We actually want to remove null and undefined naively here (rather than using getNonNullableType),
-                //    since we don't want to end up with a worse error like "`Foo` is not assignable to `NonNullable<T>`"
-                //    when dealing with generics.
-                //  * We also don't deal with primitive source types, since we already halt elaboration below.
-                if (target.flags & 1048576 /* Union */ && source.flags & 524288 /* Object */ &&
-                    target.types.length <= 3 && maybeTypeOfKind(target, 98304 /* Nullable */)) {
-                    var nullStrippedTarget = extractTypesOfKind(target, ~98304 /* Nullable */);
-                    if (!(nullStrippedTarget.flags & (1048576 /* Union */ | 131072 /* Never */))) {
-                        if (source === nullStrippedTarget)
-                            return -1 /* True */;
-                        target = nullStrippedTarget;
+            }
+        }
+        function getSignaturesOfSymbol(symbol) {
+            if (!symbol)
+                return ts.emptyArray;
+            var result = [];
+            for (var i = 0; i < symbol.declarations.length; i++) {
+                var decl = symbol.declarations[i];
+                if (!ts.isFunctionLike(decl))
+                    continue;
+                // Don't include signature if node is the implementation of an overloaded function. A node is considered
+                // an implementation node if it has a body and the previous node is of the same kind and immediately
+                // precedes the implementation node (i.e. has the same parent and ends where the implementation starts).
+                if (i > 0 && decl.body) {
+                    var previous = symbol.declarations[i - 1];
+                    if (decl.parent === previous.parent && decl.kind === previous.kind && decl.pos === previous.end) {
+                        continue;
                     }
                 }
-                if (relation === comparableRelation && !(target.flags & 131072 /* Never */) && isSimpleTypeRelatedTo(target, source, relation) ||
-                    isSimpleTypeRelatedTo(source, target, relation, reportErrors ? reportError : undefined))
-                    return -1 /* True */;
-                var isComparingJsxAttributes = !!(ts.getObjectFlags(source) & 4096 /* JsxAttributes */);
-                var isPerformingExcessPropertyChecks = !(intersectionState & 2 /* Target */) && (isObjectLiteralType(source) && ts.getObjectFlags(source) & 32768 /* FreshLiteral */);
-                if (isPerformingExcessPropertyChecks) {
-                    if (hasExcessProperties(source, target, reportErrors)) {
-                        if (reportErrors) {
-                            reportRelationError(headMessage, source, target);
-                        }
-                        return 0 /* False */;
-                    }
+                result.push(getSignatureFromDeclaration(decl));
+            }
+            return result;
+        }
+        function resolveExternalModuleTypeByLiteral(name) {
+            var moduleSym = resolveExternalModuleName(name, name);
+            if (moduleSym) {
+                var resolvedModuleSymbol = resolveExternalModuleSymbol(moduleSym);
+                if (resolvedModuleSymbol) {
+                    return getTypeOfSymbol(resolvedModuleSymbol);
                 }
-                var isPerformingCommonPropertyChecks = relation !== comparableRelation && !(intersectionState & 2 /* Target */) &&
-                    source.flags & (131068 /* Primitive */ | 524288 /* Object */ | 2097152 /* Intersection */) && source !== globalObjectType &&
-                    target.flags & (524288 /* Object */ | 2097152 /* Intersection */) && isWeakType(target) &&
-                    (getPropertiesOfType(source).length > 0 || typeHasCallOrConstructSignatures(source));
-                if (isPerformingCommonPropertyChecks && !hasCommonProperties(source, target, isComparingJsxAttributes)) {
-                    if (reportErrors) {
-                        var calls = getSignaturesOfType(source, 0 /* Call */);
-                        var constructs = getSignaturesOfType(source, 1 /* Construct */);
-                        if (calls.length > 0 && isRelatedTo(getReturnTypeOfSignature(calls[0]), target, /*reportErrors*/ false) ||
-                            constructs.length > 0 && isRelatedTo(getReturnTypeOfSignature(constructs[0]), target, /*reportErrors*/ false)) {
-                            reportError(ts.Diagnostics.Value_of_type_0_has_no_properties_in_common_with_type_1_Did_you_mean_to_call_it, typeToString(source), typeToString(target));
-                        }
-                        else {
-                            reportError(ts.Diagnostics.Type_0_has_no_properties_in_common_with_type_1, typeToString(source), typeToString(target));
-                        }
-                    }
-                    return 0 /* False */;
+            }
+            return anyType;
+        }
+        function getThisTypeOfSignature(signature) {
+            if (signature.thisParameter) {
+                return getTypeOfSymbol(signature.thisParameter);
+            }
+        }
+        function getTypePredicateOfSignature(signature) {
+            if (!signature.resolvedTypePredicate) {
+                if (signature.target) {
+                    var targetTypePredicate = getTypePredicateOfSignature(signature.target);
+                    signature.resolvedTypePredicate = targetTypePredicate ? instantiateTypePredicate(targetTypePredicate, signature.mapper) : noTypePredicate;
                 }
-                var result = 0 /* False */;
-                var saveErrorInfo = captureErrorCalculationState();
-                // Note that these checks are specifically ordered to produce correct results. In particular,
-                // we need to deconstruct unions before intersections (because unions are always at the top),
-                // and we need to handle "each" relations before "some" relations for the same kind of type.
-                if (source.flags & 1048576 /* Union */) {
-                    result = relation === comparableRelation ?
-                        someTypeRelatedToType(source, target, reportErrors && !(source.flags & 131068 /* Primitive */), intersectionState) :
-                        eachTypeRelatedToType(source, target, reportErrors && !(source.flags & 131068 /* Primitive */), intersectionState);
+                else if (signature.unionSignatures) {
+                    signature.resolvedTypePredicate = getUnionTypePredicate(signature.unionSignatures) || noTypePredicate;
                 }
                 else {
-                    if (target.flags & 1048576 /* Union */) {
-                        result = typeRelatedToSomeType(getRegularTypeOfObjectLiteral(source), target, reportErrors && !(source.flags & 131068 /* Primitive */) && !(target.flags & 131068 /* Primitive */));
-                    }
-                    else if (target.flags & 2097152 /* Intersection */) {
-                        result = typeRelatedToEachType(getRegularTypeOfObjectLiteral(source), target, reportErrors, 2 /* Target */);
-                    }
-                    else if (source.flags & 2097152 /* Intersection */) {
-                        // Check to see if any constituents of the intersection are immediately related to the target.
-                        //
-                        // Don't report errors though. Checking whether a constituent is related to the source is not actually
-                        // useful and leads to some confusing error messages. Instead it is better to let the below checks
-                        // take care of this, or to not elaborate at all. For instance,
-                        //
-                        //    - For an object type (such as 'C = A & B'), users are usually more interested in structural errors.
-                        //
-                        //    - For a union type (such as '(A | B) = (C & D)'), it's better to hold onto the whole intersection
-                        //          than to report that 'D' is not assignable to 'A' or 'B'.
-                        //
-                        //    - For a primitive type or type parameter (such as 'number = A & B') there is no point in
-                        //          breaking the intersection apart.
-                        result = someTypeRelatedToType(source, target, /*reportErrors*/ false, 1 /* Source */);
-                    }
-                    if (!result && (source.flags & 66846720 /* StructuredOrInstantiable */ || target.flags & 66846720 /* StructuredOrInstantiable */)) {
-                        if (result = recursiveTypeRelatedTo(source, target, reportErrors, intersectionState)) {
-                            resetErrorInfo(saveErrorInfo);
+                    var type = signature.declaration && ts.getEffectiveReturnTypeNode(signature.declaration);
+                    var jsdocPredicate = void 0;
+                    if (!type && ts.isInJSFile(signature.declaration)) {
+                        var jsdocSignature = getSignatureOfTypeTag(signature.declaration);
+                        if (jsdocSignature && signature !== jsdocSignature) {
+                            jsdocPredicate = getTypePredicateOfSignature(jsdocSignature);
                         }
                     }
+                    signature.resolvedTypePredicate = type && ts.isTypePredicateNode(type) ?
+                        createTypePredicateFromTypePredicateNode(type, signature) :
+                        jsdocPredicate || noTypePredicate;
                 }
-                if (!result && source.flags & (2097152 /* Intersection */ | 262144 /* TypeParameter */)) {
-                    // The combined constraint of an intersection type is the intersection of the constraints of
-                    // the constituents. When an intersection type contains instantiable types with union type
-                    // constraints, there are situations where we need to examine the combined constraint. One is
-                    // when the target is a union type. Another is when the intersection contains types belonging
-                    // to one of the disjoint domains. For example, given type variables T and U, each with the
-                    // constraint 'string | number', the combined constraint of 'T & U' is 'string | number' and
-                    // we need to check this constraint against a union on the target side. Also, given a type
-                    // variable V constrained to 'string | number', 'V & number' has a combined constraint of
-                    // 'string & number | number & number' which reduces to just 'number'.
-                    // This also handles type parameters, as a type parameter with a union constraint compared against a union
-                    // needs to have its constraint hoisted into an intersection with said type parameter, this way
-                    // the type param can be compared with itself in the target (with the influence of its constraint to match other parts)
-                    // For example, if `T extends 1 | 2` and `U extends 2 | 3` and we compare `T & U` to `T & U & (1 | 2 | 3)`
-                    var constraint = getEffectiveConstraintOfIntersection(source.flags & 2097152 /* Intersection */ ? source.types : [source], !!(target.flags & 1048576 /* Union */));
-                    if (constraint && (source.flags & 2097152 /* Intersection */ || target.flags & 1048576 /* Union */)) {
-                        if (everyType(constraint, function (c) { return c !== source; })) { // Skip comparison if expansion contains the source itself
-                            // TODO: Stack errors so we get a pyramid for the "normal" comparison above, _and_ a second for this
-                            if (result = isRelatedTo(constraint, target, /*reportErrors*/ false, /*headMessage*/ undefined, intersectionState)) {
-                                resetErrorInfo(saveErrorInfo);
-                            }
-                        }
-                    }
+                ts.Debug.assert(!!signature.resolvedTypePredicate);
+            }
+            return signature.resolvedTypePredicate === noTypePredicate ? undefined : signature.resolvedTypePredicate;
+        }
+        function createTypePredicateFromTypePredicateNode(node, signature) {
+            var parameterName = node.parameterName;
+            var type = node.type && getTypeFromTypeNode(node.type);
+            return parameterName.kind === 187 /* ThisType */ ?
+                createTypePredicate(node.assertsModifier ? 2 /* AssertsThis */ : 0 /* This */, /*parameterName*/ undefined, /*parameterIndex*/ undefined, type) :
+                createTypePredicate(node.assertsModifier ? 3 /* AssertsIdentifier */ : 1 /* Identifier */, parameterName.escapedText, ts.findIndex(signature.parameters, function (p) { return p.escapedName === parameterName.escapedText; }), type);
+        }
+        function getReturnTypeOfSignature(signature) {
+            if (!signature.resolvedReturnType) {
+                if (!pushTypeResolution(signature, 3 /* ResolvedReturnType */)) {
+                    return errorType;
                 }
-                // For certain combinations involving intersections and optional, excess, or mismatched properties we need
-                // an extra property check where the intersection is viewed as a single object. The following are motivating
-                // examples that all should be errors, but aren't without this extra property check:
-                //
-                //   let obj: { a: { x: string } } & { c: number } = { a: { x: 'hello', y: 2 }, c: 5 };  // Nested excess property
-                //
-                //   declare let wrong: { a: { y: string } };
-                //   let weak: { a?: { x?: number } } & { c?: string } = wrong;  // Nested weak object type
-                //
-                //   function foo<T extends object>(x: { a?: string }, y: T & { a: boolean }) {
-                //     x = y;  // Mismatched property in source intersection
-                //   }
-                //
-                // We suppress recursive intersection property checks because they can generate lots of work when relating
-                // recursive intersections that are structurally similar but not exactly identical. See #37854.
-                if (result && !inPropertyCheck && (target.flags & 2097152 /* Intersection */ && (isPerformingExcessPropertyChecks || isPerformingCommonPropertyChecks) ||
-                    isNonGenericObjectType(target) && !isArrayType(target) && !isTupleType(target) && source.flags & 2097152 /* Intersection */ && getApparentType(source).flags & 3670016 /* StructuredType */ && !ts.some(source.types, function (t) { return !!(ts.getObjectFlags(t) & 2097152 /* NonInferrableType */); }))) {
-                    inPropertyCheck = true;
-                    result &= recursiveTypeRelatedTo(source, target, reportErrors, 4 /* PropertyCheck */);
-                    inPropertyCheck = false;
+                var type = signature.target ? instantiateType(getReturnTypeOfSignature(signature.target), signature.mapper) :
+                    signature.unionSignatures ? getUnionType(ts.map(signature.unionSignatures, getReturnTypeOfSignature), 2 /* Subtype */) :
+                        getReturnTypeFromAnnotation(signature.declaration) ||
+                            (ts.nodeIsMissing(signature.declaration.body) ? anyType : getReturnTypeFromBody(signature.declaration));
+                if (signature.flags & 4 /* IsInnerCallChain */) {
+                    type = addOptionalTypeMarker(type);
                 }
-                reportErrorResults(source, target, result, isComparingJsxAttributes);
-                return result;
-                function reportErrorResults(source, target, result, isComparingJsxAttributes) {
-                    if (!result && reportErrors) {
-                        source = originalSource.aliasSymbol ? originalSource : source;
-                        target = originalTarget.aliasSymbol ? originalTarget : target;
-                        var maybeSuppress = overrideNextErrorInfo > 0;
-                        if (maybeSuppress) {
-                            overrideNextErrorInfo--;
+                else if (signature.flags & 8 /* IsOuterCallChain */) {
+                    type = getOptionalType(type);
+                }
+                if (!popTypeResolution()) {
+                    if (signature.declaration) {
+                        var typeNode = ts.getEffectiveReturnTypeNode(signature.declaration);
+                        if (typeNode) {
+                            error(typeNode, ts.Diagnostics.Return_type_annotation_circularly_references_itself);
                         }
-                        if (source.flags & 524288 /* Object */ && target.flags & 524288 /* Object */) {
-                            var currentError = errorInfo;
-                            tryElaborateArrayLikeErrors(source, target, reportErrors);
-                            if (errorInfo !== currentError) {
-                                maybeSuppress = !!errorInfo;
+                        else if (noImplicitAny) {
+                            var declaration = signature.declaration;
+                            var name = ts.getNameOfDeclaration(declaration);
+                            if (name) {
+                                error(name, ts.Diagnostics._0_implicitly_has_return_type_any_because_it_does_not_have_a_return_type_annotation_and_is_referenced_directly_or_indirectly_in_one_of_its_return_expressions, ts.declarationNameToString(name));
                             }
-                        }
-                        if (source.flags & 524288 /* Object */ && target.flags & 131068 /* Primitive */) {
-                            tryElaborateErrorsForPrimitivesAndObjects(source, target);
-                        }
-                        else if (source.symbol && source.flags & 524288 /* Object */ && globalObjectType === source) {
-                            reportError(ts.Diagnostics.The_Object_type_is_assignable_to_very_few_other_types_Did_you_mean_to_use_the_any_type_instead);
-                        }
-                        else if (isComparingJsxAttributes && target.flags & 2097152 /* Intersection */) {
-                            var targetTypes = target.types;
-                            var intrinsicAttributes = getJsxType(JsxNames.IntrinsicAttributes, errorNode);
-                            var intrinsicClassAttributes = getJsxType(JsxNames.IntrinsicClassAttributes, errorNode);
-                            if (intrinsicAttributes !== errorType && intrinsicClassAttributes !== errorType &&
-                                (ts.contains(targetTypes, intrinsicAttributes) || ts.contains(targetTypes, intrinsicClassAttributes))) {
-                                // do not report top error
-                                return result;
+                            else {
+                                error(declaration, ts.Diagnostics.Function_implicitly_has_return_type_any_because_it_does_not_have_a_return_type_annotation_and_is_referenced_directly_or_indirectly_in_one_of_its_return_expressions);
                             }
                         }
-                        else {
-                            errorInfo = elaborateNeverIntersection(errorInfo, originalTarget);
-                        }
-                        if (!headMessage && maybeSuppress) {
-                            lastSkippedInfo = [source, target];
-                            // Used by, eg, missing property checking to replace the top-level message with a more informative one
-                            return result;
-                        }
-                        reportRelationError(headMessage, source, target);
                     }
+                    type = anyType;
                 }
+                signature.resolvedReturnType = type;
             }
-            function isIdenticalTo(source, target) {
-                var flags = source.flags & target.flags;
-                if (!(flags & 66584576 /* Substructure */)) {
-                    return 0 /* False */;
-                }
-                if (flags & 3145728 /* UnionOrIntersection */) {
-                    var result_5 = eachTypeRelatedToSomeType(source, target);
-                    if (result_5) {
-                        result_5 &= eachTypeRelatedToSomeType(target, source);
-                    }
-                    return result_5;
-                }
-                return recursiveTypeRelatedTo(source, target, /*reportErrors*/ false, 0 /* None */);
+            return signature.resolvedReturnType;
+        }
+        function getReturnTypeFromAnnotation(declaration) {
+            if (declaration.kind === 166 /* Constructor */) {
+                return getDeclaredTypeOfClassOrInterface(getMergedSymbol(declaration.parent.symbol));
             }
-            function getTypeOfPropertyInTypes(types, name) {
-                var appendPropType = function (propTypes, type) {
-                    type = getApparentType(type);
-                    var prop = type.flags & 3145728 /* UnionOrIntersection */ ? getPropertyOfUnionOrIntersectionType(type, name) : getPropertyOfObjectType(type, name);
-                    var propType = prop && getTypeOfSymbol(prop) || isNumericLiteralName(name) && getIndexTypeOfType(type, 1 /* Number */) || getIndexTypeOfType(type, 0 /* String */) || undefinedType;
-                    return ts.append(propTypes, propType);
-                };
-                return getUnionType(ts.reduceLeft(types, appendPropType, /*initial*/ undefined) || ts.emptyArray);
+            if (ts.isJSDocConstructSignature(declaration)) {
+                return getTypeFromTypeNode(declaration.parameters[0].type); // TODO: GH#18217
             }
-            function hasExcessProperties(source, target, reportErrors) {
-                if (!isExcessPropertyCheckTarget(target) || !noImplicitAny && ts.getObjectFlags(target) & 16384 /* JSLiteral */) {
-                    return false; // Disable excess property checks on JS literals to simulate having an implicit "index signature" - but only outside of noImplicitAny
-                }
-                var isComparingJsxAttributes = !!(ts.getObjectFlags(source) & 4096 /* JsxAttributes */);
-                if ((relation === assignableRelation || relation === comparableRelation) &&
-                    (isTypeSubsetOf(globalObjectType, target) || (!isComparingJsxAttributes && isEmptyObjectType(target)))) {
-                    return false;
-                }
-                var reducedTarget = target;
-                var checkTypes;
-                if (target.flags & 1048576 /* Union */) {
-                    reducedTarget = findMatchingDiscriminantType(source, target, isRelatedTo) || filterPrimitivesIfContainsNonPrimitive(target);
-                    checkTypes = reducedTarget.flags & 1048576 /* Union */ ? reducedTarget.types : [reducedTarget];
+            var typeNode = ts.getEffectiveReturnTypeNode(declaration);
+            if (typeNode) {
+                return getTypeFromTypeNode(typeNode);
+            }
+            if (declaration.kind === 167 /* GetAccessor */ && !hasNonBindableDynamicName(declaration)) {
+                var jsDocType = ts.isInJSFile(declaration) && getTypeForDeclarationFromJSDocComment(declaration);
+                if (jsDocType) {
+                    return jsDocType;
                 }
-                var _loop_13 = function (prop) {
-                    if (shouldCheckAsExcessProperty(prop, source.symbol) && !isIgnoredJsxProperty(source, prop)) {
-                        if (!isKnownProperty(reducedTarget, prop.escapedName, isComparingJsxAttributes)) {
-                            if (reportErrors) {
-                                // Report error in terms of object types in the target as those are the only ones
-                                // we check in isKnownProperty.
-                                var errorTarget = filterType(reducedTarget, isExcessPropertyCheckTarget);
-                                // We know *exactly* where things went wrong when comparing the types.
-                                // Use this property as the error node as this will be more helpful in
-                                // reasoning about what went wrong.
-                                if (!errorNode)
-                                    return { value: ts.Debug.fail() };
-                                if (ts.isJsxAttributes(errorNode) || ts.isJsxOpeningLikeElement(errorNode) || ts.isJsxOpeningLikeElement(errorNode.parent)) {
-                                    // JsxAttributes has an object-literal flag and undergo same type-assignablity check as normal object-literal.
-                                    // However, using an object-literal error message will be very confusing to the users so we give different a message.
-                                    // TODO: Spelling suggestions for excess jsx attributes (needs new diagnostic messages)
-                                    if (prop.valueDeclaration && ts.isJsxAttribute(prop.valueDeclaration) && ts.getSourceFileOfNode(errorNode) === ts.getSourceFileOfNode(prop.valueDeclaration.name)) {
-                                        // Note that extraneous children (as in `<NoChild>extra</NoChild>`) don't pass this check,
-                                        // since `children` is a SyntaxKind.PropertySignature instead of a SyntaxKind.JsxAttribute.
-                                        errorNode = prop.valueDeclaration.name;
-                                    }
-                                    reportError(ts.Diagnostics.Property_0_does_not_exist_on_type_1, symbolToString(prop), typeToString(errorTarget));
-                                }
-                                else {
-                                    // use the property's value declaration if the property is assigned inside the literal itself
-                                    var objectLiteralDeclaration_1 = source.symbol && ts.firstOrUndefined(source.symbol.declarations);
-                                    var suggestion = void 0;
-                                    if (prop.valueDeclaration && ts.findAncestor(prop.valueDeclaration, function (d) { return d === objectLiteralDeclaration_1; }) && ts.getSourceFileOfNode(objectLiteralDeclaration_1) === ts.getSourceFileOfNode(errorNode)) {
-                                        var propDeclaration = prop.valueDeclaration;
-                                        ts.Debug.assertNode(propDeclaration, ts.isObjectLiteralElementLike);
-                                        errorNode = propDeclaration;
-                                        var name = propDeclaration.name;
-                                        if (ts.isIdentifier(name)) {
-                                            suggestion = getSuggestionForNonexistentProperty(name, errorTarget);
-                                        }
-                                    }
-                                    if (suggestion !== undefined) {
-                                        reportError(ts.Diagnostics.Object_literal_may_only_specify_known_properties_but_0_does_not_exist_in_type_1_Did_you_mean_to_write_2, symbolToString(prop), typeToString(errorTarget), suggestion);
-                                    }
-                                    else {
-                                        reportError(ts.Diagnostics.Object_literal_may_only_specify_known_properties_and_0_does_not_exist_in_type_1, symbolToString(prop), typeToString(errorTarget));
-                                    }
-                                }
-                            }
-                            return { value: true };
-                        }
-                        if (checkTypes && !isRelatedTo(getTypeOfSymbol(prop), getTypeOfPropertyInTypes(checkTypes, prop.escapedName), reportErrors)) {
-                            if (reportErrors) {
-                                reportIncompatibleError(ts.Diagnostics.Types_of_property_0_are_incompatible, symbolToString(prop));
-                            }
-                            return { value: true };
-                        }
-                    }
-                };
-                for (var _i = 0, _a = getPropertiesOfType(source); _i < _a.length; _i++) {
-                    var prop = _a[_i];
-                    var state_5 = _loop_13(prop);
-                    if (typeof state_5 === "object")
-                        return state_5.value;
+                var setter = ts.getDeclarationOfKind(getSymbolOfNode(declaration), 168 /* SetAccessor */);
+                var setterType = getAnnotatedAccessorType(setter);
+                if (setterType) {
+                    return setterType;
                 }
-                return false;
             }
-            function shouldCheckAsExcessProperty(prop, container) {
-                return prop.valueDeclaration && container.valueDeclaration && prop.valueDeclaration.parent === container.valueDeclaration;
+            return getReturnTypeOfTypeTag(declaration);
+        }
+        function isResolvingReturnTypeOfSignature(signature) {
+            return !signature.resolvedReturnType && findResolutionCycleStartIndex(signature, 3 /* ResolvedReturnType */) >= 0;
+        }
+        function getRestTypeOfSignature(signature) {
+            return tryGetRestTypeOfSignature(signature) || anyType;
+        }
+        function tryGetRestTypeOfSignature(signature) {
+            if (signatureHasRestParameter(signature)) {
+                var sigRestType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
+                var restType = isTupleType(sigRestType) ? getRestTypeOfTupleType(sigRestType) : sigRestType;
+                return restType && getIndexTypeOfType(restType, 1 /* Number */);
             }
-            function eachTypeRelatedToSomeType(source, target) {
-                var result = -1 /* True */;
-                var sourceTypes = source.types;
-                for (var _i = 0, sourceTypes_1 = sourceTypes; _i < sourceTypes_1.length; _i++) {
-                    var sourceType = sourceTypes_1[_i];
-                    var related = typeRelatedToSomeType(sourceType, target, /*reportErrors*/ false);
-                    if (!related) {
-                        return 0 /* False */;
-                    }
-                    result &= related;
+            return undefined;
+        }
+        function getSignatureInstantiation(signature, typeArguments, isJavascript, inferredTypeParameters) {
+            var instantiatedSignature = getSignatureInstantiationWithoutFillingInTypeArguments(signature, fillMissingTypeArguments(typeArguments, signature.typeParameters, getMinTypeArgumentCount(signature.typeParameters), isJavascript));
+            if (inferredTypeParameters) {
+                var returnSignature = getSingleCallOrConstructSignature(getReturnTypeOfSignature(instantiatedSignature));
+                if (returnSignature) {
+                    var newReturnSignature = cloneSignature(returnSignature);
+                    newReturnSignature.typeParameters = inferredTypeParameters;
+                    var newInstantiatedSignature = cloneSignature(instantiatedSignature);
+                    newInstantiatedSignature.resolvedReturnType = getOrCreateTypeFromSignature(newReturnSignature);
+                    return newInstantiatedSignature;
                 }
-                return result;
             }
-            function typeRelatedToSomeType(source, target, reportErrors) {
-                var targetTypes = target.types;
-                if (target.flags & 1048576 /* Union */ && containsType(targetTypes, source)) {
-                    return -1 /* True */;
-                }
-                for (var _i = 0, targetTypes_1 = targetTypes; _i < targetTypes_1.length; _i++) {
-                    var type = targetTypes_1[_i];
-                    var related = isRelatedTo(source, type, /*reportErrors*/ false);
-                    if (related) {
-                        return related;
-                    }
-                }
-                if (reportErrors) {
-                    var bestMatchingType = getBestMatchingType(source, target, isRelatedTo);
-                    isRelatedTo(source, bestMatchingType || targetTypes[targetTypes.length - 1], /*reportErrors*/ true);
-                }
-                return 0 /* False */;
+            return instantiatedSignature;
+        }
+        function getSignatureInstantiationWithoutFillingInTypeArguments(signature, typeArguments) {
+            var instantiations = signature.instantiations || (signature.instantiations = new ts.Map());
+            var id = getTypeListId(typeArguments);
+            var instantiation = instantiations.get(id);
+            if (!instantiation) {
+                instantiations.set(id, instantiation = createSignatureInstantiation(signature, typeArguments));
             }
-            function typeRelatedToEachType(source, target, reportErrors, intersectionState) {
-                var result = -1 /* True */;
-                var targetTypes = target.types;
-                for (var _i = 0, targetTypes_2 = targetTypes; _i < targetTypes_2.length; _i++) {
-                    var targetType = targetTypes_2[_i];
-                    var related = isRelatedTo(source, targetType, reportErrors, /*headMessage*/ undefined, intersectionState);
-                    if (!related) {
-                        return 0 /* False */;
-                    }
-                    result &= related;
-                }
-                return result;
+            return instantiation;
+        }
+        function createSignatureInstantiation(signature, typeArguments) {
+            return instantiateSignature(signature, createSignatureTypeMapper(signature, typeArguments), /*eraseTypeParameters*/ true);
+        }
+        function createSignatureTypeMapper(signature, typeArguments) {
+            return createTypeMapper(signature.typeParameters, typeArguments);
+        }
+        function getErasedSignature(signature) {
+            return signature.typeParameters ?
+                signature.erasedSignatureCache || (signature.erasedSignatureCache = createErasedSignature(signature)) :
+                signature;
+        }
+        function createErasedSignature(signature) {
+            // Create an instantiation of the signature where all type arguments are the any type.
+            return instantiateSignature(signature, createTypeEraser(signature.typeParameters), /*eraseTypeParameters*/ true);
+        }
+        function getCanonicalSignature(signature) {
+            return signature.typeParameters ?
+                signature.canonicalSignatureCache || (signature.canonicalSignatureCache = createCanonicalSignature(signature)) :
+                signature;
+        }
+        function createCanonicalSignature(signature) {
+            // Create an instantiation of the signature where each unconstrained type parameter is replaced with
+            // its original. When a generic class or interface is instantiated, each generic method in the class or
+            // interface is instantiated with a fresh set of cloned type parameters (which we need to handle scenarios
+            // where different generations of the same type parameter are in scope). This leads to a lot of new type
+            // identities, and potentially a lot of work comparing those identities, so here we create an instantiation
+            // that uses the original type identities for all unconstrained type parameters.
+            return getSignatureInstantiation(signature, ts.map(signature.typeParameters, function (tp) { return tp.target && !getConstraintOfTypeParameter(tp.target) ? tp.target : tp; }), ts.isInJSFile(signature.declaration));
+        }
+        function getBaseSignature(signature) {
+            var typeParameters = signature.typeParameters;
+            if (typeParameters) {
+                var typeEraser_1 = createTypeEraser(typeParameters);
+                var baseConstraints = ts.map(typeParameters, function (tp) { return instantiateType(getBaseConstraintOfType(tp), typeEraser_1) || unknownType; });
+                return instantiateSignature(signature, createTypeMapper(typeParameters, baseConstraints), /*eraseTypeParameters*/ true);
             }
-            function someTypeRelatedToType(source, target, reportErrors, intersectionState) {
-                var sourceTypes = source.types;
-                if (source.flags & 1048576 /* Union */ && containsType(sourceTypes, target)) {
-                    return -1 /* True */;
-                }
-                var len = sourceTypes.length;
-                for (var i = 0; i < len; i++) {
-                    var related = isRelatedTo(sourceTypes[i], target, reportErrors && i === len - 1, /*headMessage*/ undefined, intersectionState);
-                    if (related) {
-                        return related;
-                    }
-                }
-                return 0 /* False */;
+            return signature;
+        }
+        function getOrCreateTypeFromSignature(signature) {
+            // There are two ways to declare a construct signature, one is by declaring a class constructor
+            // using the constructor keyword, and the other is declaring a bare construct signature in an
+            // object type literal or interface (using the new keyword). Each way of declaring a constructor
+            // will result in a different declaration kind.
+            if (!signature.isolatedSignatureType) {
+                var kind = signature.declaration ? signature.declaration.kind : 0 /* Unknown */;
+                var isConstructor = kind === 166 /* Constructor */ || kind === 170 /* ConstructSignature */ || kind === 175 /* ConstructorType */;
+                var type = createObjectType(16 /* Anonymous */);
+                type.members = emptySymbols;
+                type.properties = ts.emptyArray;
+                type.callSignatures = !isConstructor ? [signature] : ts.emptyArray;
+                type.constructSignatures = isConstructor ? [signature] : ts.emptyArray;
+                signature.isolatedSignatureType = type;
             }
-            function eachTypeRelatedToType(source, target, reportErrors, intersectionState) {
-                var result = -1 /* True */;
-                var sourceTypes = source.types;
-                for (var i = 0; i < sourceTypes.length; i++) {
-                    var sourceType = sourceTypes[i];
-                    if (target.flags & 1048576 /* Union */ && target.types.length === sourceTypes.length) {
-                        // many unions are mappings of one another; in such cases, simply comparing members at the same index can shortcut the comparison
-                        var related_1 = isRelatedTo(sourceType, target.types[i], /*reportErrors*/ false, /*headMessage*/ undefined, intersectionState);
-                        if (related_1) {
-                            result &= related_1;
-                            continue;
+            return signature.isolatedSignatureType;
+        }
+        function getIndexSymbol(symbol) {
+            return symbol.members.get("__index" /* Index */);
+        }
+        function getIndexDeclarationOfSymbol(symbol, kind) {
+            var syntaxKind = kind === 1 /* Number */ ? 144 /* NumberKeyword */ : 147 /* StringKeyword */;
+            var indexSymbol = getIndexSymbol(symbol);
+            if (indexSymbol) {
+                for (var _i = 0, _a = indexSymbol.declarations; _i < _a.length; _i++) {
+                    var decl = _a[_i];
+                    var node = ts.cast(decl, ts.isIndexSignatureDeclaration);
+                    if (node.parameters.length === 1) {
+                        var parameter = node.parameters[0];
+                        if (parameter.type && parameter.type.kind === syntaxKind) {
+                            return node;
                         }
                     }
-                    var related = isRelatedTo(sourceType, target, reportErrors, /*headMessage*/ undefined, intersectionState);
-                    if (!related) {
-                        return 0 /* False */;
-                    }
-                    result &= related;
                 }
-                return result;
             }
-            function typeArgumentsRelatedTo(sources, targets, variances, reportErrors, intersectionState) {
-                if (sources === void 0) { sources = ts.emptyArray; }
-                if (targets === void 0) { targets = ts.emptyArray; }
-                if (variances === void 0) { variances = ts.emptyArray; }
-                if (sources.length !== targets.length && relation === identityRelation) {
-                    return 0 /* False */;
-                }
-                var length = sources.length <= targets.length ? sources.length : targets.length;
-                var result = -1 /* True */;
-                for (var i = 0; i < length; i++) {
-                    // When variance information isn't available we default to covariance. This happens
-                    // in the process of computing variance information for recursive types and when
-                    // comparing 'this' type arguments.
-                    var varianceFlags = i < variances.length ? variances[i] : 1 /* Covariant */;
-                    var variance = varianceFlags & 7 /* VarianceMask */;
-                    // We ignore arguments for independent type parameters (because they're never witnessed).
-                    if (variance !== 4 /* Independent */) {
-                        var s = sources[i];
-                        var t = targets[i];
-                        var related = -1 /* True */;
-                        if (varianceFlags & 8 /* Unmeasurable */) {
-                            // Even an `Unmeasurable` variance works out without a structural check if the source and target are _identical_.
-                            // We can't simply assume invariance, because `Unmeasurable` marks nonlinear relations, for example, a relation tained by
-                            // the `-?` modifier in a mapped type (where, no matter how the inputs are related, the outputs still might not be)
-                            related = relation === identityRelation ? isRelatedTo(s, t, /*reportErrors*/ false) : compareTypesIdentical(s, t);
-                        }
-                        else if (variance === 1 /* Covariant */) {
-                            related = isRelatedTo(s, t, reportErrors, /*headMessage*/ undefined, intersectionState);
-                        }
-                        else if (variance === 2 /* Contravariant */) {
-                            related = isRelatedTo(t, s, reportErrors, /*headMessage*/ undefined, intersectionState);
-                        }
-                        else if (variance === 3 /* Bivariant */) {
-                            // In the bivariant case we first compare contravariantly without reporting
-                            // errors. Then, if that doesn't succeed, we compare covariantly with error
-                            // reporting. Thus, error elaboration will be based on the the covariant check,
-                            // which is generally easier to reason about.
-                            related = isRelatedTo(t, s, /*reportErrors*/ false);
-                            if (!related) {
-                                related = isRelatedTo(s, t, reportErrors, /*headMessage*/ undefined, intersectionState);
+            return undefined;
+        }
+        function createIndexInfo(type, isReadonly, declaration) {
+            return { type: type, isReadonly: isReadonly, declaration: declaration };
+        }
+        function getIndexInfoOfSymbol(symbol, kind) {
+            var declaration = getIndexDeclarationOfSymbol(symbol, kind);
+            if (declaration) {
+                return createIndexInfo(declaration.type ? getTypeFromTypeNode(declaration.type) : anyType, ts.hasEffectiveModifier(declaration, 64 /* Readonly */), declaration);
+            }
+            return undefined;
+        }
+        function getConstraintDeclaration(type) {
+            return ts.mapDefined(ts.filter(type.symbol && type.symbol.declarations, ts.isTypeParameterDeclaration), ts.getEffectiveConstraintOfTypeParameter)[0];
+        }
+        function getInferredTypeParameterConstraint(typeParameter) {
+            var inferences;
+            if (typeParameter.symbol) {
+                for (var _i = 0, _a = typeParameter.symbol.declarations; _i < _a.length; _i++) {
+                    var declaration = _a[_i];
+                    if (declaration.parent.kind === 185 /* InferType */) {
+                        // When an 'infer T' declaration is immediately contained in a type reference node
+                        // (such as 'Foo<infer T>'), T's constraint is inferred from the constraint of the
+                        // corresponding type parameter in 'Foo'. When multiple 'infer T' declarations are
+                        // present, we form an intersection of the inferred constraint types.
+                        var _b = ts.walkUpParenthesizedTypesAndGetParentAndChild(declaration.parent.parent), _c = _b[0], childTypeParameter = _c === void 0 ? declaration.parent : _c, grandParent = _b[1];
+                        if (grandParent.kind === 173 /* TypeReference */) {
+                            var typeReference = grandParent;
+                            var typeParameters = getTypeParametersForTypeReference(typeReference);
+                            if (typeParameters) {
+                                var index = typeReference.typeArguments.indexOf(childTypeParameter);
+                                if (index < typeParameters.length) {
+                                    var declaredConstraint = getConstraintOfTypeParameter(typeParameters[index]);
+                                    if (declaredConstraint) {
+                                        // Type parameter constraints can reference other type parameters so
+                                        // constraints need to be instantiated. If instantiation produces the
+                                        // type parameter itself, we discard that inference. For example, in
+                                        //   type Foo<T extends string, U extends T> = [T, U];
+                                        //   type Bar<T> = T extends Foo<infer X, infer X> ? Foo<X, X> : T;
+                                        // the instantiated constraint for U is X, so we discard that inference.
+                                        var mapper = createTypeMapper(typeParameters, getEffectiveTypeArguments(typeReference, typeParameters));
+                                        var constraint = instantiateType(declaredConstraint, mapper);
+                                        if (constraint !== typeParameter) {
+                                            inferences = ts.append(inferences, constraint);
+                                        }
+                                    }
+                                }
                             }
                         }
-                        else {
-                            // In the invariant case we first compare covariantly, and only when that
-                            // succeeds do we proceed to compare contravariantly. Thus, error elaboration
-                            // will typically be based on the covariant check.
-                            related = isRelatedTo(s, t, reportErrors, /*headMessage*/ undefined, intersectionState);
-                            if (related) {
-                                related &= isRelatedTo(t, s, reportErrors, /*headMessage*/ undefined, intersectionState);
-                            }
+                        // When an 'infer T' declaration is immediately contained in a rest parameter declaration, a rest type
+                        // or a named rest tuple element, we infer an 'unknown[]' constraint.
+                        else if (grandParent.kind === 160 /* Parameter */ && grandParent.dotDotDotToken ||
+                            grandParent.kind === 181 /* RestType */ ||
+                            grandParent.kind === 192 /* NamedTupleMember */ && grandParent.dotDotDotToken) {
+                            inferences = ts.append(inferences, createArrayType(unknownType));
                         }
-                        if (!related) {
-                            return 0 /* False */;
+                        // When an 'infer T' declaration is immediately contained in a string template type, we infer a 'string'
+                        // constraint.
+                        else if (grandParent.kind === 194 /* TemplateLiteralTypeSpan */) {
+                            inferences = ts.append(inferences, stringType);
                         }
-                        result &= related;
                     }
                 }
-                return result;
             }
-            // Determine if possibly recursive types are related. First, check if the result is already available in the global cache.
-            // Second, check if we have already started a comparison of the given two types in which case we assume the result to be true.
-            // Third, check if both types are part of deeply nested chains of generic type instantiations and if so assume the types are
-            // equal and infinitely expanding. Fourth, if we have reached a depth of 100 nested comparisons, assume we have runaway recursion
-            // and issue an error. Otherwise, actually compare the structure of the two types.
-            function recursiveTypeRelatedTo(source, target, reportErrors, intersectionState) {
-                if (overflow) {
-                    return 0 /* False */;
+            return inferences && getIntersectionType(inferences);
+        }
+        /** This is a worker function. Use getConstraintOfTypeParameter which guards against circular constraints. */
+        function getConstraintFromTypeParameter(typeParameter) {
+            if (!typeParameter.constraint) {
+                if (typeParameter.target) {
+                    var targetConstraint = getConstraintOfTypeParameter(typeParameter.target);
+                    typeParameter.constraint = targetConstraint ? instantiateType(targetConstraint, typeParameter.mapper) : noConstraintType;
                 }
-                var id = getRelationKey(source, target, intersectionState | (inPropertyCheck ? 8 /* InPropertyCheck */ : 0), relation);
-                var entry = relation.get(id);
-                if (entry !== undefined) {
-                    if (reportErrors && entry & 2 /* Failed */ && !(entry & 4 /* Reported */)) {
-                        // We are elaborating errors and the cached result is an unreported failure. The result will be reported
-                        // as a failure, and should be updated as a reported failure by the bottom of this function.
+                else {
+                    var constraintDeclaration = getConstraintDeclaration(typeParameter);
+                    if (!constraintDeclaration) {
+                        typeParameter.constraint = getInferredTypeParameterConstraint(typeParameter) || noConstraintType;
                     }
                     else {
-                        if (outofbandVarianceMarkerHandler) {
-                            // We're in the middle of variance checking - integrate any unmeasurable/unreliable flags from this cached component
-                            var saved = entry & 24 /* ReportsMask */;
-                            if (saved & 8 /* ReportsUnmeasurable */) {
-                                instantiateType(source, makeFunctionTypeMapper(reportUnmeasurableMarkers));
-                            }
-                            if (saved & 16 /* ReportsUnreliable */) {
-                                instantiateType(source, makeFunctionTypeMapper(reportUnreliableMarkers));
-                            }
+                        var type = getTypeFromTypeNode(constraintDeclaration);
+                        if (type.flags & 1 /* Any */ && type !== errorType) { // Allow errorType to propegate to keep downstream errors suppressed
+                            // use keyofConstraintType as the base constraint for mapped type key constraints (unknown isn;t assignable to that, but `any` was),
+                            // use unknown otherwise
+                            type = constraintDeclaration.parent.parent.kind === 190 /* MappedType */ ? keyofConstraintType : unknownType;
                         }
-                        return entry & 1 /* Succeeded */ ? -1 /* True */ : 0 /* False */;
+                        typeParameter.constraint = type;
                     }
                 }
-                if (!maybeKeys) {
-                    maybeKeys = [];
-                    sourceStack = [];
-                    targetStack = [];
-                }
-                else {
-                    for (var i = 0; i < maybeCount; i++) {
-                        // If source and target are already being compared, consider them related with assumptions
-                        if (id === maybeKeys[i]) {
-                            return 1 /* Maybe */;
-                        }
+            }
+            return typeParameter.constraint === noConstraintType ? undefined : typeParameter.constraint;
+        }
+        function getParentSymbolOfTypeParameter(typeParameter) {
+            var tp = ts.getDeclarationOfKind(typeParameter.symbol, 159 /* TypeParameter */);
+            var host = ts.isJSDocTemplateTag(tp.parent) ? ts.getHostSignatureFromJSDoc(tp.parent) : tp.parent;
+            return host && getSymbolOfNode(host);
+        }
+        function getTypeListId(types) {
+            var result = "";
+            if (types) {
+                var length_4 = types.length;
+                var i = 0;
+                while (i < length_4) {
+                    var startId = types[i].id;
+                    var count = 1;
+                    while (i + count < length_4 && types[i + count].id === startId + count) {
+                        count++;
                     }
-                    if (depth === 100) {
-                        overflow = true;
-                        return 0 /* False */;
+                    if (result.length) {
+                        result += ",";
+                    }
+                    result += startId;
+                    if (count > 1) {
+                        result += ":" + count;
                     }
+                    i += count;
                 }
-                var maybeStart = maybeCount;
-                maybeKeys[maybeCount] = id;
-                maybeCount++;
-                sourceStack[depth] = source;
-                targetStack[depth] = target;
-                depth++;
-                var saveExpandingFlags = expandingFlags;
-                if (!(expandingFlags & 1 /* Source */) && isDeeplyNestedType(source, sourceStack, depth))
-                    expandingFlags |= 1 /* Source */;
-                if (!(expandingFlags & 2 /* Target */) && isDeeplyNestedType(target, targetStack, depth))
-                    expandingFlags |= 2 /* Target */;
-                var originalHandler;
-                var propagatingVarianceFlags = 0;
-                if (outofbandVarianceMarkerHandler) {
-                    originalHandler = outofbandVarianceMarkerHandler;
-                    outofbandVarianceMarkerHandler = function (onlyUnreliable) {
-                        propagatingVarianceFlags |= onlyUnreliable ? 16 /* ReportsUnreliable */ : 8 /* ReportsUnmeasurable */;
-                        return originalHandler(onlyUnreliable);
-                    };
+            }
+            return result;
+        }
+        // This function is used to propagate certain flags when creating new object type references and union types.
+        // It is only necessary to do so if a constituent type might be the undefined type, the null type, the type
+        // of an object literal or the anyFunctionType. This is because there are operations in the type checker
+        // that care about the presence of such types at arbitrary depth in a containing type.
+        function getPropagatingFlagsOfTypes(types, excludeKinds) {
+            var result = 0;
+            for (var _i = 0, types_9 = types; _i < types_9.length; _i++) {
+                var type = types_9[_i];
+                if (!(type.flags & excludeKinds)) {
+                    result |= ts.getObjectFlags(type);
                 }
-                var result = expandingFlags !== 3 /* Both */ ? structuredTypeRelatedTo(source, target, reportErrors, intersectionState) : 1 /* Maybe */;
-                if (outofbandVarianceMarkerHandler) {
-                    outofbandVarianceMarkerHandler = originalHandler;
+            }
+            return result & 3670016 /* PropagatingFlags */;
+        }
+        function createTypeReference(target, typeArguments) {
+            var id = getTypeListId(typeArguments);
+            var type = target.instantiations.get(id);
+            if (!type) {
+                type = createObjectType(4 /* Reference */, target.symbol);
+                target.instantiations.set(id, type);
+                type.objectFlags |= typeArguments ? getPropagatingFlagsOfTypes(typeArguments, /*excludeKinds*/ 0) : 0;
+                type.target = target;
+                type.resolvedTypeArguments = typeArguments;
+            }
+            return type;
+        }
+        function cloneTypeReference(source) {
+            var type = createType(source.flags);
+            type.symbol = source.symbol;
+            type.objectFlags = source.objectFlags;
+            type.target = source.target;
+            type.resolvedTypeArguments = source.resolvedTypeArguments;
+            return type;
+        }
+        function createDeferredTypeReference(target, node, mapper) {
+            var aliasSymbol = getAliasSymbolForTypeNode(node);
+            var aliasTypeArguments = getTypeArgumentsForAliasSymbol(aliasSymbol);
+            var type = createObjectType(4 /* Reference */, target.symbol);
+            type.target = target;
+            type.node = node;
+            type.mapper = mapper;
+            type.aliasSymbol = aliasSymbol;
+            type.aliasTypeArguments = mapper ? instantiateTypes(aliasTypeArguments, mapper) : aliasTypeArguments;
+            return type;
+        }
+        function getTypeArguments(type) {
+            var _a, _b;
+            if (!type.resolvedTypeArguments) {
+                if (!pushTypeResolution(type, 6 /* ResolvedTypeArguments */)) {
+                    return ((_a = type.target.localTypeParameters) === null || _a === void 0 ? void 0 : _a.map(function () { return errorType; })) || ts.emptyArray;
                 }
-                expandingFlags = saveExpandingFlags;
-                depth--;
-                if (result) {
-                    if (result === -1 /* True */ || depth === 0) {
-                        // If result is definitely true, record all maybe keys as having succeeded
-                        for (var i = maybeStart; i < maybeCount; i++) {
-                            relation.set(maybeKeys[i], 1 /* Succeeded */ | propagatingVarianceFlags);
-                        }
-                        maybeCount = maybeStart;
-                    }
+                var node = type.node;
+                var typeArguments = !node ? ts.emptyArray :
+                    node.kind === 173 /* TypeReference */ ? ts.concatenate(type.target.outerTypeParameters, getEffectiveTypeArguments(node, type.target.localTypeParameters)) :
+                        node.kind === 178 /* ArrayType */ ? [getTypeFromTypeNode(node.elementType)] :
+                            ts.map(node.elements, getTypeFromTypeNode);
+                if (popTypeResolution()) {
+                    type.resolvedTypeArguments = type.mapper ? instantiateTypes(typeArguments, type.mapper) : typeArguments;
                 }
                 else {
-                    // A false result goes straight into global cache (when something is false under
-                    // assumptions it will also be false without assumptions)
-                    relation.set(id, (reportErrors ? 4 /* Reported */ : 0) | 2 /* Failed */ | propagatingVarianceFlags);
-                    maybeCount = maybeStart;
+                    type.resolvedTypeArguments = ((_b = type.target.localTypeParameters) === null || _b === void 0 ? void 0 : _b.map(function () { return errorType; })) || ts.emptyArray;
+                    error(type.node || currentNode, type.target.symbol ? ts.Diagnostics.Type_arguments_for_0_circularly_reference_themselves : ts.Diagnostics.Tuple_type_arguments_circularly_reference_themselves, type.target.symbol && symbolToString(type.target.symbol));
                 }
-                return result;
             }
-            function structuredTypeRelatedTo(source, target, reportErrors, intersectionState) {
-                if (intersectionState & 4 /* PropertyCheck */) {
-                    return propertiesRelatedTo(source, target, reportErrors, /*excludedProperties*/ undefined, 0 /* None */);
-                }
-                var flags = source.flags & target.flags;
-                if (relation === identityRelation && !(flags & 524288 /* Object */)) {
-                    if (flags & 4194304 /* Index */) {
-                        return isRelatedTo(source.type, target.type, /*reportErrors*/ false);
-                    }
-                    var result_6 = 0 /* False */;
-                    if (flags & 8388608 /* IndexedAccess */) {
-                        if (result_6 = isRelatedTo(source.objectType, target.objectType, /*reportErrors*/ false)) {
-                            if (result_6 &= isRelatedTo(source.indexType, target.indexType, /*reportErrors*/ false)) {
-                                return result_6;
-                            }
-                        }
-                    }
-                    if (flags & 16777216 /* Conditional */) {
-                        if (source.root.isDistributive === target.root.isDistributive) {
-                            if (result_6 = isRelatedTo(source.checkType, target.checkType, /*reportErrors*/ false)) {
-                                if (result_6 &= isRelatedTo(source.extendsType, target.extendsType, /*reportErrors*/ false)) {
-                                    if (result_6 &= isRelatedTo(getTrueTypeFromConditionalType(source), getTrueTypeFromConditionalType(target), /*reportErrors*/ false)) {
-                                        if (result_6 &= isRelatedTo(getFalseTypeFromConditionalType(source), getFalseTypeFromConditionalType(target), /*reportErrors*/ false)) {
-                                            return result_6;
-                                        }
-                                    }
-                                }
-                            }
-                        }
-                    }
-                    if (flags & 33554432 /* Substitution */) {
-                        return isRelatedTo(source.substitute, target.substitute, /*reportErrors*/ false);
+            return type.resolvedTypeArguments;
+        }
+        function getTypeReferenceArity(type) {
+            return ts.length(type.target.typeParameters);
+        }
+        /**
+         * Get type from type-reference that reference to class or interface
+         */
+        function getTypeFromClassOrInterfaceReference(node, symbol) {
+            var type = getDeclaredTypeOfSymbol(getMergedSymbol(symbol));
+            var typeParameters = type.localTypeParameters;
+            if (typeParameters) {
+                var numTypeArguments = ts.length(node.typeArguments);
+                var minTypeArgumentCount = getMinTypeArgumentCount(typeParameters);
+                var isJs = ts.isInJSFile(node);
+                var isJsImplicitAny = !noImplicitAny && isJs;
+                if (!isJsImplicitAny && (numTypeArguments < minTypeArgumentCount || numTypeArguments > typeParameters.length)) {
+                    var missingAugmentsTag = isJs && ts.isExpressionWithTypeArguments(node) && !ts.isJSDocAugmentsTag(node.parent);
+                    var diag = minTypeArgumentCount === typeParameters.length ?
+                        missingAugmentsTag ?
+                            ts.Diagnostics.Expected_0_type_arguments_provide_these_with_an_extends_tag :
+                            ts.Diagnostics.Generic_type_0_requires_1_type_argument_s :
+                        missingAugmentsTag ?
+                            ts.Diagnostics.Expected_0_1_type_arguments_provide_these_with_an_extends_tag :
+                            ts.Diagnostics.Generic_type_0_requires_between_1_and_2_type_arguments;
+                    var typeStr = typeToString(type, /*enclosingDeclaration*/ undefined, 2 /* WriteArrayAsGenericType */);
+                    error(node, diag, typeStr, minTypeArgumentCount, typeParameters.length);
+                    if (!isJs) {
+                        // TODO: Adopt same permissive behavior in TS as in JS to reduce follow-on editing experience failures (requires editing fillMissingTypeArguments)
+                        return errorType;
                     }
-                    return 0 /* False */;
                 }
-                var result;
-                var originalErrorInfo;
-                var varianceCheckFailed = false;
-                var saveErrorInfo = captureErrorCalculationState();
-                // We limit alias variance probing to only object and conditional types since their alias behavior
-                // is more predictable than other, interned types, which may or may not have an alias depending on
-                // the order in which things were checked.
-                if (source.flags & (524288 /* Object */ | 16777216 /* Conditional */) && source.aliasSymbol &&
-                    source.aliasTypeArguments && source.aliasSymbol === target.aliasSymbol &&
-                    !(source.aliasTypeArgumentsContainsMarker || target.aliasTypeArgumentsContainsMarker)) {
-                    var variances = getAliasVariances(source.aliasSymbol);
-                    if (variances === ts.emptyArray) {
-                        return 1 /* Maybe */;
-                    }
-                    var varianceResult = relateVariances(source.aliasTypeArguments, target.aliasTypeArguments, variances, intersectionState);
-                    if (varianceResult !== undefined) {
-                        return varianceResult;
-                    }
+                if (node.kind === 173 /* TypeReference */ && isDeferredTypeReferenceNode(node, ts.length(node.typeArguments) !== typeParameters.length)) {
+                    return createDeferredTypeReference(type, node, /*mapper*/ undefined);
                 }
-                if (target.flags & 262144 /* TypeParameter */) {
-                    // A source type { [P in Q]: X } is related to a target type T if keyof T is related to Q and X is related to T[Q].
-                    if (ts.getObjectFlags(source) & 32 /* Mapped */ && isRelatedTo(getIndexType(target), getConstraintTypeFromMappedType(source))) {
-                        if (!(getMappedTypeModifiers(source) & 4 /* IncludeOptional */)) {
-                            var templateType = getTemplateTypeFromMappedType(source);
-                            var indexedAccessType = getIndexedAccessType(target, getTypeParameterFromMappedType(source));
-                            if (result = isRelatedTo(templateType, indexedAccessType, reportErrors)) {
-                                return result;
-                            }
-                        }
+                // In a type reference, the outer type parameters of the referenced class or interface are automatically
+                // supplied as type arguments and the type reference only specifies arguments for the local type parameters
+                // of the class or interface.
+                var typeArguments = ts.concatenate(type.outerTypeParameters, fillMissingTypeArguments(typeArgumentsFromTypeReferenceNode(node), typeParameters, minTypeArgumentCount, isJs));
+                return createTypeReference(type, typeArguments);
+            }
+            return checkNoTypeArguments(node, symbol) ? type : errorType;
+        }
+        function getTypeAliasInstantiation(symbol, typeArguments) {
+            var type = getDeclaredTypeOfSymbol(symbol);
+            if (type === intrinsicMarkerType && intrinsicTypeKinds.has(symbol.escapedName) && typeArguments && typeArguments.length === 1) {
+                return getStringMappingType(symbol, typeArguments[0]);
+            }
+            var links = getSymbolLinks(symbol);
+            var typeParameters = links.typeParameters;
+            var id = getTypeListId(typeArguments);
+            var instantiation = links.instantiations.get(id);
+            if (!instantiation) {
+                links.instantiations.set(id, instantiation = instantiateType(type, createTypeMapper(typeParameters, fillMissingTypeArguments(typeArguments, typeParameters, getMinTypeArgumentCount(typeParameters), ts.isInJSFile(symbol.valueDeclaration)))));
+            }
+            return instantiation;
+        }
+        /**
+         * Get type from reference to type alias. When a type alias is generic, the declared type of the type alias may include
+         * references to the type parameters of the alias. We replace those with the actual type arguments by instantiating the
+         * declared type. Instantiations are cached using the type identities of the type arguments as the key.
+         */
+        function getTypeFromTypeAliasReference(node, symbol) {
+            var type = getDeclaredTypeOfSymbol(symbol);
+            var typeParameters = getSymbolLinks(symbol).typeParameters;
+            if (typeParameters) {
+                var numTypeArguments = ts.length(node.typeArguments);
+                var minTypeArgumentCount = getMinTypeArgumentCount(typeParameters);
+                if (numTypeArguments < minTypeArgumentCount || numTypeArguments > typeParameters.length) {
+                    error(node, minTypeArgumentCount === typeParameters.length ?
+                        ts.Diagnostics.Generic_type_0_requires_1_type_argument_s :
+                        ts.Diagnostics.Generic_type_0_requires_between_1_and_2_type_arguments, symbolToString(symbol), minTypeArgumentCount, typeParameters.length);
+                    return errorType;
+                }
+                return getTypeAliasInstantiation(symbol, typeArgumentsFromTypeReferenceNode(node));
+            }
+            return checkNoTypeArguments(node, symbol) ? type : errorType;
+        }
+        function getTypeReferenceName(node) {
+            switch (node.kind) {
+                case 173 /* TypeReference */:
+                    return node.typeName;
+                case 223 /* ExpressionWithTypeArguments */:
+                    // We only support expressions that are simple qualified names. For other
+                    // expressions this produces undefined.
+                    var expr = node.expression;
+                    if (ts.isEntityNameExpression(expr)) {
+                        return expr;
                     }
+                // fall through;
+            }
+            return undefined;
+        }
+        function resolveTypeReferenceName(typeReferenceName, meaning, ignoreErrors) {
+            if (!typeReferenceName) {
+                return unknownSymbol;
+            }
+            return resolveEntityName(typeReferenceName, meaning, ignoreErrors) || unknownSymbol;
+        }
+        function getTypeReferenceType(node, symbol) {
+            if (symbol === unknownSymbol) {
+                return errorType;
+            }
+            symbol = getExpandoSymbol(symbol) || symbol;
+            if (symbol.flags & (32 /* Class */ | 64 /* Interface */)) {
+                return getTypeFromClassOrInterfaceReference(node, symbol);
+            }
+            if (symbol.flags & 524288 /* TypeAlias */) {
+                return getTypeFromTypeAliasReference(node, symbol);
+            }
+            // Get type from reference to named type that cannot be generic (enum or type parameter)
+            var res = tryGetDeclaredTypeOfSymbol(symbol);
+            if (res) {
+                return checkNoTypeArguments(node, symbol) ? getRegularTypeOfLiteralType(res) : errorType;
+            }
+            if (symbol.flags & 111551 /* Value */ && isJSDocTypeReference(node)) {
+                var jsdocType = getTypeFromJSDocValueReference(node, symbol);
+                if (jsdocType) {
+                    return jsdocType;
                 }
-                else if (target.flags & 4194304 /* Index */) {
-                    // A keyof S is related to a keyof T if T is related to S.
-                    if (source.flags & 4194304 /* Index */) {
-                        if (result = isRelatedTo(target.type, source.type, /*reportErrors*/ false)) {
-                            return result;
-                        }
+                else {
+                    // Resolve the type reference as a Type for the purpose of reporting errors.
+                    resolveTypeReferenceName(getTypeReferenceName(node), 788968 /* Type */);
+                    return getTypeOfSymbol(symbol);
+                }
+            }
+            return errorType;
+        }
+        /**
+         * A JSdoc TypeReference may be to a value, but resolve it as a type anyway.
+         * Example: import('./b').ConstructorFunction
+         */
+        function getTypeFromJSDocValueReference(node, symbol) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedJSDocType) {
+                var valueType = getTypeOfSymbol(symbol);
+                var typeType = valueType;
+                if (symbol.valueDeclaration) {
+                    var isImportTypeWithQualifier = node.kind === 195 /* ImportType */ && node.qualifier;
+                    // valueType might not have a symbol, eg, {import('./b').STRING_LITERAL}
+                    if (valueType.symbol && valueType.symbol !== symbol && isImportTypeWithQualifier) {
+                        typeType = getTypeReferenceType(node, valueType.symbol);
                     }
-                    // A type S is assignable to keyof T if S is assignable to keyof C, where C is the
-                    // simplified form of T or, if T doesn't simplify, the constraint of T.
-                    var constraint = getSimplifiedTypeOrConstraint(target.type);
+                }
+                links.resolvedJSDocType = typeType;
+            }
+            return links.resolvedJSDocType;
+        }
+        function getSubstitutionType(baseType, substitute) {
+            if (substitute.flags & 3 /* AnyOrUnknown */ || substitute === baseType) {
+                return baseType;
+            }
+            var id = getTypeId(baseType) + ">" + getTypeId(substitute);
+            var cached = substitutionTypes.get(id);
+            if (cached) {
+                return cached;
+            }
+            var result = createType(33554432 /* Substitution */);
+            result.baseType = baseType;
+            result.substitute = substitute;
+            substitutionTypes.set(id, result);
+            return result;
+        }
+        function isUnaryTupleTypeNode(node) {
+            return node.kind === 179 /* TupleType */ && node.elements.length === 1;
+        }
+        function getImpliedConstraint(type, checkNode, extendsNode) {
+            return isUnaryTupleTypeNode(checkNode) && isUnaryTupleTypeNode(extendsNode) ? getImpliedConstraint(type, checkNode.elements[0], extendsNode.elements[0]) :
+                getActualTypeVariable(getTypeFromTypeNode(checkNode)) === type ? getTypeFromTypeNode(extendsNode) :
+                    undefined;
+        }
+        function getConditionalFlowTypeOfType(type, node) {
+            var constraints;
+            while (node && !ts.isStatement(node) && node.kind !== 311 /* JSDocComment */) {
+                var parent = node.parent;
+                if (parent.kind === 184 /* ConditionalType */ && node === parent.trueType) {
+                    var constraint = getImpliedConstraint(type, parent.checkType, parent.extendsType);
                     if (constraint) {
-                        // We require Ternary.True here such that circular constraints don't cause
-                        // false positives. For example, given 'T extends { [K in keyof T]: string }',
-                        // 'keyof T' has itself as its constraint and produces a Ternary.Maybe when
-                        // related to other types.
-                        if (isRelatedTo(source, getIndexType(constraint, target.stringsOnly), reportErrors) === -1 /* True */) {
-                            return -1 /* True */;
-                        }
+                        constraints = ts.append(constraints, constraint);
                     }
                 }
-                else if (target.flags & 8388608 /* IndexedAccess */) {
-                    // A type S is related to a type T[K] if S is related to C, where C is the base
-                    // constraint of T[K] for writing.
-                    if (relation !== identityRelation) {
-                        var objectType = target.objectType;
-                        var indexType = target.indexType;
-                        var baseObjectType = getBaseConstraintOfType(objectType) || objectType;
-                        var baseIndexType = getBaseConstraintOfType(indexType) || indexType;
-                        if (!isGenericObjectType(baseObjectType) && !isGenericIndexType(baseIndexType)) {
-                            var accessFlags = 2 /* Writing */ | (baseObjectType !== objectType ? 1 /* NoIndexSignatures */ : 0);
-                            var constraint = getIndexedAccessTypeOrUndefined(baseObjectType, baseIndexType, /*accessNode*/ undefined, accessFlags);
-                            if (constraint && (result = isRelatedTo(source, constraint, reportErrors))) {
-                                return result;
+                node = parent;
+            }
+            return constraints ? getSubstitutionType(type, getIntersectionType(ts.append(constraints, type))) : type;
+        }
+        function isJSDocTypeReference(node) {
+            return !!(node.flags & 4194304 /* JSDoc */) && (node.kind === 173 /* TypeReference */ || node.kind === 195 /* ImportType */);
+        }
+        function checkNoTypeArguments(node, symbol) {
+            if (node.typeArguments) {
+                error(node, ts.Diagnostics.Type_0_is_not_generic, symbol ? symbolToString(symbol) : node.typeName ? ts.declarationNameToString(node.typeName) : anon);
+                return false;
+            }
+            return true;
+        }
+        function getIntendedTypeFromJSDocTypeReference(node) {
+            if (ts.isIdentifier(node.typeName)) {
+                var typeArgs = node.typeArguments;
+                switch (node.typeName.escapedText) {
+                    case "String":
+                        checkNoTypeArguments(node);
+                        return stringType;
+                    case "Number":
+                        checkNoTypeArguments(node);
+                        return numberType;
+                    case "Boolean":
+                        checkNoTypeArguments(node);
+                        return booleanType;
+                    case "Void":
+                        checkNoTypeArguments(node);
+                        return voidType;
+                    case "Undefined":
+                        checkNoTypeArguments(node);
+                        return undefinedType;
+                    case "Null":
+                        checkNoTypeArguments(node);
+                        return nullType;
+                    case "Function":
+                    case "function":
+                        checkNoTypeArguments(node);
+                        return globalFunctionType;
+                    case "array":
+                        return (!typeArgs || !typeArgs.length) && !noImplicitAny ? anyArrayType : undefined;
+                    case "promise":
+                        return (!typeArgs || !typeArgs.length) && !noImplicitAny ? createPromiseType(anyType) : undefined;
+                    case "Object":
+                        if (typeArgs && typeArgs.length === 2) {
+                            if (ts.isJSDocIndexSignature(node)) {
+                                var indexed = getTypeFromTypeNode(typeArgs[0]);
+                                var target = getTypeFromTypeNode(typeArgs[1]);
+                                var index = createIndexInfo(target, /*isReadonly*/ false);
+                                return createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, indexed === stringType ? index : undefined, indexed === numberType ? index : undefined);
                             }
+                            return anyType;
                         }
-                    }
+                        checkNoTypeArguments(node);
+                        return !noImplicitAny ? anyType : undefined;
                 }
-                else if (isGenericMappedType(target)) {
-                    // A source type T is related to a target type { [P in X]: T[P] }
-                    var template = getTemplateTypeFromMappedType(target);
-                    var modifiers = getMappedTypeModifiers(target);
-                    if (!(modifiers & 8 /* ExcludeOptional */)) {
-                        if (template.flags & 8388608 /* IndexedAccess */ && template.objectType === source &&
-                            template.indexType === getTypeParameterFromMappedType(target)) {
-                            return -1 /* True */;
-                        }
-                        if (!isGenericMappedType(source)) {
-                            var targetConstraint = getConstraintTypeFromMappedType(target);
-                            var sourceKeys = getIndexType(source, /*stringsOnly*/ undefined, /*noIndexSignatures*/ true);
-                            var includeOptional = modifiers & 4 /* IncludeOptional */;
-                            var filteredByApplicability = includeOptional ? intersectTypes(targetConstraint, sourceKeys) : undefined;
-                            // A source type T is related to a target type { [P in Q]: X } if Q is related to keyof T and T[Q] is related to X.
-                            // A source type T is related to a target type { [P in Q]?: X } if some constituent Q' of Q is related to keyof T and T[Q'] is related to X.
-                            if (includeOptional
-                                ? !(filteredByApplicability.flags & 131072 /* Never */)
-                                : isRelatedTo(targetConstraint, sourceKeys)) {
-                                var typeParameter = getTypeParameterFromMappedType(target);
-                                var indexingType = filteredByApplicability ? getIntersectionType([filteredByApplicability, typeParameter]) : typeParameter;
-                                var indexedAccessType = getIndexedAccessType(source, indexingType);
-                                var templateType = getTemplateTypeFromMappedType(target);
-                                if (result = isRelatedTo(indexedAccessType, templateType, reportErrors)) {
-                                    return result;
-                                }
-                            }
-                            originalErrorInfo = errorInfo;
-                            resetErrorInfo(saveErrorInfo);
-                        }
-                    }
+            }
+        }
+        function getTypeFromJSDocNullableTypeNode(node) {
+            var type = getTypeFromTypeNode(node.type);
+            return strictNullChecks ? getNullableType(type, 65536 /* Null */) : type;
+        }
+        function getTypeFromTypeReference(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                // handle LS queries on the `const` in `x as const` by resolving to the type of `x`
+                if (ts.isConstTypeReference(node) && ts.isAssertionExpression(node.parent)) {
+                    links.resolvedSymbol = unknownSymbol;
+                    return links.resolvedType = checkExpressionCached(node.parent.expression);
                 }
-                if (source.flags & 8650752 /* TypeVariable */) {
-                    if (source.flags & 8388608 /* IndexedAccess */ && target.flags & 8388608 /* IndexedAccess */) {
-                        // A type S[K] is related to a type T[J] if S is related to T and K is related to J.
-                        if (result = isRelatedTo(source.objectType, target.objectType, reportErrors)) {
-                            result &= isRelatedTo(source.indexType, target.indexType, reportErrors);
-                        }
-                        if (result) {
-                            resetErrorInfo(saveErrorInfo);
-                            return result;
-                        }
-                    }
-                    else {
-                        var constraint = getConstraintOfType(source);
-                        if (!constraint || (source.flags & 262144 /* TypeParameter */ && constraint.flags & 1 /* Any */)) {
-                            // A type variable with no constraint is not related to the non-primitive object type.
-                            if (result = isRelatedTo(emptyObjectType, extractTypesOfKind(target, ~67108864 /* NonPrimitive */))) {
-                                resetErrorInfo(saveErrorInfo);
-                                return result;
-                            }
-                        }
-                        // hi-speed no-this-instantiation check (less accurate, but avoids costly `this`-instantiation when the constraint will suffice), see #28231 for report on why this is needed
-                        else if (result = isRelatedTo(constraint, target, /*reportErrors*/ false, /*headMessage*/ undefined, intersectionState)) {
-                            resetErrorInfo(saveErrorInfo);
-                            return result;
+                var symbol = void 0;
+                var type = void 0;
+                var meaning = 788968 /* Type */;
+                if (isJSDocTypeReference(node)) {
+                    type = getIntendedTypeFromJSDocTypeReference(node);
+                    if (!type) {
+                        symbol = resolveTypeReferenceName(getTypeReferenceName(node), meaning, /*ignoreErrors*/ true);
+                        if (symbol === unknownSymbol) {
+                            symbol = resolveTypeReferenceName(getTypeReferenceName(node), meaning | 111551 /* Value */);
                         }
-                        // slower, fuller, this-instantiated check (necessary when comparing raw `this` types from base classes), see `subclassWithPolymorphicThisIsAssignable.ts` test for example
-                        else if (result = isRelatedTo(getTypeWithThisArgument(constraint, source), target, reportErrors, /*headMessage*/ undefined, intersectionState)) {
-                            resetErrorInfo(saveErrorInfo);
-                            return result;
+                        else {
+                            resolveTypeReferenceName(getTypeReferenceName(node), meaning); // Resolve again to mark errors, if any
                         }
+                        type = getTypeReferenceType(node, symbol);
                     }
                 }
-                else if (source.flags & 4194304 /* Index */) {
-                    if (result = isRelatedTo(keyofConstraintType, target, reportErrors)) {
-                        resetErrorInfo(saveErrorInfo);
-                        return result;
-                    }
+                if (!type) {
+                    symbol = resolveTypeReferenceName(getTypeReferenceName(node), meaning);
+                    type = getTypeReferenceType(node, symbol);
                 }
-                else if (source.flags & 16777216 /* Conditional */) {
-                    if (target.flags & 16777216 /* Conditional */) {
-                        // Two conditional types 'T1 extends U1 ? X1 : Y1' and 'T2 extends U2 ? X2 : Y2' are related if
-                        // one of T1 and T2 is related to the other, U1 and U2 are identical types, X1 is related to X2,
-                        // and Y1 is related to Y2.
-                        var sourceParams = source.root.inferTypeParameters;
-                        var sourceExtends = source.extendsType;
-                        var mapper = void 0;
-                        if (sourceParams) {
-                            // If the source has infer type parameters, we instantiate them in the context of the target
-                            var ctx = createInferenceContext(sourceParams, /*signature*/ undefined, 0 /* None */, isRelatedTo);
-                            inferTypes(ctx.inferences, target.extendsType, sourceExtends, 128 /* NoConstraints */ | 256 /* AlwaysStrict */);
-                            sourceExtends = instantiateType(sourceExtends, ctx.mapper);
-                            mapper = ctx.mapper;
-                        }
-                        if (isTypeIdenticalTo(sourceExtends, target.extendsType) &&
-                            (isRelatedTo(source.checkType, target.checkType) || isRelatedTo(target.checkType, source.checkType))) {
-                            if (result = isRelatedTo(instantiateType(getTrueTypeFromConditionalType(source), mapper), getTrueTypeFromConditionalType(target), reportErrors)) {
-                                result &= isRelatedTo(getFalseTypeFromConditionalType(source), getFalseTypeFromConditionalType(target), reportErrors);
-                            }
-                            if (result) {
-                                resetErrorInfo(saveErrorInfo);
-                                return result;
-                            }
-                        }
-                    }
-                    else {
-                        // conditionals aren't related to one another via distributive constraint as it is much too inaccurate and allows way
-                        // more assignments than are desirable (since it maps the source check type to its constraint, it loses information)
-                        var distributiveConstraint = getConstraintOfDistributiveConditionalType(source);
-                        if (distributiveConstraint) {
-                            if (result = isRelatedTo(distributiveConstraint, target, reportErrors)) {
-                                resetErrorInfo(saveErrorInfo);
-                                return result;
-                            }
-                        }
-                    }
-                    // conditionals _can_ be related to one another via normal constraint, as, eg, `A extends B ? O : never` should be assignable to `O`
-                    // when `O` is a conditional (`never` is trivially aissgnable to `O`, as is `O`!).
-                    var defaultConstraint = getDefaultConstraintOfConditionalType(source);
-                    if (defaultConstraint) {
-                        if (result = isRelatedTo(defaultConstraint, target, reportErrors)) {
-                            resetErrorInfo(saveErrorInfo);
-                            return result;
-                        }
+                // Cache both the resolved symbol and the resolved type. The resolved symbol is needed when we check the
+                // type reference in checkTypeReferenceNode.
+                links.resolvedSymbol = symbol;
+                links.resolvedType = type;
+            }
+            return links.resolvedType;
+        }
+        function typeArgumentsFromTypeReferenceNode(node) {
+            return ts.map(node.typeArguments, getTypeFromTypeNode);
+        }
+        function getTypeFromTypeQueryNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                // TypeScript 1.0 spec (April 2014): 3.6.3
+                // The expression is processed as an identifier expression (section 4.3)
+                // or property access expression(section 4.10),
+                // the widened type(section 3.9) of which becomes the result.
+                links.resolvedType = getRegularTypeOfLiteralType(getWidenedType(checkExpression(node.exprName)));
+            }
+            return links.resolvedType;
+        }
+        function getTypeOfGlobalSymbol(symbol, arity) {
+            function getTypeDeclaration(symbol) {
+                var declarations = symbol.declarations;
+                for (var _i = 0, declarations_3 = declarations; _i < declarations_3.length; _i++) {
+                    var declaration = declarations_3[_i];
+                    switch (declaration.kind) {
+                        case 252 /* ClassDeclaration */:
+                        case 253 /* InterfaceDeclaration */:
+                        case 255 /* EnumDeclaration */:
+                            return declaration;
                     }
                 }
+            }
+            if (!symbol) {
+                return arity ? emptyGenericType : emptyObjectType;
+            }
+            var type = getDeclaredTypeOfSymbol(symbol);
+            if (!(type.flags & 524288 /* Object */)) {
+                error(getTypeDeclaration(symbol), ts.Diagnostics.Global_type_0_must_be_a_class_or_interface_type, ts.symbolName(symbol));
+                return arity ? emptyGenericType : emptyObjectType;
+            }
+            if (ts.length(type.typeParameters) !== arity) {
+                error(getTypeDeclaration(symbol), ts.Diagnostics.Global_type_0_must_have_1_type_parameter_s, ts.symbolName(symbol), arity);
+                return arity ? emptyGenericType : emptyObjectType;
+            }
+            return type;
+        }
+        function getGlobalValueSymbol(name, reportErrors) {
+            return getGlobalSymbol(name, 111551 /* Value */, reportErrors ? ts.Diagnostics.Cannot_find_global_value_0 : undefined);
+        }
+        function getGlobalTypeSymbol(name, reportErrors) {
+            return getGlobalSymbol(name, 788968 /* Type */, reportErrors ? ts.Diagnostics.Cannot_find_global_type_0 : undefined);
+        }
+        function getGlobalSymbol(name, meaning, diagnostic) {
+            // Don't track references for global symbols anyway, so value if `isReference` is arbitrary
+            return resolveName(undefined, name, meaning, diagnostic, name, /*isUse*/ false);
+        }
+        function getGlobalType(name, arity, reportErrors) {
+            var symbol = getGlobalTypeSymbol(name, reportErrors);
+            return symbol || reportErrors ? getTypeOfGlobalSymbol(symbol, arity) : undefined;
+        }
+        function getGlobalTypedPropertyDescriptorType() {
+            return deferredGlobalTypedPropertyDescriptorType || (deferredGlobalTypedPropertyDescriptorType = getGlobalType("TypedPropertyDescriptor", /*arity*/ 1, /*reportErrors*/ true)) || emptyGenericType;
+        }
+        function getGlobalTemplateStringsArrayType() {
+            return deferredGlobalTemplateStringsArrayType || (deferredGlobalTemplateStringsArrayType = getGlobalType("TemplateStringsArray", /*arity*/ 0, /*reportErrors*/ true)) || emptyObjectType;
+        }
+        function getGlobalImportMetaType() {
+            return deferredGlobalImportMetaType || (deferredGlobalImportMetaType = getGlobalType("ImportMeta", /*arity*/ 0, /*reportErrors*/ true)) || emptyObjectType;
+        }
+        function getGlobalESSymbolConstructorSymbol(reportErrors) {
+            return deferredGlobalESSymbolConstructorSymbol || (deferredGlobalESSymbolConstructorSymbol = getGlobalValueSymbol("Symbol", reportErrors));
+        }
+        function getGlobalESSymbolType(reportErrors) {
+            return deferredGlobalESSymbolType || (deferredGlobalESSymbolType = getGlobalType("Symbol", /*arity*/ 0, reportErrors)) || emptyObjectType;
+        }
+        function getGlobalPromiseType(reportErrors) {
+            return deferredGlobalPromiseType || (deferredGlobalPromiseType = getGlobalType("Promise", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalPromiseLikeType(reportErrors) {
+            return deferredGlobalPromiseLikeType || (deferredGlobalPromiseLikeType = getGlobalType("PromiseLike", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalPromiseConstructorSymbol(reportErrors) {
+            return deferredGlobalPromiseConstructorSymbol || (deferredGlobalPromiseConstructorSymbol = getGlobalValueSymbol("Promise", reportErrors));
+        }
+        function getGlobalPromiseConstructorLikeType(reportErrors) {
+            return deferredGlobalPromiseConstructorLikeType || (deferredGlobalPromiseConstructorLikeType = getGlobalType("PromiseConstructorLike", /*arity*/ 0, reportErrors)) || emptyObjectType;
+        }
+        function getGlobalAsyncIterableType(reportErrors) {
+            return deferredGlobalAsyncIterableType || (deferredGlobalAsyncIterableType = getGlobalType("AsyncIterable", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalAsyncIteratorType(reportErrors) {
+            return deferredGlobalAsyncIteratorType || (deferredGlobalAsyncIteratorType = getGlobalType("AsyncIterator", /*arity*/ 3, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalAsyncIterableIteratorType(reportErrors) {
+            return deferredGlobalAsyncIterableIteratorType || (deferredGlobalAsyncIterableIteratorType = getGlobalType("AsyncIterableIterator", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalAsyncGeneratorType(reportErrors) {
+            return deferredGlobalAsyncGeneratorType || (deferredGlobalAsyncGeneratorType = getGlobalType("AsyncGenerator", /*arity*/ 3, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalIterableType(reportErrors) {
+            return deferredGlobalIterableType || (deferredGlobalIterableType = getGlobalType("Iterable", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalIteratorType(reportErrors) {
+            return deferredGlobalIteratorType || (deferredGlobalIteratorType = getGlobalType("Iterator", /*arity*/ 3, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalIterableIteratorType(reportErrors) {
+            return deferredGlobalIterableIteratorType || (deferredGlobalIterableIteratorType = getGlobalType("IterableIterator", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalGeneratorType(reportErrors) {
+            return deferredGlobalGeneratorType || (deferredGlobalGeneratorType = getGlobalType("Generator", /*arity*/ 3, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalIteratorYieldResultType(reportErrors) {
+            return deferredGlobalIteratorYieldResultType || (deferredGlobalIteratorYieldResultType = getGlobalType("IteratorYieldResult", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalIteratorReturnResultType(reportErrors) {
+            return deferredGlobalIteratorReturnResultType || (deferredGlobalIteratorReturnResultType = getGlobalType("IteratorReturnResult", /*arity*/ 1, reportErrors)) || emptyGenericType;
+        }
+        function getGlobalTypeOrUndefined(name, arity) {
+            if (arity === void 0) { arity = 0; }
+            var symbol = getGlobalSymbol(name, 788968 /* Type */, /*diagnostic*/ undefined);
+            return symbol && getTypeOfGlobalSymbol(symbol, arity);
+        }
+        function getGlobalExtractSymbol() {
+            return deferredGlobalExtractSymbol || (deferredGlobalExtractSymbol = getGlobalSymbol("Extract", 524288 /* TypeAlias */, ts.Diagnostics.Cannot_find_global_type_0)); // TODO: GH#18217
+        }
+        function getGlobalOmitSymbol() {
+            return deferredGlobalOmitSymbol || (deferredGlobalOmitSymbol = getGlobalSymbol("Omit", 524288 /* TypeAlias */, ts.Diagnostics.Cannot_find_global_type_0)); // TODO: GH#18217
+        }
+        function getGlobalBigIntType(reportErrors) {
+            return deferredGlobalBigIntType || (deferredGlobalBigIntType = getGlobalType("BigInt", /*arity*/ 0, reportErrors)) || emptyObjectType;
+        }
+        /**
+         * Instantiates a global type that is generic with some element type, and returns that instantiation.
+         */
+        function createTypeFromGenericGlobalType(genericGlobalType, typeArguments) {
+            return genericGlobalType !== emptyGenericType ? createTypeReference(genericGlobalType, typeArguments) : emptyObjectType;
+        }
+        function createTypedPropertyDescriptorType(propertyType) {
+            return createTypeFromGenericGlobalType(getGlobalTypedPropertyDescriptorType(), [propertyType]);
+        }
+        function createIterableType(iteratedType) {
+            return createTypeFromGenericGlobalType(getGlobalIterableType(/*reportErrors*/ true), [iteratedType]);
+        }
+        function createArrayType(elementType, readonly) {
+            return createTypeFromGenericGlobalType(readonly ? globalReadonlyArrayType : globalArrayType, [elementType]);
+        }
+        function getTupleElementFlags(node) {
+            switch (node.kind) {
+                case 180 /* OptionalType */:
+                    return 2 /* Optional */;
+                case 181 /* RestType */:
+                    return getRestTypeElementFlags(node);
+                case 192 /* NamedTupleMember */:
+                    return node.questionToken ? 2 /* Optional */ :
+                        node.dotDotDotToken ? getRestTypeElementFlags(node) :
+                            1 /* Required */;
+                default:
+                    return 1 /* Required */;
+            }
+        }
+        function getRestTypeElementFlags(node) {
+            return getArrayElementTypeNode(node.type) ? 4 /* Rest */ : 8 /* Variadic */;
+        }
+        function getArrayOrTupleTargetType(node) {
+            var readonly = isReadonlyTypeOperator(node.parent);
+            var elementType = getArrayElementTypeNode(node);
+            if (elementType) {
+                return readonly ? globalReadonlyArrayType : globalArrayType;
+            }
+            var elementFlags = ts.map(node.elements, getTupleElementFlags);
+            var missingName = ts.some(node.elements, function (e) { return e.kind !== 192 /* NamedTupleMember */; });
+            return getTupleTargetType(elementFlags, readonly, /*associatedNames*/ missingName ? undefined : node.elements);
+        }
+        // Return true if the given type reference node is directly aliased or if it needs to be deferred
+        // because it is possibly contained in a circular chain of eagerly resolved types.
+        function isDeferredTypeReferenceNode(node, hasDefaultTypeArguments) {
+            return !!getAliasSymbolForTypeNode(node) || isResolvedByTypeAlias(node) && (node.kind === 178 /* ArrayType */ ? mayResolveTypeAlias(node.elementType) :
+                node.kind === 179 /* TupleType */ ? ts.some(node.elements, mayResolveTypeAlias) :
+                    hasDefaultTypeArguments || ts.some(node.typeArguments, mayResolveTypeAlias));
+        }
+        // Return true when the given node is transitively contained in type constructs that eagerly
+        // resolve their constituent types. We include SyntaxKind.TypeReference because type arguments
+        // of type aliases are eagerly resolved.
+        function isResolvedByTypeAlias(node) {
+            var parent = node.parent;
+            switch (parent.kind) {
+                case 186 /* ParenthesizedType */:
+                case 192 /* NamedTupleMember */:
+                case 173 /* TypeReference */:
+                case 182 /* UnionType */:
+                case 183 /* IntersectionType */:
+                case 189 /* IndexedAccessType */:
+                case 184 /* ConditionalType */:
+                case 188 /* TypeOperator */:
+                case 178 /* ArrayType */:
+                case 179 /* TupleType */:
+                    return isResolvedByTypeAlias(parent);
+                case 254 /* TypeAliasDeclaration */:
+                    return true;
+            }
+            return false;
+        }
+        // Return true if resolving the given node (i.e. getTypeFromTypeNode) possibly causes resolution
+        // of a type alias.
+        function mayResolveTypeAlias(node) {
+            switch (node.kind) {
+                case 173 /* TypeReference */:
+                    return isJSDocTypeReference(node) || !!(resolveTypeReferenceName(node.typeName, 788968 /* Type */).flags & 524288 /* TypeAlias */);
+                case 176 /* TypeQuery */:
+                    return true;
+                case 188 /* TypeOperator */:
+                    return node.operator !== 151 /* UniqueKeyword */ && mayResolveTypeAlias(node.type);
+                case 186 /* ParenthesizedType */:
+                case 180 /* OptionalType */:
+                case 192 /* NamedTupleMember */:
+                case 307 /* JSDocOptionalType */:
+                case 305 /* JSDocNullableType */:
+                case 306 /* JSDocNonNullableType */:
+                case 301 /* JSDocTypeExpression */:
+                    return mayResolveTypeAlias(node.type);
+                case 181 /* RestType */:
+                    return node.type.kind !== 178 /* ArrayType */ || mayResolveTypeAlias(node.type.elementType);
+                case 182 /* UnionType */:
+                case 183 /* IntersectionType */:
+                    return ts.some(node.types, mayResolveTypeAlias);
+                case 189 /* IndexedAccessType */:
+                    return mayResolveTypeAlias(node.objectType) || mayResolveTypeAlias(node.indexType);
+                case 184 /* ConditionalType */:
+                    return mayResolveTypeAlias(node.checkType) || mayResolveTypeAlias(node.extendsType) ||
+                        mayResolveTypeAlias(node.trueType) || mayResolveTypeAlias(node.falseType);
+            }
+            return false;
+        }
+        function getTypeFromArrayOrTupleTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                var target = getArrayOrTupleTargetType(node);
+                if (target === emptyGenericType) {
+                    links.resolvedType = emptyObjectType;
+                }
+                else if (!(node.kind === 179 /* TupleType */ && ts.some(node.elements, function (e) { return !!(getTupleElementFlags(e) & 8 /* Variadic */); })) && isDeferredTypeReferenceNode(node)) {
+                    links.resolvedType = node.kind === 179 /* TupleType */ && node.elements.length === 0 ? target :
+                        createDeferredTypeReference(target, node, /*mapper*/ undefined);
+                }
                 else {
-                    // An empty object type is related to any mapped type that includes a '?' modifier.
-                    if (relation !== subtypeRelation && relation !== strictSubtypeRelation && isPartialMappedType(target) && isEmptyObjectType(source)) {
-                        return -1 /* True */;
-                    }
-                    if (isGenericMappedType(target)) {
-                        if (isGenericMappedType(source)) {
-                            if (result = mappedTypeRelatedTo(source, target, reportErrors)) {
-                                resetErrorInfo(saveErrorInfo);
-                                return result;
-                            }
-                        }
-                        return 0 /* False */;
-                    }
-                    var sourceIsPrimitive = !!(source.flags & 131068 /* Primitive */);
-                    if (relation !== identityRelation) {
-                        source = getApparentType(source);
-                    }
-                    else if (isGenericMappedType(source)) {
-                        return 0 /* False */;
-                    }
-                    if (ts.getObjectFlags(source) & 4 /* Reference */ && ts.getObjectFlags(target) & 4 /* Reference */ && source.target === target.target &&
-                        !(ts.getObjectFlags(source) & 8192 /* MarkerType */ || ts.getObjectFlags(target) & 8192 /* MarkerType */)) {
-                        // We have type references to the same generic type, and the type references are not marker
-                        // type references (which are intended by be compared structurally). Obtain the variance
-                        // information for the type parameters and relate the type arguments accordingly.
-                        var variances = getVariances(source.target);
-                        // We return Ternary.Maybe for a recursive invocation of getVariances (signalled by emptyArray). This
-                        // effectively means we measure variance only from type parameter occurrences that aren't nested in
-                        // recursive instantiations of the generic type.
-                        if (variances === ts.emptyArray) {
-                            return 1 /* Maybe */;
-                        }
-                        var varianceResult = relateVariances(getTypeArguments(source), getTypeArguments(target), variances, intersectionState);
-                        if (varianceResult !== undefined) {
-                            return varianceResult;
-                        }
-                    }
-                    else if (isReadonlyArrayType(target) ? isArrayType(source) || isTupleType(source) : isArrayType(target) && isTupleType(source) && !source.target.readonly) {
-                        if (relation !== identityRelation) {
-                            return isRelatedTo(getIndexTypeOfType(source, 1 /* Number */) || anyType, getIndexTypeOfType(target, 1 /* Number */) || anyType, reportErrors);
-                        }
-                        else {
-                            // By flags alone, we know that the `target` is a readonly array while the source is a normal array or tuple
-                            // or `target` is an array and source is a tuple - in both cases the types cannot be identical, by construction
-                            return 0 /* False */;
-                        }
-                    }
-                    // Consider a fresh empty object literal type "closed" under the subtype relationship - this way `{} <- {[idx: string]: any} <- fresh({})`
-                    // and not `{} <- fresh({}) <- {[idx: string]: any}`
-                    else if ((relation === subtypeRelation || relation === strictSubtypeRelation) && isEmptyObjectType(target) && ts.getObjectFlags(target) & 32768 /* FreshLiteral */ && !isEmptyObjectType(source)) {
-                        return 0 /* False */;
-                    }
-                    // Even if relationship doesn't hold for unions, intersections, or generic type references,
-                    // it may hold in a structural comparison.
-                    // In a check of the form X = A & B, we will have previously checked if A relates to X or B relates
-                    // to X. Failing both of those we want to check if the aggregation of A and B's members structurally
-                    // relates to X. Thus, we include intersection types on the source side here.
-                    if (source.flags & (524288 /* Object */ | 2097152 /* Intersection */) && target.flags & 524288 /* Object */) {
-                        // Report structural errors only if we haven't reported any errors yet
-                        var reportStructuralErrors = reportErrors && errorInfo === saveErrorInfo.errorInfo && !sourceIsPrimitive;
-                        result = propertiesRelatedTo(source, target, reportStructuralErrors, /*excludedProperties*/ undefined, intersectionState);
-                        if (result) {
-                            result &= signaturesRelatedTo(source, target, 0 /* Call */, reportStructuralErrors);
-                            if (result) {
-                                result &= signaturesRelatedTo(source, target, 1 /* Construct */, reportStructuralErrors);
-                                if (result) {
-                                    result &= indexTypesRelatedTo(source, target, 0 /* String */, sourceIsPrimitive, reportStructuralErrors, intersectionState);
-                                    if (result) {
-                                        result &= indexTypesRelatedTo(source, target, 1 /* Number */, sourceIsPrimitive, reportStructuralErrors, intersectionState);
-                                    }
-                                }
-                            }
-                        }
-                        if (varianceCheckFailed && result) {
-                            errorInfo = originalErrorInfo || errorInfo || saveErrorInfo.errorInfo; // Use variance error (there is no structural one) and return false
-                        }
-                        else if (result) {
-                            return result;
-                        }
-                    }
-                    // If S is an object type and T is a discriminated union, S may be related to T if
-                    // there exists a constituent of T for every combination of the discriminants of S
-                    // with respect to T. We do not report errors here, as we will use the existing
-                    // error result from checking each constituent of the union.
-                    if (source.flags & (524288 /* Object */ | 2097152 /* Intersection */) && target.flags & 1048576 /* Union */) {
-                        var objectOnlyTarget = extractTypesOfKind(target, 524288 /* Object */ | 2097152 /* Intersection */ | 33554432 /* Substitution */);
-                        if (objectOnlyTarget.flags & 1048576 /* Union */) {
-                            var result_7 = typeRelatedToDiscriminatedType(source, objectOnlyTarget);
-                            if (result_7) {
-                                return result_7;
-                            }
-                        }
-                    }
+                    var elementTypes = node.kind === 178 /* ArrayType */ ? [getTypeFromTypeNode(node.elementType)] : ts.map(node.elements, getTypeFromTypeNode);
+                    links.resolvedType = createNormalizedTypeReference(target, elementTypes);
                 }
-                return 0 /* False */;
-                function relateVariances(sourceTypeArguments, targetTypeArguments, variances, intersectionState) {
-                    if (result = typeArgumentsRelatedTo(sourceTypeArguments, targetTypeArguments, variances, reportErrors, intersectionState)) {
-                        return result;
-                    }
-                    if (ts.some(variances, function (v) { return !!(v & 24 /* AllowsStructuralFallback */); })) {
-                        // If some type parameter was `Unmeasurable` or `Unreliable`, and we couldn't pass by assuming it was identical, then we
-                        // have to allow a structural fallback check
-                        // We elide the variance-based error elaborations, since those might not be too helpful, since we'll potentially
-                        // be assuming identity of the type parameter.
-                        originalErrorInfo = undefined;
-                        resetErrorInfo(saveErrorInfo);
-                        return undefined;
-                    }
-                    var allowStructuralFallback = targetTypeArguments && hasCovariantVoidArgument(targetTypeArguments, variances);
-                    varianceCheckFailed = !allowStructuralFallback;
-                    // The type arguments did not relate appropriately, but it may be because we have no variance
-                    // information (in which case typeArgumentsRelatedTo defaulted to covariance for all type
-                    // arguments). It might also be the case that the target type has a 'void' type argument for
-                    // a covariant type parameter that is only used in return positions within the generic type
-                    // (in which case any type argument is permitted on the source side). In those cases we proceed
-                    // with a structural comparison. Otherwise, we know for certain the instantiations aren't
-                    // related and we can return here.
-                    if (variances !== ts.emptyArray && !allowStructuralFallback) {
-                        // In some cases generic types that are covariant in regular type checking mode become
-                        // invariant in --strictFunctionTypes mode because one or more type parameters are used in
-                        // both co- and contravariant positions. In order to make it easier to diagnose *why* such
-                        // types are invariant, if any of the type parameters are invariant we reset the reported
-                        // errors and instead force a structural comparison (which will include elaborations that
-                        // reveal the reason).
-                        // We can switch on `reportErrors` here, since varianceCheckFailed guarantees we return `False`,
-                        // we can return `False` early here to skip calculating the structural error message we don't need.
-                        if (varianceCheckFailed && !(reportErrors && ts.some(variances, function (v) { return (v & 7 /* VarianceMask */) === 0 /* Invariant */; }))) {
-                            return 0 /* False */;
-                        }
-                        // We remember the original error information so we can restore it in case the structural
-                        // comparison unexpectedly succeeds. This can happen when the structural comparison result
-                        // is a Ternary.Maybe for example caused by the recursion depth limiter.
-                        originalErrorInfo = errorInfo;
-                        resetErrorInfo(saveErrorInfo);
+            }
+            return links.resolvedType;
+        }
+        function isReadonlyTypeOperator(node) {
+            return ts.isTypeOperatorNode(node) && node.operator === 142 /* ReadonlyKeyword */;
+        }
+        function createTupleType(elementTypes, elementFlags, readonly, namedMemberDeclarations) {
+            if (readonly === void 0) { readonly = false; }
+            var tupleTarget = getTupleTargetType(elementFlags || ts.map(elementTypes, function (_) { return 1 /* Required */; }), readonly, namedMemberDeclarations);
+            return tupleTarget === emptyGenericType ? emptyObjectType :
+                elementTypes.length ? createNormalizedTypeReference(tupleTarget, elementTypes) :
+                    tupleTarget;
+        }
+        function getTupleTargetType(elementFlags, readonly, namedMemberDeclarations) {
+            if (elementFlags.length === 1 && elementFlags[0] & 4 /* Rest */) {
+                // [...X[]] is equivalent to just X[]
+                return readonly ? globalReadonlyArrayType : globalArrayType;
+            }
+            var key = ts.map(elementFlags, function (f) { return f & 1 /* Required */ ? "#" : f & 2 /* Optional */ ? "?" : f & 4 /* Rest */ ? "." : "*"; }).join() +
+                (readonly ? "R" : "") +
+                (namedMemberDeclarations && namedMemberDeclarations.length ? "," + ts.map(namedMemberDeclarations, getNodeId).join(",") : "");
+            var type = tupleTypes.get(key);
+            if (!type) {
+                tupleTypes.set(key, type = createTupleTargetType(elementFlags, readonly, namedMemberDeclarations));
+            }
+            return type;
+        }
+        // We represent tuple types as type references to synthesized generic interface types created by
+        // this function. The types are of the form:
+        //
+        //   interface Tuple<T0, T1, T2, ...> extends Array<T0 | T1 | T2 | ...> { 0: T0, 1: T1, 2: T2, ... }
+        //
+        // Note that the generic type created by this function has no symbol associated with it. The same
+        // is true for each of the synthesized type parameters.
+        function createTupleTargetType(elementFlags, readonly, namedMemberDeclarations) {
+            var arity = elementFlags.length;
+            var minLength = ts.findLastIndex(elementFlags, function (f) { return !!(f & (1 /* Required */ | 8 /* Variadic */)); }) + 1;
+            var typeParameters;
+            var properties = [];
+            var combinedFlags = 0;
+            if (arity) {
+                typeParameters = new Array(arity);
+                for (var i = 0; i < arity; i++) {
+                    var typeParameter = typeParameters[i] = createTypeParameter();
+                    var flags = elementFlags[i];
+                    combinedFlags |= flags;
+                    if (!(combinedFlags & 12 /* Variable */)) {
+                        var property = createSymbol(4 /* Property */ | (flags & 2 /* Optional */ ? 16777216 /* Optional */ : 0), "" + i, readonly ? 8 /* Readonly */ : 0);
+                        property.tupleLabelDeclaration = namedMemberDeclarations === null || namedMemberDeclarations === void 0 ? void 0 : namedMemberDeclarations[i];
+                        property.type = typeParameter;
+                        properties.push(property);
                     }
                 }
             }
-            function reportUnmeasurableMarkers(p) {
-                if (outofbandVarianceMarkerHandler && (p === markerSuperType || p === markerSubType || p === markerOtherType)) {
-                    outofbandVarianceMarkerHandler(/*onlyUnreliable*/ false);
-                }
-                return p;
+            var fixedLength = properties.length;
+            var lengthSymbol = createSymbol(4 /* Property */, "length");
+            if (combinedFlags & 12 /* Variable */) {
+                lengthSymbol.type = numberType;
             }
-            function reportUnreliableMarkers(p) {
-                if (outofbandVarianceMarkerHandler && (p === markerSuperType || p === markerSubType || p === markerOtherType)) {
-                    outofbandVarianceMarkerHandler(/*onlyUnreliable*/ true);
-                }
-                return p;
+            else {
+                var literalTypes_1 = [];
+                for (var i = minLength; i <= arity; i++)
+                    literalTypes_1.push(getLiteralType(i));
+                lengthSymbol.type = getUnionType(literalTypes_1);
             }
-            // A type [P in S]: X is related to a type [Q in T]: Y if T is related to S and X' is
-            // related to Y, where X' is an instantiation of X in which P is replaced with Q. Notice
-            // that S and T are contra-variant whereas X and Y are co-variant.
-            function mappedTypeRelatedTo(source, target, reportErrors) {
-                var modifiersRelated = relation === comparableRelation || (relation === identityRelation ? getMappedTypeModifiers(source) === getMappedTypeModifiers(target) :
-                    getCombinedMappedTypeOptionality(source) <= getCombinedMappedTypeOptionality(target));
-                if (modifiersRelated) {
-                    var result_8;
-                    var targetConstraint = getConstraintTypeFromMappedType(target);
-                    var sourceConstraint = instantiateType(getConstraintTypeFromMappedType(source), makeFunctionTypeMapper(getCombinedMappedTypeOptionality(source) < 0 ? reportUnmeasurableMarkers : reportUnreliableMarkers));
-                    if (result_8 = isRelatedTo(targetConstraint, sourceConstraint, reportErrors)) {
-                        var mapper = createTypeMapper([getTypeParameterFromMappedType(source)], [getTypeParameterFromMappedType(target)]);
-                        return result_8 & isRelatedTo(instantiateType(getTemplateTypeFromMappedType(source), mapper), getTemplateTypeFromMappedType(target), reportErrors);
+            properties.push(lengthSymbol);
+            var type = createObjectType(8 /* Tuple */ | 4 /* Reference */);
+            type.typeParameters = typeParameters;
+            type.outerTypeParameters = undefined;
+            type.localTypeParameters = typeParameters;
+            type.instantiations = new ts.Map();
+            type.instantiations.set(getTypeListId(type.typeParameters), type);
+            type.target = type;
+            type.resolvedTypeArguments = type.typeParameters;
+            type.thisType = createTypeParameter();
+            type.thisType.isThisType = true;
+            type.thisType.constraint = type;
+            type.declaredProperties = properties;
+            type.declaredCallSignatures = ts.emptyArray;
+            type.declaredConstructSignatures = ts.emptyArray;
+            type.declaredStringIndexInfo = undefined;
+            type.declaredNumberIndexInfo = undefined;
+            type.elementFlags = elementFlags;
+            type.minLength = minLength;
+            type.fixedLength = fixedLength;
+            type.hasRestElement = !!(combinedFlags & 12 /* Variable */);
+            type.combinedFlags = combinedFlags;
+            type.readonly = readonly;
+            type.labeledElementDeclarations = namedMemberDeclarations;
+            return type;
+        }
+        function createNormalizedTypeReference(target, typeArguments) {
+            return target.objectFlags & 8 /* Tuple */ && target.combinedFlags & 8 /* Variadic */ ?
+                createNormalizedTupleType(target, typeArguments) :
+                createTypeReference(target, typeArguments);
+        }
+        function createNormalizedTupleType(target, elementTypes) {
+            var _a, _b, _c;
+            // Transform [A, ...(X | Y | Z)] into [A, ...X] | [A, ...Y] | [A, ...Z]
+            var unionIndex = ts.findIndex(elementTypes, function (t, i) { return !!(target.elementFlags[i] & 8 /* Variadic */ && t.flags & (131072 /* Never */ | 1048576 /* Union */)); });
+            if (unionIndex >= 0) {
+                return checkCrossProductUnion(ts.map(elementTypes, function (t, i) { return target.elementFlags[i] & 8 /* Variadic */ ? t : unknownType; })) ?
+                    mapType(elementTypes[unionIndex], function (t) { return createNormalizedTupleType(target, ts.replaceElement(elementTypes, unionIndex, t)); }) :
+                    errorType;
+            }
+            // If there are no variadic elements with non-generic types, just create a type reference with the same target type.
+            var spreadIndex = ts.findIndex(elementTypes, function (t, i) { return !!(target.elementFlags[i] & 8 /* Variadic */) && !(t.flags & 58982400 /* InstantiableNonPrimitive */) && !isGenericMappedType(t); });
+            if (spreadIndex < 0) {
+                return createTypeReference(target, elementTypes);
+            }
+            // We have non-generic variadic elements that need normalization.
+            var expandedTypes = [];
+            var expandedFlags = [];
+            var expandedDeclarations = [];
+            var optionalIndex = -1;
+            var restTypes;
+            var _loop_13 = function (i) {
+                var type = elementTypes[i];
+                var flags = target.elementFlags[i];
+                if (flags & 8 /* Variadic */) {
+                    if (type.flags & 58982400 /* InstantiableNonPrimitive */ || isGenericMappedType(type)) {
+                        // Generic variadic elements stay as they are (except following a rest element).
+                        addElementOrRest(type, 8 /* Variadic */, (_a = target.labeledElementDeclarations) === null || _a === void 0 ? void 0 : _a[i]);
+                    }
+                    else if (isTupleType(type)) {
+                        // Spread variadic elements with tuple types into the resulting tuple.
+                        ts.forEach(getTypeArguments(type), function (t, n) { var _a; return addElementOrRest(t, type.target.elementFlags[n], (_a = type.target.labeledElementDeclarations) === null || _a === void 0 ? void 0 : _a[n]); });
                     }
-                }
-                return 0 /* False */;
-            }
-            function typeRelatedToDiscriminatedType(source, target) {
-                // 1. Generate the combinations of discriminant properties & types 'source' can satisfy.
-                //    a. If the number of combinations is above a set limit, the comparison is too complex.
-                // 2. Filter 'target' to the subset of types whose discriminants exist in the matrix.
-                //    a. If 'target' does not satisfy all discriminants in the matrix, 'source' is not related.
-                // 3. For each type in the filtered 'target', determine if all non-discriminant properties of
-                //    'target' are related to a property in 'source'.
-                //
-                // NOTE: See ~/tests/cases/conformance/types/typeRelationships/assignmentCompatibility/assignmentCompatWithDiscriminatedUnion.ts
-                //       for examples.
-                var sourceProperties = getPropertiesOfType(source);
-                var sourcePropertiesFiltered = findDiscriminantProperties(sourceProperties, target);
-                if (!sourcePropertiesFiltered)
-                    return 0 /* False */;
-                // Though we could compute the number of combinations as we generate
-                // the matrix, this would incur additional memory overhead due to
-                // array allocations. To reduce this overhead, we first compute
-                // the number of combinations to ensure we will not surpass our
-                // fixed limit before incurring the cost of any allocations:
-                var numCombinations = 1;
-                for (var _i = 0, sourcePropertiesFiltered_1 = sourcePropertiesFiltered; _i < sourcePropertiesFiltered_1.length; _i++) {
-                    var sourceProperty = sourcePropertiesFiltered_1[_i];
-                    numCombinations *= countTypes(getTypeOfSymbol(sourceProperty));
-                    if (numCombinations > 25) {
-                        // We've reached the complexity limit.
-                        return 0 /* False */;
+                    else {
+                        // Treat everything else as an array type and create a rest element.
+                        addElementOrRest(isArrayLikeType(type) && getIndexTypeOfType(type, 1 /* Number */) || errorType, 4 /* Rest */, (_b = target.labeledElementDeclarations) === null || _b === void 0 ? void 0 : _b[i]);
                     }
                 }
-                // Compute the set of types for each discriminant property.
-                var sourceDiscriminantTypes = new Array(sourcePropertiesFiltered.length);
-                var excludedProperties = ts.createUnderscoreEscapedMap();
-                for (var i = 0; i < sourcePropertiesFiltered.length; i++) {
-                    var sourceProperty = sourcePropertiesFiltered[i];
-                    var sourcePropertyType = getTypeOfSymbol(sourceProperty);
-                    sourceDiscriminantTypes[i] = sourcePropertyType.flags & 1048576 /* Union */
-                        ? sourcePropertyType.types
-                        : [sourcePropertyType];
-                    excludedProperties.set(sourceProperty.escapedName, true);
+                else {
+                    // Copy other element kinds with no change.
+                    addElementOrRest(type, flags, (_c = target.labeledElementDeclarations) === null || _c === void 0 ? void 0 : _c[i]);
                 }
-                // Match each combination of the cartesian product of discriminant properties to one or more
-                // constituents of 'target'. If any combination does not have a match then 'source' is not relatable.
-                var discriminantCombinations = ts.cartesianProduct(sourceDiscriminantTypes);
-                var matchingTypes = [];
-                var _loop_14 = function (combination) {
-                    var hasMatch = false;
-                    outer: for (var _i = 0, _a = target.types; _i < _a.length; _i++) {
-                        var type = _a[_i];
-                        var _loop_15 = function (i) {
-                            var sourceProperty = sourcePropertiesFiltered[i];
-                            var targetProperty = getPropertyOfType(type, sourceProperty.escapedName);
-                            if (!targetProperty)
-                                return "continue-outer";
-                            if (sourceProperty === targetProperty)
-                                return "continue";
-                            // We compare the source property to the target in the context of a single discriminant type.
-                            var related = propertyRelatedTo(source, target, sourceProperty, targetProperty, function (_) { return combination[i]; }, /*reportErrors*/ false, 0 /* None */, /*skipOptional*/ strictNullChecks || relation === comparableRelation);
-                            // If the target property could not be found, or if the properties were not related,
-                            // then this constituent is not a match.
-                            if (!related) {
-                                return "continue-outer";
-                            }
-                        };
-                        for (var i = 0; i < sourcePropertiesFiltered.length; i++) {
-                            var state_7 = _loop_15(i);
-                            switch (state_7) {
-                                case "continue-outer": continue outer;
-                            }
+            };
+            for (var i = 0; i < elementTypes.length; i++) {
+                _loop_13(i);
+            }
+            if (restTypes) {
+                // Create a union of the collected rest element types.
+                expandedTypes[expandedTypes.length - 1] = getUnionType(restTypes);
+            }
+            var tupleTarget = getTupleTargetType(expandedFlags, target.readonly, expandedDeclarations);
+            return tupleTarget === emptyGenericType ? emptyObjectType :
+                expandedFlags.length ? createTypeReference(tupleTarget, expandedTypes) :
+                    tupleTarget;
+            function addElementOrRest(type, flags, declaration) {
+                if (restTypes) {
+                    // A rest element was previously added, so simply collect the type of this element.
+                    restTypes.push(flags & 8 /* Variadic */ ? getIndexedAccessType(type, numberType) : type);
+                }
+                else {
+                    if (flags & 1 /* Required */ && optionalIndex >= 0) {
+                        // Turn preceding optional elements into required elements
+                        for (var i = optionalIndex; i < expandedFlags.length; i++) {
+                            if (expandedFlags[i] & 2 /* Optional */)
+                                expandedFlags[i] = 1 /* Required */;
                         }
-                        ts.pushIfUnique(matchingTypes, type, ts.equateValues);
-                        hasMatch = true;
+                        optionalIndex = -1;
                     }
-                    if (!hasMatch) {
-                        return { value: 0 /* False */ };
+                    else if (flags & 2 /* Optional */ && optionalIndex < 0) {
+                        optionalIndex = expandedFlags.length;
                     }
-                };
-                for (var _a = 0, discriminantCombinations_1 = discriminantCombinations; _a < discriminantCombinations_1.length; _a++) {
-                    var combination = discriminantCombinations_1[_a];
-                    var state_6 = _loop_14(combination);
-                    if (typeof state_6 === "object")
-                        return state_6.value;
-                }
-                // Compare the remaining non-discriminant properties of each match.
-                var result = -1 /* True */;
-                for (var _b = 0, matchingTypes_1 = matchingTypes; _b < matchingTypes_1.length; _b++) {
-                    var type = matchingTypes_1[_b];
-                    result &= propertiesRelatedTo(source, type, /*reportErrors*/ false, excludedProperties, 0 /* None */);
-                    if (result) {
-                        result &= signaturesRelatedTo(source, type, 0 /* Call */, /*reportStructuralErrors*/ false);
-                        if (result) {
-                            result &= signaturesRelatedTo(source, type, 1 /* Construct */, /*reportStructuralErrors*/ false);
-                            if (result) {
-                                result &= indexTypesRelatedTo(source, type, 0 /* String */, /*sourceIsPrimitive*/ false, /*reportStructuralErrors*/ false, 0 /* None */);
-                                if (result) {
-                                    result &= indexTypesRelatedTo(source, type, 1 /* Number */, /*sourceIsPrimitive*/ false, /*reportStructuralErrors*/ false, 0 /* None */);
-                                }
-                            }
-                        }
+                    else if (flags & 4 /* Rest */) {
+                        // Start collecting element types when a rest element is added.
+                        restTypes = [type];
                     }
-                    if (!result) {
-                        return result;
+                    expandedTypes.push(type);
+                    expandedFlags.push(flags);
+                    if (expandedDeclarations && declaration) {
+                        expandedDeclarations.push(declaration);
+                    }
+                    else {
+                        expandedDeclarations = undefined;
                     }
                 }
-                return result;
             }
-            function excludeProperties(properties, excludedProperties) {
-                if (!excludedProperties || properties.length === 0)
-                    return properties;
-                var result;
-                for (var i = 0; i < properties.length; i++) {
-                    if (!excludedProperties.has(properties[i].escapedName)) {
-                        if (result) {
-                            result.push(properties[i]);
-                        }
+        }
+        function sliceTupleType(type, index, endSkipCount) {
+            if (endSkipCount === void 0) { endSkipCount = 0; }
+            var target = type.target;
+            var endIndex = getTypeReferenceArity(type) - endSkipCount;
+            return index > target.fixedLength ? getRestArrayTypeOfTupleType(type) || createTupleType(ts.emptyArray) :
+                createTupleType(getTypeArguments(type).slice(index, endIndex), target.elementFlags.slice(index, endIndex), 
+                /*readonly*/ false, target.labeledElementDeclarations && target.labeledElementDeclarations.slice(index, endIndex));
+        }
+        function getKnownKeysOfTupleType(type) {
+            return getUnionType(ts.append(ts.arrayOf(type.target.fixedLength, function (i) { return getLiteralType("" + i); }), getIndexType(type.target.readonly ? globalReadonlyArrayType : globalArrayType)));
+        }
+        function getTypeFromOptionalTypeNode(node) {
+            var type = getTypeFromTypeNode(node.type);
+            return strictNullChecks ? getOptionalType(type) : type;
+        }
+        function getTypeId(type) {
+            return type.id;
+        }
+        function containsType(types, type) {
+            return ts.binarySearch(types, type, getTypeId, ts.compareValues) >= 0;
+        }
+        function insertType(types, type) {
+            var index = ts.binarySearch(types, type, getTypeId, ts.compareValues);
+            if (index < 0) {
+                types.splice(~index, 0, type);
+                return true;
+            }
+            return false;
+        }
+        function addTypeToUnion(typeSet, includes, type) {
+            var flags = type.flags;
+            if (flags & 1048576 /* Union */) {
+                return addTypesToUnion(typeSet, includes, type.types);
+            }
+            // We ignore 'never' types in unions
+            if (!(flags & 131072 /* Never */)) {
+                includes |= flags & 205258751 /* IncludesMask */;
+                if (flags & 469499904 /* StructuredOrInstantiable */)
+                    includes |= 262144 /* IncludesStructuredOrInstantiable */;
+                if (type === wildcardType)
+                    includes |= 8388608 /* IncludesWildcard */;
+                if (!strictNullChecks && flags & 98304 /* Nullable */) {
+                    if (!(ts.getObjectFlags(type) & 524288 /* ContainsWideningType */))
+                        includes |= 4194304 /* IncludesNonWideningType */;
+                }
+                else {
+                    var len = typeSet.length;
+                    var index = len && type.id > typeSet[len - 1].id ? ~len : ts.binarySearch(typeSet, type, getTypeId, ts.compareValues);
+                    if (index < 0) {
+                        typeSet.splice(~index, 0, type);
                     }
-                    else if (!result) {
-                        result = properties.slice(0, i);
+                }
+            }
+            return includes;
+        }
+        // Add the given types to the given type set. Order is preserved, duplicates are removed,
+        // and nested types of the given kind are flattened into the set.
+        function addTypesToUnion(typeSet, includes, types) {
+            for (var _i = 0, types_10 = types; _i < types_10.length; _i++) {
+                var type = types_10[_i];
+                includes = addTypeToUnion(typeSet, includes, type);
+            }
+            return includes;
+        }
+        function isSetOfLiteralsFromSameEnum(types) {
+            var first = types[0];
+            if (first.flags & 1024 /* EnumLiteral */) {
+                var firstEnum = getParentOfSymbol(first.symbol);
+                for (var i = 1; i < types.length; i++) {
+                    var other = types[i];
+                    if (!(other.flags & 1024 /* EnumLiteral */) || (firstEnum !== getParentOfSymbol(other.symbol))) {
+                        return false;
                     }
                 }
-                return result || properties;
+                return true;
             }
-            function isPropertySymbolTypeRelated(sourceProp, targetProp, getTypeOfSourceProperty, reportErrors, intersectionState) {
-                var targetIsOptional = strictNullChecks && !!(ts.getCheckFlags(targetProp) & 48 /* Partial */);
-                var source = getTypeOfSourceProperty(sourceProp);
-                if (ts.getCheckFlags(targetProp) & 65536 /* DeferredType */ && !getSymbolLinks(targetProp).type) {
-                    // Rather than resolving (and normalizing) the type, relate constituent-by-constituent without performing normalization or seconadary passes
-                    var links = getSymbolLinks(targetProp);
-                    ts.Debug.assertIsDefined(links.deferralParent);
-                    ts.Debug.assertIsDefined(links.deferralConstituents);
-                    var unionParent = !!(links.deferralParent.flags & 1048576 /* Union */);
-                    var result_9 = unionParent ? 0 /* False */ : -1 /* True */;
-                    var targetTypes = links.deferralConstituents;
-                    for (var _i = 0, targetTypes_3 = targetTypes; _i < targetTypes_3.length; _i++) {
-                        var targetType = targetTypes_3[_i];
-                        var related = isRelatedTo(source, targetType, /*reportErrors*/ false, /*headMessage*/ undefined, unionParent ? 0 : 2 /* Target */);
-                        if (!unionParent) {
-                            if (!related) {
-                                // Can't assign to a target individually - have to fallback to assigning to the _whole_ intersection (which forces normalization)
-                                return isRelatedTo(source, addOptionality(getTypeOfSymbol(targetProp), targetIsOptional), reportErrors);
+            return false;
+        }
+        function removeSubtypes(types, primitivesOnly) {
+            var len = types.length;
+            if (len === 0 || isSetOfLiteralsFromSameEnum(types)) {
+                return true;
+            }
+            var i = len;
+            var count = 0;
+            while (i > 0) {
+                i--;
+                var source = types[i];
+                for (var _i = 0, types_11 = types; _i < types_11.length; _i++) {
+                    var target = types_11[_i];
+                    if (source !== target) {
+                        if (count === 100000) {
+                            // After 100000 subtype checks we estimate the remaining amount of work by assuming the
+                            // same ratio of checks per element. If the estimated number of remaining type checks is
+                            // greater than an upper limit we deem the union type too complex to represent. The
+                            // upper limit is 25M for unions of primitives only, and 1M otherwise. This for example
+                            // caps union types at 5000 unique literal types and 1000 unique object types.
+                            var estimatedCount = (count / (len - i)) * len;
+                            if (estimatedCount > (primitivesOnly ? 25000000 : 1000000)) {
+                                ts.tracing.instant("check" /* Check */, "removeSubtypes_DepthLimit", { typeIds: types.map(function (t) { return t.id; }) });
+                                error(currentNode, ts.Diagnostics.Expression_produces_a_union_type_that_is_too_complex_to_represent);
+                                return false;
                             }
-                            result_9 &= related;
                         }
-                        else {
-                            if (related) {
-                                return related;
-                            }
+                        count++;
+                        if (isTypeRelatedTo(source, target, strictSubtypeRelation) && (!(ts.getObjectFlags(getTargetType(source)) & 1 /* Class */) ||
+                            !(ts.getObjectFlags(getTargetType(target)) & 1 /* Class */) ||
+                            isTypeDerivedFrom(source, target))) {
+                            ts.orderedRemoveItemAt(types, i);
+                            break;
                         }
                     }
-                    if (unionParent && !result_9 && targetIsOptional) {
-                        result_9 = isRelatedTo(source, undefinedType);
-                    }
-                    if (unionParent && !result_9 && reportErrors) {
-                        // The easiest way to get the right errors here is to un-defer (which may be costly)
-                        // If it turns out this is too costly too often, we can replicate the error handling logic within
-                        // typeRelatedToSomeType without the discriminatable type branch (as that requires a manifest union
-                        // type on which to hand discriminable properties, which we are expressly trying to avoid here)
-                        return isRelatedTo(source, addOptionality(getTypeOfSymbol(targetProp), targetIsOptional), reportErrors);
-                    }
-                    return result_9;
                 }
-                else {
-                    return isRelatedTo(source, addOptionality(getTypeOfSymbol(targetProp), targetIsOptional), reportErrors, /*headMessage*/ undefined, intersectionState);
+            }
+            return true;
+        }
+        function removeRedundantLiteralTypes(types, includes) {
+            var i = types.length;
+            while (i > 0) {
+                i--;
+                var t = types[i];
+                var remove = t.flags & 128 /* StringLiteral */ && includes & 4 /* String */ ||
+                    t.flags & 256 /* NumberLiteral */ && includes & 8 /* Number */ ||
+                    t.flags & 2048 /* BigIntLiteral */ && includes & 64 /* BigInt */ ||
+                    t.flags & 8192 /* UniqueESSymbol */ && includes & 4096 /* ESSymbol */ ||
+                    isFreshLiteralType(t) && containsType(types, t.regularType);
+                if (remove) {
+                    ts.orderedRemoveItemAt(types, i);
                 }
             }
-            function propertyRelatedTo(source, target, sourceProp, targetProp, getTypeOfSourceProperty, reportErrors, intersectionState, skipOptional) {
-                var sourcePropFlags = ts.getDeclarationModifierFlagsFromSymbol(sourceProp);
-                var targetPropFlags = ts.getDeclarationModifierFlagsFromSymbol(targetProp);
-                if (sourcePropFlags & 8 /* Private */ || targetPropFlags & 8 /* Private */) {
-                    if (sourceProp.valueDeclaration !== targetProp.valueDeclaration) {
-                        if (reportErrors) {
-                            if (sourcePropFlags & 8 /* Private */ && targetPropFlags & 8 /* Private */) {
-                                reportError(ts.Diagnostics.Types_have_separate_declarations_of_a_private_property_0, symbolToString(targetProp));
-                            }
-                            else {
-                                reportError(ts.Diagnostics.Property_0_is_private_in_type_1_but_not_in_type_2, symbolToString(targetProp), typeToString(sourcePropFlags & 8 /* Private */ ? source : target), typeToString(sourcePropFlags & 8 /* Private */ ? target : source));
-                            }
-                        }
-                        return 0 /* False */;
+        }
+        function removeStringLiteralsMatchedByTemplateLiterals(types) {
+            var templates = ts.filter(types, isPatternLiteralType);
+            if (templates.length) {
+                var i = types.length;
+                var _loop_14 = function () {
+                    i--;
+                    var t = types[i];
+                    if (t.flags & 128 /* StringLiteral */ && ts.some(templates, function (template) { return isTypeSubtypeOf(t, template); })) {
+                        ts.orderedRemoveItemAt(types, i);
                     }
+                };
+                while (i > 0) {
+                    _loop_14();
                 }
-                else if (targetPropFlags & 16 /* Protected */) {
-                    if (!isValidOverrideOf(sourceProp, targetProp)) {
-                        if (reportErrors) {
-                            reportError(ts.Diagnostics.Property_0_is_protected_but_type_1_is_not_a_class_derived_from_2, symbolToString(targetProp), typeToString(getDeclaringClass(sourceProp) || source), typeToString(getDeclaringClass(targetProp) || target));
+            }
+        }
+        // We sort and deduplicate the constituent types based on object identity. If the subtypeReduction
+        // flag is specified we also reduce the constituent type set to only include types that aren't subtypes
+        // of other types. Subtype reduction is expensive for large union types and is possible only when union
+        // types are known not to circularly reference themselves (as is the case with union types created by
+        // expression constructs such as array literals and the || and ?: operators). Named types can
+        // circularly reference themselves and therefore cannot be subtype reduced during their declaration.
+        // For example, "type Item = string | (() => Item" is a named type that circularly references itself.
+        function getUnionType(types, unionReduction, aliasSymbol, aliasTypeArguments) {
+            if (unionReduction === void 0) { unionReduction = 1 /* Literal */; }
+            if (types.length === 0) {
+                return neverType;
+            }
+            if (types.length === 1) {
+                return types[0];
+            }
+            var typeSet = [];
+            var includes = addTypesToUnion(typeSet, 0, types);
+            if (unionReduction !== 0 /* None */) {
+                if (includes & 3 /* AnyOrUnknown */) {
+                    return includes & 1 /* Any */ ? includes & 8388608 /* IncludesWildcard */ ? wildcardType : anyType : unknownType;
+                }
+                switch (unionReduction) {
+                    case 1 /* Literal */:
+                        if (includes & (2944 /* Literal */ | 8192 /* UniqueESSymbol */)) {
+                            removeRedundantLiteralTypes(typeSet, includes);
                         }
-                        return 0 /* False */;
-                    }
+                        if (includes & 128 /* StringLiteral */ && includes & 134217728 /* TemplateLiteral */) {
+                            removeStringLiteralsMatchedByTemplateLiterals(typeSet);
+                        }
+                        break;
+                    case 2 /* Subtype */:
+                        if (!removeSubtypes(typeSet, !(includes & 262144 /* IncludesStructuredOrInstantiable */))) {
+                            return errorType;
+                        }
+                        break;
                 }
-                else if (sourcePropFlags & 16 /* Protected */) {
-                    if (reportErrors) {
-                        reportError(ts.Diagnostics.Property_0_is_protected_in_type_1_but_public_in_type_2, symbolToString(targetProp), typeToString(source), typeToString(target));
-                    }
-                    return 0 /* False */;
+                if (typeSet.length === 0) {
+                    return includes & 65536 /* Null */ ? includes & 4194304 /* IncludesNonWideningType */ ? nullType : nullWideningType :
+                        includes & 32768 /* Undefined */ ? includes & 4194304 /* IncludesNonWideningType */ ? undefinedType : undefinedWideningType :
+                            neverType;
                 }
-                // If the target comes from a partial union prop, allow `undefined` in the target type
-                var related = isPropertySymbolTypeRelated(sourceProp, targetProp, getTypeOfSourceProperty, reportErrors, intersectionState);
-                if (!related) {
-                    if (reportErrors) {
-                        reportIncompatibleError(ts.Diagnostics.Types_of_property_0_are_incompatible, symbolToString(targetProp));
-                    }
-                    return 0 /* False */;
+            }
+            var objectFlags = (includes & 469647395 /* NotPrimitiveUnion */ ? 0 : 262144 /* PrimitiveUnion */) |
+                (includes & 2097152 /* Intersection */ ? 268435456 /* ContainsIntersections */ : 0);
+            return getUnionTypeFromSortedList(typeSet, objectFlags, aliasSymbol, aliasTypeArguments);
+        }
+        function getUnionTypePredicate(signatures) {
+            var first;
+            var types = [];
+            for (var _i = 0, signatures_6 = signatures; _i < signatures_6.length; _i++) {
+                var sig = signatures_6[_i];
+                var pred = getTypePredicateOfSignature(sig);
+                if (!pred || pred.kind === 2 /* AssertsThis */ || pred.kind === 3 /* AssertsIdentifier */) {
+                    continue;
                 }
-                // When checking for comparability, be more lenient with optional properties.
-                if (!skipOptional && sourceProp.flags & 16777216 /* Optional */ && !(targetProp.flags & 16777216 /* Optional */)) {
-                    // TypeScript 1.0 spec (April 2014): 3.8.3
-                    // S is a subtype of a type T, and T is a supertype of S if ...
-                    // S' and T are object types and, for each member M in T..
-                    // M is a property and S' contains a property N where
-                    // if M is a required property, N is also a required property
-                    // (M - property in T)
-                    // (N - property in S)
-                    if (reportErrors) {
-                        reportError(ts.Diagnostics.Property_0_is_optional_in_type_1_but_required_in_type_2, symbolToString(targetProp), typeToString(source), typeToString(target));
+                if (first) {
+                    if (!typePredicateKindsMatch(first, pred)) {
+                        // No common type predicate.
+                        return undefined;
                     }
-                    return 0 /* False */;
                 }
-                return related;
+                else {
+                    first = pred;
+                }
+                types.push(pred.type);
             }
-            function reportUnmatchedProperty(source, target, unmatchedProperty, requireOptionalProperties) {
-                var shouldSkipElaboration = false;
-                // give specific error in case where private names have the same description
-                if (unmatchedProperty.valueDeclaration
-                    && ts.isNamedDeclaration(unmatchedProperty.valueDeclaration)
-                    && ts.isPrivateIdentifier(unmatchedProperty.valueDeclaration.name)
-                    && source.symbol
-                    && source.symbol.flags & 32 /* Class */) {
-                    var privateIdentifierDescription = unmatchedProperty.valueDeclaration.name.escapedText;
-                    var symbolTableKey = ts.getSymbolNameForPrivateIdentifier(source.symbol, privateIdentifierDescription);
-                    if (symbolTableKey && getPropertyOfType(source, symbolTableKey)) {
-                        var sourceName = ts.getDeclarationName(source.symbol.valueDeclaration);
-                        var targetName = ts.getDeclarationName(target.symbol.valueDeclaration);
-                        reportError(ts.Diagnostics.Property_0_in_type_1_refers_to_a_different_member_that_cannot_be_accessed_from_within_type_2, diagnosticName(privateIdentifierDescription), diagnosticName(sourceName.escapedText === "" ? anon : sourceName), diagnosticName(targetName.escapedText === "" ? anon : targetName));
-                        return;
-                    }
+            if (!first) {
+                // No union signatures had a type predicate.
+                return undefined;
+            }
+            var unionType = getUnionType(types);
+            return createTypePredicate(first.kind, first.parameterName, first.parameterIndex, unionType);
+        }
+        function typePredicateKindsMatch(a, b) {
+            return a.kind === b.kind && a.parameterIndex === b.parameterIndex;
+        }
+        // This function assumes the constituent type list is sorted and deduplicated.
+        function getUnionTypeFromSortedList(types, objectFlags, aliasSymbol, aliasTypeArguments) {
+            if (types.length === 0) {
+                return neverType;
+            }
+            if (types.length === 1) {
+                return types[0];
+            }
+            var id = getTypeListId(types);
+            var type = unionTypes.get(id);
+            if (!type) {
+                type = createType(1048576 /* Union */);
+                unionTypes.set(id, type);
+                type.objectFlags = objectFlags | getPropagatingFlagsOfTypes(types, /*excludeKinds*/ 98304 /* Nullable */);
+                type.types = types;
+                /*
+                Note: This is the alias symbol (or lack thereof) that we see when we first encounter this union type.
+                For aliases of identical unions, eg `type T = A | B; type U = A | B`, the symbol of the first alias encountered is the aliasSymbol.
+                (In the language service, the order may depend on the order in which a user takes actions, such as hovering over symbols.)
+                It's important that we create equivalent union types only once, so that's an unfortunate side effect.
+                */
+                type.aliasSymbol = aliasSymbol;
+                type.aliasTypeArguments = aliasTypeArguments;
+            }
+            return type;
+        }
+        function getTypeFromUnionTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                var aliasSymbol = getAliasSymbolForTypeNode(node);
+                links.resolvedType = getUnionType(ts.map(node.types, getTypeFromTypeNode), 1 /* Literal */, aliasSymbol, getTypeArgumentsForAliasSymbol(aliasSymbol));
+            }
+            return links.resolvedType;
+        }
+        function addTypeToIntersection(typeSet, includes, type) {
+            var flags = type.flags;
+            if (flags & 2097152 /* Intersection */) {
+                return addTypesToIntersection(typeSet, includes, type.types);
+            }
+            if (isEmptyAnonymousObjectType(type)) {
+                if (!(includes & 16777216 /* IncludesEmptyObject */)) {
+                    includes |= 16777216 /* IncludesEmptyObject */;
+                    typeSet.set(type.id.toString(), type);
                 }
-                var props = ts.arrayFrom(getUnmatchedProperties(source, target, requireOptionalProperties, /*matchDiscriminantProperties*/ false));
-                if (!headMessage || (headMessage.code !== ts.Diagnostics.Class_0_incorrectly_implements_interface_1.code &&
-                    headMessage.code !== ts.Diagnostics.Class_0_incorrectly_implements_class_1_Did_you_mean_to_extend_1_and_inherit_its_members_as_a_subclass.code)) {
-                    shouldSkipElaboration = true; // Retain top-level error for interface implementing issues, otherwise omit it
+            }
+            else {
+                if (flags & 3 /* AnyOrUnknown */) {
+                    if (type === wildcardType)
+                        includes |= 8388608 /* IncludesWildcard */;
                 }
-                if (props.length === 1) {
-                    var propName = symbolToString(unmatchedProperty);
-                    reportError.apply(void 0, __spreadArrays([ts.Diagnostics.Property_0_is_missing_in_type_1_but_required_in_type_2, propName], getTypeNamesForErrorDisplay(source, target)));
-                    if (ts.length(unmatchedProperty.declarations)) {
-                        associateRelatedInfo(ts.createDiagnosticForNode(unmatchedProperty.declarations[0], ts.Diagnostics._0_is_declared_here, propName));
-                    }
-                    if (shouldSkipElaboration && errorInfo) {
-                        overrideNextErrorInfo++;
+                else if ((strictNullChecks || !(flags & 98304 /* Nullable */)) && !typeSet.has(type.id.toString())) {
+                    if (type.flags & 109440 /* Unit */ && includes & 109440 /* Unit */) {
+                        // We have seen two distinct unit types which means we should reduce to an
+                        // empty intersection. Adding TypeFlags.NonPrimitive causes that to happen.
+                        includes |= 67108864 /* NonPrimitive */;
                     }
+                    typeSet.set(type.id.toString(), type);
                 }
-                else if (tryElaborateArrayLikeErrors(source, target, /*reportErrors*/ false)) {
-                    if (props.length > 5) { // arbitrary cutoff for too-long list form
-                        reportError(ts.Diagnostics.Type_0_is_missing_the_following_properties_from_type_1_Colon_2_and_3_more, typeToString(source), typeToString(target), ts.map(props.slice(0, 4), function (p) { return symbolToString(p); }).join(", "), props.length - 4);
+                includes |= flags & 205258751 /* IncludesMask */;
+            }
+            return includes;
+        }
+        // Add the given types to the given type set. Order is preserved, freshness is removed from literal
+        // types, duplicates are removed, and nested types of the given kind are flattened into the set.
+        function addTypesToIntersection(typeSet, includes, types) {
+            for (var _i = 0, types_12 = types; _i < types_12.length; _i++) {
+                var type = types_12[_i];
+                includes = addTypeToIntersection(typeSet, includes, getRegularTypeOfLiteralType(type));
+            }
+            return includes;
+        }
+        function removeRedundantPrimitiveTypes(types, includes) {
+            var i = types.length;
+            while (i > 0) {
+                i--;
+                var t = types[i];
+                var remove = t.flags & 4 /* String */ && includes & 128 /* StringLiteral */ ||
+                    t.flags & 8 /* Number */ && includes & 256 /* NumberLiteral */ ||
+                    t.flags & 64 /* BigInt */ && includes & 2048 /* BigIntLiteral */ ||
+                    t.flags & 4096 /* ESSymbol */ && includes & 8192 /* UniqueESSymbol */;
+                if (remove) {
+                    ts.orderedRemoveItemAt(types, i);
+                }
+            }
+        }
+        // Check that the given type has a match in every union. A given type is matched by
+        // an identical type, and a literal type is additionally matched by its corresponding
+        // primitive type.
+        function eachUnionContains(unionTypes, type) {
+            for (var _i = 0, unionTypes_1 = unionTypes; _i < unionTypes_1.length; _i++) {
+                var u = unionTypes_1[_i];
+                if (!containsType(u.types, type)) {
+                    var primitive = type.flags & 128 /* StringLiteral */ ? stringType :
+                        type.flags & 256 /* NumberLiteral */ ? numberType :
+                            type.flags & 2048 /* BigIntLiteral */ ? bigintType :
+                                type.flags & 8192 /* UniqueESSymbol */ ? esSymbolType :
+                                    undefined;
+                    if (!primitive || !containsType(u.types, primitive)) {
+                        return false;
                     }
-                    else {
-                        reportError(ts.Diagnostics.Type_0_is_missing_the_following_properties_from_type_1_Colon_2, typeToString(source), typeToString(target), ts.map(props, function (p) { return symbolToString(p); }).join(", "));
+                }
+            }
+            return true;
+        }
+        /**
+         * Returns `true` if the intersection of the template literals and string literals is the empty set, eg `get${string}` & "setX", and should reduce to `never`
+         */
+        function extractRedundantTemplateLiterals(types) {
+            var i = types.length;
+            var literals = ts.filter(types, function (t) { return !!(t.flags & 128 /* StringLiteral */); });
+            while (i > 0) {
+                i--;
+                var t = types[i];
+                if (!(t.flags & 134217728 /* TemplateLiteral */))
+                    continue;
+                for (var _i = 0, literals_1 = literals; _i < literals_1.length; _i++) {
+                    var t2 = literals_1[_i];
+                    if (isTypeSubtypeOf(t2, t)) {
+                        // eg, ``get${T}` & "getX"` is just `"getX"`
+                        ts.orderedRemoveItemAt(types, i);
+                        break;
                     }
-                    if (shouldSkipElaboration && errorInfo) {
-                        overrideNextErrorInfo++;
+                    else if (isPatternLiteralType(t)) {
+                        return true;
                     }
                 }
-                // No array like or unmatched property error - just issue top level error (errorInfo = undefined)
             }
-            function propertiesRelatedTo(source, target, reportErrors, excludedProperties, intersectionState) {
-                if (relation === identityRelation) {
-                    return propertiesIdenticalTo(source, target, excludedProperties);
+            return false;
+        }
+        function extractIrreducible(types, flag) {
+            if (ts.every(types, function (t) { return !!(t.flags & 1048576 /* Union */) && ts.some(t.types, function (tt) { return !!(tt.flags & flag); }); })) {
+                for (var i = 0; i < types.length; i++) {
+                    types[i] = filterType(types[i], function (t) { return !(t.flags & flag); });
                 }
-                var requireOptionalProperties = (relation === subtypeRelation || relation === strictSubtypeRelation) && !isObjectLiteralType(source) && !isEmptyArrayLiteralType(source) && !isTupleType(source);
-                var unmatchedProperty = getUnmatchedProperty(source, target, requireOptionalProperties, /*matchDiscriminantProperties*/ false);
-                if (unmatchedProperty) {
-                    if (reportErrors) {
-                        reportUnmatchedProperty(source, target, unmatchedProperty, requireOptionalProperties);
-                    }
-                    return 0 /* False */;
+                return true;
+            }
+            return false;
+        }
+        // If the given list of types contains more than one union of primitive types, replace the
+        // first with a union containing an intersection of those primitive types, then remove the
+        // other unions and return true. Otherwise, do nothing and return false.
+        function intersectUnionsOfPrimitiveTypes(types) {
+            var unionTypes;
+            var index = ts.findIndex(types, function (t) { return !!(ts.getObjectFlags(t) & 262144 /* PrimitiveUnion */); });
+            if (index < 0) {
+                return false;
+            }
+            var i = index + 1;
+            // Remove all but the first union of primitive types and collect them in
+            // the unionTypes array.
+            while (i < types.length) {
+                var t = types[i];
+                if (ts.getObjectFlags(t) & 262144 /* PrimitiveUnion */) {
+                    (unionTypes || (unionTypes = [types[index]])).push(t);
+                    ts.orderedRemoveItemAt(types, i);
                 }
-                if (isObjectLiteralType(target)) {
-                    for (var _i = 0, _a = excludeProperties(getPropertiesOfType(source), excludedProperties); _i < _a.length; _i++) {
-                        var sourceProp = _a[_i];
-                        if (!getPropertyOfObjectType(target, sourceProp.escapedName)) {
-                            var sourceType = getTypeOfSymbol(sourceProp);
-                            if (!(sourceType === undefinedType || sourceType === undefinedWideningType || sourceType === optionalType)) {
-                                if (reportErrors) {
-                                    reportError(ts.Diagnostics.Property_0_does_not_exist_on_type_1, symbolToString(sourceProp), typeToString(target));
-                                }
-                                return 0 /* False */;
-                            }
-                        }
-                    }
+                else {
+                    i++;
                 }
-                var result = -1 /* True */;
-                if (isTupleType(target)) {
-                    var targetRestType = getRestTypeOfTupleType(target);
-                    if (targetRestType) {
-                        if (!isTupleType(source)) {
-                            return 0 /* False */;
-                        }
-                        var sourceRestType = getRestTypeOfTupleType(source);
-                        if (sourceRestType && !isRelatedTo(sourceRestType, targetRestType, reportErrors)) {
-                            if (reportErrors) {
-                                reportError(ts.Diagnostics.Rest_signatures_are_incompatible);
-                            }
-                            return 0 /* False */;
-                        }
-                        var targetCount = getTypeReferenceArity(target) - 1;
-                        var sourceCount = getTypeReferenceArity(source) - (sourceRestType ? 1 : 0);
-                        var sourceTypeArguments = getTypeArguments(source);
-                        for (var i = targetCount; i < sourceCount; i++) {
-                            var related = isRelatedTo(sourceTypeArguments[i], targetRestType, reportErrors);
-                            if (!related) {
-                                if (reportErrors) {
-                                    reportError(ts.Diagnostics.Property_0_is_incompatible_with_rest_element_type, "" + i);
-                                }
-                                return 0 /* False */;
-                            }
-                            result &= related;
+            }
+            // Return false if there was only one union of primitive types
+            if (!unionTypes) {
+                return false;
+            }
+            // We have more than one union of primitive types, now intersect them. For each
+            // type in each union we check if the type is matched in every union and if so
+            // we include it in the result.
+            var checked = [];
+            var result = [];
+            for (var _i = 0, unionTypes_2 = unionTypes; _i < unionTypes_2.length; _i++) {
+                var u = unionTypes_2[_i];
+                for (var _a = 0, _b = u.types; _a < _b.length; _a++) {
+                    var t = _b[_a];
+                    if (insertType(checked, t)) {
+                        if (eachUnionContains(unionTypes, t)) {
+                            insertType(result, t);
                         }
                     }
                 }
-                // We only call this for union target types when we're attempting to do excess property checking - in those cases, we want to get _all possible props_
-                // from the target union, across all members
-                var properties = getPropertiesOfType(target);
-                var numericNamesOnly = isTupleType(source) && isTupleType(target);
-                for (var _b = 0, _c = excludeProperties(properties, excludedProperties); _b < _c.length; _b++) {
-                    var targetProp = _c[_b];
-                    var name = targetProp.escapedName;
-                    if (!(targetProp.flags & 4194304 /* Prototype */) && (!numericNamesOnly || isNumericLiteralName(name) || name === "length")) {
-                        var sourceProp = getPropertyOfType(source, name);
-                        if (sourceProp && sourceProp !== targetProp) {
-                            var related = propertyRelatedTo(source, target, sourceProp, targetProp, getTypeOfSymbol, reportErrors, intersectionState, relation === comparableRelation);
-                            if (!related) {
-                                return 0 /* False */;
-                            }
-                            result &= related;
+            }
+            // Finally replace the first union with the result
+            types[index] = getUnionTypeFromSortedList(result, 262144 /* PrimitiveUnion */);
+            return true;
+        }
+        function createIntersectionType(types, aliasSymbol, aliasTypeArguments) {
+            var result = createType(2097152 /* Intersection */);
+            result.objectFlags = getPropagatingFlagsOfTypes(types, /*excludeKinds*/ 98304 /* Nullable */);
+            result.types = types;
+            result.aliasSymbol = aliasSymbol; // See comment in `getUnionTypeFromSortedList`.
+            result.aliasTypeArguments = aliasTypeArguments;
+            return result;
+        }
+        // We normalize combinations of intersection and union types based on the distributive property of the '&'
+        // operator. Specifically, because X & (A | B) is equivalent to X & A | X & B, we can transform intersection
+        // types with union type constituents into equivalent union types with intersection type constituents and
+        // effectively ensure that union types are always at the top level in type representations.
+        //
+        // We do not perform structural deduplication on intersection types. Intersection types are created only by the &
+        // type operator and we can't reduce those because we want to support recursive intersection types. For example,
+        // a type alias of the form "type List<T> = T & { next: List<T> }" cannot be reduced during its declaration.
+        // Also, unlike union types, the order of the constituent types is preserved in order that overload resolution
+        // for intersections of types with signatures can be deterministic.
+        function getIntersectionType(types, aliasSymbol, aliasTypeArguments) {
+            var typeMembershipMap = new ts.Map();
+            var includes = addTypesToIntersection(typeMembershipMap, 0, types);
+            var typeSet = ts.arrayFrom(typeMembershipMap.values());
+            // An intersection type is considered empty if it contains
+            // the type never, or
+            // more than one unit type or,
+            // an object type and a nullable type (null or undefined), or
+            // a string-like type and a type known to be non-string-like, or
+            // a number-like type and a type known to be non-number-like, or
+            // a symbol-like type and a type known to be non-symbol-like, or
+            // a void-like type and a type known to be non-void-like, or
+            // a non-primitive type and a type known to be primitive.
+            if (includes & 131072 /* Never */ ||
+                strictNullChecks && includes & 98304 /* Nullable */ && includes & (524288 /* Object */ | 67108864 /* NonPrimitive */ | 16777216 /* IncludesEmptyObject */) ||
+                includes & 67108864 /* NonPrimitive */ && includes & (469892092 /* DisjointDomains */ & ~67108864 /* NonPrimitive */) ||
+                includes & 402653316 /* StringLike */ && includes & (469892092 /* DisjointDomains */ & ~402653316 /* StringLike */) ||
+                includes & 296 /* NumberLike */ && includes & (469892092 /* DisjointDomains */ & ~296 /* NumberLike */) ||
+                includes & 2112 /* BigIntLike */ && includes & (469892092 /* DisjointDomains */ & ~2112 /* BigIntLike */) ||
+                includes & 12288 /* ESSymbolLike */ && includes & (469892092 /* DisjointDomains */ & ~12288 /* ESSymbolLike */) ||
+                includes & 49152 /* VoidLike */ && includes & (469892092 /* DisjointDomains */ & ~49152 /* VoidLike */)) {
+                return neverType;
+            }
+            if (includes & 134217728 /* TemplateLiteral */ && includes & 128 /* StringLiteral */ && extractRedundantTemplateLiterals(typeSet)) {
+                return neverType;
+            }
+            if (includes & 1 /* Any */) {
+                return includes & 8388608 /* IncludesWildcard */ ? wildcardType : anyType;
+            }
+            if (!strictNullChecks && includes & 98304 /* Nullable */) {
+                return includes & 32768 /* Undefined */ ? undefinedType : nullType;
+            }
+            if (includes & 4 /* String */ && includes & 128 /* StringLiteral */ ||
+                includes & 8 /* Number */ && includes & 256 /* NumberLiteral */ ||
+                includes & 64 /* BigInt */ && includes & 2048 /* BigIntLiteral */ ||
+                includes & 4096 /* ESSymbol */ && includes & 8192 /* UniqueESSymbol */) {
+                removeRedundantPrimitiveTypes(typeSet, includes);
+            }
+            if (includes & 16777216 /* IncludesEmptyObject */ && includes & 524288 /* Object */) {
+                ts.orderedRemoveItemAt(typeSet, ts.findIndex(typeSet, isEmptyAnonymousObjectType));
+            }
+            if (typeSet.length === 0) {
+                return unknownType;
+            }
+            if (typeSet.length === 1) {
+                return typeSet[0];
+            }
+            var id = getTypeListId(typeSet);
+            var result = intersectionTypes.get(id);
+            if (!result) {
+                if (includes & 1048576 /* Union */) {
+                    if (intersectUnionsOfPrimitiveTypes(typeSet)) {
+                        // When the intersection creates a reduced set (which might mean that *all* union types have
+                        // disappeared), we restart the operation to get a new set of combined flags. Once we have
+                        // reduced we'll never reduce again, so this occurs at most once.
+                        result = getIntersectionType(typeSet, aliasSymbol, aliasTypeArguments);
+                    }
+                    else if (extractIrreducible(typeSet, 32768 /* Undefined */)) {
+                        result = getUnionType([getIntersectionType(typeSet), undefinedType], 1 /* Literal */, aliasSymbol, aliasTypeArguments);
+                    }
+                    else if (extractIrreducible(typeSet, 65536 /* Null */)) {
+                        result = getUnionType([getIntersectionType(typeSet), nullType], 1 /* Literal */, aliasSymbol, aliasTypeArguments);
+                    }
+                    else {
+                        // We are attempting to construct a type of the form X & (A | B) & Y. Transform this into a type of
+                        // the form X & A & Y | X & B & Y and recursively reduce until no union type constituents remain.
+                        // If the estimated size of the resulting union type exceeds 100000 constituents, report an error.
+                        if (!checkCrossProductUnion(typeSet)) {
+                            return errorType;
                         }
+                        var unionIndex_1 = ts.findIndex(typeSet, function (t) { return (t.flags & 1048576 /* Union */) !== 0; });
+                        var unionType = typeSet[unionIndex_1];
+                        result = getUnionType(ts.map(unionType.types, function (t) { return getIntersectionType(ts.replaceElement(typeSet, unionIndex_1, t)); }), 1 /* Literal */, aliasSymbol, aliasTypeArguments);
                     }
                 }
-                return result;
-            }
-            function propertiesIdenticalTo(source, target, excludedProperties) {
-                if (!(source.flags & 524288 /* Object */ && target.flags & 524288 /* Object */)) {
-                    return 0 /* False */;
-                }
-                var sourceProperties = excludeProperties(getPropertiesOfObjectType(source), excludedProperties);
-                var targetProperties = excludeProperties(getPropertiesOfObjectType(target), excludedProperties);
-                if (sourceProperties.length !== targetProperties.length) {
-                    return 0 /* False */;
+                else {
+                    result = createIntersectionType(typeSet, aliasSymbol, aliasTypeArguments);
                 }
-                var result = -1 /* True */;
-                for (var _i = 0, sourceProperties_1 = sourceProperties; _i < sourceProperties_1.length; _i++) {
-                    var sourceProp = sourceProperties_1[_i];
-                    var targetProp = getPropertyOfObjectType(target, sourceProp.escapedName);
-                    if (!targetProp) {
-                        return 0 /* False */;
+                intersectionTypes.set(id, result);
+            }
+            return result;
+        }
+        function checkCrossProductUnion(types) {
+            var size = ts.reduceLeft(types, function (n, t) { return n * (t.flags & 1048576 /* Union */ ? t.types.length : t.flags & 131072 /* Never */ ? 0 : 1); }, 1);
+            if (size >= 100000) {
+                ts.tracing.instant("check" /* Check */, "checkCrossProductUnion_DepthLimit", { typeIds: types.map(function (t) { return t.id; }), size: size });
+                error(currentNode, ts.Diagnostics.Expression_produces_a_union_type_that_is_too_complex_to_represent);
+                return false;
+            }
+            return true;
+        }
+        function getTypeFromIntersectionTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                var aliasSymbol = getAliasSymbolForTypeNode(node);
+                links.resolvedType = getIntersectionType(ts.map(node.types, getTypeFromTypeNode), aliasSymbol, getTypeArgumentsForAliasSymbol(aliasSymbol));
+            }
+            return links.resolvedType;
+        }
+        function createIndexType(type, stringsOnly) {
+            var result = createType(4194304 /* Index */);
+            result.type = type;
+            result.stringsOnly = stringsOnly;
+            return result;
+        }
+        function getIndexTypeForGenericType(type, stringsOnly) {
+            return stringsOnly ?
+                type.resolvedStringIndexType || (type.resolvedStringIndexType = createIndexType(type, /*stringsOnly*/ true)) :
+                type.resolvedIndexType || (type.resolvedIndexType = createIndexType(type, /*stringsOnly*/ false));
+        }
+        function getIndexTypeForMappedType(type, noIndexSignatures) {
+            var constraint = filterType(getConstraintTypeFromMappedType(type), function (t) { return !(noIndexSignatures && t.flags & (1 /* Any */ | 4 /* String */)); });
+            var nameType = type.declaration.nameType && getTypeFromTypeNode(type.declaration.nameType);
+            return nameType ?
+                mapType(constraint, function (t) { return instantiateType(nameType, appendTypeMapping(type.mapper, getTypeParameterFromMappedType(type), t)); }) :
+                constraint;
+        }
+        // Ordinarily we reduce a keyof M where M is a mapped type { [P in K as N<P>]: X } to simply N<K>. This however presumes
+        // that N distributes over union types, i.e. that N<A | B | C> is equivalent to N<A> | N<B> | N<C>. That presumption is
+        // generally true, except when N is a non-distributive conditional type or an instantiable type with non-distributive
+        // conditional type as a constituent. In those cases, we cannot reduce keyof M and need to preserve it as is.
+        function isNonDistributiveNameType(type) {
+            return !!(type && (type.flags & 16777216 /* Conditional */ && !type.root.isDistributive ||
+                type.flags & (3145728 /* UnionOrIntersection */ | 134217728 /* TemplateLiteral */) && ts.some(type.types, isNonDistributiveNameType) ||
+                type.flags & (4194304 /* Index */ | 268435456 /* StringMapping */) && isNonDistributiveNameType(type.type) ||
+                type.flags & 8388608 /* IndexedAccess */ && isNonDistributiveNameType(type.indexType) ||
+                type.flags & 33554432 /* Substitution */ && isNonDistributiveNameType(type.substitute)));
+        }
+        function getLiteralTypeFromPropertyName(name) {
+            if (ts.isPrivateIdentifier(name)) {
+                return neverType;
+            }
+            return ts.isIdentifier(name) ? getLiteralType(ts.unescapeLeadingUnderscores(name.escapedText)) :
+                getRegularTypeOfLiteralType(ts.isComputedPropertyName(name) ? checkComputedPropertyName(name) : checkExpression(name));
+        }
+        function getBigIntLiteralType(node) {
+            return getLiteralType({
+                negative: false,
+                base10Value: ts.parsePseudoBigInt(node.text)
+            });
+        }
+        function getLiteralTypeFromProperty(prop, include) {
+            if (!(ts.getDeclarationModifierFlagsFromSymbol(prop) & 24 /* NonPublicAccessibilityModifier */)) {
+                var type = getSymbolLinks(getLateBoundSymbol(prop)).nameType;
+                if (!type && !ts.isKnownSymbol(prop)) {
+                    if (prop.escapedName === "default" /* Default */) {
+                        type = getLiteralType("default");
                     }
-                    var related = compareProperties(sourceProp, targetProp, isRelatedTo);
-                    if (!related) {
-                        return 0 /* False */;
+                    else {
+                        var name = prop.valueDeclaration && ts.getNameOfDeclaration(prop.valueDeclaration);
+                        type = name && getLiteralTypeFromPropertyName(name) || getLiteralType(ts.symbolName(prop));
                     }
-                    result &= related;
                 }
-                return result;
-            }
-            function signaturesRelatedTo(source, target, kind, reportErrors) {
-                if (relation === identityRelation) {
-                    return signaturesIdenticalTo(source, target, kind);
+                if (type && type.flags & include) {
+                    return type;
                 }
-                if (target === anyFunctionType || source === anyFunctionType) {
-                    return -1 /* True */;
+            }
+            return neverType;
+        }
+        function getLiteralTypeFromProperties(type, include) {
+            return getUnionType(ts.map(getPropertiesOfType(type), function (p) { return getLiteralTypeFromProperty(p, include); }));
+        }
+        function getNonEnumNumberIndexInfo(type) {
+            var numberIndexInfo = getIndexInfoOfType(type, 1 /* Number */);
+            return numberIndexInfo !== enumNumberIndexInfo ? numberIndexInfo : undefined;
+        }
+        function getIndexType(type, stringsOnly, noIndexSignatures) {
+            if (stringsOnly === void 0) { stringsOnly = keyofStringsOnly; }
+            type = getReducedType(type);
+            return type.flags & 1048576 /* Union */ ? getIntersectionType(ts.map(type.types, function (t) { return getIndexType(t, stringsOnly, noIndexSignatures); })) :
+                type.flags & 2097152 /* Intersection */ ? getUnionType(ts.map(type.types, function (t) { return getIndexType(t, stringsOnly, noIndexSignatures); })) :
+                    type.flags & 58982400 /* InstantiableNonPrimitive */ || isGenericTupleType(type) || isGenericMappedType(type) && isNonDistributiveNameType(getNameTypeFromMappedType(type)) ? getIndexTypeForGenericType(type, stringsOnly) :
+                        ts.getObjectFlags(type) & 32 /* Mapped */ ? getIndexTypeForMappedType(type, noIndexSignatures) :
+                            type === wildcardType ? wildcardType :
+                                type.flags & 2 /* Unknown */ ? neverType :
+                                    type.flags & (1 /* Any */ | 131072 /* Never */) ? keyofConstraintType :
+                                        stringsOnly ? !noIndexSignatures && getIndexInfoOfType(type, 0 /* String */) ? stringType : getLiteralTypeFromProperties(type, 128 /* StringLiteral */) :
+                                            !noIndexSignatures && getIndexInfoOfType(type, 0 /* String */) ? getUnionType([stringType, numberType, getLiteralTypeFromProperties(type, 8192 /* UniqueESSymbol */)]) :
+                                                getNonEnumNumberIndexInfo(type) ? getUnionType([numberType, getLiteralTypeFromProperties(type, 128 /* StringLiteral */ | 8192 /* UniqueESSymbol */)]) :
+                                                    getLiteralTypeFromProperties(type, 8576 /* StringOrNumberLiteralOrUnique */);
+        }
+        function getExtractStringType(type) {
+            if (keyofStringsOnly) {
+                return type;
+            }
+            var extractTypeAlias = getGlobalExtractSymbol();
+            return extractTypeAlias ? getTypeAliasInstantiation(extractTypeAlias, [type, stringType]) : stringType;
+        }
+        function getIndexTypeOrString(type) {
+            var indexType = getExtractStringType(getIndexType(type));
+            return indexType.flags & 131072 /* Never */ ? stringType : indexType;
+        }
+        function getTypeFromTypeOperatorNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                switch (node.operator) {
+                    case 138 /* KeyOfKeyword */:
+                        links.resolvedType = getIndexType(getTypeFromTypeNode(node.type));
+                        break;
+                    case 151 /* UniqueKeyword */:
+                        links.resolvedType = node.type.kind === 148 /* SymbolKeyword */
+                            ? getESSymbolLikeTypeForNode(ts.walkUpParenthesizedTypes(node.parent))
+                            : errorType;
+                        break;
+                    case 142 /* ReadonlyKeyword */:
+                        links.resolvedType = getTypeFromTypeNode(node.type);
+                        break;
+                    default:
+                        throw ts.Debug.assertNever(node.operator);
                 }
-                var sourceIsJSConstructor = source.symbol && isJSConstructor(source.symbol.valueDeclaration);
-                var targetIsJSConstructor = target.symbol && isJSConstructor(target.symbol.valueDeclaration);
-                var sourceSignatures = getSignaturesOfType(source, (sourceIsJSConstructor && kind === 1 /* Construct */) ?
-                    0 /* Call */ : kind);
-                var targetSignatures = getSignaturesOfType(target, (targetIsJSConstructor && kind === 1 /* Construct */) ?
-                    0 /* Call */ : kind);
-                if (kind === 1 /* Construct */ && sourceSignatures.length && targetSignatures.length) {
-                    if (ts.isAbstractConstructorType(source) && !ts.isAbstractConstructorType(target)) {
-                        // An abstract constructor type is not assignable to a non-abstract constructor type
-                        // as it would otherwise be possible to new an abstract class. Note that the assignability
-                        // check we perform for an extends clause excludes construct signatures from the target,
-                        // so this check never proceeds.
-                        if (reportErrors) {
-                            reportError(ts.Diagnostics.Cannot_assign_an_abstract_constructor_type_to_a_non_abstract_constructor_type);
-                        }
-                        return 0 /* False */;
+            }
+            return links.resolvedType;
+        }
+        function getTypeFromTemplateTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                links.resolvedType = getTemplateLiteralType(__spreadArrays([node.head.text], ts.map(node.templateSpans, function (span) { return span.literal.text; })), ts.map(node.templateSpans, function (span) { return getTypeFromTypeNode(span.type); }));
+            }
+            return links.resolvedType;
+        }
+        function getTemplateLiteralType(texts, types) {
+            var unionIndex = ts.findIndex(types, function (t) { return !!(t.flags & (131072 /* Never */ | 1048576 /* Union */)); });
+            if (unionIndex >= 0) {
+                return checkCrossProductUnion(types) ?
+                    mapType(types[unionIndex], function (t) { return getTemplateLiteralType(texts, ts.replaceElement(types, unionIndex, t)); }) :
+                    errorType;
+            }
+            if (ts.contains(types, wildcardType)) {
+                return wildcardType;
+            }
+            var newTypes = [];
+            var newTexts = [];
+            var text = texts[0];
+            if (!addSpans(texts, types)) {
+                return stringType;
+            }
+            if (newTypes.length === 0) {
+                return getLiteralType(text);
+            }
+            newTexts.push(text);
+            var id = getTypeListId(newTypes) + "|" + ts.map(newTexts, function (t) { return t.length; }).join(",") + "|" + newTexts.join("");
+            var type = templateLiteralTypes.get(id);
+            if (!type) {
+                templateLiteralTypes.set(id, type = createTemplateLiteralType(newTexts, newTypes));
+            }
+            return type;
+            function addSpans(texts, types) {
+                for (var i = 0; i < types.length; i++) {
+                    var t = types[i];
+                    if (t.flags & (2944 /* Literal */ | 65536 /* Null */ | 32768 /* Undefined */)) {
+                        text += getTemplateStringForType(t) || "";
+                        text += texts[i + 1];
+                    }
+                    else if (t.flags & 134217728 /* TemplateLiteral */) {
+                        text += t.texts[0];
+                        if (!addSpans(t.texts, t.types))
+                            return false;
+                        text += texts[i + 1];
                     }
-                    if (!constructorVisibilitiesAreCompatible(sourceSignatures[0], targetSignatures[0], reportErrors)) {
-                        return 0 /* False */;
+                    else if (isGenericIndexType(t) || isPatternLiteralPlaceholderType(t)) {
+                        newTypes.push(t);
+                        newTexts.push(text);
+                        text = texts[i + 1];
+                    }
+                    else {
+                        return false;
                     }
                 }
-                var result = -1 /* True */;
-                var saveErrorInfo = captureErrorCalculationState();
-                var incompatibleReporter = kind === 1 /* Construct */ ? reportIncompatibleConstructSignatureReturn : reportIncompatibleCallSignatureReturn;
-                if (ts.getObjectFlags(source) & 64 /* Instantiated */ && ts.getObjectFlags(target) & 64 /* Instantiated */ && source.symbol === target.symbol) {
-                    // We have instantiations of the same anonymous type (which typically will be the type of a
-                    // method). Simply do a pairwise comparison of the signatures in the two signature lists instead
-                    // of the much more expensive N * M comparison matrix we explore below. We erase type parameters
-                    // as they are known to always be the same.
-                    for (var i = 0; i < targetSignatures.length; i++) {
-                        var related = signatureRelatedTo(sourceSignatures[i], targetSignatures[i], /*erase*/ true, reportErrors, incompatibleReporter(sourceSignatures[i], targetSignatures[i]));
-                        if (!related) {
-                            return 0 /* False */;
+                return true;
+            }
+        }
+        function getTemplateStringForType(type) {
+            return type.flags & 128 /* StringLiteral */ ? type.value :
+                type.flags & 256 /* NumberLiteral */ ? "" + type.value :
+                    type.flags & 2048 /* BigIntLiteral */ ? ts.pseudoBigIntToString(type.value) :
+                        type.flags & 512 /* BooleanLiteral */ ? type.intrinsicName :
+                            type.flags & 65536 /* Null */ ? "null" :
+                                type.flags & 32768 /* Undefined */ ? "undefined" :
+                                    undefined;
+        }
+        function createTemplateLiteralType(texts, types) {
+            var type = createType(134217728 /* TemplateLiteral */);
+            type.texts = texts;
+            type.types = types;
+            return type;
+        }
+        function getStringMappingType(symbol, type) {
+            return type.flags & (1048576 /* Union */ | 131072 /* Never */) ? mapType(type, function (t) { return getStringMappingType(symbol, t); }) :
+                isGenericIndexType(type) ? getStringMappingTypeForGenericType(symbol, type) :
+                    type.flags & 128 /* StringLiteral */ ? getLiteralType(applyStringMapping(symbol, type.value)) :
+                        type;
+        }
+        function applyStringMapping(symbol, str) {
+            switch (intrinsicTypeKinds.get(symbol.escapedName)) {
+                case 0 /* Uppercase */: return str.toUpperCase();
+                case 1 /* Lowercase */: return str.toLowerCase();
+                case 2 /* Capitalize */: return str.charAt(0).toUpperCase() + str.slice(1);
+                case 3 /* Uncapitalize */: return str.charAt(0).toLowerCase() + str.slice(1);
+            }
+            return str;
+        }
+        function getStringMappingTypeForGenericType(symbol, type) {
+            var id = getSymbolId(symbol) + "," + getTypeId(type);
+            var result = stringMappingTypes.get(id);
+            if (!result) {
+                stringMappingTypes.set(id, result = createStringMappingType(symbol, type));
+            }
+            return result;
+        }
+        function createStringMappingType(symbol, type) {
+            var result = createType(268435456 /* StringMapping */);
+            result.symbol = symbol;
+            result.type = type;
+            return result;
+        }
+        function createIndexedAccessType(objectType, indexType, aliasSymbol, aliasTypeArguments, shouldIncludeUndefined) {
+            var type = createType(8388608 /* IndexedAccess */);
+            type.objectType = objectType;
+            type.indexType = indexType;
+            type.aliasSymbol = aliasSymbol;
+            type.aliasTypeArguments = aliasTypeArguments;
+            type.noUncheckedIndexedAccessCandidate = shouldIncludeUndefined;
+            return type;
+        }
+        /**
+         * Returns if a type is or consists of a JSLiteral object type
+         * In addition to objects which are directly literals,
+         * * unions where every element is a jsliteral
+         * * intersections where at least one element is a jsliteral
+         * * and instantiable types constrained to a jsliteral
+         * Should all count as literals and not print errors on access or assignment of possibly existing properties.
+         * This mirrors the behavior of the index signature propagation, to which this behaves similarly (but doesn't affect assignability or inference).
+         */
+        function isJSLiteralType(type) {
+            if (noImplicitAny) {
+                return false; // Flag is meaningless under `noImplicitAny` mode
+            }
+            if (ts.getObjectFlags(type) & 16384 /* JSLiteral */) {
+                return true;
+            }
+            if (type.flags & 1048576 /* Union */) {
+                return ts.every(type.types, isJSLiteralType);
+            }
+            if (type.flags & 2097152 /* Intersection */) {
+                return ts.some(type.types, isJSLiteralType);
+            }
+            if (type.flags & 465829888 /* Instantiable */) {
+                return isJSLiteralType(getResolvedBaseConstraint(type));
+            }
+            return false;
+        }
+        function getPropertyNameFromIndex(indexType, accessNode) {
+            var accessExpression = accessNode && accessNode.kind === 202 /* ElementAccessExpression */ ? accessNode : undefined;
+            return isTypeUsableAsPropertyName(indexType) ?
+                getPropertyNameFromType(indexType) :
+                accessExpression && checkThatExpressionIsProperSymbolReference(accessExpression.argumentExpression, indexType, /*reportError*/ false) ?
+                    ts.getPropertyNameForKnownSymbolName(ts.idText(accessExpression.argumentExpression.name)) :
+                    accessNode && ts.isPropertyName(accessNode) ?
+                        // late bound names are handled in the first branch, so here we only need to handle normal names
+                        ts.getPropertyNameForPropertyNameNode(accessNode) :
+                        undefined;
+        }
+        function isUncalledFunctionReference(node, symbol) {
+            return !(symbol.flags & (16 /* Function */ | 8192 /* Method */))
+                || !ts.isCallLikeExpression(ts.findAncestor(node, function (n) { return !ts.isAccessExpression(n); }) || node.parent)
+                    && ts.every(symbol.declarations, function (d) { return !ts.isFunctionLike(d) || !!(ts.getCombinedNodeFlags(d) & 134217728 /* Deprecated */); });
+        }
+        function getPropertyTypeForIndexType(originalObjectType, objectType, indexType, fullIndexType, suppressNoImplicitAnyError, accessNode, accessFlags, noUncheckedIndexedAccessCandidate, reportDeprecated) {
+            var _a;
+            var accessExpression = accessNode && accessNode.kind === 202 /* ElementAccessExpression */ ? accessNode : undefined;
+            var propName = accessNode && ts.isPrivateIdentifier(accessNode) ? undefined : getPropertyNameFromIndex(indexType, accessNode);
+            if (propName !== undefined) {
+                var prop = getPropertyOfType(objectType, propName);
+                if (prop) {
+                    if (reportDeprecated && accessNode && getDeclarationNodeFlagsFromSymbol(prop) & 134217728 /* Deprecated */ && isUncalledFunctionReference(accessNode, prop)) {
+                        var deprecatedNode = (_a = accessExpression === null || accessExpression === void 0 ? void 0 : accessExpression.argumentExpression) !== null && _a !== void 0 ? _a : (ts.isIndexedAccessTypeNode(accessNode) ? accessNode.indexType : accessNode);
+                        errorOrSuggestion(/* isError */ false, deprecatedNode, ts.Diagnostics._0_is_deprecated, propName);
+                    }
+                    if (accessExpression) {
+                        markPropertyAsReferenced(prop, accessExpression, /*isThisAccess*/ accessExpression.expression.kind === 107 /* ThisKeyword */);
+                        if (isAssignmentToReadonlyEntity(accessExpression, prop, ts.getAssignmentTargetKind(accessExpression))) {
+                            error(accessExpression.argumentExpression, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_read_only_property, symbolToString(prop));
+                            return undefined;
+                        }
+                        if (accessFlags & 4 /* CacheSymbol */) {
+                            getNodeLinks(accessNode).resolvedSymbol = prop;
+                        }
+                        if (isThisPropertyAccessInConstructor(accessExpression, prop)) {
+                            return autoType;
                         }
-                        result &= related;
                     }
+                    var propType = getTypeOfSymbol(prop);
+                    return accessExpression && ts.getAssignmentTargetKind(accessExpression) !== 1 /* Definite */ ?
+                        getFlowTypeOfReference(accessExpression, propType) :
+                        propType;
                 }
-                else if (sourceSignatures.length === 1 && targetSignatures.length === 1) {
-                    // For simple functions (functions with a single signature) we only erase type parameters for
-                    // the comparable relation. Otherwise, if the source signature is generic, we instantiate it
-                    // in the context of the target signature before checking the relationship. Ideally we'd do
-                    // this regardless of the number of signatures, but the potential costs are prohibitive due
-                    // to the quadratic nature of the logic below.
-                    var eraseGenerics = relation === comparableRelation || !!compilerOptions.noStrictGenericChecks;
-                    result = signatureRelatedTo(sourceSignatures[0], targetSignatures[0], eraseGenerics, reportErrors, incompatibleReporter(sourceSignatures[0], targetSignatures[0]));
-                }
-                else {
-                    outer: for (var _i = 0, targetSignatures_1 = targetSignatures; _i < targetSignatures_1.length; _i++) {
-                        var t = targetSignatures_1[_i];
-                        // Only elaborate errors from the first failure
-                        var shouldElaborateErrors = reportErrors;
-                        for (var _a = 0, sourceSignatures_1 = sourceSignatures; _a < sourceSignatures_1.length; _a++) {
-                            var s = sourceSignatures_1[_a];
-                            var related = signatureRelatedTo(s, t, /*erase*/ true, shouldElaborateErrors, incompatibleReporter(s, t));
-                            if (related) {
-                                result &= related;
-                                resetErrorInfo(saveErrorInfo);
-                                continue outer;
-                            }
-                            shouldElaborateErrors = false;
+                if (everyType(objectType, isTupleType) && isNumericLiteralName(propName) && +propName >= 0) {
+                    if (accessNode && everyType(objectType, function (t) { return !t.target.hasRestElement; }) && !(accessFlags & 8 /* NoTupleBoundsCheck */)) {
+                        var indexNode = getIndexNodeForAccessExpression(accessNode);
+                        if (isTupleType(objectType)) {
+                            error(indexNode, ts.Diagnostics.Tuple_type_0_of_length_1_has_no_element_at_index_2, typeToString(objectType), getTypeReferenceArity(objectType), ts.unescapeLeadingUnderscores(propName));
                         }
-                        if (shouldElaborateErrors) {
-                            reportError(ts.Diagnostics.Type_0_provides_no_match_for_the_signature_1, typeToString(source), signatureToString(t, /*enclosingDeclaration*/ undefined, /*flags*/ undefined, kind));
+                        else {
+                            error(indexNode, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.unescapeLeadingUnderscores(propName), typeToString(objectType));
                         }
-                        return 0 /* False */;
                     }
+                    errorIfWritingToReadonlyIndex(getIndexInfoOfType(objectType, 1 /* Number */));
+                    return mapType(objectType, function (t) {
+                        var restType = getRestTypeOfTupleType(t) || undefinedType;
+                        return noUncheckedIndexedAccessCandidate ? getUnionType([restType, undefinedType]) : restType;
+                    });
                 }
-                return result;
-            }
-            function reportIncompatibleCallSignatureReturn(siga, sigb) {
-                if (siga.parameters.length === 0 && sigb.parameters.length === 0) {
-                    return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1, typeToString(source), typeToString(target)); };
-                }
-                return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Call_signature_return_types_0_and_1_are_incompatible, typeToString(source), typeToString(target)); };
-            }
-            function reportIncompatibleConstructSignatureReturn(siga, sigb) {
-                if (siga.parameters.length === 0 && sigb.parameters.length === 0) {
-                    return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1, typeToString(source), typeToString(target)); };
-                }
-                return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible, typeToString(source), typeToString(target)); };
-            }
-            /**
-             * See signatureAssignableTo, compareSignaturesIdentical
-             */
-            function signatureRelatedTo(source, target, erase, reportErrors, incompatibleReporter) {
-                return compareSignaturesRelated(erase ? getErasedSignature(source) : source, erase ? getErasedSignature(target) : target, relation === strictSubtypeRelation ? 8 /* StrictArity */ : 0, reportErrors, reportError, incompatibleReporter, isRelatedTo, makeFunctionTypeMapper(reportUnreliableMarkers));
             }
-            function signaturesIdenticalTo(source, target, kind) {
-                var sourceSignatures = getSignaturesOfType(source, kind);
-                var targetSignatures = getSignaturesOfType(target, kind);
-                if (sourceSignatures.length !== targetSignatures.length) {
-                    return 0 /* False */;
+            if (!(indexType.flags & 98304 /* Nullable */) && isTypeAssignableToKind(indexType, 402653316 /* StringLike */ | 296 /* NumberLike */ | 12288 /* ESSymbolLike */)) {
+                if (objectType.flags & (1 /* Any */ | 131072 /* Never */)) {
+                    return objectType;
                 }
-                var result = -1 /* True */;
-                for (var i = 0; i < sourceSignatures.length; i++) {
-                    var related = compareSignaturesIdentical(sourceSignatures[i], targetSignatures[i], /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ false, isRelatedTo);
-                    if (!related) {
-                        return 0 /* False */;
+                var stringIndexInfo = getIndexInfoOfType(objectType, 0 /* String */);
+                var indexInfo = isTypeAssignableToKind(indexType, 296 /* NumberLike */) && getIndexInfoOfType(objectType, 1 /* Number */) || stringIndexInfo;
+                if (indexInfo) {
+                    if (accessFlags & 1 /* NoIndexSignatures */ && indexInfo === stringIndexInfo) {
+                        if (accessExpression) {
+                            error(accessExpression, ts.Diagnostics.Type_0_cannot_be_used_to_index_type_1, typeToString(indexType), typeToString(originalObjectType));
+                        }
+                        return undefined;
+                    }
+                    if (accessNode && !isTypeAssignableToKind(indexType, 4 /* String */ | 8 /* Number */)) {
+                        var indexNode = getIndexNodeForAccessExpression(accessNode);
+                        error(indexNode, ts.Diagnostics.Type_0_cannot_be_used_as_an_index_type, typeToString(indexType));
+                        return noUncheckedIndexedAccessCandidate ? getUnionType([indexInfo.type, undefinedType]) : indexInfo.type;
                     }
-                    result &= related;
+                    errorIfWritingToReadonlyIndex(indexInfo);
+                    return noUncheckedIndexedAccessCandidate ? getUnionType([indexInfo.type, undefinedType]) : indexInfo.type;
                 }
-                return result;
-            }
-            function eachPropertyRelatedTo(source, target, kind, reportErrors) {
-                var result = -1 /* True */;
-                var props = source.flags & 2097152 /* Intersection */ ? getPropertiesOfUnionOrIntersectionType(source) : getPropertiesOfObjectType(source);
-                for (var _i = 0, props_2 = props; _i < props_2.length; _i++) {
-                    var prop = props_2[_i];
-                    // Skip over ignored JSX and symbol-named members
-                    if (isIgnoredJsxProperty(source, prop)) {
-                        continue;
-                    }
-                    var nameType = getSymbolLinks(prop).nameType;
-                    if (nameType && nameType.flags & 8192 /* UniqueESSymbol */) {
-                        continue;
+                if (indexType.flags & 131072 /* Never */) {
+                    return neverType;
+                }
+                if (isJSLiteralType(objectType)) {
+                    return anyType;
+                }
+                if (accessExpression && !isConstEnumObjectType(objectType)) {
+                    if (objectType.symbol === globalThisSymbol && propName !== undefined && globalThisSymbol.exports.has(propName) && (globalThisSymbol.exports.get(propName).flags & 418 /* BlockScoped */)) {
+                        error(accessExpression, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.unescapeLeadingUnderscores(propName), typeToString(objectType));
                     }
-                    if (kind === 0 /* String */ || isNumericLiteralName(prop.escapedName)) {
-                        var related = isRelatedTo(getTypeOfSymbol(prop), target, reportErrors);
-                        if (!related) {
-                            if (reportErrors) {
-                                reportError(ts.Diagnostics.Property_0_is_incompatible_with_index_signature, symbolToString(prop));
+                    else if (noImplicitAny && !compilerOptions.suppressImplicitAnyIndexErrors && !suppressNoImplicitAnyError) {
+                        if (propName !== undefined && typeHasStaticProperty(propName, objectType)) {
+                            error(accessExpression, ts.Diagnostics.Property_0_is_a_static_member_of_type_1, propName, typeToString(objectType));
+                        }
+                        else if (getIndexTypeOfType(objectType, 1 /* Number */)) {
+                            error(accessExpression.argumentExpression, ts.Diagnostics.Element_implicitly_has_an_any_type_because_index_expression_is_not_of_type_number);
+                        }
+                        else {
+                            var suggestion = void 0;
+                            if (propName !== undefined && (suggestion = getSuggestionForNonexistentProperty(propName, objectType))) {
+                                if (suggestion !== undefined) {
+                                    error(accessExpression.argumentExpression, ts.Diagnostics.Property_0_does_not_exist_on_type_1_Did_you_mean_2, propName, typeToString(objectType), suggestion);
+                                }
+                            }
+                            else {
+                                var suggestion_1 = getSuggestionForNonexistentIndexSignature(objectType, accessExpression, indexType);
+                                if (suggestion_1 !== undefined) {
+                                    error(accessExpression, ts.Diagnostics.Element_implicitly_has_an_any_type_because_type_0_has_no_index_signature_Did_you_mean_to_call_1, typeToString(objectType), suggestion_1);
+                                }
+                                else {
+                                    var errorInfo = void 0;
+                                    if (indexType.flags & 1024 /* EnumLiteral */) {
+                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, "[" + typeToString(indexType) + "]", typeToString(objectType));
+                                    }
+                                    else if (indexType.flags & 8192 /* UniqueESSymbol */) {
+                                        var symbolName_2 = getFullyQualifiedName(indexType.symbol, accessExpression);
+                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, "[" + symbolName_2 + "]", typeToString(objectType));
+                                    }
+                                    else if (indexType.flags & 128 /* StringLiteral */) {
+                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, indexType.value, typeToString(objectType));
+                                    }
+                                    else if (indexType.flags & 256 /* NumberLiteral */) {
+                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.Property_0_does_not_exist_on_type_1, indexType.value, typeToString(objectType));
+                                    }
+                                    else if (indexType.flags & (8 /* Number */ | 4 /* String */)) {
+                                        errorInfo = ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics.No_index_signature_with_a_parameter_of_type_0_was_found_on_type_1, typeToString(indexType), typeToString(objectType));
+                                    }
+                                    errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Element_implicitly_has_an_any_type_because_expression_of_type_0_can_t_be_used_to_index_type_1, typeToString(fullIndexType), typeToString(objectType));
+                                    diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(accessExpression, errorInfo));
+                                }
                             }
-                            return 0 /* False */;
                         }
-                        result &= related;
                     }
+                    return undefined;
                 }
-                return result;
             }
-            function indexTypeRelatedTo(sourceType, targetType, reportErrors) {
-                var related = isRelatedTo(sourceType, targetType, reportErrors);
-                if (!related && reportErrors) {
-                    reportError(ts.Diagnostics.Index_signatures_are_incompatible);
-                }
-                return related;
+            if (isJSLiteralType(objectType)) {
+                return anyType;
             }
-            function indexTypesRelatedTo(source, target, kind, sourceIsPrimitive, reportErrors, intersectionState) {
-                if (relation === identityRelation) {
-                    return indexTypesIdenticalTo(source, target, kind);
-                }
-                var targetType = getIndexTypeOfType(target, kind);
-                if (!targetType || targetType.flags & 1 /* Any */ && !sourceIsPrimitive) {
-                    // Index signature of type any permits assignment from everything but primitives
-                    return -1 /* True */;
+            if (accessNode) {
+                var indexNode = getIndexNodeForAccessExpression(accessNode);
+                if (indexType.flags & (128 /* StringLiteral */ | 256 /* NumberLiteral */)) {
+                    error(indexNode, ts.Diagnostics.Property_0_does_not_exist_on_type_1, "" + indexType.value, typeToString(objectType));
                 }
-                if (isGenericMappedType(source)) {
-                    // A generic mapped type { [P in K]: T } is related to an index signature { [x: string]: U }
-                    // if T is related to U.
-                    return kind === 0 /* String */ ? isRelatedTo(getTemplateTypeFromMappedType(source), targetType, reportErrors) : 0 /* False */;
+                else if (indexType.flags & (4 /* String */ | 8 /* Number */)) {
+                    error(indexNode, ts.Diagnostics.Type_0_has_no_matching_index_signature_for_type_1, typeToString(objectType), typeToString(indexType));
                 }
-                var indexType = getIndexTypeOfType(source, kind) || kind === 1 /* Number */ && getIndexTypeOfType(source, 0 /* String */);
-                if (indexType) {
-                    return indexTypeRelatedTo(indexType, targetType, reportErrors);
+                else {
+                    error(indexNode, ts.Diagnostics.Type_0_cannot_be_used_as_an_index_type, typeToString(indexType));
                 }
-                if (!(intersectionState & 1 /* Source */) && isObjectTypeWithInferableIndex(source)) {
-                    // Intersection constituents are never considered to have an inferred index signature
-                    var related = eachPropertyRelatedTo(source, targetType, kind, reportErrors);
-                    if (related && kind === 0 /* String */) {
-                        var numberIndexType = getIndexTypeOfType(source, 1 /* Number */);
-                        if (numberIndexType) {
-                            related &= indexTypeRelatedTo(numberIndexType, targetType, reportErrors);
-                        }
-                    }
-                    return related;
+            }
+            if (isTypeAny(indexType)) {
+                return indexType;
+            }
+            return undefined;
+            function errorIfWritingToReadonlyIndex(indexInfo) {
+                if (indexInfo && indexInfo.isReadonly && accessExpression && (ts.isAssignmentTarget(accessExpression) || ts.isDeleteTarget(accessExpression))) {
+                    error(accessExpression, ts.Diagnostics.Index_signature_in_type_0_only_permits_reading, typeToString(objectType));
                 }
-                if (reportErrors) {
-                    reportError(ts.Diagnostics.Index_signature_is_missing_in_type_0, typeToString(source));
+            }
+        }
+        function getIndexNodeForAccessExpression(accessNode) {
+            return accessNode.kind === 202 /* ElementAccessExpression */ ? accessNode.argumentExpression :
+                accessNode.kind === 189 /* IndexedAccessType */ ? accessNode.indexType :
+                    accessNode.kind === 158 /* ComputedPropertyName */ ? accessNode.expression :
+                        accessNode;
+        }
+        function isPatternLiteralPlaceholderType(type) {
+            return templateConstraintType.types.indexOf(type) !== -1 || !!(type.flags & 1 /* Any */);
+        }
+        function isPatternLiteralType(type) {
+            return !!(type.flags & 134217728 /* TemplateLiteral */) && ts.every(type.types, isPatternLiteralPlaceholderType);
+        }
+        function isGenericObjectType(type) {
+            if (type.flags & 3145728 /* UnionOrIntersection */) {
+                if (!(type.objectFlags & 4194304 /* IsGenericObjectTypeComputed */)) {
+                    type.objectFlags |= 4194304 /* IsGenericObjectTypeComputed */ |
+                        (ts.some(type.types, isGenericObjectType) ? 8388608 /* IsGenericObjectType */ : 0);
                 }
-                return 0 /* False */;
+                return !!(type.objectFlags & 8388608 /* IsGenericObjectType */);
             }
-            function indexTypesIdenticalTo(source, target, indexKind) {
-                var targetInfo = getIndexInfoOfType(target, indexKind);
-                var sourceInfo = getIndexInfoOfType(source, indexKind);
-                if (!sourceInfo && !targetInfo) {
-                    return -1 /* True */;
+            return !!(type.flags & 58982400 /* InstantiableNonPrimitive */) || isGenericMappedType(type) || isGenericTupleType(type);
+        }
+        function isGenericIndexType(type) {
+            if (type.flags & 3145728 /* UnionOrIntersection */) {
+                if (!(type.objectFlags & 16777216 /* IsGenericIndexTypeComputed */)) {
+                    type.objectFlags |= 16777216 /* IsGenericIndexTypeComputed */ |
+                        (ts.some(type.types, isGenericIndexType) ? 33554432 /* IsGenericIndexType */ : 0);
                 }
-                if (sourceInfo && targetInfo && sourceInfo.isReadonly === targetInfo.isReadonly) {
-                    return isRelatedTo(sourceInfo.type, targetInfo.type);
+                return !!(type.objectFlags & 33554432 /* IsGenericIndexType */);
+            }
+            return !!(type.flags & (58982400 /* InstantiableNonPrimitive */ | 4194304 /* Index */ | 134217728 /* TemplateLiteral */ | 268435456 /* StringMapping */)) && !isPatternLiteralType(type);
+        }
+        function isThisTypeParameter(type) {
+            return !!(type.flags & 262144 /* TypeParameter */ && type.isThisType);
+        }
+        function getSimplifiedType(type, writing) {
+            return type.flags & 8388608 /* IndexedAccess */ ? getSimplifiedIndexedAccessType(type, writing) :
+                type.flags & 16777216 /* Conditional */ ? getSimplifiedConditionalType(type, writing) :
+                    type;
+        }
+        function distributeIndexOverObjectType(objectType, indexType, writing) {
+            // (T | U)[K] -> T[K] | U[K] (reading)
+            // (T | U)[K] -> T[K] & U[K] (writing)
+            // (T & U)[K] -> T[K] & U[K]
+            if (objectType.flags & 3145728 /* UnionOrIntersection */) {
+                var types = ts.map(objectType.types, function (t) { return getSimplifiedType(getIndexedAccessType(t, indexType), writing); });
+                return objectType.flags & 2097152 /* Intersection */ || writing ? getIntersectionType(types) : getUnionType(types);
+            }
+        }
+        function distributeObjectOverIndexType(objectType, indexType, writing) {
+            // T[A | B] -> T[A] | T[B] (reading)
+            // T[A | B] -> T[A] & T[B] (writing)
+            if (indexType.flags & 1048576 /* Union */) {
+                var types = ts.map(indexType.types, function (t) { return getSimplifiedType(getIndexedAccessType(objectType, t), writing); });
+                return writing ? getIntersectionType(types) : getUnionType(types);
+            }
+        }
+        function unwrapSubstitution(type) {
+            if (type.flags & 33554432 /* Substitution */) {
+                return type.substitute;
+            }
+            return type;
+        }
+        // Transform an indexed access to a simpler form, if possible. Return the simpler form, or return
+        // the type itself if no transformation is possible. The writing flag indicates that the type is
+        // the target of an assignment.
+        function getSimplifiedIndexedAccessType(type, writing) {
+            var cache = writing ? "simplifiedForWriting" : "simplifiedForReading";
+            if (type[cache]) {
+                return type[cache] === circularConstraintType ? type : type[cache];
+            }
+            type[cache] = circularConstraintType;
+            // We recursively simplify the object type as it may in turn be an indexed access type. For example, with
+            // '{ [P in T]: { [Q in U]: number } }[T][U]' we want to first simplify the inner indexed access type.
+            var objectType = unwrapSubstitution(getSimplifiedType(type.objectType, writing));
+            var indexType = getSimplifiedType(type.indexType, writing);
+            // T[A | B] -> T[A] | T[B] (reading)
+            // T[A | B] -> T[A] & T[B] (writing)
+            var distributedOverIndex = distributeObjectOverIndexType(objectType, indexType, writing);
+            if (distributedOverIndex) {
+                return type[cache] = distributedOverIndex;
+            }
+            // Only do the inner distributions if the index can no longer be instantiated to cause index distribution again
+            if (!(indexType.flags & 465829888 /* Instantiable */)) {
+                // (T | U)[K] -> T[K] | U[K] (reading)
+                // (T | U)[K] -> T[K] & U[K] (writing)
+                // (T & U)[K] -> T[K] & U[K]
+                var distributedOverObject = distributeIndexOverObjectType(objectType, indexType, writing);
+                if (distributedOverObject) {
+                    return type[cache] = distributedOverObject;
                 }
-                return 0 /* False */;
             }
-            function constructorVisibilitiesAreCompatible(sourceSignature, targetSignature, reportErrors) {
-                if (!sourceSignature.declaration || !targetSignature.declaration) {
-                    return true;
+            // So ultimately (reading):
+            // ((A & B) | C)[K1 | K2] -> ((A & B) | C)[K1] | ((A & B) | C)[K2] -> (A & B)[K1] | C[K1] | (A & B)[K2] | C[K2] -> (A[K1] & B[K1]) | C[K1] | (A[K2] & B[K2]) | C[K2]
+            // A generic tuple type indexed by a number exists only when the index type doesn't select a
+            // fixed element. We simplify to either the combined type of all elements (when the index type
+            // the actual number type) or to the combined type of all non-fixed elements.
+            if (isGenericTupleType(objectType) && indexType.flags & 296 /* NumberLike */) {
+                var elementType = getElementTypeOfSliceOfTupleType(objectType, indexType.flags & 8 /* Number */ ? 0 : objectType.target.fixedLength, /*endSkipCount*/ 0, writing);
+                if (elementType) {
+                    return type[cache] = elementType;
                 }
-                var sourceAccessibility = ts.getSelectedModifierFlags(sourceSignature.declaration, 24 /* NonPublicAccessibilityModifier */);
-                var targetAccessibility = ts.getSelectedModifierFlags(targetSignature.declaration, 24 /* NonPublicAccessibilityModifier */);
-                // A public, protected and private signature is assignable to a private signature.
-                if (targetAccessibility === 8 /* Private */) {
-                    return true;
+            }
+            // If the object type is a mapped type { [P in K]: E }, where K is generic, instantiate E using a mapper
+            // that substitutes the index type for P. For example, for an index access { [P in K]: Box<T[P]> }[X], we
+            // construct the type Box<T[X]>.
+            if (isGenericMappedType(objectType)) {
+                return type[cache] = mapType(substituteIndexedMappedType(objectType, type.indexType), function (t) { return getSimplifiedType(t, writing); });
+            }
+            return type[cache] = type;
+        }
+        function getSimplifiedConditionalType(type, writing) {
+            var checkType = type.checkType;
+            var extendsType = type.extendsType;
+            var trueType = getTrueTypeFromConditionalType(type);
+            var falseType = getFalseTypeFromConditionalType(type);
+            // Simplifications for types of the form `T extends U ? T : never` and `T extends U ? never : T`.
+            if (falseType.flags & 131072 /* Never */ && getActualTypeVariable(trueType) === getActualTypeVariable(checkType)) {
+                if (checkType.flags & 1 /* Any */ || isTypeAssignableTo(getRestrictiveInstantiation(checkType), getRestrictiveInstantiation(extendsType))) { // Always true
+                    return getSimplifiedType(trueType, writing);
                 }
-                // A public and protected signature is assignable to a protected signature.
-                if (targetAccessibility === 16 /* Protected */ && sourceAccessibility !== 8 /* Private */) {
-                    return true;
+                else if (isIntersectionEmpty(checkType, extendsType)) { // Always false
+                    return neverType;
                 }
-                // Only a public signature is assignable to public signature.
-                if (targetAccessibility !== 16 /* Protected */ && !sourceAccessibility) {
-                    return true;
+            }
+            else if (trueType.flags & 131072 /* Never */ && getActualTypeVariable(falseType) === getActualTypeVariable(checkType)) {
+                if (!(checkType.flags & 1 /* Any */) && isTypeAssignableTo(getRestrictiveInstantiation(checkType), getRestrictiveInstantiation(extendsType))) { // Always true
+                    return neverType;
                 }
-                if (reportErrors) {
-                    reportError(ts.Diagnostics.Cannot_assign_a_0_constructor_type_to_a_1_constructor_type, visibilityToString(sourceAccessibility), visibilityToString(targetAccessibility));
+                else if (checkType.flags & 1 /* Any */ || isIntersectionEmpty(checkType, extendsType)) { // Always false
+                    return getSimplifiedType(falseType, writing);
                 }
-                return false;
             }
+            return type;
         }
-        function getBestMatchingType(source, target, isRelatedTo) {
-            if (isRelatedTo === void 0) { isRelatedTo = compareTypesAssignable; }
-            return findMatchingDiscriminantType(source, target, isRelatedTo, /*skipPartial*/ true) ||
-                findMatchingTypeReferenceOrTypeAliasReference(source, target) ||
-                findBestTypeForObjectLiteral(source, target) ||
-                findBestTypeForInvokable(source, target) ||
-                findMostOverlappyType(source, target);
+        /**
+         * Invokes union simplification logic to determine if an intersection is considered empty as a union constituent
+         */
+        function isIntersectionEmpty(type1, type2) {
+            return !!(getUnionType([intersectTypes(type1, type2), neverType]).flags & 131072 /* Never */);
         }
-        function discriminateTypeByDiscriminableItems(target, discriminators, related, defaultValue, skipPartial) {
-            // undefined=unknown, true=discriminated, false=not discriminated
-            // The state of each type progresses from left to right. Discriminated types stop at 'true'.
-            var discriminable = target.types.map(function (_) { return undefined; });
-            for (var _i = 0, discriminators_1 = discriminators; _i < discriminators_1.length; _i++) {
-                var _a = discriminators_1[_i], getDiscriminatingType = _a[0], propertyName = _a[1];
-                var targetProp = getUnionOrIntersectionProperty(target, propertyName);
-                if (skipPartial && targetProp && ts.getCheckFlags(targetProp) & 16 /* ReadPartial */) {
-                    continue;
+        function substituteIndexedMappedType(objectType, index) {
+            var mapper = createTypeMapper([getTypeParameterFromMappedType(objectType)], [index]);
+            var templateMapper = combineTypeMappers(objectType.mapper, mapper);
+            return instantiateType(getTemplateTypeFromMappedType(objectType), templateMapper);
+        }
+        function getIndexedAccessType(objectType, indexType, noUncheckedIndexedAccessCandidate, accessNode, aliasSymbol, aliasTypeArguments, accessFlags) {
+            if (accessFlags === void 0) { accessFlags = 0 /* None */; }
+            return getIndexedAccessTypeOrUndefined(objectType, indexType, noUncheckedIndexedAccessCandidate, accessNode, accessFlags, aliasSymbol, aliasTypeArguments) || (accessNode ? errorType : unknownType);
+        }
+        function indexTypeLessThan(indexType, limit) {
+            return everyType(indexType, function (t) {
+                if (t.flags & 384 /* StringOrNumberLiteral */) {
+                    var propName = getPropertyNameFromType(t);
+                    if (isNumericLiteralName(propName)) {
+                        var index = +propName;
+                        return index >= 0 && index < limit;
+                    }
                 }
-                var i = 0;
-                for (var _b = 0, _c = target.types; _b < _c.length; _b++) {
-                    var type = _c[_b];
-                    var targetType = getTypeOfPropertyOfType(type, propertyName);
-                    if (targetType && related(getDiscriminatingType(), targetType)) {
-                        discriminable[i] = discriminable[i] === undefined ? true : discriminable[i];
+                return false;
+            });
+        }
+        function getIndexedAccessTypeOrUndefined(objectType, indexType, noUncheckedIndexedAccessCandidate, accessNode, accessFlags, aliasSymbol, aliasTypeArguments) {
+            if (accessFlags === void 0) { accessFlags = 0 /* None */; }
+            if (objectType === wildcardType || indexType === wildcardType) {
+                return wildcardType;
+            }
+            var shouldIncludeUndefined = noUncheckedIndexedAccessCandidate ||
+                (!!compilerOptions.noUncheckedIndexedAccess &&
+                    (accessFlags & (2 /* Writing */ | 16 /* ExpressionPosition */)) === 16 /* ExpressionPosition */);
+            // If the object type has a string index signature and no other members we know that the result will
+            // always be the type of that index signature and we can simplify accordingly.
+            if (isStringIndexSignatureOnlyType(objectType) && !(indexType.flags & 98304 /* Nullable */) && isTypeAssignableToKind(indexType, 4 /* String */ | 8 /* Number */)) {
+                indexType = stringType;
+            }
+            // If the index type is generic, or if the object type is generic and doesn't originate in an expression and
+            // the operation isn't exclusively indexing the fixed (non-variadic) portion of a tuple type, we are performing
+            // a higher-order index access where we cannot meaningfully access the properties of the object type. Note that
+            // for a generic T and a non-generic K, we eagerly resolve T[K] if it originates in an expression. This is to
+            // preserve backwards compatibility. For example, an element access 'this["foo"]' has always been resolved
+            // eagerly using the constraint type of 'this' at the given location.
+            if (isGenericIndexType(indexType) || (accessNode && accessNode.kind !== 189 /* IndexedAccessType */ ?
+                isGenericTupleType(objectType) && !indexTypeLessThan(indexType, objectType.target.fixedLength) :
+                isGenericObjectType(objectType) && !(isTupleType(objectType) && indexTypeLessThan(indexType, objectType.target.fixedLength)))) {
+                if (objectType.flags & 3 /* AnyOrUnknown */) {
+                    return objectType;
+                }
+                // Defer the operation by creating an indexed access type.
+                var id = objectType.id + "," + indexType.id + (shouldIncludeUndefined ? "?" : "");
+                var type = indexedAccessTypes.get(id);
+                if (!type) {
+                    indexedAccessTypes.set(id, type = createIndexedAccessType(objectType, indexType, aliasSymbol, aliasTypeArguments, shouldIncludeUndefined));
+                }
+                return type;
+            }
+            // In the following we resolve T[K] to the type of the property in T selected by K.
+            // We treat boolean as different from other unions to improve errors;
+            // skipping straight to getPropertyTypeForIndexType gives errors with 'boolean' instead of 'true'.
+            var apparentObjectType = getReducedApparentType(objectType);
+            if (indexType.flags & 1048576 /* Union */ && !(indexType.flags & 16 /* Boolean */)) {
+                var propTypes = [];
+                var wasMissingProp = false;
+                for (var _i = 0, _a = indexType.types; _i < _a.length; _i++) {
+                    var t = _a[_i];
+                    var propType = getPropertyTypeForIndexType(objectType, apparentObjectType, t, indexType, wasMissingProp, accessNode, accessFlags, shouldIncludeUndefined);
+                    if (propType) {
+                        propTypes.push(propType);
+                    }
+                    else if (!accessNode) {
+                        // If there's no error node, we can immeditely stop, since error reporting is off
+                        return undefined;
                     }
                     else {
-                        discriminable[i] = false;
+                        // Otherwise we set a flag and return at the end of the loop so we still mark all errors
+                        wasMissingProp = true;
                     }
-                    i++;
                 }
+                if (wasMissingProp) {
+                    return undefined;
+                }
+                return accessFlags & 2 /* Writing */
+                    ? getIntersectionType(propTypes, aliasSymbol, aliasTypeArguments)
+                    : getUnionType(propTypes, 1 /* Literal */, aliasSymbol, aliasTypeArguments);
             }
-            var match = discriminable.indexOf(/*searchElement*/ true);
-            // make sure exactly 1 matches before returning it
-            return match === -1 || discriminable.indexOf(/*searchElement*/ true, match + 1) !== -1 ? defaultValue : target.types[match];
+            return getPropertyTypeForIndexType(objectType, apparentObjectType, indexType, indexType, /* supressNoImplicitAnyError */ false, accessNode, accessFlags | 4 /* CacheSymbol */, shouldIncludeUndefined, /* reportDeprecated */ true);
         }
-        /**
-         * A type is 'weak' if it is an object type with at least one optional property
-         * and no required properties, call/construct signatures or index signatures
-         */
-        function isWeakType(type) {
-            if (type.flags & 524288 /* Object */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                return resolved.callSignatures.length === 0 && resolved.constructSignatures.length === 0 &&
-                    !resolved.stringIndexInfo && !resolved.numberIndexInfo &&
-                    resolved.properties.length > 0 &&
-                    ts.every(resolved.properties, function (p) { return !!(p.flags & 16777216 /* Optional */); });
-            }
-            if (type.flags & 2097152 /* Intersection */) {
-                return ts.every(type.types, isWeakType);
+        function getTypeFromIndexedAccessTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                var objectType = getTypeFromTypeNode(node.objectType);
+                var indexType = getTypeFromTypeNode(node.indexType);
+                var potentialAlias = getAliasSymbolForTypeNode(node);
+                var resolved = getIndexedAccessType(objectType, indexType, /*noUncheckedIndexedAccessCandidate*/ undefined, node, potentialAlias, getTypeArgumentsForAliasSymbol(potentialAlias));
+                links.resolvedType = resolved.flags & 8388608 /* IndexedAccess */ &&
+                    resolved.objectType === objectType &&
+                    resolved.indexType === indexType ?
+                    getConditionalFlowTypeOfType(resolved, node) : resolved;
             }
-            return false;
+            return links.resolvedType;
         }
-        function hasCommonProperties(source, target, isComparingJsxAttributes) {
-            for (var _i = 0, _a = getPropertiesOfType(source); _i < _a.length; _i++) {
-                var prop = _a[_i];
-                if (isKnownProperty(target, prop.escapedName, isComparingJsxAttributes)) {
-                    return true;
-                }
+        function getTypeFromMappedTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                var type = createObjectType(32 /* Mapped */, node.symbol);
+                type.declaration = node;
+                type.aliasSymbol = getAliasSymbolForTypeNode(node);
+                type.aliasTypeArguments = getTypeArgumentsForAliasSymbol(type.aliasSymbol);
+                links.resolvedType = type;
+                // Eagerly resolve the constraint type which forces an error if the constraint type circularly
+                // references itself through one or more type aliases.
+                getConstraintTypeFromMappedType(type);
             }
-            return false;
-        }
-        // Return a type reference where the source type parameter is replaced with the target marker
-        // type, and flag the result as a marker type reference.
-        function getMarkerTypeReference(type, source, target) {
-            var result = createTypeReference(type, ts.map(type.typeParameters, function (t) { return t === source ? target : t; }));
-            result.objectFlags |= 8192 /* MarkerType */;
-            return result;
+            return links.resolvedType;
         }
-        function getAliasVariances(symbol) {
-            var links = getSymbolLinks(symbol);
-            return getVariancesWorker(links.typeParameters, links, function (_links, param, marker) {
-                var type = getTypeAliasInstantiation(symbol, instantiateTypes(links.typeParameters, makeUnaryTypeMapper(param, marker)));
-                type.aliasTypeArgumentsContainsMarker = true;
-                return type;
-            });
+        function getActualTypeVariable(type) {
+            if (type.flags & 33554432 /* Substitution */) {
+                return type.baseType;
+            }
+            if (type.flags & 8388608 /* IndexedAccess */ && (type.objectType.flags & 33554432 /* Substitution */ ||
+                type.indexType.flags & 33554432 /* Substitution */)) {
+                return getIndexedAccessType(getActualTypeVariable(type.objectType), getActualTypeVariable(type.indexType));
+            }
+            return type;
         }
-        // Return an array containing the variance of each type parameter. The variance is effectively
-        // a digest of the type comparisons that occur for each type argument when instantiations of the
-        // generic type are structurally compared. We infer the variance information by comparing
-        // instantiations of the generic type for type arguments with known relations. The function
-        // returns the emptyArray singleton when invoked recursively for the given generic type.
-        function getVariancesWorker(typeParameters, cache, createMarkerType) {
-            if (typeParameters === void 0) { typeParameters = ts.emptyArray; }
-            var variances = cache.variances;
-            if (!variances) {
-                // The emptyArray singleton is used to signal a recursive invocation.
-                cache.variances = ts.emptyArray;
-                variances = [];
-                var _loop_16 = function (tp) {
-                    var unmeasurable = false;
-                    var unreliable = false;
-                    var oldHandler = outofbandVarianceMarkerHandler;
-                    outofbandVarianceMarkerHandler = function (onlyUnreliable) { return onlyUnreliable ? unreliable = true : unmeasurable = true; };
-                    // We first compare instantiations where the type parameter is replaced with
-                    // marker types that have a known subtype relationship. From this we can infer
-                    // invariance, covariance, contravariance or bivariance.
-                    var typeWithSuper = createMarkerType(cache, tp, markerSuperType);
-                    var typeWithSub = createMarkerType(cache, tp, markerSubType);
-                    var variance = (isTypeAssignableTo(typeWithSub, typeWithSuper) ? 1 /* Covariant */ : 0) |
-                        (isTypeAssignableTo(typeWithSuper, typeWithSub) ? 2 /* Contravariant */ : 0);
-                    // If the instantiations appear to be related bivariantly it may be because the
-                    // type parameter is independent (i.e. it isn't witnessed anywhere in the generic
-                    // type). To determine this we compare instantiations where the type parameter is
-                    // replaced with marker types that are known to be unrelated.
-                    if (variance === 3 /* Bivariant */ && isTypeAssignableTo(createMarkerType(cache, tp, markerOtherType), typeWithSuper)) {
-                        variance = 4 /* Independent */;
+        function getConditionalType(root, mapper) {
+            var result;
+            var extraTypes;
+            var _loop_15 = function () {
+                var checkType = instantiateType(root.checkType, mapper);
+                var checkTypeInstantiable = isGenericObjectType(checkType) || isGenericIndexType(checkType);
+                var extendsType = instantiateType(root.extendsType, mapper);
+                if (checkType === wildcardType || extendsType === wildcardType) {
+                    return { value: wildcardType };
+                }
+                var combinedMapper = void 0;
+                if (root.inferTypeParameters) {
+                    var context = createInferenceContext(root.inferTypeParameters, /*signature*/ undefined, 0 /* None */);
+                    // We skip inference of the possible `infer` types unles the `extendsType` _is_ an infer type
+                    // if it was, it's trivial to say that extendsType = checkType, however such a pattern is used to
+                    // "reset" the type being build up during constraint calculation and avoid making an apparently "infinite" constraint
+                    // so in those cases we refain from performing inference and retain the uninfered type parameter
+                    if (!checkTypeInstantiable || !ts.some(root.inferTypeParameters, function (t) { return t === extendsType; })) {
+                        // We don't want inferences from constraints as they may cause us to eagerly resolve the
+                        // conditional type instead of deferring resolution. Also, we always want strict function
+                        // types rules (i.e. proper contravariance) for inferences.
+                        inferTypes(context.inferences, checkType, extendsType, 256 /* NoConstraints */ | 512 /* AlwaysStrict */);
                     }
-                    outofbandVarianceMarkerHandler = oldHandler;
-                    if (unmeasurable || unreliable) {
-                        if (unmeasurable) {
-                            variance |= 8 /* Unmeasurable */;
+                    combinedMapper = mergeTypeMappers(mapper, context.mapper);
+                }
+                // Instantiate the extends type including inferences for 'infer T' type parameters
+                var inferredExtendsType = combinedMapper ? instantiateType(root.extendsType, combinedMapper) : extendsType;
+                // We attempt to resolve the conditional type only when the check and extends types are non-generic
+                if (!checkTypeInstantiable && !isGenericObjectType(inferredExtendsType) && !isGenericIndexType(inferredExtendsType)) {
+                    // Return falseType for a definitely false extends check. We check an instantiations of the two
+                    // types with type parameters mapped to the wildcard type, the most permissive instantiations
+                    // possible (the wildcard type is assignable to and from all types). If those are not related,
+                    // then no instantiations will be and we can just return the false branch type.
+                    if (!(inferredExtendsType.flags & 3 /* AnyOrUnknown */) && (checkType.flags & 1 /* Any */ || !isTypeAssignableTo(getPermissiveInstantiation(checkType), getPermissiveInstantiation(inferredExtendsType)))) {
+                        // Return union of trueType and falseType for 'any' since it matches anything
+                        if (checkType.flags & 1 /* Any */) {
+                            (extraTypes || (extraTypes = [])).push(instantiateType(getTypeFromTypeNode(root.node.trueType), combinedMapper || mapper));
                         }
-                        if (unreliable) {
-                            variance |= 16 /* Unreliable */;
+                        // If falseType is an immediately nested conditional type that isn't distributive or has an
+                        // identical checkType, switch to that type and loop.
+                        var falseType_1 = getTypeFromTypeNode(root.node.falseType);
+                        if (falseType_1.flags & 16777216 /* Conditional */) {
+                            var newRoot = falseType_1.root;
+                            if (newRoot.node.parent === root.node && (!newRoot.isDistributive || newRoot.checkType === root.checkType)) {
+                                root = newRoot;
+                                return "continue";
+                            }
                         }
+                        result = instantiateType(falseType_1, mapper);
+                        return "break";
+                    }
+                    // Return trueType for a definitely true extends check. We check instantiations of the two
+                    // types with type parameters mapped to their restrictive form, i.e. a form of the type parameter
+                    // that has no constraint. This ensures that, for example, the type
+                    //   type Foo<T extends { x: any }> = T extends { x: string } ? string : number
+                    // doesn't immediately resolve to 'string' instead of being deferred.
+                    if (inferredExtendsType.flags & 3 /* AnyOrUnknown */ || isTypeAssignableTo(getRestrictiveInstantiation(checkType), getRestrictiveInstantiation(inferredExtendsType))) {
+                        result = instantiateType(getTypeFromTypeNode(root.node.trueType), combinedMapper || mapper);
+                        return "break";
                     }
-                    variances.push(variance);
-                };
-                for (var _i = 0, typeParameters_1 = typeParameters; _i < typeParameters_1.length; _i++) {
-                    var tp = typeParameters_1[_i];
-                    _loop_16(tp);
                 }
-                cache.variances = variances;
+                // Return a deferred type for a check that is neither definitely true nor definitely false
+                var erasedCheckType = getActualTypeVariable(checkType);
+                result = createType(16777216 /* Conditional */);
+                result.root = root;
+                result.checkType = erasedCheckType;
+                result.extendsType = extendsType;
+                result.mapper = mapper;
+                result.combinedMapper = combinedMapper;
+                result.aliasSymbol = root.aliasSymbol;
+                result.aliasTypeArguments = instantiateTypes(root.aliasTypeArguments, mapper); // TODO: GH#18217
+                return "break";
+            };
+            // We loop here for an immediately nested conditional type in the false position, effectively treating
+            // types of the form 'A extends B ? X : C extends D ? Y : E extends F ? Z : ...' as a single construct for
+            // purposes of resolution. This means such types aren't subject to the instatiation depth limiter.
+            while (true) {
+                var state_4 = _loop_15();
+                if (typeof state_4 === "object")
+                    return state_4.value;
+                if (state_4 === "break")
+                    break;
             }
-            return variances;
+            return extraTypes ? getUnionType(ts.append(extraTypes, result)) : result;
         }
-        function getVariances(type) {
-            // Arrays and tuples are known to be covariant, no need to spend time computing this.
-            if (type === globalArrayType || type === globalReadonlyArrayType || type.objectFlags & 8 /* Tuple */) {
-                return arrayVariances;
+        function getTrueTypeFromConditionalType(type) {
+            return type.resolvedTrueType || (type.resolvedTrueType = instantiateType(getTypeFromTypeNode(type.root.node.trueType), type.mapper));
+        }
+        function getFalseTypeFromConditionalType(type) {
+            return type.resolvedFalseType || (type.resolvedFalseType = instantiateType(getTypeFromTypeNode(type.root.node.falseType), type.mapper));
+        }
+        function getInferredTrueTypeFromConditionalType(type) {
+            return type.resolvedInferredTrueType || (type.resolvedInferredTrueType = type.combinedMapper ? instantiateType(getTypeFromTypeNode(type.root.node.trueType), type.combinedMapper) : getTrueTypeFromConditionalType(type));
+        }
+        function getInferTypeParameters(node) {
+            var result;
+            if (node.locals) {
+                node.locals.forEach(function (symbol) {
+                    if (symbol.flags & 262144 /* TypeParameter */) {
+                        result = ts.append(result, getDeclaredTypeOfSymbol(symbol));
+                    }
+                });
             }
-            return getVariancesWorker(type.typeParameters, type, getMarkerTypeReference);
+            return result;
         }
-        // Return true if the given type reference has a 'void' type argument for a covariant type parameter.
-        // See comment at call in recursiveTypeRelatedTo for when this case matters.
-        function hasCovariantVoidArgument(typeArguments, variances) {
-            for (var i = 0; i < variances.length; i++) {
-                if ((variances[i] & 7 /* VarianceMask */) === 1 /* Covariant */ && typeArguments[i].flags & 16384 /* Void */) {
-                    return true;
+        function getTypeFromConditionalTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                var checkType = getTypeFromTypeNode(node.checkType);
+                var aliasSymbol = getAliasSymbolForTypeNode(node);
+                var aliasTypeArguments = getTypeArgumentsForAliasSymbol(aliasSymbol);
+                var allOuterTypeParameters = getOuterTypeParameters(node, /*includeThisTypes*/ true);
+                var outerTypeParameters = aliasTypeArguments ? allOuterTypeParameters : ts.filter(allOuterTypeParameters, function (tp) { return isTypeParameterPossiblyReferenced(tp, node); });
+                var root = {
+                    node: node,
+                    checkType: checkType,
+                    extendsType: getTypeFromTypeNode(node.extendsType),
+                    isDistributive: !!(checkType.flags & 262144 /* TypeParameter */),
+                    inferTypeParameters: getInferTypeParameters(node),
+                    outerTypeParameters: outerTypeParameters,
+                    instantiations: undefined,
+                    aliasSymbol: aliasSymbol,
+                    aliasTypeArguments: aliasTypeArguments
+                };
+                links.resolvedType = getConditionalType(root, /*mapper*/ undefined);
+                if (outerTypeParameters) {
+                    root.instantiations = new ts.Map();
+                    root.instantiations.set(getTypeListId(outerTypeParameters), links.resolvedType);
                 }
             }
-            return false;
-        }
-        function isUnconstrainedTypeParameter(type) {
-            return type.flags & 262144 /* TypeParameter */ && !getConstraintOfTypeParameter(type);
+            return links.resolvedType;
         }
-        function isNonDeferredTypeReference(type) {
-            return !!(ts.getObjectFlags(type) & 4 /* Reference */) && !type.node;
+        function getTypeFromInferTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                links.resolvedType = getDeclaredTypeOfTypeParameter(getSymbolOfNode(node.typeParameter));
+            }
+            return links.resolvedType;
         }
-        function isTypeReferenceWithGenericArguments(type) {
-            return isNonDeferredTypeReference(type) && ts.some(getTypeArguments(type), function (t) { return isUnconstrainedTypeParameter(t) || isTypeReferenceWithGenericArguments(t); });
+        function getIdentifierChain(node) {
+            if (ts.isIdentifier(node)) {
+                return [node];
+            }
+            else {
+                return ts.append(getIdentifierChain(node.left), node.right);
+            }
         }
-        /**
-         * getTypeReferenceId(A<T, number, U>) returns "111=0-12=1"
-         *   where A.id=111 and number.id=12
-         */
-        function getTypeReferenceId(type, typeParameters, depth) {
-            if (depth === void 0) { depth = 0; }
-            var result = "" + type.target.id;
-            for (var _i = 0, _a = getTypeArguments(type); _i < _a.length; _i++) {
-                var t = _a[_i];
-                if (isUnconstrainedTypeParameter(t)) {
-                    var index = typeParameters.indexOf(t);
-                    if (index < 0) {
-                        index = typeParameters.length;
-                        typeParameters.push(t);
-                    }
-                    result += "=" + index;
+        function getTypeFromImportTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                if (node.isTypeOf && node.typeArguments) { // Only the non-typeof form can make use of type arguments
+                    error(node, ts.Diagnostics.Type_arguments_cannot_be_used_here);
+                    links.resolvedSymbol = unknownSymbol;
+                    return links.resolvedType = errorType;
                 }
-                else if (depth < 4 && isTypeReferenceWithGenericArguments(t)) {
-                    result += "<" + getTypeReferenceId(t, typeParameters, depth + 1) + ">";
+                if (!ts.isLiteralImportTypeNode(node)) {
+                    error(node.argument, ts.Diagnostics.String_literal_expected);
+                    links.resolvedSymbol = unknownSymbol;
+                    return links.resolvedType = errorType;
+                }
+                var targetMeaning = node.isTypeOf ? 111551 /* Value */ : node.flags & 4194304 /* JSDoc */ ? 111551 /* Value */ | 788968 /* Type */ : 788968 /* Type */;
+                // TODO: Future work: support unions/generics/whatever via a deferred import-type
+                var innerModuleSymbol = resolveExternalModuleName(node, node.argument.literal);
+                if (!innerModuleSymbol) {
+                    links.resolvedSymbol = unknownSymbol;
+                    return links.resolvedType = errorType;
+                }
+                var moduleSymbol = resolveExternalModuleSymbol(innerModuleSymbol, /*dontResolveAlias*/ false);
+                if (!ts.nodeIsMissing(node.qualifier)) {
+                    var nameStack = getIdentifierChain(node.qualifier);
+                    var currentNamespace = moduleSymbol;
+                    var current = void 0;
+                    while (current = nameStack.shift()) {
+                        var meaning = nameStack.length ? 1920 /* Namespace */ : targetMeaning;
+                        // typeof a.b.c is normally resolved using `checkExpression` which in turn defers to `checkQualifiedName`
+                        // That, in turn, ultimately uses `getPropertyOfType` on the type of the symbol, which differs slightly from
+                        // the `exports` lookup process that only looks up namespace members which is used for most type references
+                        var mergedResolvedSymbol = getMergedSymbol(resolveSymbol(currentNamespace));
+                        var next = node.isTypeOf
+                            ? getPropertyOfType(getTypeOfSymbol(mergedResolvedSymbol), current.escapedText)
+                            : getSymbol(getExportsOfSymbol(mergedResolvedSymbol), current.escapedText, meaning);
+                        if (!next) {
+                            error(current, ts.Diagnostics.Namespace_0_has_no_exported_member_1, getFullyQualifiedName(currentNamespace), ts.declarationNameToString(current));
+                            return links.resolvedType = errorType;
+                        }
+                        getNodeLinks(current).resolvedSymbol = next;
+                        getNodeLinks(current.parent).resolvedSymbol = next;
+                        currentNamespace = next;
+                    }
+                    links.resolvedType = resolveImportSymbolType(node, links, currentNamespace, targetMeaning);
                 }
                 else {
-                    result += "-" + t.id;
+                    if (moduleSymbol.flags & targetMeaning) {
+                        links.resolvedType = resolveImportSymbolType(node, links, moduleSymbol, targetMeaning);
+                    }
+                    else {
+                        var errorMessage = targetMeaning === 111551 /* Value */
+                            ? ts.Diagnostics.Module_0_does_not_refer_to_a_value_but_is_used_as_a_value_here
+                            : ts.Diagnostics.Module_0_does_not_refer_to_a_type_but_is_used_as_a_type_here_Did_you_mean_typeof_import_0;
+                        error(node, errorMessage, node.argument.literal.text);
+                        links.resolvedSymbol = unknownSymbol;
+                        links.resolvedType = errorType;
+                    }
                 }
             }
-            return result;
+            return links.resolvedType;
         }
-        /**
-         * To improve caching, the relation key for two generic types uses the target's id plus ids of the type parameters.
-         * For other cases, the types ids are used.
-         */
-        function getRelationKey(source, target, intersectionState, relation) {
-            if (relation === identityRelation && source.id > target.id) {
-                var temp = source;
-                source = target;
-                target = temp;
+        function resolveImportSymbolType(node, links, symbol, meaning) {
+            var resolvedSymbol = resolveSymbol(symbol);
+            links.resolvedSymbol = resolvedSymbol;
+            if (meaning === 111551 /* Value */) {
+                return getTypeOfSymbol(symbol); // intentionally doesn't use resolved symbol so type is cached as expected on the alias
             }
-            var postFix = intersectionState ? ":" + intersectionState : "";
-            if (isTypeReferenceWithGenericArguments(source) && isTypeReferenceWithGenericArguments(target)) {
-                var typeParameters = [];
-                return getTypeReferenceId(source, typeParameters) + "," + getTypeReferenceId(target, typeParameters) + postFix;
+            else {
+                return getTypeReferenceType(node, resolvedSymbol); // getTypeReferenceType doesn't handle aliases - it must get the resolved symbol
             }
-            return source.id + "," + target.id + postFix;
         }
-        // Invoke the callback for each underlying property symbol of the given symbol and return the first
-        // value that isn't undefined.
-        function forEachProperty(prop, callback) {
-            if (ts.getCheckFlags(prop) & 6 /* Synthetic */) {
-                for (var _i = 0, _a = prop.containingType.types; _i < _a.length; _i++) {
-                    var t = _a[_i];
-                    var p = getPropertyOfType(t, prop.escapedName);
-                    var result = p && forEachProperty(p, callback);
-                    if (result) {
-                        return result;
+        function getTypeFromTypeLiteralOrFunctionOrConstructorTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                // Deferred resolution of members is handled by resolveObjectTypeMembers
+                var aliasSymbol = getAliasSymbolForTypeNode(node);
+                if (getMembersOfSymbol(node.symbol).size === 0 && !aliasSymbol) {
+                    links.resolvedType = emptyTypeLiteralType;
+                }
+                else {
+                    var type = createObjectType(16 /* Anonymous */, node.symbol);
+                    type.aliasSymbol = aliasSymbol;
+                    type.aliasTypeArguments = getTypeArgumentsForAliasSymbol(aliasSymbol);
+                    if (ts.isJSDocTypeLiteral(node) && node.isArrayType) {
+                        type = createArrayType(type);
                     }
+                    links.resolvedType = type;
                 }
-                return undefined;
             }
-            return callback(prop);
+            return links.resolvedType;
         }
-        // Return the declaring class type of a property or undefined if property not declared in class
-        function getDeclaringClass(prop) {
-            return prop.parent && prop.parent.flags & 32 /* Class */ ? getDeclaredTypeOfSymbol(getParentOfSymbol(prop)) : undefined;
+        function getAliasSymbolForTypeNode(node) {
+            var host = node.parent;
+            while (ts.isParenthesizedTypeNode(host) || ts.isJSDocTypeExpression(host) || ts.isTypeOperatorNode(host) && host.operator === 142 /* ReadonlyKeyword */) {
+                host = host.parent;
+            }
+            return ts.isTypeAlias(host) ? getSymbolOfNode(host) : undefined;
         }
-        // Return true if some underlying source property is declared in a class that derives
-        // from the given base class.
-        function isPropertyInClassDerivedFrom(prop, baseClass) {
-            return forEachProperty(prop, function (sp) {
-                var sourceClass = getDeclaringClass(sp);
-                return sourceClass ? hasBaseType(sourceClass, baseClass) : false;
-            });
+        function getTypeArgumentsForAliasSymbol(symbol) {
+            return symbol ? getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(symbol) : undefined;
         }
-        // Return true if source property is a valid override of protected parts of target property.
-        function isValidOverrideOf(sourceProp, targetProp) {
-            return !forEachProperty(targetProp, function (tp) { return ts.getDeclarationModifierFlagsFromSymbol(tp) & 16 /* Protected */ ?
-                !isPropertyInClassDerivedFrom(sourceProp, getDeclaringClass(tp)) : false; });
+        function isNonGenericObjectType(type) {
+            return !!(type.flags & 524288 /* Object */) && !isGenericMappedType(type);
         }
-        // Return true if the given class derives from each of the declaring classes of the protected
-        // constituents of the given property.
-        function isClassDerivedFromDeclaringClasses(checkClass, prop) {
-            return forEachProperty(prop, function (p) { return ts.getDeclarationModifierFlagsFromSymbol(p) & 16 /* Protected */ ?
-                !hasBaseType(checkClass, getDeclaringClass(p)) : false; }) ? undefined : checkClass;
+        function isEmptyObjectTypeOrSpreadsIntoEmptyObject(type) {
+            return isEmptyObjectType(type) || !!(type.flags & (65536 /* Null */ | 32768 /* Undefined */ | 528 /* BooleanLike */ | 296 /* NumberLike */ | 2112 /* BigIntLike */ | 402653316 /* StringLike */ | 1056 /* EnumLike */ | 67108864 /* NonPrimitive */ | 4194304 /* Index */));
         }
-        // Return true if the given type is deeply nested. We consider this to be the case when structural type comparisons
-        // for 5 or more occurrences or instantiations of the type have been recorded on the given stack. It is possible,
-        // though highly unlikely, for this test to be true in a situation where a chain of instantiations is not infinitely
-        // expanding. Effectively, we will generate a false positive when two types are structurally equal to at least 5
-        // levels, but unequal at some level beyond that.
-        // In addition, this will also detect when an indexed access has been chained off of 5 or more times (which is essentially
-        // the dual of the structural comparison), and likewise mark the type as deeply nested, potentially adding false positives
-        // for finite but deeply expanding indexed accesses (eg, for `Q[P1][P2][P3][P4][P5]`).
-        function isDeeplyNestedType(type, stack, depth) {
-            // We track all object types that have an associated symbol (representing the origin of the type)
-            if (depth >= 5 && type.flags & 524288 /* Object */ && !isObjectOrArrayLiteralType(type)) {
-                var symbol = type.symbol;
-                if (symbol) {
-                    var count = 0;
-                    for (var i = 0; i < depth; i++) {
-                        var t = stack[i];
-                        if (t.flags & 524288 /* Object */ && t.symbol === symbol) {
-                            count++;
-                            if (count >= 5)
-                                return true;
-                        }
-                    }
+        function tryMergeUnionOfObjectTypeAndEmptyObject(type, readonly) {
+            if (type.types.length === 2) {
+                var firstType = type.types[0];
+                var secondType = type.types[1];
+                if (ts.every(type.types, isEmptyObjectTypeOrSpreadsIntoEmptyObject)) {
+                    return isEmptyObjectType(firstType) ? firstType : isEmptyObjectType(secondType) ? secondType : emptyObjectType;
+                }
+                if (isEmptyObjectTypeOrSpreadsIntoEmptyObject(firstType)) {
+                    return getAnonymousPartialType(secondType);
+                }
+                if (isEmptyObjectTypeOrSpreadsIntoEmptyObject(secondType)) {
+                    return getAnonymousPartialType(firstType);
                 }
             }
-            if (depth >= 5 && type.flags & 8388608 /* IndexedAccess */) {
-                var root = getRootObjectTypeFromIndexedAccessChain(type);
-                var count = 0;
-                for (var i = 0; i < depth; i++) {
-                    var t = stack[i];
-                    if (getRootObjectTypeFromIndexedAccessChain(t) === root) {
-                        count++;
-                        if (count >= 5)
-                            return true;
+            function getAnonymousPartialType(type) {
+                // gets the type as if it had been spread, but where everything in the spread is made optional
+                var members = ts.createSymbolTable();
+                for (var _i = 0, _a = getPropertiesOfType(type); _i < _a.length; _i++) {
+                    var prop = _a[_i];
+                    if (ts.getDeclarationModifierFlagsFromSymbol(prop) & (8 /* Private */ | 16 /* Protected */)) {
+                        // do nothing, skip privates
+                    }
+                    else if (isSpreadableProperty(prop)) {
+                        var isSetonlyAccessor = prop.flags & 65536 /* SetAccessor */ && !(prop.flags & 32768 /* GetAccessor */);
+                        var flags = 4 /* Property */ | 16777216 /* Optional */;
+                        var result = createSymbol(flags, prop.escapedName, readonly ? 8 /* Readonly */ : 0);
+                        result.type = isSetonlyAccessor ? undefinedType : getTypeOfSymbol(prop);
+                        result.declarations = prop.declarations;
+                        result.nameType = getSymbolLinks(prop).nameType;
+                        result.syntheticOrigin = prop;
+                        members.set(prop.escapedName, result);
                     }
                 }
+                var spread = createAnonymousType(type.symbol, members, ts.emptyArray, ts.emptyArray, getIndexInfoOfType(type, 0 /* String */), getIndexInfoOfType(type, 1 /* Number */));
+                spread.objectFlags |= 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
+                return spread;
             }
-            return false;
         }
         /**
-         * Gets the leftmost object type in a chain of indexed accesses, eg, in A[P][Q], returns A
+         * Since the source of spread types are object literals, which are not binary,
+         * this function should be called in a left folding style, with left = previous result of getSpreadType
+         * and right = the new element to be spread.
          */
-        function getRootObjectTypeFromIndexedAccessChain(type) {
-            var t = type;
-            while (t.flags & 8388608 /* IndexedAccess */) {
-                t = t.objectType;
+        function getSpreadType(left, right, symbol, objectFlags, readonly) {
+            if (left.flags & 1 /* Any */ || right.flags & 1 /* Any */) {
+                return anyType;
             }
-            return t;
-        }
-        function isPropertyIdenticalTo(sourceProp, targetProp) {
-            return compareProperties(sourceProp, targetProp, compareTypesIdentical) !== 0 /* False */;
-        }
-        function compareProperties(sourceProp, targetProp, compareTypes) {
-            // Two members are considered identical when
-            // - they are public properties with identical names, optionality, and types,
-            // - they are private or protected properties originating in the same declaration and having identical types
-            if (sourceProp === targetProp) {
-                return -1 /* True */;
+            if (left.flags & 2 /* Unknown */ || right.flags & 2 /* Unknown */) {
+                return unknownType;
             }
-            var sourcePropAccessibility = ts.getDeclarationModifierFlagsFromSymbol(sourceProp) & 24 /* NonPublicAccessibilityModifier */;
-            var targetPropAccessibility = ts.getDeclarationModifierFlagsFromSymbol(targetProp) & 24 /* NonPublicAccessibilityModifier */;
-            if (sourcePropAccessibility !== targetPropAccessibility) {
-                return 0 /* False */;
+            if (left.flags & 131072 /* Never */) {
+                return right;
             }
-            if (sourcePropAccessibility) {
-                if (getTargetSymbol(sourceProp) !== getTargetSymbol(targetProp)) {
-                    return 0 /* False */;
-                }
+            if (right.flags & 131072 /* Never */) {
+                return left;
             }
-            else {
-                if ((sourceProp.flags & 16777216 /* Optional */) !== (targetProp.flags & 16777216 /* Optional */)) {
-                    return 0 /* False */;
+            if (left.flags & 1048576 /* Union */) {
+                var merged = tryMergeUnionOfObjectTypeAndEmptyObject(left, readonly);
+                if (merged) {
+                    return getSpreadType(merged, right, symbol, objectFlags, readonly);
                 }
+                return checkCrossProductUnion([left, right])
+                    ? mapType(left, function (t) { return getSpreadType(t, right, symbol, objectFlags, readonly); })
+                    : errorType;
             }
-            if (isReadonlySymbol(sourceProp) !== isReadonlySymbol(targetProp)) {
-                return 0 /* False */;
-            }
-            return compareTypes(getTypeOfSymbol(sourceProp), getTypeOfSymbol(targetProp));
-        }
-        function isMatchingSignature(source, target, partialMatch) {
-            var sourceParameterCount = getParameterCount(source);
-            var targetParameterCount = getParameterCount(target);
-            var sourceMinArgumentCount = getMinArgumentCount(source);
-            var targetMinArgumentCount = getMinArgumentCount(target);
-            var sourceHasRestParameter = hasEffectiveRestParameter(source);
-            var targetHasRestParameter = hasEffectiveRestParameter(target);
-            // A source signature matches a target signature if the two signatures have the same number of required,
-            // optional, and rest parameters.
-            if (sourceParameterCount === targetParameterCount &&
-                sourceMinArgumentCount === targetMinArgumentCount &&
-                sourceHasRestParameter === targetHasRestParameter) {
-                return true;
+            if (right.flags & 1048576 /* Union */) {
+                var merged = tryMergeUnionOfObjectTypeAndEmptyObject(right, readonly);
+                if (merged) {
+                    return getSpreadType(left, merged, symbol, objectFlags, readonly);
+                }
+                return checkCrossProductUnion([left, right])
+                    ? mapType(right, function (t) { return getSpreadType(left, t, symbol, objectFlags, readonly); })
+                    : errorType;
             }
-            // A source signature partially matches a target signature if the target signature has no fewer required
-            // parameters
-            if (partialMatch && sourceMinArgumentCount <= targetMinArgumentCount) {
-                return true;
+            if (right.flags & (528 /* BooleanLike */ | 296 /* NumberLike */ | 2112 /* BigIntLike */ | 402653316 /* StringLike */ | 1056 /* EnumLike */ | 67108864 /* NonPrimitive */ | 4194304 /* Index */)) {
+                return left;
             }
-            return false;
-        }
-        /**
-         * See signatureRelatedTo, compareSignaturesIdentical
-         */
-        function compareSignaturesIdentical(source, target, partialMatch, ignoreThisTypes, ignoreReturnTypes, compareTypes) {
-            // TODO (drosen): De-duplicate code between related functions.
-            if (source === target) {
-                return -1 /* True */;
+            if (isGenericObjectType(left) || isGenericObjectType(right)) {
+                if (isEmptyObjectType(left)) {
+                    return right;
+                }
+                // When the left type is an intersection, we may need to merge the last constituent of the
+                // intersection with the right type. For example when the left type is 'T & { a: string }'
+                // and the right type is '{ b: string }' we produce 'T & { a: string, b: string }'.
+                if (left.flags & 2097152 /* Intersection */) {
+                    var types = left.types;
+                    var lastLeft = types[types.length - 1];
+                    if (isNonGenericObjectType(lastLeft) && isNonGenericObjectType(right)) {
+                        return getIntersectionType(ts.concatenate(types.slice(0, types.length - 1), [getSpreadType(lastLeft, right, symbol, objectFlags, readonly)]));
+                    }
+                }
+                return getIntersectionType([left, right]);
             }
-            if (!(isMatchingSignature(source, target, partialMatch))) {
-                return 0 /* False */;
+            var members = ts.createSymbolTable();
+            var skippedPrivateMembers = new ts.Set();
+            var stringIndexInfo;
+            var numberIndexInfo;
+            if (left === emptyObjectType) {
+                // for the first spread element, left === emptyObjectType, so take the right's string indexer
+                stringIndexInfo = getIndexInfoOfType(right, 0 /* String */);
+                numberIndexInfo = getIndexInfoOfType(right, 1 /* Number */);
             }
-            // Check that the two signatures have the same number of type parameters.
-            if (ts.length(source.typeParameters) !== ts.length(target.typeParameters)) {
-                return 0 /* False */;
+            else {
+                stringIndexInfo = unionSpreadIndexInfos(getIndexInfoOfType(left, 0 /* String */), getIndexInfoOfType(right, 0 /* String */));
+                numberIndexInfo = unionSpreadIndexInfos(getIndexInfoOfType(left, 1 /* Number */), getIndexInfoOfType(right, 1 /* Number */));
             }
-            // Check that type parameter constraints and defaults match. If they do, instantiate the source
-            // signature with the type parameters of the target signature and continue the comparison.
-            if (target.typeParameters) {
-                var mapper = createTypeMapper(source.typeParameters, target.typeParameters);
-                for (var i = 0; i < target.typeParameters.length; i++) {
-                    var s = source.typeParameters[i];
-                    var t = target.typeParameters[i];
-                    if (!(s === t || compareTypes(instantiateType(getConstraintFromTypeParameter(s), mapper) || unknownType, getConstraintFromTypeParameter(t) || unknownType) &&
-                        compareTypes(instantiateType(getDefaultFromTypeParameter(s), mapper) || unknownType, getDefaultFromTypeParameter(t) || unknownType))) {
-                        return 0 /* False */;
-                    }
+            for (var _i = 0, _a = getPropertiesOfType(right); _i < _a.length; _i++) {
+                var rightProp = _a[_i];
+                if (ts.getDeclarationModifierFlagsFromSymbol(rightProp) & (8 /* Private */ | 16 /* Protected */)) {
+                    skippedPrivateMembers.add(rightProp.escapedName);
+                }
+                else if (isSpreadableProperty(rightProp)) {
+                    members.set(rightProp.escapedName, getSpreadSymbol(rightProp, readonly));
                 }
-                source = instantiateSignature(source, mapper, /*eraseTypeParameters*/ true);
             }
-            var result = -1 /* True */;
-            if (!ignoreThisTypes) {
-                var sourceThisType = getThisTypeOfSignature(source);
-                if (sourceThisType) {
-                    var targetThisType = getThisTypeOfSignature(target);
-                    if (targetThisType) {
-                        var related = compareTypes(sourceThisType, targetThisType);
-                        if (!related) {
-                            return 0 /* False */;
-                        }
-                        result &= related;
+            for (var _b = 0, _c = getPropertiesOfType(left); _b < _c.length; _b++) {
+                var leftProp = _c[_b];
+                if (skippedPrivateMembers.has(leftProp.escapedName) || !isSpreadableProperty(leftProp)) {
+                    continue;
+                }
+                if (members.has(leftProp.escapedName)) {
+                    var rightProp = members.get(leftProp.escapedName);
+                    var rightType = getTypeOfSymbol(rightProp);
+                    if (rightProp.flags & 16777216 /* Optional */) {
+                        var declarations = ts.concatenate(leftProp.declarations, rightProp.declarations);
+                        var flags = 4 /* Property */ | (leftProp.flags & 16777216 /* Optional */);
+                        var result = createSymbol(flags, leftProp.escapedName);
+                        result.type = getUnionType([getTypeOfSymbol(leftProp), getTypeWithFacts(rightType, 524288 /* NEUndefined */)]);
+                        result.leftSpread = leftProp;
+                        result.rightSpread = rightProp;
+                        result.declarations = declarations;
+                        result.nameType = getSymbolLinks(leftProp).nameType;
+                        members.set(leftProp.escapedName, result);
                     }
                 }
-            }
-            var targetLen = getParameterCount(target);
-            for (var i = 0; i < targetLen; i++) {
-                var s = getTypeAtPosition(source, i);
-                var t = getTypeAtPosition(target, i);
-                var related = compareTypes(t, s);
-                if (!related) {
-                    return 0 /* False */;
+                else {
+                    members.set(leftProp.escapedName, getSpreadSymbol(leftProp, readonly));
                 }
-                result &= related;
             }
-            if (!ignoreReturnTypes) {
-                var sourceTypePredicate = getTypePredicateOfSignature(source);
-                var targetTypePredicate = getTypePredicateOfSignature(target);
-                result &= sourceTypePredicate || targetTypePredicate ?
-                    compareTypePredicatesIdentical(sourceTypePredicate, targetTypePredicate, compareTypes) :
-                    compareTypes(getReturnTypeOfSignature(source), getReturnTypeOfSignature(target));
+            var spread = createAnonymousType(symbol, members, ts.emptyArray, ts.emptyArray, getIndexInfoWithReadonly(stringIndexInfo, readonly), getIndexInfoWithReadonly(numberIndexInfo, readonly));
+            spread.objectFlags |= 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */ | 1024 /* ContainsSpread */ | objectFlags;
+            return spread;
+        }
+        /** We approximate own properties as non-methods plus methods that are inside the object literal */
+        function isSpreadableProperty(prop) {
+            return !ts.some(prop.declarations, ts.isPrivateIdentifierPropertyDeclaration) &&
+                (!(prop.flags & (8192 /* Method */ | 32768 /* GetAccessor */ | 65536 /* SetAccessor */)) ||
+                    !prop.declarations.some(function (decl) { return ts.isClassLike(decl.parent); }));
+        }
+        function getSpreadSymbol(prop, readonly) {
+            var isSetonlyAccessor = prop.flags & 65536 /* SetAccessor */ && !(prop.flags & 32768 /* GetAccessor */);
+            if (!isSetonlyAccessor && readonly === isReadonlySymbol(prop)) {
+                return prop;
             }
+            var flags = 4 /* Property */ | (prop.flags & 16777216 /* Optional */);
+            var result = createSymbol(flags, prop.escapedName, readonly ? 8 /* Readonly */ : 0);
+            result.type = isSetonlyAccessor ? undefinedType : getTypeOfSymbol(prop);
+            result.declarations = prop.declarations;
+            result.nameType = getSymbolLinks(prop).nameType;
+            result.syntheticOrigin = prop;
             return result;
         }
-        function compareTypePredicatesIdentical(source, target, compareTypes) {
-            return !(source && target && typePredicateKindsMatch(source, target)) ? 0 /* False */ :
-                source.type === target.type ? -1 /* True */ :
-                    source.type && target.type ? compareTypes(source.type, target.type) :
-                        0 /* False */;
+        function getIndexInfoWithReadonly(info, readonly) {
+            return info && info.isReadonly !== readonly ? createIndexInfo(info.type, readonly, info.declaration) : info;
         }
-        function literalTypesWithSameBaseType(types) {
-            var commonBaseType;
-            for (var _i = 0, types_12 = types; _i < types_12.length; _i++) {
-                var t = types_12[_i];
-                var baseType = getBaseTypeOfLiteralType(t);
-                if (!commonBaseType) {
-                    commonBaseType = baseType;
-                }
-                if (baseType === t || baseType !== commonBaseType) {
-                    return false;
+        function createLiteralType(flags, value, symbol) {
+            var type = createType(flags);
+            type.symbol = symbol;
+            type.value = value;
+            return type;
+        }
+        function getFreshTypeOfLiteralType(type) {
+            if (type.flags & 2944 /* Literal */) {
+                if (!type.freshType) {
+                    var freshType = createLiteralType(type.flags, type.value, type.symbol);
+                    freshType.regularType = type;
+                    freshType.freshType = freshType;
+                    type.freshType = freshType;
                 }
+                return type.freshType;
             }
-            return true;
+            return type;
         }
-        // When the candidate types are all literal types with the same base type, return a union
-        // of those literal types. Otherwise, return the leftmost type for which no type to the
-        // right is a supertype.
-        function getSupertypeOrUnion(types) {
-            return literalTypesWithSameBaseType(types) ?
-                getUnionType(types) :
-                ts.reduceLeft(types, function (s, t) { return isTypeSubtypeOf(s, t) ? t : s; });
+        function getRegularTypeOfLiteralType(type) {
+            return type.flags & 2944 /* Literal */ ? type.regularType :
+                type.flags & 1048576 /* Union */ ? (type.regularType || (type.regularType = getUnionType(ts.sameMap(type.types, getRegularTypeOfLiteralType)))) :
+                    type;
         }
-        function getCommonSupertype(types) {
-            if (!strictNullChecks) {
-                return getSupertypeOrUnion(types);
+        function isFreshLiteralType(type) {
+            return !!(type.flags & 2944 /* Literal */) && type.freshType === type;
+        }
+        function getLiteralType(value, enumId, symbol) {
+            // We store all literal types in a single map with keys of the form '#NNN' and '@SSS',
+            // where NNN is the text representation of a numeric literal and SSS are the characters
+            // of a string literal. For literal enum members we use 'EEE#NNN' and 'EEE@SSS', where
+            // EEE is a unique id for the containing enum type.
+            var qualifier = typeof value === "number" ? "#" : typeof value === "string" ? "@" : "n";
+            var key = (enumId ? enumId : "") + qualifier + (typeof value === "object" ? ts.pseudoBigIntToString(value) : value);
+            var type = literalTypes.get(key);
+            if (!type) {
+                var flags = (typeof value === "number" ? 256 /* NumberLiteral */ :
+                    typeof value === "string" ? 128 /* StringLiteral */ : 2048 /* BigIntLiteral */) |
+                    (enumId ? 1024 /* EnumLiteral */ : 0);
+                literalTypes.set(key, type = createLiteralType(flags, value, symbol));
+                type.regularType = type;
             }
-            var primaryTypes = ts.filter(types, function (t) { return !(t.flags & 98304 /* Nullable */); });
-            return primaryTypes.length ?
-                getNullableType(getSupertypeOrUnion(primaryTypes), getFalsyFlagsOfTypes(types) & 98304 /* Nullable */) :
-                getUnionType(types, 2 /* Subtype */);
+            return type;
         }
-        // Return the leftmost type for which no type to the right is a subtype.
-        function getCommonSubtype(types) {
-            return ts.reduceLeft(types, function (s, t) { return isTypeSubtypeOf(t, s) ? t : s; });
+        function getTypeFromLiteralTypeNode(node) {
+            if (node.literal.kind === 103 /* NullKeyword */) {
+                return nullType;
+            }
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                links.resolvedType = getRegularTypeOfLiteralType(checkExpression(node.literal));
+            }
+            return links.resolvedType;
         }
-        function isArrayType(type) {
-            return !!(ts.getObjectFlags(type) & 4 /* Reference */) && (type.target === globalArrayType || type.target === globalReadonlyArrayType);
+        function createUniqueESSymbolType(symbol) {
+            var type = createType(8192 /* UniqueESSymbol */);
+            type.symbol = symbol;
+            type.escapedName = "__@" + type.symbol.escapedName + "@" + getSymbolId(type.symbol);
+            return type;
         }
-        function isReadonlyArrayType(type) {
-            return !!(ts.getObjectFlags(type) & 4 /* Reference */) && type.target === globalReadonlyArrayType;
+        function getESSymbolLikeTypeForNode(node) {
+            if (ts.isValidESSymbolDeclaration(node)) {
+                var symbol = getSymbolOfNode(node);
+                var links = getSymbolLinks(symbol);
+                return links.uniqueESSymbolType || (links.uniqueESSymbolType = createUniqueESSymbolType(symbol));
+            }
+            return esSymbolType;
         }
-        function isMutableArrayOrTuple(type) {
-            return isArrayType(type) && !isReadonlyArrayType(type) || isTupleType(type) && !type.target.readonly;
+        function getThisType(node) {
+            var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
+            var parent = container && container.parent;
+            if (parent && (ts.isClassLike(parent) || parent.kind === 253 /* InterfaceDeclaration */)) {
+                if (!ts.hasSyntacticModifier(container, 32 /* Static */) &&
+                    (!ts.isConstructorDeclaration(container) || ts.isNodeDescendantOf(node, container.body))) {
+                    return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(parent)).thisType;
+                }
+            }
+            // inside x.prototype = { ... }
+            if (parent && ts.isObjectLiteralExpression(parent) && ts.isBinaryExpression(parent.parent) && ts.getAssignmentDeclarationKind(parent.parent) === 6 /* Prototype */) {
+                return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(parent.parent.left).parent).thisType;
+            }
+            // /** @return {this} */
+            // x.prototype.m = function() { ... }
+            var host = node.flags & 4194304 /* JSDoc */ ? ts.getHostSignatureFromJSDoc(node) : undefined;
+            if (host && ts.isFunctionExpression(host) && ts.isBinaryExpression(host.parent) && ts.getAssignmentDeclarationKind(host.parent) === 3 /* PrototypeProperty */) {
+                return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(host.parent.left).parent).thisType;
+            }
+            // inside constructor function C() { ... }
+            if (isJSConstructor(container) && ts.isNodeDescendantOf(node, container.body)) {
+                return getDeclaredTypeOfClassOrInterface(getSymbolOfNode(container)).thisType;
+            }
+            error(node, ts.Diagnostics.A_this_type_is_available_only_in_a_non_static_member_of_a_class_or_interface);
+            return errorType;
         }
-        function getElementTypeOfArrayType(type) {
-            return isArrayType(type) ? getTypeArguments(type)[0] : undefined;
+        function getTypeFromThisTypeNode(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                links.resolvedType = getThisType(node);
+            }
+            return links.resolvedType;
         }
-        function isArrayLikeType(type) {
-            // A type is array-like if it is a reference to the global Array or global ReadonlyArray type,
-            // or if it is not the undefined or null type and if it is assignable to ReadonlyArray<any>
-            return isArrayType(type) || !(type.flags & 98304 /* Nullable */) && isTypeAssignableTo(type, anyReadonlyArrayType);
+        function getTypeFromRestTypeNode(node) {
+            return getTypeFromTypeNode(getArrayElementTypeNode(node.type) || node.type);
         }
-        function isEmptyArrayLiteralType(type) {
-            var elementType = isArrayType(type) ? getTypeArguments(type)[0] : undefined;
-            return elementType === undefinedWideningType || elementType === implicitNeverType;
+        function getArrayElementTypeNode(node) {
+            switch (node.kind) {
+                case 186 /* ParenthesizedType */:
+                    return getArrayElementTypeNode(node.type);
+                case 179 /* TupleType */:
+                    if (node.elements.length === 1) {
+                        node = node.elements[0];
+                        if (node.kind === 181 /* RestType */ || node.kind === 192 /* NamedTupleMember */ && node.dotDotDotToken) {
+                            return getArrayElementTypeNode(node.type);
+                        }
+                    }
+                    break;
+                case 178 /* ArrayType */:
+                    return node.elementType;
+            }
+            return undefined;
         }
-        function isTupleLikeType(type) {
-            return isTupleType(type) || !!getPropertyOfType(type, "0");
+        function getTypeFromNamedTupleTypeNode(node) {
+            var links = getNodeLinks(node);
+            return links.resolvedType || (links.resolvedType =
+                node.dotDotDotToken ? getTypeFromRestTypeNode(node) :
+                    node.questionToken && strictNullChecks ? getOptionalType(getTypeFromTypeNode(node.type)) :
+                        getTypeFromTypeNode(node.type));
         }
-        function isArrayOrTupleLikeType(type) {
-            return isArrayLikeType(type) || isTupleLikeType(type);
+        function getTypeFromTypeNode(node) {
+            return getConditionalFlowTypeOfType(getTypeFromTypeNodeWorker(node), node);
         }
-        function getTupleElementType(type, index) {
-            var propType = getTypeOfPropertyOfType(type, "" + index);
-            if (propType) {
-                return propType;
+        function getTypeFromTypeNodeWorker(node) {
+            switch (node.kind) {
+                case 128 /* AnyKeyword */:
+                case 303 /* JSDocAllType */:
+                case 304 /* JSDocUnknownType */:
+                    return anyType;
+                case 152 /* UnknownKeyword */:
+                    return unknownType;
+                case 147 /* StringKeyword */:
+                    return stringType;
+                case 144 /* NumberKeyword */:
+                    return numberType;
+                case 155 /* BigIntKeyword */:
+                    return bigintType;
+                case 131 /* BooleanKeyword */:
+                    return booleanType;
+                case 148 /* SymbolKeyword */:
+                    return esSymbolType;
+                case 113 /* VoidKeyword */:
+                    return voidType;
+                case 150 /* UndefinedKeyword */:
+                    return undefinedType;
+                case 103 /* NullKeyword */:
+                    // TODO(rbuckton): `NullKeyword` is no longer a `TypeNode`, but we defensively allow it here because of incorrect casts in the Language Service.
+                    return nullType;
+                case 141 /* NeverKeyword */:
+                    return neverType;
+                case 145 /* ObjectKeyword */:
+                    return node.flags & 131072 /* JavaScriptFile */ && !noImplicitAny ? anyType : nonPrimitiveType;
+                case 136 /* IntrinsicKeyword */:
+                    return intrinsicMarkerType;
+                case 187 /* ThisType */:
+                case 107 /* ThisKeyword */:
+                    // TODO(rbuckton): `ThisKeyword` is no longer a `TypeNode`, but we defensively allow it here because of incorrect casts in the Language Service and because of `isPartOfTypeNode`.
+                    return getTypeFromThisTypeNode(node);
+                case 191 /* LiteralType */:
+                    return getTypeFromLiteralTypeNode(node);
+                case 173 /* TypeReference */:
+                    return getTypeFromTypeReference(node);
+                case 172 /* TypePredicate */:
+                    return node.assertsModifier ? voidType : booleanType;
+                case 223 /* ExpressionWithTypeArguments */:
+                    return getTypeFromTypeReference(node);
+                case 176 /* TypeQuery */:
+                    return getTypeFromTypeQueryNode(node);
+                case 178 /* ArrayType */:
+                case 179 /* TupleType */:
+                    return getTypeFromArrayOrTupleTypeNode(node);
+                case 180 /* OptionalType */:
+                    return getTypeFromOptionalTypeNode(node);
+                case 182 /* UnionType */:
+                    return getTypeFromUnionTypeNode(node);
+                case 183 /* IntersectionType */:
+                    return getTypeFromIntersectionTypeNode(node);
+                case 305 /* JSDocNullableType */:
+                    return getTypeFromJSDocNullableTypeNode(node);
+                case 307 /* JSDocOptionalType */:
+                    return addOptionality(getTypeFromTypeNode(node.type));
+                case 192 /* NamedTupleMember */:
+                    return getTypeFromNamedTupleTypeNode(node);
+                case 186 /* ParenthesizedType */:
+                case 306 /* JSDocNonNullableType */:
+                case 301 /* JSDocTypeExpression */:
+                    return getTypeFromTypeNode(node.type);
+                case 181 /* RestType */:
+                    return getTypeFromRestTypeNode(node);
+                case 309 /* JSDocVariadicType */:
+                    return getTypeFromJSDocVariadicType(node);
+                case 174 /* FunctionType */:
+                case 175 /* ConstructorType */:
+                case 177 /* TypeLiteral */:
+                case 312 /* JSDocTypeLiteral */:
+                case 308 /* JSDocFunctionType */:
+                case 313 /* JSDocSignature */:
+                    return getTypeFromTypeLiteralOrFunctionOrConstructorTypeNode(node);
+                case 188 /* TypeOperator */:
+                    return getTypeFromTypeOperatorNode(node);
+                case 189 /* IndexedAccessType */:
+                    return getTypeFromIndexedAccessTypeNode(node);
+                case 190 /* MappedType */:
+                    return getTypeFromMappedTypeNode(node);
+                case 184 /* ConditionalType */:
+                    return getTypeFromConditionalTypeNode(node);
+                case 185 /* InferType */:
+                    return getTypeFromInferTypeNode(node);
+                case 193 /* TemplateLiteralType */:
+                    return getTypeFromTemplateTypeNode(node);
+                case 195 /* ImportType */:
+                    return getTypeFromImportTypeNode(node);
+                // This function assumes that an identifier, qualified name, or property access expression is a type expression
+                // Callers should first ensure this by calling `isPartOfTypeNode`
+                // TODO(rbuckton): These aren't valid TypeNodes, but we treat them as such because of `isPartOfTypeNode`, which returns `true` for things that aren't `TypeNode`s.
+                case 78 /* Identifier */:
+                case 157 /* QualifiedName */:
+                case 201 /* PropertyAccessExpression */:
+                    var symbol = getSymbolAtLocation(node);
+                    return symbol ? getDeclaredTypeOfSymbol(symbol) : errorType;
+                default:
+                    return errorType;
             }
-            if (everyType(type, isTupleType)) {
-                return mapType(type, function (t) { return getRestTypeOfTupleType(t) || undefinedType; });
+        }
+        function instantiateList(items, mapper, instantiator) {
+            if (items && items.length) {
+                for (var i = 0; i < items.length; i++) {
+                    var item = items[i];
+                    var mapped = instantiator(item, mapper);
+                    if (item !== mapped) {
+                        var result = i === 0 ? [] : items.slice(0, i);
+                        result.push(mapped);
+                        for (i++; i < items.length; i++) {
+                            result.push(instantiator(items[i], mapper));
+                        }
+                        return result;
+                    }
+                }
             }
-            return undefined;
+            return items;
         }
-        function isNeitherUnitTypeNorNever(type) {
-            return !(type.flags & (109440 /* Unit */ | 131072 /* Never */));
+        function instantiateTypes(types, mapper) {
+            return instantiateList(types, mapper, instantiateType);
         }
-        function isUnitType(type) {
-            return !!(type.flags & 109440 /* Unit */);
+        function instantiateSignatures(signatures, mapper) {
+            return instantiateList(signatures, mapper, instantiateSignature);
         }
-        function isLiteralType(type) {
-            return type.flags & 16 /* Boolean */ ? true :
-                type.flags & 1048576 /* Union */ ? type.flags & 1024 /* EnumLiteral */ ? true : ts.every(type.types, isUnitType) :
-                    isUnitType(type);
+        function createTypeMapper(sources, targets) {
+            return sources.length === 1 ? makeUnaryTypeMapper(sources[0], targets ? targets[0] : anyType) : makeArrayTypeMapper(sources, targets);
         }
-        function getBaseTypeOfLiteralType(type) {
-            return type.flags & 1024 /* EnumLiteral */ ? getBaseTypeOfEnumLiteralType(type) :
-                type.flags & 128 /* StringLiteral */ ? stringType :
-                    type.flags & 256 /* NumberLiteral */ ? numberType :
-                        type.flags & 2048 /* BigIntLiteral */ ? bigintType :
-                            type.flags & 512 /* BooleanLiteral */ ? booleanType :
-                                type.flags & 1048576 /* Union */ ? getUnionType(ts.sameMap(type.types, getBaseTypeOfLiteralType)) :
-                                    type;
+        function getMappedType(type, mapper) {
+            switch (mapper.kind) {
+                case 0 /* Simple */:
+                    return type === mapper.source ? mapper.target : type;
+                case 1 /* Array */:
+                    var sources = mapper.sources;
+                    var targets = mapper.targets;
+                    for (var i = 0; i < sources.length; i++) {
+                        if (type === sources[i]) {
+                            return targets ? targets[i] : anyType;
+                        }
+                    }
+                    return type;
+                case 2 /* Function */:
+                    return mapper.func(type);
+                case 3 /* Composite */:
+                case 4 /* Merged */:
+                    var t1 = getMappedType(type, mapper.mapper1);
+                    return t1 !== type && mapper.kind === 3 /* Composite */ ? instantiateType(t1, mapper.mapper2) : getMappedType(t1, mapper.mapper2);
+            }
         }
-        function getWidenedLiteralType(type) {
-            return type.flags & 1024 /* EnumLiteral */ && isFreshLiteralType(type) ? getBaseTypeOfEnumLiteralType(type) :
-                type.flags & 128 /* StringLiteral */ && isFreshLiteralType(type) ? stringType :
-                    type.flags & 256 /* NumberLiteral */ && isFreshLiteralType(type) ? numberType :
-                        type.flags & 2048 /* BigIntLiteral */ && isFreshLiteralType(type) ? bigintType :
-                            type.flags & 512 /* BooleanLiteral */ && isFreshLiteralType(type) ? booleanType :
-                                type.flags & 1048576 /* Union */ ? getUnionType(ts.sameMap(type.types, getWidenedLiteralType)) :
-                                    type;
+        function makeUnaryTypeMapper(source, target) {
+            return { kind: 0 /* Simple */, source: source, target: target };
         }
-        function getWidenedUniqueESSymbolType(type) {
-            return type.flags & 8192 /* UniqueESSymbol */ ? esSymbolType :
-                type.flags & 1048576 /* Union */ ? getUnionType(ts.sameMap(type.types, getWidenedUniqueESSymbolType)) :
-                    type;
+        function makeArrayTypeMapper(sources, targets) {
+            return { kind: 1 /* Array */, sources: sources, targets: targets };
         }
-        function getWidenedLiteralLikeTypeForContextualType(type, contextualType) {
-            if (!isLiteralOfContextualType(type, contextualType)) {
-                type = getWidenedUniqueESSymbolType(getWidenedLiteralType(type));
-            }
-            return type;
+        function makeFunctionTypeMapper(func) {
+            return { kind: 2 /* Function */, func: func };
         }
-        function getWidenedLiteralLikeTypeForContextualReturnTypeIfNeeded(type, contextualSignatureReturnType, isAsync) {
-            if (type && isUnitType(type)) {
-                var contextualType = !contextualSignatureReturnType ? undefined :
-                    isAsync ? getPromisedTypeOfPromise(contextualSignatureReturnType) :
-                        contextualSignatureReturnType;
-                type = getWidenedLiteralLikeTypeForContextualType(type, contextualType);
-            }
-            return type;
+        function makeCompositeTypeMapper(kind, mapper1, mapper2) {
+            return { kind: kind, mapper1: mapper1, mapper2: mapper2 };
         }
-        function getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(type, contextualSignatureReturnType, kind, isAsyncGenerator) {
-            if (type && isUnitType(type)) {
-                var contextualType = !contextualSignatureReturnType ? undefined :
-                    getIterationTypeOfGeneratorFunctionReturnType(kind, contextualSignatureReturnType, isAsyncGenerator);
-                type = getWidenedLiteralLikeTypeForContextualType(type, contextualType);
-            }
-            return type;
+        function createTypeEraser(sources) {
+            return createTypeMapper(sources, /*targets*/ undefined);
         }
         /**
-         * Check if a Type was written as a tuple type literal.
-         * Prefer using isTupleLikeType() unless the use of `elementTypes`/`getTypeArguments` is required.
+         * Maps forward-references to later types parameters to the empty object type.
+         * This is used during inference when instantiating type parameter defaults.
          */
-        function isTupleType(type) {
-            return !!(ts.getObjectFlags(type) & 4 /* Reference */ && type.target.objectFlags & 8 /* Tuple */);
+        function createBackreferenceMapper(context, index) {
+            return makeFunctionTypeMapper(function (t) { return ts.findIndex(context.inferences, function (info) { return info.typeParameter === t; }) >= index ? unknownType : t; });
         }
-        function getRestTypeOfTupleType(type) {
-            return type.target.hasRestElement ? getTypeArguments(type)[type.target.typeParameters.length - 1] : undefined;
+        function combineTypeMappers(mapper1, mapper2) {
+            return mapper1 ? makeCompositeTypeMapper(3 /* Composite */, mapper1, mapper2) : mapper2;
         }
-        function getRestArrayTypeOfTupleType(type) {
-            var restType = getRestTypeOfTupleType(type);
-            return restType && createArrayType(restType);
+        function mergeTypeMappers(mapper1, mapper2) {
+            return mapper1 ? makeCompositeTypeMapper(4 /* Merged */, mapper1, mapper2) : mapper2;
         }
-        function getLengthOfTupleType(type) {
-            return getTypeReferenceArity(type) - (type.target.hasRestElement ? 1 : 0);
+        function prependTypeMapping(source, target, mapper) {
+            return !mapper ? makeUnaryTypeMapper(source, target) : makeCompositeTypeMapper(4 /* Merged */, makeUnaryTypeMapper(source, target), mapper);
         }
-        function isZeroBigInt(_a) {
-            var value = _a.value;
-            return value.base10Value === "0";
+        function appendTypeMapping(mapper, source, target) {
+            return !mapper ? makeUnaryTypeMapper(source, target) : makeCompositeTypeMapper(4 /* Merged */, mapper, makeUnaryTypeMapper(source, target));
         }
-        function getFalsyFlagsOfTypes(types) {
-            var result = 0;
-            for (var _i = 0, types_13 = types; _i < types_13.length; _i++) {
-                var t = types_13[_i];
-                result |= getFalsyFlags(t);
-            }
-            return result;
+        function getRestrictiveTypeParameter(tp) {
+            return tp.constraint === unknownType ? tp : tp.restrictiveInstantiation || (tp.restrictiveInstantiation = createTypeParameter(tp.symbol),
+                tp.restrictiveInstantiation.constraint = unknownType,
+                tp.restrictiveInstantiation);
         }
-        // Returns the String, Number, Boolean, StringLiteral, NumberLiteral, BooleanLiteral, Void, Undefined, or Null
-        // flags for the string, number, boolean, "", 0, false, void, undefined, or null types respectively. Returns
-        // no flags for all other types (including non-falsy literal types).
-        function getFalsyFlags(type) {
-            return type.flags & 1048576 /* Union */ ? getFalsyFlagsOfTypes(type.types) :
-                type.flags & 128 /* StringLiteral */ ? type.value === "" ? 128 /* StringLiteral */ : 0 :
-                    type.flags & 256 /* NumberLiteral */ ? type.value === 0 ? 256 /* NumberLiteral */ : 0 :
-                        type.flags & 2048 /* BigIntLiteral */ ? isZeroBigInt(type) ? 2048 /* BigIntLiteral */ : 0 :
-                            type.flags & 512 /* BooleanLiteral */ ? (type === falseType || type === regularFalseType) ? 512 /* BooleanLiteral */ : 0 :
-                                type.flags & 117724 /* PossiblyFalsy */;
+        function cloneTypeParameter(typeParameter) {
+            var result = createTypeParameter(typeParameter.symbol);
+            result.target = typeParameter;
+            return result;
         }
-        function removeDefinitelyFalsyTypes(type) {
-            return getFalsyFlags(type) & 117632 /* DefinitelyFalsy */ ?
-                filterType(type, function (t) { return !(getFalsyFlags(t) & 117632 /* DefinitelyFalsy */); }) :
-                type;
+        function instantiateTypePredicate(predicate, mapper) {
+            return createTypePredicate(predicate.kind, predicate.parameterName, predicate.parameterIndex, instantiateType(predicate.type, mapper));
         }
-        function extractDefinitelyFalsyTypes(type) {
-            return mapType(type, getDefinitelyFalsyPartOfType);
+        function instantiateSignature(signature, mapper, eraseTypeParameters) {
+            var freshTypeParameters;
+            if (signature.typeParameters && !eraseTypeParameters) {
+                // First create a fresh set of type parameters, then include a mapping from the old to the
+                // new type parameters in the mapper function. Finally store this mapper in the new type
+                // parameters such that we can use it when instantiating constraints.
+                freshTypeParameters = ts.map(signature.typeParameters, cloneTypeParameter);
+                mapper = combineTypeMappers(createTypeMapper(signature.typeParameters, freshTypeParameters), mapper);
+                for (var _i = 0, freshTypeParameters_1 = freshTypeParameters; _i < freshTypeParameters_1.length; _i++) {
+                    var tp = freshTypeParameters_1[_i];
+                    tp.mapper = mapper;
+                }
+            }
+            // Don't compute resolvedReturnType and resolvedTypePredicate now,
+            // because using `mapper` now could trigger inferences to become fixed. (See `createInferenceContext`.)
+            // See GH#17600.
+            var result = createSignature(signature.declaration, freshTypeParameters, signature.thisParameter && instantiateSymbol(signature.thisParameter, mapper), instantiateList(signature.parameters, mapper, instantiateSymbol), 
+            /*resolvedReturnType*/ undefined, 
+            /*resolvedTypePredicate*/ undefined, signature.minArgumentCount, signature.flags & 19 /* PropagatingFlags */);
+            result.target = signature;
+            result.mapper = mapper;
+            return result;
         }
-        function getDefinitelyFalsyPartOfType(type) {
-            return type.flags & 4 /* String */ ? emptyStringType :
-                type.flags & 8 /* Number */ ? zeroType :
-                    type.flags & 64 /* BigInt */ ? zeroBigIntType :
-                        type === regularFalseType ||
-                            type === falseType ||
-                            type.flags & (16384 /* Void */ | 32768 /* Undefined */ | 65536 /* Null */) ||
-                            type.flags & 128 /* StringLiteral */ && type.value === "" ||
-                            type.flags & 256 /* NumberLiteral */ && type.value === 0 ||
-                            type.flags & 2048 /* BigIntLiteral */ && isZeroBigInt(type) ? type :
-                            neverType;
+        function instantiateSymbol(symbol, mapper) {
+            var links = getSymbolLinks(symbol);
+            if (links.type && !couldContainTypeVariables(links.type)) {
+                // If the type of the symbol is already resolved, and if that type could not possibly
+                // be affected by instantiation, simply return the symbol itself.
+                return symbol;
+            }
+            if (ts.getCheckFlags(symbol) & 1 /* Instantiated */) {
+                // If symbol being instantiated is itself a instantiation, fetch the original target and combine the
+                // type mappers. This ensures that original type identities are properly preserved and that aliases
+                // always reference a non-aliases.
+                symbol = links.target;
+                mapper = combineTypeMappers(links.mapper, mapper);
+            }
+            // Keep the flags from the symbol we're instantiating.  Mark that is instantiated, and
+            // also transient so that we can just store data on it directly.
+            var result = createSymbol(symbol.flags, symbol.escapedName, 1 /* Instantiated */ | ts.getCheckFlags(symbol) & (8 /* Readonly */ | 4096 /* Late */ | 16384 /* OptionalParameter */ | 32768 /* RestParameter */));
+            result.declarations = symbol.declarations;
+            result.parent = symbol.parent;
+            result.target = symbol;
+            result.mapper = mapper;
+            if (symbol.valueDeclaration) {
+                result.valueDeclaration = symbol.valueDeclaration;
+            }
+            if (links.nameType) {
+                result.nameType = links.nameType;
+            }
+            return result;
         }
-        /**
-         * Add undefined or null or both to a type if they are missing.
-         * @param type - type to add undefined and/or null to if not present
-         * @param flags - Either TypeFlags.Undefined or TypeFlags.Null, or both
-         */
-        function getNullableType(type, flags) {
-            var missing = (flags & ~type.flags) & (32768 /* Undefined */ | 65536 /* Null */);
-            return missing === 0 ? type :
-                missing === 32768 /* Undefined */ ? getUnionType([type, undefinedType]) :
-                    missing === 65536 /* Null */ ? getUnionType([type, nullType]) :
-                        getUnionType([type, undefinedType, nullType]);
+        function getObjectTypeInstantiation(type, mapper) {
+            var declaration = type.objectFlags & 4 /* Reference */ ? type.node : type.symbol.declarations[0];
+            var links = getNodeLinks(declaration);
+            var target = type.objectFlags & 4 /* Reference */ ? links.resolvedType :
+                type.objectFlags & 64 /* Instantiated */ ? type.target : type;
+            var typeParameters = links.outerTypeParameters;
+            if (!typeParameters) {
+                // The first time an anonymous type is instantiated we compute and store a list of the type
+                // parameters that are in scope (and therefore potentially referenced). For type literals that
+                // aren't the right hand side of a generic type alias declaration we optimize by reducing the
+                // set of type parameters to those that are possibly referenced in the literal.
+                var outerTypeParameters = getOuterTypeParameters(declaration, /*includeThisTypes*/ true);
+                if (isJSConstructor(declaration)) {
+                    var templateTagParameters = getTypeParametersFromDeclaration(declaration);
+                    outerTypeParameters = ts.addRange(outerTypeParameters, templateTagParameters);
+                }
+                typeParameters = outerTypeParameters || ts.emptyArray;
+                typeParameters = (target.objectFlags & 4 /* Reference */ || target.symbol.flags & 2048 /* TypeLiteral */) && !target.aliasTypeArguments ?
+                    ts.filter(typeParameters, function (tp) { return isTypeParameterPossiblyReferenced(tp, declaration); }) :
+                    typeParameters;
+                links.outerTypeParameters = typeParameters;
+            }
+            if (typeParameters.length) {
+                // We are instantiating an anonymous type that has one or more type parameters in scope. Apply the
+                // mapper to the type parameters to produce the effective list of type arguments, and compute the
+                // instantiation cache key from the type IDs of the type arguments.
+                var combinedMapper_1 = combineTypeMappers(type.mapper, mapper);
+                var typeArguments = ts.map(typeParameters, function (t) { return getMappedType(t, combinedMapper_1); });
+                var id = getTypeListId(typeArguments);
+                if (!target.instantiations) {
+                    target.instantiations = new ts.Map();
+                    target.instantiations.set(getTypeListId(typeParameters), target);
+                }
+                var result = target.instantiations.get(id);
+                if (!result) {
+                    var newMapper = createTypeMapper(typeParameters, typeArguments);
+                    result = target.objectFlags & 4 /* Reference */ ? createDeferredTypeReference(type.target, type.node, newMapper) :
+                        target.objectFlags & 32 /* Mapped */ ? instantiateMappedType(target, newMapper) :
+                            instantiateAnonymousType(target, newMapper);
+                    target.instantiations.set(id, result);
+                }
+                return result;
+            }
+            return type;
         }
-        function getOptionalType(type) {
-            ts.Debug.assert(strictNullChecks);
-            return type.flags & 32768 /* Undefined */ ? type : getUnionType([type, undefinedType]);
+        function maybeTypeParameterReference(node) {
+            return !(node.kind === 157 /* QualifiedName */ ||
+                node.parent.kind === 173 /* TypeReference */ && node.parent.typeArguments && node === node.parent.typeName ||
+                node.parent.kind === 195 /* ImportType */ && node.parent.typeArguments && node === node.parent.qualifier);
         }
-        function getGlobalNonNullableTypeInstantiation(type) {
-            if (!deferredGlobalNonNullableTypeAlias) {
-                deferredGlobalNonNullableTypeAlias = getGlobalSymbol("NonNullable", 524288 /* TypeAlias */, /*diagnostic*/ undefined) || unknownSymbol;
+        function isTypeParameterPossiblyReferenced(tp, node) {
+            // If the type parameter doesn't have exactly one declaration, if there are invening statement blocks
+            // between the node and the type parameter declaration, if the node contains actual references to the
+            // type parameter, or if the node contains type queries, we consider the type parameter possibly referenced.
+            if (tp.symbol && tp.symbol.declarations && tp.symbol.declarations.length === 1) {
+                var container = tp.symbol.declarations[0].parent;
+                for (var n = node; n !== container; n = n.parent) {
+                    if (!n || n.kind === 230 /* Block */ || n.kind === 184 /* ConditionalType */ && ts.forEachChild(n.extendsType, containsReference)) {
+                        return true;
+                    }
+                }
+                return !!ts.forEachChild(node, containsReference);
             }
-            // Use NonNullable global type alias if available to improve quick info/declaration emit
-            if (deferredGlobalNonNullableTypeAlias !== unknownSymbol) {
-                return getTypeAliasInstantiation(deferredGlobalNonNullableTypeAlias, [type]);
+            return true;
+            function containsReference(node) {
+                switch (node.kind) {
+                    case 187 /* ThisType */:
+                        return !!tp.isThisType;
+                    case 78 /* Identifier */:
+                        return !tp.isThisType && ts.isPartOfTypeNode(node) && maybeTypeParameterReference(node) &&
+                            getTypeFromTypeNodeWorker(node) === tp; // use worker because we're looking for === equality
+                    case 176 /* TypeQuery */:
+                        return true;
+                }
+                return !!ts.forEachChild(node, containsReference);
             }
-            return getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */); // Type alias unavailable, fall back to non-higher-order behavior
-        }
-        function getNonNullableType(type) {
-            return strictNullChecks ? getGlobalNonNullableTypeInstantiation(type) : type;
         }
-        function addOptionalTypeMarker(type) {
-            return strictNullChecks ? getUnionType([type, optionalType]) : type;
+        function getHomomorphicTypeVariable(type) {
+            var constraintType = getConstraintTypeFromMappedType(type);
+            if (constraintType.flags & 4194304 /* Index */) {
+                var typeVariable = getActualTypeVariable(constraintType.type);
+                if (typeVariable.flags & 262144 /* TypeParameter */) {
+                    return typeVariable;
+                }
+            }
+            return undefined;
         }
-        function isNotOptionalTypeMarker(type) {
-            return type !== optionalType;
+        function instantiateMappedType(type, mapper) {
+            // For a homomorphic mapped type { [P in keyof T]: X }, where T is some type variable, the mapping
+            // operation depends on T as follows:
+            // * If T is a primitive type no mapping is performed and the result is simply T.
+            // * If T is a union type we distribute the mapped type over the union.
+            // * If T is an array we map to an array where the element type has been transformed.
+            // * If T is a tuple we map to a tuple where the element types have been transformed.
+            // * Otherwise we map to an object type where the type of each property has been transformed.
+            // For example, when T is instantiated to a union type A | B, we produce { [P in keyof A]: X } |
+            // { [P in keyof B]: X }, and when when T is instantiated to a union type A | undefined, we produce
+            // { [P in keyof A]: X } | undefined.
+            var typeVariable = getHomomorphicTypeVariable(type);
+            if (typeVariable) {
+                var mappedTypeVariable = instantiateType(typeVariable, mapper);
+                if (typeVariable !== mappedTypeVariable) {
+                    return mapType(getReducedType(mappedTypeVariable), function (t) {
+                        if (t.flags & (3 /* AnyOrUnknown */ | 58982400 /* InstantiableNonPrimitive */ | 524288 /* Object */ | 2097152 /* Intersection */) && t !== wildcardType && t !== errorType) {
+                            if (!type.declaration.nameType) {
+                                if (isArrayType(t)) {
+                                    return instantiateMappedArrayType(t, type, prependTypeMapping(typeVariable, t, mapper));
+                                }
+                                if (isGenericTupleType(t)) {
+                                    return instantiateMappedGenericTupleType(t, type, typeVariable, mapper);
+                                }
+                                if (isTupleType(t)) {
+                                    return instantiateMappedTupleType(t, type, prependTypeMapping(typeVariable, t, mapper));
+                                }
+                            }
+                            return instantiateAnonymousType(type, prependTypeMapping(typeVariable, t, mapper));
+                        }
+                        return t;
+                    });
+                }
+            }
+            return instantiateAnonymousType(type, mapper);
         }
-        function removeOptionalTypeMarker(type) {
-            return strictNullChecks ? filterType(type, isNotOptionalTypeMarker) : type;
+        function getModifiedReadonlyState(state, modifiers) {
+            return modifiers & 1 /* IncludeReadonly */ ? true : modifiers & 2 /* ExcludeReadonly */ ? false : state;
         }
-        function propagateOptionalTypeMarker(type, node, wasOptional) {
-            return wasOptional ? ts.isOutermostOptionalChain(node) ? getOptionalType(type) : addOptionalTypeMarker(type) : type;
+        function instantiateMappedGenericTupleType(tupleType, mappedType, typeVariable, mapper) {
+            // When a tuple type is generic (i.e. when it contains variadic elements), we want to eagerly map the
+            // non-generic elements and defer mapping the generic elements. In order to facilitate this, we transform
+            // M<[A, B?, ...T, ...C[]] into [...M<[A]>, ...M<[B?]>, ...M<T>, ...M<C[]>] and then rely on tuple type
+            // normalization to resolve the non-generic parts of the resulting tuple.
+            var elementFlags = tupleType.target.elementFlags;
+            var elementTypes = ts.map(getTypeArguments(tupleType), function (t, i) {
+                var singleton = elementFlags[i] & 8 /* Variadic */ ? t :
+                    elementFlags[i] & 4 /* Rest */ ? createArrayType(t) :
+                        createTupleType([t], [elementFlags[i]]);
+                // The singleton is never a generic tuple type, so it is safe to recurse here.
+                return instantiateMappedType(mappedType, prependTypeMapping(typeVariable, singleton, mapper));
+            });
+            var newReadonly = getModifiedReadonlyState(tupleType.target.readonly, getMappedTypeModifiers(mappedType));
+            return createTupleType(elementTypes, ts.map(elementTypes, function (_) { return 8 /* Variadic */; }), newReadonly);
         }
-        function getOptionalExpressionType(exprType, expression) {
-            return ts.isExpressionOfOptionalChainRoot(expression) ? getNonNullableType(exprType) :
-                ts.isOptionalChain(expression) ? removeOptionalTypeMarker(exprType) :
-                    exprType;
+        function instantiateMappedArrayType(arrayType, mappedType, mapper) {
+            var elementType = instantiateMappedTypeTemplate(mappedType, numberType, /*isOptional*/ true, mapper);
+            return elementType === errorType ? errorType :
+                createArrayType(elementType, getModifiedReadonlyState(isReadonlyArrayType(arrayType), getMappedTypeModifiers(mappedType)));
         }
-        /**
-         * Is source potentially coercible to target type under `==`.
-         * Assumes that `source` is a constituent of a union, hence
-         * the boolean literal flag on the LHS, but not on the RHS.
-         *
-         * This does not fully replicate the semantics of `==`. The
-         * intention is to catch cases that are clearly not right.
-         *
-         * Comparing (string | number) to number should not remove the
-         * string element.
-         *
-         * Comparing (string | number) to 1 will remove the string
-         * element, though this is not sound. This is a pragmatic
-         * choice.
-         *
-         * @see narrowTypeByEquality
-         *
-         * @param source
-         * @param target
-         */
-        function isCoercibleUnderDoubleEquals(source, target) {
-            return ((source.flags & (8 /* Number */ | 4 /* String */ | 512 /* BooleanLiteral */)) !== 0)
-                && ((target.flags & (8 /* Number */ | 4 /* String */ | 16 /* Boolean */)) !== 0);
+        function instantiateMappedTupleType(tupleType, mappedType, mapper) {
+            var elementFlags = tupleType.target.elementFlags;
+            var elementTypes = ts.map(getTypeArguments(tupleType), function (_, i) {
+                return instantiateMappedTypeTemplate(mappedType, getLiteralType("" + i), !!(elementFlags[i] & 2 /* Optional */), mapper);
+            });
+            var modifiers = getMappedTypeModifiers(mappedType);
+            var newTupleModifiers = modifiers & 4 /* IncludeOptional */ ? ts.map(elementFlags, function (f) { return f & 1 /* Required */ ? 2 /* Optional */ : f; }) :
+                modifiers & 8 /* ExcludeOptional */ ? ts.map(elementFlags, function (f) { return f & 2 /* Optional */ ? 1 /* Required */ : f; }) :
+                    elementFlags;
+            var newReadonly = getModifiedReadonlyState(tupleType.target.readonly, modifiers);
+            return ts.contains(elementTypes, errorType) ? errorType :
+                createTupleType(elementTypes, newTupleModifiers, newReadonly, tupleType.target.labeledElementDeclarations);
         }
-        /**
-         * Return true if type was inferred from an object literal, written as an object type literal, or is the shape of a module
-         * with no call or construct signatures.
-         */
-        function isObjectTypeWithInferableIndex(type) {
-            return type.flags & 2097152 /* Intersection */ ? ts.every(type.types, isObjectTypeWithInferableIndex) :
-                !!(type.symbol && (type.symbol.flags & (4096 /* ObjectLiteral */ | 2048 /* TypeLiteral */ | 384 /* Enum */ | 512 /* ValueModule */)) !== 0 &&
-                    !typeHasCallOrConstructSignatures(type)) || !!(ts.getObjectFlags(type) & 2048 /* ReverseMapped */ && isObjectTypeWithInferableIndex(type.source));
+        function instantiateMappedTypeTemplate(type, key, isOptional, mapper) {
+            var templateMapper = appendTypeMapping(mapper, getTypeParameterFromMappedType(type), key);
+            var propType = instantiateType(getTemplateTypeFromMappedType(type.target || type), templateMapper);
+            var modifiers = getMappedTypeModifiers(type);
+            return strictNullChecks && modifiers & 4 /* IncludeOptional */ && !maybeTypeOfKind(propType, 32768 /* Undefined */ | 16384 /* Void */) ? getOptionalType(propType) :
+                strictNullChecks && modifiers & 8 /* ExcludeOptional */ && isOptional ? getTypeWithFacts(propType, 524288 /* NEUndefined */) :
+                    propType;
         }
-        function createSymbolWithType(source, type) {
-            var symbol = createSymbol(source.flags, source.escapedName, ts.getCheckFlags(source) & 8 /* Readonly */);
-            symbol.declarations = source.declarations;
-            symbol.parent = source.parent;
-            symbol.type = type;
-            symbol.target = source;
-            if (source.valueDeclaration) {
-                symbol.valueDeclaration = source.valueDeclaration;
+        function instantiateAnonymousType(type, mapper) {
+            var result = createObjectType(type.objectFlags | 64 /* Instantiated */, type.symbol);
+            if (type.objectFlags & 32 /* Mapped */) {
+                result.declaration = type.declaration;
+                // C.f. instantiateSignature
+                var origTypeParameter = getTypeParameterFromMappedType(type);
+                var freshTypeParameter = cloneTypeParameter(origTypeParameter);
+                result.typeParameter = freshTypeParameter;
+                mapper = combineTypeMappers(makeUnaryTypeMapper(origTypeParameter, freshTypeParameter), mapper);
+                freshTypeParameter.mapper = mapper;
             }
-            var nameType = getSymbolLinks(source).nameType;
-            if (nameType) {
-                symbol.nameType = nameType;
+            result.target = type;
+            result.mapper = mapper;
+            result.aliasSymbol = type.aliasSymbol;
+            result.aliasTypeArguments = instantiateTypes(type.aliasTypeArguments, mapper);
+            return result;
+        }
+        function getConditionalTypeInstantiation(type, mapper) {
+            var root = type.root;
+            if (root.outerTypeParameters) {
+                // We are instantiating a conditional type that has one or more type parameters in scope. Apply the
+                // mapper to the type parameters to produce the effective list of type arguments, and compute the
+                // instantiation cache key from the type IDs of the type arguments.
+                var typeArguments = ts.map(root.outerTypeParameters, function (t) { return getMappedType(t, mapper); });
+                var id = getTypeListId(typeArguments);
+                var result = root.instantiations.get(id);
+                if (!result) {
+                    var newMapper = createTypeMapper(root.outerTypeParameters, typeArguments);
+                    result = instantiateConditionalType(root, newMapper);
+                    root.instantiations.set(id, result);
+                }
+                return result;
             }
-            return symbol;
+            return type;
         }
-        function transformTypeOfMembers(type, f) {
-            var members = ts.createSymbolTable();
-            for (var _i = 0, _a = getPropertiesOfObjectType(type); _i < _a.length; _i++) {
-                var property = _a[_i];
-                var original = getTypeOfSymbol(property);
-                var updated = f(original);
-                members.set(property.escapedName, updated === original ? property : createSymbolWithType(property, updated));
+        function instantiateConditionalType(root, mapper) {
+            // Check if we have a conditional type where the check type is a naked type parameter. If so,
+            // the conditional type is distributive over union types and when T is instantiated to a union
+            // type A | B, we produce (A extends U ? X : Y) | (B extends U ? X : Y).
+            if (root.isDistributive) {
+                var checkType_1 = root.checkType;
+                var instantiatedType = getMappedType(checkType_1, mapper);
+                if (checkType_1 !== instantiatedType && instantiatedType.flags & (1048576 /* Union */ | 131072 /* Never */)) {
+                    return mapType(instantiatedType, function (t) { return getConditionalType(root, prependTypeMapping(checkType_1, t, mapper)); });
+                }
             }
-            return members;
+            return getConditionalType(root, mapper);
         }
-        /**
-         * If the the provided object literal is subject to the excess properties check,
-         * create a new that is exempt. Recursively mark object literal members as exempt.
-         * Leave signatures alone since they are not subject to the check.
-         */
-        function getRegularTypeOfObjectLiteral(type) {
-            if (!(isObjectLiteralType(type) && ts.getObjectFlags(type) & 32768 /* FreshLiteral */)) {
+        function instantiateType(type, mapper) {
+            if (!(type && mapper && couldContainTypeVariables(type))) {
                 return type;
             }
-            var regularType = type.regularType;
-            if (regularType) {
-                return regularType;
+            if (instantiationDepth === 50 || instantiationCount >= 5000000) {
+                // We have reached 50 recursive type instantiations and there is a very high likelyhood we're dealing
+                // with a combination of infinite generic types that perpetually generate new type identities. We stop
+                // the recursion here by yielding the error type.
+                ts.tracing.instant("check" /* Check */, "instantiateType_DepthLimit", { typeId: type.id, instantiationDepth: instantiationDepth, instantiationCount: instantiationCount });
+                error(currentNode, ts.Diagnostics.Type_instantiation_is_excessively_deep_and_possibly_infinite);
+                return errorType;
             }
-            var resolved = type;
-            var members = transformTypeOfMembers(type, getRegularTypeOfObjectLiteral);
-            var regularNew = createAnonymousType(resolved.symbol, members, resolved.callSignatures, resolved.constructSignatures, resolved.stringIndexInfo, resolved.numberIndexInfo);
-            regularNew.flags = resolved.flags;
-            regularNew.objectFlags |= resolved.objectFlags & ~32768 /* FreshLiteral */;
-            type.regularType = regularNew;
-            return regularNew;
-        }
-        function createWideningContext(parent, propertyName, siblings) {
-            return { parent: parent, propertyName: propertyName, siblings: siblings, resolvedProperties: undefined };
+            totalInstantiationCount++;
+            instantiationCount++;
+            instantiationDepth++;
+            var result = instantiateTypeWorker(type, mapper);
+            instantiationDepth--;
+            return result;
         }
-        function getSiblingsOfContext(context) {
-            if (!context.siblings) {
-                var siblings_1 = [];
-                for (var _i = 0, _a = getSiblingsOfContext(context.parent); _i < _a.length; _i++) {
-                    var type = _a[_i];
-                    if (isObjectLiteralType(type)) {
-                        var prop = getPropertyOfObjectType(type, context.propertyName);
-                        if (prop) {
-                            forEachType(getTypeOfSymbol(prop), function (t) {
-                                siblings_1.push(t);
-                            });
-                        }
-                    }
-                }
-                context.siblings = siblings_1;
+        function instantiateTypeWorker(type, mapper) {
+            var flags = type.flags;
+            if (flags & 262144 /* TypeParameter */) {
+                return getMappedType(type, mapper);
             }
-            return context.siblings;
-        }
-        function getPropertiesOfContext(context) {
-            if (!context.resolvedProperties) {
-                var names = ts.createMap();
-                for (var _i = 0, _a = getSiblingsOfContext(context); _i < _a.length; _i++) {
-                    var t = _a[_i];
-                    if (isObjectLiteralType(t) && !(ts.getObjectFlags(t) & 1024 /* ContainsSpread */)) {
-                        for (var _b = 0, _c = getPropertiesOfType(t); _b < _c.length; _b++) {
-                            var prop = _c[_b];
-                            names.set(prop.escapedName, prop);
-                        }
+            if (flags & 524288 /* Object */) {
+                var objectFlags = type.objectFlags;
+                if (objectFlags & (4 /* Reference */ | 16 /* Anonymous */ | 32 /* Mapped */)) {
+                    if (objectFlags & 4 /* Reference */ && !(type.node)) {
+                        var resolvedTypeArguments = type.resolvedTypeArguments;
+                        var newTypeArguments = instantiateTypes(resolvedTypeArguments, mapper);
+                        return newTypeArguments !== resolvedTypeArguments ? createNormalizedTypeReference(type.target, newTypeArguments) : type;
                     }
+                    return getObjectTypeInstantiation(type, mapper);
                 }
-                context.resolvedProperties = ts.arrayFrom(names.values());
+                return type;
             }
-            return context.resolvedProperties;
-        }
-        function getWidenedProperty(prop, context) {
-            if (!(prop.flags & 4 /* Property */)) {
-                // Since get accessors already widen their return value there is no need to
-                // widen accessor based properties here.
-                return prop;
+            if (flags & 3145728 /* UnionOrIntersection */) {
+                var types = type.types;
+                var newTypes = instantiateTypes(types, mapper);
+                return newTypes === types ? type :
+                    flags & 2097152 /* Intersection */ ?
+                        getIntersectionType(newTypes, type.aliasSymbol, instantiateTypes(type.aliasTypeArguments, mapper)) :
+                        getUnionType(newTypes, 1 /* Literal */, type.aliasSymbol, instantiateTypes(type.aliasTypeArguments, mapper));
             }
-            var original = getTypeOfSymbol(prop);
-            var propContext = context && createWideningContext(context, prop.escapedName, /*siblings*/ undefined);
-            var widened = getWidenedTypeWithContext(original, propContext);
-            return widened === original ? prop : createSymbolWithType(prop, widened);
-        }
-        function getUndefinedProperty(prop) {
-            var cached = undefinedProperties.get(prop.escapedName);
-            if (cached) {
-                return cached;
+            if (flags & 4194304 /* Index */) {
+                return getIndexType(instantiateType(type.type, mapper));
             }
-            var result = createSymbolWithType(prop, undefinedType);
-            result.flags |= 16777216 /* Optional */;
-            undefinedProperties.set(prop.escapedName, result);
-            return result;
-        }
-        function getWidenedTypeOfObjectLiteral(type, context) {
-            var members = ts.createSymbolTable();
-            for (var _i = 0, _a = getPropertiesOfObjectType(type); _i < _a.length; _i++) {
-                var prop = _a[_i];
-                members.set(prop.escapedName, getWidenedProperty(prop, context));
+            if (flags & 134217728 /* TemplateLiteral */) {
+                return getTemplateLiteralType(type.texts, instantiateTypes(type.types, mapper));
             }
-            if (context) {
-                for (var _b = 0, _c = getPropertiesOfContext(context); _b < _c.length; _b++) {
-                    var prop = _c[_b];
-                    if (!members.has(prop.escapedName)) {
-                        members.set(prop.escapedName, getUndefinedProperty(prop));
+            if (flags & 268435456 /* StringMapping */) {
+                return getStringMappingType(type.symbol, instantiateType(type.type, mapper));
+            }
+            if (flags & 8388608 /* IndexedAccess */) {
+                return getIndexedAccessType(instantiateType(type.objectType, mapper), instantiateType(type.indexType, mapper), type.noUncheckedIndexedAccessCandidate, /*accessNode*/ undefined, type.aliasSymbol, instantiateTypes(type.aliasTypeArguments, mapper));
+            }
+            if (flags & 16777216 /* Conditional */) {
+                return getConditionalTypeInstantiation(type, combineTypeMappers(type.mapper, mapper));
+            }
+            if (flags & 33554432 /* Substitution */) {
+                var maybeVariable = instantiateType(type.baseType, mapper);
+                if (maybeVariable.flags & 8650752 /* TypeVariable */) {
+                    return getSubstitutionType(maybeVariable, instantiateType(type.substitute, mapper));
+                }
+                else {
+                    var sub = instantiateType(type.substitute, mapper);
+                    if (sub.flags & 3 /* AnyOrUnknown */ || isTypeAssignableTo(getRestrictiveInstantiation(maybeVariable), getRestrictiveInstantiation(sub))) {
+                        return maybeVariable;
                     }
+                    return sub;
                 }
             }
-            var stringIndexInfo = getIndexInfoOfType(type, 0 /* String */);
-            var numberIndexInfo = getIndexInfoOfType(type, 1 /* Number */);
-            var result = createAnonymousType(type.symbol, members, ts.emptyArray, ts.emptyArray, stringIndexInfo && createIndexInfo(getWidenedType(stringIndexInfo.type), stringIndexInfo.isReadonly), numberIndexInfo && createIndexInfo(getWidenedType(numberIndexInfo.type), numberIndexInfo.isReadonly));
-            result.objectFlags |= (ts.getObjectFlags(type) & (16384 /* JSLiteral */ | 2097152 /* NonInferrableType */)); // Retain js literal flag through widening
-            return result;
+            return type;
         }
-        function getWidenedType(type) {
-            return getWidenedTypeWithContext(type, /*context*/ undefined);
+        function getPermissiveInstantiation(type) {
+            return type.flags & (131068 /* Primitive */ | 3 /* AnyOrUnknown */ | 131072 /* Never */) ? type :
+                type.permissiveInstantiation || (type.permissiveInstantiation = instantiateType(type, permissiveMapper));
         }
-        function getWidenedTypeWithContext(type, context) {
-            if (ts.getObjectFlags(type) & 1572864 /* RequiresWidening */) {
-                if (context === undefined && type.widened) {
-                    return type.widened;
-                }
-                var result = void 0;
-                if (type.flags & (1 /* Any */ | 98304 /* Nullable */)) {
-                    result = anyType;
-                }
-                else if (isObjectLiteralType(type)) {
-                    result = getWidenedTypeOfObjectLiteral(type, context);
+        function getRestrictiveInstantiation(type) {
+            if (type.flags & (131068 /* Primitive */ | 3 /* AnyOrUnknown */ | 131072 /* Never */)) {
+                return type;
+            }
+            if (type.restrictiveInstantiation) {
+                return type.restrictiveInstantiation;
+            }
+            type.restrictiveInstantiation = instantiateType(type, restrictiveMapper);
+            // We set the following so we don't attempt to set the restrictive instance of a restrictive instance
+            // which is redundant - we'll produce new type identities, but all type params have already been mapped.
+            // This also gives us a way to detect restrictive instances upon comparisons and _disable_ the "distributeive constraint"
+            // assignability check for them, which is distinctly unsafe, as once you have a restrctive instance, all the type parameters
+            // are constrained to `unknown` and produce tons of false positives/negatives!
+            type.restrictiveInstantiation.restrictiveInstantiation = type.restrictiveInstantiation;
+            return type.restrictiveInstantiation;
+        }
+        function instantiateIndexInfo(info, mapper) {
+            return info && createIndexInfo(instantiateType(info.type, mapper), info.isReadonly, info.declaration);
+        }
+        // Returns true if the given expression contains (at any level of nesting) a function or arrow expression
+        // that is subject to contextual typing.
+        function isContextSensitive(node) {
+            ts.Debug.assert(node.kind !== 165 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
+            switch (node.kind) {
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 165 /* MethodDeclaration */:
+                case 251 /* FunctionDeclaration */: // Function declarations can have context when annotated with a jsdoc @type
+                    return isContextSensitiveFunctionLikeDeclaration(node);
+                case 200 /* ObjectLiteralExpression */:
+                    return ts.some(node.properties, isContextSensitive);
+                case 199 /* ArrayLiteralExpression */:
+                    return ts.some(node.elements, isContextSensitive);
+                case 217 /* ConditionalExpression */:
+                    return isContextSensitive(node.whenTrue) ||
+                        isContextSensitive(node.whenFalse);
+                case 216 /* BinaryExpression */:
+                    return (node.operatorToken.kind === 56 /* BarBarToken */ || node.operatorToken.kind === 60 /* QuestionQuestionToken */) &&
+                        (isContextSensitive(node.left) || isContextSensitive(node.right));
+                case 288 /* PropertyAssignment */:
+                    return isContextSensitive(node.initializer);
+                case 207 /* ParenthesizedExpression */:
+                    return isContextSensitive(node.expression);
+                case 281 /* JsxAttributes */:
+                    return ts.some(node.properties, isContextSensitive) || ts.isJsxOpeningElement(node.parent) && ts.some(node.parent.parent.children, isContextSensitive);
+                case 280 /* JsxAttribute */: {
+                    // If there is no initializer, JSX attribute has a boolean value of true which is not context sensitive.
+                    var initializer = node.initializer;
+                    return !!initializer && isContextSensitive(initializer);
                 }
-                else if (type.flags & 1048576 /* Union */) {
-                    var unionContext_1 = context || createWideningContext(/*parent*/ undefined, /*propertyName*/ undefined, type.types);
-                    var widenedTypes = ts.sameMap(type.types, function (t) { return t.flags & 98304 /* Nullable */ ? t : getWidenedTypeWithContext(t, unionContext_1); });
-                    // Widening an empty object literal transitions from a highly restrictive type to
-                    // a highly inclusive one. For that reason we perform subtype reduction here if the
-                    // union includes empty object types (e.g. reducing {} | string to just {}).
-                    result = getUnionType(widenedTypes, ts.some(widenedTypes, isEmptyObjectType) ? 2 /* Subtype */ : 1 /* Literal */);
+                case 283 /* JsxExpression */: {
+                    // It is possible to that node.expression is undefined (e.g <div x={} />)
+                    var expression = node.expression;
+                    return !!expression && isContextSensitive(expression);
                 }
-                else if (type.flags & 2097152 /* Intersection */) {
-                    result = getIntersectionType(ts.sameMap(type.types, getWidenedType));
+            }
+            return false;
+        }
+        function isContextSensitiveFunctionLikeDeclaration(node) {
+            return (!ts.isFunctionDeclaration(node) || ts.isInJSFile(node) && !!getTypeForDeclarationFromJSDocComment(node)) &&
+                (hasContextSensitiveParameters(node) || hasContextSensitiveReturnExpression(node));
+        }
+        function hasContextSensitiveParameters(node) {
+            // Functions with type parameters are not context sensitive.
+            if (!node.typeParameters) {
+                // Functions with any parameters that lack type annotations are context sensitive.
+                if (ts.some(node.parameters, function (p) { return !ts.getEffectiveTypeAnnotationNode(p); })) {
+                    return true;
                 }
-                else if (isArrayType(type) || isTupleType(type)) {
-                    result = createTypeReference(type.target, ts.sameMap(getTypeArguments(type), getWidenedType));
+                if (node.kind !== 209 /* ArrowFunction */) {
+                    // If the first parameter is not an explicit 'this' parameter, then the function has
+                    // an implicit 'this' parameter which is subject to contextual typing.
+                    var parameter = ts.firstOrUndefined(node.parameters);
+                    if (!(parameter && ts.parameterIsThisKeyword(parameter))) {
+                        return true;
+                    }
                 }
-                if (result && context === undefined) {
-                    type.widened = result;
+            }
+            return false;
+        }
+        function hasContextSensitiveReturnExpression(node) {
+            // TODO(anhans): A block should be context-sensitive if it has a context-sensitive return value.
+            return !node.typeParameters && !ts.getEffectiveReturnTypeNode(node) && !!node.body && node.body.kind !== 230 /* Block */ && isContextSensitive(node.body);
+        }
+        function isContextSensitiveFunctionOrObjectLiteralMethod(func) {
+            return (ts.isInJSFile(func) && ts.isFunctionDeclaration(func) || isFunctionExpressionOrArrowFunction(func) || ts.isObjectLiteralMethod(func)) &&
+                isContextSensitiveFunctionLikeDeclaration(func);
+        }
+        function getTypeWithoutSignatures(type) {
+            if (type.flags & 524288 /* Object */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                if (resolved.constructSignatures.length || resolved.callSignatures.length) {
+                    var result = createObjectType(16 /* Anonymous */, type.symbol);
+                    result.members = resolved.members;
+                    result.properties = resolved.properties;
+                    result.callSignatures = ts.emptyArray;
+                    result.constructSignatures = ts.emptyArray;
+                    return result;
                 }
-                return result || type;
+            }
+            else if (type.flags & 2097152 /* Intersection */) {
+                return getIntersectionType(ts.map(type.types, getTypeWithoutSignatures));
             }
             return type;
         }
+        // TYPE CHECKING
+        function isTypeIdenticalTo(source, target) {
+            return isTypeRelatedTo(source, target, identityRelation);
+        }
+        function compareTypesIdentical(source, target) {
+            return isTypeRelatedTo(source, target, identityRelation) ? -1 /* True */ : 0 /* False */;
+        }
+        function compareTypesAssignable(source, target) {
+            return isTypeRelatedTo(source, target, assignableRelation) ? -1 /* True */ : 0 /* False */;
+        }
+        function compareTypesSubtypeOf(source, target) {
+            return isTypeRelatedTo(source, target, subtypeRelation) ? -1 /* True */ : 0 /* False */;
+        }
+        function isTypeSubtypeOf(source, target) {
+            return isTypeRelatedTo(source, target, subtypeRelation);
+        }
+        function isTypeAssignableTo(source, target) {
+            return isTypeRelatedTo(source, target, assignableRelation);
+        }
+        // An object type S is considered to be derived from an object type T if
+        // S is a union type and every constituent of S is derived from T,
+        // T is a union type and S is derived from at least one constituent of T, or
+        // S is a type variable with a base constraint that is derived from T,
+        // T is one of the global types Object and Function and S is a subtype of T, or
+        // T occurs directly or indirectly in an 'extends' clause of S.
+        // Note that this check ignores type parameters and only considers the
+        // inheritance hierarchy.
+        function isTypeDerivedFrom(source, target) {
+            return source.flags & 1048576 /* Union */ ? ts.every(source.types, function (t) { return isTypeDerivedFrom(t, target); }) :
+                target.flags & 1048576 /* Union */ ? ts.some(target.types, function (t) { return isTypeDerivedFrom(source, t); }) :
+                    source.flags & 58982400 /* InstantiableNonPrimitive */ ? isTypeDerivedFrom(getBaseConstraintOfType(source) || unknownType, target) :
+                        target === globalObjectType ? !!(source.flags & (524288 /* Object */ | 67108864 /* NonPrimitive */)) :
+                            target === globalFunctionType ? !!(source.flags & 524288 /* Object */) && isFunctionObjectType(source) :
+                                hasBaseType(source, getTargetType(target)) || (isArrayType(target) && !isReadonlyArrayType(target) && isTypeDerivedFrom(source, globalReadonlyArrayType));
+        }
         /**
-         * Reports implicit any errors that occur as a result of widening 'null' and 'undefined'
-         * to 'any'. A call to reportWideningErrorsInType is normally accompanied by a call to
-         * getWidenedType. But in some cases getWidenedType is called without reporting errors
-         * (type argument inference is an example).
+         * This is *not* a bi-directional relationship.
+         * If one needs to check both directions for comparability, use a second call to this function or 'checkTypeComparableTo'.
          *
-         * The return value indicates whether an error was in fact reported. The particular circumstances
-         * are on a best effort basis. Currently, if the null or undefined that causes widening is inside
-         * an object literal property (arbitrarily deeply), this function reports an error. If no error is
-         * reported, reportImplicitAnyError is a suitable fallback to report a general error.
+         * A type S is comparable to a type T if some (but not necessarily all) of the possible values of S are also possible values of T.
+         * It is used to check following cases:
+         *   - the types of the left and right sides of equality/inequality operators (`===`, `!==`, `==`, `!=`).
+         *   - the types of `case` clause expressions and their respective `switch` expressions.
+         *   - the type of an expression in a type assertion with the type being asserted.
          */
-        function reportWideningErrorsInType(type) {
-            var errorReported = false;
-            if (ts.getObjectFlags(type) & 524288 /* ContainsWideningType */) {
-                if (type.flags & 1048576 /* Union */) {
-                    if (ts.some(type.types, isEmptyObjectType)) {
-                        errorReported = true;
-                    }
-                    else {
-                        for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
-                            var t = _a[_i];
-                            if (reportWideningErrorsInType(t)) {
-                                errorReported = true;
-                            }
-                        }
-                    }
-                }
-                if (isArrayType(type) || isTupleType(type)) {
-                    for (var _b = 0, _c = getTypeArguments(type); _b < _c.length; _b++) {
-                        var t = _c[_b];
-                        if (reportWideningErrorsInType(t)) {
-                            errorReported = true;
-                        }
-                    }
-                }
-                if (isObjectLiteralType(type)) {
-                    for (var _d = 0, _e = getPropertiesOfObjectType(type); _d < _e.length; _d++) {
-                        var p = _e[_d];
-                        var t = getTypeOfSymbol(p);
-                        if (ts.getObjectFlags(t) & 524288 /* ContainsWideningType */) {
-                            if (!reportWideningErrorsInType(t)) {
-                                error(p.valueDeclaration, ts.Diagnostics.Object_literal_s_property_0_implicitly_has_an_1_type, symbolToString(p), typeToString(getWidenedType(t)));
-                            }
-                            errorReported = true;
-                        }
-                    }
-                }
+        function isTypeComparableTo(source, target) {
+            return isTypeRelatedTo(source, target, comparableRelation);
+        }
+        function areTypesComparable(type1, type2) {
+            return isTypeComparableTo(type1, type2) || isTypeComparableTo(type2, type1);
+        }
+        function checkTypeAssignableTo(source, target, errorNode, headMessage, containingMessageChain, errorOutputObject) {
+            return checkTypeRelatedTo(source, target, assignableRelation, errorNode, headMessage, containingMessageChain, errorOutputObject);
+        }
+        /**
+         * Like `checkTypeAssignableTo`, but if it would issue an error, instead performs structural comparisons of the types using the given expression node to
+         * attempt to issue more specific errors on, for example, specific object literal properties or tuple members.
+         */
+        function checkTypeAssignableToAndOptionallyElaborate(source, target, errorNode, expr, headMessage, containingMessageChain) {
+            return checkTypeRelatedToAndOptionallyElaborate(source, target, assignableRelation, errorNode, expr, headMessage, containingMessageChain, /*errorOutputContainer*/ undefined);
+        }
+        function checkTypeRelatedToAndOptionallyElaborate(source, target, relation, errorNode, expr, headMessage, containingMessageChain, errorOutputContainer) {
+            if (isTypeRelatedTo(source, target, relation))
+                return true;
+            if (!errorNode || !elaborateError(expr, source, target, relation, headMessage, containingMessageChain, errorOutputContainer)) {
+                return checkTypeRelatedTo(source, target, relation, errorNode, headMessage, containingMessageChain, errorOutputContainer);
             }
-            return errorReported;
+            return false;
         }
-        function reportImplicitAny(declaration, type, wideningKind) {
-            var typeAsString = typeToString(getWidenedType(type));
-            if (ts.isInJSFile(declaration) && !ts.isCheckJsEnabledForFile(ts.getSourceFileOfNode(declaration), compilerOptions)) {
-                // Only report implicit any errors/suggestions in TS and ts-check JS files
-                return;
+        function isOrHasGenericConditional(type) {
+            return !!(type.flags & 16777216 /* Conditional */ || (type.flags & 2097152 /* Intersection */ && ts.some(type.types, isOrHasGenericConditional)));
+        }
+        function elaborateError(node, source, target, relation, headMessage, containingMessageChain, errorOutputContainer) {
+            if (!node || isOrHasGenericConditional(target))
+                return false;
+            if (!checkTypeRelatedTo(source, target, relation, /*errorNode*/ undefined)
+                && elaborateDidYouMeanToCallOrConstruct(node, source, target, relation, headMessage, containingMessageChain, errorOutputContainer)) {
+                return true;
             }
-            var diagnostic;
-            switch (declaration.kind) {
-                case 209 /* BinaryExpression */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                    diagnostic = noImplicitAny ? ts.Diagnostics.Member_0_implicitly_has_an_1_type : ts.Diagnostics.Member_0_implicitly_has_an_1_type_but_a_better_type_may_be_inferred_from_usage;
-                    break;
-                case 156 /* Parameter */:
-                    var param = declaration;
-                    if (ts.isIdentifier(param.name) &&
-                        (ts.isCallSignatureDeclaration(param.parent) || ts.isMethodSignature(param.parent) || ts.isFunctionTypeNode(param.parent)) &&
-                        param.parent.parameters.indexOf(param) > -1 &&
-                        (resolveName(param, param.name.escapedText, 788968 /* Type */, undefined, param.name.escapedText, /*isUse*/ true) ||
-                            param.name.originalKeywordKind && ts.isTypeNodeKind(param.name.originalKeywordKind))) {
-                        var newName = "arg" + param.parent.parameters.indexOf(param);
-                        errorOrSuggestion(noImplicitAny, declaration, ts.Diagnostics.Parameter_has_a_name_but_no_type_Did_you_mean_0_Colon_1, newName, ts.declarationNameToString(param.name));
-                        return;
-                    }
-                    diagnostic = declaration.dotDotDotToken ?
-                        noImplicitAny ? ts.Diagnostics.Rest_parameter_0_implicitly_has_an_any_type : ts.Diagnostics.Rest_parameter_0_implicitly_has_an_any_type_but_a_better_type_may_be_inferred_from_usage :
-                        noImplicitAny ? ts.Diagnostics.Parameter_0_implicitly_has_an_1_type : ts.Diagnostics.Parameter_0_implicitly_has_an_1_type_but_a_better_type_may_be_inferred_from_usage;
-                    break;
-                case 191 /* BindingElement */:
-                    diagnostic = ts.Diagnostics.Binding_element_0_implicitly_has_an_1_type;
-                    if (!noImplicitAny) {
-                        // Don't issue a suggestion for binding elements since the codefix doesn't yet support them.
-                        return;
-                    }
-                    break;
-                case 300 /* JSDocFunctionType */:
-                    error(declaration, ts.Diagnostics.Function_type_which_lacks_return_type_annotation_implicitly_has_an_0_return_type, typeAsString);
-                    return;
-                case 244 /* FunctionDeclaration */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                    if (noImplicitAny && !declaration.name) {
-                        if (wideningKind === 3 /* GeneratorYield */) {
-                            error(declaration, ts.Diagnostics.Generator_implicitly_has_yield_type_0_because_it_does_not_yield_any_values_Consider_supplying_a_return_type_annotation, typeAsString);
-                        }
-                        else {
-                            error(declaration, ts.Diagnostics.Function_expression_which_lacks_return_type_annotation_implicitly_has_an_0_return_type, typeAsString);
-                        }
-                        return;
+            switch (node.kind) {
+                case 283 /* JsxExpression */:
+                case 207 /* ParenthesizedExpression */:
+                    return elaborateError(node.expression, source, target, relation, headMessage, containingMessageChain, errorOutputContainer);
+                case 216 /* BinaryExpression */:
+                    switch (node.operatorToken.kind) {
+                        case 62 /* EqualsToken */:
+                        case 27 /* CommaToken */:
+                            return elaborateError(node.right, source, target, relation, headMessage, containingMessageChain, errorOutputContainer);
                     }
-                    diagnostic = !noImplicitAny ? ts.Diagnostics._0_implicitly_has_an_1_return_type_but_a_better_type_may_be_inferred_from_usage :
-                        wideningKind === 3 /* GeneratorYield */ ? ts.Diagnostics._0_which_lacks_return_type_annotation_implicitly_has_an_1_yield_type :
-                            ts.Diagnostics._0_which_lacks_return_type_annotation_implicitly_has_an_1_return_type;
                     break;
-                case 186 /* MappedType */:
-                    if (noImplicitAny) {
-                        error(declaration, ts.Diagnostics.Mapped_object_type_implicitly_has_an_any_template_type);
-                    }
-                    return;
-                default:
-                    diagnostic = noImplicitAny ? ts.Diagnostics.Variable_0_implicitly_has_an_1_type : ts.Diagnostics.Variable_0_implicitly_has_an_1_type_but_a_better_type_may_be_inferred_from_usage;
+                case 200 /* ObjectLiteralExpression */:
+                    return elaborateObjectLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer);
+                case 199 /* ArrayLiteralExpression */:
+                    return elaborateArrayLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer);
+                case 281 /* JsxAttributes */:
+                    return elaborateJsxComponents(node, source, target, relation, containingMessageChain, errorOutputContainer);
+                case 209 /* ArrowFunction */:
+                    return elaborateArrowFunction(node, source, target, relation, containingMessageChain, errorOutputContainer);
             }
-            errorOrSuggestion(noImplicitAny, declaration, diagnostic, ts.declarationNameToString(ts.getNameOfDeclaration(declaration)), typeAsString);
+            return false;
         }
-        function reportErrorsFromWidening(declaration, type, wideningKind) {
-            if (produceDiagnostics && noImplicitAny && ts.getObjectFlags(type) & 524288 /* ContainsWideningType */ && (!wideningKind || !getContextualSignatureForFunctionLikeDeclaration(declaration))) {
-                // Report implicit any error within type if possible, otherwise report error on declaration
-                if (!reportWideningErrorsInType(type)) {
-                    reportImplicitAny(declaration, type, wideningKind);
+        function elaborateDidYouMeanToCallOrConstruct(node, source, target, relation, headMessage, containingMessageChain, errorOutputContainer) {
+            var callSignatures = getSignaturesOfType(source, 0 /* Call */);
+            var constructSignatures = getSignaturesOfType(source, 1 /* Construct */);
+            for (var _i = 0, _a = [constructSignatures, callSignatures]; _i < _a.length; _i++) {
+                var signatures = _a[_i];
+                if (ts.some(signatures, function (s) {
+                    var returnType = getReturnTypeOfSignature(s);
+                    return !(returnType.flags & (1 /* Any */ | 131072 /* Never */)) && checkTypeRelatedTo(returnType, target, relation, /*errorNode*/ undefined);
+                })) {
+                    var resultObj = errorOutputContainer || {};
+                    checkTypeAssignableTo(source, target, node, headMessage, containingMessageChain, resultObj);
+                    var diagnostic = resultObj.errors[resultObj.errors.length - 1];
+                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(node, signatures === constructSignatures ? ts.Diagnostics.Did_you_mean_to_use_new_with_this_expression : ts.Diagnostics.Did_you_mean_to_call_this_expression));
+                    return true;
                 }
             }
+            return false;
         }
-        function applyToParameterTypes(source, target, callback) {
-            var sourceCount = getParameterCount(source);
-            var targetCount = getParameterCount(target);
-            var sourceRestType = getEffectiveRestType(source);
-            var targetRestType = getEffectiveRestType(target);
-            var targetNonRestCount = targetRestType ? targetCount - 1 : targetCount;
-            var paramCount = sourceRestType ? targetNonRestCount : Math.min(sourceCount, targetNonRestCount);
-            var sourceThisType = getThisTypeOfSignature(source);
-            if (sourceThisType) {
-                var targetThisType = getThisTypeOfSignature(target);
-                if (targetThisType) {
-                    callback(sourceThisType, targetThisType);
-                }
-            }
-            for (var i = 0; i < paramCount; i++) {
-                callback(getTypeAtPosition(source, i), getTypeAtPosition(target, i));
+        function elaborateArrowFunction(node, source, target, relation, containingMessageChain, errorOutputContainer) {
+            // Don't elaborate blocks
+            if (ts.isBlock(node.body)) {
+                return false;
             }
-            if (targetRestType) {
-                callback(getRestTypeAtPosition(source, paramCount), targetRestType);
+            // Or functions with annotated parameter types
+            if (ts.some(node.parameters, ts.hasType)) {
+                return false;
             }
-        }
-        function applyToReturnTypes(source, target, callback) {
-            var sourceTypePredicate = getTypePredicateOfSignature(source);
-            var targetTypePredicate = getTypePredicateOfSignature(target);
-            if (sourceTypePredicate && targetTypePredicate && typePredicateKindsMatch(sourceTypePredicate, targetTypePredicate) && sourceTypePredicate.type && targetTypePredicate.type) {
-                callback(sourceTypePredicate.type, targetTypePredicate.type);
+            var sourceSig = getSingleCallSignature(source);
+            if (!sourceSig) {
+                return false;
             }
-            else {
-                callback(getReturnTypeOfSignature(source), getReturnTypeOfSignature(target));
+            var targetSignatures = getSignaturesOfType(target, 0 /* Call */);
+            if (!ts.length(targetSignatures)) {
+                return false;
             }
-        }
-        function createInferenceContext(typeParameters, signature, flags, compareTypes) {
-            return createInferenceContextWorker(typeParameters.map(createInferenceInfo), signature, flags, compareTypes || compareTypesAssignable);
-        }
-        function cloneInferenceContext(context, extraFlags) {
-            if (extraFlags === void 0) { extraFlags = 0; }
-            return context && createInferenceContextWorker(ts.map(context.inferences, cloneInferenceInfo), context.signature, context.flags | extraFlags, context.compareTypes);
-        }
-        function createInferenceContextWorker(inferences, signature, flags, compareTypes) {
-            var context = {
-                inferences: inferences,
-                signature: signature,
-                flags: flags,
-                compareTypes: compareTypes,
-                mapper: makeFunctionTypeMapper(function (t) { return mapToInferredType(context, t, /*fix*/ true); }),
-                nonFixingMapper: makeFunctionTypeMapper(function (t) { return mapToInferredType(context, t, /*fix*/ false); }),
-            };
-            return context;
-        }
-        function mapToInferredType(context, t, fix) {
-            var inferences = context.inferences;
-            for (var i = 0; i < inferences.length; i++) {
-                var inference = inferences[i];
-                if (t === inference.typeParameter) {
-                    if (fix && !inference.isFixed) {
-                        clearCachedInferences(inferences);
-                        inference.isFixed = true;
+            var returnExpression = node.body;
+            var sourceReturn = getReturnTypeOfSignature(sourceSig);
+            var targetReturn = getUnionType(ts.map(targetSignatures, getReturnTypeOfSignature));
+            if (!checkTypeRelatedTo(sourceReturn, targetReturn, relation, /*errorNode*/ undefined)) {
+                var elaborated = returnExpression && elaborateError(returnExpression, sourceReturn, targetReturn, relation, /*headMessage*/ undefined, containingMessageChain, errorOutputContainer);
+                if (elaborated) {
+                    return elaborated;
+                }
+                var resultObj = errorOutputContainer || {};
+                checkTypeRelatedTo(sourceReturn, targetReturn, relation, returnExpression, /*message*/ undefined, containingMessageChain, resultObj);
+                if (resultObj.errors) {
+                    if (target.symbol && ts.length(target.symbol.declarations)) {
+                        ts.addRelatedInfo(resultObj.errors[resultObj.errors.length - 1], ts.createDiagnosticForNode(target.symbol.declarations[0], ts.Diagnostics.The_expected_type_comes_from_the_return_type_of_this_signature));
                     }
-                    return getInferredType(context, i);
+                    if ((ts.getFunctionFlags(node) & 2 /* Async */) === 0
+                        // exclude cases where source itself is promisy - this way we don't make a suggestion when relating
+                        // an IPromise and a Promise that are slightly different
+                        && !getTypeOfPropertyOfType(sourceReturn, "then")
+                        && checkTypeRelatedTo(createPromiseType(sourceReturn), targetReturn, relation, /*errorNode*/ undefined)) {
+                        ts.addRelatedInfo(resultObj.errors[resultObj.errors.length - 1], ts.createDiagnosticForNode(node, ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async));
+                    }
+                    return true;
                 }
             }
-            return t;
+            return false;
         }
-        function clearCachedInferences(inferences) {
-            for (var _i = 0, inferences_1 = inferences; _i < inferences_1.length; _i++) {
-                var inference = inferences_1[_i];
-                if (!inference.isFixed) {
-                    inference.inferredType = undefined;
+        function getBestMatchIndexedAccessTypeOrUndefined(source, target, nameType) {
+            var idx = getIndexedAccessTypeOrUndefined(target, nameType);
+            if (idx) {
+                return idx;
+            }
+            if (target.flags & 1048576 /* Union */) {
+                var best = getBestMatchingType(source, target);
+                if (best) {
+                    return getIndexedAccessTypeOrUndefined(best, nameType);
                 }
             }
         }
-        function createInferenceInfo(typeParameter) {
-            return {
-                typeParameter: typeParameter,
-                candidates: undefined,
-                contraCandidates: undefined,
-                inferredType: undefined,
-                priority: undefined,
-                topLevel: true,
-                isFixed: false
-            };
-        }
-        function cloneInferenceInfo(inference) {
-            return {
-                typeParameter: inference.typeParameter,
-                candidates: inference.candidates && inference.candidates.slice(),
-                contraCandidates: inference.contraCandidates && inference.contraCandidates.slice(),
-                inferredType: inference.inferredType,
-                priority: inference.priority,
-                topLevel: inference.topLevel,
-                isFixed: inference.isFixed
-            };
-        }
-        function cloneInferredPartOfContext(context) {
-            var inferences = ts.filter(context.inferences, hasInferenceCandidates);
-            return inferences.length ?
-                createInferenceContextWorker(ts.map(inferences, cloneInferenceInfo), context.signature, context.flags, context.compareTypes) :
-                undefined;
-        }
-        function getMapperFromContext(context) {
-            return context && context.mapper;
-        }
-        // Return true if the given type could possibly reference a type parameter for which
-        // we perform type inference (i.e. a type parameter of a generic function). We cache
-        // results for union and intersection types for performance reasons.
-        function couldContainTypeVariables(type) {
-            var objectFlags = ts.getObjectFlags(type);
-            if (objectFlags & 67108864 /* CouldContainTypeVariablesComputed */) {
-                return !!(objectFlags & 134217728 /* CouldContainTypeVariables */);
+        function checkExpressionForMutableLocationWithContextualType(next, sourcePropType) {
+            next.contextualType = sourcePropType;
+            try {
+                return checkExpressionForMutableLocation(next, 1 /* Contextual */, sourcePropType);
             }
-            var result = !!(type.flags & 63176704 /* Instantiable */ ||
-                objectFlags & 4 /* Reference */ && (type.node || ts.forEach(getTypeArguments(type), couldContainTypeVariables)) ||
-                objectFlags & 16 /* Anonymous */ && type.symbol && type.symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 2048 /* TypeLiteral */ | 4096 /* ObjectLiteral */) && type.symbol.declarations ||
-                objectFlags & (32 /* Mapped */ | 131072 /* ObjectRestType */) ||
-                type.flags & 3145728 /* UnionOrIntersection */ && !(type.flags & 1024 /* EnumLiteral */) && ts.some(type.types, couldContainTypeVariables));
-            if (type.flags & 3899393 /* ObjectFlagsType */) {
-                type.objectFlags |= 67108864 /* CouldContainTypeVariablesComputed */ | (result ? 134217728 /* CouldContainTypeVariables */ : 0);
+            finally {
+                next.contextualType = undefined;
             }
-            return result;
-        }
-        function isTypeParameterAtTopLevel(type, typeParameter) {
-            return !!(type === typeParameter ||
-                type.flags & 3145728 /* UnionOrIntersection */ && ts.some(type.types, function (t) { return isTypeParameterAtTopLevel(t, typeParameter); }) ||
-                type.flags & 16777216 /* Conditional */ && (isTypeParameterAtTopLevel(getTrueTypeFromConditionalType(type), typeParameter) ||
-                    isTypeParameterAtTopLevel(getFalseTypeFromConditionalType(type), typeParameter)));
-        }
-        /** Create an object with properties named in the string literal type. Every property has type `any` */
-        function createEmptyObjectTypeFromStringLiteral(type) {
-            var members = ts.createSymbolTable();
-            forEachType(type, function (t) {
-                if (!(t.flags & 128 /* StringLiteral */)) {
-                    return;
-                }
-                var name = ts.escapeLeadingUnderscores(t.value);
-                var literalProp = createSymbol(4 /* Property */, name);
-                literalProp.type = anyType;
-                if (t.symbol) {
-                    literalProp.declarations = t.symbol.declarations;
-                    literalProp.valueDeclaration = t.symbol.valueDeclaration;
-                }
-                members.set(name, literalProp);
-            });
-            var indexInfo = type.flags & 4 /* String */ ? createIndexInfo(emptyObjectType, /*isReadonly*/ false) : undefined;
-            return createAnonymousType(undefined, members, ts.emptyArray, ts.emptyArray, indexInfo, undefined);
         }
         /**
-         * Infer a suitable input type for a homomorphic mapped type { [P in keyof T]: X }. We construct
-         * an object type with the same set of properties as the source type, where the type of each
-         * property is computed by inferring from the source property type to X for the type
-         * variable T[P] (i.e. we treat the type T[P] as the type variable we're inferring for).
+         * For every element returned from the iterator, checks that element to issue an error on a property of that element's type
+         * If that element would issue an error, we first attempt to dive into that element's inner expression and issue a more specific error by recuring into `elaborateError`
+         * Otherwise, we issue an error on _every_ element which fail the assignability check
          */
-        function inferTypeForHomomorphicMappedType(source, target, constraint) {
-            var key = source.id + "," + target.id + "," + constraint.id;
-            if (reverseMappedCache.has(key)) {
-                return reverseMappedCache.get(key);
-            }
-            reverseMappedCache.set(key, undefined);
-            var type = createReverseMappedType(source, target, constraint);
-            reverseMappedCache.set(key, type);
-            return type;
-        }
-        // We consider a type to be partially inferable if it isn't marked non-inferable or if it is
-        // an object literal type with at least one property of an inferable type. For example, an object
-        // literal { a: 123, b: x => true } is marked non-inferable because it contains a context sensitive
-        // arrow function, but is considered partially inferable because property 'a' has an inferable type.
-        function isPartiallyInferableType(type) {
-            return !(ts.getObjectFlags(type) & 2097152 /* NonInferrableType */) ||
-                isObjectLiteralType(type) && ts.some(getPropertiesOfType(type), function (prop) { return isPartiallyInferableType(getTypeOfSymbol(prop)); });
-        }
-        function createReverseMappedType(source, target, constraint) {
-            // We consider a source type reverse mappable if it has a string index signature or if
-            // it has one or more properties and is of a partially inferable type.
-            if (!(getIndexInfoOfType(source, 0 /* String */) || getPropertiesOfType(source).length !== 0 && isPartiallyInferableType(source))) {
-                return undefined;
-            }
-            // For arrays and tuples we infer new arrays and tuples where the reverse mapping has been
-            // applied to the element type(s).
-            if (isArrayType(source)) {
-                return createArrayType(inferReverseMappedType(getTypeArguments(source)[0], target, constraint), isReadonlyArrayType(source));
-            }
-            if (isTupleType(source)) {
-                var elementTypes = ts.map(getTypeArguments(source), function (t) { return inferReverseMappedType(t, target, constraint); });
-                var minLength = getMappedTypeModifiers(target) & 4 /* IncludeOptional */ ?
-                    getTypeReferenceArity(source) - (source.target.hasRestElement ? 1 : 0) : source.target.minLength;
-                return createTupleType(elementTypes, minLength, source.target.hasRestElement, source.target.readonly, source.target.associatedNames);
+        function elaborateElementwise(iterator, source, target, relation, containingMessageChain, errorOutputContainer) {
+            // Assignability failure - check each prop individually, and if that fails, fall back on the bad error span
+            var reportedError = false;
+            for (var status = iterator.next(); !status.done; status = iterator.next()) {
+                var _a = status.value, prop = _a.errorNode, next = _a.innerExpression, nameType = _a.nameType, errorMessage = _a.errorMessage;
+                var targetPropType = getBestMatchIndexedAccessTypeOrUndefined(source, target, nameType);
+                if (!targetPropType || targetPropType.flags & 8388608 /* IndexedAccess */)
+                    continue; // Don't elaborate on indexes on generic variables
+                var sourcePropType = getIndexedAccessTypeOrUndefined(source, nameType);
+                if (sourcePropType && !checkTypeRelatedTo(sourcePropType, targetPropType, relation, /*errorNode*/ undefined)) {
+                    var elaborated = next && elaborateError(next, sourcePropType, targetPropType, relation, /*headMessage*/ undefined, containingMessageChain, errorOutputContainer);
+                    if (elaborated) {
+                        reportedError = true;
+                    }
+                    else {
+                        // Issue error on the prop itself, since the prop couldn't elaborate the error
+                        var resultObj = errorOutputContainer || {};
+                        // Use the expression type, if available
+                        var specificSource = next ? checkExpressionForMutableLocationWithContextualType(next, sourcePropType) : sourcePropType;
+                        var result = checkTypeRelatedTo(specificSource, targetPropType, relation, prop, errorMessage, containingMessageChain, resultObj);
+                        if (result && specificSource !== sourcePropType) {
+                            // If for whatever reason the expression type doesn't yield an error, make sure we still issue an error on the sourcePropType
+                            checkTypeRelatedTo(sourcePropType, targetPropType, relation, prop, errorMessage, containingMessageChain, resultObj);
+                        }
+                        if (resultObj.errors) {
+                            var reportedDiag = resultObj.errors[resultObj.errors.length - 1];
+                            var propertyName = isTypeUsableAsPropertyName(nameType) ? getPropertyNameFromType(nameType) : undefined;
+                            var targetProp = propertyName !== undefined ? getPropertyOfType(target, propertyName) : undefined;
+                            var issuedElaboration = false;
+                            if (!targetProp) {
+                                var indexInfo = isTypeAssignableToKind(nameType, 296 /* NumberLike */) && getIndexInfoOfType(target, 1 /* Number */) ||
+                                    getIndexInfoOfType(target, 0 /* String */) ||
+                                    undefined;
+                                if (indexInfo && indexInfo.declaration && !ts.getSourceFileOfNode(indexInfo.declaration).hasNoDefaultLib) {
+                                    issuedElaboration = true;
+                                    ts.addRelatedInfo(reportedDiag, ts.createDiagnosticForNode(indexInfo.declaration, ts.Diagnostics.The_expected_type_comes_from_this_index_signature));
+                                }
+                            }
+                            if (!issuedElaboration && (targetProp && ts.length(targetProp.declarations) || target.symbol && ts.length(target.symbol.declarations))) {
+                                var targetNode = targetProp && ts.length(targetProp.declarations) ? targetProp.declarations[0] : target.symbol.declarations[0];
+                                if (!ts.getSourceFileOfNode(targetNode).hasNoDefaultLib) {
+                                    ts.addRelatedInfo(reportedDiag, ts.createDiagnosticForNode(targetNode, ts.Diagnostics.The_expected_type_comes_from_property_0_which_is_declared_here_on_type_1, propertyName && !(nameType.flags & 8192 /* UniqueESSymbol */) ? ts.unescapeLeadingUnderscores(propertyName) : typeToString(nameType), typeToString(target)));
+                                }
+                            }
+                        }
+                        reportedError = true;
+                    }
+                }
             }
-            // For all other object types we infer a new object type where the reverse mapping has been
-            // applied to the type of each property.
-            var reversed = createObjectType(2048 /* ReverseMapped */ | 16 /* Anonymous */, /*symbol*/ undefined);
-            reversed.source = source;
-            reversed.mappedType = target;
-            reversed.constraintType = constraint;
-            return reversed;
-        }
-        function getTypeOfReverseMappedSymbol(symbol) {
-            return inferReverseMappedType(symbol.propertyType, symbol.mappedType, symbol.constraintType);
+            return reportedError;
         }
-        function inferReverseMappedType(sourceType, target, constraint) {
-            var typeParameter = getIndexedAccessType(constraint.type, getTypeParameterFromMappedType(target));
-            var templateType = getTemplateTypeFromMappedType(target);
-            var inference = createInferenceInfo(typeParameter);
-            inferTypes([inference], sourceType, templateType);
-            return getTypeFromInference(inference) || unknownType;
+        function generateJsxAttributes(node) {
+            var _i, _a, prop;
+            return __generator(this, function (_b) {
+                switch (_b.label) {
+                    case 0:
+                        if (!ts.length(node.properties))
+                            return [2 /*return*/];
+                        _i = 0, _a = node.properties;
+                        _b.label = 1;
+                    case 1:
+                        if (!(_i < _a.length)) return [3 /*break*/, 4];
+                        prop = _a[_i];
+                        if (ts.isJsxSpreadAttribute(prop))
+                            return [3 /*break*/, 3];
+                        return [4 /*yield*/, { errorNode: prop.name, innerExpression: prop.initializer, nameType: getLiteralType(ts.idText(prop.name)) }];
+                    case 2:
+                        _b.sent();
+                        _b.label = 3;
+                    case 3:
+                        _i++;
+                        return [3 /*break*/, 1];
+                    case 4: return [2 /*return*/];
+                }
+            });
         }
-        function getUnmatchedProperties(source, target, requireOptionalProperties, matchDiscriminantProperties) {
-            var properties, _i, properties_2, targetProp, sourceProp, targetType, sourceType;
+        function generateJsxChildren(node, getInvalidTextDiagnostic) {
+            var memberOffset, i, child, nameType, elem;
             return __generator(this, function (_a) {
                 switch (_a.label) {
                     case 0:
-                        properties = getPropertiesOfType(target);
-                        _i = 0, properties_2 = properties;
+                        if (!ts.length(node.children))
+                            return [2 /*return*/];
+                        memberOffset = 0;
+                        i = 0;
                         _a.label = 1;
                     case 1:
-                        if (!(_i < properties_2.length)) return [3 /*break*/, 6];
-                        targetProp = properties_2[_i];
-                        // TODO: remove this when we support static private identifier fields and find other solutions to get privateNamesAndStaticFields test to pass
-                        if (isStaticPrivateIdentifierProperty(targetProp)) {
-                            return [3 /*break*/, 5];
-                        }
-                        if (!(requireOptionalProperties || !(targetProp.flags & 16777216 /* Optional */ || ts.getCheckFlags(targetProp) & 48 /* Partial */))) return [3 /*break*/, 5];
-                        sourceProp = getPropertyOfType(source, targetProp.escapedName);
-                        if (!!sourceProp) return [3 /*break*/, 3];
-                        return [4 /*yield*/, targetProp];
+                        if (!(i < node.children.length)) return [3 /*break*/, 5];
+                        child = node.children[i];
+                        nameType = getLiteralType(i - memberOffset);
+                        elem = getElaborationElementForJsxChild(child, nameType, getInvalidTextDiagnostic);
+                        if (!elem) return [3 /*break*/, 3];
+                        return [4 /*yield*/, elem];
                     case 2:
                         _a.sent();
-                        return [3 /*break*/, 5];
+                        return [3 /*break*/, 4];
                     case 3:
-                        if (!matchDiscriminantProperties) return [3 /*break*/, 5];
-                        targetType = getTypeOfSymbol(targetProp);
-                        if (!(targetType.flags & 109440 /* Unit */)) return [3 /*break*/, 5];
-                        sourceType = getTypeOfSymbol(sourceProp);
-                        if (!!(sourceType.flags & 1 /* Any */ || getRegularTypeOfLiteralType(sourceType) === getRegularTypeOfLiteralType(targetType))) return [3 /*break*/, 5];
-                        return [4 /*yield*/, targetProp];
+                        memberOffset++;
+                        _a.label = 4;
                     case 4:
-                        _a.sent();
-                        _a.label = 5;
-                    case 5:
-                        _i++;
+                        i++;
                         return [3 /*break*/, 1];
-                    case 6: return [2 /*return*/];
+                    case 5: return [2 /*return*/];
                 }
             });
         }
-        function getUnmatchedProperty(source, target, requireOptionalProperties, matchDiscriminantProperties) {
-            var result = getUnmatchedProperties(source, target, requireOptionalProperties, matchDiscriminantProperties).next();
-            if (!result.done)
-                return result.value;
-        }
-        function tupleTypesDefinitelyUnrelated(source, target) {
-            return target.target.minLength > source.target.minLength ||
-                !getRestTypeOfTupleType(target) && (!!getRestTypeOfTupleType(source) || getLengthOfTupleType(target) < getLengthOfTupleType(source));
-        }
-        function typesDefinitelyUnrelated(source, target) {
-            // Two tuple types with incompatible arities are definitely unrelated.
-            // Two object types that each have a property that is unmatched in the other are definitely unrelated.
-            return isTupleType(source) && isTupleType(target) && tupleTypesDefinitelyUnrelated(source, target) ||
-                !!getUnmatchedProperty(source, target, /*requireOptionalProperties*/ false, /*matchDiscriminantProperties*/ true) &&
-                    !!getUnmatchedProperty(target, source, /*requireOptionalProperties*/ false, /*matchDiscriminantProperties*/ true);
-        }
-        function getTypeFromInference(inference) {
-            return inference.candidates ? getUnionType(inference.candidates, 2 /* Subtype */) :
-                inference.contraCandidates ? getIntersectionType(inference.contraCandidates) :
-                    undefined;
-        }
-        function hasSkipDirectInferenceFlag(node) {
-            return !!getNodeLinks(node).skipDirectInference;
-        }
-        function isFromInferenceBlockedSource(type) {
-            return !!(type.symbol && ts.some(type.symbol.declarations, hasSkipDirectInferenceFlag));
-        }
-        function inferTypes(inferences, originalSource, originalTarget, priority, contravariant) {
-            if (priority === void 0) { priority = 0; }
-            if (contravariant === void 0) { contravariant = false; }
-            var symbolOrTypeStack;
-            var visited;
-            var bivariant = false;
-            var propagationType;
-            var inferencePriority = 512 /* MaxValue */;
-            var allowComplexConstraintInference = true;
-            inferFromTypes(originalSource, originalTarget);
-            function inferFromTypes(source, target) {
-                if (!couldContainTypeVariables(target)) {
-                    return;
-                }
-                if (source === wildcardType) {
-                    // We are inferring from an 'any' type. We want to infer this type for every type parameter
-                    // referenced in the target type, so we record it as the propagation type and infer from the
-                    // target to itself. Then, as we find candidates we substitute the propagation type.
-                    var savePropagationType = propagationType;
-                    propagationType = source;
-                    inferFromTypes(target, target);
-                    propagationType = savePropagationType;
-                    return;
-                }
-                if (source.aliasSymbol && source.aliasTypeArguments && source.aliasSymbol === target.aliasSymbol) {
-                    // Source and target are types originating in the same generic type alias declaration.
-                    // Simply infer from source type arguments to target type arguments.
-                    inferFromTypeArguments(source.aliasTypeArguments, target.aliasTypeArguments, getAliasVariances(source.aliasSymbol));
-                    return;
-                }
-                if (source === target && source.flags & 3145728 /* UnionOrIntersection */) {
-                    // When source and target are the same union or intersection type, just relate each constituent
-                    // type to itself.
-                    for (var _i = 0, _a = source.types; _i < _a.length; _i++) {
-                        var t = _a[_i];
-                        inferFromTypes(t, t);
-                    }
-                    return;
-                }
-                if (target.flags & 1048576 /* Union */) {
-                    // First, infer between identically matching source and target constituents and remove the
-                    // matching types.
-                    var _b = inferFromMatchingTypes(source.flags & 1048576 /* Union */ ? source.types : [source], target.types, isTypeOrBaseIdenticalTo), tempSources = _b[0], tempTargets = _b[1];
-                    // Next, infer between closely matching source and target constituents and remove
-                    // the matching types. Types closely match when they are instantiations of the same
-                    // object type or instantiations of the same type alias.
-                    var _c = inferFromMatchingTypes(tempSources, tempTargets, isTypeCloselyMatchedBy), sources = _c[0], targets = _c[1];
-                    if (targets.length === 0) {
-                        return;
-                    }
-                    target = getUnionType(targets);
-                    if (sources.length === 0) {
-                        // All source constituents have been matched and there is nothing further to infer from.
-                        // However, simply making no inferences is undesirable because it could ultimately mean
-                        // inferring a type parameter constraint. Instead, make a lower priority inference from
-                        // the full source to whatever remains in the target. For example, when inferring from
-                        // string to 'string | T', make a lower priority inference of string for T.
-                        inferWithPriority(source, target, 1 /* NakedTypeVariable */);
-                        return;
-                    }
-                    source = getUnionType(sources);
-                }
-                else if (target.flags & 2097152 /* Intersection */ && ts.some(target.types, function (t) { return !!getInferenceInfoForType(t) || (isGenericMappedType(t) && !!getInferenceInfoForType(getHomomorphicTypeVariable(t) || neverType)); })) {
-                    // We reduce intersection types only when they contain naked type parameters. For example, when
-                    // inferring from 'string[] & { extra: any }' to 'string[] & T' we want to remove string[] and
-                    // infer { extra: any } for T. But when inferring to 'string[] & Iterable<T>' we want to keep the
-                    // string[] on the source side and infer string for T.
-                    // Likewise, we consider a homomorphic mapped type constrainted to the target type parameter as similar to a "naked type variable"
-                    // in such scenarios.
-                    if (!(source.flags & 1048576 /* Union */)) {
-                        // Infer between identically matching source and target constituents and remove the matching types.
-                        var _d = inferFromMatchingTypes(source.flags & 2097152 /* Intersection */ ? source.types : [source], target.types, isTypeIdenticalTo), sources = _d[0], targets = _d[1];
-                        if (sources.length === 0 || targets.length === 0) {
-                            return;
-                        }
-                        source = getIntersectionType(sources);
-                        target = getIntersectionType(targets);
+        function getElaborationElementForJsxChild(child, nameType, getInvalidTextDiagnostic) {
+            switch (child.kind) {
+                case 283 /* JsxExpression */:
+                    // child is of the type of the expression
+                    return { errorNode: child, innerExpression: child.expression, nameType: nameType };
+                case 11 /* JsxText */:
+                    if (child.containsOnlyTriviaWhiteSpaces) {
+                        break; // Whitespace only jsx text isn't real jsx text
                     }
+                    // child is a string
+                    return { errorNode: child, innerExpression: undefined, nameType: nameType, errorMessage: getInvalidTextDiagnostic() };
+                case 273 /* JsxElement */:
+                case 274 /* JsxSelfClosingElement */:
+                case 277 /* JsxFragment */:
+                    // child is of type JSX.Element
+                    return { errorNode: child, innerExpression: child, nameType: nameType };
+                default:
+                    return ts.Debug.assertNever(child, "Found invalid jsx child");
+            }
+        }
+        function elaborateJsxComponents(node, source, target, relation, containingMessageChain, errorOutputContainer) {
+            var result = elaborateElementwise(generateJsxAttributes(node), source, target, relation, containingMessageChain, errorOutputContainer);
+            var invalidTextDiagnostic;
+            if (ts.isJsxOpeningElement(node.parent) && ts.isJsxElement(node.parent.parent)) {
+                var containingElement = node.parent.parent;
+                var childPropName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(node));
+                var childrenPropName = childPropName === undefined ? "children" : ts.unescapeLeadingUnderscores(childPropName);
+                var childrenNameType = getLiteralType(childrenPropName);
+                var childrenTargetType = getIndexedAccessType(target, childrenNameType);
+                var validChildren = ts.getSemanticJsxChildren(containingElement.children);
+                if (!ts.length(validChildren)) {
+                    return result;
                 }
-                else if (target.flags & (8388608 /* IndexedAccess */ | 33554432 /* Substitution */)) {
-                    target = getActualTypeVariable(target);
-                }
-                if (target.flags & 8650752 /* TypeVariable */) {
-                    // If target is a type parameter, make an inference, unless the source type contains
-                    // the anyFunctionType (the wildcard type that's used to avoid contextually typing functions).
-                    // Because the anyFunctionType is internal, it should not be exposed to the user by adding
-                    // it as an inference candidate. Hopefully, a better candidate will come along that does
-                    // not contain anyFunctionType when we come back to this argument for its second round
-                    // of inference. Also, we exclude inferences for silentNeverType (which is used as a wildcard
-                    // when constructing types from type parameters that had no inference candidates).
-                    if (ts.getObjectFlags(source) & 2097152 /* NonInferrableType */ || source === nonInferrableAnyType || source === silentNeverType ||
-                        (priority & 32 /* ReturnType */ && (source === autoType || source === autoArrayType)) || isFromInferenceBlockedSource(source)) {
-                        return;
-                    }
-                    var inference = getInferenceInfoForType(target);
-                    if (inference) {
-                        if (!inference.isFixed) {
-                            if (inference.priority === undefined || priority < inference.priority) {
-                                inference.candidates = undefined;
-                                inference.contraCandidates = undefined;
-                                inference.topLevel = true;
-                                inference.priority = priority;
-                            }
-                            if (priority === inference.priority) {
-                                var candidate = propagationType || source;
-                                // We make contravariant inferences only if we are in a pure contravariant position,
-                                // i.e. only if we have not descended into a bivariant position.
-                                if (contravariant && !bivariant) {
-                                    if (!ts.contains(inference.contraCandidates, candidate)) {
-                                        inference.contraCandidates = ts.append(inference.contraCandidates, candidate);
-                                        clearCachedInferences(inferences);
-                                    }
-                                }
-                                else if (!ts.contains(inference.candidates, candidate)) {
-                                    inference.candidates = ts.append(inference.candidates, candidate);
-                                    clearCachedInferences(inferences);
-                                }
-                            }
-                            if (!(priority & 32 /* ReturnType */) && target.flags & 262144 /* TypeParameter */ && inference.topLevel && !isTypeParameterAtTopLevel(originalTarget, target)) {
-                                inference.topLevel = false;
-                                clearCachedInferences(inferences);
-                            }
-                        }
-                        inferencePriority = Math.min(inferencePriority, priority);
-                        return;
+                var moreThanOneRealChildren = ts.length(validChildren) > 1;
+                var arrayLikeTargetParts = filterType(childrenTargetType, isArrayOrTupleLikeType);
+                var nonArrayLikeTargetParts = filterType(childrenTargetType, function (t) { return !isArrayOrTupleLikeType(t); });
+                if (moreThanOneRealChildren) {
+                    if (arrayLikeTargetParts !== neverType) {
+                        var realSource = createTupleType(checkJsxChildren(containingElement, 0 /* Normal */));
+                        var children = generateJsxChildren(containingElement, getInvalidTextualChildDiagnostic);
+                        result = elaborateElementwise(children, realSource, arrayLikeTargetParts, relation, containingMessageChain, errorOutputContainer) || result;
                     }
-                    else {
-                        // Infer to the simplified version of an indexed access, if possible, to (hopefully) expose more bare type parameters to the inference engine
-                        var simplified = getSimplifiedType(target, /*writing*/ false);
-                        if (simplified !== target) {
-                            invokeOnce(source, simplified, inferFromTypes);
-                        }
-                        else if (target.flags & 8388608 /* IndexedAccess */) {
-                            var indexType = getSimplifiedType(target.indexType, /*writing*/ false);
-                            // Generally simplifications of instantiable indexes are avoided to keep relationship checking correct, however if our target is an access, we can consider
-                            // that key of that access to be "instantiated", since we're looking to find the infernce goal in any way we can.
-                            if (indexType.flags & 63176704 /* Instantiable */) {
-                                var simplified_1 = distributeIndexOverObjectType(getSimplifiedType(target.objectType, /*writing*/ false), indexType, /*writing*/ false);
-                                if (simplified_1 && simplified_1 !== target) {
-                                    invokeOnce(source, simplified_1, inferFromTypes);
-                                }
-                            }
+                    else if (!isTypeRelatedTo(getIndexedAccessType(source, childrenNameType), childrenTargetType, relation)) {
+                        // arity mismatch
+                        result = true;
+                        var diag = error(containingElement.openingElement.tagName, ts.Diagnostics.This_JSX_tag_s_0_prop_expects_a_single_child_of_type_1_but_multiple_children_were_provided, childrenPropName, typeToString(childrenTargetType));
+                        if (errorOutputContainer && errorOutputContainer.skipLogging) {
+                            (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
                         }
                     }
                 }
-                if (ts.getObjectFlags(source) & 4 /* Reference */ && ts.getObjectFlags(target) & 4 /* Reference */ && (source.target === target.target || isArrayType(source) && isArrayType(target)) &&
-                    !(source.node && target.node)) {
-                    // If source and target are references to the same generic type, infer from type arguments
-                    inferFromTypeArguments(getTypeArguments(source), getTypeArguments(target), getVariances(source.target));
-                }
-                else if (source.flags & 4194304 /* Index */ && target.flags & 4194304 /* Index */) {
-                    contravariant = !contravariant;
-                    inferFromTypes(source.type, target.type);
-                    contravariant = !contravariant;
-                }
-                else if ((isLiteralType(source) || source.flags & 4 /* String */) && target.flags & 4194304 /* Index */) {
-                    var empty = createEmptyObjectTypeFromStringLiteral(source);
-                    contravariant = !contravariant;
-                    inferWithPriority(empty, target.type, 64 /* LiteralKeyof */);
-                    contravariant = !contravariant;
-                }
-                else if (source.flags & 8388608 /* IndexedAccess */ && target.flags & 8388608 /* IndexedAccess */) {
-                    inferFromTypes(source.objectType, target.objectType);
-                    inferFromTypes(source.indexType, target.indexType);
-                }
-                else if (source.flags & 16777216 /* Conditional */ && target.flags & 16777216 /* Conditional */) {
-                    inferFromTypes(source.checkType, target.checkType);
-                    inferFromTypes(source.extendsType, target.extendsType);
-                    inferFromTypes(getTrueTypeFromConditionalType(source), getTrueTypeFromConditionalType(target));
-                    inferFromTypes(getFalseTypeFromConditionalType(source), getFalseTypeFromConditionalType(target));
-                }
-                else if (target.flags & 16777216 /* Conditional */) {
-                    var savePriority = priority;
-                    priority |= contravariant ? 16 /* ContravariantConditional */ : 0;
-                    var targetTypes = [getTrueTypeFromConditionalType(target), getFalseTypeFromConditionalType(target)];
-                    inferToMultipleTypes(source, targetTypes, target.flags);
-                    priority = savePriority;
-                }
-                else if (target.flags & 3145728 /* UnionOrIntersection */) {
-                    inferToMultipleTypes(source, target.types, target.flags);
-                }
-                else if (source.flags & 1048576 /* Union */) {
-                    // Source is a union or intersection type, infer from each constituent type
-                    var sourceTypes = source.types;
-                    for (var _e = 0, sourceTypes_2 = sourceTypes; _e < sourceTypes_2.length; _e++) {
-                        var sourceType = sourceTypes_2[_e];
-                        inferFromTypes(sourceType, target);
-                    }
-                }
                 else {
-                    source = getReducedType(source);
-                    if (!(priority & 128 /* NoConstraints */ && source.flags & (2097152 /* Intersection */ | 63176704 /* Instantiable */))) {
-                        var apparentSource = getApparentType(source);
-                        // getApparentType can return _any_ type, since an indexed access or conditional may simplify to any other type.
-                        // If that occurs and it doesn't simplify to an object or intersection, we'll need to restart `inferFromTypes`
-                        // with the simplified source.
-                        if (apparentSource !== source && allowComplexConstraintInference && !(apparentSource.flags & (524288 /* Object */ | 2097152 /* Intersection */))) {
-                            // TODO: The `allowComplexConstraintInference` flag is a hack! This forbids inference from complex constraints within constraints!
-                            // This isn't required algorithmically, but rather is used to lower the memory burden caused by performing inference
-                            // that is _too good_ in projects with complicated constraints (eg, fp-ts). In such cases, if we did not limit ourselves
-                            // here, we might produce more valid inferences for types, causing us to do more checks and perform more instantiations
-                            // (in addition to the extra stack depth here) which, in turn, can push the already close process over its limit.
-                            // TL;DR: If we ever become generally more memory efficient (or our resource budget ever increases), we should just
-                            // remove this `allowComplexConstraintInference` flag.
-                            allowComplexConstraintInference = false;
-                            return inferFromTypes(apparentSource, target);
+                    if (nonArrayLikeTargetParts !== neverType) {
+                        var child = validChildren[0];
+                        var elem_1 = getElaborationElementForJsxChild(child, childrenNameType, getInvalidTextualChildDiagnostic);
+                        if (elem_1) {
+                            result = elaborateElementwise((function () { return __generator(this, function (_a) {
+                                switch (_a.label) {
+                                    case 0: return [4 /*yield*/, elem_1];
+                                    case 1:
+                                        _a.sent();
+                                        return [2 /*return*/];
+                                }
+                            }); })(), source, target, relation, containingMessageChain, errorOutputContainer) || result;
                         }
-                        source = apparentSource;
-                    }
-                    if (source.flags & (524288 /* Object */ | 2097152 /* Intersection */)) {
-                        invokeOnce(source, target, inferFromObjectTypes);
-                    }
-                }
-                if (source.flags & 25165824 /* Simplifiable */) {
-                    var simplified = getSimplifiedType(source, contravariant);
-                    if (simplified !== source) {
-                        inferFromTypes(simplified, target);
                     }
-                }
-            }
-            function inferWithPriority(source, target, newPriority) {
-                var savePriority = priority;
-                priority |= newPriority;
-                inferFromTypes(source, target);
-                priority = savePriority;
-            }
-            function invokeOnce(source, target, action) {
-                var key = source.id + "," + target.id;
-                var status = visited && visited.get(key);
-                if (status !== undefined) {
-                    inferencePriority = Math.min(inferencePriority, status);
-                    return;
-                }
-                (visited || (visited = ts.createMap())).set(key, -1 /* Circularity */);
-                var saveInferencePriority = inferencePriority;
-                inferencePriority = 512 /* MaxValue */;
-                action(source, target);
-                visited.set(key, inferencePriority);
-                inferencePriority = Math.min(inferencePriority, saveInferencePriority);
-            }
-            function inferFromMatchingTypes(sources, targets, matches) {
-                var matchedSources;
-                var matchedTargets;
-                for (var _i = 0, targets_1 = targets; _i < targets_1.length; _i++) {
-                    var t = targets_1[_i];
-                    for (var _a = 0, sources_1 = sources; _a < sources_1.length; _a++) {
-                        var s = sources_1[_a];
-                        if (matches(s, t)) {
-                            inferFromTypes(s, t);
-                            matchedSources = ts.appendIfUnique(matchedSources, s);
-                            matchedTargets = ts.appendIfUnique(matchedTargets, t);
+                    else if (!isTypeRelatedTo(getIndexedAccessType(source, childrenNameType), childrenTargetType, relation)) {
+                        // arity mismatch
+                        result = true;
+                        var diag = error(containingElement.openingElement.tagName, ts.Diagnostics.This_JSX_tag_s_0_prop_expects_type_1_which_requires_multiple_children_but_only_a_single_child_was_provided, childrenPropName, typeToString(childrenTargetType));
+                        if (errorOutputContainer && errorOutputContainer.skipLogging) {
+                            (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
                         }
                     }
                 }
-                return [
-                    matchedSources ? ts.filter(sources, function (t) { return !ts.contains(matchedSources, t); }) : sources,
-                    matchedTargets ? ts.filter(targets, function (t) { return !ts.contains(matchedTargets, t); }) : targets,
-                ];
             }
-            function inferFromTypeArguments(sourceTypes, targetTypes, variances) {
-                var count = sourceTypes.length < targetTypes.length ? sourceTypes.length : targetTypes.length;
-                for (var i = 0; i < count; i++) {
-                    if (i < variances.length && (variances[i] & 7 /* VarianceMask */) === 2 /* Contravariant */) {
-                        inferFromContravariantTypes(sourceTypes[i], targetTypes[i]);
-                    }
-                    else {
-                        inferFromTypes(sourceTypes[i], targetTypes[i]);
-                    }
+            return result;
+            function getInvalidTextualChildDiagnostic() {
+                if (!invalidTextDiagnostic) {
+                    var tagNameText = ts.getTextOfNode(node.parent.tagName);
+                    var childPropName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(node));
+                    var childrenPropName = childPropName === undefined ? "children" : ts.unescapeLeadingUnderscores(childPropName);
+                    var childrenTargetType = getIndexedAccessType(target, getLiteralType(childrenPropName));
+                    var diagnostic = ts.Diagnostics._0_components_don_t_accept_text_as_child_elements_Text_in_JSX_has_the_type_string_but_the_expected_type_of_1_is_2;
+                    invalidTextDiagnostic = __assign(__assign({}, diagnostic), { key: "!!ALREADY FORMATTED!!", message: ts.formatMessage(/*_dummy*/ undefined, diagnostic, tagNameText, childrenPropName, typeToString(childrenTargetType)) });
                 }
+                return invalidTextDiagnostic;
             }
-            function inferFromContravariantTypes(source, target) {
-                if (strictFunctionTypes || priority & 256 /* AlwaysStrict */) {
-                    contravariant = !contravariant;
-                    inferFromTypes(source, target);
-                    contravariant = !contravariant;
-                }
-                else {
-                    inferFromTypes(source, target);
+        }
+        function generateLimitedTupleElements(node, target) {
+            var len, i, elem, nameType;
+            return __generator(this, function (_a) {
+                switch (_a.label) {
+                    case 0:
+                        len = ts.length(node.elements);
+                        if (!len)
+                            return [2 /*return*/];
+                        i = 0;
+                        _a.label = 1;
+                    case 1:
+                        if (!(i < len)) return [3 /*break*/, 4];
+                        // Skip elements which do not exist in the target - a length error on the tuple overall is likely better than an error on a mismatched index signature
+                        if (isTupleLikeType(target) && !getPropertyOfType(target, ("" + i)))
+                            return [3 /*break*/, 3];
+                        elem = node.elements[i];
+                        if (ts.isOmittedExpression(elem))
+                            return [3 /*break*/, 3];
+                        nameType = getLiteralType(i);
+                        return [4 /*yield*/, { errorNode: elem, innerExpression: elem, nameType: nameType }];
+                    case 2:
+                        _a.sent();
+                        _a.label = 3;
+                    case 3:
+                        i++;
+                        return [3 /*break*/, 1];
+                    case 4: return [2 /*return*/];
                 }
+            });
+        }
+        function elaborateArrayLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer) {
+            if (target.flags & 131068 /* Primitive */)
+                return false;
+            if (isTupleLikeType(source)) {
+                return elaborateElementwise(generateLimitedTupleElements(node, target), source, target, relation, containingMessageChain, errorOutputContainer);
             }
-            function getInferenceInfoForType(type) {
-                if (type.flags & 8650752 /* TypeVariable */) {
-                    for (var _i = 0, inferences_2 = inferences; _i < inferences_2.length; _i++) {
-                        var inference = inferences_2[_i];
-                        if (type === inference.typeParameter) {
-                            return inference;
-                        }
-                    }
+            // recreate a tuple from the elements, if possible
+            // Since we're re-doing the expression type, we need to reapply the contextual type
+            var oldContext = node.contextualType;
+            node.contextualType = target;
+            try {
+                var tupleizedType = checkArrayLiteral(node, 1 /* Contextual */, /*forceTuple*/ true);
+                node.contextualType = oldContext;
+                if (isTupleLikeType(tupleizedType)) {
+                    return elaborateElementwise(generateLimitedTupleElements(node, target), tupleizedType, target, relation, containingMessageChain, errorOutputContainer);
                 }
-                return undefined;
+                return false;
             }
-            function getSingleTypeVariableFromIntersectionTypes(types) {
-                var typeVariable;
-                for (var _i = 0, types_14 = types; _i < types_14.length; _i++) {
-                    var type = types_14[_i];
-                    var t = type.flags & 2097152 /* Intersection */ && ts.find(type.types, function (t) { return !!getInferenceInfoForType(t); });
-                    if (!t || typeVariable && t !== typeVariable) {
-                        return undefined;
-                    }
-                    typeVariable = t;
-                }
-                return typeVariable;
+            finally {
+                node.contextualType = oldContext;
             }
-            function inferToMultipleTypes(source, targets, targetFlags) {
-                var typeVariableCount = 0;
-                if (targetFlags & 1048576 /* Union */) {
-                    var nakedTypeVariable = void 0;
-                    var sources = source.flags & 1048576 /* Union */ ? source.types : [source];
-                    var matched_1 = new Array(sources.length);
-                    var inferenceCircularity = false;
-                    // First infer to types that are not naked type variables. For each source type we
-                    // track whether inferences were made from that particular type to some target with
-                    // equal priority (i.e. of equal quality) to what we would infer for a naked type
-                    // parameter.
-                    for (var _i = 0, targets_2 = targets; _i < targets_2.length; _i++) {
-                        var t = targets_2[_i];
-                        if (getInferenceInfoForType(t)) {
-                            nakedTypeVariable = t;
-                            typeVariableCount++;
-                        }
-                        else {
-                            for (var i = 0; i < sources.length; i++) {
-                                var saveInferencePriority = inferencePriority;
-                                inferencePriority = 512 /* MaxValue */;
-                                inferFromTypes(sources[i], t);
-                                if (inferencePriority === priority)
-                                    matched_1[i] = true;
-                                inferenceCircularity = inferenceCircularity || inferencePriority === -1 /* Circularity */;
-                                inferencePriority = Math.min(inferencePriority, saveInferencePriority);
-                            }
-                        }
-                    }
-                    if (typeVariableCount === 0) {
-                        // If every target is an intersection of types containing a single naked type variable,
-                        // make a lower priority inference to that type variable. This handles inferring from
-                        // 'A | B' to 'T & (X | Y)' where we want to infer 'A | B' for T.
-                        var intersectionTypeVariable = getSingleTypeVariableFromIntersectionTypes(targets);
-                        if (intersectionTypeVariable) {
-                            inferWithPriority(source, intersectionTypeVariable, 1 /* NakedTypeVariable */);
-                        }
-                        return;
-                    }
-                    // If the target has a single naked type variable and no inference circularities were
-                    // encountered above (meaning we explored the types fully), create a union of the source
-                    // types from which no inferences have been made so far and infer from that union to the
-                    // naked type variable.
-                    if (typeVariableCount === 1 && !inferenceCircularity) {
-                        var unmatched = ts.flatMap(sources, function (s, i) { return matched_1[i] ? undefined : s; });
-                        if (unmatched.length) {
-                            inferFromTypes(getUnionType(unmatched), nakedTypeVariable);
-                            return;
-                        }
-                    }
-                }
-                else {
-                    // We infer from types that are not naked type variables first so that inferences we
-                    // make from nested naked type variables and given slightly higher priority by virtue
-                    // of being first in the candidates array.
-                    for (var _a = 0, targets_3 = targets; _a < targets_3.length; _a++) {
-                        var t = targets_3[_a];
-                        if (getInferenceInfoForType(t)) {
-                            typeVariableCount++;
-                        }
-                        else {
-                            inferFromTypes(source, t);
-                        }
-                    }
-                }
-                // Inferences directly to naked type variables are given lower priority as they are
-                // less specific. For example, when inferring from Promise<string> to T | Promise<T>,
-                // we want to infer string for T, not Promise<string> | string. For intersection types
-                // we only infer to single naked type variables.
-                if (targetFlags & 2097152 /* Intersection */ ? typeVariableCount === 1 : typeVariableCount > 0) {
-                    for (var _b = 0, targets_4 = targets; _b < targets_4.length; _b++) {
-                        var t = targets_4[_b];
-                        if (getInferenceInfoForType(t)) {
-                            inferWithPriority(source, t, 1 /* NakedTypeVariable */);
+        }
+        function generateObjectLiteralElements(node) {
+            var _i, _a, prop, type, _b;
+            return __generator(this, function (_c) {
+                switch (_c.label) {
+                    case 0:
+                        if (!ts.length(node.properties))
+                            return [2 /*return*/];
+                        _i = 0, _a = node.properties;
+                        _c.label = 1;
+                    case 1:
+                        if (!(_i < _a.length)) return [3 /*break*/, 8];
+                        prop = _a[_i];
+                        if (ts.isSpreadAssignment(prop))
+                            return [3 /*break*/, 7];
+                        type = getLiteralTypeFromProperty(getSymbolOfNode(prop), 8576 /* StringOrNumberLiteralOrUnique */);
+                        if (!type || (type.flags & 131072 /* Never */)) {
+                            return [3 /*break*/, 7];
                         }
-                    }
-                }
-            }
-            function inferToMappedType(source, target, constraintType) {
-                if (constraintType.flags & 1048576 /* Union */) {
-                    var result = false;
-                    for (var _i = 0, _a = constraintType.types; _i < _a.length; _i++) {
-                        var type = _a[_i];
-                        result = inferToMappedType(source, target, type) || result;
-                    }
-                    return result;
-                }
-                if (constraintType.flags & 4194304 /* Index */) {
-                    // We're inferring from some source type S to a homomorphic mapped type { [P in keyof T]: X },
-                    // where T is a type variable. Use inferTypeForHomomorphicMappedType to infer a suitable source
-                    // type and then make a secondary inference from that type to T. We make a secondary inference
-                    // such that direct inferences to T get priority over inferences to Partial<T>, for example.
-                    var inference = getInferenceInfoForType(constraintType.type);
-                    if (inference && !inference.isFixed && !isFromInferenceBlockedSource(source)) {
-                        var inferredType = inferTypeForHomomorphicMappedType(source, target, constraintType);
-                        if (inferredType) {
-                            // We assign a lower priority to inferences made from types containing non-inferrable
-                            // types because we may only have a partial result (i.e. we may have failed to make
-                            // reverse inferences for some properties).
-                            inferWithPriority(inferredType, inference.typeParameter, ts.getObjectFlags(source) & 2097152 /* NonInferrableType */ ?
-                                4 /* PartialHomomorphicMappedType */ :
-                                2 /* HomomorphicMappedType */);
+                        _b = prop.kind;
+                        switch (_b) {
+                            case 168 /* SetAccessor */: return [3 /*break*/, 2];
+                            case 167 /* GetAccessor */: return [3 /*break*/, 2];
+                            case 165 /* MethodDeclaration */: return [3 /*break*/, 2];
+                            case 289 /* ShorthandPropertyAssignment */: return [3 /*break*/, 2];
+                            case 288 /* PropertyAssignment */: return [3 /*break*/, 4];
                         }
-                    }
-                    return true;
-                }
-                if (constraintType.flags & 262144 /* TypeParameter */) {
-                    // We're inferring from some source type S to a mapped type { [P in K]: X }, where K is a type
-                    // parameter. First infer from 'keyof S' to K.
-                    inferWithPriority(getIndexType(source), constraintType, 8 /* MappedTypeConstraint */);
-                    // If K is constrained to a type C, also infer to C. Thus, for a mapped type { [P in K]: X },
-                    // where K extends keyof T, we make the same inferences as for a homomorphic mapped type
-                    // { [P in keyof T]: X }. This enables us to make meaningful inferences when the target is a
-                    // Pick<T, K>.
-                    var extendedConstraint = getConstraintOfType(constraintType);
-                    if (extendedConstraint && inferToMappedType(source, target, extendedConstraint)) {
-                        return true;
-                    }
-                    // If no inferences can be made to K's constraint, infer from a union of the property types
-                    // in the source to the template type X.
-                    var propTypes = ts.map(getPropertiesOfType(source), getTypeOfSymbol);
-                    var stringIndexType = getIndexTypeOfType(source, 0 /* String */);
-                    var numberIndexInfo = getNonEnumNumberIndexInfo(source);
-                    var numberIndexType = numberIndexInfo && numberIndexInfo.type;
-                    inferFromTypes(getUnionType(ts.append(ts.append(propTypes, stringIndexType), numberIndexType)), getTemplateTypeFromMappedType(target));
-                    return true;
+                        return [3 /*break*/, 6];
+                    case 2: return [4 /*yield*/, { errorNode: prop.name, innerExpression: undefined, nameType: type }];
+                    case 3:
+                        _c.sent();
+                        return [3 /*break*/, 7];
+                    case 4: return [4 /*yield*/, { errorNode: prop.name, innerExpression: prop.initializer, nameType: type, errorMessage: ts.isComputedNonLiteralName(prop.name) ? ts.Diagnostics.Type_of_computed_property_s_value_is_0_which_is_not_assignable_to_type_1 : undefined }];
+                    case 5:
+                        _c.sent();
+                        return [3 /*break*/, 7];
+                    case 6:
+                        ts.Debug.assertNever(prop);
+                        _c.label = 7;
+                    case 7:
+                        _i++;
+                        return [3 /*break*/, 1];
+                    case 8: return [2 /*return*/];
                 }
+            });
+        }
+        function elaborateObjectLiteral(node, source, target, relation, containingMessageChain, errorOutputContainer) {
+            if (target.flags & 131068 /* Primitive */)
                 return false;
+            return elaborateElementwise(generateObjectLiteralElements(node), source, target, relation, containingMessageChain, errorOutputContainer);
+        }
+        /**
+         * This is *not* a bi-directional relationship.
+         * If one needs to check both directions for comparability, use a second call to this function or 'isTypeComparableTo'.
+         */
+        function checkTypeComparableTo(source, target, errorNode, headMessage, containingMessageChain) {
+            return checkTypeRelatedTo(source, target, comparableRelation, errorNode, headMessage, containingMessageChain);
+        }
+        function isSignatureAssignableTo(source, target, ignoreReturnTypes) {
+            return compareSignaturesRelated(source, target, ignoreReturnTypes ? 4 /* IgnoreReturnTypes */ : 0, /*reportErrors*/ false, 
+            /*errorReporter*/ undefined, /*errorReporter*/ undefined, compareTypesAssignable, /*reportUnreliableMarkers*/ undefined) !== 0 /* False */;
+        }
+        /**
+         * Returns true if `s` is `(...args: any[]) => any` or `(this: any, ...args: any[]) => any`
+         */
+        function isAnySignature(s) {
+            return !s.typeParameters && (!s.thisParameter || isTypeAny(getTypeOfParameter(s.thisParameter))) && s.parameters.length === 1 &&
+                signatureHasRestParameter(s) && (getTypeOfParameter(s.parameters[0]) === anyArrayType || isTypeAny(getTypeOfParameter(s.parameters[0]))) &&
+                isTypeAny(getReturnTypeOfSignature(s));
+        }
+        /**
+         * See signatureRelatedTo, compareSignaturesIdentical
+         */
+        function compareSignaturesRelated(source, target, checkMode, reportErrors, errorReporter, incompatibleErrorReporter, compareTypes, reportUnreliableMarkers) {
+            // TODO (drosen): De-duplicate code between related functions.
+            if (source === target) {
+                return -1 /* True */;
             }
-            function inferFromObjectTypes(source, target) {
-                // If we are already processing another target type with the same associated symbol (such as
-                // an instantiation of the same generic type), we do not explore this target as it would yield
-                // no further inferences. We exclude the static side of classes from this check since it shares
-                // its symbol with the instance side which would lead to false positives.
-                var isNonConstructorObject = target.flags & 524288 /* Object */ &&
-                    !(ts.getObjectFlags(target) & 16 /* Anonymous */ && target.symbol && target.symbol.flags & 32 /* Class */);
-                var symbolOrType = isNonConstructorObject ? isTupleType(target) ? target.target : target.symbol : undefined;
-                if (symbolOrType) {
-                    if (ts.contains(symbolOrTypeStack, symbolOrType)) {
-                        inferencePriority = -1 /* Circularity */;
-                        return;
-                    }
-                    (symbolOrTypeStack || (symbolOrTypeStack = [])).push(symbolOrType);
-                    inferFromObjectTypesWorker(source, target);
-                    symbolOrTypeStack.pop();
-                }
-                else {
-                    inferFromObjectTypesWorker(source, target);
-                }
-            }
-            function inferFromObjectTypesWorker(source, target) {
-                if (ts.getObjectFlags(source) & 4 /* Reference */ && ts.getObjectFlags(target) & 4 /* Reference */ && (source.target === target.target || isArrayType(source) && isArrayType(target))) {
-                    // If source and target are references to the same generic type, infer from type arguments
-                    inferFromTypeArguments(getTypeArguments(source), getTypeArguments(target), getVariances(source.target));
-                    return;
-                }
-                if (isGenericMappedType(source) && isGenericMappedType(target)) {
-                    // The source and target types are generic types { [P in S]: X } and { [P in T]: Y }, so we infer
-                    // from S to T and from X to Y.
-                    inferFromTypes(getConstraintTypeFromMappedType(source), getConstraintTypeFromMappedType(target));
-                    inferFromTypes(getTemplateTypeFromMappedType(source), getTemplateTypeFromMappedType(target));
-                }
-                if (ts.getObjectFlags(target) & 32 /* Mapped */) {
-                    var constraintType = getConstraintTypeFromMappedType(target);
-                    if (inferToMappedType(source, target, constraintType)) {
-                        return;
-                    }
-                }
-                // Infer from the members of source and target only if the two types are possibly related
-                if (!typesDefinitelyUnrelated(source, target)) {
-                    if (isArrayType(source) || isTupleType(source)) {
-                        if (isTupleType(target)) {
-                            var sourceLength = isTupleType(source) ? getLengthOfTupleType(source) : 0;
-                            var targetLength = getLengthOfTupleType(target);
-                            var sourceRestType = isTupleType(source) ? getRestTypeOfTupleType(source) : getElementTypeOfArrayType(source);
-                            var targetRestType = getRestTypeOfTupleType(target);
-                            var fixedLength = targetLength < sourceLength || sourceRestType ? targetLength : sourceLength;
-                            for (var i = 0; i < fixedLength; i++) {
-                                inferFromTypes(i < sourceLength ? getTypeArguments(source)[i] : sourceRestType, getTypeArguments(target)[i]);
-                            }
-                            if (targetRestType) {
-                                var types = fixedLength < sourceLength ? getTypeArguments(source).slice(fixedLength, sourceLength) : [];
-                                if (sourceRestType) {
-                                    types.push(sourceRestType);
-                                }
-                                if (types.length) {
-                                    inferFromTypes(getUnionType(types), targetRestType);
-                                }
-                            }
-                            return;
-                        }
-                        if (isArrayType(target)) {
-                            inferFromIndexTypes(source, target);
-                            return;
-                        }
-                    }
-                    inferFromProperties(source, target);
-                    inferFromSignatures(source, target, 0 /* Call */);
-                    inferFromSignatures(source, target, 1 /* Construct */);
-                    inferFromIndexTypes(source, target);
-                }
+            if (isAnySignature(target)) {
+                return -1 /* True */;
             }
-            function inferFromProperties(source, target) {
-                var properties = getPropertiesOfObjectType(target);
-                for (var _i = 0, properties_3 = properties; _i < properties_3.length; _i++) {
-                    var targetProp = properties_3[_i];
-                    var sourceProp = getPropertyOfType(source, targetProp.escapedName);
-                    if (sourceProp) {
-                        inferFromTypes(getTypeOfSymbol(sourceProp), getTypeOfSymbol(targetProp));
-                    }
-                }
+            var targetCount = getParameterCount(target);
+            var sourceHasMoreParameters = !hasEffectiveRestParameter(target) &&
+                (checkMode & 8 /* StrictArity */ ? hasEffectiveRestParameter(source) || getParameterCount(source) > targetCount : getMinArgumentCount(source) > targetCount);
+            if (sourceHasMoreParameters) {
+                return 0 /* False */;
             }
-            function inferFromSignatures(source, target, kind) {
-                var sourceSignatures = getSignaturesOfType(source, kind);
-                var targetSignatures = getSignaturesOfType(target, kind);
-                var sourceLen = sourceSignatures.length;
-                var targetLen = targetSignatures.length;
-                var len = sourceLen < targetLen ? sourceLen : targetLen;
-                var skipParameters = !!(ts.getObjectFlags(source) & 2097152 /* NonInferrableType */);
-                for (var i = 0; i < len; i++) {
-                    inferFromSignature(getBaseSignature(sourceSignatures[sourceLen - len + i]), getErasedSignature(targetSignatures[targetLen - len + i]), skipParameters);
-                }
+            if (source.typeParameters && source.typeParameters !== target.typeParameters) {
+                target = getCanonicalSignature(target);
+                source = instantiateSignatureInContextOf(source, target, /*inferenceContext*/ undefined, compareTypes);
             }
-            function inferFromSignature(source, target, skipParameters) {
-                if (!skipParameters) {
-                    var saveBivariant = bivariant;
-                    var kind = target.declaration ? target.declaration.kind : 0 /* Unknown */;
-                    // Once we descend into a bivariant signature we remain bivariant for all nested inferences
-                    bivariant = bivariant || kind === 161 /* MethodDeclaration */ || kind === 160 /* MethodSignature */ || kind === 162 /* Constructor */;
-                    applyToParameterTypes(source, target, inferFromContravariantTypes);
-                    bivariant = saveBivariant;
-                }
-                applyToReturnTypes(source, target, inferFromTypes);
+            var sourceCount = getParameterCount(source);
+            var sourceRestType = getNonArrayRestType(source);
+            var targetRestType = getNonArrayRestType(target);
+            if (sourceRestType || targetRestType) {
+                void instantiateType(sourceRestType || targetRestType, reportUnreliableMarkers);
             }
-            function inferFromIndexTypes(source, target) {
-                var targetStringIndexType = getIndexTypeOfType(target, 0 /* String */);
-                if (targetStringIndexType) {
-                    var sourceIndexType = getIndexTypeOfType(source, 0 /* String */) ||
-                        getImplicitIndexTypeOfType(source, 0 /* String */);
-                    if (sourceIndexType) {
-                        inferFromTypes(sourceIndexType, targetStringIndexType);
+            if (sourceRestType && targetRestType && sourceCount !== targetCount) {
+                // We're not able to relate misaligned complex rest parameters
+                return 0 /* False */;
+            }
+            var kind = target.declaration ? target.declaration.kind : 0 /* Unknown */;
+            var strictVariance = !(checkMode & 3 /* Callback */) && strictFunctionTypes && kind !== 165 /* MethodDeclaration */ &&
+                kind !== 164 /* MethodSignature */ && kind !== 166 /* Constructor */;
+            var result = -1 /* True */;
+            var sourceThisType = getThisTypeOfSignature(source);
+            if (sourceThisType && sourceThisType !== voidType) {
+                var targetThisType = getThisTypeOfSignature(target);
+                if (targetThisType) {
+                    // void sources are assignable to anything.
+                    var related = !strictVariance && compareTypes(sourceThisType, targetThisType, /*reportErrors*/ false)
+                        || compareTypes(targetThisType, sourceThisType, reportErrors);
+                    if (!related) {
+                        if (reportErrors) {
+                            errorReporter(ts.Diagnostics.The_this_types_of_each_signature_are_incompatible);
+                        }
+                        return 0 /* False */;
                     }
+                    result &= related;
                 }
-                var targetNumberIndexType = getIndexTypeOfType(target, 1 /* Number */);
-                if (targetNumberIndexType) {
-                    var sourceIndexType = getIndexTypeOfType(source, 1 /* Number */) ||
-                        getIndexTypeOfType(source, 0 /* String */) ||
-                        getImplicitIndexTypeOfType(source, 1 /* Number */);
-                    if (sourceIndexType) {
-                        inferFromTypes(sourceIndexType, targetNumberIndexType);
+            }
+            var paramCount = sourceRestType || targetRestType ? Math.min(sourceCount, targetCount) : Math.max(sourceCount, targetCount);
+            var restIndex = sourceRestType || targetRestType ? paramCount - 1 : -1;
+            for (var i = 0; i < paramCount; i++) {
+                var sourceType = i === restIndex ? getRestTypeAtPosition(source, i) : tryGetTypeAtPosition(source, i);
+                var targetType = i === restIndex ? getRestTypeAtPosition(target, i) : tryGetTypeAtPosition(target, i);
+                if (sourceType && targetType) {
+                    // In order to ensure that any generic type Foo<T> is at least co-variant with respect to T no matter
+                    // how Foo uses T, we need to relate parameters bi-variantly (given that parameters are input positions,
+                    // they naturally relate only contra-variantly). However, if the source and target parameters both have
+                    // function types with a single call signature, we know we are relating two callback parameters. In
+                    // that case it is sufficient to only relate the parameters of the signatures co-variantly because,
+                    // similar to return values, callback parameters are output positions. This means that a Promise<T>,
+                    // where T is used only in callback parameter positions, will be co-variant (as opposed to bi-variant)
+                    // with respect to T.
+                    var sourceSig = checkMode & 3 /* Callback */ ? undefined : getSingleCallSignature(getNonNullableType(sourceType));
+                    var targetSig = checkMode & 3 /* Callback */ ? undefined : getSingleCallSignature(getNonNullableType(targetType));
+                    var callbacks = sourceSig && targetSig && !getTypePredicateOfSignature(sourceSig) && !getTypePredicateOfSignature(targetSig) &&
+                        (getFalsyFlags(sourceType) & 98304 /* Nullable */) === (getFalsyFlags(targetType) & 98304 /* Nullable */);
+                    var related = callbacks ?
+                        compareSignaturesRelated(targetSig, sourceSig, (checkMode & 8 /* StrictArity */) | (strictVariance ? 2 /* StrictCallback */ : 1 /* BivariantCallback */), reportErrors, errorReporter, incompatibleErrorReporter, compareTypes, reportUnreliableMarkers) :
+                        !(checkMode & 3 /* Callback */) && !strictVariance && compareTypes(sourceType, targetType, /*reportErrors*/ false) || compareTypes(targetType, sourceType, reportErrors);
+                    // With strict arity, (x: number | undefined) => void is a subtype of (x?: number | undefined) => void
+                    if (related && checkMode & 8 /* StrictArity */ && i >= getMinArgumentCount(source) && i < getMinArgumentCount(target) && compareTypes(sourceType, targetType, /*reportErrors*/ false)) {
+                        related = 0 /* False */;
+                    }
+                    if (!related) {
+                        if (reportErrors) {
+                            errorReporter(ts.Diagnostics.Types_of_parameters_0_and_1_are_incompatible, ts.unescapeLeadingUnderscores(getParameterNameAtPosition(source, i)), ts.unescapeLeadingUnderscores(getParameterNameAtPosition(target, i)));
+                        }
+                        return 0 /* False */;
                     }
+                    result &= related;
                 }
             }
-        }
-        function isTypeOrBaseIdenticalTo(s, t) {
-            return isTypeIdenticalTo(s, t) || !!(t.flags & 4 /* String */ && s.flags & 128 /* StringLiteral */ || t.flags & 8 /* Number */ && s.flags & 256 /* NumberLiteral */);
-        }
-        function isTypeCloselyMatchedBy(s, t) {
-            return !!(s.flags & 524288 /* Object */ && t.flags & 524288 /* Object */ && s.symbol && s.symbol === t.symbol ||
-                s.aliasSymbol && s.aliasTypeArguments && s.aliasSymbol === t.aliasSymbol);
-        }
-        function hasPrimitiveConstraint(type) {
-            var constraint = getConstraintOfTypeParameter(type);
-            return !!constraint && maybeTypeOfKind(constraint.flags & 16777216 /* Conditional */ ? getDefaultConstraintOfConditionalType(constraint) : constraint, 131068 /* Primitive */ | 4194304 /* Index */);
-        }
-        function isObjectLiteralType(type) {
-            return !!(ts.getObjectFlags(type) & 128 /* ObjectLiteral */);
-        }
-        function isObjectOrArrayLiteralType(type) {
-            return !!(ts.getObjectFlags(type) & (128 /* ObjectLiteral */ | 65536 /* ArrayLiteral */));
-        }
-        function unionObjectAndArrayLiteralCandidates(candidates) {
-            if (candidates.length > 1) {
-                var objectLiterals = ts.filter(candidates, isObjectOrArrayLiteralType);
-                if (objectLiterals.length) {
-                    var literalsType = getUnionType(objectLiterals, 2 /* Subtype */);
-                    return ts.concatenate(ts.filter(candidates, function (t) { return !isObjectOrArrayLiteralType(t); }), [literalsType]);
+            if (!(checkMode & 4 /* IgnoreReturnTypes */)) {
+                // If a signature resolution is already in-flight, skip issuing a circularity error
+                // here and just use the `any` type directly
+                var targetReturnType = isResolvingReturnTypeOfSignature(target) ? anyType
+                    : target.declaration && isJSConstructor(target.declaration) ? getDeclaredTypeOfClassOrInterface(getMergedSymbol(target.declaration.symbol))
+                        : getReturnTypeOfSignature(target);
+                if (targetReturnType === voidType) {
+                    return result;
                 }
-            }
-            return candidates;
-        }
-        function getContravariantInference(inference) {
-            return inference.priority & 104 /* PriorityImpliesCombination */ ? getIntersectionType(inference.contraCandidates) : getCommonSubtype(inference.contraCandidates);
-        }
-        function getCovariantInference(inference, signature) {
-            // Extract all object and array literal types and replace them with a single widened and normalized type.
-            var candidates = unionObjectAndArrayLiteralCandidates(inference.candidates);
-            // We widen inferred literal types if
-            // all inferences were made to top-level occurrences of the type parameter, and
-            // the type parameter has no constraint or its constraint includes no primitive or literal types, and
-            // the type parameter was fixed during inference or does not occur at top-level in the return type.
-            var primitiveConstraint = hasPrimitiveConstraint(inference.typeParameter);
-            var widenLiteralTypes = !primitiveConstraint && inference.topLevel &&
-                (inference.isFixed || !isTypeParameterAtTopLevel(getReturnTypeOfSignature(signature), inference.typeParameter));
-            var baseCandidates = primitiveConstraint ? ts.sameMap(candidates, getRegularTypeOfLiteralType) :
-                widenLiteralTypes ? ts.sameMap(candidates, getWidenedLiteralType) :
-                    candidates;
-            // If all inferences were made from a position that implies a combined result, infer a union type.
-            // Otherwise, infer a common supertype.
-            var unwidenedType = inference.priority & 104 /* PriorityImpliesCombination */ ?
-                getUnionType(baseCandidates, 2 /* Subtype */) :
-                getCommonSupertype(baseCandidates);
-            return getWidenedType(unwidenedType);
-        }
-        function getInferredType(context, index) {
-            var inference = context.inferences[index];
-            if (!inference.inferredType) {
-                var inferredType = void 0;
-                var signature = context.signature;
-                if (signature) {
-                    var inferredCovariantType = inference.candidates ? getCovariantInference(inference, signature) : undefined;
-                    if (inference.contraCandidates) {
-                        var inferredContravariantType = getContravariantInference(inference);
-                        // If we have both co- and contra-variant inferences, we prefer the contra-variant inference
-                        // unless the co-variant inference is a subtype and not 'never'.
-                        inferredType = inferredCovariantType && !(inferredCovariantType.flags & 131072 /* Never */) &&
-                            isTypeSubtypeOf(inferredCovariantType, inferredContravariantType) ?
-                            inferredCovariantType : inferredContravariantType;
-                    }
-                    else if (inferredCovariantType) {
-                        inferredType = inferredCovariantType;
-                    }
-                    else if (context.flags & 1 /* NoDefault */) {
-                        // We use silentNeverType as the wildcard that signals no inferences.
-                        inferredType = silentNeverType;
+                var sourceReturnType = isResolvingReturnTypeOfSignature(source) ? anyType
+                    : source.declaration && isJSConstructor(source.declaration) ? getDeclaredTypeOfClassOrInterface(getMergedSymbol(source.declaration.symbol))
+                        : getReturnTypeOfSignature(source);
+                // The following block preserves behavior forbidding boolean returning functions from being assignable to type guard returning functions
+                var targetTypePredicate = getTypePredicateOfSignature(target);
+                if (targetTypePredicate) {
+                    var sourceTypePredicate = getTypePredicateOfSignature(source);
+                    if (sourceTypePredicate) {
+                        result &= compareTypePredicateRelatedTo(sourceTypePredicate, targetTypePredicate, reportErrors, errorReporter, compareTypes);
                     }
-                    else {
-                        // Infer either the default or the empty object type when no inferences were
-                        // made. It is important to remember that in this case, inference still
-                        // succeeds, meaning there is no error for not having inference candidates. An
-                        // inference error only occurs when there are *conflicting* candidates, i.e.
-                        // candidates with no common supertype.
-                        var defaultType = getDefaultFromTypeParameter(inference.typeParameter);
-                        if (defaultType) {
-                            // Instantiate the default type. Any forward reference to a type
-                            // parameter should be instantiated to the empty object type.
-                            inferredType = instantiateType(defaultType, mergeTypeMappers(createBackreferenceMapper(context, index), context.nonFixingMapper));
+                    else if (ts.isIdentifierTypePredicate(targetTypePredicate)) {
+                        if (reportErrors) {
+                            errorReporter(ts.Diagnostics.Signature_0_must_be_a_type_predicate, signatureToString(source));
                         }
+                        return 0 /* False */;
                     }
                 }
                 else {
-                    inferredType = getTypeFromInference(inference);
-                }
-                inference.inferredType = inferredType || getDefaultTypeArgumentType(!!(context.flags & 2 /* AnyDefault */));
-                var constraint = getConstraintOfTypeParameter(inference.typeParameter);
-                if (constraint) {
-                    var instantiatedConstraint = instantiateType(constraint, context.nonFixingMapper);
-                    if (!inferredType || !context.compareTypes(inferredType, getTypeWithThisArgument(instantiatedConstraint, inferredType))) {
-                        inference.inferredType = inferredType = instantiatedConstraint;
+                    // When relating callback signatures, we still need to relate return types bi-variantly as otherwise
+                    // the containing type wouldn't be co-variant. For example, interface Foo<T> { add(cb: () => T): void }
+                    // wouldn't be co-variant for T without this rule.
+                    result &= checkMode & 1 /* BivariantCallback */ && compareTypes(targetReturnType, sourceReturnType, /*reportErrors*/ false) ||
+                        compareTypes(sourceReturnType, targetReturnType, reportErrors);
+                    if (!result && reportErrors && incompatibleErrorReporter) {
+                        incompatibleErrorReporter(sourceReturnType, targetReturnType);
                     }
                 }
             }
-            return inference.inferredType;
-        }
-        function getDefaultTypeArgumentType(isInJavaScriptFile) {
-            return isInJavaScriptFile ? anyType : unknownType;
-        }
-        function getInferredTypes(context) {
-            var result = [];
-            for (var i = 0; i < context.inferences.length; i++) {
-                result.push(getInferredType(context, i));
-            }
             return result;
         }
-        // EXPRESSION TYPE CHECKING
-        function getCannotFindNameDiagnosticForName(node) {
-            switch (node.escapedText) {
-                case "document":
-                case "console":
-                    return ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_include_dom;
-                case "$":
-                    return compilerOptions.types
-                        ? ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_types_Slashjquery_and_then_add_jquery_to_the_types_field_in_your_tsconfig
-                        : ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_types_Slashjquery;
-                case "describe":
-                case "suite":
-                case "it":
-                case "test":
-                    return compilerOptions.types
-                        ? ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_types_Slashjest_or_npm_i_types_Slashmocha_and_then_add_jest_or_mocha_to_the_types_field_in_your_tsconfig
-                        : ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_types_Slashjest_or_npm_i_types_Slashmocha;
-                case "process":
-                case "require":
-                case "Buffer":
-                case "module":
-                    return compilerOptions.types
-                        ? ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_types_Slashnode_and_then_add_node_to_the_types_field_in_your_tsconfig
-                        : ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_types_Slashnode;
-                case "Map":
-                case "Set":
-                case "Promise":
-                case "Symbol":
-                case "WeakMap":
-                case "WeakSet":
-                case "Iterator":
-                case "AsyncIterator":
-                    return ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_es2015_or_later;
-                default:
-                    if (node.parent.kind === 282 /* ShorthandPropertyAssignment */) {
-                        return ts.Diagnostics.No_value_exists_in_scope_for_the_shorthand_property_0_Either_declare_one_or_provide_an_initializer;
-                    }
-                    else {
-                        return ts.Diagnostics.Cannot_find_name_0;
-                    }
-            }
-        }
-        function getResolvedSymbol(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedSymbol) {
-                links.resolvedSymbol = !ts.nodeIsMissing(node) &&
-                    resolveName(node, node.escapedText, 111551 /* Value */ | 1048576 /* ExportValue */, getCannotFindNameDiagnosticForName(node), node, !ts.isWriteOnlyAccess(node), 
-                    /*excludeGlobals*/ false, ts.Diagnostics.Cannot_find_name_0_Did_you_mean_1) || unknownSymbol;
+        function compareTypePredicateRelatedTo(source, target, reportErrors, errorReporter, compareTypes) {
+            if (source.kind !== target.kind) {
+                if (reportErrors) {
+                    errorReporter(ts.Diagnostics.A_this_based_type_guard_is_not_compatible_with_a_parameter_based_type_guard);
+                    errorReporter(ts.Diagnostics.Type_predicate_0_is_not_assignable_to_1, typePredicateToString(source), typePredicateToString(target));
+                }
+                return 0 /* False */;
             }
-            return links.resolvedSymbol;
-        }
-        function isInTypeQuery(node) {
-            // TypeScript 1.0 spec (April 2014): 3.6.3
-            // A type query consists of the keyword typeof followed by an expression.
-            // The expression is restricted to a single identifier or a sequence of identifiers separated by periods
-            return !!ts.findAncestor(node, function (n) { return n.kind === 172 /* TypeQuery */ ? true : n.kind === 75 /* Identifier */ || n.kind === 153 /* QualifiedName */ ? false : "quit"; });
-        }
-        // Return the flow cache key for a "dotted name" (i.e. a sequence of identifiers
-        // separated by dots). The key consists of the id of the symbol referenced by the
-        // leftmost identifier followed by zero or more property names separated by dots.
-        // The result is undefined if the reference isn't a dotted name. We prefix nodes
-        // occurring in an apparent type position with '@' because the control flow type
-        // of such nodes may be based on the apparent type instead of the declared type.
-        function getFlowCacheKey(node, declaredType, initialType, flowContainer) {
-            switch (node.kind) {
-                case 75 /* Identifier */:
-                    var symbol = getResolvedSymbol(node);
-                    return symbol !== unknownSymbol ? (flowContainer ? getNodeId(flowContainer) : "-1") + "|" + getTypeId(declaredType) + "|" + getTypeId(initialType) + "|" + (isConstraintPosition(node) ? "@" : "") + getSymbolId(symbol) : undefined;
-                case 104 /* ThisKeyword */:
-                    return "0";
-                case 218 /* NonNullExpression */:
-                case 200 /* ParenthesizedExpression */:
-                    return getFlowCacheKey(node.expression, declaredType, initialType, flowContainer);
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    var propName = getAccessedPropertyName(node);
-                    if (propName !== undefined) {
-                        var key = getFlowCacheKey(node.expression, declaredType, initialType, flowContainer);
-                        return key && key + "." + propName;
+            if (source.kind === 1 /* Identifier */ || source.kind === 3 /* AssertsIdentifier */) {
+                if (source.parameterIndex !== target.parameterIndex) {
+                    if (reportErrors) {
+                        errorReporter(ts.Diagnostics.Parameter_0_is_not_in_the_same_position_as_parameter_1, source.parameterName, target.parameterName);
+                        errorReporter(ts.Diagnostics.Type_predicate_0_is_not_assignable_to_1, typePredicateToString(source), typePredicateToString(target));
                     }
+                    return 0 /* False */;
+                }
             }
-            return undefined;
-        }
-        function isMatchingReference(source, target) {
-            switch (target.kind) {
-                case 200 /* ParenthesizedExpression */:
-                case 218 /* NonNullExpression */:
-                    return isMatchingReference(source, target.expression);
+            var related = source.type === target.type ? -1 /* True */ :
+                source.type && target.type ? compareTypes(source.type, target.type, reportErrors) :
+                    0 /* False */;
+            if (related === 0 /* False */ && reportErrors) {
+                errorReporter(ts.Diagnostics.Type_predicate_0_is_not_assignable_to_1, typePredicateToString(source), typePredicateToString(target));
             }
-            switch (source.kind) {
-                case 75 /* Identifier */:
-                    return target.kind === 75 /* Identifier */ && getResolvedSymbol(source) === getResolvedSymbol(target) ||
-                        (target.kind === 242 /* VariableDeclaration */ || target.kind === 191 /* BindingElement */) &&
-                            getExportSymbolOfValueSymbolIfExported(getResolvedSymbol(source)) === getSymbolOfNode(target);
-                case 104 /* ThisKeyword */:
-                    return target.kind === 104 /* ThisKeyword */;
-                case 102 /* SuperKeyword */:
-                    return target.kind === 102 /* SuperKeyword */;
-                case 218 /* NonNullExpression */:
-                case 200 /* ParenthesizedExpression */:
-                    return isMatchingReference(source.expression, target);
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    return ts.isAccessExpression(target) &&
-                        getAccessedPropertyName(source) === getAccessedPropertyName(target) &&
-                        isMatchingReference(source.expression, target.expression);
+            return related;
+        }
+        function isImplementationCompatibleWithOverload(implementation, overload) {
+            var erasedSource = getErasedSignature(implementation);
+            var erasedTarget = getErasedSignature(overload);
+            // First see if the return types are compatible in either direction.
+            var sourceReturnType = getReturnTypeOfSignature(erasedSource);
+            var targetReturnType = getReturnTypeOfSignature(erasedTarget);
+            if (targetReturnType === voidType
+                || isTypeRelatedTo(targetReturnType, sourceReturnType, assignableRelation)
+                || isTypeRelatedTo(sourceReturnType, targetReturnType, assignableRelation)) {
+                return isSignatureAssignableTo(erasedSource, erasedTarget, /*ignoreReturnTypes*/ true);
             }
             return false;
         }
-        // Given a source x, check if target matches x or is an && operation with an operand that matches x.
-        function containsTruthyCheck(source, target) {
-            return isMatchingReference(source, target) ||
-                (target.kind === 209 /* BinaryExpression */ && target.operatorToken.kind === 55 /* AmpersandAmpersandToken */ &&
-                    (containsTruthyCheck(source, target.left) || containsTruthyCheck(source, target.right)));
+        function isEmptyResolvedType(t) {
+            return t !== anyFunctionType &&
+                t.properties.length === 0 &&
+                t.callSignatures.length === 0 &&
+                t.constructSignatures.length === 0 &&
+                !t.stringIndexInfo &&
+                !t.numberIndexInfo;
         }
-        function getAccessedPropertyName(access) {
-            return access.kind === 194 /* PropertyAccessExpression */ ? access.name.escapedText :
-                ts.isStringOrNumericLiteralLike(access.argumentExpression) ? ts.escapeLeadingUnderscores(access.argumentExpression.text) :
-                    undefined;
+        function isEmptyObjectType(type) {
+            return type.flags & 524288 /* Object */ ? !isGenericMappedType(type) && isEmptyResolvedType(resolveStructuredTypeMembers(type)) :
+                type.flags & 67108864 /* NonPrimitive */ ? true :
+                    type.flags & 1048576 /* Union */ ? ts.some(type.types, isEmptyObjectType) :
+                        type.flags & 2097152 /* Intersection */ ? ts.every(type.types, isEmptyObjectType) :
+                            false;
         }
-        function containsMatchingReference(source, target) {
-            while (ts.isAccessExpression(source)) {
-                source = source.expression;
-                if (isMatchingReference(source, target)) {
-                    return true;
-                }
-            }
-            return false;
+        function isEmptyAnonymousObjectType(type) {
+            return !!(ts.getObjectFlags(type) & 16 /* Anonymous */ && (type.members && isEmptyResolvedType(type) ||
+                type.symbol && type.symbol.flags & 2048 /* TypeLiteral */ && getMembersOfSymbol(type.symbol).size === 0));
         }
-        function optionalChainContainsReference(source, target) {
-            while (ts.isOptionalChain(source)) {
-                source = source.expression;
-                if (isMatchingReference(source, target)) {
-                    return true;
-                }
-            }
-            return false;
+        function isStringIndexSignatureOnlyType(type) {
+            return type.flags & 524288 /* Object */ && !isGenericMappedType(type) && getPropertiesOfType(type).length === 0 && getIndexInfoOfType(type, 0 /* String */) && !getIndexInfoOfType(type, 1 /* Number */) ||
+                type.flags & 3145728 /* UnionOrIntersection */ && ts.every(type.types, isStringIndexSignatureOnlyType) ||
+                false;
         }
-        function isDiscriminantProperty(type, name) {
-            if (type && type.flags & 1048576 /* Union */) {
-                var prop = getUnionOrIntersectionProperty(type, name);
-                if (prop && ts.getCheckFlags(prop) & 2 /* SyntheticProperty */) {
-                    if (prop.isDiscriminantProperty === undefined) {
-                        prop.isDiscriminantProperty =
-                            (prop.checkFlags & 192 /* Discriminant */) === 192 /* Discriminant */ &&
-                                !maybeTypeOfKind(getTypeOfSymbol(prop), 63176704 /* Instantiable */);
-                    }
-                    return !!prop.isDiscriminantProperty;
-                }
+        function isEnumTypeRelatedTo(sourceSymbol, targetSymbol, errorReporter) {
+            if (sourceSymbol === targetSymbol) {
+                return true;
             }
-            return false;
-        }
-        function findDiscriminantProperties(sourceProperties, target) {
-            var result;
-            for (var _i = 0, sourceProperties_2 = sourceProperties; _i < sourceProperties_2.length; _i++) {
-                var sourceProperty = sourceProperties_2[_i];
-                if (isDiscriminantProperty(target, sourceProperty.escapedName)) {
-                    if (result) {
-                        result.push(sourceProperty);
-                        continue;
+            var id = getSymbolId(sourceSymbol) + "," + getSymbolId(targetSymbol);
+            var entry = enumRelation.get(id);
+            if (entry !== undefined && !(!(entry & 4 /* Reported */) && entry & 2 /* Failed */ && errorReporter)) {
+                return !!(entry & 1 /* Succeeded */);
+            }
+            if (sourceSymbol.escapedName !== targetSymbol.escapedName || !(sourceSymbol.flags & 256 /* RegularEnum */) || !(targetSymbol.flags & 256 /* RegularEnum */)) {
+                enumRelation.set(id, 2 /* Failed */ | 4 /* Reported */);
+                return false;
+            }
+            var targetEnumType = getTypeOfSymbol(targetSymbol);
+            for (var _i = 0, _a = getPropertiesOfType(getTypeOfSymbol(sourceSymbol)); _i < _a.length; _i++) {
+                var property = _a[_i];
+                if (property.flags & 8 /* EnumMember */) {
+                    var targetProperty = getPropertyOfType(targetEnumType, property.escapedName);
+                    if (!targetProperty || !(targetProperty.flags & 8 /* EnumMember */)) {
+                        if (errorReporter) {
+                            errorReporter(ts.Diagnostics.Property_0_is_missing_in_type_1, ts.symbolName(property), typeToString(getDeclaredTypeOfSymbol(targetSymbol), /*enclosingDeclaration*/ undefined, 64 /* UseFullyQualifiedType */));
+                            enumRelation.set(id, 2 /* Failed */ | 4 /* Reported */);
+                        }
+                        else {
+                            enumRelation.set(id, 2 /* Failed */);
+                        }
+                        return false;
                     }
-                    result = [sourceProperty];
                 }
             }
-            return result;
-        }
-        function isOrContainsMatchingReference(source, target) {
-            return isMatchingReference(source, target) || containsMatchingReference(source, target);
+            enumRelation.set(id, 1 /* Succeeded */);
+            return true;
         }
-        function hasMatchingArgument(callExpression, reference) {
-            if (callExpression.arguments) {
-                for (var _i = 0, _a = callExpression.arguments; _i < _a.length; _i++) {
-                    var argument = _a[_i];
-                    if (isOrContainsMatchingReference(reference, argument)) {
-                        return true;
-                    }
-                }
+        function isSimpleTypeRelatedTo(source, target, relation, errorReporter) {
+            var s = source.flags;
+            var t = target.flags;
+            if (t & 3 /* AnyOrUnknown */ || s & 131072 /* Never */ || source === wildcardType)
+                return true;
+            if (t & 131072 /* Never */)
+                return false;
+            if (s & 402653316 /* StringLike */ && t & 4 /* String */)
+                return true;
+            if (s & 128 /* StringLiteral */ && s & 1024 /* EnumLiteral */ &&
+                t & 128 /* StringLiteral */ && !(t & 1024 /* EnumLiteral */) &&
+                source.value === target.value)
+                return true;
+            if (s & 296 /* NumberLike */ && t & 8 /* Number */)
+                return true;
+            if (s & 256 /* NumberLiteral */ && s & 1024 /* EnumLiteral */ &&
+                t & 256 /* NumberLiteral */ && !(t & 1024 /* EnumLiteral */) &&
+                source.value === target.value)
+                return true;
+            if (s & 2112 /* BigIntLike */ && t & 64 /* BigInt */)
+                return true;
+            if (s & 528 /* BooleanLike */ && t & 16 /* Boolean */)
+                return true;
+            if (s & 12288 /* ESSymbolLike */ && t & 4096 /* ESSymbol */)
+                return true;
+            if (s & 32 /* Enum */ && t & 32 /* Enum */ && isEnumTypeRelatedTo(source.symbol, target.symbol, errorReporter))
+                return true;
+            if (s & 1024 /* EnumLiteral */ && t & 1024 /* EnumLiteral */) {
+                if (s & 1048576 /* Union */ && t & 1048576 /* Union */ && isEnumTypeRelatedTo(source.symbol, target.symbol, errorReporter))
+                    return true;
+                if (s & 2944 /* Literal */ && t & 2944 /* Literal */ &&
+                    source.value === target.value &&
+                    isEnumTypeRelatedTo(getParentOfSymbol(source.symbol), getParentOfSymbol(target.symbol), errorReporter))
+                    return true;
             }
-            if (callExpression.expression.kind === 194 /* PropertyAccessExpression */ &&
-                isOrContainsMatchingReference(reference, callExpression.expression.expression)) {
+            if (s & 32768 /* Undefined */ && (!strictNullChecks || t & (32768 /* Undefined */ | 16384 /* Void */)))
+                return true;
+            if (s & 65536 /* Null */ && (!strictNullChecks || t & 65536 /* Null */))
+                return true;
+            if (s & 524288 /* Object */ && t & 67108864 /* NonPrimitive */)
                 return true;
+            if (relation === assignableRelation || relation === comparableRelation) {
+                if (s & 1 /* Any */)
+                    return true;
+                // Type number or any numeric literal type is assignable to any numeric enum type or any
+                // numeric enum literal type. This rule exists for backwards compatibility reasons because
+                // bit-flag enum types sometimes look like literal enum types with numeric literal values.
+                if (s & (8 /* Number */ | 256 /* NumberLiteral */) && !(s & 1024 /* EnumLiteral */) && (t & 32 /* Enum */ || t & 256 /* NumberLiteral */ && t & 1024 /* EnumLiteral */))
+                    return true;
             }
             return false;
         }
-        function getFlowNodeId(flow) {
-            if (!flow.id || flow.id < 0) {
-                flow.id = nextFlowId;
-                nextFlowId++;
+        function isTypeRelatedTo(source, target, relation) {
+            if (isFreshLiteralType(source)) {
+                source = source.regularType;
             }
-            return flow.id;
-        }
-        function typeMaybeAssignableTo(source, target) {
-            if (!(source.flags & 1048576 /* Union */)) {
-                return isTypeAssignableTo(source, target);
+            if (isFreshLiteralType(target)) {
+                target = target.regularType;
             }
-            for (var _i = 0, _a = source.types; _i < _a.length; _i++) {
-                var t = _a[_i];
-                if (isTypeAssignableTo(t, target)) {
+            if (source === target) {
+                return true;
+            }
+            if (relation !== identityRelation) {
+                if (relation === comparableRelation && !(target.flags & 131072 /* Never */) && isSimpleTypeRelatedTo(target, source, relation) || isSimpleTypeRelatedTo(source, target, relation)) {
                     return true;
                 }
             }
-            return false;
-        }
-        // Remove those constituent types of declaredType to which no constituent type of assignedType is assignable.
-        // For example, when a variable of type number | string | boolean is assigned a value of type number | boolean,
-        // we remove type string.
-        function getAssignmentReducedType(declaredType, assignedType) {
-            if (declaredType !== assignedType) {
-                if (assignedType.flags & 131072 /* Never */) {
-                    return assignedType;
-                }
-                var reducedType = filterType(declaredType, function (t) { return typeMaybeAssignableTo(assignedType, t); });
-                if (assignedType.flags & 512 /* BooleanLiteral */ && isFreshLiteralType(assignedType)) {
-                    reducedType = mapType(reducedType, getFreshTypeOfLiteralType); // Ensure that if the assignment is a fresh type, that we narrow to fresh types
-                }
-                // Our crude heuristic produces an invalid result in some cases: see GH#26130.
-                // For now, when that happens, we give up and don't narrow at all.  (This also
-                // means we'll never narrow for erroneous assignments where the assigned type
-                // is not assignable to the declared type.)
-                if (isTypeAssignableTo(assignedType, reducedType)) {
-                    return reducedType;
+            else {
+                if (!(source.flags & 3145728 /* UnionOrIntersection */) && !(target.flags & 3145728 /* UnionOrIntersection */) &&
+                    source.flags !== target.flags && !(source.flags & 469237760 /* Substructure */))
+                    return false;
+            }
+            if (source.flags & 524288 /* Object */ && target.flags & 524288 /* Object */) {
+                var related = relation.get(getRelationKey(source, target, 0 /* None */, relation));
+                if (related !== undefined) {
+                    return !!(related & 1 /* Succeeded */);
                 }
             }
-            return declaredType;
-        }
-        function getTypeFactsOfTypes(types) {
-            var result = 0 /* None */;
-            for (var _i = 0, types_15 = types; _i < types_15.length; _i++) {
-                var t = types_15[_i];
-                result |= getTypeFacts(t);
+            if (source.flags & 469499904 /* StructuredOrInstantiable */ || target.flags & 469499904 /* StructuredOrInstantiable */) {
+                return checkTypeRelatedTo(source, target, relation, /*errorNode*/ undefined);
             }
-            return result;
+            return false;
         }
-        function isFunctionObjectType(type) {
-            // We do a quick check for a "bind" property before performing the more expensive subtype
-            // check. This gives us a quicker out in the common case where an object type is not a function.
-            var resolved = resolveStructuredTypeMembers(type);
-            return !!(resolved.callSignatures.length || resolved.constructSignatures.length ||
-                resolved.members.get("bind") && isTypeSubtypeOf(type, globalFunctionType));
+        function isIgnoredJsxProperty(source, sourceProp) {
+            return ts.getObjectFlags(source) & 4096 /* JsxAttributes */ && !isUnhyphenatedJsxName(sourceProp.escapedName);
         }
-        function getTypeFacts(type) {
-            var flags = type.flags;
-            if (flags & 4 /* String */) {
-                return strictNullChecks ? 16317953 /* StringStrictFacts */ : 16776705 /* StringFacts */;
-            }
-            if (flags & 128 /* StringLiteral */) {
-                var isEmpty = type.value === "";
-                return strictNullChecks ?
-                    isEmpty ? 12123649 /* EmptyStringStrictFacts */ : 7929345 /* NonEmptyStringStrictFacts */ :
-                    isEmpty ? 12582401 /* EmptyStringFacts */ : 16776705 /* NonEmptyStringFacts */;
-            }
-            if (flags & (8 /* Number */ | 32 /* Enum */)) {
-                return strictNullChecks ? 16317698 /* NumberStrictFacts */ : 16776450 /* NumberFacts */;
-            }
-            if (flags & 256 /* NumberLiteral */) {
-                var isZero = type.value === 0;
-                return strictNullChecks ?
-                    isZero ? 12123394 /* ZeroNumberStrictFacts */ : 7929090 /* NonZeroNumberStrictFacts */ :
-                    isZero ? 12582146 /* ZeroNumberFacts */ : 16776450 /* NonZeroNumberFacts */;
-            }
-            if (flags & 64 /* BigInt */) {
-                return strictNullChecks ? 16317188 /* BigIntStrictFacts */ : 16775940 /* BigIntFacts */;
-            }
-            if (flags & 2048 /* BigIntLiteral */) {
-                var isZero = isZeroBigInt(type);
-                return strictNullChecks ?
-                    isZero ? 12122884 /* ZeroBigIntStrictFacts */ : 7928580 /* NonZeroBigIntStrictFacts */ :
-                    isZero ? 12581636 /* ZeroBigIntFacts */ : 16775940 /* NonZeroBigIntFacts */;
-            }
-            if (flags & 16 /* Boolean */) {
-                return strictNullChecks ? 16316168 /* BooleanStrictFacts */ : 16774920 /* BooleanFacts */;
-            }
-            if (flags & 528 /* BooleanLike */) {
-                return strictNullChecks ?
-                    (type === falseType || type === regularFalseType) ? 12121864 /* FalseStrictFacts */ : 7927560 /* TrueStrictFacts */ :
-                    (type === falseType || type === regularFalseType) ? 12580616 /* FalseFacts */ : 16774920 /* TrueFacts */;
-            }
-            if (flags & 524288 /* Object */) {
-                return ts.getObjectFlags(type) & 16 /* Anonymous */ && isEmptyObjectType(type) ?
-                    strictNullChecks ? 16318463 /* EmptyObjectStrictFacts */ : 16777215 /* EmptyObjectFacts */ :
-                    isFunctionObjectType(type) ?
-                        strictNullChecks ? 7880640 /* FunctionStrictFacts */ : 16728000 /* FunctionFacts */ :
-                        strictNullChecks ? 7888800 /* ObjectStrictFacts */ : 16736160 /* ObjectFacts */;
-            }
-            if (flags & (16384 /* Void */ | 32768 /* Undefined */)) {
-                return 9830144 /* UndefinedFacts */;
-            }
-            if (flags & 65536 /* Null */) {
-                return 9363232 /* NullFacts */;
-            }
-            if (flags & 12288 /* ESSymbolLike */) {
-                return strictNullChecks ? 7925520 /* SymbolStrictFacts */ : 16772880 /* SymbolFacts */;
-            }
-            if (flags & 67108864 /* NonPrimitive */) {
-                return strictNullChecks ? 7888800 /* ObjectStrictFacts */ : 16736160 /* ObjectFacts */;
-            }
-            if (flags & 131072 /* Never */) {
-                return 0 /* None */;
+        function getNormalizedType(type, writing) {
+            while (true) {
+                var t = isFreshLiteralType(type) ? type.regularType :
+                    ts.getObjectFlags(type) & 4 /* Reference */ && type.node ? createTypeReference(type.target, getTypeArguments(type)) :
+                        type.flags & 3145728 /* UnionOrIntersection */ ? getReducedType(type) :
+                            type.flags & 33554432 /* Substitution */ ? writing ? type.baseType : type.substitute :
+                                type.flags & 25165824 /* Simplifiable */ ? getSimplifiedType(type, writing) :
+                                    type;
+                if (t === type)
+                    break;
+                type = t;
             }
-            if (flags & 63176704 /* Instantiable */) {
-                return getTypeFacts(getBaseConstraintOfType(type) || unknownType);
+            return type;
+        }
+        /**
+         * Checks if 'source' is related to 'target' (e.g.: is a assignable to).
+         * @param source The left-hand-side of the relation.
+         * @param target The right-hand-side of the relation.
+         * @param relation The relation considered. One of 'identityRelation', 'subtypeRelation', 'assignableRelation', or 'comparableRelation'.
+         * Used as both to determine which checks are performed and as a cache of previously computed results.
+         * @param errorNode The suggested node upon which all errors will be reported, if defined. This may or may not be the actual node used.
+         * @param headMessage If the error chain should be prepended by a head message, then headMessage will be used.
+         * @param containingMessageChain A chain of errors to prepend any new errors found.
+         * @param errorOutputContainer Return the diagnostic. Do not log if 'skipLogging' is truthy.
+         */
+        function checkTypeRelatedTo(source, target, relation, errorNode, headMessage, containingMessageChain, errorOutputContainer) {
+            var errorInfo;
+            var relatedInfo;
+            var maybeKeys;
+            var sourceStack;
+            var targetStack;
+            var maybeCount = 0;
+            var depth = 0;
+            var expandingFlags = 0 /* None */;
+            var overflow = false;
+            var overrideNextErrorInfo = 0; // How many `reportRelationError` calls should be skipped in the elaboration pyramid
+            var lastSkippedInfo;
+            var incompatibleStack = [];
+            var inPropertyCheck = false;
+            ts.Debug.assert(relation !== identityRelation || !errorNode, "no error reporting in identity checking");
+            var result = isRelatedTo(source, target, /*reportErrors*/ !!errorNode, headMessage);
+            if (incompatibleStack.length) {
+                reportIncompatibleStack();
             }
-            if (flags & 3145728 /* UnionOrIntersection */) {
-                return getTypeFactsOfTypes(type.types);
+            if (overflow) {
+                ts.tracing.instant("check" /* Check */, "checkTypeRelatedTo_DepthLimit", { sourceId: source.id, targetId: target.id, depth: depth });
+                var diag = error(errorNode || currentNode, ts.Diagnostics.Excessive_stack_depth_comparing_types_0_and_1, typeToString(source), typeToString(target));
+                if (errorOutputContainer) {
+                    (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
+                }
             }
-            return 16777215 /* All */;
-        }
-        function getTypeWithFacts(type, include) {
-            return filterType(type, function (t) { return (getTypeFacts(t) & include) !== 0; });
-        }
-        function getTypeWithDefault(type, defaultExpression) {
-            if (defaultExpression) {
-                var defaultType = getTypeOfExpression(defaultExpression);
-                return getUnionType([getTypeWithFacts(type, 524288 /* NEUndefined */), defaultType]);
+            else if (errorInfo) {
+                if (containingMessageChain) {
+                    var chain = containingMessageChain();
+                    if (chain) {
+                        ts.concatenateDiagnosticMessageChains(chain, errorInfo);
+                        errorInfo = chain;
+                    }
+                }
+                var relatedInformation = void 0;
+                // Check if we should issue an extra diagnostic to produce a quickfix for a slightly incorrect import statement
+                if (headMessage && errorNode && !result && source.symbol) {
+                    var links = getSymbolLinks(source.symbol);
+                    if (links.originatingImport && !ts.isImportCall(links.originatingImport)) {
+                        var helpfulRetry = checkTypeRelatedTo(getTypeOfSymbol(links.target), target, relation, /*errorNode*/ undefined);
+                        if (helpfulRetry) {
+                            // Likely an incorrect import. Issue a helpful diagnostic to produce a quickfix to change the import
+                            var diag_1 = ts.createDiagnosticForNode(links.originatingImport, ts.Diagnostics.Type_originates_at_this_import_A_namespace_style_import_cannot_be_called_or_constructed_and_will_cause_a_failure_at_runtime_Consider_using_a_default_import_or_import_require_here_instead);
+                            relatedInformation = ts.append(relatedInformation, diag_1); // Cause the error to appear with the error that triggered it
+                        }
+                    }
+                }
+                var diag = ts.createDiagnosticForNodeFromMessageChain(errorNode, errorInfo, relatedInformation);
+                if (relatedInfo) {
+                    ts.addRelatedInfo.apply(void 0, __spreadArrays([diag], relatedInfo));
+                }
+                if (errorOutputContainer) {
+                    (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
+                }
+                if (!errorOutputContainer || !errorOutputContainer.skipLogging) {
+                    diagnostics.add(diag);
+                }
             }
-            return type;
-        }
-        function getTypeOfDestructuredProperty(type, name) {
-            var nameType = getLiteralTypeFromPropertyName(name);
-            if (!isTypeUsableAsPropertyName(nameType))
-                return errorType;
-            var text = getPropertyNameFromType(nameType);
-            return getConstraintForLocation(getTypeOfPropertyOfType(type, text), name) ||
-                isNumericLiteralName(text) && getIndexTypeOfType(type, 1 /* Number */) ||
-                getIndexTypeOfType(type, 0 /* String */) ||
-                errorType;
-        }
-        function getTypeOfDestructuredArrayElement(type, index) {
-            return everyType(type, isTupleLikeType) && getTupleElementType(type, index) ||
-                checkIteratedTypeOrElementType(65 /* Destructuring */, type, undefinedType, /*errorNode*/ undefined) ||
-                errorType;
-        }
-        function getTypeOfDestructuredSpreadExpression(type) {
-            return createArrayType(checkIteratedTypeOrElementType(65 /* Destructuring */, type, undefinedType, /*errorNode*/ undefined) || errorType);
-        }
-        function getAssignedTypeOfBinaryExpression(node) {
-            var isDestructuringDefaultAssignment = node.parent.kind === 192 /* ArrayLiteralExpression */ && isDestructuringAssignmentTarget(node.parent) ||
-                node.parent.kind === 281 /* PropertyAssignment */ && isDestructuringAssignmentTarget(node.parent.parent);
-            return isDestructuringDefaultAssignment ?
-                getTypeWithDefault(getAssignedType(node), node.right) :
-                getTypeOfExpression(node.right);
-        }
-        function isDestructuringAssignmentTarget(parent) {
-            return parent.parent.kind === 209 /* BinaryExpression */ && parent.parent.left === parent ||
-                parent.parent.kind === 232 /* ForOfStatement */ && parent.parent.initializer === parent;
-        }
-        function getAssignedTypeOfArrayLiteralElement(node, element) {
-            return getTypeOfDestructuredArrayElement(getAssignedType(node), node.elements.indexOf(element));
-        }
-        function getAssignedTypeOfSpreadExpression(node) {
-            return getTypeOfDestructuredSpreadExpression(getAssignedType(node.parent));
-        }
-        function getAssignedTypeOfPropertyAssignment(node) {
-            return getTypeOfDestructuredProperty(getAssignedType(node.parent), node.name);
-        }
-        function getAssignedTypeOfShorthandPropertyAssignment(node) {
-            return getTypeWithDefault(getAssignedTypeOfPropertyAssignment(node), node.objectAssignmentInitializer);
-        }
-        function getAssignedType(node) {
-            var parent = node.parent;
-            switch (parent.kind) {
-                case 231 /* ForInStatement */:
-                    return stringType;
-                case 232 /* ForOfStatement */:
-                    return checkRightHandSideOfForOf(parent) || errorType;
-                case 209 /* BinaryExpression */:
-                    return getAssignedTypeOfBinaryExpression(parent);
-                case 203 /* DeleteExpression */:
-                    return undefinedType;
-                case 192 /* ArrayLiteralExpression */:
-                    return getAssignedTypeOfArrayLiteralElement(parent, node);
-                case 213 /* SpreadElement */:
-                    return getAssignedTypeOfSpreadExpression(parent);
-                case 281 /* PropertyAssignment */:
-                    return getAssignedTypeOfPropertyAssignment(parent);
-                case 282 /* ShorthandPropertyAssignment */:
-                    return getAssignedTypeOfShorthandPropertyAssignment(parent);
+            if (errorNode && errorOutputContainer && errorOutputContainer.skipLogging && result === 0 /* False */) {
+                ts.Debug.assert(!!errorOutputContainer.errors, "missed opportunity to interact with error.");
             }
-            return errorType;
-        }
-        function getInitialTypeOfBindingElement(node) {
-            var pattern = node.parent;
-            var parentType = getInitialType(pattern.parent);
-            var type = pattern.kind === 189 /* ObjectBindingPattern */ ?
-                getTypeOfDestructuredProperty(parentType, node.propertyName || node.name) :
-                !node.dotDotDotToken ?
-                    getTypeOfDestructuredArrayElement(parentType, pattern.elements.indexOf(node)) :
-                    getTypeOfDestructuredSpreadExpression(parentType);
-            return getTypeWithDefault(type, node.initializer);
-        }
-        function getTypeOfInitializer(node) {
-            // Return the cached type if one is available. If the type of the variable was inferred
-            // from its initializer, we'll already have cached the type. Otherwise we compute it now
-            // without caching such that transient types are reflected.
-            var links = getNodeLinks(node);
-            return links.resolvedType || getTypeOfExpression(node);
-        }
-        function getInitialTypeOfVariableDeclaration(node) {
-            if (node.initializer) {
-                return getTypeOfInitializer(node.initializer);
+            return result !== 0 /* False */;
+            function resetErrorInfo(saved) {
+                errorInfo = saved.errorInfo;
+                lastSkippedInfo = saved.lastSkippedInfo;
+                incompatibleStack = saved.incompatibleStack;
+                overrideNextErrorInfo = saved.overrideNextErrorInfo;
+                relatedInfo = saved.relatedInfo;
             }
-            if (node.parent.parent.kind === 231 /* ForInStatement */) {
-                return stringType;
+            function captureErrorCalculationState() {
+                return {
+                    errorInfo: errorInfo,
+                    lastSkippedInfo: lastSkippedInfo,
+                    incompatibleStack: incompatibleStack.slice(),
+                    overrideNextErrorInfo: overrideNextErrorInfo,
+                    relatedInfo: !relatedInfo ? undefined : relatedInfo.slice()
+                };
             }
-            if (node.parent.parent.kind === 232 /* ForOfStatement */) {
-                return checkRightHandSideOfForOf(node.parent.parent) || errorType;
+            function reportIncompatibleError(message, arg0, arg1, arg2, arg3) {
+                overrideNextErrorInfo++; // Suppress the next relation error
+                lastSkippedInfo = undefined; // Reset skipped info cache
+                incompatibleStack.push([message, arg0, arg1, arg2, arg3]);
             }
-            return errorType;
-        }
-        function getInitialType(node) {
-            return node.kind === 242 /* VariableDeclaration */ ?
-                getInitialTypeOfVariableDeclaration(node) :
-                getInitialTypeOfBindingElement(node);
-        }
-        function isEmptyArrayAssignment(node) {
-            return node.kind === 242 /* VariableDeclaration */ && node.initializer &&
-                isEmptyArrayLiteral(node.initializer) ||
-                node.kind !== 191 /* BindingElement */ && node.parent.kind === 209 /* BinaryExpression */ &&
-                    isEmptyArrayLiteral(node.parent.right);
-        }
-        function getReferenceCandidate(node) {
-            switch (node.kind) {
-                case 200 /* ParenthesizedExpression */:
-                    return getReferenceCandidate(node.expression);
-                case 209 /* BinaryExpression */:
-                    switch (node.operatorToken.kind) {
-                        case 62 /* EqualsToken */:
-                            return getReferenceCandidate(node.left);
-                        case 27 /* CommaToken */:
-                            return getReferenceCandidate(node.right);
+            function reportIncompatibleStack() {
+                var stack = incompatibleStack;
+                incompatibleStack = [];
+                var info = lastSkippedInfo;
+                lastSkippedInfo = undefined;
+                if (stack.length === 1) {
+                    reportError.apply(void 0, stack[0]);
+                    if (info) {
+                        // Actually do the last relation error
+                        reportRelationError.apply(void 0, __spreadArrays([/*headMessage*/ undefined], info));
+                    }
+                    return;
+                }
+                // The first error will be the innermost, while the last will be the outermost - so by popping off the end,
+                // we can build from left to right
+                var path = "";
+                var secondaryRootErrors = [];
+                while (stack.length) {
+                    var _a = stack.pop(), msg = _a[0], args = _a.slice(1);
+                    switch (msg.code) {
+                        case ts.Diagnostics.Types_of_property_0_are_incompatible.code: {
+                            // Parenthesize a `new` if there is one
+                            if (path.indexOf("new ") === 0) {
+                                path = "(" + path + ")";
+                            }
+                            var str = "" + args[0];
+                            // If leading, just print back the arg (irrespective of if it's a valid identifier)
+                            if (path.length === 0) {
+                                path = "" + str;
+                            }
+                            // Otherwise write a dotted name if possible
+                            else if (ts.isIdentifierText(str, compilerOptions.target)) {
+                                path = path + "." + str;
+                            }
+                            // Failing that, check if the name is already a computed name
+                            else if (str[0] === "[" && str[str.length - 1] === "]") {
+                                path = "" + path + str;
+                            }
+                            // And finally write out a computed name as a last resort
+                            else {
+                                path = path + "[" + str + "]";
+                            }
+                            break;
+                        }
+                        case ts.Diagnostics.Call_signature_return_types_0_and_1_are_incompatible.code:
+                        case ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible.code:
+                        case ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code:
+                        case ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code: {
+                            if (path.length === 0) {
+                                // Don't flatten signature compatability errors at the start of a chain - instead prefer
+                                // to unify (the with no arguments bit is excessive for printback) and print them back
+                                var mappedMsg = msg;
+                                if (msg.code === ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code) {
+                                    mappedMsg = ts.Diagnostics.Call_signature_return_types_0_and_1_are_incompatible;
+                                }
+                                else if (msg.code === ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code) {
+                                    mappedMsg = ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible;
+                                }
+                                secondaryRootErrors.unshift([mappedMsg, args[0], args[1]]);
+                            }
+                            else {
+                                var prefix = (msg.code === ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible.code ||
+                                    msg.code === ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code)
+                                    ? "new "
+                                    : "";
+                                var params = (msg.code === ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code ||
+                                    msg.code === ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1.code)
+                                    ? ""
+                                    : "...";
+                                path = "" + prefix + path + "(" + params + ")";
+                            }
+                            break;
+                        }
+                        default:
+                            return ts.Debug.fail("Unhandled Diagnostic: " + msg.code);
                     }
+                }
+                if (path) {
+                    reportError(path[path.length - 1] === ")"
+                        ? ts.Diagnostics.The_types_returned_by_0_are_incompatible_between_these_types
+                        : ts.Diagnostics.The_types_of_0_are_incompatible_between_these_types, path);
+                }
+                else {
+                    // Remove the innermost secondary error as it will duplicate the error already reported by `reportRelationError` on entry
+                    secondaryRootErrors.shift();
+                }
+                for (var _i = 0, secondaryRootErrors_1 = secondaryRootErrors; _i < secondaryRootErrors_1.length; _i++) {
+                    var _b = secondaryRootErrors_1[_i], msg = _b[0], args = _b.slice(1);
+                    var originalValue = msg.elidedInCompatabilityPyramid;
+                    msg.elidedInCompatabilityPyramid = false; // Teporarily override elision to ensure error is reported
+                    reportError.apply(void 0, __spreadArrays([msg], args));
+                    msg.elidedInCompatabilityPyramid = originalValue;
+                }
+                if (info) {
+                    // Actually do the last relation error
+                    reportRelationError.apply(void 0, __spreadArrays([/*headMessage*/ undefined], info));
+                }
             }
-            return node;
-        }
-        function getReferenceRoot(node) {
-            var parent = node.parent;
-            return parent.kind === 200 /* ParenthesizedExpression */ ||
-                parent.kind === 209 /* BinaryExpression */ && parent.operatorToken.kind === 62 /* EqualsToken */ && parent.left === node ||
-                parent.kind === 209 /* BinaryExpression */ && parent.operatorToken.kind === 27 /* CommaToken */ && parent.right === node ?
-                getReferenceRoot(parent) : node;
-        }
-        function getTypeOfSwitchClause(clause) {
-            if (clause.kind === 277 /* CaseClause */) {
-                return getRegularTypeOfLiteralType(getTypeOfExpression(clause.expression));
+            function reportError(message, arg0, arg1, arg2, arg3) {
+                ts.Debug.assert(!!errorNode);
+                if (incompatibleStack.length)
+                    reportIncompatibleStack();
+                if (message.elidedInCompatabilityPyramid)
+                    return;
+                errorInfo = ts.chainDiagnosticMessages(errorInfo, message, arg0, arg1, arg2, arg3);
             }
-            return neverType;
-        }
-        function getSwitchClauseTypes(switchStatement) {
-            var links = getNodeLinks(switchStatement);
-            if (!links.switchTypes) {
-                links.switchTypes = [];
-                for (var _i = 0, _a = switchStatement.caseBlock.clauses; _i < _a.length; _i++) {
-                    var clause = _a[_i];
-                    links.switchTypes.push(getTypeOfSwitchClause(clause));
+            function associateRelatedInfo(info) {
+                ts.Debug.assert(!!errorInfo);
+                if (!relatedInfo) {
+                    relatedInfo = [info];
+                }
+                else {
+                    relatedInfo.push(info);
                 }
             }
-            return links.switchTypes;
-        }
-        function getSwitchClauseTypeOfWitnesses(switchStatement, retainDefault) {
-            var witnesses = [];
-            for (var _i = 0, _a = switchStatement.caseBlock.clauses; _i < _a.length; _i++) {
-                var clause = _a[_i];
-                if (clause.kind === 277 /* CaseClause */) {
-                    if (ts.isStringLiteralLike(clause.expression)) {
-                        witnesses.push(clause.expression.text);
-                        continue;
+            function reportRelationError(message, source, target) {
+                if (incompatibleStack.length)
+                    reportIncompatibleStack();
+                var _a = getTypeNamesForErrorDisplay(source, target), sourceType = _a[0], targetType = _a[1];
+                var generalizedSource = source;
+                var generalizedSourceType = sourceType;
+                if (isLiteralType(source) && !typeCouldHaveTopLevelSingletonTypes(target)) {
+                    generalizedSource = getBaseTypeOfLiteralType(source);
+                    ts.Debug.assert(!isTypeAssignableTo(generalizedSource, target), "generalized source shouldn't be assignable");
+                    generalizedSourceType = getTypeNameForErrorDisplay(generalizedSource);
+                }
+                if (target.flags & 262144 /* TypeParameter */) {
+                    var constraint = getBaseConstraintOfType(target);
+                    var needsOriginalSource = void 0;
+                    if (constraint && (isTypeAssignableTo(generalizedSource, constraint) || (needsOriginalSource = isTypeAssignableTo(source, constraint)))) {
+                        reportError(ts.Diagnostics._0_is_assignable_to_the_constraint_of_type_1_but_1_could_be_instantiated_with_a_different_subtype_of_constraint_2, needsOriginalSource ? sourceType : generalizedSourceType, targetType, typeToString(constraint));
+                    }
+                    else {
+                        reportError(ts.Diagnostics._0_could_be_instantiated_with_an_arbitrary_type_which_could_be_unrelated_to_1, targetType, generalizedSourceType);
                     }
-                    return ts.emptyArray;
                 }
-                if (retainDefault)
-                    witnesses.push(/*explicitDefaultStatement*/ undefined);
+                if (!message) {
+                    if (relation === comparableRelation) {
+                        message = ts.Diagnostics.Type_0_is_not_comparable_to_type_1;
+                    }
+                    else if (sourceType === targetType) {
+                        message = ts.Diagnostics.Type_0_is_not_assignable_to_type_1_Two_different_types_with_this_name_exist_but_they_are_unrelated;
+                    }
+                    else {
+                        message = ts.Diagnostics.Type_0_is_not_assignable_to_type_1;
+                    }
+                }
+                reportError(message, generalizedSourceType, targetType);
             }
-            return witnesses;
-        }
-        function eachTypeContainedIn(source, types) {
-            return source.flags & 1048576 /* Union */ ? !ts.forEach(source.types, function (t) { return !ts.contains(types, t); }) : ts.contains(types, source);
-        }
-        function isTypeSubsetOf(source, target) {
-            return source === target || target.flags & 1048576 /* Union */ && isTypeSubsetOfUnion(source, target);
-        }
-        function isTypeSubsetOfUnion(source, target) {
-            if (source.flags & 1048576 /* Union */) {
-                for (var _i = 0, _a = source.types; _i < _a.length; _i++) {
-                    var t = _a[_i];
-                    if (!containsType(target.types, t)) {
+            function tryElaborateErrorsForPrimitivesAndObjects(source, target) {
+                var sourceType = symbolValueDeclarationIsContextSensitive(source.symbol) ? typeToString(source, source.symbol.valueDeclaration) : typeToString(source);
+                var targetType = symbolValueDeclarationIsContextSensitive(target.symbol) ? typeToString(target, target.symbol.valueDeclaration) : typeToString(target);
+                if ((globalStringType === source && stringType === target) ||
+                    (globalNumberType === source && numberType === target) ||
+                    (globalBooleanType === source && booleanType === target) ||
+                    (getGlobalESSymbolType(/*reportErrors*/ false) === source && esSymbolType === target)) {
+                    reportError(ts.Diagnostics._0_is_a_primitive_but_1_is_a_wrapper_object_Prefer_using_0_when_possible, targetType, sourceType);
+                }
+            }
+            /**
+             * Try and elaborate array and tuple errors. Returns false
+             * if we have found an elaboration, or we should ignore
+             * any other elaborations when relating the `source` and
+             * `target` types.
+             */
+            function tryElaborateArrayLikeErrors(source, target, reportErrors) {
+                /**
+                 * The spec for elaboration is:
+                 * - If the source is a readonly tuple and the target is a mutable array or tuple, elaborate on mutability and skip property elaborations.
+                 * - If the source is a tuple then skip property elaborations if the target is an array or tuple.
+                 * - If the source is a readonly array and the target is a mutable array or tuple, elaborate on mutability and skip property elaborations.
+                 * - If the source an array then skip property elaborations if the target is a tuple.
+                 */
+                if (isTupleType(source)) {
+                    if (source.target.readonly && isMutableArrayOrTuple(target)) {
+                        if (reportErrors) {
+                            reportError(ts.Diagnostics.The_type_0_is_readonly_and_cannot_be_assigned_to_the_mutable_type_1, typeToString(source), typeToString(target));
+                        }
                         return false;
                     }
+                    return isTupleType(target) || isArrayType(target);
+                }
+                if (isReadonlyArrayType(source) && isMutableArrayOrTuple(target)) {
+                    if (reportErrors) {
+                        reportError(ts.Diagnostics.The_type_0_is_readonly_and_cannot_be_assigned_to_the_mutable_type_1, typeToString(source), typeToString(target));
+                    }
+                    return false;
+                }
+                if (isTupleType(target)) {
+                    return isArrayType(source);
                 }
                 return true;
             }
-            if (source.flags & 1024 /* EnumLiteral */ && getBaseTypeOfEnumLiteralType(source) === target) {
-                return true;
-            }
-            return containsType(target.types, source);
-        }
-        function forEachType(type, f) {
-            return type.flags & 1048576 /* Union */ ? ts.forEach(type.types, f) : f(type);
-        }
-        function everyType(type, f) {
-            return type.flags & 1048576 /* Union */ ? ts.every(type.types, f) : f(type);
-        }
-        function filterType(type, f) {
-            if (type.flags & 1048576 /* Union */) {
-                var types = type.types;
-                var filtered = ts.filter(types, f);
-                return filtered === types ? type : getUnionTypeFromSortedList(filtered, type.objectFlags);
-            }
-            return type.flags & 131072 /* Never */ || f(type) ? type : neverType;
-        }
-        function countTypes(type) {
-            return type.flags & 1048576 /* Union */ ? type.types.length : 1;
-        }
-        function mapType(type, mapper, noReductions) {
-            if (type.flags & 131072 /* Never */) {
-                return type;
-            }
-            if (!(type.flags & 1048576 /* Union */)) {
-                return mapper(type);
-            }
-            var mappedTypes;
-            for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
-                var t = _a[_i];
-                var mapped = mapper(t);
-                if (mapped) {
-                    if (!mappedTypes) {
-                        mappedTypes = [mapped];
+            /**
+             * Compare two types and return
+             * * Ternary.True if they are related with no assumptions,
+             * * Ternary.Maybe if they are related with assumptions of other relationships, or
+             * * Ternary.False if they are not related.
+             */
+            function isRelatedTo(originalSource, originalTarget, reportErrors, headMessage, intersectionState) {
+                if (reportErrors === void 0) { reportErrors = false; }
+                if (intersectionState === void 0) { intersectionState = 0 /* None */; }
+                // Before normalization: if `source` is type an object type, and `target` is primitive,
+                // skip all the checks we don't need and just return `isSimpleTypeRelatedTo` result
+                if (originalSource.flags & 524288 /* Object */ && originalTarget.flags & 131068 /* Primitive */) {
+                    if (isSimpleTypeRelatedTo(originalSource, originalTarget, relation, reportErrors ? reportError : undefined)) {
+                        return -1 /* True */;
+                    }
+                    reportErrorResults(originalSource, originalTarget, 0 /* False */, !!(ts.getObjectFlags(originalSource) & 4096 /* JsxAttributes */));
+                    return 0 /* False */;
+                }
+                // Normalize the source and target types: Turn fresh literal types into regular literal types,
+                // turn deferred type references into regular type references, simplify indexed access and
+                // conditional types, and resolve substitution types to either the substitution (on the source
+                // side) or the type variable (on the target side).
+                var source = getNormalizedType(originalSource, /*writing*/ false);
+                var target = getNormalizedType(originalTarget, /*writing*/ true);
+                if (source === target)
+                    return -1 /* True */;
+                if (relation === identityRelation) {
+                    return isIdenticalTo(source, target);
+                }
+                // We fastpath comparing a type parameter to exactly its constraint, as this is _super_ common,
+                // and otherwise, for type parameters in large unions, causes us to need to compare the union to itself,
+                // as we break down the _target_ union first, _then_ get the source constraint - so for every
+                // member of the target, we attempt to find a match in the source. This avoids that in cases where
+                // the target is exactly the constraint.
+                if (source.flags & 262144 /* TypeParameter */ && getConstraintOfType(source) === target) {
+                    return -1 /* True */;
+                }
+                // Try to see if we're relating something like `Foo` -> `Bar | null | undefined`.
+                // If so, reporting the `null` and `undefined` in the type is hardly useful.
+                // First, see if we're even relating an object type to a union.
+                // Then see if the target is stripped down to a single non-union type.
+                // Note
+                //  * We actually want to remove null and undefined naively here (rather than using getNonNullableType),
+                //    since we don't want to end up with a worse error like "`Foo` is not assignable to `NonNullable<T>`"
+                //    when dealing with generics.
+                //  * We also don't deal with primitive source types, since we already halt elaboration below.
+                if (target.flags & 1048576 /* Union */ && source.flags & 524288 /* Object */ &&
+                    target.types.length <= 3 && maybeTypeOfKind(target, 98304 /* Nullable */)) {
+                    var nullStrippedTarget = extractTypesOfKind(target, ~98304 /* Nullable */);
+                    if (!(nullStrippedTarget.flags & (1048576 /* Union */ | 131072 /* Never */))) {
+                        if (source === nullStrippedTarget)
+                            return -1 /* True */;
+                        target = nullStrippedTarget;
                     }
-                    else {
-                        mappedTypes.push(mapped);
+                }
+                if (relation === comparableRelation && !(target.flags & 131072 /* Never */) && isSimpleTypeRelatedTo(target, source, relation) ||
+                    isSimpleTypeRelatedTo(source, target, relation, reportErrors ? reportError : undefined))
+                    return -1 /* True */;
+                var isComparingJsxAttributes = !!(ts.getObjectFlags(source) & 4096 /* JsxAttributes */);
+                var isPerformingExcessPropertyChecks = !(intersectionState & 2 /* Target */) && (isObjectLiteralType(source) && ts.getObjectFlags(source) & 32768 /* FreshLiteral */);
+                if (isPerformingExcessPropertyChecks) {
+                    if (hasExcessProperties(source, target, reportErrors)) {
+                        if (reportErrors) {
+                            reportRelationError(headMessage, source, target);
+                        }
+                        return 0 /* False */;
                     }
                 }
-            }
-            return mappedTypes && getUnionType(mappedTypes, noReductions ? 0 /* None */ : 1 /* Literal */);
-        }
-        function extractTypesOfKind(type, kind) {
-            return filterType(type, function (t) { return (t.flags & kind) !== 0; });
-        }
-        // Return a new type in which occurrences of the string and number primitive types in
-        // typeWithPrimitives have been replaced with occurrences of string literals and numeric
-        // literals in typeWithLiterals, respectively.
-        function replacePrimitivesWithLiterals(typeWithPrimitives, typeWithLiterals) {
-            if (isTypeSubsetOf(stringType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, 128 /* StringLiteral */) ||
-                isTypeSubsetOf(numberType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, 256 /* NumberLiteral */) ||
-                isTypeSubsetOf(bigintType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, 2048 /* BigIntLiteral */)) {
-                return mapType(typeWithPrimitives, function (t) {
-                    return t.flags & 4 /* String */ ? extractTypesOfKind(typeWithLiterals, 4 /* String */ | 128 /* StringLiteral */) :
-                        t.flags & 8 /* Number */ ? extractTypesOfKind(typeWithLiterals, 8 /* Number */ | 256 /* NumberLiteral */) :
-                            t.flags & 64 /* BigInt */ ? extractTypesOfKind(typeWithLiterals, 64 /* BigInt */ | 2048 /* BigIntLiteral */) : t;
-                });
-            }
-            return typeWithPrimitives;
-        }
-        function isIncomplete(flowType) {
-            return flowType.flags === 0;
-        }
-        function getTypeFromFlowType(flowType) {
-            return flowType.flags === 0 ? flowType.type : flowType;
-        }
-        function createFlowType(type, incomplete) {
-            return incomplete ? { flags: 0, type: type } : type;
-        }
-        // An evolving array type tracks the element types that have so far been seen in an
-        // 'x.push(value)' or 'x[n] = value' operation along the control flow graph. Evolving
-        // array types are ultimately converted into manifest array types (using getFinalArrayType)
-        // and never escape the getFlowTypeOfReference function.
-        function createEvolvingArrayType(elementType) {
-            var result = createObjectType(256 /* EvolvingArray */);
-            result.elementType = elementType;
-            return result;
-        }
-        function getEvolvingArrayType(elementType) {
-            return evolvingArrayTypes[elementType.id] || (evolvingArrayTypes[elementType.id] = createEvolvingArrayType(elementType));
-        }
-        // When adding evolving array element types we do not perform subtype reduction. Instead,
-        // we defer subtype reduction until the evolving array type is finalized into a manifest
-        // array type.
-        function addEvolvingArrayElementType(evolvingArrayType, node) {
-            var elementType = getBaseTypeOfLiteralType(getContextFreeTypeOfExpression(node));
-            return isTypeSubsetOf(elementType, evolvingArrayType.elementType) ? evolvingArrayType : getEvolvingArrayType(getUnionType([evolvingArrayType.elementType, elementType]));
-        }
-        function createFinalArrayType(elementType) {
-            return elementType.flags & 131072 /* Never */ ?
-                autoArrayType :
-                createArrayType(elementType.flags & 1048576 /* Union */ ?
-                    getUnionType(elementType.types, 2 /* Subtype */) :
-                    elementType);
-        }
-        // We perform subtype reduction upon obtaining the final array type from an evolving array type.
-        function getFinalArrayType(evolvingArrayType) {
-            return evolvingArrayType.finalArrayType || (evolvingArrayType.finalArrayType = createFinalArrayType(evolvingArrayType.elementType));
-        }
-        function finalizeEvolvingArrayType(type) {
-            return ts.getObjectFlags(type) & 256 /* EvolvingArray */ ? getFinalArrayType(type) : type;
-        }
-        function getElementTypeOfEvolvingArrayType(type) {
-            return ts.getObjectFlags(type) & 256 /* EvolvingArray */ ? type.elementType : neverType;
-        }
-        function isEvolvingArrayTypeList(types) {
-            var hasEvolvingArrayType = false;
-            for (var _i = 0, types_16 = types; _i < types_16.length; _i++) {
-                var t = types_16[_i];
-                if (!(t.flags & 131072 /* Never */)) {
-                    if (!(ts.getObjectFlags(t) & 256 /* EvolvingArray */)) {
-                        return false;
+                var isPerformingCommonPropertyChecks = relation !== comparableRelation && !(intersectionState & 2 /* Target */) &&
+                    source.flags & (131068 /* Primitive */ | 524288 /* Object */ | 2097152 /* Intersection */) && source !== globalObjectType &&
+                    target.flags & (524288 /* Object */ | 2097152 /* Intersection */) && isWeakType(target) &&
+                    (getPropertiesOfType(source).length > 0 || typeHasCallOrConstructSignatures(source));
+                if (isPerformingCommonPropertyChecks && !hasCommonProperties(source, target, isComparingJsxAttributes)) {
+                    if (reportErrors) {
+                        var calls = getSignaturesOfType(source, 0 /* Call */);
+                        var constructs = getSignaturesOfType(source, 1 /* Construct */);
+                        if (calls.length > 0 && isRelatedTo(getReturnTypeOfSignature(calls[0]), target, /*reportErrors*/ false) ||
+                            constructs.length > 0 && isRelatedTo(getReturnTypeOfSignature(constructs[0]), target, /*reportErrors*/ false)) {
+                            reportError(ts.Diagnostics.Value_of_type_0_has_no_properties_in_common_with_type_1_Did_you_mean_to_call_it, typeToString(source), typeToString(target));
+                        }
+                        else {
+                            reportError(ts.Diagnostics.Type_0_has_no_properties_in_common_with_type_1, typeToString(source), typeToString(target));
+                        }
                     }
-                    hasEvolvingArrayType = true;
+                    return 0 /* False */;
                 }
-            }
-            return hasEvolvingArrayType;
-        }
-        // At flow control branch or loop junctions, if the type along every antecedent code path
-        // is an evolving array type, we construct a combined evolving array type. Otherwise we
-        // finalize all evolving array types.
-        function getUnionOrEvolvingArrayType(types, subtypeReduction) {
-            return isEvolvingArrayTypeList(types) ?
-                getEvolvingArrayType(getUnionType(ts.map(types, getElementTypeOfEvolvingArrayType))) :
-                getUnionType(ts.sameMap(types, finalizeEvolvingArrayType), subtypeReduction);
-        }
-        // Return true if the given node is 'x' in an 'x.length', x.push(value)', 'x.unshift(value)' or
-        // 'x[n] = value' operation, where 'n' is an expression of type any, undefined, or a number-like type.
-        function isEvolvingArrayOperationTarget(node) {
-            var root = getReferenceRoot(node);
-            var parent = root.parent;
-            var isLengthPushOrUnshift = ts.isPropertyAccessExpression(parent) && (parent.name.escapedText === "length" ||
-                parent.parent.kind === 196 /* CallExpression */
-                    && ts.isIdentifier(parent.name)
-                    && ts.isPushOrUnshiftIdentifier(parent.name));
-            var isElementAssignment = parent.kind === 195 /* ElementAccessExpression */ &&
-                parent.expression === root &&
-                parent.parent.kind === 209 /* BinaryExpression */ &&
-                parent.parent.operatorToken.kind === 62 /* EqualsToken */ &&
-                parent.parent.left === parent &&
-                !ts.isAssignmentTarget(parent.parent) &&
-                isTypeAssignableToKind(getTypeOfExpression(parent.argumentExpression), 296 /* NumberLike */);
-            return isLengthPushOrUnshift || isElementAssignment;
-        }
-        function isDeclarationWithExplicitTypeAnnotation(declaration) {
-            return (declaration.kind === 242 /* VariableDeclaration */ || declaration.kind === 156 /* Parameter */ ||
-                declaration.kind === 159 /* PropertyDeclaration */ || declaration.kind === 158 /* PropertySignature */) &&
-                !!ts.getEffectiveTypeAnnotationNode(declaration);
-        }
-        function getExplicitTypeOfSymbol(symbol, diagnostic) {
-            if (symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 512 /* ValueModule */)) {
-                return getTypeOfSymbol(symbol);
-            }
-            if (symbol.flags & (3 /* Variable */ | 4 /* Property */)) {
-                var declaration = symbol.valueDeclaration;
-                if (declaration) {
-                    if (isDeclarationWithExplicitTypeAnnotation(declaration)) {
-                        return getTypeOfSymbol(symbol);
+                var result = 0 /* False */;
+                var saveErrorInfo = captureErrorCalculationState();
+                // Note that these checks are specifically ordered to produce correct results. In particular,
+                // we need to deconstruct unions before intersections (because unions are always at the top),
+                // and we need to handle "each" relations before "some" relations for the same kind of type.
+                if (source.flags & 1048576 /* Union */) {
+                    result = relation === comparableRelation ?
+                        someTypeRelatedToType(source, target, reportErrors && !(source.flags & 131068 /* Primitive */), intersectionState) :
+                        eachTypeRelatedToType(source, target, reportErrors && !(source.flags & 131068 /* Primitive */), intersectionState);
+                }
+                else {
+                    if (target.flags & 1048576 /* Union */) {
+                        result = typeRelatedToSomeType(getRegularTypeOfObjectLiteral(source), target, reportErrors && !(source.flags & 131068 /* Primitive */) && !(target.flags & 131068 /* Primitive */));
                     }
-                    if (ts.isVariableDeclaration(declaration) && declaration.parent.parent.kind === 232 /* ForOfStatement */) {
-                        var statement = declaration.parent.parent;
-                        var expressionType = getTypeOfDottedName(statement.expression, /*diagnostic*/ undefined);
-                        if (expressionType) {
-                            var use = statement.awaitModifier ? 15 /* ForAwaitOf */ : 13 /* ForOf */;
-                            return checkIteratedTypeOrElementType(use, expressionType, undefinedType, /*errorNode*/ undefined);
+                    else if (target.flags & 2097152 /* Intersection */) {
+                        result = typeRelatedToEachType(getRegularTypeOfObjectLiteral(source), target, reportErrors, 2 /* Target */);
+                    }
+                    else if (source.flags & 2097152 /* Intersection */) {
+                        // Check to see if any constituents of the intersection are immediately related to the target.
+                        //
+                        // Don't report errors though. Checking whether a constituent is related to the source is not actually
+                        // useful and leads to some confusing error messages. Instead it is better to let the below checks
+                        // take care of this, or to not elaborate at all. For instance,
+                        //
+                        //    - For an object type (such as 'C = A & B'), users are usually more interested in structural errors.
+                        //
+                        //    - For a union type (such as '(A | B) = (C & D)'), it's better to hold onto the whole intersection
+                        //          than to report that 'D' is not assignable to 'A' or 'B'.
+                        //
+                        //    - For a primitive type or type parameter (such as 'number = A & B') there is no point in
+                        //          breaking the intersection apart.
+                        result = someTypeRelatedToType(source, target, /*reportErrors*/ false, 1 /* Source */);
+                    }
+                    if (!result && (source.flags & 469499904 /* StructuredOrInstantiable */ || target.flags & 469499904 /* StructuredOrInstantiable */)) {
+                        if (result = recursiveTypeRelatedTo(source, target, reportErrors, intersectionState)) {
+                            resetErrorInfo(saveErrorInfo);
                         }
                     }
-                    if (diagnostic) {
-                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(declaration, ts.Diagnostics._0_needs_an_explicit_type_annotation, symbolToString(symbol)));
+                }
+                if (!result && source.flags & (2097152 /* Intersection */ | 262144 /* TypeParameter */)) {
+                    // The combined constraint of an intersection type is the intersection of the constraints of
+                    // the constituents. When an intersection type contains instantiable types with union type
+                    // constraints, there are situations where we need to examine the combined constraint. One is
+                    // when the target is a union type. Another is when the intersection contains types belonging
+                    // to one of the disjoint domains. For example, given type variables T and U, each with the
+                    // constraint 'string | number', the combined constraint of 'T & U' is 'string | number' and
+                    // we need to check this constraint against a union on the target side. Also, given a type
+                    // variable V constrained to 'string | number', 'V & number' has a combined constraint of
+                    // 'string & number | number & number' which reduces to just 'number'.
+                    // This also handles type parameters, as a type parameter with a union constraint compared against a union
+                    // needs to have its constraint hoisted into an intersection with said type parameter, this way
+                    // the type param can be compared with itself in the target (with the influence of its constraint to match other parts)
+                    // For example, if `T extends 1 | 2` and `U extends 2 | 3` and we compare `T & U` to `T & U & (1 | 2 | 3)`
+                    var constraint = getEffectiveConstraintOfIntersection(source.flags & 2097152 /* Intersection */ ? source.types : [source], !!(target.flags & 1048576 /* Union */));
+                    if (constraint && (source.flags & 2097152 /* Intersection */ || target.flags & 1048576 /* Union */)) {
+                        if (everyType(constraint, function (c) { return c !== source; })) { // Skip comparison if expansion contains the source itself
+                            // TODO: Stack errors so we get a pyramid for the "normal" comparison above, _and_ a second for this
+                            if (result = isRelatedTo(constraint, target, /*reportErrors*/ false, /*headMessage*/ undefined, intersectionState)) {
+                                resetErrorInfo(saveErrorInfo);
+                            }
+                        }
                     }
                 }
-            }
-        }
-        // We require the dotted function name in an assertion expression to be comprised of identifiers
-        // that reference function, method, class or value module symbols; or variable, property or
-        // parameter symbols with declarations that have explicit type annotations. Such references are
-        // resolvable with no possibility of triggering circularities in control flow analysis.
-        function getTypeOfDottedName(node, diagnostic) {
-            if (!(node.flags & 16777216 /* InWithStatement */)) {
-                switch (node.kind) {
-                    case 75 /* Identifier */:
-                        var symbol = getExportSymbolOfValueSymbolIfExported(getResolvedSymbol(node));
-                        return getExplicitTypeOfSymbol(symbol.flags & 2097152 /* Alias */ ? resolveAlias(symbol) : symbol, diagnostic);
-                    case 104 /* ThisKeyword */:
-                        return getExplicitThisType(node);
-                    case 102 /* SuperKeyword */:
-                        return checkSuperExpression(node);
-                    case 194 /* PropertyAccessExpression */:
-                        var type = getTypeOfDottedName(node.expression, diagnostic);
-                        var prop = type && getPropertyOfType(type, node.name.escapedText);
-                        return prop && getExplicitTypeOfSymbol(prop, diagnostic);
-                    case 200 /* ParenthesizedExpression */:
-                        return getTypeOfDottedName(node.expression, diagnostic);
+                // For certain combinations involving intersections and optional, excess, or mismatched properties we need
+                // an extra property check where the intersection is viewed as a single object. The following are motivating
+                // examples that all should be errors, but aren't without this extra property check:
+                //
+                //   let obj: { a: { x: string } } & { c: number } = { a: { x: 'hello', y: 2 }, c: 5 };  // Nested excess property
+                //
+                //   declare let wrong: { a: { y: string } };
+                //   let weak: { a?: { x?: number } } & { c?: string } = wrong;  // Nested weak object type
+                //
+                //   function foo<T extends object>(x: { a?: string }, y: T & { a: boolean }) {
+                //     x = y;  // Mismatched property in source intersection
+                //   }
+                //
+                // We suppress recursive intersection property checks because they can generate lots of work when relating
+                // recursive intersections that are structurally similar but not exactly identical. See #37854.
+                if (result && !inPropertyCheck && (target.flags & 2097152 /* Intersection */ && (isPerformingExcessPropertyChecks || isPerformingCommonPropertyChecks) ||
+                    isNonGenericObjectType(target) && !isArrayType(target) && !isTupleType(target) && source.flags & 2097152 /* Intersection */ && getApparentType(source).flags & 3670016 /* StructuredType */ && !ts.some(source.types, function (t) { return !!(ts.getObjectFlags(t) & 2097152 /* NonInferrableType */); }))) {
+                    inPropertyCheck = true;
+                    result &= recursiveTypeRelatedTo(source, target, reportErrors, 4 /* PropertyCheck */);
+                    inPropertyCheck = false;
+                }
+                reportErrorResults(source, target, result, isComparingJsxAttributes);
+                return result;
+                function reportErrorResults(source, target, result, isComparingJsxAttributes) {
+                    if (!result && reportErrors) {
+                        source = originalSource.aliasSymbol ? originalSource : source;
+                        target = originalTarget.aliasSymbol ? originalTarget : target;
+                        var maybeSuppress = overrideNextErrorInfo > 0;
+                        if (maybeSuppress) {
+                            overrideNextErrorInfo--;
+                        }
+                        if (source.flags & 524288 /* Object */ && target.flags & 524288 /* Object */) {
+                            var currentError = errorInfo;
+                            tryElaborateArrayLikeErrors(source, target, reportErrors);
+                            if (errorInfo !== currentError) {
+                                maybeSuppress = !!errorInfo;
+                            }
+                        }
+                        if (source.flags & 524288 /* Object */ && target.flags & 131068 /* Primitive */) {
+                            tryElaborateErrorsForPrimitivesAndObjects(source, target);
+                        }
+                        else if (source.symbol && source.flags & 524288 /* Object */ && globalObjectType === source) {
+                            reportError(ts.Diagnostics.The_Object_type_is_assignable_to_very_few_other_types_Did_you_mean_to_use_the_any_type_instead);
+                        }
+                        else if (isComparingJsxAttributes && target.flags & 2097152 /* Intersection */) {
+                            var targetTypes = target.types;
+                            var intrinsicAttributes = getJsxType(JsxNames.IntrinsicAttributes, errorNode);
+                            var intrinsicClassAttributes = getJsxType(JsxNames.IntrinsicClassAttributes, errorNode);
+                            if (intrinsicAttributes !== errorType && intrinsicClassAttributes !== errorType &&
+                                (ts.contains(targetTypes, intrinsicAttributes) || ts.contains(targetTypes, intrinsicClassAttributes))) {
+                                // do not report top error
+                                return result;
+                            }
+                        }
+                        else {
+                            errorInfo = elaborateNeverIntersection(errorInfo, originalTarget);
+                        }
+                        if (!headMessage && maybeSuppress) {
+                            lastSkippedInfo = [source, target];
+                            // Used by, eg, missing property checking to replace the top-level message with a more informative one
+                            return result;
+                        }
+                        reportRelationError(headMessage, source, target);
+                    }
                 }
             }
-        }
-        function getEffectsSignature(node) {
-            var links = getNodeLinks(node);
-            var signature = links.effectsSignature;
-            if (signature === undefined) {
-                // A call expression parented by an expression statement is a potential assertion. Other call
-                // expressions are potential type predicate function calls. In order to avoid triggering
-                // circularities in control flow analysis, we use getTypeOfDottedName when resolving the call
-                // target expression of an assertion.
-                var funcType = void 0;
-                if (node.parent.kind === 226 /* ExpressionStatement */) {
-                    funcType = getTypeOfDottedName(node.expression, /*diagnostic*/ undefined);
+            function isIdenticalTo(source, target) {
+                var flags = source.flags & target.flags;
+                if (!(flags & 469237760 /* Substructure */)) {
+                    return 0 /* False */;
                 }
-                else if (node.expression.kind !== 102 /* SuperKeyword */) {
-                    if (ts.isOptionalChain(node)) {
-                        funcType = checkNonNullType(getOptionalExpressionType(checkExpression(node.expression), node.expression), node.expression);
-                    }
-                    else {
-                        funcType = checkNonNullExpression(node.expression);
+                if (flags & 3145728 /* UnionOrIntersection */) {
+                    var result_6 = eachTypeRelatedToSomeType(source, target);
+                    if (result_6) {
+                        result_6 &= eachTypeRelatedToSomeType(target, source);
                     }
+                    return result_6;
                 }
-                var signatures = getSignaturesOfType(funcType && getApparentType(funcType) || unknownType, 0 /* Call */);
-                var candidate = signatures.length === 1 && !signatures[0].typeParameters ? signatures[0] :
-                    ts.some(signatures, hasTypePredicateOrNeverReturnType) ? getResolvedSignature(node) :
-                        undefined;
-                signature = links.effectsSignature = candidate && hasTypePredicateOrNeverReturnType(candidate) ? candidate : unknownSignature;
+                return recursiveTypeRelatedTo(source, target, /*reportErrors*/ false, 0 /* None */);
             }
-            return signature === unknownSignature ? undefined : signature;
-        }
-        function hasTypePredicateOrNeverReturnType(signature) {
-            return !!(getTypePredicateOfSignature(signature) ||
-                signature.declaration && (getReturnTypeFromAnnotation(signature.declaration) || unknownType).flags & 131072 /* Never */);
-        }
-        function getTypePredicateArgument(predicate, callExpression) {
-            if (predicate.kind === 1 /* Identifier */ || predicate.kind === 3 /* AssertsIdentifier */) {
-                return callExpression.arguments[predicate.parameterIndex];
+            function getTypeOfPropertyInTypes(types, name) {
+                var appendPropType = function (propTypes, type) {
+                    type = getApparentType(type);
+                    var prop = type.flags & 3145728 /* UnionOrIntersection */ ? getPropertyOfUnionOrIntersectionType(type, name) : getPropertyOfObjectType(type, name);
+                    var propType = prop && getTypeOfSymbol(prop) || isNumericLiteralName(name) && getIndexTypeOfType(type, 1 /* Number */) || getIndexTypeOfType(type, 0 /* String */) || undefinedType;
+                    return ts.append(propTypes, propType);
+                };
+                return getUnionType(ts.reduceLeft(types, appendPropType, /*initial*/ undefined) || ts.emptyArray);
             }
-            var invokedExpression = ts.skipParentheses(callExpression.expression);
-            return ts.isAccessExpression(invokedExpression) ? ts.skipParentheses(invokedExpression.expression) : undefined;
-        }
-        function reportFlowControlError(node) {
-            var block = ts.findAncestor(node, ts.isFunctionOrModuleBlock);
-            var sourceFile = ts.getSourceFileOfNode(node);
-            var span = ts.getSpanOfTokenAtPosition(sourceFile, block.statements.pos);
-            diagnostics.add(ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.The_containing_function_or_module_body_is_too_large_for_control_flow_analysis));
-        }
-        function isReachableFlowNode(flow) {
-            var result = isReachableFlowNodeWorker(flow, /*noCacheCheck*/ false);
-            lastFlowNode = flow;
-            lastFlowNodeReachable = result;
-            return result;
-        }
-        function isFalseExpression(expr) {
-            var node = ts.skipParentheses(expr);
-            return node.kind === 91 /* FalseKeyword */ || node.kind === 209 /* BinaryExpression */ && (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */ && (isFalseExpression(node.left) || isFalseExpression(node.right)) ||
-                node.operatorToken.kind === 56 /* BarBarToken */ && isFalseExpression(node.left) && isFalseExpression(node.right));
-        }
-        function isReachableFlowNodeWorker(flow, noCacheCheck) {
-            while (true) {
-                if (flow === lastFlowNode) {
-                    return lastFlowNodeReachable;
+            function hasExcessProperties(source, target, reportErrors) {
+                if (!isExcessPropertyCheckTarget(target) || !noImplicitAny && ts.getObjectFlags(target) & 16384 /* JSLiteral */) {
+                    return false; // Disable excess property checks on JS literals to simulate having an implicit "index signature" - but only outside of noImplicitAny
                 }
-                var flags = flow.flags;
-                if (flags & 4096 /* Shared */) {
-                    if (!noCacheCheck) {
-                        var id = getFlowNodeId(flow);
-                        var reachable = flowNodeReachable[id];
-                        return reachable !== undefined ? reachable : (flowNodeReachable[id] = isReachableFlowNodeWorker(flow, /*noCacheCheck*/ true));
-                    }
-                    noCacheCheck = false;
+                var isComparingJsxAttributes = !!(ts.getObjectFlags(source) & 4096 /* JsxAttributes */);
+                if ((relation === assignableRelation || relation === comparableRelation) &&
+                    (isTypeSubsetOf(globalObjectType, target) || (!isComparingJsxAttributes && isEmptyObjectType(target)))) {
+                    return false;
                 }
-                if (flags & (16 /* Assignment */ | 96 /* Condition */ | 256 /* ArrayMutation */)) {
-                    flow = flow.antecedent;
+                var reducedTarget = target;
+                var checkTypes;
+                if (target.flags & 1048576 /* Union */) {
+                    reducedTarget = findMatchingDiscriminantType(source, target, isRelatedTo) || filterPrimitivesIfContainsNonPrimitive(target);
+                    checkTypes = reducedTarget.flags & 1048576 /* Union */ ? reducedTarget.types : [reducedTarget];
                 }
-                else if (flags & 512 /* Call */) {
-                    var signature = getEffectsSignature(flow.node);
-                    if (signature) {
-                        var predicate = getTypePredicateOfSignature(signature);
-                        if (predicate && predicate.kind === 3 /* AssertsIdentifier */) {
-                            var predicateArgument = flow.node.arguments[predicate.parameterIndex];
-                            if (predicateArgument && isFalseExpression(predicateArgument)) {
-                                return false;
+                var _loop_16 = function (prop) {
+                    if (shouldCheckAsExcessProperty(prop, source.symbol) && !isIgnoredJsxProperty(source, prop)) {
+                        if (!isKnownProperty(reducedTarget, prop.escapedName, isComparingJsxAttributes)) {
+                            if (reportErrors) {
+                                // Report error in terms of object types in the target as those are the only ones
+                                // we check in isKnownProperty.
+                                var errorTarget = filterType(reducedTarget, isExcessPropertyCheckTarget);
+                                // We know *exactly* where things went wrong when comparing the types.
+                                // Use this property as the error node as this will be more helpful in
+                                // reasoning about what went wrong.
+                                if (!errorNode)
+                                    return { value: ts.Debug.fail() };
+                                if (ts.isJsxAttributes(errorNode) || ts.isJsxOpeningLikeElement(errorNode) || ts.isJsxOpeningLikeElement(errorNode.parent)) {
+                                    // JsxAttributes has an object-literal flag and undergo same type-assignablity check as normal object-literal.
+                                    // However, using an object-literal error message will be very confusing to the users so we give different a message.
+                                    if (prop.valueDeclaration && ts.isJsxAttribute(prop.valueDeclaration) && ts.getSourceFileOfNode(errorNode) === ts.getSourceFileOfNode(prop.valueDeclaration.name)) {
+                                        // Note that extraneous children (as in `<NoChild>extra</NoChild>`) don't pass this check,
+                                        // since `children` is a SyntaxKind.PropertySignature instead of a SyntaxKind.JsxAttribute.
+                                        errorNode = prop.valueDeclaration.name;
+                                    }
+                                    var propName = symbolToString(prop);
+                                    var suggestionSymbol = getSuggestedSymbolForNonexistentJSXAttribute(propName, errorTarget);
+                                    var suggestion = suggestionSymbol ? symbolToString(suggestionSymbol) : undefined;
+                                    if (suggestion) {
+                                        reportError(ts.Diagnostics.Property_0_does_not_exist_on_type_1_Did_you_mean_2, propName, typeToString(errorTarget), suggestion);
+                                    }
+                                    else {
+                                        reportError(ts.Diagnostics.Property_0_does_not_exist_on_type_1, propName, typeToString(errorTarget));
+                                    }
+                                }
+                                else {
+                                    // use the property's value declaration if the property is assigned inside the literal itself
+                                    var objectLiteralDeclaration_1 = source.symbol && ts.firstOrUndefined(source.symbol.declarations);
+                                    var suggestion = void 0;
+                                    if (prop.valueDeclaration && ts.findAncestor(prop.valueDeclaration, function (d) { return d === objectLiteralDeclaration_1; }) && ts.getSourceFileOfNode(objectLiteralDeclaration_1) === ts.getSourceFileOfNode(errorNode)) {
+                                        var propDeclaration = prop.valueDeclaration;
+                                        ts.Debug.assertNode(propDeclaration, ts.isObjectLiteralElementLike);
+                                        errorNode = propDeclaration;
+                                        var name = propDeclaration.name;
+                                        if (ts.isIdentifier(name)) {
+                                            suggestion = getSuggestionForNonexistentProperty(name, errorTarget);
+                                        }
+                                    }
+                                    if (suggestion !== undefined) {
+                                        reportError(ts.Diagnostics.Object_literal_may_only_specify_known_properties_but_0_does_not_exist_in_type_1_Did_you_mean_to_write_2, symbolToString(prop), typeToString(errorTarget), suggestion);
+                                    }
+                                    else {
+                                        reportError(ts.Diagnostics.Object_literal_may_only_specify_known_properties_and_0_does_not_exist_in_type_1, symbolToString(prop), typeToString(errorTarget));
+                                    }
+                                }
                             }
+                            return { value: true };
                         }
-                        if (getReturnTypeOfSignature(signature).flags & 131072 /* Never */) {
-                            return false;
+                        if (checkTypes && !isRelatedTo(getTypeOfSymbol(prop), getTypeOfPropertyInTypes(checkTypes, prop.escapedName), reportErrors)) {
+                            if (reportErrors) {
+                                reportIncompatibleError(ts.Diagnostics.Types_of_property_0_are_incompatible, symbolToString(prop));
+                            }
+                            return { value: true };
                         }
                     }
-                    flow = flow.antecedent;
+                };
+                for (var _i = 0, _a = getPropertiesOfType(source); _i < _a.length; _i++) {
+                    var prop = _a[_i];
+                    var state_5 = _loop_16(prop);
+                    if (typeof state_5 === "object")
+                        return state_5.value;
                 }
-                else if (flags & 4 /* BranchLabel */) {
-                    // A branching point is reachable if any branch is reachable.
-                    return ts.some(flow.antecedents, function (f) { return isReachableFlowNodeWorker(f, /*noCacheCheck*/ false); });
+                return false;
+            }
+            function shouldCheckAsExcessProperty(prop, container) {
+                return prop.valueDeclaration && container.valueDeclaration && prop.valueDeclaration.parent === container.valueDeclaration;
+            }
+            function eachTypeRelatedToSomeType(source, target) {
+                var result = -1 /* True */;
+                var sourceTypes = source.types;
+                for (var _i = 0, sourceTypes_1 = sourceTypes; _i < sourceTypes_1.length; _i++) {
+                    var sourceType = sourceTypes_1[_i];
+                    var related = typeRelatedToSomeType(sourceType, target, /*reportErrors*/ false);
+                    if (!related) {
+                        return 0 /* False */;
+                    }
+                    result &= related;
                 }
-                else if (flags & 8 /* LoopLabel */) {
-                    // A loop is reachable if the control flow path that leads to the top is reachable.
-                    flow = flow.antecedents[0];
+                return result;
+            }
+            function typeRelatedToSomeType(source, target, reportErrors) {
+                var targetTypes = target.types;
+                if (target.flags & 1048576 /* Union */ && containsType(targetTypes, source)) {
+                    return -1 /* True */;
                 }
-                else if (flags & 128 /* SwitchClause */) {
-                    // The control flow path representing an unmatched value in a switch statement with
-                    // no default clause is unreachable if the switch statement is exhaustive.
-                    if (flow.clauseStart === flow.clauseEnd && isExhaustiveSwitchStatement(flow.switchStatement)) {
-                        return false;
+                for (var _i = 0, targetTypes_1 = targetTypes; _i < targetTypes_1.length; _i++) {
+                    var type = targetTypes_1[_i];
+                    var related = isRelatedTo(source, type, /*reportErrors*/ false);
+                    if (related) {
+                        return related;
                     }
-                    flow = flow.antecedent;
-                }
-                else if (flags & 1024 /* ReduceLabel */) {
-                    // Cache is unreliable once we start adjusting labels
-                    lastFlowNode = undefined;
-                    var target = flow.target;
-                    var saveAntecedents = target.antecedents;
-                    target.antecedents = flow.antecedents;
-                    var result = isReachableFlowNodeWorker(flow.antecedent, /*noCacheCheck*/ false);
-                    target.antecedents = saveAntecedents;
-                    return result;
                 }
-                else {
-                    return !(flags & 1 /* Unreachable */);
+                if (reportErrors) {
+                    var bestMatchingType = getBestMatchingType(source, target, isRelatedTo);
+                    isRelatedTo(source, bestMatchingType || targetTypes[targetTypes.length - 1], /*reportErrors*/ true);
                 }
+                return 0 /* False */;
             }
-        }
-        function getFlowTypeOfReference(reference, declaredType, initialType, flowContainer, couldBeUninitialized) {
-            if (initialType === void 0) { initialType = declaredType; }
-            var key;
-            var keySet = false;
-            var flowDepth = 0;
-            if (flowAnalysisDisabled) {
-                return errorType;
-            }
-            if (!reference.flowNode || !couldBeUninitialized && !(declaredType.flags & 133970943 /* Narrowable */)) {
-                return declaredType;
+            function typeRelatedToEachType(source, target, reportErrors, intersectionState) {
+                var result = -1 /* True */;
+                var targetTypes = target.types;
+                for (var _i = 0, targetTypes_2 = targetTypes; _i < targetTypes_2.length; _i++) {
+                    var targetType = targetTypes_2[_i];
+                    var related = isRelatedTo(source, targetType, reportErrors, /*headMessage*/ undefined, intersectionState);
+                    if (!related) {
+                        return 0 /* False */;
+                    }
+                    result &= related;
+                }
+                return result;
             }
-            flowInvocationCount++;
-            var sharedFlowStart = sharedFlowCount;
-            var evolvedType = getTypeFromFlowType(getTypeAtFlowNode(reference.flowNode));
-            sharedFlowCount = sharedFlowStart;
-            // When the reference is 'x' in an 'x.length', 'x.push(value)', 'x.unshift(value)' or x[n] = value' operation,
-            // we give type 'any[]' to 'x' instead of using the type determined by control flow analysis such that operations
-            // on empty arrays are possible without implicit any errors and new element types can be inferred without
-            // type mismatch errors.
-            var resultType = ts.getObjectFlags(evolvedType) & 256 /* EvolvingArray */ && isEvolvingArrayOperationTarget(reference) ? autoArrayType : finalizeEvolvingArrayType(evolvedType);
-            if (resultType === unreachableNeverType || reference.parent && reference.parent.kind === 218 /* NonNullExpression */ && getTypeWithFacts(resultType, 2097152 /* NEUndefinedOrNull */).flags & 131072 /* Never */) {
-                return declaredType;
+            function someTypeRelatedToType(source, target, reportErrors, intersectionState) {
+                var sourceTypes = source.types;
+                if (source.flags & 1048576 /* Union */ && containsType(sourceTypes, target)) {
+                    return -1 /* True */;
+                }
+                var len = sourceTypes.length;
+                for (var i = 0; i < len; i++) {
+                    var related = isRelatedTo(sourceTypes[i], target, reportErrors && i === len - 1, /*headMessage*/ undefined, intersectionState);
+                    if (related) {
+                        return related;
+                    }
+                }
+                return 0 /* False */;
             }
-            return resultType;
-            function getOrSetCacheKey() {
-                if (keySet) {
-                    return key;
+            function eachTypeRelatedToType(source, target, reportErrors, intersectionState) {
+                var result = -1 /* True */;
+                var sourceTypes = source.types;
+                for (var i = 0; i < sourceTypes.length; i++) {
+                    var sourceType = sourceTypes[i];
+                    if (target.flags & 1048576 /* Union */ && target.types.length === sourceTypes.length) {
+                        // many unions are mappings of one another; in such cases, simply comparing members at the same index can shortcut the comparison
+                        var related_1 = isRelatedTo(sourceType, target.types[i], /*reportErrors*/ false, /*headMessage*/ undefined, intersectionState);
+                        if (related_1) {
+                            result &= related_1;
+                            continue;
+                        }
+                    }
+                    var related = isRelatedTo(sourceType, target, reportErrors, /*headMessage*/ undefined, intersectionState);
+                    if (!related) {
+                        return 0 /* False */;
+                    }
+                    result &= related;
                 }
-                keySet = true;
-                return key = getFlowCacheKey(reference, declaredType, initialType, flowContainer);
+                return result;
             }
-            function getTypeAtFlowNode(flow) {
-                if (flowDepth === 2000) {
-                    // We have made 2000 recursive invocations. To avoid overflowing the call stack we report an error
-                    // and disable further control flow analysis in the containing function or module body.
-                    flowAnalysisDisabled = true;
-                    reportFlowControlError(reference);
-                    return errorType;
+            function typeArgumentsRelatedTo(sources, targets, variances, reportErrors, intersectionState) {
+                if (sources === void 0) { sources = ts.emptyArray; }
+                if (targets === void 0) { targets = ts.emptyArray; }
+                if (variances === void 0) { variances = ts.emptyArray; }
+                if (sources.length !== targets.length && relation === identityRelation) {
+                    return 0 /* False */;
                 }
-                flowDepth++;
-                while (true) {
-                    var flags = flow.flags;
-                    if (flags & 4096 /* Shared */) {
-                        // We cache results of flow type resolution for shared nodes that were previously visited in
-                        // the same getFlowTypeOfReference invocation. A node is considered shared when it is the
-                        // antecedent of more than one node.
-                        for (var i = sharedFlowStart; i < sharedFlowCount; i++) {
-                            if (sharedFlowNodes[i] === flow) {
-                                flowDepth--;
-                                return sharedFlowTypes[i];
+                var length = sources.length <= targets.length ? sources.length : targets.length;
+                var result = -1 /* True */;
+                for (var i = 0; i < length; i++) {
+                    // When variance information isn't available we default to covariance. This happens
+                    // in the process of computing variance information for recursive types and when
+                    // comparing 'this' type arguments.
+                    var varianceFlags = i < variances.length ? variances[i] : 1 /* Covariant */;
+                    var variance = varianceFlags & 7 /* VarianceMask */;
+                    // We ignore arguments for independent type parameters (because they're never witnessed).
+                    if (variance !== 4 /* Independent */) {
+                        var s = sources[i];
+                        var t = targets[i];
+                        var related = -1 /* True */;
+                        if (varianceFlags & 8 /* Unmeasurable */) {
+                            // Even an `Unmeasurable` variance works out without a structural check if the source and target are _identical_.
+                            // We can't simply assume invariance, because `Unmeasurable` marks nonlinear relations, for example, a relation tained by
+                            // the `-?` modifier in a mapped type (where, no matter how the inputs are related, the outputs still might not be)
+                            related = relation === identityRelation ? isRelatedTo(s, t, /*reportErrors*/ false) : compareTypesIdentical(s, t);
+                        }
+                        else if (variance === 1 /* Covariant */) {
+                            related = isRelatedTo(s, t, reportErrors, /*headMessage*/ undefined, intersectionState);
+                        }
+                        else if (variance === 2 /* Contravariant */) {
+                            related = isRelatedTo(t, s, reportErrors, /*headMessage*/ undefined, intersectionState);
+                        }
+                        else if (variance === 3 /* Bivariant */) {
+                            // In the bivariant case we first compare contravariantly without reporting
+                            // errors. Then, if that doesn't succeed, we compare covariantly with error
+                            // reporting. Thus, error elaboration will be based on the the covariant check,
+                            // which is generally easier to reason about.
+                            related = isRelatedTo(t, s, /*reportErrors*/ false);
+                            if (!related) {
+                                related = isRelatedTo(s, t, reportErrors, /*headMessage*/ undefined, intersectionState);
                             }
                         }
-                    }
-                    var type = void 0;
-                    if (flags & 16 /* Assignment */) {
-                        type = getTypeAtFlowAssignment(flow);
-                        if (!type) {
-                            flow = flow.antecedent;
-                            continue;
+                        else {
+                            // In the invariant case we first compare covariantly, and only when that
+                            // succeeds do we proceed to compare contravariantly. Thus, error elaboration
+                            // will typically be based on the covariant check.
+                            related = isRelatedTo(s, t, reportErrors, /*headMessage*/ undefined, intersectionState);
+                            if (related) {
+                                related &= isRelatedTo(t, s, reportErrors, /*headMessage*/ undefined, intersectionState);
+                            }
                         }
-                    }
-                    else if (flags & 512 /* Call */) {
-                        type = getTypeAtFlowCall(flow);
-                        if (!type) {
-                            flow = flow.antecedent;
-                            continue;
+                        if (!related) {
+                            return 0 /* False */;
                         }
+                        result &= related;
                     }
-                    else if (flags & 96 /* Condition */) {
-                        type = getTypeAtFlowCondition(flow);
-                    }
-                    else if (flags & 128 /* SwitchClause */) {
-                        type = getTypeAtSwitchClause(flow);
+                }
+                return result;
+            }
+            // Determine if possibly recursive types are related. First, check if the result is already available in the global cache.
+            // Second, check if we have already started a comparison of the given two types in which case we assume the result to be true.
+            // Third, check if both types are part of deeply nested chains of generic type instantiations and if so assume the types are
+            // equal and infinitely expanding. Fourth, if we have reached a depth of 100 nested comparisons, assume we have runaway recursion
+            // and issue an error. Otherwise, actually compare the structure of the two types.
+            function recursiveTypeRelatedTo(source, target, reportErrors, intersectionState) {
+                if (overflow) {
+                    return 0 /* False */;
+                }
+                var id = getRelationKey(source, target, intersectionState | (inPropertyCheck ? 8 /* InPropertyCheck */ : 0), relation);
+                var entry = relation.get(id);
+                if (entry !== undefined) {
+                    if (reportErrors && entry & 2 /* Failed */ && !(entry & 4 /* Reported */)) {
+                        // We are elaborating errors and the cached result is an unreported failure. The result will be reported
+                        // as a failure, and should be updated as a reported failure by the bottom of this function.
                     }
-                    else if (flags & 12 /* Label */) {
-                        if (flow.antecedents.length === 1) {
-                            flow = flow.antecedents[0];
-                            continue;
+                    else {
+                        if (outofbandVarianceMarkerHandler) {
+                            // We're in the middle of variance checking - integrate any unmeasurable/unreliable flags from this cached component
+                            var saved = entry & 24 /* ReportsMask */;
+                            if (saved & 8 /* ReportsUnmeasurable */) {
+                                instantiateType(source, makeFunctionTypeMapper(reportUnmeasurableMarkers));
+                            }
+                            if (saved & 16 /* ReportsUnreliable */) {
+                                instantiateType(source, makeFunctionTypeMapper(reportUnreliableMarkers));
+                            }
                         }
-                        type = flags & 4 /* BranchLabel */ ?
-                            getTypeAtFlowBranchLabel(flow) :
-                            getTypeAtFlowLoopLabel(flow);
+                        return entry & 1 /* Succeeded */ ? -1 /* True */ : 0 /* False */;
                     }
-                    else if (flags & 256 /* ArrayMutation */) {
-                        type = getTypeAtFlowArrayMutation(flow);
-                        if (!type) {
-                            flow = flow.antecedent;
-                            continue;
+                }
+                if (!maybeKeys) {
+                    maybeKeys = [];
+                    sourceStack = [];
+                    targetStack = [];
+                }
+                else {
+                    for (var i = 0; i < maybeCount; i++) {
+                        // If source and target are already being compared, consider them related with assumptions
+                        if (id === maybeKeys[i]) {
+                            return 3 /* Maybe */;
                         }
                     }
-                    else if (flags & 1024 /* ReduceLabel */) {
-                        var target = flow.target;
-                        var saveAntecedents = target.antecedents;
-                        target.antecedents = flow.antecedents;
-                        type = getTypeAtFlowNode(flow.antecedent);
-                        target.antecedents = saveAntecedents;
+                    if (depth === 100) {
+                        overflow = true;
+                        return 0 /* False */;
                     }
-                    else if (flags & 2 /* Start */) {
-                        // Check if we should continue with the control flow of the containing function.
-                        var container = flow.node;
-                        if (container && container !== flowContainer &&
-                            reference.kind !== 194 /* PropertyAccessExpression */ &&
-                            reference.kind !== 195 /* ElementAccessExpression */ &&
-                            reference.kind !== 104 /* ThisKeyword */) {
-                            flow = container.flowNode;
-                            continue;
+                }
+                var maybeStart = maybeCount;
+                maybeKeys[maybeCount] = id;
+                maybeCount++;
+                sourceStack[depth] = source;
+                targetStack[depth] = target;
+                depth++;
+                var saveExpandingFlags = expandingFlags;
+                if (!(expandingFlags & 1 /* Source */) && isDeeplyNestedType(source, sourceStack, depth))
+                    expandingFlags |= 1 /* Source */;
+                if (!(expandingFlags & 2 /* Target */) && isDeeplyNestedType(target, targetStack, depth))
+                    expandingFlags |= 2 /* Target */;
+                var originalHandler;
+                var propagatingVarianceFlags = 0;
+                if (outofbandVarianceMarkerHandler) {
+                    originalHandler = outofbandVarianceMarkerHandler;
+                    outofbandVarianceMarkerHandler = function (onlyUnreliable) {
+                        propagatingVarianceFlags |= onlyUnreliable ? 16 /* ReportsUnreliable */ : 8 /* ReportsUnmeasurable */;
+                        return originalHandler(onlyUnreliable);
+                    };
+                }
+                if (expandingFlags === 3 /* Both */) {
+                    ts.tracing.instant("check" /* Check */, "recursiveTypeRelatedTo_DepthLimit", {
+                        sourceId: source.id,
+                        sourceIdStack: sourceStack.map(function (t) { return t.id; }),
+                        targetId: target.id,
+                        targetIdStack: targetStack.map(function (t) { return t.id; }),
+                        depth: depth,
+                    });
+                }
+                var result = expandingFlags !== 3 /* Both */ ? structuredTypeRelatedTo(source, target, reportErrors, intersectionState) : 3 /* Maybe */;
+                if (outofbandVarianceMarkerHandler) {
+                    outofbandVarianceMarkerHandler = originalHandler;
+                }
+                expandingFlags = saveExpandingFlags;
+                depth--;
+                if (result) {
+                    if (result === -1 /* True */ || depth === 0) {
+                        if (result === -1 /* True */ || result === 3 /* Maybe */) {
+                            // If result is definitely true, record all maybe keys as having succeeded. Also, record Ternary.Maybe
+                            // results as having succeeded once we reach depth 0, but never record Ternary.Unknown results.
+                            for (var i = maybeStart; i < maybeCount; i++) {
+                                relation.set(maybeKeys[i], 1 /* Succeeded */ | propagatingVarianceFlags);
+                            }
                         }
-                        // At the top of the flow we have the initial type.
-                        type = initialType;
-                    }
-                    else {
-                        // Unreachable code errors are reported in the binding phase. Here we
-                        // simply return the non-auto declared type to reduce follow-on errors.
-                        type = convertAutoToAny(declaredType);
-                    }
-                    if (flags & 4096 /* Shared */) {
-                        // Record visited node and the associated type in the cache.
-                        sharedFlowNodes[sharedFlowCount] = flow;
-                        sharedFlowTypes[sharedFlowCount] = type;
-                        sharedFlowCount++;
+                        maybeCount = maybeStart;
                     }
-                    flowDepth--;
-                    return type;
                 }
+                else {
+                    // A false result goes straight into global cache (when something is false under
+                    // assumptions it will also be false without assumptions)
+                    relation.set(id, (reportErrors ? 4 /* Reported */ : 0) | 2 /* Failed */ | propagatingVarianceFlags);
+                    maybeCount = maybeStart;
+                }
+                return result;
             }
-            function getInitialOrAssignedType(flow) {
-                var node = flow.node;
-                return getConstraintForLocation(node.kind === 242 /* VariableDeclaration */ || node.kind === 191 /* BindingElement */ ?
-                    getInitialType(node) :
-                    getAssignedType(node), reference);
+            function structuredTypeRelatedTo(source, target, reportErrors, intersectionState) {
+                ts.tracing.push("check" /* Check */, "structuredTypeRelatedTo", { sourceId: source.id, targetId: target.id });
+                var result = structuredTypeRelatedToWorker(source, target, reportErrors, intersectionState);
+                ts.tracing.pop();
+                return result;
             }
-            function getTypeAtFlowAssignment(flow) {
-                var node = flow.node;
-                // Assignments only narrow the computed type if the declared type is a union type. Thus, we
-                // only need to evaluate the assigned type if the declared type is a union type.
-                if (isMatchingReference(reference, node)) {
-                    if (!isReachableFlowNode(flow)) {
-                        return unreachableNeverType;
+            function structuredTypeRelatedToWorker(source, target, reportErrors, intersectionState) {
+                if (intersectionState & 4 /* PropertyCheck */) {
+                    return propertiesRelatedTo(source, target, reportErrors, /*excludedProperties*/ undefined, 0 /* None */);
+                }
+                var flags = source.flags & target.flags;
+                if (relation === identityRelation && !(flags & 524288 /* Object */)) {
+                    if (flags & 4194304 /* Index */) {
+                        return isRelatedTo(source.type, target.type, /*reportErrors*/ false);
                     }
-                    if (ts.getAssignmentTargetKind(node) === 2 /* Compound */) {
-                        var flowType = getTypeAtFlowNode(flow.antecedent);
-                        return createFlowType(getBaseTypeOfLiteralType(getTypeFromFlowType(flowType)), isIncomplete(flowType));
+                    var result_7 = 0 /* False */;
+                    if (flags & 8388608 /* IndexedAccess */) {
+                        if (result_7 = isRelatedTo(source.objectType, target.objectType, /*reportErrors*/ false)) {
+                            if (result_7 &= isRelatedTo(source.indexType, target.indexType, /*reportErrors*/ false)) {
+                                return result_7;
+                            }
+                        }
                     }
-                    if (declaredType === autoType || declaredType === autoArrayType) {
-                        if (isEmptyArrayAssignment(node)) {
-                            return getEvolvingArrayType(neverType);
+                    if (flags & 16777216 /* Conditional */) {
+                        if (source.root.isDistributive === target.root.isDistributive) {
+                            if (result_7 = isRelatedTo(source.checkType, target.checkType, /*reportErrors*/ false)) {
+                                if (result_7 &= isRelatedTo(source.extendsType, target.extendsType, /*reportErrors*/ false)) {
+                                    if (result_7 &= isRelatedTo(getTrueTypeFromConditionalType(source), getTrueTypeFromConditionalType(target), /*reportErrors*/ false)) {
+                                        if (result_7 &= isRelatedTo(getFalseTypeFromConditionalType(source), getFalseTypeFromConditionalType(target), /*reportErrors*/ false)) {
+                                            return result_7;
+                                        }
+                                    }
+                                }
+                            }
                         }
-                        var assignedType = getBaseTypeOfLiteralType(getInitialOrAssignedType(flow));
-                        return isTypeAssignableTo(assignedType, declaredType) ? assignedType : anyArrayType;
                     }
-                    if (declaredType.flags & 1048576 /* Union */) {
-                        return getAssignmentReducedType(declaredType, getInitialOrAssignedType(flow));
+                    if (flags & 33554432 /* Substitution */) {
+                        return isRelatedTo(source.substitute, target.substitute, /*reportErrors*/ false);
                     }
-                    return declaredType;
+                    return 0 /* False */;
                 }
-                // We didn't have a direct match. However, if the reference is a dotted name, this
-                // may be an assignment to a left hand part of the reference. For example, for a
-                // reference 'x.y.z', we may be at an assignment to 'x.y' or 'x'. In that case,
-                // return the declared type.
-                if (containsMatchingReference(reference, node)) {
-                    if (!isReachableFlowNode(flow)) {
-                        return unreachableNeverType;
+                var result;
+                var originalErrorInfo;
+                var varianceCheckFailed = false;
+                var saveErrorInfo = captureErrorCalculationState();
+                // We limit alias variance probing to only object and conditional types since their alias behavior
+                // is more predictable than other, interned types, which may or may not have an alias depending on
+                // the order in which things were checked.
+                if (source.flags & (524288 /* Object */ | 16777216 /* Conditional */) && source.aliasSymbol &&
+                    source.aliasTypeArguments && source.aliasSymbol === target.aliasSymbol &&
+                    !(source.aliasTypeArgumentsContainsMarker || target.aliasTypeArgumentsContainsMarker)) {
+                    var variances = getAliasVariances(source.aliasSymbol);
+                    if (variances === ts.emptyArray) {
+                        return 1 /* Unknown */;
                     }
-                    // A matching dotted name might also be an expando property on a function *expression*,
-                    // in which case we continue control flow analysis back to the function's declaration
-                    if (ts.isVariableDeclaration(node) && (ts.isInJSFile(node) || ts.isVarConst(node))) {
-                        var init = ts.getDeclaredExpandoInitializer(node);
-                        if (init && (init.kind === 201 /* FunctionExpression */ || init.kind === 202 /* ArrowFunction */)) {
-                            return getTypeAtFlowNode(flow.antecedent);
-                        }
+                    var varianceResult = relateVariances(source.aliasTypeArguments, target.aliasTypeArguments, variances, intersectionState);
+                    if (varianceResult !== undefined) {
+                        return varianceResult;
                     }
-                    return declaredType;
                 }
-                // for (const _ in ref) acts as a nonnull on ref
-                if (ts.isVariableDeclaration(node) && node.parent.parent.kind === 231 /* ForInStatement */ && isMatchingReference(reference, node.parent.parent.expression)) {
-                    return getNonNullableTypeIfNeeded(getTypeFromFlowType(getTypeAtFlowNode(flow.antecedent)));
+                // For a generic type T and a type U that is assignable to T, [...U] is assignable to T, U is assignable to readonly [...T],
+                // and U is assignable to [...T] when U is constrained to a mutable array or tuple type.
+                if (isSingleElementGenericTupleType(source) && !source.target.readonly && (result = isRelatedTo(getTypeArguments(source)[0], target)) ||
+                    isSingleElementGenericTupleType(target) && (target.target.readonly || isMutableArrayOrTuple(getBaseConstraintOfType(source) || source)) && (result = isRelatedTo(source, getTypeArguments(target)[0]))) {
+                    return result;
                 }
-                // Assignment doesn't affect reference
-                return undefined;
-            }
-            function narrowTypeByAssertion(type, expr) {
-                var node = ts.skipParentheses(expr);
-                if (node.kind === 91 /* FalseKeyword */) {
-                    return unreachableNeverType;
+                if (target.flags & 262144 /* TypeParameter */) {
+                    // A source type { [P in Q]: X } is related to a target type T if keyof T is related to Q and X is related to T[Q].
+                    if (ts.getObjectFlags(source) & 32 /* Mapped */ && !source.declaration.nameType && isRelatedTo(getIndexType(target), getConstraintTypeFromMappedType(source))) {
+                        if (!(getMappedTypeModifiers(source) & 4 /* IncludeOptional */)) {
+                            var templateType = getTemplateTypeFromMappedType(source);
+                            var indexedAccessType = getIndexedAccessType(target, getTypeParameterFromMappedType(source));
+                            if (result = isRelatedTo(templateType, indexedAccessType, reportErrors)) {
+                                return result;
+                            }
+                        }
+                    }
                 }
-                if (node.kind === 209 /* BinaryExpression */) {
-                    if (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */) {
-                        return narrowTypeByAssertion(narrowTypeByAssertion(type, node.left), node.right);
+                else if (target.flags & 4194304 /* Index */) {
+                    var targetType = target.type;
+                    // A keyof S is related to a keyof T if T is related to S.
+                    if (source.flags & 4194304 /* Index */) {
+                        if (result = isRelatedTo(targetType, source.type, /*reportErrors*/ false)) {
+                            return result;
+                        }
                     }
-                    if (node.operatorToken.kind === 56 /* BarBarToken */) {
-                        return getUnionType([narrowTypeByAssertion(type, node.left), narrowTypeByAssertion(type, node.right)]);
+                    if (isTupleType(targetType)) {
+                        // An index type can have a tuple type target when the tuple type contains variadic elements.
+                        // Check if the source is related to the known keys of the tuple type.
+                        if (result = isRelatedTo(source, getKnownKeysOfTupleType(targetType), reportErrors)) {
+                            return result;
+                        }
                     }
-                }
-                return narrowType(type, node, /*assumeTrue*/ true);
-            }
-            function getTypeAtFlowCall(flow) {
-                var signature = getEffectsSignature(flow.node);
-                if (signature) {
-                    var predicate = getTypePredicateOfSignature(signature);
-                    if (predicate && (predicate.kind === 2 /* AssertsThis */ || predicate.kind === 3 /* AssertsIdentifier */)) {
-                        var flowType = getTypeAtFlowNode(flow.antecedent);
-                        var type = finalizeEvolvingArrayType(getTypeFromFlowType(flowType));
-                        var narrowedType = predicate.type ? narrowTypeByTypePredicate(type, predicate, flow.node, /*assumeTrue*/ true) :
-                            predicate.kind === 3 /* AssertsIdentifier */ && predicate.parameterIndex >= 0 && predicate.parameterIndex < flow.node.arguments.length ? narrowTypeByAssertion(type, flow.node.arguments[predicate.parameterIndex]) :
-                                type;
-                        return narrowedType === type ? flowType : createFlowType(narrowedType, isIncomplete(flowType));
+                    else {
+                        // A type S is assignable to keyof T if S is assignable to keyof C, where C is the
+                        // simplified form of T or, if T doesn't simplify, the constraint of T.
+                        var constraint = getSimplifiedTypeOrConstraint(targetType);
+                        if (constraint) {
+                            // We require Ternary.True here such that circular constraints don't cause
+                            // false positives. For example, given 'T extends { [K in keyof T]: string }',
+                            // 'keyof T' has itself as its constraint and produces a Ternary.Maybe when
+                            // related to other types.
+                            if (isRelatedTo(source, getIndexType(constraint, target.stringsOnly), reportErrors) === -1 /* True */) {
+                                return -1 /* True */;
+                            }
+                        }
                     }
-                    if (getReturnTypeOfSignature(signature).flags & 131072 /* Never */) {
-                        return unreachableNeverType;
+                }
+                else if (target.flags & 8388608 /* IndexedAccess */) {
+                    // A type S is related to a type T[K] if S is related to C, where C is the base
+                    // constraint of T[K] for writing.
+                    if (relation === assignableRelation || relation === comparableRelation) {
+                        var objectType = target.objectType;
+                        var indexType = target.indexType;
+                        var baseObjectType = getBaseConstraintOfType(objectType) || objectType;
+                        var baseIndexType = getBaseConstraintOfType(indexType) || indexType;
+                        if (!isGenericObjectType(baseObjectType) && !isGenericIndexType(baseIndexType)) {
+                            var accessFlags = 2 /* Writing */ | (baseObjectType !== objectType ? 1 /* NoIndexSignatures */ : 0);
+                            var constraint = getIndexedAccessTypeOrUndefined(baseObjectType, baseIndexType, target.noUncheckedIndexedAccessCandidate, /*accessNode*/ undefined, accessFlags);
+                            if (constraint && (result = isRelatedTo(source, constraint, reportErrors))) {
+                                return result;
+                            }
+                        }
                     }
                 }
-                return undefined;
-            }
-            function getTypeAtFlowArrayMutation(flow) {
-                if (declaredType === autoType || declaredType === autoArrayType) {
-                    var node = flow.node;
-                    var expr = node.kind === 196 /* CallExpression */ ?
-                        node.expression.expression :
-                        node.left.expression;
-                    if (isMatchingReference(reference, getReferenceCandidate(expr))) {
-                        var flowType = getTypeAtFlowNode(flow.antecedent);
-                        var type = getTypeFromFlowType(flowType);
-                        if (ts.getObjectFlags(type) & 256 /* EvolvingArray */) {
-                            var evolvedType_1 = type;
-                            if (node.kind === 196 /* CallExpression */) {
-                                for (var _i = 0, _a = node.arguments; _i < _a.length; _i++) {
-                                    var arg = _a[_i];
-                                    evolvedType_1 = addEvolvingArrayElementType(evolvedType_1, arg);
+                else if (isGenericMappedType(target) && !target.declaration.nameType) {
+                    // A source type T is related to a target type { [P in X]: T[P] }
+                    var template = getTemplateTypeFromMappedType(target);
+                    var modifiers = getMappedTypeModifiers(target);
+                    if (!(modifiers & 8 /* ExcludeOptional */)) {
+                        if (template.flags & 8388608 /* IndexedAccess */ && template.objectType === source &&
+                            template.indexType === getTypeParameterFromMappedType(target)) {
+                            return -1 /* True */;
+                        }
+                        if (!isGenericMappedType(source)) {
+                            var targetConstraint = getConstraintTypeFromMappedType(target);
+                            var sourceKeys = getIndexType(source, /*stringsOnly*/ undefined, /*noIndexSignatures*/ true);
+                            var includeOptional = modifiers & 4 /* IncludeOptional */;
+                            var filteredByApplicability = includeOptional ? intersectTypes(targetConstraint, sourceKeys) : undefined;
+                            // A source type T is related to a target type { [P in Q]: X } if Q is related to keyof T and T[Q] is related to X.
+                            // A source type T is related to a target type { [P in Q]?: X } if some constituent Q' of Q is related to keyof T and T[Q'] is related to X.
+                            if (includeOptional
+                                ? !(filteredByApplicability.flags & 131072 /* Never */)
+                                : isRelatedTo(targetConstraint, sourceKeys)) {
+                                var templateType = getTemplateTypeFromMappedType(target);
+                                var typeParameter = getTypeParameterFromMappedType(target);
+                                // Fastpath: When the template has the form Obj[P] where P is the mapped type parameter, directly compare `source` with `Obj`
+                                // to avoid creating the (potentially very large) number of new intermediate types made by manufacturing `source[P]`
+                                var nonNullComponent = extractTypesOfKind(templateType, ~98304 /* Nullable */);
+                                if (nonNullComponent.flags & 8388608 /* IndexedAccess */ && nonNullComponent.indexType === typeParameter) {
+                                    if (result = isRelatedTo(source, nonNullComponent.objectType, reportErrors)) {
+                                        return result;
+                                    }
                                 }
-                            }
-                            else {
-                                // We must get the context free expression type so as to not recur in an uncached fashion on the LHS (which causes exponential blowup in compile time)
-                                var indexType = getContextFreeTypeOfExpression(node.left.argumentExpression);
-                                if (isTypeAssignableToKind(indexType, 296 /* NumberLike */)) {
-                                    evolvedType_1 = addEvolvingArrayElementType(evolvedType_1, node.right);
+                                else {
+                                    var indexingType = filteredByApplicability ? getIntersectionType([filteredByApplicability, typeParameter]) : typeParameter;
+                                    var indexedAccessType = getIndexedAccessType(source, indexingType);
+                                    if (result = isRelatedTo(indexedAccessType, templateType, reportErrors)) {
+                                        return result;
+                                    }
                                 }
                             }
-                            return evolvedType_1 === type ? flowType : createFlowType(evolvedType_1, isIncomplete(flowType));
+                            originalErrorInfo = errorInfo;
+                            resetErrorInfo(saveErrorInfo);
                         }
-                        return flowType;
                     }
                 }
-                return undefined;
-            }
-            function getTypeAtFlowCondition(flow) {
-                var flowType = getTypeAtFlowNode(flow.antecedent);
-                var type = getTypeFromFlowType(flowType);
-                if (type.flags & 131072 /* Never */) {
-                    return flowType;
-                }
-                // If we have an antecedent type (meaning we're reachable in some way), we first
-                // attempt to narrow the antecedent type. If that produces the never type, and if
-                // the antecedent type is incomplete (i.e. a transient type in a loop), then we
-                // take the type guard as an indication that control *could* reach here once we
-                // have the complete type. We proceed by switching to the silent never type which
-                // doesn't report errors when operators are applied to it. Note that this is the
-                // *only* place a silent never type is ever generated.
-                var assumeTrue = (flow.flags & 32 /* TrueCondition */) !== 0;
-                var nonEvolvingType = finalizeEvolvingArrayType(type);
-                var narrowedType = narrowType(nonEvolvingType, flow.node, assumeTrue);
-                if (narrowedType === nonEvolvingType) {
-                    return flowType;
-                }
-                var incomplete = isIncomplete(flowType);
-                var resultType = incomplete && narrowedType.flags & 131072 /* Never */ ? silentNeverType : narrowedType;
-                return createFlowType(resultType, incomplete);
-            }
-            function getTypeAtSwitchClause(flow) {
-                var expr = flow.switchStatement.expression;
-                var flowType = getTypeAtFlowNode(flow.antecedent);
-                var type = getTypeFromFlowType(flowType);
-                if (isMatchingReference(reference, expr)) {
-                    type = narrowTypeBySwitchOnDiscriminant(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd);
-                }
-                else if (expr.kind === 204 /* TypeOfExpression */ && isMatchingReference(reference, expr.expression)) {
-                    type = narrowBySwitchOnTypeOf(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd);
+                else if (target.flags & 134217728 /* TemplateLiteral */ && source.flags & 128 /* StringLiteral */) {
+                    if (isPatternLiteralType(target)) {
+                        // match all non-`string` segments
+                        var result_8 = inferLiteralsFromTemplateLiteralType(source, target);
+                        if (result_8 && ts.every(result_8, function (r, i) { return isStringLiteralTypeValueParsableAsType(r, target.types[i]); })) {
+                            return -1 /* True */;
+                        }
+                    }
                 }
-                else {
-                    if (strictNullChecks) {
-                        if (optionalChainContainsReference(expr, reference)) {
-                            type = narrowTypeBySwitchOptionalChainContainment(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd, function (t) { return !(t.flags & (32768 /* Undefined */ | 131072 /* Never */)); });
+                if (source.flags & 8650752 /* TypeVariable */) {
+                    if (source.flags & 8388608 /* IndexedAccess */ && target.flags & 8388608 /* IndexedAccess */) {
+                        // A type S[K] is related to a type T[J] if S is related to T and K is related to J.
+                        if (result = isRelatedTo(source.objectType, target.objectType, reportErrors)) {
+                            result &= isRelatedTo(source.indexType, target.indexType, reportErrors);
+                        }
+                        if (result) {
+                            resetErrorInfo(saveErrorInfo);
+                            return result;
+                        }
+                    }
+                    else {
+                        var constraint = getConstraintOfType(source);
+                        if (!constraint || (source.flags & 262144 /* TypeParameter */ && constraint.flags & 1 /* Any */)) {
+                            // A type variable with no constraint is not related to the non-primitive object type.
+                            if (result = isRelatedTo(emptyObjectType, extractTypesOfKind(target, ~67108864 /* NonPrimitive */))) {
+                                resetErrorInfo(saveErrorInfo);
+                                return result;
+                            }
+                        }
+                        // hi-speed no-this-instantiation check (less accurate, but avoids costly `this`-instantiation when the constraint will suffice), see #28231 for report on why this is needed
+                        else if (result = isRelatedTo(constraint, target, /*reportErrors*/ false, /*headMessage*/ undefined, intersectionState)) {
+                            resetErrorInfo(saveErrorInfo);
+                            return result;
                         }
-                        else if (expr.kind === 204 /* TypeOfExpression */ && optionalChainContainsReference(expr.expression, reference)) {
-                            type = narrowTypeBySwitchOptionalChainContainment(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd, function (t) { return !(t.flags & 131072 /* Never */ || t.flags & 128 /* StringLiteral */ && t.value === "undefined"); });
+                        // slower, fuller, this-instantiated check (necessary when comparing raw `this` types from base classes), see `subclassWithPolymorphicThisIsAssignable.ts` test for example
+                        else if (result = isRelatedTo(getTypeWithThisArgument(constraint, source), target, reportErrors, /*headMessage*/ undefined, intersectionState)) {
+                            resetErrorInfo(saveErrorInfo);
+                            return result;
                         }
                     }
-                    if (isMatchingReferenceDiscriminant(expr, type)) {
-                        type = narrowTypeByDiscriminant(type, expr, function (t) { return narrowTypeBySwitchOnDiscriminant(t, flow.switchStatement, flow.clauseStart, flow.clauseEnd); });
+                }
+                else if (source.flags & 4194304 /* Index */) {
+                    if (result = isRelatedTo(keyofConstraintType, target, reportErrors)) {
+                        resetErrorInfo(saveErrorInfo);
+                        return result;
                     }
                 }
-                return createFlowType(type, isIncomplete(flowType));
-            }
-            function getTypeAtFlowBranchLabel(flow) {
-                var antecedentTypes = [];
-                var subtypeReduction = false;
-                var seenIncomplete = false;
-                var bypassFlow;
-                for (var _i = 0, _a = flow.antecedents; _i < _a.length; _i++) {
-                    var antecedent = _a[_i];
-                    if (!bypassFlow && antecedent.flags & 128 /* SwitchClause */ && antecedent.clauseStart === antecedent.clauseEnd) {
-                        // The antecedent is the bypass branch of a potentially exhaustive switch statement.
-                        bypassFlow = antecedent;
-                        continue;
+                else if (source.flags & 134217728 /* TemplateLiteral */) {
+                    if (target.flags & 134217728 /* TemplateLiteral */ &&
+                        source.texts.length === target.texts.length &&
+                        source.types.length === target.types.length &&
+                        ts.every(source.texts, function (t, i) { return t === target.texts[i]; }) &&
+                        ts.every(instantiateType(source, makeFunctionTypeMapper(reportUnreliableMarkers)).types, function (t, i) { return !!(target.types[i].flags & (1 /* Any */ | 4 /* String */)) || !!isRelatedTo(t, target.types[i], /*reportErrors*/ false); })) {
+                        return -1 /* True */;
                     }
-                    var flowType = getTypeAtFlowNode(antecedent);
-                    var type = getTypeFromFlowType(flowType);
-                    // If the type at a particular antecedent path is the declared type and the
-                    // reference is known to always be assigned (i.e. when declared and initial types
-                    // are the same), there is no reason to process more antecedents since the only
-                    // possible outcome is subtypes that will be removed in the final union type anyway.
-                    if (type === declaredType && declaredType === initialType) {
-                        return type;
+                    var constraint = getBaseConstraintOfType(source);
+                    if (constraint && constraint !== source && (result = isRelatedTo(constraint, target, reportErrors))) {
+                        resetErrorInfo(saveErrorInfo);
+                        return result;
                     }
-                    ts.pushIfUnique(antecedentTypes, type);
-                    // If an antecedent type is not a subset of the declared type, we need to perform
-                    // subtype reduction. This happens when a "foreign" type is injected into the control
-                    // flow using the instanceof operator or a user defined type predicate.
-                    if (!isTypeSubsetOf(type, declaredType)) {
-                        subtypeReduction = true;
+                }
+                else if (source.flags & 268435456 /* StringMapping */) {
+                    if (target.flags & 268435456 /* StringMapping */ && source.symbol === target.symbol) {
+                        if (result = isRelatedTo(source.type, target.type, reportErrors)) {
+                            resetErrorInfo(saveErrorInfo);
+                            return result;
+                        }
                     }
-                    if (isIncomplete(flowType)) {
-                        seenIncomplete = true;
+                    else {
+                        var constraint = getBaseConstraintOfType(source);
+                        if (constraint && (result = isRelatedTo(constraint, target, reportErrors))) {
+                            resetErrorInfo(saveErrorInfo);
+                            return result;
+                        }
                     }
                 }
-                if (bypassFlow) {
-                    var flowType = getTypeAtFlowNode(bypassFlow);
-                    var type = getTypeFromFlowType(flowType);
-                    // If the bypass flow contributes a type we haven't seen yet and the switch statement
-                    // isn't exhaustive, process the bypass flow type. Since exhaustiveness checks increase
-                    // the risk of circularities, we only want to perform them when they make a difference.
-                    if (!ts.contains(antecedentTypes, type) && !isExhaustiveSwitchStatement(bypassFlow.switchStatement)) {
-                        if (type === declaredType && declaredType === initialType) {
-                            return type;
+                else if (source.flags & 16777216 /* Conditional */) {
+                    if (target.flags & 16777216 /* Conditional */) {
+                        // Two conditional types 'T1 extends U1 ? X1 : Y1' and 'T2 extends U2 ? X2 : Y2' are related if
+                        // one of T1 and T2 is related to the other, U1 and U2 are identical types, X1 is related to X2,
+                        // and Y1 is related to Y2.
+                        var sourceParams = source.root.inferTypeParameters;
+                        var sourceExtends = source.extendsType;
+                        var mapper = void 0;
+                        if (sourceParams) {
+                            // If the source has infer type parameters, we instantiate them in the context of the target
+                            var ctx = createInferenceContext(sourceParams, /*signature*/ undefined, 0 /* None */, isRelatedTo);
+                            inferTypes(ctx.inferences, target.extendsType, sourceExtends, 256 /* NoConstraints */ | 512 /* AlwaysStrict */);
+                            sourceExtends = instantiateType(sourceExtends, ctx.mapper);
+                            mapper = ctx.mapper;
                         }
-                        antecedentTypes.push(type);
-                        if (!isTypeSubsetOf(type, declaredType)) {
-                            subtypeReduction = true;
+                        if (isTypeIdenticalTo(sourceExtends, target.extendsType) &&
+                            (isRelatedTo(source.checkType, target.checkType) || isRelatedTo(target.checkType, source.checkType))) {
+                            if (result = isRelatedTo(instantiateType(getTrueTypeFromConditionalType(source), mapper), getTrueTypeFromConditionalType(target), reportErrors)) {
+                                result &= isRelatedTo(getFalseTypeFromConditionalType(source), getFalseTypeFromConditionalType(target), reportErrors);
+                            }
+                            if (result) {
+                                resetErrorInfo(saveErrorInfo);
+                                return result;
+                            }
                         }
-                        if (isIncomplete(flowType)) {
-                            seenIncomplete = true;
+                    }
+                    else {
+                        // conditionals aren't related to one another via distributive constraint as it is much too inaccurate and allows way
+                        // more assignments than are desirable (since it maps the source check type to its constraint, it loses information)
+                        var distributiveConstraint = getConstraintOfDistributiveConditionalType(source);
+                        if (distributiveConstraint) {
+                            if (result = isRelatedTo(distributiveConstraint, target, reportErrors)) {
+                                resetErrorInfo(saveErrorInfo);
+                                return result;
+                            }
                         }
                     }
-                }
-                return createFlowType(getUnionOrEvolvingArrayType(antecedentTypes, subtypeReduction ? 2 /* Subtype */ : 1 /* Literal */), seenIncomplete);
-            }
-            function getTypeAtFlowLoopLabel(flow) {
-                // If we have previously computed the control flow type for the reference at
-                // this flow loop junction, return the cached type.
-                var id = getFlowNodeId(flow);
-                var cache = flowLoopCaches[id] || (flowLoopCaches[id] = ts.createMap());
-                var key = getOrSetCacheKey();
-                if (!key) {
-                    // No cache key is generated when binding patterns are in unnarrowable situations
-                    return declaredType;
-                }
-                var cached = cache.get(key);
-                if (cached) {
-                    return cached;
-                }
-                // If this flow loop junction and reference are already being processed, return
-                // the union of the types computed for each branch so far, marked as incomplete.
-                // It is possible to see an empty array in cases where loops are nested and the
-                // back edge of the outer loop reaches an inner loop that is already being analyzed.
-                // In such cases we restart the analysis of the inner loop, which will then see
-                // a non-empty in-process array for the outer loop and eventually terminate because
-                // the first antecedent of a loop junction is always the non-looping control flow
-                // path that leads to the top.
-                for (var i = flowLoopStart; i < flowLoopCount; i++) {
-                    if (flowLoopNodes[i] === flow && flowLoopKeys[i] === key && flowLoopTypes[i].length) {
-                        return createFlowType(getUnionOrEvolvingArrayType(flowLoopTypes[i], 1 /* Literal */), /*incomplete*/ true);
+                    // conditionals _can_ be related to one another via normal constraint, as, eg, `A extends B ? O : never` should be assignable to `O`
+                    // when `O` is a conditional (`never` is trivially aissgnable to `O`, as is `O`!).
+                    var defaultConstraint = getDefaultConstraintOfConditionalType(source);
+                    if (defaultConstraint) {
+                        if (result = isRelatedTo(defaultConstraint, target, reportErrors)) {
+                            resetErrorInfo(saveErrorInfo);
+                            return result;
+                        }
                     }
                 }
-                // Add the flow loop junction and reference to the in-process stack and analyze
-                // each antecedent code path.
-                var antecedentTypes = [];
-                var subtypeReduction = false;
-                var firstAntecedentType;
-                for (var _i = 0, _a = flow.antecedents; _i < _a.length; _i++) {
-                    var antecedent = _a[_i];
-                    var flowType = void 0;
-                    if (!firstAntecedentType) {
-                        // The first antecedent of a loop junction is always the non-looping control
-                        // flow path that leads to the top.
-                        flowType = firstAntecedentType = getTypeAtFlowNode(antecedent);
+                else {
+                    // An empty object type is related to any mapped type that includes a '?' modifier.
+                    if (relation !== subtypeRelation && relation !== strictSubtypeRelation && isPartialMappedType(target) && isEmptyObjectType(source)) {
+                        return -1 /* True */;
                     }
-                    else {
-                        // All but the first antecedent are the looping control flow paths that lead
-                        // back to the loop junction. We track these on the flow loop stack.
-                        flowLoopNodes[flowLoopCount] = flow;
-                        flowLoopKeys[flowLoopCount] = key;
-                        flowLoopTypes[flowLoopCount] = antecedentTypes;
-                        flowLoopCount++;
-                        var saveFlowTypeCache = flowTypeCache;
-                        flowTypeCache = undefined;
-                        flowType = getTypeAtFlowNode(antecedent);
-                        flowTypeCache = saveFlowTypeCache;
-                        flowLoopCount--;
-                        // If we see a value appear in the cache it is a sign that control flow analysis
-                        // was restarted and completed by checkExpressionCached. We can simply pick up
-                        // the resulting type and bail out.
-                        var cached_1 = cache.get(key);
-                        if (cached_1) {
-                            return cached_1;
+                    if (isGenericMappedType(target)) {
+                        if (isGenericMappedType(source)) {
+                            if (result = mappedTypeRelatedTo(source, target, reportErrors)) {
+                                resetErrorInfo(saveErrorInfo);
+                                return result;
+                            }
                         }
+                        return 0 /* False */;
                     }
-                    var type = getTypeFromFlowType(flowType);
-                    ts.pushIfUnique(antecedentTypes, type);
-                    // If an antecedent type is not a subset of the declared type, we need to perform
-                    // subtype reduction. This happens when a "foreign" type is injected into the control
-                    // flow using the instanceof operator or a user defined type predicate.
-                    if (!isTypeSubsetOf(type, declaredType)) {
-                        subtypeReduction = true;
+                    var sourceIsPrimitive = !!(source.flags & 131068 /* Primitive */);
+                    if (relation !== identityRelation) {
+                        source = getApparentType(source);
                     }
-                    // If the type at a particular antecedent path is the declared type there is no
-                    // reason to process more antecedents since the only possible outcome is subtypes
-                    // that will be removed in the final union type anyway.
-                    if (type === declaredType) {
-                        break;
+                    else if (isGenericMappedType(source)) {
+                        return 0 /* False */;
                     }
-                }
-                // The result is incomplete if the first antecedent (the non-looping control flow path)
-                // is incomplete.
-                var result = getUnionOrEvolvingArrayType(antecedentTypes, subtypeReduction ? 2 /* Subtype */ : 1 /* Literal */);
-                if (isIncomplete(firstAntecedentType)) {
-                    return createFlowType(result, /*incomplete*/ true);
-                }
-                cache.set(key, result);
-                return result;
-            }
-            function isMatchingReferenceDiscriminant(expr, computedType) {
-                if (!(computedType.flags & 1048576 /* Union */) || !ts.isAccessExpression(expr)) {
-                    return false;
-                }
-                var name = getAccessedPropertyName(expr);
-                if (name === undefined) {
-                    return false;
-                }
-                return isMatchingReference(reference, expr.expression) && isDiscriminantProperty(computedType, name);
-            }
-            function narrowTypeByDiscriminant(type, access, narrowType) {
-                var propName = getAccessedPropertyName(access);
-                if (propName === undefined) {
-                    return type;
-                }
-                var propType = getTypeOfPropertyOfType(type, propName);
-                if (!propType) {
-                    return type;
-                }
-                var narrowedPropType = narrowType(propType);
-                return filterType(type, function (t) {
-                    var discriminantType = getTypeOfPropertyOrIndexSignature(t, propName);
-                    return !(discriminantType.flags & 131072 /* Never */) && isTypeComparableTo(discriminantType, narrowedPropType);
-                });
-            }
-            function narrowTypeByTruthiness(type, expr, assumeTrue) {
-                if (isMatchingReference(reference, expr)) {
-                    return getTypeWithFacts(type, assumeTrue ? 4194304 /* Truthy */ : 8388608 /* Falsy */);
-                }
-                if (strictNullChecks && assumeTrue && optionalChainContainsReference(expr, reference)) {
-                    type = getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
-                }
-                if (isMatchingReferenceDiscriminant(expr, declaredType)) {
-                    return narrowTypeByDiscriminant(type, expr, function (t) { return getTypeWithFacts(t, assumeTrue ? 4194304 /* Truthy */ : 8388608 /* Falsy */); });
-                }
-                return type;
-            }
-            function isTypePresencePossible(type, propName, assumeTrue) {
-                if (getIndexInfoOfType(type, 0 /* String */)) {
-                    return true;
-                }
-                var prop = getPropertyOfType(type, propName);
-                if (prop) {
-                    return prop.flags & 16777216 /* Optional */ ? true : assumeTrue;
-                }
-                return !assumeTrue;
-            }
-            function narrowByInKeyword(type, literal, assumeTrue) {
-                if (type.flags & (1048576 /* Union */ | 524288 /* Object */) || isThisTypeParameter(type)) {
-                    var propName_1 = ts.escapeLeadingUnderscores(literal.text);
-                    return filterType(type, function (t) { return isTypePresencePossible(t, propName_1, assumeTrue); });
-                }
-                return type;
-            }
-            function narrowTypeByBinaryExpression(type, expr, assumeTrue) {
-                switch (expr.operatorToken.kind) {
-                    case 62 /* EqualsToken */:
-                        return narrowTypeByTruthiness(narrowType(type, expr.right, assumeTrue), expr.left, assumeTrue);
-                    case 34 /* EqualsEqualsToken */:
-                    case 35 /* ExclamationEqualsToken */:
-                    case 36 /* EqualsEqualsEqualsToken */:
-                    case 37 /* ExclamationEqualsEqualsToken */:
-                        var operator_1 = expr.operatorToken.kind;
-                        var left_1 = getReferenceCandidate(expr.left);
-                        var right_1 = getReferenceCandidate(expr.right);
-                        if (left_1.kind === 204 /* TypeOfExpression */ && ts.isStringLiteralLike(right_1)) {
-                            return narrowTypeByTypeof(type, left_1, operator_1, right_1, assumeTrue);
+                    if (ts.getObjectFlags(source) & 4 /* Reference */ && ts.getObjectFlags(target) & 4 /* Reference */ && source.target === target.target &&
+                        !(ts.getObjectFlags(source) & 8192 /* MarkerType */ || ts.getObjectFlags(target) & 8192 /* MarkerType */)) {
+                        // We have type references to the same generic type, and the type references are not marker
+                        // type references (which are intended by be compared structurally). Obtain the variance
+                        // information for the type parameters and relate the type arguments accordingly.
+                        var variances = getVariances(source.target);
+                        // We return Ternary.Maybe for a recursive invocation of getVariances (signalled by emptyArray). This
+                        // effectively means we measure variance only from type parameter occurrences that aren't nested in
+                        // recursive instantiations of the generic type.
+                        if (variances === ts.emptyArray) {
+                            return 1 /* Unknown */;
                         }
-                        if (right_1.kind === 204 /* TypeOfExpression */ && ts.isStringLiteralLike(left_1)) {
-                            return narrowTypeByTypeof(type, right_1, operator_1, left_1, assumeTrue);
+                        var varianceResult = relateVariances(getTypeArguments(source), getTypeArguments(target), variances, intersectionState);
+                        if (varianceResult !== undefined) {
+                            return varianceResult;
                         }
-                        if (isMatchingReference(reference, left_1)) {
-                            return narrowTypeByEquality(type, operator_1, right_1, assumeTrue);
+                    }
+                    else if (isReadonlyArrayType(target) ? isArrayType(source) || isTupleType(source) : isArrayType(target) && isTupleType(source) && !source.target.readonly) {
+                        if (relation !== identityRelation) {
+                            return isRelatedTo(getIndexTypeOfType(source, 1 /* Number */) || anyType, getIndexTypeOfType(target, 1 /* Number */) || anyType, reportErrors);
                         }
-                        if (isMatchingReference(reference, right_1)) {
-                            return narrowTypeByEquality(type, operator_1, left_1, assumeTrue);
+                        else {
+                            // By flags alone, we know that the `target` is a readonly array while the source is a normal array or tuple
+                            // or `target` is an array and source is a tuple - in both cases the types cannot be identical, by construction
+                            return 0 /* False */;
                         }
-                        if (strictNullChecks) {
-                            if (optionalChainContainsReference(left_1, reference)) {
-                                type = narrowTypeByOptionalChainContainment(type, operator_1, right_1, assumeTrue);
-                            }
-                            else if (optionalChainContainsReference(right_1, reference)) {
-                                type = narrowTypeByOptionalChainContainment(type, operator_1, left_1, assumeTrue);
+                    }
+                    // Consider a fresh empty object literal type "closed" under the subtype relationship - this way `{} <- {[idx: string]: any} <- fresh({})`
+                    // and not `{} <- fresh({}) <- {[idx: string]: any}`
+                    else if ((relation === subtypeRelation || relation === strictSubtypeRelation) && isEmptyObjectType(target) && ts.getObjectFlags(target) & 32768 /* FreshLiteral */ && !isEmptyObjectType(source)) {
+                        return 0 /* False */;
+                    }
+                    // Even if relationship doesn't hold for unions, intersections, or generic type references,
+                    // it may hold in a structural comparison.
+                    // In a check of the form X = A & B, we will have previously checked if A relates to X or B relates
+                    // to X. Failing both of those we want to check if the aggregation of A and B's members structurally
+                    // relates to X. Thus, we include intersection types on the source side here.
+                    if (source.flags & (524288 /* Object */ | 2097152 /* Intersection */) && target.flags & 524288 /* Object */) {
+                        // Report structural errors only if we haven't reported any errors yet
+                        var reportStructuralErrors = reportErrors && errorInfo === saveErrorInfo.errorInfo && !sourceIsPrimitive;
+                        result = propertiesRelatedTo(source, target, reportStructuralErrors, /*excludedProperties*/ undefined, intersectionState);
+                        if (result) {
+                            result &= signaturesRelatedTo(source, target, 0 /* Call */, reportStructuralErrors);
+                            if (result) {
+                                result &= signaturesRelatedTo(source, target, 1 /* Construct */, reportStructuralErrors);
+                                if (result) {
+                                    result &= indexTypesRelatedTo(source, target, 0 /* String */, sourceIsPrimitive, reportStructuralErrors, intersectionState);
+                                    if (result) {
+                                        result &= indexTypesRelatedTo(source, target, 1 /* Number */, sourceIsPrimitive, reportStructuralErrors, intersectionState);
+                                    }
+                                }
                             }
                         }
-                        if (isMatchingReferenceDiscriminant(left_1, declaredType)) {
-                            return narrowTypeByDiscriminant(type, left_1, function (t) { return narrowTypeByEquality(t, operator_1, right_1, assumeTrue); });
-                        }
-                        if (isMatchingReferenceDiscriminant(right_1, declaredType)) {
-                            return narrowTypeByDiscriminant(type, right_1, function (t) { return narrowTypeByEquality(t, operator_1, left_1, assumeTrue); });
-                        }
-                        if (isMatchingConstructorReference(left_1)) {
-                            return narrowTypeByConstructor(type, operator_1, right_1, assumeTrue);
+                        if (varianceCheckFailed && result) {
+                            errorInfo = originalErrorInfo || errorInfo || saveErrorInfo.errorInfo; // Use variance error (there is no structural one) and return false
                         }
-                        if (isMatchingConstructorReference(right_1)) {
-                            return narrowTypeByConstructor(type, operator_1, left_1, assumeTrue);
+                        else if (result) {
+                            return result;
                         }
-                        break;
-                    case 98 /* InstanceOfKeyword */:
-                        return narrowTypeByInstanceof(type, expr, assumeTrue);
-                    case 97 /* InKeyword */:
-                        var target = getReferenceCandidate(expr.right);
-                        if (ts.isStringLiteralLike(expr.left) && isMatchingReference(reference, target)) {
-                            return narrowByInKeyword(type, expr.left, assumeTrue);
+                    }
+                    // If S is an object type and T is a discriminated union, S may be related to T if
+                    // there exists a constituent of T for every combination of the discriminants of S
+                    // with respect to T. We do not report errors here, as we will use the existing
+                    // error result from checking each constituent of the union.
+                    if (source.flags & (524288 /* Object */ | 2097152 /* Intersection */) && target.flags & 1048576 /* Union */) {
+                        var objectOnlyTarget = extractTypesOfKind(target, 524288 /* Object */ | 2097152 /* Intersection */ | 33554432 /* Substitution */);
+                        if (objectOnlyTarget.flags & 1048576 /* Union */) {
+                            var result_9 = typeRelatedToDiscriminatedType(source, objectOnlyTarget);
+                            if (result_9) {
+                                return result_9;
+                            }
                         }
-                        break;
-                    case 27 /* CommaToken */:
-                        return narrowType(type, expr.right, assumeTrue);
-                }
-                return type;
-            }
-            function narrowTypeByOptionalChainContainment(type, operator, value, assumeTrue) {
-                // We are in a branch of obj?.foo === value (or any one of the other equality operators). We narrow obj as follows:
-                // When operator is === and type of value excludes undefined, null and undefined is removed from type of obj in true branch.
-                // When operator is !== and type of value excludes undefined, null and undefined is removed from type of obj in false branch.
-                // When operator is == and type of value excludes null and undefined, null and undefined is removed from type of obj in true branch.
-                // When operator is != and type of value excludes null and undefined, null and undefined is removed from type of obj in false branch.
-                // When operator is === and type of value is undefined, null and undefined is removed from type of obj in false branch.
-                // When operator is !== and type of value is undefined, null and undefined is removed from type of obj in true branch.
-                // When operator is == and type of value is null or undefined, null and undefined is removed from type of obj in false branch.
-                // When operator is != and type of value is null or undefined, null and undefined is removed from type of obj in true branch.
-                var equalsOperator = operator === 34 /* EqualsEqualsToken */ || operator === 36 /* EqualsEqualsEqualsToken */;
-                var nullableFlags = operator === 34 /* EqualsEqualsToken */ || operator === 35 /* ExclamationEqualsToken */ ? 98304 /* Nullable */ : 32768 /* Undefined */;
-                var valueType = getTypeOfExpression(value);
-                // Note that we include any and unknown in the exclusion test because their domain includes null and undefined.
-                var removeNullable = equalsOperator !== assumeTrue && everyType(valueType, function (t) { return !!(t.flags & nullableFlags); }) ||
-                    equalsOperator === assumeTrue && everyType(valueType, function (t) { return !(t.flags & (3 /* AnyOrUnknown */ | nullableFlags)); });
-                return removeNullable ? getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */) : type;
-            }
-            function narrowTypeByEquality(type, operator, value, assumeTrue) {
-                if (type.flags & 1 /* Any */) {
-                    return type;
-                }
-                if (operator === 35 /* ExclamationEqualsToken */ || operator === 37 /* ExclamationEqualsEqualsToken */) {
-                    assumeTrue = !assumeTrue;
+                    }
                 }
-                var valueType = getTypeOfExpression(value);
-                if ((type.flags & 2 /* Unknown */) && assumeTrue && (operator === 36 /* EqualsEqualsEqualsToken */ || operator === 37 /* ExclamationEqualsEqualsToken */)) {
-                    if (valueType.flags & (131068 /* Primitive */ | 67108864 /* NonPrimitive */)) {
-                        return valueType;
+                return 0 /* False */;
+                function relateVariances(sourceTypeArguments, targetTypeArguments, variances, intersectionState) {
+                    if (result = typeArgumentsRelatedTo(sourceTypeArguments, targetTypeArguments, variances, reportErrors, intersectionState)) {
+                        return result;
                     }
-                    if (valueType.flags & 524288 /* Object */) {
-                        return nonPrimitiveType;
+                    if (ts.some(variances, function (v) { return !!(v & 24 /* AllowsStructuralFallback */); })) {
+                        // If some type parameter was `Unmeasurable` or `Unreliable`, and we couldn't pass by assuming it was identical, then we
+                        // have to allow a structural fallback check
+                        // We elide the variance-based error elaborations, since those might not be too helpful, since we'll potentially
+                        // be assuming identity of the type parameter.
+                        originalErrorInfo = undefined;
+                        resetErrorInfo(saveErrorInfo);
+                        return undefined;
                     }
-                    return type;
-                }
-                if (valueType.flags & 98304 /* Nullable */) {
-                    if (!strictNullChecks) {
-                        return type;
+                    var allowStructuralFallback = targetTypeArguments && hasCovariantVoidArgument(targetTypeArguments, variances);
+                    varianceCheckFailed = !allowStructuralFallback;
+                    // The type arguments did not relate appropriately, but it may be because we have no variance
+                    // information (in which case typeArgumentsRelatedTo defaulted to covariance for all type
+                    // arguments). It might also be the case that the target type has a 'void' type argument for
+                    // a covariant type parameter that is only used in return positions within the generic type
+                    // (in which case any type argument is permitted on the source side). In those cases we proceed
+                    // with a structural comparison. Otherwise, we know for certain the instantiations aren't
+                    // related and we can return here.
+                    if (variances !== ts.emptyArray && !allowStructuralFallback) {
+                        // In some cases generic types that are covariant in regular type checking mode become
+                        // invariant in --strictFunctionTypes mode because one or more type parameters are used in
+                        // both co- and contravariant positions. In order to make it easier to diagnose *why* such
+                        // types are invariant, if any of the type parameters are invariant we reset the reported
+                        // errors and instead force a structural comparison (which will include elaborations that
+                        // reveal the reason).
+                        // We can switch on `reportErrors` here, since varianceCheckFailed guarantees we return `False`,
+                        // we can return `False` early here to skip calculating the structural error message we don't need.
+                        if (varianceCheckFailed && !(reportErrors && ts.some(variances, function (v) { return (v & 7 /* VarianceMask */) === 0 /* Invariant */; }))) {
+                            return 0 /* False */;
+                        }
+                        // We remember the original error information so we can restore it in case the structural
+                        // comparison unexpectedly succeeds. This can happen when the structural comparison result
+                        // is a Ternary.Maybe for example caused by the recursion depth limiter.
+                        originalErrorInfo = errorInfo;
+                        resetErrorInfo(saveErrorInfo);
                     }
-                    var doubleEquals = operator === 34 /* EqualsEqualsToken */ || operator === 35 /* ExclamationEqualsToken */;
-                    var facts = doubleEquals ?
-                        assumeTrue ? 262144 /* EQUndefinedOrNull */ : 2097152 /* NEUndefinedOrNull */ :
-                        valueType.flags & 65536 /* Null */ ?
-                            assumeTrue ? 131072 /* EQNull */ : 1048576 /* NENull */ :
-                            assumeTrue ? 65536 /* EQUndefined */ : 524288 /* NEUndefined */;
-                    return getTypeWithFacts(type, facts);
-                }
-                if (type.flags & 67637251 /* NotUnionOrUnit */) {
-                    return type;
                 }
-                if (assumeTrue) {
-                    var filterFn = operator === 34 /* EqualsEqualsToken */ ?
-                        (function (t) { return areTypesComparable(t, valueType) || isCoercibleUnderDoubleEquals(t, valueType); }) :
-                        function (t) { return areTypesComparable(t, valueType); };
-                    var narrowedType = filterType(type, filterFn);
-                    return narrowedType.flags & 131072 /* Never */ ? type : replacePrimitivesWithLiterals(narrowedType, valueType);
+            }
+            function reportUnmeasurableMarkers(p) {
+                if (outofbandVarianceMarkerHandler && (p === markerSuperType || p === markerSubType || p === markerOtherType)) {
+                    outofbandVarianceMarkerHandler(/*onlyUnreliable*/ false);
                 }
-                if (isUnitType(valueType)) {
-                    var regularType_1 = getRegularTypeOfLiteralType(valueType);
-                    return filterType(type, function (t) { return isUnitType(t) ? !areTypesComparable(t, valueType) : getRegularTypeOfLiteralType(t) !== regularType_1; });
+                return p;
+            }
+            function reportUnreliableMarkers(p) {
+                if (outofbandVarianceMarkerHandler && (p === markerSuperType || p === markerSubType || p === markerOtherType)) {
+                    outofbandVarianceMarkerHandler(/*onlyUnreliable*/ true);
                 }
-                return type;
+                return p;
             }
-            function narrowTypeByTypeof(type, typeOfExpr, operator, literal, assumeTrue) {
-                // We have '==', '!=', '===', or !==' operator with 'typeof xxx' and string literal operands
-                if (operator === 35 /* ExclamationEqualsToken */ || operator === 37 /* ExclamationEqualsEqualsToken */) {
-                    assumeTrue = !assumeTrue;
+            // A type [P in S]: X is related to a type [Q in T]: Y if T is related to S and X' is
+            // related to Y, where X' is an instantiation of X in which P is replaced with Q. Notice
+            // that S and T are contra-variant whereas X and Y are co-variant.
+            function mappedTypeRelatedTo(source, target, reportErrors) {
+                var modifiersRelated = relation === comparableRelation || (relation === identityRelation ? getMappedTypeModifiers(source) === getMappedTypeModifiers(target) :
+                    getCombinedMappedTypeOptionality(source) <= getCombinedMappedTypeOptionality(target));
+                if (modifiersRelated) {
+                    var result_10;
+                    var targetConstraint = getConstraintTypeFromMappedType(target);
+                    var sourceConstraint = instantiateType(getConstraintTypeFromMappedType(source), makeFunctionTypeMapper(getCombinedMappedTypeOptionality(source) < 0 ? reportUnmeasurableMarkers : reportUnreliableMarkers));
+                    if (result_10 = isRelatedTo(targetConstraint, sourceConstraint, reportErrors)) {
+                        var mapper = createTypeMapper([getTypeParameterFromMappedType(source)], [getTypeParameterFromMappedType(target)]);
+                        if (instantiateType(getNameTypeFromMappedType(source), mapper) === instantiateType(getNameTypeFromMappedType(target), mapper)) {
+                            return result_10 & isRelatedTo(instantiateType(getTemplateTypeFromMappedType(source), mapper), getTemplateTypeFromMappedType(target), reportErrors);
+                        }
+                    }
                 }
-                var target = getReferenceCandidate(typeOfExpr.expression);
-                if (!isMatchingReference(reference, target)) {
-                    if (strictNullChecks && optionalChainContainsReference(target, reference) && assumeTrue === (literal.text !== "undefined")) {
-                        return getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
+                return 0 /* False */;
+            }
+            function typeRelatedToDiscriminatedType(source, target) {
+                // 1. Generate the combinations of discriminant properties & types 'source' can satisfy.
+                //    a. If the number of combinations is above a set limit, the comparison is too complex.
+                // 2. Filter 'target' to the subset of types whose discriminants exist in the matrix.
+                //    a. If 'target' does not satisfy all discriminants in the matrix, 'source' is not related.
+                // 3. For each type in the filtered 'target', determine if all non-discriminant properties of
+                //    'target' are related to a property in 'source'.
+                //
+                // NOTE: See ~/tests/cases/conformance/types/typeRelationships/assignmentCompatibility/assignmentCompatWithDiscriminatedUnion.ts
+                //       for examples.
+                var sourceProperties = getPropertiesOfType(source);
+                var sourcePropertiesFiltered = findDiscriminantProperties(sourceProperties, target);
+                if (!sourcePropertiesFiltered)
+                    return 0 /* False */;
+                // Though we could compute the number of combinations as we generate
+                // the matrix, this would incur additional memory overhead due to
+                // array allocations. To reduce this overhead, we first compute
+                // the number of combinations to ensure we will not surpass our
+                // fixed limit before incurring the cost of any allocations:
+                var numCombinations = 1;
+                for (var _i = 0, sourcePropertiesFiltered_1 = sourcePropertiesFiltered; _i < sourcePropertiesFiltered_1.length; _i++) {
+                    var sourceProperty = sourcePropertiesFiltered_1[_i];
+                    numCombinations *= countTypes(getTypeOfSymbol(sourceProperty));
+                    if (numCombinations > 25) {
+                        // We've reached the complexity limit.
+                        ts.tracing.instant("check" /* Check */, "typeRelatedToDiscriminatedType_DepthLimit", { sourceId: source.id, targetId: target.id, numCombinations: numCombinations });
+                        return 0 /* False */;
                     }
-                    return type;
                 }
-                if (type.flags & 1 /* Any */ && literal.text === "function") {
-                    return type;
+                // Compute the set of types for each discriminant property.
+                var sourceDiscriminantTypes = new Array(sourcePropertiesFiltered.length);
+                var excludedProperties = new ts.Set();
+                for (var i = 0; i < sourcePropertiesFiltered.length; i++) {
+                    var sourceProperty = sourcePropertiesFiltered[i];
+                    var sourcePropertyType = getTypeOfSymbol(sourceProperty);
+                    sourceDiscriminantTypes[i] = sourcePropertyType.flags & 1048576 /* Union */
+                        ? sourcePropertyType.types
+                        : [sourcePropertyType];
+                    excludedProperties.add(sourceProperty.escapedName);
                 }
-                if (assumeTrue && type.flags & 2 /* Unknown */ && literal.text === "object") {
-                    // The pattern x && typeof x === 'object', where x is of type unknown, narrows x to type object. We don't
-                    // need to check for the reverse typeof x === 'object' && x since that already narrows correctly.
-                    if (typeOfExpr.parent.parent.kind === 209 /* BinaryExpression */) {
-                        var expr = typeOfExpr.parent.parent;
-                        if (expr.operatorToken.kind === 55 /* AmpersandAmpersandToken */ && expr.right === typeOfExpr.parent && containsTruthyCheck(reference, expr.left)) {
-                            return nonPrimitiveType;
+                // Match each combination of the cartesian product of discriminant properties to one or more
+                // constituents of 'target'. If any combination does not have a match then 'source' is not relatable.
+                var discriminantCombinations = ts.cartesianProduct(sourceDiscriminantTypes);
+                var matchingTypes = [];
+                var _loop_17 = function (combination) {
+                    var hasMatch = false;
+                    outer: for (var _i = 0, _a = target.types; _i < _a.length; _i++) {
+                        var type = _a[_i];
+                        var _loop_18 = function (i) {
+                            var sourceProperty = sourcePropertiesFiltered[i];
+                            var targetProperty = getPropertyOfType(type, sourceProperty.escapedName);
+                            if (!targetProperty)
+                                return "continue-outer";
+                            if (sourceProperty === targetProperty)
+                                return "continue";
+                            // We compare the source property to the target in the context of a single discriminant type.
+                            var related = propertyRelatedTo(source, target, sourceProperty, targetProperty, function (_) { return combination[i]; }, /*reportErrors*/ false, 0 /* None */, /*skipOptional*/ strictNullChecks || relation === comparableRelation);
+                            // If the target property could not be found, or if the properties were not related,
+                            // then this constituent is not a match.
+                            if (!related) {
+                                return "continue-outer";
+                            }
+                        };
+                        for (var i = 0; i < sourcePropertiesFiltered.length; i++) {
+                            var state_7 = _loop_18(i);
+                            switch (state_7) {
+                                case "continue-outer": continue outer;
+                            }
                         }
+                        ts.pushIfUnique(matchingTypes, type, ts.equateValues);
+                        hasMatch = true;
                     }
-                    return getUnionType([nonPrimitiveType, nullType]);
+                    if (!hasMatch) {
+                        return { value: 0 /* False */ };
+                    }
+                };
+                for (var _a = 0, discriminantCombinations_1 = discriminantCombinations; _a < discriminantCombinations_1.length; _a++) {
+                    var combination = discriminantCombinations_1[_a];
+                    var state_6 = _loop_17(combination);
+                    if (typeof state_6 === "object")
+                        return state_6.value;
                 }
-                var facts = assumeTrue ?
-                    typeofEQFacts.get(literal.text) || 128 /* TypeofEQHostObject */ :
-                    typeofNEFacts.get(literal.text) || 32768 /* TypeofNEHostObject */;
-                return getTypeWithFacts(assumeTrue ? mapType(type, narrowTypeForTypeof) : type, facts);
-                function narrowTypeForTypeof(type) {
-                    // We narrow a non-union type to an exact primitive type if the non-union type
-                    // is a supertype of that primitive type. For example, type 'any' can be narrowed
-                    // to one of the primitive types.
-                    var targetType = literal.text === "function" ? globalFunctionType : typeofTypesByName.get(literal.text);
-                    if (targetType) {
-                        if (isTypeSubtypeOf(type, targetType)) {
-                            return type;
-                        }
-                        if (isTypeSubtypeOf(targetType, type)) {
-                            return targetType;
-                        }
-                        if (type.flags & 63176704 /* Instantiable */) {
-                            var constraint = getBaseConstraintOfType(type) || anyType;
-                            if (isTypeSubtypeOf(targetType, constraint)) {
-                                return getIntersectionType([type, targetType]);
+                // Compare the remaining non-discriminant properties of each match.
+                var result = -1 /* True */;
+                for (var _b = 0, matchingTypes_1 = matchingTypes; _b < matchingTypes_1.length; _b++) {
+                    var type = matchingTypes_1[_b];
+                    result &= propertiesRelatedTo(source, type, /*reportErrors*/ false, excludedProperties, 0 /* None */);
+                    if (result) {
+                        result &= signaturesRelatedTo(source, type, 0 /* Call */, /*reportStructuralErrors*/ false);
+                        if (result) {
+                            result &= signaturesRelatedTo(source, type, 1 /* Construct */, /*reportStructuralErrors*/ false);
+                            if (result) {
+                                result &= indexTypesRelatedTo(source, type, 0 /* String */, /*sourceIsPrimitive*/ false, /*reportStructuralErrors*/ false, 0 /* None */);
+                                // Comparing numeric index types when both `source` and `type` are tuples is unnecessary as the
+                                // element types should be sufficiently covered by `propertiesRelatedTo`. It also causes problems
+                                // with index type assignability as the types for the excluded discriminants are still included
+                                // in the index type.
+                                if (result && !(isTupleType(source) && isTupleType(type))) {
+                                    result &= indexTypesRelatedTo(source, type, 1 /* Number */, /*sourceIsPrimitive*/ false, /*reportStructuralErrors*/ false, 0 /* None */);
+                                }
                             }
                         }
                     }
-                    return type;
+                    if (!result) {
+                        return result;
+                    }
                 }
+                return result;
             }
-            function narrowTypeBySwitchOptionalChainContainment(type, switchStatement, clauseStart, clauseEnd, clauseCheck) {
-                var everyClauseChecks = clauseStart !== clauseEnd && ts.every(getSwitchClauseTypes(switchStatement).slice(clauseStart, clauseEnd), clauseCheck);
-                return everyClauseChecks ? getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */) : type;
-            }
-            function narrowTypeBySwitchOnDiscriminant(type, switchStatement, clauseStart, clauseEnd) {
-                // We only narrow if all case expressions specify
-                // values with unit types, except for the case where
-                // `type` is unknown. In this instance we map object
-                // types to the nonPrimitive type and narrow with that.
-                var switchTypes = getSwitchClauseTypes(switchStatement);
-                if (!switchTypes.length) {
-                    return type;
-                }
-                var clauseTypes = switchTypes.slice(clauseStart, clauseEnd);
-                var hasDefaultClause = clauseStart === clauseEnd || ts.contains(clauseTypes, neverType);
-                if ((type.flags & 2 /* Unknown */) && !hasDefaultClause) {
-                    var groundClauseTypes = void 0;
-                    for (var i = 0; i < clauseTypes.length; i += 1) {
-                        var t = clauseTypes[i];
-                        if (t.flags & (131068 /* Primitive */ | 67108864 /* NonPrimitive */)) {
-                            if (groundClauseTypes !== undefined) {
-                                groundClauseTypes.push(t);
-                            }
+            function excludeProperties(properties, excludedProperties) {
+                if (!excludedProperties || properties.length === 0)
+                    return properties;
+                var result;
+                for (var i = 0; i < properties.length; i++) {
+                    if (!excludedProperties.has(properties[i].escapedName)) {
+                        if (result) {
+                            result.push(properties[i]);
                         }
-                        else if (t.flags & 524288 /* Object */) {
-                            if (groundClauseTypes === undefined) {
-                                groundClauseTypes = clauseTypes.slice(0, i);
+                    }
+                    else if (!result) {
+                        result = properties.slice(0, i);
+                    }
+                }
+                return result || properties;
+            }
+            function isPropertySymbolTypeRelated(sourceProp, targetProp, getTypeOfSourceProperty, reportErrors, intersectionState) {
+                var targetIsOptional = strictNullChecks && !!(ts.getCheckFlags(targetProp) & 48 /* Partial */);
+                var source = getTypeOfSourceProperty(sourceProp);
+                if (ts.getCheckFlags(targetProp) & 65536 /* DeferredType */ && !getSymbolLinks(targetProp).type) {
+                    // Rather than resolving (and normalizing) the type, relate constituent-by-constituent without performing normalization or seconadary passes
+                    var links = getSymbolLinks(targetProp);
+                    ts.Debug.assertIsDefined(links.deferralParent);
+                    ts.Debug.assertIsDefined(links.deferralConstituents);
+                    var unionParent = !!(links.deferralParent.flags & 1048576 /* Union */);
+                    var result_11 = unionParent ? 0 /* False */ : -1 /* True */;
+                    var targetTypes = links.deferralConstituents;
+                    for (var _i = 0, targetTypes_3 = targetTypes; _i < targetTypes_3.length; _i++) {
+                        var targetType = targetTypes_3[_i];
+                        var related = isRelatedTo(source, targetType, /*reportErrors*/ false, /*headMessage*/ undefined, unionParent ? 0 : 2 /* Target */);
+                        if (!unionParent) {
+                            if (!related) {
+                                // Can't assign to a target individually - have to fallback to assigning to the _whole_ intersection (which forces normalization)
+                                return isRelatedTo(source, addOptionality(getTypeOfSymbol(targetProp), targetIsOptional), reportErrors);
                             }
-                            groundClauseTypes.push(nonPrimitiveType);
+                            result_11 &= related;
                         }
                         else {
-                            return type;
+                            if (related) {
+                                return related;
+                            }
                         }
                     }
-                    return getUnionType(groundClauseTypes === undefined ? clauseTypes : groundClauseTypes);
-                }
-                var discriminantType = getUnionType(clauseTypes);
-                var caseType = discriminantType.flags & 131072 /* Never */ ? neverType :
-                    replacePrimitivesWithLiterals(filterType(type, function (t) { return areTypesComparable(discriminantType, t); }), discriminantType);
-                if (!hasDefaultClause) {
-                    return caseType;
+                    if (unionParent && !result_11 && targetIsOptional) {
+                        result_11 = isRelatedTo(source, undefinedType);
+                    }
+                    if (unionParent && !result_11 && reportErrors) {
+                        // The easiest way to get the right errors here is to un-defer (which may be costly)
+                        // If it turns out this is too costly too often, we can replicate the error handling logic within
+                        // typeRelatedToSomeType without the discriminatable type branch (as that requires a manifest union
+                        // type on which to hand discriminable properties, which we are expressly trying to avoid here)
+                        return isRelatedTo(source, addOptionality(getTypeOfSymbol(targetProp), targetIsOptional), reportErrors);
+                    }
+                    return result_11;
                 }
-                var defaultType = filterType(type, function (t) { return !(isUnitType(t) && ts.contains(switchTypes, getRegularTypeOfLiteralType(t))); });
-                return caseType.flags & 131072 /* Never */ ? defaultType : getUnionType([caseType, defaultType]);
-            }
-            function getImpliedTypeFromTypeofCase(type, text) {
-                switch (text) {
-                    case "function":
-                        return type.flags & 1 /* Any */ ? type : globalFunctionType;
-                    case "object":
-                        return type.flags & 2 /* Unknown */ ? getUnionType([nonPrimitiveType, nullType]) : type;
-                    default:
-                        return typeofTypesByName.get(text) || type;
+                else {
+                    return isRelatedTo(source, addOptionality(getTypeOfSymbol(targetProp), targetIsOptional), reportErrors, /*headMessage*/ undefined, intersectionState);
                 }
             }
-            function narrowTypeForTypeofSwitch(candidate) {
-                return function (type) {
-                    if (isTypeSubtypeOf(candidate, type)) {
-                        return candidate;
-                    }
-                    if (type.flags & 63176704 /* Instantiable */) {
-                        var constraint = getBaseConstraintOfType(type) || anyType;
-                        if (isTypeSubtypeOf(candidate, constraint)) {
-                            return getIntersectionType([type, candidate]);
+            function propertyRelatedTo(source, target, sourceProp, targetProp, getTypeOfSourceProperty, reportErrors, intersectionState, skipOptional) {
+                var sourcePropFlags = ts.getDeclarationModifierFlagsFromSymbol(sourceProp);
+                var targetPropFlags = ts.getDeclarationModifierFlagsFromSymbol(targetProp);
+                if (sourcePropFlags & 8 /* Private */ || targetPropFlags & 8 /* Private */) {
+                    if (sourceProp.valueDeclaration !== targetProp.valueDeclaration) {
+                        if (reportErrors) {
+                            if (sourcePropFlags & 8 /* Private */ && targetPropFlags & 8 /* Private */) {
+                                reportError(ts.Diagnostics.Types_have_separate_declarations_of_a_private_property_0, symbolToString(targetProp));
+                            }
+                            else {
+                                reportError(ts.Diagnostics.Property_0_is_private_in_type_1_but_not_in_type_2, symbolToString(targetProp), typeToString(sourcePropFlags & 8 /* Private */ ? source : target), typeToString(sourcePropFlags & 8 /* Private */ ? target : source));
+                            }
                         }
+                        return 0 /* False */;
                     }
-                    return type;
-                };
-            }
-            function narrowBySwitchOnTypeOf(type, switchStatement, clauseStart, clauseEnd) {
-                var switchWitnesses = getSwitchClauseTypeOfWitnesses(switchStatement, /*retainDefault*/ true);
-                if (!switchWitnesses.length) {
-                    return type;
                 }
-                //  Equal start and end denotes implicit fallthrough; undefined marks explicit default clause
-                var defaultCaseLocation = ts.findIndex(switchWitnesses, function (elem) { return elem === undefined; });
-                var hasDefaultClause = clauseStart === clauseEnd || (defaultCaseLocation >= clauseStart && defaultCaseLocation < clauseEnd);
-                var clauseWitnesses;
-                var switchFacts;
-                if (defaultCaseLocation > -1) {
-                    // We no longer need the undefined denoting an
-                    // explicit default case. Remove the undefined and
-                    // fix-up clauseStart and clauseEnd.  This means
-                    // that we don't have to worry about undefined
-                    // in the witness array.
-                    var witnesses = switchWitnesses.filter(function (witness) { return witness !== undefined; });
-                    // The adjusted clause start and end after removing the `default` statement.
-                    var fixedClauseStart = defaultCaseLocation < clauseStart ? clauseStart - 1 : clauseStart;
-                    var fixedClauseEnd = defaultCaseLocation < clauseEnd ? clauseEnd - 1 : clauseEnd;
-                    clauseWitnesses = witnesses.slice(fixedClauseStart, fixedClauseEnd);
-                    switchFacts = getFactsFromTypeofSwitch(fixedClauseStart, fixedClauseEnd, witnesses, hasDefaultClause);
+                else if (targetPropFlags & 16 /* Protected */) {
+                    if (!isValidOverrideOf(sourceProp, targetProp)) {
+                        if (reportErrors) {
+                            reportError(ts.Diagnostics.Property_0_is_protected_but_type_1_is_not_a_class_derived_from_2, symbolToString(targetProp), typeToString(getDeclaringClass(sourceProp) || source), typeToString(getDeclaringClass(targetProp) || target));
+                        }
+                        return 0 /* False */;
+                    }
                 }
-                else {
-                    clauseWitnesses = switchWitnesses.slice(clauseStart, clauseEnd);
-                    switchFacts = getFactsFromTypeofSwitch(clauseStart, clauseEnd, switchWitnesses, hasDefaultClause);
+                else if (sourcePropFlags & 16 /* Protected */) {
+                    if (reportErrors) {
+                        reportError(ts.Diagnostics.Property_0_is_protected_in_type_1_but_public_in_type_2, symbolToString(targetProp), typeToString(source), typeToString(target));
+                    }
+                    return 0 /* False */;
                 }
-                if (hasDefaultClause) {
-                    return filterType(type, function (t) { return (getTypeFacts(t) & switchFacts) === switchFacts; });
+                // If the target comes from a partial union prop, allow `undefined` in the target type
+                var related = isPropertySymbolTypeRelated(sourceProp, targetProp, getTypeOfSourceProperty, reportErrors, intersectionState);
+                if (!related) {
+                    if (reportErrors) {
+                        reportIncompatibleError(ts.Diagnostics.Types_of_property_0_are_incompatible, symbolToString(targetProp));
+                    }
+                    return 0 /* False */;
                 }
-                /*
-                  The implied type is the raw type suggested by a
-                  value being caught in this clause.
-
-                  When the clause contains a default case we ignore
-                  the implied type and try to narrow using any facts
-                  we can learn: see `switchFacts`.
-
-                  Example:
-                  switch (typeof x) {
-                      case 'number':
-                      case 'string': break;
-                      default: break;
-                      case 'number':
-                      case 'boolean': break
-                  }
-
-                  In the first clause (case `number` and `string`) the
-                  implied type is number | string.
-
-                  In the default clause we de not compute an implied type.
-
-                  In the third clause (case `number` and `boolean`)
-                  the naive implied type is number | boolean, however
-                  we use the type facts to narrow the implied type to
-                  boolean. We know that number cannot be selected
-                  because it is caught in the first clause.
-                */
-                var impliedType = getTypeWithFacts(getUnionType(clauseWitnesses.map(function (text) { return getImpliedTypeFromTypeofCase(type, text); })), switchFacts);
-                if (impliedType.flags & 1048576 /* Union */) {
-                    impliedType = getAssignmentReducedType(impliedType, getBaseConstraintOrType(type));
+                // When checking for comparability, be more lenient with optional properties.
+                if (!skipOptional && sourceProp.flags & 16777216 /* Optional */ && !(targetProp.flags & 16777216 /* Optional */)) {
+                    // TypeScript 1.0 spec (April 2014): 3.8.3
+                    // S is a subtype of a type T, and T is a supertype of S if ...
+                    // S' and T are object types and, for each member M in T..
+                    // M is a property and S' contains a property N where
+                    // if M is a required property, N is also a required property
+                    // (M - property in T)
+                    // (N - property in S)
+                    if (reportErrors) {
+                        reportError(ts.Diagnostics.Property_0_is_optional_in_type_1_but_required_in_type_2, symbolToString(targetProp), typeToString(source), typeToString(target));
+                    }
+                    return 0 /* False */;
                 }
-                return getTypeWithFacts(mapType(type, narrowTypeForTypeofSwitch(impliedType)), switchFacts);
-            }
-            function isMatchingConstructorReference(expr) {
-                return (ts.isPropertyAccessExpression(expr) && ts.idText(expr.name) === "constructor" ||
-                    ts.isElementAccessExpression(expr) && ts.isStringLiteralLike(expr.argumentExpression) && expr.argumentExpression.text === "constructor") &&
-                    isMatchingReference(reference, expr.expression);
+                return related;
             }
-            function narrowTypeByConstructor(type, operator, identifier, assumeTrue) {
-                // Do not narrow when checking inequality.
-                if (assumeTrue ? (operator !== 34 /* EqualsEqualsToken */ && operator !== 36 /* EqualsEqualsEqualsToken */) : (operator !== 35 /* ExclamationEqualsToken */ && operator !== 37 /* ExclamationEqualsEqualsToken */)) {
-                    return type;
+            function reportUnmatchedProperty(source, target, unmatchedProperty, requireOptionalProperties) {
+                var shouldSkipElaboration = false;
+                // give specific error in case where private names have the same description
+                if (unmatchedProperty.valueDeclaration
+                    && ts.isNamedDeclaration(unmatchedProperty.valueDeclaration)
+                    && ts.isPrivateIdentifier(unmatchedProperty.valueDeclaration.name)
+                    && source.symbol
+                    && source.symbol.flags & 32 /* Class */) {
+                    var privateIdentifierDescription = unmatchedProperty.valueDeclaration.name.escapedText;
+                    var symbolTableKey = ts.getSymbolNameForPrivateIdentifier(source.symbol, privateIdentifierDescription);
+                    if (symbolTableKey && getPropertyOfType(source, symbolTableKey)) {
+                        var sourceName = ts.factory.getDeclarationName(source.symbol.valueDeclaration);
+                        var targetName = ts.factory.getDeclarationName(target.symbol.valueDeclaration);
+                        reportError(ts.Diagnostics.Property_0_in_type_1_refers_to_a_different_member_that_cannot_be_accessed_from_within_type_2, diagnosticName(privateIdentifierDescription), diagnosticName(sourceName.escapedText === "" ? anon : sourceName), diagnosticName(targetName.escapedText === "" ? anon : targetName));
+                        return;
+                    }
                 }
-                // Get the type of the constructor identifier expression, if it is not a function then do not narrow.
-                var identifierType = getTypeOfExpression(identifier);
-                if (!isFunctionType(identifierType) && !isConstructorType(identifierType)) {
-                    return type;
+                var props = ts.arrayFrom(getUnmatchedProperties(source, target, requireOptionalProperties, /*matchDiscriminantProperties*/ false));
+                if (!headMessage || (headMessage.code !== ts.Diagnostics.Class_0_incorrectly_implements_interface_1.code &&
+                    headMessage.code !== ts.Diagnostics.Class_0_incorrectly_implements_class_1_Did_you_mean_to_extend_1_and_inherit_its_members_as_a_subclass.code)) {
+                    shouldSkipElaboration = true; // Retain top-level error for interface implementing issues, otherwise omit it
                 }
-                // Get the prototype property of the type identifier so we can find out its type.
-                var prototypeProperty = getPropertyOfType(identifierType, "prototype");
-                if (!prototypeProperty) {
-                    return type;
+                if (props.length === 1) {
+                    var propName = symbolToString(unmatchedProperty);
+                    reportError.apply(void 0, __spreadArrays([ts.Diagnostics.Property_0_is_missing_in_type_1_but_required_in_type_2, propName], getTypeNamesForErrorDisplay(source, target)));
+                    if (ts.length(unmatchedProperty.declarations)) {
+                        associateRelatedInfo(ts.createDiagnosticForNode(unmatchedProperty.declarations[0], ts.Diagnostics._0_is_declared_here, propName));
+                    }
+                    if (shouldSkipElaboration && errorInfo) {
+                        overrideNextErrorInfo++;
+                    }
                 }
-                // Get the type of the prototype, if it is undefined, or the global `Object` or `Function` types then do not narrow.
-                var prototypeType = getTypeOfSymbol(prototypeProperty);
-                var candidate = !isTypeAny(prototypeType) ? prototypeType : undefined;
-                if (!candidate || candidate === globalObjectType || candidate === globalFunctionType) {
-                    return type;
+                else if (tryElaborateArrayLikeErrors(source, target, /*reportErrors*/ false)) {
+                    if (props.length > 5) { // arbitrary cutoff for too-long list form
+                        reportError(ts.Diagnostics.Type_0_is_missing_the_following_properties_from_type_1_Colon_2_and_3_more, typeToString(source), typeToString(target), ts.map(props.slice(0, 4), function (p) { return symbolToString(p); }).join(", "), props.length - 4);
+                    }
+                    else {
+                        reportError(ts.Diagnostics.Type_0_is_missing_the_following_properties_from_type_1_Colon_2, typeToString(source), typeToString(target), ts.map(props, function (p) { return symbolToString(p); }).join(", "));
+                    }
+                    if (shouldSkipElaboration && errorInfo) {
+                        overrideNextErrorInfo++;
+                    }
                 }
-                // If the type that is being narrowed is `any` then just return the `candidate` type since every type is a subtype of `any`.
-                if (isTypeAny(type)) {
-                    return candidate;
+                // No array like or unmatched property error - just issue top level error (errorInfo = undefined)
+            }
+            function propertiesRelatedTo(source, target, reportErrors, excludedProperties, intersectionState) {
+                if (relation === identityRelation) {
+                    return propertiesIdenticalTo(source, target, excludedProperties);
                 }
-                // Filter out types that are not considered to be "constructed by" the `candidate` type.
-                return filterType(type, function (t) { return isConstructedBy(t, candidate); });
-                function isConstructedBy(source, target) {
-                    // If either the source or target type are a class type then we need to check that they are the same exact type.
-                    // This is because you may have a class `A` that defines some set of properties, and another class `B`
-                    // that defines the same set of properties as class `A`, in that case they are structurally the same
-                    // type, but when you do something like `instanceOfA.constructor === B` it will return false.
-                    if (source.flags & 524288 /* Object */ && ts.getObjectFlags(source) & 1 /* Class */ ||
-                        target.flags & 524288 /* Object */ && ts.getObjectFlags(target) & 1 /* Class */) {
-                        return source.symbol === target.symbol;
+                var result = -1 /* True */;
+                if (isTupleType(target)) {
+                    if (isArrayType(source) || isTupleType(source)) {
+                        if (!target.target.readonly && (isReadonlyArrayType(source) || isTupleType(source) && source.target.readonly)) {
+                            return 0 /* False */;
+                        }
+                        var sourceArity = getTypeReferenceArity(source);
+                        var targetArity = getTypeReferenceArity(target);
+                        var sourceRestFlag = isTupleType(source) ? source.target.combinedFlags & 4 /* Rest */ : 4 /* Rest */;
+                        var targetRestFlag = target.target.combinedFlags & 4 /* Rest */;
+                        var sourceMinLength = isTupleType(source) ? source.target.minLength : 0;
+                        var targetMinLength = target.target.minLength;
+                        if (!sourceRestFlag && sourceArity < targetMinLength) {
+                            if (reportErrors) {
+                                reportError(ts.Diagnostics.Source_has_0_element_s_but_target_requires_1, sourceArity, targetMinLength);
+                            }
+                            return 0 /* False */;
+                        }
+                        if (!targetRestFlag && targetArity < sourceMinLength) {
+                            if (reportErrors) {
+                                reportError(ts.Diagnostics.Source_has_0_element_s_but_target_allows_only_1, sourceMinLength, targetArity);
+                            }
+                            return 0 /* False */;
+                        }
+                        if (!targetRestFlag && sourceRestFlag) {
+                            if (reportErrors) {
+                                if (sourceMinLength < targetMinLength) {
+                                    reportError(ts.Diagnostics.Target_requires_0_element_s_but_source_may_have_fewer, targetMinLength);
+                                }
+                                else {
+                                    reportError(ts.Diagnostics.Target_allows_only_0_element_s_but_source_may_have_more, targetArity);
+                                }
+                            }
+                            return 0 /* False */;
+                        }
+                        var maxArity = Math.max(sourceArity, targetArity);
+                        for (var i = 0; i < maxArity; i++) {
+                            var targetFlags = i < targetArity ? target.target.elementFlags[i] : targetRestFlag;
+                            var sourceFlags = isTupleType(source) && i < sourceArity ? source.target.elementFlags[i] : sourceRestFlag;
+                            var canExcludeDiscriminants = !!excludedProperties;
+                            if (sourceFlags && targetFlags) {
+                                if (targetFlags & 8 /* Variadic */ && !(sourceFlags & 8 /* Variadic */) ||
+                                    (sourceFlags & 8 /* Variadic */ && !(targetFlags & 12 /* Variable */))) {
+                                    if (reportErrors) {
+                                        reportError(ts.Diagnostics.Element_at_index_0_is_variadic_in_one_type_but_not_in_the_other, i);
+                                    }
+                                    return 0 /* False */;
+                                }
+                                if (targetFlags & 1 /* Required */) {
+                                    if (!(sourceFlags & 1 /* Required */)) {
+                                        if (reportErrors) {
+                                            reportError(ts.Diagnostics.Property_0_is_optional_in_type_1_but_required_in_type_2, i, typeToString(source), typeToString(target));
+                                        }
+                                        return 0 /* False */;
+                                    }
+                                }
+                                // We can only exclude discriminant properties if we have not yet encountered a variable-length element.
+                                if (canExcludeDiscriminants) {
+                                    if (sourceFlags & 12 /* Variable */ || targetFlags & 12 /* Variable */) {
+                                        canExcludeDiscriminants = false;
+                                    }
+                                    if (canExcludeDiscriminants && (excludedProperties === null || excludedProperties === void 0 ? void 0 : excludedProperties.has(("" + i)))) {
+                                        continue;
+                                    }
+                                }
+                                var sourceType = getTypeArguments(source)[Math.min(i, sourceArity - 1)];
+                                var targetType = getTypeArguments(target)[Math.min(i, targetArity - 1)];
+                                var targetCheckType = sourceFlags & 8 /* Variadic */ && targetFlags & 4 /* Rest */ ? createArrayType(targetType) : targetType;
+                                var related = isRelatedTo(sourceType, targetCheckType, reportErrors, /*headMessage*/ undefined, intersectionState);
+                                if (!related) {
+                                    if (reportErrors) {
+                                        reportIncompatibleError(ts.Diagnostics.Types_of_property_0_are_incompatible, i);
+                                    }
+                                    return 0 /* False */;
+                                }
+                                result &= related;
+                            }
+                        }
+                        return result;
+                    }
+                    if (target.target.combinedFlags & 12 /* Variable */) {
+                        return 0 /* False */;
                     }
-                    // For all other types just check that the `source` type is a subtype of the `target` type.
-                    return isTypeSubtypeOf(source, target);
                 }
-            }
-            function narrowTypeByInstanceof(type, expr, assumeTrue) {
-                var left = getReferenceCandidate(expr.left);
-                if (!isMatchingReference(reference, left)) {
-                    if (assumeTrue && strictNullChecks && optionalChainContainsReference(left, reference)) {
-                        return getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
+                var requireOptionalProperties = (relation === subtypeRelation || relation === strictSubtypeRelation) && !isObjectLiteralType(source) && !isEmptyArrayLiteralType(source) && !isTupleType(source);
+                var unmatchedProperty = getUnmatchedProperty(source, target, requireOptionalProperties, /*matchDiscriminantProperties*/ false);
+                if (unmatchedProperty) {
+                    if (reportErrors) {
+                        reportUnmatchedProperty(source, target, unmatchedProperty, requireOptionalProperties);
                     }
-                    return type;
+                    return 0 /* False */;
                 }
-                // Check that right operand is a function type with a prototype property
-                var rightType = getTypeOfExpression(expr.right);
-                if (!isTypeDerivedFrom(rightType, globalFunctionType)) {
-                    return type;
+                if (isObjectLiteralType(target)) {
+                    for (var _i = 0, _a = excludeProperties(getPropertiesOfType(source), excludedProperties); _i < _a.length; _i++) {
+                        var sourceProp = _a[_i];
+                        if (!getPropertyOfObjectType(target, sourceProp.escapedName)) {
+                            var sourceType = getTypeOfSymbol(sourceProp);
+                            if (!(sourceType === undefinedType || sourceType === undefinedWideningType || sourceType === optionalType)) {
+                                if (reportErrors) {
+                                    reportError(ts.Diagnostics.Property_0_does_not_exist_on_type_1, symbolToString(sourceProp), typeToString(target));
+                                }
+                                return 0 /* False */;
+                            }
+                        }
+                    }
                 }
-                var targetType;
-                var prototypeProperty = getPropertyOfType(rightType, "prototype");
-                if (prototypeProperty) {
-                    // Target type is type of the prototype property
-                    var prototypePropertyType = getTypeOfSymbol(prototypeProperty);
-                    if (!isTypeAny(prototypePropertyType)) {
-                        targetType = prototypePropertyType;
+                // We only call this for union target types when we're attempting to do excess property checking - in those cases, we want to get _all possible props_
+                // from the target union, across all members
+                var properties = getPropertiesOfType(target);
+                var numericNamesOnly = isTupleType(source) && isTupleType(target);
+                for (var _b = 0, _c = excludeProperties(properties, excludedProperties); _b < _c.length; _b++) {
+                    var targetProp = _c[_b];
+                    var name = targetProp.escapedName;
+                    if (!(targetProp.flags & 4194304 /* Prototype */) && (!numericNamesOnly || isNumericLiteralName(name) || name === "length")) {
+                        var sourceProp = getPropertyOfType(source, name);
+                        if (sourceProp && sourceProp !== targetProp) {
+                            var related = propertyRelatedTo(source, target, sourceProp, targetProp, getTypeOfSymbol, reportErrors, intersectionState, relation === comparableRelation);
+                            if (!related) {
+                                return 0 /* False */;
+                            }
+                            result &= related;
+                        }
                     }
                 }
-                // Don't narrow from 'any' if the target type is exactly 'Object' or 'Function'
-                if (isTypeAny(type) && (targetType === globalObjectType || targetType === globalFunctionType)) {
-                    return type;
+                return result;
+            }
+            function propertiesIdenticalTo(source, target, excludedProperties) {
+                if (!(source.flags & 524288 /* Object */ && target.flags & 524288 /* Object */)) {
+                    return 0 /* False */;
                 }
-                if (!targetType) {
-                    var constructSignatures = getSignaturesOfType(rightType, 1 /* Construct */);
-                    targetType = constructSignatures.length ?
-                        getUnionType(ts.map(constructSignatures, function (signature) { return getReturnTypeOfSignature(getErasedSignature(signature)); })) :
-                        emptyObjectType;
+                var sourceProperties = excludeProperties(getPropertiesOfObjectType(source), excludedProperties);
+                var targetProperties = excludeProperties(getPropertiesOfObjectType(target), excludedProperties);
+                if (sourceProperties.length !== targetProperties.length) {
+                    return 0 /* False */;
                 }
-                return getNarrowedType(type, targetType, assumeTrue, isTypeDerivedFrom);
+                var result = -1 /* True */;
+                for (var _i = 0, sourceProperties_1 = sourceProperties; _i < sourceProperties_1.length; _i++) {
+                    var sourceProp = sourceProperties_1[_i];
+                    var targetProp = getPropertyOfObjectType(target, sourceProp.escapedName);
+                    if (!targetProp) {
+                        return 0 /* False */;
+                    }
+                    var related = compareProperties(sourceProp, targetProp, isRelatedTo);
+                    if (!related) {
+                        return 0 /* False */;
+                    }
+                    result &= related;
+                }
+                return result;
             }
-            function getNarrowedType(type, candidate, assumeTrue, isRelated) {
-                if (!assumeTrue) {
-                    return filterType(type, function (t) { return !isRelated(t, candidate); });
+            function signaturesRelatedTo(source, target, kind, reportErrors) {
+                var _a, _b;
+                if (relation === identityRelation) {
+                    return signaturesIdenticalTo(source, target, kind);
                 }
-                // If the current type is a union type, remove all constituents that couldn't be instances of
-                // the candidate type. If one or more constituents remain, return a union of those.
-                if (type.flags & 1048576 /* Union */) {
-                    var assignableType = filterType(type, function (t) { return isRelated(t, candidate); });
-                    if (!(assignableType.flags & 131072 /* Never */)) {
-                        return assignableType;
+                if (target === anyFunctionType || source === anyFunctionType) {
+                    return -1 /* True */;
+                }
+                var sourceIsJSConstructor = source.symbol && isJSConstructor(source.symbol.valueDeclaration);
+                var targetIsJSConstructor = target.symbol && isJSConstructor(target.symbol.valueDeclaration);
+                var sourceSignatures = getSignaturesOfType(source, (sourceIsJSConstructor && kind === 1 /* Construct */) ?
+                    0 /* Call */ : kind);
+                var targetSignatures = getSignaturesOfType(target, (targetIsJSConstructor && kind === 1 /* Construct */) ?
+                    0 /* Call */ : kind);
+                if (kind === 1 /* Construct */ && sourceSignatures.length && targetSignatures.length) {
+                    if (ts.isAbstractConstructorType(source) && !ts.isAbstractConstructorType(target)) {
+                        // An abstract constructor type is not assignable to a non-abstract constructor type
+                        // as it would otherwise be possible to new an abstract class. Note that the assignability
+                        // check we perform for an extends clause excludes construct signatures from the target,
+                        // so this check never proceeds.
+                        if (reportErrors) {
+                            reportError(ts.Diagnostics.Cannot_assign_an_abstract_constructor_type_to_a_non_abstract_constructor_type);
+                        }
+                        return 0 /* False */;
+                    }
+                    if (!constructorVisibilitiesAreCompatible(sourceSignatures[0], targetSignatures[0], reportErrors)) {
+                        return 0 /* False */;
+                    }
+                }
+                var result = -1 /* True */;
+                var saveErrorInfo = captureErrorCalculationState();
+                var incompatibleReporter = kind === 1 /* Construct */ ? reportIncompatibleConstructSignatureReturn : reportIncompatibleCallSignatureReturn;
+                var sourceObjectFlags = ts.getObjectFlags(source);
+                var targetObjectFlags = ts.getObjectFlags(target);
+                if (sourceObjectFlags & 64 /* Instantiated */ && targetObjectFlags & 64 /* Instantiated */ && source.symbol === target.symbol) {
+                    // We have instantiations of the same anonymous type (which typically will be the type of a
+                    // method). Simply do a pairwise comparison of the signatures in the two signature lists instead
+                    // of the much more expensive N * M comparison matrix we explore below. We erase type parameters
+                    // as they are known to always be the same.
+                    for (var i = 0; i < targetSignatures.length; i++) {
+                        var related = signatureRelatedTo(sourceSignatures[i], targetSignatures[i], /*erase*/ true, reportErrors, incompatibleReporter(sourceSignatures[i], targetSignatures[i]));
+                        if (!related) {
+                            return 0 /* False */;
+                        }
+                        result &= related;
                     }
                 }
-                // If the candidate type is a subtype of the target type, narrow to the candidate type.
-                // Otherwise, if the target type is assignable to the candidate type, keep the target type.
-                // Otherwise, if the candidate type is assignable to the target type, narrow to the candidate
-                // type. Otherwise, the types are completely unrelated, so narrow to an intersection of the
-                // two types.
-                return isTypeSubtypeOf(candidate, type) ? candidate :
-                    isTypeAssignableTo(type, candidate) ? type :
-                        isTypeAssignableTo(candidate, type) ? candidate :
-                            getIntersectionType([type, candidate]);
-            }
-            function narrowTypeByCallExpression(type, callExpression, assumeTrue) {
-                if (hasMatchingArgument(callExpression, reference)) {
-                    var signature = assumeTrue || !ts.isCallChain(callExpression) ? getEffectsSignature(callExpression) : undefined;
-                    var predicate = signature && getTypePredicateOfSignature(signature);
-                    if (predicate && (predicate.kind === 0 /* This */ || predicate.kind === 1 /* Identifier */)) {
-                        return narrowTypeByTypePredicate(type, predicate, callExpression, assumeTrue);
+                else if (sourceSignatures.length === 1 && targetSignatures.length === 1) {
+                    // For simple functions (functions with a single signature) we only erase type parameters for
+                    // the comparable relation. Otherwise, if the source signature is generic, we instantiate it
+                    // in the context of the target signature before checking the relationship. Ideally we'd do
+                    // this regardless of the number of signatures, but the potential costs are prohibitive due
+                    // to the quadratic nature of the logic below.
+                    var eraseGenerics = relation === comparableRelation || !!compilerOptions.noStrictGenericChecks;
+                    var sourceSignature = ts.first(sourceSignatures);
+                    var targetSignature = ts.first(targetSignatures);
+                    result = signatureRelatedTo(sourceSignature, targetSignature, eraseGenerics, reportErrors, incompatibleReporter(sourceSignature, targetSignature));
+                    if (!result && reportErrors && kind === 1 /* Construct */ && (sourceObjectFlags & targetObjectFlags) &&
+                        (((_a = targetSignature.declaration) === null || _a === void 0 ? void 0 : _a.kind) === 166 /* Constructor */ || ((_b = sourceSignature.declaration) === null || _b === void 0 ? void 0 : _b.kind) === 166 /* Constructor */)) {
+                        var constructSignatureToString = function (signature) {
+                            return signatureToString(signature, /*enclosingDeclaration*/ undefined, 262144 /* WriteArrowStyleSignature */, kind);
+                        };
+                        reportError(ts.Diagnostics.Type_0_is_not_assignable_to_type_1, constructSignatureToString(sourceSignature), constructSignatureToString(targetSignature));
+                        reportError(ts.Diagnostics.Types_of_construct_signatures_are_incompatible);
+                        return result;
                     }
                 }
-                return type;
-            }
-            function narrowTypeByTypePredicate(type, predicate, callExpression, assumeTrue) {
-                // Don't narrow from 'any' if the predicate type is exactly 'Object' or 'Function'
-                if (predicate.type && !(isTypeAny(type) && (predicate.type === globalObjectType || predicate.type === globalFunctionType))) {
-                    var predicateArgument = getTypePredicateArgument(predicate, callExpression);
-                    if (predicateArgument) {
-                        if (isMatchingReference(reference, predicateArgument)) {
-                            return getNarrowedType(type, predicate.type, assumeTrue, isTypeSubtypeOf);
-                        }
-                        if (strictNullChecks && assumeTrue && optionalChainContainsReference(predicateArgument, reference) &&
-                            !(getTypeFacts(predicate.type) & 65536 /* EQUndefined */)) {
-                            type = getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
+                else {
+                    outer: for (var _i = 0, targetSignatures_1 = targetSignatures; _i < targetSignatures_1.length; _i++) {
+                        var t = targetSignatures_1[_i];
+                        // Only elaborate errors from the first failure
+                        var shouldElaborateErrors = reportErrors;
+                        for (var _c = 0, sourceSignatures_1 = sourceSignatures; _c < sourceSignatures_1.length; _c++) {
+                            var s = sourceSignatures_1[_c];
+                            var related = signatureRelatedTo(s, t, /*erase*/ true, shouldElaborateErrors, incompatibleReporter(s, t));
+                            if (related) {
+                                result &= related;
+                                resetErrorInfo(saveErrorInfo);
+                                continue outer;
+                            }
+                            shouldElaborateErrors = false;
                         }
-                        if (isMatchingReferenceDiscriminant(predicateArgument, declaredType)) {
-                            return narrowTypeByDiscriminant(type, predicateArgument, function (t) { return getNarrowedType(t, predicate.type, assumeTrue, isTypeSubtypeOf); });
+                        if (shouldElaborateErrors) {
+                            reportError(ts.Diagnostics.Type_0_provides_no_match_for_the_signature_1, typeToString(source), signatureToString(t, /*enclosingDeclaration*/ undefined, /*flags*/ undefined, kind));
                         }
+                        return 0 /* False */;
                     }
                 }
-                return type;
+                return result;
             }
-            // Narrow the given type based on the given expression having the assumed boolean value. The returned type
-            // will be a subtype or the same type as the argument.
-            function narrowType(type, expr, assumeTrue) {
-                // for `a?.b`, we emulate a synthetic `a !== null && a !== undefined` condition for `a`
-                if (ts.isExpressionOfOptionalChainRoot(expr) ||
-                    ts.isBinaryExpression(expr.parent) && expr.parent.operatorToken.kind === 60 /* QuestionQuestionToken */ && expr.parent.left === expr) {
-                    return narrowTypeByOptionality(type, expr, assumeTrue);
-                }
-                switch (expr.kind) {
-                    case 75 /* Identifier */:
-                    case 104 /* ThisKeyword */:
-                    case 102 /* SuperKeyword */:
-                    case 194 /* PropertyAccessExpression */:
-                    case 195 /* ElementAccessExpression */:
-                        return narrowTypeByTruthiness(type, expr, assumeTrue);
-                    case 196 /* CallExpression */:
-                        return narrowTypeByCallExpression(type, expr, assumeTrue);
-                    case 200 /* ParenthesizedExpression */:
-                        return narrowType(type, expr.expression, assumeTrue);
-                    case 209 /* BinaryExpression */:
-                        return narrowTypeByBinaryExpression(type, expr, assumeTrue);
-                    case 207 /* PrefixUnaryExpression */:
-                        if (expr.operator === 53 /* ExclamationToken */) {
-                            return narrowType(type, expr.operand, !assumeTrue);
-                        }
-                        break;
+            function reportIncompatibleCallSignatureReturn(siga, sigb) {
+                if (siga.parameters.length === 0 && sigb.parameters.length === 0) {
+                    return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Call_signatures_with_no_arguments_have_incompatible_return_types_0_and_1, typeToString(source), typeToString(target)); };
                 }
-                return type;
+                return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Call_signature_return_types_0_and_1_are_incompatible, typeToString(source), typeToString(target)); };
             }
-            function narrowTypeByOptionality(type, expr, assumePresent) {
-                if (isMatchingReference(reference, expr)) {
-                    return getTypeWithFacts(type, assumePresent ? 2097152 /* NEUndefinedOrNull */ : 262144 /* EQUndefinedOrNull */);
-                }
-                if (isMatchingReferenceDiscriminant(expr, declaredType)) {
-                    return narrowTypeByDiscriminant(type, expr, function (t) { return getTypeWithFacts(t, assumePresent ? 2097152 /* NEUndefinedOrNull */ : 262144 /* EQUndefinedOrNull */); });
+            function reportIncompatibleConstructSignatureReturn(siga, sigb) {
+                if (siga.parameters.length === 0 && sigb.parameters.length === 0) {
+                    return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Construct_signatures_with_no_arguments_have_incompatible_return_types_0_and_1, typeToString(source), typeToString(target)); };
                 }
-                return type;
+                return function (source, target) { return reportIncompatibleError(ts.Diagnostics.Construct_signature_return_types_0_and_1_are_incompatible, typeToString(source), typeToString(target)); };
             }
-        }
-        function getTypeOfSymbolAtLocation(symbol, location) {
-            symbol = symbol.exportSymbol || symbol;
-            // If we have an identifier or a property access at the given location, if the location is
-            // an dotted name expression, and if the location is not an assignment target, obtain the type
-            // of the expression (which will reflect control flow analysis). If the expression indeed
-            // resolved to the given symbol, return the narrowed type.
-            if (location.kind === 75 /* Identifier */) {
-                if (ts.isRightSideOfQualifiedNameOrPropertyAccess(location)) {
-                    location = location.parent;
+            /**
+             * See signatureAssignableTo, compareSignaturesIdentical
+             */
+            function signatureRelatedTo(source, target, erase, reportErrors, incompatibleReporter) {
+                return compareSignaturesRelated(erase ? getErasedSignature(source) : source, erase ? getErasedSignature(target) : target, relation === strictSubtypeRelation ? 8 /* StrictArity */ : 0, reportErrors, reportError, incompatibleReporter, isRelatedTo, makeFunctionTypeMapper(reportUnreliableMarkers));
+            }
+            function signaturesIdenticalTo(source, target, kind) {
+                var sourceSignatures = getSignaturesOfType(source, kind);
+                var targetSignatures = getSignaturesOfType(target, kind);
+                if (sourceSignatures.length !== targetSignatures.length) {
+                    return 0 /* False */;
                 }
-                if (ts.isExpressionNode(location) && !ts.isAssignmentTarget(location)) {
-                    var type = getTypeOfExpression(location);
-                    if (getExportSymbolOfValueSymbolIfExported(getNodeLinks(location).resolvedSymbol) === symbol) {
-                        return type;
+                var result = -1 /* True */;
+                for (var i = 0; i < sourceSignatures.length; i++) {
+                    var related = compareSignaturesIdentical(sourceSignatures[i], targetSignatures[i], /*partialMatch*/ false, /*ignoreThisTypes*/ false, /*ignoreReturnTypes*/ false, isRelatedTo);
+                    if (!related) {
+                        return 0 /* False */;
                     }
+                    result &= related;
                 }
+                return result;
             }
-            // The location isn't a reference to the given symbol, meaning we're being asked
-            // a hypothetical question of what type the symbol would have if there was a reference
-            // to it at the given location. Since we have no control flow information for the
-            // hypothetical reference (control flow information is created and attached by the
-            // binder), we simply return the declared type of the symbol.
-            return getTypeOfSymbol(symbol);
-        }
-        function getControlFlowContainer(node) {
-            return ts.findAncestor(node.parent, function (node) {
-                return ts.isFunctionLike(node) && !ts.getImmediatelyInvokedFunctionExpression(node) ||
-                    node.kind === 250 /* ModuleBlock */ ||
-                    node.kind === 290 /* SourceFile */ ||
-                    node.kind === 159 /* PropertyDeclaration */;
-            });
-        }
-        // Check if a parameter is assigned anywhere within its declaring function.
-        function isParameterAssigned(symbol) {
-            var func = ts.getRootDeclaration(symbol.valueDeclaration).parent;
-            var links = getNodeLinks(func);
-            if (!(links.flags & 8388608 /* AssignmentsMarked */)) {
-                links.flags |= 8388608 /* AssignmentsMarked */;
-                if (!hasParentWithAssignmentsMarked(func)) {
-                    markParameterAssignments(func);
-                }
-            }
-            return symbol.isAssigned || false;
-        }
-        function hasParentWithAssignmentsMarked(node) {
-            return !!ts.findAncestor(node.parent, function (node) { return ts.isFunctionLike(node) && !!(getNodeLinks(node).flags & 8388608 /* AssignmentsMarked */); });
-        }
-        function markParameterAssignments(node) {
-            if (node.kind === 75 /* Identifier */) {
-                if (ts.isAssignmentTarget(node)) {
-                    var symbol = getResolvedSymbol(node);
-                    if (symbol.valueDeclaration && ts.getRootDeclaration(symbol.valueDeclaration).kind === 156 /* Parameter */) {
-                        symbol.isAssigned = true;
+            function eachPropertyRelatedTo(source, target, kind, reportErrors) {
+                var result = -1 /* True */;
+                var props = source.flags & 2097152 /* Intersection */ ? getPropertiesOfUnionOrIntersectionType(source) : getPropertiesOfObjectType(source);
+                for (var _i = 0, props_2 = props; _i < props_2.length; _i++) {
+                    var prop = props_2[_i];
+                    // Skip over ignored JSX and symbol-named members
+                    if (isIgnoredJsxProperty(source, prop)) {
+                        continue;
+                    }
+                    var nameType = getSymbolLinks(prop).nameType;
+                    if (nameType && nameType.flags & 8192 /* UniqueESSymbol */) {
+                        continue;
+                    }
+                    if (kind === 0 /* String */ || isNumericLiteralName(prop.escapedName)) {
+                        var related = isRelatedTo(getTypeOfSymbol(prop), target, reportErrors);
+                        if (!related) {
+                            if (reportErrors) {
+                                reportError(ts.Diagnostics.Property_0_is_incompatible_with_index_signature, symbolToString(prop));
+                            }
+                            return 0 /* False */;
+                        }
+                        result &= related;
                     }
                 }
+                return result;
             }
-            else {
-                ts.forEachChild(node, markParameterAssignments);
-            }
-        }
-        function isConstVariable(symbol) {
-            return symbol.flags & 3 /* Variable */ && (getDeclarationNodeFlagsFromSymbol(symbol) & 2 /* Const */) !== 0 && getTypeOfSymbol(symbol) !== autoArrayType;
-        }
-        /** remove undefined from the annotated type of a parameter when there is an initializer (that doesn't include undefined) */
-        function removeOptionalityFromDeclaredType(declaredType, declaration) {
-            if (pushTypeResolution(declaration.symbol, 2 /* DeclaredType */)) {
-                var annotationIncludesUndefined = strictNullChecks &&
-                    declaration.kind === 156 /* Parameter */ &&
-                    declaration.initializer &&
-                    getFalsyFlags(declaredType) & 32768 /* Undefined */ &&
-                    !(getFalsyFlags(checkExpression(declaration.initializer)) & 32768 /* Undefined */);
-                popTypeResolution();
-                return annotationIncludesUndefined ? getTypeWithFacts(declaredType, 524288 /* NEUndefined */) : declaredType;
-            }
-            else {
-                reportCircularityError(declaration.symbol);
-                return declaredType;
-            }
-        }
-        function isConstraintPosition(node) {
-            var parent = node.parent;
-            return parent.kind === 194 /* PropertyAccessExpression */ ||
-                parent.kind === 196 /* CallExpression */ && parent.expression === node ||
-                parent.kind === 195 /* ElementAccessExpression */ && parent.expression === node ||
-                parent.kind === 191 /* BindingElement */ && parent.name === node && !!parent.initializer;
-        }
-        function typeHasNullableConstraint(type) {
-            return type.flags & 58982400 /* InstantiableNonPrimitive */ && maybeTypeOfKind(getBaseConstraintOfType(type) || unknownType, 98304 /* Nullable */);
-        }
-        function getConstraintForLocation(type, node) {
-            // When a node is the left hand expression of a property access, element access, or call expression,
-            // and the type of the node includes type variables with constraints that are nullable, we fetch the
-            // apparent type of the node *before* performing control flow analysis such that narrowings apply to
-            // the constraint type.
-            if (type && isConstraintPosition(node) && forEachType(type, typeHasNullableConstraint)) {
-                return mapType(getWidenedType(type), getBaseConstraintOrType);
+            function indexTypeRelatedTo(sourceType, targetType, reportErrors) {
+                var related = isRelatedTo(sourceType, targetType, reportErrors);
+                if (!related && reportErrors) {
+                    reportError(ts.Diagnostics.Index_signatures_are_incompatible);
+                }
+                return related;
             }
-            return type;
-        }
-        function isExportOrExportExpression(location) {
-            return !!ts.findAncestor(location, function (e) { return e.parent && ts.isExportAssignment(e.parent) && e.parent.expression === e && ts.isEntityNameExpression(e); });
-        }
-        function markAliasReferenced(symbol, location) {
-            if (isNonLocalAlias(symbol, /*excludes*/ 111551 /* Value */) && !isInTypeQuery(location) && !getTypeOnlyAliasDeclaration(symbol)) {
-                if (compilerOptions.preserveConstEnums && isExportOrExportExpression(location) || !isConstEnumOrConstEnumOnlyModule(resolveAlias(symbol))) {
-                    markAliasSymbolAsReferenced(symbol);
+            function indexTypesRelatedTo(source, target, kind, sourceIsPrimitive, reportErrors, intersectionState) {
+                if (relation === identityRelation) {
+                    return indexTypesIdenticalTo(source, target, kind);
                 }
-                else {
-                    markConstEnumAliasAsReferenced(symbol);
+                var targetType = getIndexTypeOfType(target, kind);
+                if (!targetType || targetType.flags & 1 /* Any */ && !sourceIsPrimitive) {
+                    // Index signature of type any permits assignment from everything but primitives
+                    return -1 /* True */;
                 }
-            }
-        }
-        function checkIdentifier(node) {
-            var symbol = getResolvedSymbol(node);
-            if (symbol === unknownSymbol) {
-                return errorType;
-            }
-            // As noted in ECMAScript 6 language spec, arrow functions never have an arguments objects.
-            // Although in down-level emit of arrow function, we emit it using function expression which means that
-            // arguments objects will be bound to the inner object; emitting arrow function natively in ES6, arguments objects
-            // will be bound to non-arrow function that contain this arrow function. This results in inconsistent behavior.
-            // To avoid that we will give an error to users if they use arguments objects in arrow function so that they
-            // can explicitly bound arguments objects
-            if (symbol === argumentsSymbol) {
-                var container = ts.getContainingFunction(node);
-                if (languageVersion < 2 /* ES2015 */) {
-                    if (container.kind === 202 /* ArrowFunction */) {
-                        error(node, ts.Diagnostics.The_arguments_object_cannot_be_referenced_in_an_arrow_function_in_ES3_and_ES5_Consider_using_a_standard_function_expression);
-                    }
-                    else if (ts.hasModifier(container, 256 /* Async */)) {
-                        error(node, ts.Diagnostics.The_arguments_object_cannot_be_referenced_in_an_async_function_or_method_in_ES3_and_ES5_Consider_using_a_standard_function_or_method);
-                    }
+                if (isGenericMappedType(source)) {
+                    // A generic mapped type { [P in K]: T } is related to a type with an index signature
+                    // { [x: string]: U }, and optionally with an index signature { [x: number]: V },
+                    // if T is related to U and V.
+                    return getIndexTypeOfType(target, 0 /* String */) ? isRelatedTo(getTemplateTypeFromMappedType(source), targetType, reportErrors) : 0 /* False */;
                 }
-                getNodeLinks(container).flags |= 8192 /* CaptureArguments */;
-                return getTypeOfSymbol(symbol);
-            }
-            // We should only mark aliases as referenced if there isn't a local value declaration
-            // for the symbol. Also, don't mark any property access expression LHS - checkPropertyAccessExpression will handle that
-            if (!(node.parent && ts.isPropertyAccessExpression(node.parent) && node.parent.expression === node)) {
-                markAliasReferenced(symbol, node);
-            }
-            var localOrExportSymbol = getExportSymbolOfValueSymbolIfExported(symbol);
-            var declaration = localOrExportSymbol.valueDeclaration;
-            if (localOrExportSymbol.flags & 32 /* Class */) {
-                // Due to the emit for class decorators, any reference to the class from inside of the class body
-                // must instead be rewritten to point to a temporary variable to avoid issues with the double-bind
-                // behavior of class names in ES6.
-                if (declaration.kind === 245 /* ClassDeclaration */
-                    && ts.nodeIsDecorated(declaration)) {
-                    var container = ts.getContainingClass(node);
-                    while (container !== undefined) {
-                        if (container === declaration && container.name !== node) {
-                            getNodeLinks(declaration).flags |= 16777216 /* ClassWithConstructorReference */;
-                            getNodeLinks(node).flags |= 33554432 /* ConstructorReferenceInClass */;
-                            break;
-                        }
-                        container = ts.getContainingClass(container);
-                    }
+                var indexType = getIndexTypeOfType(source, kind) || kind === 1 /* Number */ && getIndexTypeOfType(source, 0 /* String */);
+                if (indexType) {
+                    return indexTypeRelatedTo(indexType, targetType, reportErrors);
                 }
-                else if (declaration.kind === 214 /* ClassExpression */) {
-                    // When we emit a class expression with static members that contain a reference
-                    // to the constructor in the initializer, we will need to substitute that
-                    // binding with an alias as the class name is not in scope.
-                    var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
-                    while (container.kind !== 290 /* SourceFile */) {
-                        if (container.parent === declaration) {
-                            if (container.kind === 159 /* PropertyDeclaration */ && ts.hasModifier(container, 32 /* Static */)) {
-                                getNodeLinks(declaration).flags |= 16777216 /* ClassWithConstructorReference */;
-                                getNodeLinks(node).flags |= 33554432 /* ConstructorReferenceInClass */;
-                            }
-                            break;
+                if (!(intersectionState & 1 /* Source */) && isObjectTypeWithInferableIndex(source)) {
+                    // Intersection constituents are never considered to have an inferred index signature
+                    var related = eachPropertyRelatedTo(source, targetType, kind, reportErrors);
+                    if (related && kind === 0 /* String */) {
+                        var numberIndexType = getIndexTypeOfType(source, 1 /* Number */);
+                        if (numberIndexType) {
+                            related &= indexTypeRelatedTo(numberIndexType, targetType, reportErrors);
                         }
-                        container = ts.getThisContainer(container, /*includeArrowFunctions*/ false);
                     }
+                    return related;
+                }
+                if (reportErrors) {
+                    reportError(ts.Diagnostics.Index_signature_is_missing_in_type_0, typeToString(source));
                 }
+                return 0 /* False */;
             }
-            checkNestedBlockScopedBinding(node, symbol);
-            var type = getConstraintForLocation(getTypeOfSymbol(localOrExportSymbol), node);
-            var assignmentKind = ts.getAssignmentTargetKind(node);
-            if (assignmentKind) {
-                if (!(localOrExportSymbol.flags & 3 /* Variable */) &&
-                    !(ts.isInJSFile(node) && localOrExportSymbol.flags & 512 /* ValueModule */)) {
-                    error(node, ts.Diagnostics.Cannot_assign_to_0_because_it_is_not_a_variable, symbolToString(symbol));
-                    return errorType;
+            function indexTypesIdenticalTo(source, target, indexKind) {
+                var targetInfo = getIndexInfoOfType(target, indexKind);
+                var sourceInfo = getIndexInfoOfType(source, indexKind);
+                if (!sourceInfo && !targetInfo) {
+                    return -1 /* True */;
                 }
-                if (isReadonlySymbol(localOrExportSymbol)) {
-                    if (localOrExportSymbol.flags & 3 /* Variable */) {
-                        error(node, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_constant, symbolToString(symbol));
-                    }
-                    else {
-                        error(node, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_read_only_property, symbolToString(symbol));
-                    }
-                    return errorType;
+                if (sourceInfo && targetInfo && sourceInfo.isReadonly === targetInfo.isReadonly) {
+                    return isRelatedTo(sourceInfo.type, targetInfo.type);
                 }
+                return 0 /* False */;
             }
-            var isAlias = localOrExportSymbol.flags & 2097152 /* Alias */;
-            // We only narrow variables and parameters occurring in a non-assignment position. For all other
-            // entities we simply return the declared type.
-            if (localOrExportSymbol.flags & 3 /* Variable */) {
-                if (assignmentKind === 1 /* Definite */) {
-                    return type;
+            function constructorVisibilitiesAreCompatible(sourceSignature, targetSignature, reportErrors) {
+                if (!sourceSignature.declaration || !targetSignature.declaration) {
+                    return true;
                 }
+                var sourceAccessibility = ts.getSelectedEffectiveModifierFlags(sourceSignature.declaration, 24 /* NonPublicAccessibilityModifier */);
+                var targetAccessibility = ts.getSelectedEffectiveModifierFlags(targetSignature.declaration, 24 /* NonPublicAccessibilityModifier */);
+                // A public, protected and private signature is assignable to a private signature.
+                if (targetAccessibility === 8 /* Private */) {
+                    return true;
+                }
+                // A public and protected signature is assignable to a protected signature.
+                if (targetAccessibility === 16 /* Protected */ && sourceAccessibility !== 8 /* Private */) {
+                    return true;
+                }
+                // Only a public signature is assignable to public signature.
+                if (targetAccessibility !== 16 /* Protected */ && !sourceAccessibility) {
+                    return true;
+                }
+                if (reportErrors) {
+                    reportError(ts.Diagnostics.Cannot_assign_a_0_constructor_type_to_a_1_constructor_type, visibilityToString(sourceAccessibility), visibilityToString(targetAccessibility));
+                }
+                return false;
             }
-            else if (isAlias) {
-                declaration = ts.find(symbol.declarations, isSomeImportDeclaration);
-            }
-            else {
-                return type;
-            }
-            if (!declaration) {
-                return type;
+        }
+        function typeCouldHaveTopLevelSingletonTypes(type) {
+            // Okay, yes, 'boolean' is a union of 'true | false', but that's not useful
+            // in error reporting scenarios. If you need to use this function but that detail matters,
+            // feel free to add a flag.
+            if (type.flags & 16 /* Boolean */) {
+                return false;
             }
-            // The declaration container is the innermost function that encloses the declaration of the variable
-            // or parameter. The flow container is the innermost function starting with which we analyze the control
-            // flow graph to determine the control flow based type.
-            var isParameter = ts.getRootDeclaration(declaration).kind === 156 /* Parameter */;
-            var declarationContainer = getControlFlowContainer(declaration);
-            var flowContainer = getControlFlowContainer(node);
-            var isOuterVariable = flowContainer !== declarationContainer;
-            var isSpreadDestructuringAssignmentTarget = node.parent && node.parent.parent && ts.isSpreadAssignment(node.parent) && isDestructuringAssignmentTarget(node.parent.parent);
-            var isModuleExports = symbol.flags & 134217728 /* ModuleExports */;
-            // When the control flow originates in a function expression or arrow function and we are referencing
-            // a const variable or parameter from an outer function, we extend the origin of the control flow
-            // analysis to include the immediately enclosing function.
-            while (flowContainer !== declarationContainer && (flowContainer.kind === 201 /* FunctionExpression */ ||
-                flowContainer.kind === 202 /* ArrowFunction */ || ts.isObjectLiteralOrClassExpressionMethod(flowContainer)) &&
-                (isConstVariable(localOrExportSymbol) || isParameter && !isParameterAssigned(localOrExportSymbol))) {
-                flowContainer = getControlFlowContainer(flowContainer);
+            if (type.flags & 3145728 /* UnionOrIntersection */) {
+                return !!ts.forEach(type.types, typeCouldHaveTopLevelSingletonTypes);
             }
-            // We only look for uninitialized variables in strict null checking mode, and only when we can analyze
-            // the entire control flow graph from the variable's declaration (i.e. when the flow container and
-            // declaration container are the same).
-            var assumeInitialized = isParameter || isAlias || isOuterVariable || isSpreadDestructuringAssignmentTarget || isModuleExports || ts.isBindingElement(declaration) ||
-                type !== autoType && type !== autoArrayType && (!strictNullChecks || (type.flags & (3 /* AnyOrUnknown */ | 16384 /* Void */)) !== 0 ||
-                    isInTypeQuery(node) || node.parent.kind === 263 /* ExportSpecifier */) ||
-                node.parent.kind === 218 /* NonNullExpression */ ||
-                declaration.kind === 242 /* VariableDeclaration */ && declaration.exclamationToken ||
-                declaration.flags & 8388608 /* Ambient */;
-            var initialType = assumeInitialized ? (isParameter ? removeOptionalityFromDeclaredType(type, declaration) : type) :
-                type === autoType || type === autoArrayType ? undefinedType :
-                    getOptionalType(type);
-            var flowType = getFlowTypeOfReference(node, type, initialType, flowContainer, !assumeInitialized);
-            // A variable is considered uninitialized when it is possible to analyze the entire control flow graph
-            // from declaration to use, and when the variable's declared type doesn't include undefined but the
-            // control flow based type does include undefined.
-            if (!isEvolvingArrayOperationTarget(node) && (type === autoType || type === autoArrayType)) {
-                if (flowType === autoType || flowType === autoArrayType) {
-                    if (noImplicitAny) {
-                        error(ts.getNameOfDeclaration(declaration), ts.Diagnostics.Variable_0_implicitly_has_type_1_in_some_locations_where_its_type_cannot_be_determined, symbolToString(symbol), typeToString(flowType));
-                        error(node, ts.Diagnostics.Variable_0_implicitly_has_an_1_type, symbolToString(symbol), typeToString(flowType));
-                    }
-                    return convertAutoToAny(flowType);
+            if (type.flags & 465829888 /* Instantiable */) {
+                var constraint = getConstraintOfType(type);
+                if (constraint && constraint !== type) {
+                    return typeCouldHaveTopLevelSingletonTypes(constraint);
                 }
             }
-            else if (!assumeInitialized && !(getFalsyFlags(type) & 32768 /* Undefined */) && getFalsyFlags(flowType) & 32768 /* Undefined */) {
-                error(node, ts.Diagnostics.Variable_0_is_used_before_being_assigned, symbolToString(symbol));
-                // Return the declared type to reduce follow-on errors
-                return type;
-            }
-            return assignmentKind ? getBaseTypeOfLiteralType(flowType) : flowType;
-        }
-        function isInsideFunction(node, threshold) {
-            return !!ts.findAncestor(node, function (n) { return n === threshold ? "quit" : ts.isFunctionLike(n); });
+            return isUnitType(type) || !!(type.flags & 134217728 /* TemplateLiteral */);
         }
-        function getPartOfForStatementContainingNode(node, container) {
-            return ts.findAncestor(node, function (n) { return n === container ? "quit" : n === container.initializer || n === container.condition || n === container.incrementor || n === container.statement; });
+        function getBestMatchingType(source, target, isRelatedTo) {
+            if (isRelatedTo === void 0) { isRelatedTo = compareTypesAssignable; }
+            return findMatchingDiscriminantType(source, target, isRelatedTo, /*skipPartial*/ true) ||
+                findMatchingTypeReferenceOrTypeAliasReference(source, target) ||
+                findBestTypeForObjectLiteral(source, target) ||
+                findBestTypeForInvokable(source, target) ||
+                findMostOverlappyType(source, target);
         }
-        function checkNestedBlockScopedBinding(node, symbol) {
-            if (languageVersion >= 2 /* ES2015 */ ||
-                (symbol.flags & (2 /* BlockScopedVariable */ | 32 /* Class */)) === 0 ||
-                ts.isSourceFile(symbol.valueDeclaration) ||
-                symbol.valueDeclaration.parent.kind === 280 /* CatchClause */) {
-                return;
-            }
-            // 1. walk from the use site up to the declaration and check
-            // if there is anything function like between declaration and use-site (is binding/class is captured in function).
-            // 2. walk from the declaration up to the boundary of lexical environment and check
-            // if there is an iteration statement in between declaration and boundary (is binding/class declared inside iteration statement)
-            var container = ts.getEnclosingBlockScopeContainer(symbol.valueDeclaration);
-            var usedInFunction = isInsideFunction(node.parent, container);
-            var current = container;
-            var containedInIterationStatement = false;
-            while (current && !ts.nodeStartsNewLexicalEnvironment(current)) {
-                if (ts.isIterationStatement(current, /*lookInLabeledStatements*/ false)) {
-                    containedInIterationStatement = true;
-                    break;
+        function discriminateTypeByDiscriminableItems(target, discriminators, related, defaultValue, skipPartial) {
+            // undefined=unknown, true=discriminated, false=not discriminated
+            // The state of each type progresses from left to right. Discriminated types stop at 'true'.
+            var discriminable = target.types.map(function (_) { return undefined; });
+            for (var _i = 0, discriminators_1 = discriminators; _i < discriminators_1.length; _i++) {
+                var _a = discriminators_1[_i], getDiscriminatingType = _a[0], propertyName = _a[1];
+                var targetProp = getUnionOrIntersectionProperty(target, propertyName);
+                if (skipPartial && targetProp && ts.getCheckFlags(targetProp) & 16 /* ReadPartial */) {
+                    continue;
                 }
-                current = current.parent;
-            }
-            if (containedInIterationStatement) {
-                if (usedInFunction) {
-                    // mark iteration statement as containing block-scoped binding captured in some function
-                    var capturesBlockScopeBindingInLoopBody = true;
-                    if (ts.isForStatement(container)) {
-                        var varDeclList = ts.getAncestor(symbol.valueDeclaration, 243 /* VariableDeclarationList */);
-                        if (varDeclList && varDeclList.parent === container) {
-                            var part = getPartOfForStatementContainingNode(node.parent, container);
-                            if (part) {
-                                var links = getNodeLinks(part);
-                                links.flags |= 131072 /* ContainsCapturedBlockScopeBinding */;
-                                var capturedBindings = links.capturedBlockScopeBindings || (links.capturedBlockScopeBindings = []);
-                                ts.pushIfUnique(capturedBindings, symbol);
-                                if (part === container.initializer) {
-                                    capturesBlockScopeBindingInLoopBody = false; // Initializer is outside of loop body
-                                }
-                            }
-                        }
-                    }
-                    if (capturesBlockScopeBindingInLoopBody) {
-                        getNodeLinks(current).flags |= 65536 /* LoopWithCapturedBlockScopedBinding */;
+                var i = 0;
+                for (var _b = 0, _c = target.types; _b < _c.length; _b++) {
+                    var type = _c[_b];
+                    var targetType = getTypeOfPropertyOfType(type, propertyName);
+                    if (targetType && related(getDiscriminatingType(), targetType)) {
+                        discriminable[i] = discriminable[i] === undefined ? true : discriminable[i];
                     }
-                }
-                // mark variables that are declared in loop initializer and reassigned inside the body of ForStatement.
-                // if body of ForStatement will be converted to function then we'll need a extra machinery to propagate reassigned values back.
-                if (ts.isForStatement(container)) {
-                    var varDeclList = ts.getAncestor(symbol.valueDeclaration, 243 /* VariableDeclarationList */);
-                    if (varDeclList && varDeclList.parent === container && isAssignedInBodyOfForStatement(node, container)) {
-                        getNodeLinks(symbol.valueDeclaration).flags |= 4194304 /* NeedsLoopOutParameter */;
+                    else {
+                        discriminable[i] = false;
                     }
+                    i++;
                 }
-                // set 'declared inside loop' bit on the block-scoped binding
-                getNodeLinks(symbol.valueDeclaration).flags |= 524288 /* BlockScopedBindingInLoop */;
-            }
-            if (usedInFunction) {
-                getNodeLinks(symbol.valueDeclaration).flags |= 262144 /* CapturedBlockScopedBinding */;
-            }
-        }
-        function isBindingCapturedByNode(node, decl) {
-            var links = getNodeLinks(node);
-            return !!links && ts.contains(links.capturedBlockScopeBindings, getSymbolOfNode(decl));
-        }
-        function isAssignedInBodyOfForStatement(node, container) {
-            // skip parenthesized nodes
-            var current = node;
-            while (current.parent.kind === 200 /* ParenthesizedExpression */) {
-                current = current.parent;
-            }
-            // check if node is used as LHS in some assignment expression
-            var isAssigned = false;
-            if (ts.isAssignmentTarget(current)) {
-                isAssigned = true;
-            }
-            else if ((current.parent.kind === 207 /* PrefixUnaryExpression */ || current.parent.kind === 208 /* PostfixUnaryExpression */)) {
-                var expr = current.parent;
-                isAssigned = expr.operator === 45 /* PlusPlusToken */ || expr.operator === 46 /* MinusMinusToken */;
-            }
-            if (!isAssigned) {
-                return false;
-            }
-            // at this point we know that node is the target of assignment
-            // now check that modification happens inside the statement part of the ForStatement
-            return !!ts.findAncestor(current, function (n) { return n === container ? "quit" : n === container.statement; });
-        }
-        function captureLexicalThis(node, container) {
-            getNodeLinks(node).flags |= 2 /* LexicalThis */;
-            if (container.kind === 159 /* PropertyDeclaration */ || container.kind === 162 /* Constructor */) {
-                var classNode = container.parent;
-                getNodeLinks(classNode).flags |= 4 /* CaptureThis */;
             }
-            else {
-                getNodeLinks(container).flags |= 4 /* CaptureThis */;
-            }
-        }
-        function findFirstSuperCall(n) {
-            if (ts.isSuperCall(n)) {
-                return n;
+            var match = discriminable.indexOf(/*searchElement*/ true);
+            if (match === -1) {
+                return defaultValue;
             }
-            else if (ts.isFunctionLike(n)) {
-                return undefined;
+            // make sure exactly 1 matches before returning it
+            var nextMatch = discriminable.indexOf(/*searchElement*/ true, match + 1);
+            while (nextMatch !== -1) {
+                if (!isTypeIdenticalTo(target.types[match], target.types[nextMatch])) {
+                    return defaultValue;
+                }
+                nextMatch = discriminable.indexOf(/*searchElement*/ true, nextMatch + 1);
             }
-            return ts.forEachChild(n, findFirstSuperCall);
+            return target.types[match];
         }
         /**
-         * Return a cached result if super-statement is already found.
-         * Otherwise, find a super statement in a given constructor function and cache the result in the node-links of the constructor
-         *
-         * @param constructor constructor-function to look for super statement
+         * A type is 'weak' if it is an object type with at least one optional property
+         * and no required properties, call/construct signatures or index signatures
          */
-        function getSuperCallInConstructor(constructor) {
-            var links = getNodeLinks(constructor);
-            // Only trying to find super-call if we haven't yet tried to find one.  Once we try, we will record the result
-            if (links.hasSuperCall === undefined) {
-                links.superCall = findFirstSuperCall(constructor.body);
-                links.hasSuperCall = links.superCall ? true : false;
+        function isWeakType(type) {
+            if (type.flags & 524288 /* Object */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                return resolved.callSignatures.length === 0 && resolved.constructSignatures.length === 0 &&
+                    !resolved.stringIndexInfo && !resolved.numberIndexInfo &&
+                    resolved.properties.length > 0 &&
+                    ts.every(resolved.properties, function (p) { return !!(p.flags & 16777216 /* Optional */); });
             }
-            return links.superCall;
-        }
-        /**
-         * Check if the given class-declaration extends null then return true.
-         * Otherwise, return false
-         * @param classDecl a class declaration to check if it extends null
-         */
-        function classDeclarationExtendsNull(classDecl) {
-            var classSymbol = getSymbolOfNode(classDecl);
-            var classInstanceType = getDeclaredTypeOfSymbol(classSymbol);
-            var baseConstructorType = getBaseConstructorTypeOfClass(classInstanceType);
-            return baseConstructorType === nullWideningType;
+            if (type.flags & 2097152 /* Intersection */) {
+                return ts.every(type.types, isWeakType);
+            }
+            return false;
         }
-        function checkThisBeforeSuper(node, container, diagnosticMessage) {
-            var containingClassDecl = container.parent;
-            var baseTypeNode = ts.getClassExtendsHeritageElement(containingClassDecl);
-            // If a containing class does not have extends clause or the class extends null
-            // skip checking whether super statement is called before "this" accessing.
-            if (baseTypeNode && !classDeclarationExtendsNull(containingClassDecl)) {
-                var superCall = getSuperCallInConstructor(container);
-                // We should give an error in the following cases:
-                //      - No super-call
-                //      - "this" is accessing before super-call.
-                //          i.e super(this)
-                //              this.x; super();
-                // We want to make sure that super-call is done before accessing "this" so that
-                // "this" is not accessed as a parameter of the super-call.
-                if (!superCall || superCall.end > node.pos) {
-                    // In ES6, super inside constructor of class-declaration has to precede "this" accessing
-                    error(node, diagnosticMessage);
+        function hasCommonProperties(source, target, isComparingJsxAttributes) {
+            for (var _i = 0, _a = getPropertiesOfType(source); _i < _a.length; _i++) {
+                var prop = _a[_i];
+                if (isKnownProperty(target, prop.escapedName, isComparingJsxAttributes)) {
+                    return true;
                 }
             }
+            return false;
         }
-        function checkThisExpression(node) {
-            // Stop at the first arrow function so that we can
-            // tell whether 'this' needs to be captured.
-            var container = ts.getThisContainer(node, /* includeArrowFunctions */ true);
-            var capturedByArrowFunction = false;
-            if (container.kind === 162 /* Constructor */) {
-                checkThisBeforeSuper(node, container, ts.Diagnostics.super_must_be_called_before_accessing_this_in_the_constructor_of_a_derived_class);
-            }
-            // Now skip arrow functions to get the "real" owner of 'this'.
-            if (container.kind === 202 /* ArrowFunction */) {
-                container = ts.getThisContainer(container, /* includeArrowFunctions */ false);
-                capturedByArrowFunction = true;
-            }
-            switch (container.kind) {
-                case 249 /* ModuleDeclaration */:
-                    error(node, ts.Diagnostics.this_cannot_be_referenced_in_a_module_or_namespace_body);
-                    // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
-                    break;
-                case 248 /* EnumDeclaration */:
-                    error(node, ts.Diagnostics.this_cannot_be_referenced_in_current_location);
-                    // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
-                    break;
-                case 162 /* Constructor */:
-                    if (isInConstructorArgumentInitializer(node, container)) {
-                        error(node, ts.Diagnostics.this_cannot_be_referenced_in_constructor_arguments);
-                        // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
-                    }
-                    break;
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                    if (ts.hasModifier(container, 32 /* Static */) && !(compilerOptions.target === 99 /* ESNext */ && compilerOptions.useDefineForClassFields)) {
-                        error(node, ts.Diagnostics.this_cannot_be_referenced_in_a_static_property_initializer);
-                        // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
+        // Return a type reference where the source type parameter is replaced with the target marker
+        // type, and flag the result as a marker type reference.
+        function getMarkerTypeReference(type, source, target) {
+            var result = createTypeReference(type, ts.map(type.typeParameters, function (t) { return t === source ? target : t; }));
+            result.objectFlags |= 8192 /* MarkerType */;
+            return result;
+        }
+        function getAliasVariances(symbol) {
+            var links = getSymbolLinks(symbol);
+            return getVariancesWorker(links.typeParameters, links, function (_links, param, marker) {
+                var type = getTypeAliasInstantiation(symbol, instantiateTypes(links.typeParameters, makeUnaryTypeMapper(param, marker)));
+                type.aliasTypeArgumentsContainsMarker = true;
+                return type;
+            });
+        }
+        // Return an array containing the variance of each type parameter. The variance is effectively
+        // a digest of the type comparisons that occur for each type argument when instantiations of the
+        // generic type are structurally compared. We infer the variance information by comparing
+        // instantiations of the generic type for type arguments with known relations. The function
+        // returns the emptyArray singleton when invoked recursively for the given generic type.
+        function getVariancesWorker(typeParameters, cache, createMarkerType) {
+            var _a, _b, _c;
+            if (typeParameters === void 0) { typeParameters = ts.emptyArray; }
+            var variances = cache.variances;
+            if (!variances) {
+                ts.tracing.push("check" /* Check */, "getVariancesWorker", { arity: typeParameters.length, id: (_c = (_a = cache.id) !== null && _a !== void 0 ? _a : (_b = cache.declaredType) === null || _b === void 0 ? void 0 : _b.id) !== null && _c !== void 0 ? _c : -1 });
+                // The emptyArray singleton is used to signal a recursive invocation.
+                cache.variances = ts.emptyArray;
+                variances = [];
+                var _loop_19 = function (tp) {
+                    var unmeasurable = false;
+                    var unreliable = false;
+                    var oldHandler = outofbandVarianceMarkerHandler;
+                    outofbandVarianceMarkerHandler = function (onlyUnreliable) { return onlyUnreliable ? unreliable = true : unmeasurable = true; };
+                    // We first compare instantiations where the type parameter is replaced with
+                    // marker types that have a known subtype relationship. From this we can infer
+                    // invariance, covariance, contravariance or bivariance.
+                    var typeWithSuper = createMarkerType(cache, tp, markerSuperType);
+                    var typeWithSub = createMarkerType(cache, tp, markerSubType);
+                    var variance = (isTypeAssignableTo(typeWithSub, typeWithSuper) ? 1 /* Covariant */ : 0) |
+                        (isTypeAssignableTo(typeWithSuper, typeWithSub) ? 2 /* Contravariant */ : 0);
+                    // If the instantiations appear to be related bivariantly it may be because the
+                    // type parameter is independent (i.e. it isn't witnessed anywhere in the generic
+                    // type). To determine this we compare instantiations where the type parameter is
+                    // replaced with marker types that are known to be unrelated.
+                    if (variance === 3 /* Bivariant */ && isTypeAssignableTo(createMarkerType(cache, tp, markerOtherType), typeWithSuper)) {
+                        variance = 4 /* Independent */;
                     }
-                    break;
-                case 154 /* ComputedPropertyName */:
-                    error(node, ts.Diagnostics.this_cannot_be_referenced_in_a_computed_property_name);
-                    break;
-            }
-            // When targeting es6, mark that we'll need to capture `this` in its lexically bound scope.
-            if (capturedByArrowFunction && languageVersion < 2 /* ES2015 */) {
-                captureLexicalThis(node, container);
-            }
-            var type = tryGetThisTypeAt(node, /*includeGlobalThis*/ true, container);
-            if (noImplicitThis) {
-                var globalThisType_1 = getTypeOfSymbol(globalThisSymbol);
-                if (type === globalThisType_1 && capturedByArrowFunction) {
-                    error(node, ts.Diagnostics.The_containing_arrow_function_captures_the_global_value_of_this);
-                }
-                else if (!type) {
-                    // With noImplicitThis, functions may not reference 'this' if it has type 'any'
-                    var diag = error(node, ts.Diagnostics.this_implicitly_has_type_any_because_it_does_not_have_a_type_annotation);
-                    if (!ts.isSourceFile(container)) {
-                        var outsideThis = tryGetThisTypeAt(container);
-                        if (outsideThis && outsideThis !== globalThisType_1) {
-                            ts.addRelatedInfo(diag, ts.createDiagnosticForNode(container, ts.Diagnostics.An_outer_value_of_this_is_shadowed_by_this_container));
+                    outofbandVarianceMarkerHandler = oldHandler;
+                    if (unmeasurable || unreliable) {
+                        if (unmeasurable) {
+                            variance |= 8 /* Unmeasurable */;
+                        }
+                        if (unreliable) {
+                            variance |= 16 /* Unreliable */;
                         }
                     }
+                    variances.push(variance);
+                };
+                for (var _i = 0, typeParameters_1 = typeParameters; _i < typeParameters_1.length; _i++) {
+                    var tp = typeParameters_1[_i];
+                    _loop_19(tp);
                 }
+                cache.variances = variances;
+                ts.tracing.pop();
             }
-            return type || anyType;
+            return variances;
         }
-        function tryGetThisTypeAt(node, includeGlobalThis, container) {
-            if (includeGlobalThis === void 0) { includeGlobalThis = true; }
-            if (container === void 0) { container = ts.getThisContainer(node, /*includeArrowFunctions*/ false); }
-            var isInJS = ts.isInJSFile(node);
-            if (ts.isFunctionLike(container) &&
-                (!isInParameterInitializerBeforeContainingFunction(node) || ts.getThisParameter(container))) {
-                // Note: a parameter initializer should refer to class-this unless function-this is explicitly annotated.
-                // If this is a function in a JS file, it might be a class method.
-                var className = getClassNameFromPrototypeMethod(container);
-                if (isInJS && className) {
-                    var classSymbol = checkExpression(className).symbol;
-                    if (classSymbol && classSymbol.members && (classSymbol.flags & 16 /* Function */)) {
-                        var classType = getDeclaredTypeOfSymbol(classSymbol).thisType;
-                        if (classType) {
-                            return getFlowTypeOfReference(node, classType);
-                        }
-                    }
-                }
-                // Check if it's a constructor definition, can be either a variable decl or function decl
-                // i.e.
-                //   * /** @constructor */ function [name]() { ... }
-                //   * /** @constructor */ var x = function() { ... }
-                else if (isInJS &&
-                    (container.kind === 201 /* FunctionExpression */ || container.kind === 244 /* FunctionDeclaration */) &&
-                    ts.getJSDocClassTag(container)) {
-                    var classType = getDeclaredTypeOfSymbol(getMergedSymbol(container.symbol)).thisType;
-                    return getFlowTypeOfReference(node, classType);
-                }
-                var thisType = getThisTypeOfDeclaration(container) || getContextualThisParameterType(container);
-                if (thisType) {
-                    return getFlowTypeOfReference(node, thisType);
-                }
-            }
-            if (ts.isClassLike(container.parent)) {
-                var symbol = getSymbolOfNode(container.parent);
-                var type = ts.hasModifier(container, 32 /* Static */) ? getTypeOfSymbol(symbol) : getDeclaredTypeOfSymbol(symbol).thisType;
-                return getFlowTypeOfReference(node, type);
-            }
-            if (isInJS) {
-                var type = getTypeForThisExpressionFromJSDoc(container);
-                if (type && type !== errorType) {
-                    return getFlowTypeOfReference(node, type);
-                }
-            }
-            if (ts.isSourceFile(container)) {
-                // look up in the source file's locals or exports
-                if (container.commonJsModuleIndicator) {
-                    var fileSymbol = getSymbolOfNode(container);
-                    return fileSymbol && getTypeOfSymbol(fileSymbol);
-                }
-                else if (includeGlobalThis) {
-                    return getTypeOfSymbol(globalThisSymbol);
-                }
+        function getVariances(type) {
+            // Arrays and tuples are known to be covariant, no need to spend time computing this.
+            if (type === globalArrayType || type === globalReadonlyArrayType || type.objectFlags & 8 /* Tuple */) {
+                return arrayVariances;
             }
+            return getVariancesWorker(type.typeParameters, type, getMarkerTypeReference);
         }
-        function getExplicitThisType(node) {
-            var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
-            if (ts.isFunctionLike(container)) {
-                var signature = getSignatureFromDeclaration(container);
-                if (signature.thisParameter) {
-                    return getExplicitTypeOfSymbol(signature.thisParameter);
+        // Return true if the given type reference has a 'void' type argument for a covariant type parameter.
+        // See comment at call in recursiveTypeRelatedTo for when this case matters.
+        function hasCovariantVoidArgument(typeArguments, variances) {
+            for (var i = 0; i < variances.length; i++) {
+                if ((variances[i] & 7 /* VarianceMask */) === 1 /* Covariant */ && typeArguments[i].flags & 16384 /* Void */) {
+                    return true;
                 }
             }
-            if (ts.isClassLike(container.parent)) {
-                var symbol = getSymbolOfNode(container.parent);
-                return ts.hasModifier(container, 32 /* Static */) ? getTypeOfSymbol(symbol) : getDeclaredTypeOfSymbol(symbol).thisType;
-            }
+            return false;
         }
-        function getClassNameFromPrototypeMethod(container) {
-            // Check if it's the RHS of a x.prototype.y = function [name]() { .... }
-            if (container.kind === 201 /* FunctionExpression */ &&
-                ts.isBinaryExpression(container.parent) &&
-                ts.getAssignmentDeclarationKind(container.parent) === 3 /* PrototypeProperty */) {
-                // Get the 'x' of 'x.prototype.y = container'
-                return container.parent // x.prototype.y = container
-                    .left // x.prototype.y
-                    .expression // x.prototype
-                    .expression; // x
-            }
-            // x.prototype = { method() { } }
-            else if (container.kind === 161 /* MethodDeclaration */ &&
-                container.parent.kind === 193 /* ObjectLiteralExpression */ &&
-                ts.isBinaryExpression(container.parent.parent) &&
-                ts.getAssignmentDeclarationKind(container.parent.parent) === 6 /* Prototype */) {
-                return container.parent.parent.left.expression;
-            }
-            // x.prototype = { method: function() { } }
-            else if (container.kind === 201 /* FunctionExpression */ &&
-                container.parent.kind === 281 /* PropertyAssignment */ &&
-                container.parent.parent.kind === 193 /* ObjectLiteralExpression */ &&
-                ts.isBinaryExpression(container.parent.parent.parent) &&
-                ts.getAssignmentDeclarationKind(container.parent.parent.parent) === 6 /* Prototype */) {
-                return container.parent.parent.parent.left.expression;
-            }
-            // Object.defineProperty(x, "method", { value: function() { } });
-            // Object.defineProperty(x, "method", { set: (x: () => void) => void });
-            // Object.defineProperty(x, "method", { get: () => function() { }) });
-            else if (container.kind === 201 /* FunctionExpression */ &&
-                ts.isPropertyAssignment(container.parent) &&
-                ts.isIdentifier(container.parent.name) &&
-                (container.parent.name.escapedText === "value" || container.parent.name.escapedText === "get" || container.parent.name.escapedText === "set") &&
-                ts.isObjectLiteralExpression(container.parent.parent) &&
-                ts.isCallExpression(container.parent.parent.parent) &&
-                container.parent.parent.parent.arguments[2] === container.parent.parent &&
-                ts.getAssignmentDeclarationKind(container.parent.parent.parent) === 9 /* ObjectDefinePrototypeProperty */) {
-                return container.parent.parent.parent.arguments[0].expression;
-            }
-            // Object.defineProperty(x, "method", { value() { } });
-            // Object.defineProperty(x, "method", { set(x: () => void) {} });
-            // Object.defineProperty(x, "method", { get() { return () => {} } });
-            else if (ts.isMethodDeclaration(container) &&
-                ts.isIdentifier(container.name) &&
-                (container.name.escapedText === "value" || container.name.escapedText === "get" || container.name.escapedText === "set") &&
-                ts.isObjectLiteralExpression(container.parent) &&
-                ts.isCallExpression(container.parent.parent) &&
-                container.parent.parent.arguments[2] === container.parent &&
-                ts.getAssignmentDeclarationKind(container.parent.parent) === 9 /* ObjectDefinePrototypeProperty */) {
-                return container.parent.parent.arguments[0].expression;
-            }
+        function isUnconstrainedTypeParameter(type) {
+            return type.flags & 262144 /* TypeParameter */ && !getConstraintOfTypeParameter(type);
         }
-        function getTypeForThisExpressionFromJSDoc(node) {
-            var jsdocType = ts.getJSDocType(node);
-            if (jsdocType && jsdocType.kind === 300 /* JSDocFunctionType */) {
-                var jsDocFunctionType = jsdocType;
-                if (jsDocFunctionType.parameters.length > 0 &&
-                    jsDocFunctionType.parameters[0].name &&
-                    jsDocFunctionType.parameters[0].name.escapedText === "this" /* This */) {
-                    return getTypeFromTypeNode(jsDocFunctionType.parameters[0].type);
-                }
-            }
-            var thisTag = ts.getJSDocThisTag(node);
-            if (thisTag && thisTag.typeExpression) {
-                return getTypeFromTypeNode(thisTag.typeExpression);
-            }
+        function isNonDeferredTypeReference(type) {
+            return !!(ts.getObjectFlags(type) & 4 /* Reference */) && !type.node;
         }
-        function isInConstructorArgumentInitializer(node, constructorDecl) {
-            return !!ts.findAncestor(node, function (n) { return ts.isFunctionLikeDeclaration(n) ? "quit" : n.kind === 156 /* Parameter */ && n.parent === constructorDecl; });
+        function isTypeReferenceWithGenericArguments(type) {
+            return isNonDeferredTypeReference(type) && ts.some(getTypeArguments(type), function (t) { return isUnconstrainedTypeParameter(t) || isTypeReferenceWithGenericArguments(t); });
         }
-        function checkSuperExpression(node) {
-            var isCallExpression = node.parent.kind === 196 /* CallExpression */ && node.parent.expression === node;
-            var container = ts.getSuperContainer(node, /*stopOnFunctions*/ true);
-            var needToCaptureLexicalThis = false;
-            // adjust the container reference in case if super is used inside arrow functions with arbitrarily deep nesting
-            if (!isCallExpression) {
-                while (container && container.kind === 202 /* ArrowFunction */) {
-                    container = ts.getSuperContainer(container, /*stopOnFunctions*/ true);
-                    needToCaptureLexicalThis = languageVersion < 2 /* ES2015 */;
-                }
-            }
-            var canUseSuperExpression = isLegalUsageOfSuperExpression(container);
-            var nodeCheckFlag = 0;
-            if (!canUseSuperExpression) {
-                // issue more specific error if super is used in computed property name
-                // class A { foo() { return "1" }}
-                // class B {
-                //     [super.foo()]() {}
-                // }
-                var current = ts.findAncestor(node, function (n) { return n === container ? "quit" : n.kind === 154 /* ComputedPropertyName */; });
-                if (current && current.kind === 154 /* ComputedPropertyName */) {
-                    error(node, ts.Diagnostics.super_cannot_be_referenced_in_a_computed_property_name);
-                }
-                else if (isCallExpression) {
-                    error(node, ts.Diagnostics.Super_calls_are_not_permitted_outside_constructors_or_in_nested_functions_inside_constructors);
-                }
-                else if (!container || !container.parent || !(ts.isClassLike(container.parent) || container.parent.kind === 193 /* ObjectLiteralExpression */)) {
-                    error(node, ts.Diagnostics.super_can_only_be_referenced_in_members_of_derived_classes_or_object_literal_expressions);
-                }
-                else {
-                    error(node, ts.Diagnostics.super_property_access_is_permitted_only_in_a_constructor_member_function_or_member_accessor_of_a_derived_class);
-                }
-                return errorType;
-            }
-            if (!isCallExpression && container.kind === 162 /* Constructor */) {
-                checkThisBeforeSuper(node, container, ts.Diagnostics.super_must_be_called_before_accessing_a_property_of_super_in_the_constructor_of_a_derived_class);
-            }
-            if (ts.hasModifier(container, 32 /* Static */) || isCallExpression) {
-                nodeCheckFlag = 512 /* SuperStatic */;
-            }
-            else {
-                nodeCheckFlag = 256 /* SuperInstance */;
-            }
-            getNodeLinks(node).flags |= nodeCheckFlag;
-            // Due to how we emit async functions, we need to specialize the emit for an async method that contains a `super` reference.
-            // This is due to the fact that we emit the body of an async function inside of a generator function. As generator
-            // functions cannot reference `super`, we emit a helper inside of the method body, but outside of the generator. This helper
-            // uses an arrow function, which is permitted to reference `super`.
-            //
-            // There are two primary ways we can access `super` from within an async method. The first is getting the value of a property
-            // or indexed access on super, either as part of a right-hand-side expression or call expression. The second is when setting the value
-            // of a property or indexed access, either as part of an assignment expression or destructuring assignment.
-            //
-            // The simplest case is reading a value, in which case we will emit something like the following:
-            //
-            //  // ts
-            //  ...
-            //  async asyncMethod() {
-            //    let x = await super.asyncMethod();
-            //    return x;
-            //  }
-            //  ...
-            //
-            //  // js
-            //  ...
-            //  asyncMethod() {
-            //      const _super = Object.create(null, {
-            //        asyncMethod: { get: () => super.asyncMethod },
-            //      });
-            //      return __awaiter(this, arguments, Promise, function *() {
-            //          let x = yield _super.asyncMethod.call(this);
-            //          return x;
-            //      });
-            //  }
-            //  ...
-            //
-            // The more complex case is when we wish to assign a value, especially as part of a destructuring assignment. As both cases
-            // are legal in ES6, but also likely less frequent, we only emit setters if there is an assignment:
-            //
-            //  // ts
-            //  ...
-            //  async asyncMethod(ar: Promise<any[]>) {
-            //      [super.a, super.b] = await ar;
-            //  }
-            //  ...
-            //
-            //  // js
-            //  ...
-            //  asyncMethod(ar) {
-            //      const _super = Object.create(null, {
-            //        a: { get: () => super.a, set: (v) => super.a = v },
-            //        b: { get: () => super.b, set: (v) => super.b = v }
-            //      };
-            //      return __awaiter(this, arguments, Promise, function *() {
-            //          [_super.a, _super.b] = yield ar;
-            //      });
-            //  }
-            //  ...
-            //
-            // Creating an object that has getter and setters instead of just an accessor function is required for destructuring assignments
-            // as a call expression cannot be used as the target of a destructuring assignment while a property access can.
-            //
-            // For element access expressions (`super[x]`), we emit a generic helper that forwards the element access in both situations.
-            if (container.kind === 161 /* MethodDeclaration */ && ts.hasModifier(container, 256 /* Async */)) {
-                if (ts.isSuperProperty(node.parent) && ts.isAssignmentTarget(node.parent)) {
-                    getNodeLinks(container).flags |= 4096 /* AsyncMethodWithSuperBinding */;
-                }
-                else {
-                    getNodeLinks(container).flags |= 2048 /* AsyncMethodWithSuper */;
+        /**
+         * getTypeReferenceId(A<T, number, U>) returns "111=0-12=1"
+         *   where A.id=111 and number.id=12
+         */
+        function getTypeReferenceId(type, typeParameters, depth) {
+            if (depth === void 0) { depth = 0; }
+            var result = "" + type.target.id;
+            for (var _i = 0, _a = getTypeArguments(type); _i < _a.length; _i++) {
+                var t = _a[_i];
+                if (isUnconstrainedTypeParameter(t)) {
+                    var index = typeParameters.indexOf(t);
+                    if (index < 0) {
+                        index = typeParameters.length;
+                        typeParameters.push(t);
+                    }
+                    result += "=" + index;
                 }
-            }
-            if (needToCaptureLexicalThis) {
-                // call expressions are allowed only in constructors so they should always capture correct 'this'
-                // super property access expressions can also appear in arrow functions -
-                // in this case they should also use correct lexical this
-                captureLexicalThis(node.parent, container);
-            }
-            if (container.parent.kind === 193 /* ObjectLiteralExpression */) {
-                if (languageVersion < 2 /* ES2015 */) {
-                    error(node, ts.Diagnostics.super_is_only_allowed_in_members_of_object_literal_expressions_when_option_target_is_ES2015_or_higher);
-                    return errorType;
+                else if (depth < 4 && isTypeReferenceWithGenericArguments(t)) {
+                    result += "<" + getTypeReferenceId(t, typeParameters, depth + 1) + ">";
                 }
                 else {
-                    // for object literal assume that type of 'super' is 'any'
-                    return anyType;
+                    result += "-" + t.id;
                 }
             }
-            // at this point the only legal case for parent is ClassLikeDeclaration
-            var classLikeDeclaration = container.parent;
-            if (!ts.getClassExtendsHeritageElement(classLikeDeclaration)) {
-                error(node, ts.Diagnostics.super_can_only_be_referenced_in_a_derived_class);
-                return errorType;
-            }
-            var classType = getDeclaredTypeOfSymbol(getSymbolOfNode(classLikeDeclaration));
-            var baseClassType = classType && getBaseTypes(classType)[0];
-            if (!baseClassType) {
-                return errorType;
+            return result;
+        }
+        /**
+         * To improve caching, the relation key for two generic types uses the target's id plus ids of the type parameters.
+         * For other cases, the types ids are used.
+         */
+        function getRelationKey(source, target, intersectionState, relation) {
+            if (relation === identityRelation && source.id > target.id) {
+                var temp = source;
+                source = target;
+                target = temp;
             }
-            if (container.kind === 162 /* Constructor */ && isInConstructorArgumentInitializer(node, container)) {
-                // issue custom error message for super property access in constructor arguments (to be aligned with old compiler)
-                error(node, ts.Diagnostics.super_cannot_be_referenced_in_constructor_arguments);
-                return errorType;
+            var postFix = intersectionState ? ":" + intersectionState : "";
+            if (isTypeReferenceWithGenericArguments(source) && isTypeReferenceWithGenericArguments(target)) {
+                var typeParameters = [];
+                return getTypeReferenceId(source, typeParameters) + "," + getTypeReferenceId(target, typeParameters) + postFix;
             }
-            return nodeCheckFlag === 512 /* SuperStatic */
-                ? getBaseConstructorTypeOfClass(classType)
-                : getTypeWithThisArgument(baseClassType, classType.thisType);
-            function isLegalUsageOfSuperExpression(container) {
-                if (!container) {
-                    return false;
-                }
-                if (isCallExpression) {
-                    // TS 1.0 SPEC (April 2014): 4.8.1
-                    // Super calls are only permitted in constructors of derived classes
-                    return container.kind === 162 /* Constructor */;
-                }
-                else {
-                    // TS 1.0 SPEC (April 2014)
-                    // 'super' property access is allowed
-                    // - In a constructor, instance member function, instance member accessor, or instance member variable initializer where this references a derived class instance
-                    // - In a static member function or static member accessor
-                    // topmost container must be something that is directly nested in the class declaration\object literal expression
-                    if (ts.isClassLike(container.parent) || container.parent.kind === 193 /* ObjectLiteralExpression */) {
-                        if (ts.hasModifier(container, 32 /* Static */)) {
-                            return container.kind === 161 /* MethodDeclaration */ ||
-                                container.kind === 160 /* MethodSignature */ ||
-                                container.kind === 163 /* GetAccessor */ ||
-                                container.kind === 164 /* SetAccessor */;
-                        }
-                        else {
-                            return container.kind === 161 /* MethodDeclaration */ ||
-                                container.kind === 160 /* MethodSignature */ ||
-                                container.kind === 163 /* GetAccessor */ ||
-                                container.kind === 164 /* SetAccessor */ ||
-                                container.kind === 159 /* PropertyDeclaration */ ||
-                                container.kind === 158 /* PropertySignature */ ||
-                                container.kind === 162 /* Constructor */;
-                        }
+            return source.id + "," + target.id + postFix;
+        }
+        // Invoke the callback for each underlying property symbol of the given symbol and return the first
+        // value that isn't undefined.
+        function forEachProperty(prop, callback) {
+            if (ts.getCheckFlags(prop) & 6 /* Synthetic */) {
+                for (var _i = 0, _a = prop.containingType.types; _i < _a.length; _i++) {
+                    var t = _a[_i];
+                    var p = getPropertyOfType(t, prop.escapedName);
+                    var result = p && forEachProperty(p, callback);
+                    if (result) {
+                        return result;
                     }
                 }
-                return false;
+                return undefined;
             }
+            return callback(prop);
         }
-        function getContainingObjectLiteral(func) {
-            return (func.kind === 161 /* MethodDeclaration */ ||
-                func.kind === 163 /* GetAccessor */ ||
-                func.kind === 164 /* SetAccessor */) && func.parent.kind === 193 /* ObjectLiteralExpression */ ? func.parent :
-                func.kind === 201 /* FunctionExpression */ && func.parent.kind === 281 /* PropertyAssignment */ ? func.parent.parent :
-                    undefined;
-        }
-        function getThisTypeArgument(type) {
-            return ts.getObjectFlags(type) & 4 /* Reference */ && type.target === globalThisType ? getTypeArguments(type)[0] : undefined;
-        }
-        function getThisTypeFromContextualType(type) {
-            return mapType(type, function (t) {
-                return t.flags & 2097152 /* Intersection */ ? ts.forEach(t.types, getThisTypeArgument) : getThisTypeArgument(t);
-            });
+        // Return the declaring class type of a property or undefined if property not declared in class
+        function getDeclaringClass(prop) {
+            return prop.parent && prop.parent.flags & 32 /* Class */ ? getDeclaredTypeOfSymbol(getParentOfSymbol(prop)) : undefined;
         }
-        function getContextualThisParameterType(func) {
-            if (func.kind === 202 /* ArrowFunction */) {
-                return undefined;
-            }
-            if (isContextSensitiveFunctionOrObjectLiteralMethod(func)) {
-                var contextualSignature = getContextualSignature(func);
-                if (contextualSignature) {
-                    var thisParameter = contextualSignature.thisParameter;
-                    if (thisParameter) {
-                        return getTypeOfSymbol(thisParameter);
-                    }
-                }
-            }
-            var inJs = ts.isInJSFile(func);
-            if (noImplicitThis || inJs) {
-                var containingLiteral = getContainingObjectLiteral(func);
-                if (containingLiteral) {
-                    // We have an object literal method. Check if the containing object literal has a contextual type
-                    // that includes a ThisType<T>. If so, T is the contextual type for 'this'. We continue looking in
-                    // any directly enclosing object literals.
-                    var contextualType = getApparentTypeOfContextualType(containingLiteral);
-                    var literal = containingLiteral;
-                    var type = contextualType;
-                    while (type) {
-                        var thisType = getThisTypeFromContextualType(type);
-                        if (thisType) {
-                            return instantiateType(thisType, getMapperFromContext(getInferenceContext(containingLiteral)));
-                        }
-                        if (literal.parent.kind !== 281 /* PropertyAssignment */) {
-                            break;
-                        }
-                        literal = literal.parent.parent;
-                        type = getApparentTypeOfContextualType(literal);
-                    }
-                    // There was no contextual ThisType<T> for the containing object literal, so the contextual type
-                    // for 'this' is the non-null form of the contextual type for the containing object literal or
-                    // the type of the object literal itself.
-                    return getWidenedType(contextualType ? getNonNullableType(contextualType) : checkExpressionCached(containingLiteral));
-                }
-                // In an assignment of the form 'obj.xxx = function(...)' or 'obj[xxx] = function(...)', the
-                // contextual type for 'this' is 'obj'.
-                var parent = ts.walkUpParenthesizedExpressions(func.parent);
-                if (parent.kind === 209 /* BinaryExpression */ && parent.operatorToken.kind === 62 /* EqualsToken */) {
-                    var target = parent.left;
-                    if (ts.isAccessExpression(target)) {
-                        var expression = target.expression;
-                        // Don't contextually type `this` as `exports` in `exports.Point = function(x, y) { this.x = x; this.y = y; }`
-                        if (inJs && ts.isIdentifier(expression)) {
-                            var sourceFile = ts.getSourceFileOfNode(parent);
-                            if (sourceFile.commonJsModuleIndicator && getResolvedSymbol(expression) === sourceFile.symbol) {
-                                return undefined;
+        // Return the inherited type of the given property or undefined if property doesn't exist in a base class.
+        function getTypeOfPropertyInBaseClass(property) {
+            var classType = getDeclaringClass(property);
+            var baseClassType = classType && getBaseTypes(classType)[0];
+            return baseClassType && getTypeOfPropertyOfType(baseClassType, property.escapedName);
+        }
+        // Return true if some underlying source property is declared in a class that derives
+        // from the given base class.
+        function isPropertyInClassDerivedFrom(prop, baseClass) {
+            return forEachProperty(prop, function (sp) {
+                var sourceClass = getDeclaringClass(sp);
+                return sourceClass ? hasBaseType(sourceClass, baseClass) : false;
+            });
+        }
+        // Return true if source property is a valid override of protected parts of target property.
+        function isValidOverrideOf(sourceProp, targetProp) {
+            return !forEachProperty(targetProp, function (tp) { return ts.getDeclarationModifierFlagsFromSymbol(tp) & 16 /* Protected */ ?
+                !isPropertyInClassDerivedFrom(sourceProp, getDeclaringClass(tp)) : false; });
+        }
+        // Return true if the given class derives from each of the declaring classes of the protected
+        // constituents of the given property.
+        function isClassDerivedFromDeclaringClasses(checkClass, prop) {
+            return forEachProperty(prop, function (p) { return ts.getDeclarationModifierFlagsFromSymbol(p) & 16 /* Protected */ ?
+                !hasBaseType(checkClass, getDeclaringClass(p)) : false; }) ? undefined : checkClass;
+        }
+        // Return true if the given type is deeply nested. We consider this to be the case when structural type comparisons
+        // for 5 or more occurrences or instantiations of the type have been recorded on the given stack. It is possible,
+        // though highly unlikely, for this test to be true in a situation where a chain of instantiations is not infinitely
+        // expanding. Effectively, we will generate a false positive when two types are structurally equal to at least 5
+        // levels, but unequal at some level beyond that.
+        // In addition, this will also detect when an indexed access has been chained off of 5 or more times (which is essentially
+        // the dual of the structural comparison), and likewise mark the type as deeply nested, potentially adding false positives
+        // for finite but deeply expanding indexed accesses (eg, for `Q[P1][P2][P3][P4][P5]`).
+        // It also detects when a recursive type reference has expanded 5 or more times, eg, if the true branch of
+        // `type A<T> = null extends T ? [A<NonNullable<T>>] : [T]`
+        // has expanded into `[A<NonNullable<NonNullable<NonNullable<NonNullable<NonNullable<T>>>>>>]`
+        // in such cases we need to terminate the expansion, and we do so here.
+        function isDeeplyNestedType(type, stack, depth) {
+            if (depth >= 5) {
+                var identity_1 = getRecursionIdentity(type);
+                if (identity_1) {
+                    var count = 0;
+                    for (var i = 0; i < depth; i++) {
+                        if (getRecursionIdentity(stack[i]) === identity_1) {
+                            count++;
+                            if (count >= 5) {
+                                return true;
                             }
                         }
-                        return getWidenedType(checkExpressionCached(expression));
                     }
                 }
             }
+            return false;
+        }
+        // Types with constituents that could circularly reference the type have a recursion identity. The recursion
+        // identity is some object that is common to instantiations of the type with the same origin.
+        function getRecursionIdentity(type) {
+            if (type.flags & 524288 /* Object */ && !isObjectOrArrayLiteralType(type)) {
+                if (ts.getObjectFlags(type) && 4 /* Reference */ && type.node) {
+                    // Deferred type references are tracked through their associated AST node. This gives us finer
+                    // granularity than using their associated target because each manifest type reference has a
+                    // unique AST node.
+                    return type.node;
+                }
+                if (type.symbol && !(ts.getObjectFlags(type) & 16 /* Anonymous */ && type.symbol.flags & 32 /* Class */)) {
+                    // We track all object types that have an associated symbol (representing the origin of the type), but
+                    // exclude the static side of classes from this check since it shares its symbol with the instance side.
+                    return type.symbol;
+                }
+                if (isTupleType(type)) {
+                    // Tuple types are tracked through their target type
+                    return type.target;
+                }
+            }
+            if (type.flags & 8388608 /* IndexedAccess */) {
+                // Identity is the leftmost object type in a chain of indexed accesses, eg, in A[P][Q] it is A
+                do {
+                    type = type.objectType;
+                } while (type.flags & 8388608 /* IndexedAccess */);
+                return type;
+            }
+            if (type.flags & 16777216 /* Conditional */) {
+                // The root object represents the origin of the conditional type
+                return type.root;
+            }
             return undefined;
         }
-        // Return contextual type of parameter or undefined if no contextual type is available
-        function getContextuallyTypedParameterType(parameter) {
-            var func = parameter.parent;
-            if (!isContextSensitiveFunctionOrObjectLiteralMethod(func)) {
-                return undefined;
+        function isPropertyIdenticalTo(sourceProp, targetProp) {
+            return compareProperties(sourceProp, targetProp, compareTypesIdentical) !== 0 /* False */;
+        }
+        function compareProperties(sourceProp, targetProp, compareTypes) {
+            // Two members are considered identical when
+            // - they are public properties with identical names, optionality, and types,
+            // - they are private or protected properties originating in the same declaration and having identical types
+            if (sourceProp === targetProp) {
+                return -1 /* True */;
             }
-            var iife = ts.getImmediatelyInvokedFunctionExpression(func);
-            if (iife && iife.arguments) {
-                var args = getEffectiveCallArguments(iife);
-                var indexOfParameter = func.parameters.indexOf(parameter);
-                if (parameter.dotDotDotToken) {
-                    return getSpreadArgumentType(args, indexOfParameter, args.length, anyType, /*context*/ undefined);
+            var sourcePropAccessibility = ts.getDeclarationModifierFlagsFromSymbol(sourceProp) & 24 /* NonPublicAccessibilityModifier */;
+            var targetPropAccessibility = ts.getDeclarationModifierFlagsFromSymbol(targetProp) & 24 /* NonPublicAccessibilityModifier */;
+            if (sourcePropAccessibility !== targetPropAccessibility) {
+                return 0 /* False */;
+            }
+            if (sourcePropAccessibility) {
+                if (getTargetSymbol(sourceProp) !== getTargetSymbol(targetProp)) {
+                    return 0 /* False */;
                 }
-                var links = getNodeLinks(iife);
-                var cached = links.resolvedSignature;
-                links.resolvedSignature = anySignature;
-                var type = indexOfParameter < args.length ?
-                    getWidenedLiteralType(checkExpression(args[indexOfParameter])) :
-                    parameter.initializer ? undefined : undefinedWideningType;
-                links.resolvedSignature = cached;
-                return type;
             }
-            var contextualSignature = getContextualSignature(func);
-            if (contextualSignature) {
-                var index = func.parameters.indexOf(parameter) - (ts.getThisParameter(func) ? 1 : 0);
-                return parameter.dotDotDotToken && ts.lastOrUndefined(func.parameters) === parameter ?
-                    getRestTypeAtPosition(contextualSignature, index) :
-                    tryGetTypeAtPosition(contextualSignature, index);
+            else {
+                if ((sourceProp.flags & 16777216 /* Optional */) !== (targetProp.flags & 16777216 /* Optional */)) {
+                    return 0 /* False */;
+                }
+            }
+            if (isReadonlySymbol(sourceProp) !== isReadonlySymbol(targetProp)) {
+                return 0 /* False */;
             }
+            return compareTypes(getTypeOfSymbol(sourceProp), getTypeOfSymbol(targetProp));
         }
-        function getContextualTypeForVariableLikeDeclaration(declaration) {
-            var typeNode = ts.getEffectiveTypeAnnotationNode(declaration);
-            if (typeNode) {
-                return getTypeFromTypeNode(typeNode);
+        function isMatchingSignature(source, target, partialMatch) {
+            var sourceParameterCount = getParameterCount(source);
+            var targetParameterCount = getParameterCount(target);
+            var sourceMinArgumentCount = getMinArgumentCount(source);
+            var targetMinArgumentCount = getMinArgumentCount(target);
+            var sourceHasRestParameter = hasEffectiveRestParameter(source);
+            var targetHasRestParameter = hasEffectiveRestParameter(target);
+            // A source signature matches a target signature if the two signatures have the same number of required,
+            // optional, and rest parameters.
+            if (sourceParameterCount === targetParameterCount &&
+                sourceMinArgumentCount === targetMinArgumentCount &&
+                sourceHasRestParameter === targetHasRestParameter) {
+                return true;
             }
-            switch (declaration.kind) {
-                case 156 /* Parameter */:
-                    return getContextuallyTypedParameterType(declaration);
-                case 191 /* BindingElement */:
-                    return getContextualTypeForBindingElement(declaration);
-                // By default, do nothing and return undefined - only parameters and binding elements have context implied by a parent
+            // A source signature partially matches a target signature if the target signature has no fewer required
+            // parameters
+            if (partialMatch && sourceMinArgumentCount <= targetMinArgumentCount) {
+                return true;
             }
+            return false;
         }
-        function getContextualTypeForBindingElement(declaration) {
-            var parent = declaration.parent.parent;
-            var name = declaration.propertyName || declaration.name;
-            var parentType = getContextualTypeForVariableLikeDeclaration(parent) ||
-                parent.kind !== 191 /* BindingElement */ && parent.initializer && checkDeclarationInitializer(parent);
-            if (parentType && !ts.isBindingPattern(name) && !ts.isComputedNonLiteralName(name)) {
-                var nameType = getLiteralTypeFromPropertyName(name);
-                if (isTypeUsableAsPropertyName(nameType)) {
-                    var text = getPropertyNameFromType(nameType);
-                    return getTypeOfPropertyOfType(parentType, text);
+        /**
+         * See signatureRelatedTo, compareSignaturesIdentical
+         */
+        function compareSignaturesIdentical(source, target, partialMatch, ignoreThisTypes, ignoreReturnTypes, compareTypes) {
+            // TODO (drosen): De-duplicate code between related functions.
+            if (source === target) {
+                return -1 /* True */;
+            }
+            if (!(isMatchingSignature(source, target, partialMatch))) {
+                return 0 /* False */;
+            }
+            // Check that the two signatures have the same number of type parameters.
+            if (ts.length(source.typeParameters) !== ts.length(target.typeParameters)) {
+                return 0 /* False */;
+            }
+            // Check that type parameter constraints and defaults match. If they do, instantiate the source
+            // signature with the type parameters of the target signature and continue the comparison.
+            if (target.typeParameters) {
+                var mapper = createTypeMapper(source.typeParameters, target.typeParameters);
+                for (var i = 0; i < target.typeParameters.length; i++) {
+                    var s = source.typeParameters[i];
+                    var t = target.typeParameters[i];
+                    if (!(s === t || compareTypes(instantiateType(getConstraintFromTypeParameter(s), mapper) || unknownType, getConstraintFromTypeParameter(t) || unknownType) &&
+                        compareTypes(instantiateType(getDefaultFromTypeParameter(s), mapper) || unknownType, getDefaultFromTypeParameter(t) || unknownType))) {
+                        return 0 /* False */;
+                    }
                 }
+                source = instantiateSignature(source, mapper, /*eraseTypeParameters*/ true);
             }
-        }
-        // In a variable, parameter or property declaration with a type annotation,
-        //   the contextual type of an initializer expression is the type of the variable, parameter or property.
-        // Otherwise, in a parameter declaration of a contextually typed function expression,
-        //   the contextual type of an initializer expression is the contextual type of the parameter.
-        // Otherwise, in a variable or parameter declaration with a binding pattern name,
-        //   the contextual type of an initializer expression is the type implied by the binding pattern.
-        // Otherwise, in a binding pattern inside a variable or parameter declaration,
-        //   the contextual type of an initializer expression is the type annotation of the containing declaration, if present.
-        function getContextualTypeForInitializerExpression(node) {
-            var declaration = node.parent;
-            if (ts.hasInitializer(declaration) && node === declaration.initializer) {
-                var result = getContextualTypeForVariableLikeDeclaration(declaration);
-                if (result) {
-                    return result;
+            var result = -1 /* True */;
+            if (!ignoreThisTypes) {
+                var sourceThisType = getThisTypeOfSignature(source);
+                if (sourceThisType) {
+                    var targetThisType = getThisTypeOfSignature(target);
+                    if (targetThisType) {
+                        var related = compareTypes(sourceThisType, targetThisType);
+                        if (!related) {
+                            return 0 /* False */;
+                        }
+                        result &= related;
+                    }
                 }
-                if (ts.isBindingPattern(declaration.name)) { // This is less a contextual type and more an implied shape - in some cases, this may be undesirable
-                    return getTypeFromBindingPattern(declaration.name, /*includePatternInType*/ true, /*reportErrors*/ false);
+            }
+            var targetLen = getParameterCount(target);
+            for (var i = 0; i < targetLen; i++) {
+                var s = getTypeAtPosition(source, i);
+                var t = getTypeAtPosition(target, i);
+                var related = compareTypes(t, s);
+                if (!related) {
+                    return 0 /* False */;
                 }
+                result &= related;
             }
-            return undefined;
+            if (!ignoreReturnTypes) {
+                var sourceTypePredicate = getTypePredicateOfSignature(source);
+                var targetTypePredicate = getTypePredicateOfSignature(target);
+                result &= sourceTypePredicate || targetTypePredicate ?
+                    compareTypePredicatesIdentical(sourceTypePredicate, targetTypePredicate, compareTypes) :
+                    compareTypes(getReturnTypeOfSignature(source), getReturnTypeOfSignature(target));
+            }
+            return result;
         }
-        function getContextualTypeForReturnExpression(node) {
-            var func = ts.getContainingFunction(node);
-            if (func) {
-                var functionFlags = ts.getFunctionFlags(func);
-                if (functionFlags & 1 /* Generator */) { // AsyncGenerator function or Generator function
-                    return undefined;
+        function compareTypePredicatesIdentical(source, target, compareTypes) {
+            return !(source && target && typePredicateKindsMatch(source, target)) ? 0 /* False */ :
+                source.type === target.type ? -1 /* True */ :
+                    source.type && target.type ? compareTypes(source.type, target.type) :
+                        0 /* False */;
+        }
+        function literalTypesWithSameBaseType(types) {
+            var commonBaseType;
+            for (var _i = 0, types_13 = types; _i < types_13.length; _i++) {
+                var t = types_13[_i];
+                var baseType = getBaseTypeOfLiteralType(t);
+                if (!commonBaseType) {
+                    commonBaseType = baseType;
                 }
-                var contextualReturnType = getContextualReturnType(func);
-                if (contextualReturnType) {
-                    if (functionFlags & 2 /* Async */) { // Async function
-                        var contextualAwaitedType = mapType(contextualReturnType, getAwaitedTypeOfPromise);
-                        return contextualAwaitedType && getUnionType([contextualAwaitedType, createPromiseLikeType(contextualAwaitedType)]);
-                    }
-                    return contextualReturnType; // Regular function
+                if (baseType === t || baseType !== commonBaseType) {
+                    return false;
                 }
             }
-            return undefined;
+            return true;
         }
-        function getContextualTypeForAwaitOperand(node) {
-            var contextualType = getContextualType(node);
-            if (contextualType) {
-                var contextualAwaitedType = getAwaitedType(contextualType);
-                return contextualAwaitedType && getUnionType([contextualAwaitedType, createPromiseLikeType(contextualAwaitedType)]);
+        // When the candidate types are all literal types with the same base type, return a union
+        // of those literal types. Otherwise, return the leftmost type for which no type to the
+        // right is a supertype.
+        function getSupertypeOrUnion(types) {
+            return literalTypesWithSameBaseType(types) ?
+                getUnionType(types) :
+                ts.reduceLeft(types, function (s, t) { return isTypeSubtypeOf(s, t) ? t : s; });
+        }
+        function getCommonSupertype(types) {
+            if (!strictNullChecks) {
+                return getSupertypeOrUnion(types);
             }
-            return undefined;
+            var primaryTypes = ts.filter(types, function (t) { return !(t.flags & 98304 /* Nullable */); });
+            return primaryTypes.length ?
+                getNullableType(getSupertypeOrUnion(primaryTypes), getFalsyFlagsOfTypes(types) & 98304 /* Nullable */) :
+                getUnionType(types, 2 /* Subtype */);
         }
-        function getContextualTypeForYieldOperand(node) {
-            var func = ts.getContainingFunction(node);
-            if (func) {
-                var functionFlags = ts.getFunctionFlags(func);
-                var contextualReturnType = getContextualReturnType(func);
-                if (contextualReturnType) {
-                    return node.asteriskToken
-                        ? contextualReturnType
-                        : getIterationTypeOfGeneratorFunctionReturnType(0 /* Yield */, contextualReturnType, (functionFlags & 2 /* Async */) !== 0);
-                }
+        // Return the leftmost type for which no type to the right is a subtype.
+        function getCommonSubtype(types) {
+            return ts.reduceLeft(types, function (s, t) { return isTypeSubtypeOf(t, s) ? t : s; });
+        }
+        function isArrayType(type) {
+            return !!(ts.getObjectFlags(type) & 4 /* Reference */) && (type.target === globalArrayType || type.target === globalReadonlyArrayType);
+        }
+        function isReadonlyArrayType(type) {
+            return !!(ts.getObjectFlags(type) & 4 /* Reference */) && type.target === globalReadonlyArrayType;
+        }
+        function isMutableArrayOrTuple(type) {
+            return isArrayType(type) && !isReadonlyArrayType(type) || isTupleType(type) && !type.target.readonly;
+        }
+        function getElementTypeOfArrayType(type) {
+            return isArrayType(type) ? getTypeArguments(type)[0] : undefined;
+        }
+        function isArrayLikeType(type) {
+            // A type is array-like if it is a reference to the global Array or global ReadonlyArray type,
+            // or if it is not the undefined or null type and if it is assignable to ReadonlyArray<any>
+            return isArrayType(type) || !(type.flags & 98304 /* Nullable */) && isTypeAssignableTo(type, anyReadonlyArrayType);
+        }
+        function isEmptyArrayLiteralType(type) {
+            var elementType = isArrayType(type) ? getTypeArguments(type)[0] : undefined;
+            return elementType === undefinedWideningType || elementType === implicitNeverType;
+        }
+        function isTupleLikeType(type) {
+            return isTupleType(type) || !!getPropertyOfType(type, "0");
+        }
+        function isArrayOrTupleLikeType(type) {
+            return isArrayLikeType(type) || isTupleLikeType(type);
+        }
+        function getTupleElementType(type, index) {
+            var propType = getTypeOfPropertyOfType(type, "" + index);
+            if (propType) {
+                return propType;
+            }
+            if (everyType(type, isTupleType)) {
+                return mapType(type, function (t) { return getRestTypeOfTupleType(t) || undefinedType; });
             }
             return undefined;
         }
-        function isInParameterInitializerBeforeContainingFunction(node) {
-            var inBindingInitializer = false;
-            while (node.parent && !ts.isFunctionLike(node.parent)) {
-                if (ts.isParameter(node.parent) && (inBindingInitializer || node.parent.initializer === node)) {
-                    return true;
-                }
-                if (ts.isBindingElement(node.parent) && node.parent.initializer === node) {
-                    inBindingInitializer = true;
-                }
-                node = node.parent;
+        function isNeitherUnitTypeNorNever(type) {
+            return !(type.flags & (109440 /* Unit */ | 131072 /* Never */));
+        }
+        function isUnitType(type) {
+            return !!(type.flags & 109440 /* Unit */);
+        }
+        function isLiteralType(type) {
+            return type.flags & 16 /* Boolean */ ? true :
+                type.flags & 1048576 /* Union */ ? type.flags & 1024 /* EnumLiteral */ ? true : ts.every(type.types, isUnitType) :
+                    isUnitType(type);
+        }
+        function getBaseTypeOfLiteralType(type) {
+            return type.flags & 1024 /* EnumLiteral */ ? getBaseTypeOfEnumLiteralType(type) :
+                type.flags & 128 /* StringLiteral */ ? stringType :
+                    type.flags & 256 /* NumberLiteral */ ? numberType :
+                        type.flags & 2048 /* BigIntLiteral */ ? bigintType :
+                            type.flags & 512 /* BooleanLiteral */ ? booleanType :
+                                type.flags & 1048576 /* Union */ ? getUnionType(ts.sameMap(type.types, getBaseTypeOfLiteralType)) :
+                                    type;
+        }
+        function getWidenedLiteralType(type) {
+            return type.flags & 1024 /* EnumLiteral */ && isFreshLiteralType(type) ? getBaseTypeOfEnumLiteralType(type) :
+                type.flags & 128 /* StringLiteral */ && isFreshLiteralType(type) ? stringType :
+                    type.flags & 256 /* NumberLiteral */ && isFreshLiteralType(type) ? numberType :
+                        type.flags & 2048 /* BigIntLiteral */ && isFreshLiteralType(type) ? bigintType :
+                            type.flags & 512 /* BooleanLiteral */ && isFreshLiteralType(type) ? booleanType :
+                                type.flags & 1048576 /* Union */ ? getUnionType(ts.sameMap(type.types, getWidenedLiteralType)) :
+                                    type;
+        }
+        function getWidenedUniqueESSymbolType(type) {
+            return type.flags & 8192 /* UniqueESSymbol */ ? esSymbolType :
+                type.flags & 1048576 /* Union */ ? getUnionType(ts.sameMap(type.types, getWidenedUniqueESSymbolType)) :
+                    type;
+        }
+        function getWidenedLiteralLikeTypeForContextualType(type, contextualType) {
+            if (!isLiteralOfContextualType(type, contextualType)) {
+                type = getWidenedUniqueESSymbolType(getWidenedLiteralType(type));
             }
-            return false;
+            return type;
         }
-        function getContextualIterationType(kind, functionDecl) {
-            var isAsync = !!(ts.getFunctionFlags(functionDecl) & 2 /* Async */);
-            var contextualReturnType = getContextualReturnType(functionDecl);
-            if (contextualReturnType) {
-                return getIterationTypeOfGeneratorFunctionReturnType(kind, contextualReturnType, isAsync)
-                    || undefined;
+        function getWidenedLiteralLikeTypeForContextualReturnTypeIfNeeded(type, contextualSignatureReturnType, isAsync) {
+            if (type && isUnitType(type)) {
+                var contextualType = !contextualSignatureReturnType ? undefined :
+                    isAsync ? getPromisedTypeOfPromise(contextualSignatureReturnType) :
+                        contextualSignatureReturnType;
+                type = getWidenedLiteralLikeTypeForContextualType(type, contextualType);
             }
-            return undefined;
+            return type;
         }
-        function getContextualReturnType(functionDecl) {
-            // If the containing function has a return type annotation, is a constructor, or is a get accessor whose
-            // corresponding set accessor has a type annotation, return statements in the function are contextually typed
-            var returnType = getReturnTypeFromAnnotation(functionDecl);
-            if (returnType) {
-                return returnType;
+        function getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(type, contextualSignatureReturnType, kind, isAsyncGenerator) {
+            if (type && isUnitType(type)) {
+                var contextualType = !contextualSignatureReturnType ? undefined :
+                    getIterationTypeOfGeneratorFunctionReturnType(kind, contextualSignatureReturnType, isAsyncGenerator);
+                type = getWidenedLiteralLikeTypeForContextualType(type, contextualType);
             }
-            // Otherwise, if the containing function is contextually typed by a function type with exactly one call signature
-            // and that call signature is non-generic, return statements are contextually typed by the return type of the signature
-            var signature = getContextualSignatureForFunctionLikeDeclaration(functionDecl);
-            if (signature && !isResolvingReturnTypeOfSignature(signature)) {
-                return getReturnTypeOfSignature(signature);
+            return type;
+        }
+        /**
+         * Check if a Type was written as a tuple type literal.
+         * Prefer using isTupleLikeType() unless the use of `elementTypes`/`getTypeArguments` is required.
+         */
+        function isTupleType(type) {
+            return !!(ts.getObjectFlags(type) & 4 /* Reference */ && type.target.objectFlags & 8 /* Tuple */);
+        }
+        function isGenericTupleType(type) {
+            return isTupleType(type) && !!(type.target.combinedFlags & 8 /* Variadic */);
+        }
+        function isSingleElementGenericTupleType(type) {
+            return isGenericTupleType(type) && type.target.elementFlags.length === 1;
+        }
+        function getRestTypeOfTupleType(type) {
+            return getElementTypeOfSliceOfTupleType(type, type.target.fixedLength);
+        }
+        function getRestArrayTypeOfTupleType(type) {
+            var restType = getRestTypeOfTupleType(type);
+            return restType && createArrayType(restType);
+        }
+        function getEndLengthOfType(type) {
+            return isTupleType(type) ? getTypeReferenceArity(type) - ts.findLastIndex(type.target.elementFlags, function (f) { return !(f & (1 /* Required */ | 2 /* Optional */)); }) - 1 : 0;
+        }
+        function getElementTypeOfSliceOfTupleType(type, index, endSkipCount, writing) {
+            if (endSkipCount === void 0) { endSkipCount = 0; }
+            if (writing === void 0) { writing = false; }
+            var length = getTypeReferenceArity(type) - endSkipCount;
+            if (index < length) {
+                var typeArguments = getTypeArguments(type);
+                var elementTypes = [];
+                for (var i = index; i < length; i++) {
+                    var t = typeArguments[i];
+                    elementTypes.push(type.target.elementFlags[i] & 8 /* Variadic */ ? getIndexedAccessType(t, numberType) : t);
+                }
+                return writing ? getIntersectionType(elementTypes) : getUnionType(elementTypes);
             }
             return undefined;
         }
-        // In a typed function call, an argument or substitution expression is contextually typed by the type of the corresponding parameter.
-        function getContextualTypeForArgument(callTarget, arg) {
-            var args = getEffectiveCallArguments(callTarget);
-            var argIndex = args.indexOf(arg); // -1 for e.g. the expression of a CallExpression, or the tag of a TaggedTemplateExpression
-            return argIndex === -1 ? undefined : getContextualTypeForArgumentAtIndex(callTarget, argIndex);
+        function isTupleTypeStructureMatching(t1, t2) {
+            return getTypeReferenceArity(t1) === getTypeReferenceArity(t2) &&
+                ts.every(t1.target.elementFlags, function (f, i) { return (f & 12 /* Variable */) === (t2.target.elementFlags[i] & 12 /* Variable */); });
         }
-        function getContextualTypeForArgumentAtIndex(callTarget, argIndex) {
-            // If we're already in the process of resolving the given signature, don't resolve again as
-            // that could cause infinite recursion. Instead, return anySignature.
-            var signature = getNodeLinks(callTarget).resolvedSignature === resolvingSignature ? resolvingSignature : getResolvedSignature(callTarget);
-            if (ts.isJsxOpeningLikeElement(callTarget) && argIndex === 0) {
-                return getEffectiveFirstArgumentForJsxSignature(signature, callTarget);
+        function isZeroBigInt(_a) {
+            var value = _a.value;
+            return value.base10Value === "0";
+        }
+        function getFalsyFlagsOfTypes(types) {
+            var result = 0;
+            for (var _i = 0, types_14 = types; _i < types_14.length; _i++) {
+                var t = types_14[_i];
+                result |= getFalsyFlags(t);
             }
-            return getTypeAtPosition(signature, argIndex);
+            return result;
         }
-        function getContextualTypeForSubstitutionExpression(template, substitutionExpression) {
-            if (template.parent.kind === 198 /* TaggedTemplateExpression */) {
-                return getContextualTypeForArgument(template.parent, substitutionExpression);
+        // Returns the String, Number, Boolean, StringLiteral, NumberLiteral, BooleanLiteral, Void, Undefined, or Null
+        // flags for the string, number, boolean, "", 0, false, void, undefined, or null types respectively. Returns
+        // no flags for all other types (including non-falsy literal types).
+        function getFalsyFlags(type) {
+            return type.flags & 1048576 /* Union */ ? getFalsyFlagsOfTypes(type.types) :
+                type.flags & 128 /* StringLiteral */ ? type.value === "" ? 128 /* StringLiteral */ : 0 :
+                    type.flags & 256 /* NumberLiteral */ ? type.value === 0 ? 256 /* NumberLiteral */ : 0 :
+                        type.flags & 2048 /* BigIntLiteral */ ? isZeroBigInt(type) ? 2048 /* BigIntLiteral */ : 0 :
+                            type.flags & 512 /* BooleanLiteral */ ? (type === falseType || type === regularFalseType) ? 512 /* BooleanLiteral */ : 0 :
+                                type.flags & 117724 /* PossiblyFalsy */;
+        }
+        function removeDefinitelyFalsyTypes(type) {
+            return getFalsyFlags(type) & 117632 /* DefinitelyFalsy */ ?
+                filterType(type, function (t) { return !(getFalsyFlags(t) & 117632 /* DefinitelyFalsy */); }) :
+                type;
+        }
+        function extractDefinitelyFalsyTypes(type) {
+            return mapType(type, getDefinitelyFalsyPartOfType);
+        }
+        function getDefinitelyFalsyPartOfType(type) {
+            return type.flags & 4 /* String */ ? emptyStringType :
+                type.flags & 8 /* Number */ ? zeroType :
+                    type.flags & 64 /* BigInt */ ? zeroBigIntType :
+                        type === regularFalseType ||
+                            type === falseType ||
+                            type.flags & (16384 /* Void */ | 32768 /* Undefined */ | 65536 /* Null */ | 3 /* AnyOrUnknown */) ||
+                            type.flags & 128 /* StringLiteral */ && type.value === "" ||
+                            type.flags & 256 /* NumberLiteral */ && type.value === 0 ||
+                            type.flags & 2048 /* BigIntLiteral */ && isZeroBigInt(type) ? type :
+                            neverType;
+        }
+        /**
+         * Add undefined or null or both to a type if they are missing.
+         * @param type - type to add undefined and/or null to if not present
+         * @param flags - Either TypeFlags.Undefined or TypeFlags.Null, or both
+         */
+        function getNullableType(type, flags) {
+            var missing = (flags & ~type.flags) & (32768 /* Undefined */ | 65536 /* Null */);
+            return missing === 0 ? type :
+                missing === 32768 /* Undefined */ ? getUnionType([type, undefinedType]) :
+                    missing === 65536 /* Null */ ? getUnionType([type, nullType]) :
+                        getUnionType([type, undefinedType, nullType]);
+        }
+        function getOptionalType(type) {
+            ts.Debug.assert(strictNullChecks);
+            return type.flags & 32768 /* Undefined */ ? type : getUnionType([type, undefinedType]);
+        }
+        function getGlobalNonNullableTypeInstantiation(type) {
+            if (!deferredGlobalNonNullableTypeAlias) {
+                deferredGlobalNonNullableTypeAlias = getGlobalSymbol("NonNullable", 524288 /* TypeAlias */, /*diagnostic*/ undefined) || unknownSymbol;
             }
-            return undefined;
+            // Use NonNullable global type alias if available to improve quick info/declaration emit
+            if (deferredGlobalNonNullableTypeAlias !== unknownSymbol) {
+                return getTypeAliasInstantiation(deferredGlobalNonNullableTypeAlias, [type]);
+            }
+            return getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */); // Type alias unavailable, fall back to non-higher-order behavior
         }
-        function getContextualTypeForBinaryOperand(node, contextFlags) {
-            var binaryExpression = node.parent;
-            var left = binaryExpression.left, operatorToken = binaryExpression.operatorToken, right = binaryExpression.right;
-            switch (operatorToken.kind) {
-                case 62 /* EqualsToken */:
-                    if (node !== right) {
-                        return undefined;
-                    }
-                    var contextSensitive = getIsContextSensitiveAssignmentOrContextType(binaryExpression);
-                    if (!contextSensitive) {
-                        return undefined;
-                    }
-                    return contextSensitive === true ? getTypeOfExpression(left) : contextSensitive;
-                case 56 /* BarBarToken */:
-                case 60 /* QuestionQuestionToken */:
-                    // When an || expression has a contextual type, the operands are contextually typed by that type, except
-                    // when that type originates in a binding pattern, the right operand is contextually typed by the type of
-                    // the left operand. When an || expression has no contextual type, the right operand is contextually typed
-                    // by the type of the left operand, except for the special case of Javascript declarations of the form
-                    // `namespace.prop = namespace.prop || {}`.
-                    var type = getContextualType(binaryExpression, contextFlags);
-                    return node === right && (type && type.pattern || !type && !ts.isDefaultedExpandoInitializer(binaryExpression)) ?
-                        getTypeOfExpression(left) : type;
-                case 55 /* AmpersandAmpersandToken */:
-                case 27 /* CommaToken */:
-                    return node === right ? getContextualType(binaryExpression, contextFlags) : undefined;
-                default:
-                    return undefined;
+        function getNonNullableType(type) {
+            return strictNullChecks ? getGlobalNonNullableTypeInstantiation(type) : type;
+        }
+        function addOptionalTypeMarker(type) {
+            return strictNullChecks ? getUnionType([type, optionalType]) : type;
+        }
+        function isNotOptionalTypeMarker(type) {
+            return type !== optionalType;
+        }
+        function removeOptionalTypeMarker(type) {
+            return strictNullChecks ? filterType(type, isNotOptionalTypeMarker) : type;
+        }
+        function propagateOptionalTypeMarker(type, node, wasOptional) {
+            return wasOptional ? ts.isOutermostOptionalChain(node) ? getOptionalType(type) : addOptionalTypeMarker(type) : type;
+        }
+        function getOptionalExpressionType(exprType, expression) {
+            return ts.isExpressionOfOptionalChainRoot(expression) ? getNonNullableType(exprType) :
+                ts.isOptionalChain(expression) ? removeOptionalTypeMarker(exprType) :
+                    exprType;
+        }
+        /**
+         * Is source potentially coercible to target type under `==`.
+         * Assumes that `source` is a constituent of a union, hence
+         * the boolean literal flag on the LHS, but not on the RHS.
+         *
+         * This does not fully replicate the semantics of `==`. The
+         * intention is to catch cases that are clearly not right.
+         *
+         * Comparing (string | number) to number should not remove the
+         * string element.
+         *
+         * Comparing (string | number) to 1 will remove the string
+         * element, though this is not sound. This is a pragmatic
+         * choice.
+         *
+         * @see narrowTypeByEquality
+         *
+         * @param source
+         * @param target
+         */
+        function isCoercibleUnderDoubleEquals(source, target) {
+            return ((source.flags & (8 /* Number */ | 4 /* String */ | 512 /* BooleanLiteral */)) !== 0)
+                && ((target.flags & (8 /* Number */ | 4 /* String */ | 16 /* Boolean */)) !== 0);
+        }
+        /**
+         * Return true if type was inferred from an object literal, written as an object type literal, or is the shape of a module
+         * with no call or construct signatures.
+         */
+        function isObjectTypeWithInferableIndex(type) {
+            return type.flags & 2097152 /* Intersection */ ? ts.every(type.types, isObjectTypeWithInferableIndex) :
+                !!(type.symbol && (type.symbol.flags & (4096 /* ObjectLiteral */ | 2048 /* TypeLiteral */ | 384 /* Enum */ | 512 /* ValueModule */)) !== 0 &&
+                    !typeHasCallOrConstructSignatures(type)) || !!(ts.getObjectFlags(type) & 2048 /* ReverseMapped */ && isObjectTypeWithInferableIndex(type.source));
+        }
+        function createSymbolWithType(source, type) {
+            var symbol = createSymbol(source.flags, source.escapedName, ts.getCheckFlags(source) & 8 /* Readonly */);
+            symbol.declarations = source.declarations;
+            symbol.parent = source.parent;
+            symbol.type = type;
+            symbol.target = source;
+            if (source.valueDeclaration) {
+                symbol.valueDeclaration = source.valueDeclaration;
+            }
+            var nameType = getSymbolLinks(source).nameType;
+            if (nameType) {
+                symbol.nameType = nameType;
             }
+            return symbol;
         }
-        // In an assignment expression, the right operand is contextually typed by the type of the left operand.
-        // Don't do this for assignment declarations unless there is a type tag on the assignment, to avoid circularity from checking the right operand.
-        function getIsContextSensitiveAssignmentOrContextType(binaryExpression) {
-            var kind = ts.getAssignmentDeclarationKind(binaryExpression);
-            switch (kind) {
-                case 0 /* None */:
-                    return true;
-                case 5 /* Property */:
-                case 1 /* ExportsProperty */:
-                case 6 /* Prototype */:
-                case 3 /* PrototypeProperty */:
-                    // If `binaryExpression.left` was assigned a symbol, then this is a new declaration; otherwise it is an assignment to an existing declaration.
-                    // See `bindStaticPropertyAssignment` in `binder.ts`.
-                    if (!binaryExpression.left.symbol) {
-                        return true;
-                    }
-                    else {
-                        var decl = binaryExpression.left.symbol.valueDeclaration;
-                        if (!decl) {
-                            return false;
-                        }
-                        var lhs = ts.cast(binaryExpression.left, ts.isAccessExpression);
-                        var overallAnnotation = ts.getEffectiveTypeAnnotationNode(decl);
-                        if (overallAnnotation) {
-                            return getTypeFromTypeNode(overallAnnotation);
-                        }
-                        else if (ts.isIdentifier(lhs.expression)) {
-                            var id = lhs.expression;
-                            var parentSymbol = resolveName(id, id.escapedText, 111551 /* Value */, undefined, id.escapedText, /*isUse*/ true);
-                            if (parentSymbol) {
-                                var annotated = ts.getEffectiveTypeAnnotationNode(parentSymbol.valueDeclaration);
-                                if (annotated) {
-                                    var nameStr_1 = ts.getElementOrPropertyAccessName(lhs);
-                                    if (nameStr_1 !== undefined) {
-                                        var type = getTypeOfPropertyOfContextualType(getTypeFromTypeNode(annotated), nameStr_1);
-                                        return type || false;
-                                    }
-                                }
-                                return false;
-                            }
-                        }
-                        return !ts.isInJSFile(decl);
-                    }
-                case 2 /* ModuleExports */:
-                case 4 /* ThisProperty */:
-                    if (!binaryExpression.symbol)
-                        return true;
-                    if (binaryExpression.symbol.valueDeclaration) {
-                        var annotated = ts.getEffectiveTypeAnnotationNode(binaryExpression.symbol.valueDeclaration);
-                        if (annotated) {
-                            var type = getTypeFromTypeNode(annotated);
-                            if (type) {
-                                return type;
-                            }
-                        }
-                    }
-                    if (kind === 2 /* ModuleExports */)
-                        return false;
-                    var thisAccess = ts.cast(binaryExpression.left, ts.isAccessExpression);
-                    if (!ts.isObjectLiteralMethod(ts.getThisContainer(thisAccess.expression, /*includeArrowFunctions*/ false))) {
-                        return false;
-                    }
-                    var thisType = checkThisExpression(thisAccess.expression);
-                    var nameStr = ts.getElementOrPropertyAccessName(thisAccess);
-                    return nameStr !== undefined && thisType && getTypeOfPropertyOfContextualType(thisType, nameStr) || false;
-                case 7 /* ObjectDefinePropertyValue */:
-                case 8 /* ObjectDefinePropertyExports */:
-                case 9 /* ObjectDefinePrototypeProperty */:
-                    return ts.Debug.fail("Does not apply");
-                default:
-                    return ts.Debug.assertNever(kind);
+        function transformTypeOfMembers(type, f) {
+            var members = ts.createSymbolTable();
+            for (var _i = 0, _a = getPropertiesOfObjectType(type); _i < _a.length; _i++) {
+                var property = _a[_i];
+                var original = getTypeOfSymbol(property);
+                var updated = f(original);
+                members.set(property.escapedName, updated === original ? property : createSymbolWithType(property, updated));
             }
+            return members;
         }
-        function getTypeOfPropertyOfContextualType(type, name) {
-            return mapType(type, function (t) {
-                if (isGenericMappedType(t)) {
-                    var constraint = getConstraintTypeFromMappedType(t);
-                    var constraintOfConstraint = getBaseConstraintOfType(constraint) || constraint;
-                    var propertyNameType = getLiteralType(ts.unescapeLeadingUnderscores(name));
-                    if (isTypeAssignableTo(propertyNameType, constraintOfConstraint)) {
-                        return substituteIndexedMappedType(t, propertyNameType);
+        /**
+         * If the the provided object literal is subject to the excess properties check,
+         * create a new that is exempt. Recursively mark object literal members as exempt.
+         * Leave signatures alone since they are not subject to the check.
+         */
+        function getRegularTypeOfObjectLiteral(type) {
+            if (!(isObjectLiteralType(type) && ts.getObjectFlags(type) & 32768 /* FreshLiteral */)) {
+                return type;
+            }
+            var regularType = type.regularType;
+            if (regularType) {
+                return regularType;
+            }
+            var resolved = type;
+            var members = transformTypeOfMembers(type, getRegularTypeOfObjectLiteral);
+            var regularNew = createAnonymousType(resolved.symbol, members, resolved.callSignatures, resolved.constructSignatures, resolved.stringIndexInfo, resolved.numberIndexInfo);
+            regularNew.flags = resolved.flags;
+            regularNew.objectFlags |= resolved.objectFlags & ~32768 /* FreshLiteral */;
+            type.regularType = regularNew;
+            return regularNew;
+        }
+        function createWideningContext(parent, propertyName, siblings) {
+            return { parent: parent, propertyName: propertyName, siblings: siblings, resolvedProperties: undefined };
+        }
+        function getSiblingsOfContext(context) {
+            if (!context.siblings) {
+                var siblings_1 = [];
+                for (var _i = 0, _a = getSiblingsOfContext(context.parent); _i < _a.length; _i++) {
+                    var type = _a[_i];
+                    if (isObjectLiteralType(type)) {
+                        var prop = getPropertyOfObjectType(type, context.propertyName);
+                        if (prop) {
+                            forEachType(getTypeOfSymbol(prop), function (t) {
+                                siblings_1.push(t);
+                            });
+                        }
                     }
                 }
-                else if (t.flags & 3670016 /* StructuredType */) {
-                    var prop = getPropertyOfType(t, name);
-                    if (prop) {
-                        return getTypeOfSymbol(prop);
-                    }
-                    if (isTupleType(t)) {
-                        var restType = getRestTypeOfTupleType(t);
-                        if (restType && isNumericLiteralName(name) && +name >= 0) {
-                            return restType;
+                context.siblings = siblings_1;
+            }
+            return context.siblings;
+        }
+        function getPropertiesOfContext(context) {
+            if (!context.resolvedProperties) {
+                var names = new ts.Map();
+                for (var _i = 0, _a = getSiblingsOfContext(context); _i < _a.length; _i++) {
+                    var t = _a[_i];
+                    if (isObjectLiteralType(t) && !(ts.getObjectFlags(t) & 1024 /* ContainsSpread */)) {
+                        for (var _b = 0, _c = getPropertiesOfType(t); _b < _c.length; _b++) {
+                            var prop = _c[_b];
+                            names.set(prop.escapedName, prop);
                         }
                     }
-                    return isNumericLiteralName(name) && getIndexTypeOfContextualType(t, 1 /* Number */) ||
-                        getIndexTypeOfContextualType(t, 0 /* String */);
                 }
-                return undefined;
-            }, /*noReductions*/ true);
+                context.resolvedProperties = ts.arrayFrom(names.values());
+            }
+            return context.resolvedProperties;
         }
-        function getIndexTypeOfContextualType(type, kind) {
-            return mapType(type, function (t) { return getIndexTypeOfStructuredType(t, kind); }, /*noReductions*/ true);
+        function getWidenedProperty(prop, context) {
+            if (!(prop.flags & 4 /* Property */)) {
+                // Since get accessors already widen their return value there is no need to
+                // widen accessor based properties here.
+                return prop;
+            }
+            var original = getTypeOfSymbol(prop);
+            var propContext = context && createWideningContext(context, prop.escapedName, /*siblings*/ undefined);
+            var widened = getWidenedTypeWithContext(original, propContext);
+            return widened === original ? prop : createSymbolWithType(prop, widened);
         }
-        // In an object literal contextually typed by a type T, the contextual type of a property assignment is the type of
-        // the matching property in T, if one exists. Otherwise, it is the type of the numeric index signature in T, if one
-        // exists. Otherwise, it is the type of the string index signature in T, if one exists.
-        function getContextualTypeForObjectLiteralMethod(node, contextFlags) {
-            ts.Debug.assert(ts.isObjectLiteralMethod(node));
-            if (node.flags & 16777216 /* InWithStatement */) {
-                // We cannot answer semantic questions within a with block, do not proceed any further
-                return undefined;
+        function getUndefinedProperty(prop) {
+            var cached = undefinedProperties.get(prop.escapedName);
+            if (cached) {
+                return cached;
             }
-            return getContextualTypeForObjectLiteralElement(node, contextFlags);
+            var result = createSymbolWithType(prop, undefinedType);
+            result.flags |= 16777216 /* Optional */;
+            undefinedProperties.set(prop.escapedName, result);
+            return result;
         }
-        function getContextualTypeForObjectLiteralElement(element, contextFlags) {
-            var objectLiteral = element.parent;
-            var type = getApparentTypeOfContextualType(objectLiteral, contextFlags);
-            if (type) {
-                if (!hasNonBindableDynamicName(element)) {
-                    // For a (non-symbol) computed property, there is no reason to look up the name
-                    // in the type. It will just be "__computed", which does not appear in any
-                    // SymbolTable.
-                    var symbolName_3 = getSymbolOfNode(element).escapedName;
-                    var propertyType = getTypeOfPropertyOfContextualType(type, symbolName_3);
-                    if (propertyType) {
-                        return propertyType;
+        function getWidenedTypeOfObjectLiteral(type, context) {
+            var members = ts.createSymbolTable();
+            for (var _i = 0, _a = getPropertiesOfObjectType(type); _i < _a.length; _i++) {
+                var prop = _a[_i];
+                members.set(prop.escapedName, getWidenedProperty(prop, context));
+            }
+            if (context) {
+                for (var _b = 0, _c = getPropertiesOfContext(context); _b < _c.length; _b++) {
+                    var prop = _c[_b];
+                    if (!members.has(prop.escapedName)) {
+                        members.set(prop.escapedName, getUndefinedProperty(prop));
                     }
                 }
-                return isNumericName(element.name) && getIndexTypeOfContextualType(type, 1 /* Number */) ||
-                    getIndexTypeOfContextualType(type, 0 /* String */);
             }
-            return undefined;
-        }
-        // In an array literal contextually typed by a type T, the contextual type of an element expression at index N is
-        // the type of the property with the numeric name N in T, if one exists. Otherwise, if T has a numeric index signature,
-        // it is the type of the numeric index signature in T. Otherwise, in ES6 and higher, the contextual type is the iterated
-        // type of T.
-        function getContextualTypeForElementExpression(arrayContextualType, index) {
-            return arrayContextualType && (getTypeOfPropertyOfContextualType(arrayContextualType, "" + index)
-                || getIteratedTypeOrElementType(1 /* Element */, arrayContextualType, undefinedType, /*errorNode*/ undefined, /*checkAssignability*/ false));
+            var stringIndexInfo = getIndexInfoOfType(type, 0 /* String */);
+            var numberIndexInfo = getIndexInfoOfType(type, 1 /* Number */);
+            var result = createAnonymousType(type.symbol, members, ts.emptyArray, ts.emptyArray, stringIndexInfo && createIndexInfo(getWidenedType(stringIndexInfo.type), stringIndexInfo.isReadonly), numberIndexInfo && createIndexInfo(getWidenedType(numberIndexInfo.type), numberIndexInfo.isReadonly));
+            result.objectFlags |= (ts.getObjectFlags(type) & (16384 /* JSLiteral */ | 2097152 /* NonInferrableType */)); // Retain js literal flag through widening
+            return result;
         }
-        // In a contextually typed conditional expression, the true/false expressions are contextually typed by the same type.
-        function getContextualTypeForConditionalOperand(node, contextFlags) {
-            var conditional = node.parent;
-            return node === conditional.whenTrue || node === conditional.whenFalse ? getContextualType(conditional, contextFlags) : undefined;
+        function getWidenedType(type) {
+            return getWidenedTypeWithContext(type, /*context*/ undefined);
         }
-        function getContextualTypeForChildJsxExpression(node, child) {
-            var attributesType = getApparentTypeOfContextualType(node.openingElement.tagName);
-            // JSX expression is in children of JSX Element, we will look for an "children" atttribute (we get the name from JSX.ElementAttributesProperty)
-            var jsxChildrenPropertyName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(node));
-            if (!(attributesType && !isTypeAny(attributesType) && jsxChildrenPropertyName && jsxChildrenPropertyName !== "")) {
-                return undefined;
-            }
-            var realChildren = getSemanticJsxChildren(node.children);
-            var childIndex = realChildren.indexOf(child);
-            var childFieldType = getTypeOfPropertyOfContextualType(attributesType, jsxChildrenPropertyName);
-            return childFieldType && (realChildren.length === 1 ? childFieldType : mapType(childFieldType, function (t) {
-                if (isArrayLikeType(t)) {
-                    return getIndexedAccessType(t, getLiteralType(childIndex));
+        function getWidenedTypeWithContext(type, context) {
+            if (ts.getObjectFlags(type) & 1572864 /* RequiresWidening */) {
+                if (context === undefined && type.widened) {
+                    return type.widened;
                 }
-                else {
-                    return t;
+                var result = void 0;
+                if (type.flags & (1 /* Any */ | 98304 /* Nullable */)) {
+                    result = anyType;
                 }
-            }, /*noReductions*/ true));
-        }
-        function getContextualTypeForJsxExpression(node) {
-            var exprParent = node.parent;
-            return ts.isJsxAttributeLike(exprParent)
-                ? getContextualType(node)
-                : ts.isJsxElement(exprParent)
-                    ? getContextualTypeForChildJsxExpression(exprParent, node)
-                    : undefined;
-        }
-        function getContextualTypeForJsxAttribute(attribute) {
-            // When we trying to resolve JsxOpeningLikeElement as a stateless function element, we will already give its attributes a contextual type
-            // which is a type of the parameter of the signature we are trying out.
-            // If there is no contextual type (e.g. we are trying to resolve stateful component), get attributes type from resolving element's tagName
-            if (ts.isJsxAttribute(attribute)) {
-                var attributesType = getApparentTypeOfContextualType(attribute.parent);
-                if (!attributesType || isTypeAny(attributesType)) {
-                    return undefined;
+                else if (isObjectLiteralType(type)) {
+                    result = getWidenedTypeOfObjectLiteral(type, context);
                 }
-                return getTypeOfPropertyOfContextualType(attributesType, attribute.name.escapedText);
-            }
-            else {
-                return getContextualType(attribute.parent);
-            }
-        }
-        // Return true if the given expression is possibly a discriminant value. We limit the kinds of
-        // expressions we check to those that don't depend on their contextual type in order not to cause
-        // recursive (and possibly infinite) invocations of getContextualType.
-        function isPossiblyDiscriminantValue(node) {
-            switch (node.kind) {
-                case 10 /* StringLiteral */:
-                case 8 /* NumericLiteral */:
-                case 9 /* BigIntLiteral */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
-                case 100 /* NullKeyword */:
-                case 75 /* Identifier */:
-                case 146 /* UndefinedKeyword */:
-                    return true;
-                case 194 /* PropertyAccessExpression */:
-                case 200 /* ParenthesizedExpression */:
-                    return isPossiblyDiscriminantValue(node.expression);
-                case 276 /* JsxExpression */:
-                    return !node.expression || isPossiblyDiscriminantValue(node.expression);
+                else if (type.flags & 1048576 /* Union */) {
+                    var unionContext_1 = context || createWideningContext(/*parent*/ undefined, /*propertyName*/ undefined, type.types);
+                    var widenedTypes = ts.sameMap(type.types, function (t) { return t.flags & 98304 /* Nullable */ ? t : getWidenedTypeWithContext(t, unionContext_1); });
+                    // Widening an empty object literal transitions from a highly restrictive type to
+                    // a highly inclusive one. For that reason we perform subtype reduction here if the
+                    // union includes empty object types (e.g. reducing {} | string to just {}).
+                    result = getUnionType(widenedTypes, ts.some(widenedTypes, isEmptyObjectType) ? 2 /* Subtype */ : 1 /* Literal */);
+                }
+                else if (type.flags & 2097152 /* Intersection */) {
+                    result = getIntersectionType(ts.sameMap(type.types, getWidenedType));
+                }
+                else if (isArrayType(type) || isTupleType(type)) {
+                    result = createTypeReference(type.target, ts.sameMap(getTypeArguments(type), getWidenedType));
+                }
+                if (result && context === undefined) {
+                    type.widened = result;
+                }
+                return result || type;
             }
-            return false;
-        }
-        function discriminateContextualTypeByObjectMembers(node, contextualType) {
-            return discriminateTypeByDiscriminableItems(contextualType, ts.map(ts.filter(node.properties, function (p) { return !!p.symbol && p.kind === 281 /* PropertyAssignment */ && isPossiblyDiscriminantValue(p.initializer) && isDiscriminantProperty(contextualType, p.symbol.escapedName); }), function (prop) { return [function () { return checkExpression(prop.initializer); }, prop.symbol.escapedName]; }), isTypeAssignableTo, contextualType);
-        }
-        function discriminateContextualTypeByJSXAttributes(node, contextualType) {
-            return discriminateTypeByDiscriminableItems(contextualType, ts.map(ts.filter(node.properties, function (p) { return !!p.symbol && p.kind === 273 /* JsxAttribute */ && isDiscriminantProperty(contextualType, p.symbol.escapedName) && (!p.initializer || isPossiblyDiscriminantValue(p.initializer)); }), function (prop) { return [!prop.initializer ? (function () { return trueType; }) : (function () { return checkExpression(prop.initializer); }), prop.symbol.escapedName]; }), isTypeAssignableTo, contextualType);
+            return type;
         }
-        // Return the contextual type for a given expression node. During overload resolution, a contextual type may temporarily
-        // be "pushed" onto a node using the contextualType property.
-        function getApparentTypeOfContextualType(node, contextFlags) {
-            var contextualType = ts.isObjectLiteralMethod(node) ?
-                getContextualTypeForObjectLiteralMethod(node, contextFlags) :
-                getContextualType(node, contextFlags);
-            var instantiatedType = instantiateContextualType(contextualType, node, contextFlags);
-            if (instantiatedType && !(contextFlags && contextFlags & 2 /* NoConstraints */ && instantiatedType.flags & 8650752 /* TypeVariable */)) {
-                var apparentType = mapType(instantiatedType, getApparentType, /*noReductions*/ true);
-                if (apparentType.flags & 1048576 /* Union */) {
-                    if (ts.isObjectLiteralExpression(node)) {
-                        return discriminateContextualTypeByObjectMembers(node, apparentType);
+        /**
+         * Reports implicit any errors that occur as a result of widening 'null' and 'undefined'
+         * to 'any'. A call to reportWideningErrorsInType is normally accompanied by a call to
+         * getWidenedType. But in some cases getWidenedType is called without reporting errors
+         * (type argument inference is an example).
+         *
+         * The return value indicates whether an error was in fact reported. The particular circumstances
+         * are on a best effort basis. Currently, if the null or undefined that causes widening is inside
+         * an object literal property (arbitrarily deeply), this function reports an error. If no error is
+         * reported, reportImplicitAnyError is a suitable fallback to report a general error.
+         */
+        function reportWideningErrorsInType(type) {
+            var errorReported = false;
+            if (ts.getObjectFlags(type) & 524288 /* ContainsWideningType */) {
+                if (type.flags & 1048576 /* Union */) {
+                    if (ts.some(type.types, isEmptyObjectType)) {
+                        errorReported = true;
                     }
-                    else if (ts.isJsxAttributes(node)) {
-                        return discriminateContextualTypeByJSXAttributes(node, apparentType);
+                    else {
+                        for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
+                            var t = _a[_i];
+                            if (reportWideningErrorsInType(t)) {
+                                errorReported = true;
+                            }
+                        }
+                    }
+                }
+                if (isArrayType(type) || isTupleType(type)) {
+                    for (var _b = 0, _c = getTypeArguments(type); _b < _c.length; _b++) {
+                        var t = _c[_b];
+                        if (reportWideningErrorsInType(t)) {
+                            errorReported = true;
+                        }
+                    }
+                }
+                if (isObjectLiteralType(type)) {
+                    for (var _d = 0, _e = getPropertiesOfObjectType(type); _d < _e.length; _d++) {
+                        var p = _e[_d];
+                        var t = getTypeOfSymbol(p);
+                        if (ts.getObjectFlags(t) & 524288 /* ContainsWideningType */) {
+                            if (!reportWideningErrorsInType(t)) {
+                                error(p.valueDeclaration, ts.Diagnostics.Object_literal_s_property_0_implicitly_has_an_1_type, symbolToString(p), typeToString(getWidenedType(t)));
+                            }
+                            errorReported = true;
+                        }
                     }
                 }
-                return apparentType;
             }
+            return errorReported;
         }
-        // If the given contextual type contains instantiable types and if a mapper representing
-        // return type inferences is available, instantiate those types using that mapper.
-        function instantiateContextualType(contextualType, node, contextFlags) {
-            if (contextualType && maybeTypeOfKind(contextualType, 63176704 /* Instantiable */)) {
-                var inferenceContext = getInferenceContext(node);
-                // If no inferences have been made, nothing is gained from instantiating as type parameters
-                // would just be replaced with their defaults similar to the apparent type.
-                if (inferenceContext && ts.some(inferenceContext.inferences, hasInferenceCandidates)) {
-                    // For contextual signatures we incorporate all inferences made so far, e.g. from return
-                    // types as well as arguments to the left in a function call.
-                    if (contextFlags && contextFlags & 1 /* Signature */) {
-                        return instantiateInstantiableTypes(contextualType, inferenceContext.nonFixingMapper);
+        function reportImplicitAny(declaration, type, wideningKind) {
+            var typeAsString = typeToString(getWidenedType(type));
+            if (ts.isInJSFile(declaration) && !ts.isCheckJsEnabledForFile(ts.getSourceFileOfNode(declaration), compilerOptions)) {
+                // Only report implicit any errors/suggestions in TS and ts-check JS files
+                return;
+            }
+            var diagnostic;
+            switch (declaration.kind) {
+                case 216 /* BinaryExpression */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                    diagnostic = noImplicitAny ? ts.Diagnostics.Member_0_implicitly_has_an_1_type : ts.Diagnostics.Member_0_implicitly_has_an_1_type_but_a_better_type_may_be_inferred_from_usage;
+                    break;
+                case 160 /* Parameter */:
+                    var param = declaration;
+                    if (ts.isIdentifier(param.name) &&
+                        (ts.isCallSignatureDeclaration(param.parent) || ts.isMethodSignature(param.parent) || ts.isFunctionTypeNode(param.parent)) &&
+                        param.parent.parameters.indexOf(param) > -1 &&
+                        (resolveName(param, param.name.escapedText, 788968 /* Type */, undefined, param.name.escapedText, /*isUse*/ true) ||
+                            param.name.originalKeywordKind && ts.isTypeNodeKind(param.name.originalKeywordKind))) {
+                        var newName = "arg" + param.parent.parameters.indexOf(param);
+                        errorOrSuggestion(noImplicitAny, declaration, ts.Diagnostics.Parameter_has_a_name_but_no_type_Did_you_mean_0_Colon_1, newName, ts.declarationNameToString(param.name));
+                        return;
                     }
-                    // For other purposes (e.g. determining whether to produce literal types) we only
-                    // incorporate inferences made from the return type in a function call.
-                    if (inferenceContext.returnMapper) {
-                        return instantiateInstantiableTypes(contextualType, inferenceContext.returnMapper);
+                    diagnostic = declaration.dotDotDotToken ?
+                        noImplicitAny ? ts.Diagnostics.Rest_parameter_0_implicitly_has_an_any_type : ts.Diagnostics.Rest_parameter_0_implicitly_has_an_any_type_but_a_better_type_may_be_inferred_from_usage :
+                        noImplicitAny ? ts.Diagnostics.Parameter_0_implicitly_has_an_1_type : ts.Diagnostics.Parameter_0_implicitly_has_an_1_type_but_a_better_type_may_be_inferred_from_usage;
+                    break;
+                case 198 /* BindingElement */:
+                    diagnostic = ts.Diagnostics.Binding_element_0_implicitly_has_an_1_type;
+                    if (!noImplicitAny) {
+                        // Don't issue a suggestion for binding elements since the codefix doesn't yet support them.
+                        return;
+                    }
+                    break;
+                case 308 /* JSDocFunctionType */:
+                    error(declaration, ts.Diagnostics.Function_type_which_lacks_return_type_annotation_implicitly_has_an_0_return_type, typeAsString);
+                    return;
+                case 251 /* FunctionDeclaration */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                    if (noImplicitAny && !declaration.name) {
+                        if (wideningKind === 3 /* GeneratorYield */) {
+                            error(declaration, ts.Diagnostics.Generator_implicitly_has_yield_type_0_because_it_does_not_yield_any_values_Consider_supplying_a_return_type_annotation, typeAsString);
+                        }
+                        else {
+                            error(declaration, ts.Diagnostics.Function_expression_which_lacks_return_type_annotation_implicitly_has_an_0_return_type, typeAsString);
+                        }
+                        return;
+                    }
+                    diagnostic = !noImplicitAny ? ts.Diagnostics._0_implicitly_has_an_1_return_type_but_a_better_type_may_be_inferred_from_usage :
+                        wideningKind === 3 /* GeneratorYield */ ? ts.Diagnostics._0_which_lacks_return_type_annotation_implicitly_has_an_1_yield_type :
+                            ts.Diagnostics._0_which_lacks_return_type_annotation_implicitly_has_an_1_return_type;
+                    break;
+                case 190 /* MappedType */:
+                    if (noImplicitAny) {
+                        error(declaration, ts.Diagnostics.Mapped_object_type_implicitly_has_an_any_template_type);
                     }
+                    return;
+                default:
+                    diagnostic = noImplicitAny ? ts.Diagnostics.Variable_0_implicitly_has_an_1_type : ts.Diagnostics.Variable_0_implicitly_has_an_1_type_but_a_better_type_may_be_inferred_from_usage;
+            }
+            errorOrSuggestion(noImplicitAny, declaration, diagnostic, ts.declarationNameToString(ts.getNameOfDeclaration(declaration)), typeAsString);
+        }
+        function reportErrorsFromWidening(declaration, type, wideningKind) {
+            if (produceDiagnostics && noImplicitAny && ts.getObjectFlags(type) & 524288 /* ContainsWideningType */ && (!wideningKind || !getContextualSignatureForFunctionLikeDeclaration(declaration))) {
+                // Report implicit any error within type if possible, otherwise report error on declaration
+                if (!reportWideningErrorsInType(type)) {
+                    reportImplicitAny(declaration, type, wideningKind);
                 }
             }
-            return contextualType;
         }
-        // This function is similar to instantiateType, except that (a) it only instantiates types that
-        // are classified as instantiable (i.e. it doesn't instantiate object types), and (b) it performs
-        // no reductions on instantiated union types.
-        function instantiateInstantiableTypes(type, mapper) {
-            if (type.flags & 63176704 /* Instantiable */) {
-                return instantiateType(type, mapper);
+        function applyToParameterTypes(source, target, callback) {
+            var sourceCount = getParameterCount(source);
+            var targetCount = getParameterCount(target);
+            var sourceRestType = getEffectiveRestType(source);
+            var targetRestType = getEffectiveRestType(target);
+            var targetNonRestCount = targetRestType ? targetCount - 1 : targetCount;
+            var paramCount = sourceRestType ? targetNonRestCount : Math.min(sourceCount, targetNonRestCount);
+            var sourceThisType = getThisTypeOfSignature(source);
+            if (sourceThisType) {
+                var targetThisType = getThisTypeOfSignature(target);
+                if (targetThisType) {
+                    callback(sourceThisType, targetThisType);
+                }
             }
-            if (type.flags & 1048576 /* Union */) {
-                return getUnionType(ts.map(type.types, function (t) { return instantiateInstantiableTypes(t, mapper); }), 0 /* None */);
+            for (var i = 0; i < paramCount; i++) {
+                callback(getTypeAtPosition(source, i), getTypeAtPosition(target, i));
             }
-            if (type.flags & 2097152 /* Intersection */) {
-                return getIntersectionType(ts.map(type.types, function (t) { return instantiateInstantiableTypes(t, mapper); }));
+            if (targetRestType) {
+                callback(getRestTypeAtPosition(source, paramCount), targetRestType);
             }
-            return type;
         }
-        /**
-         * Whoa! Do you really want to use this function?
-         *
-         * Unless you're trying to get the *non-apparent* type for a
-         * value-literal type or you're authoring relevant portions of this algorithm,
-         * you probably meant to use 'getApparentTypeOfContextualType'.
-         * Otherwise this may not be very useful.
-         *
-         * In cases where you *are* working on this function, you should understand
-         * when it is appropriate to use 'getContextualType' and 'getApparentTypeOfContextualType'.
-         *
-         *   - Use 'getContextualType' when you are simply going to propagate the result to the expression.
-         *   - Use 'getApparentTypeOfContextualType' when you're going to need the members of the type.
-         *
-         * @param node the expression whose contextual type will be returned.
-         * @returns the contextual type of an expression.
-         */
-        function getContextualType(node, contextFlags) {
-            if (node.flags & 16777216 /* InWithStatement */) {
-                // We cannot answer semantic questions within a with block, do not proceed any further
-                return undefined;
+        function applyToReturnTypes(source, target, callback) {
+            var sourceTypePredicate = getTypePredicateOfSignature(source);
+            var targetTypePredicate = getTypePredicateOfSignature(target);
+            if (sourceTypePredicate && targetTypePredicate && typePredicateKindsMatch(sourceTypePredicate, targetTypePredicate) && sourceTypePredicate.type && targetTypePredicate.type) {
+                callback(sourceTypePredicate.type, targetTypePredicate.type);
             }
-            if (node.contextualType) {
-                return node.contextualType;
+            else {
+                callback(getReturnTypeOfSignature(source), getReturnTypeOfSignature(target));
             }
-            var parent = node.parent;
-            switch (parent.kind) {
-                case 242 /* VariableDeclaration */:
-                case 156 /* Parameter */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                case 191 /* BindingElement */:
-                    return getContextualTypeForInitializerExpression(node);
-                case 202 /* ArrowFunction */:
-                case 235 /* ReturnStatement */:
-                    return getContextualTypeForReturnExpression(node);
-                case 212 /* YieldExpression */:
-                    return getContextualTypeForYieldOperand(parent);
-                case 206 /* AwaitExpression */:
-                    return getContextualTypeForAwaitOperand(parent);
-                case 196 /* CallExpression */:
-                    if (parent.expression.kind === 96 /* ImportKeyword */) {
-                        return stringType;
+        }
+        function createInferenceContext(typeParameters, signature, flags, compareTypes) {
+            return createInferenceContextWorker(typeParameters.map(createInferenceInfo), signature, flags, compareTypes || compareTypesAssignable);
+        }
+        function cloneInferenceContext(context, extraFlags) {
+            if (extraFlags === void 0) { extraFlags = 0; }
+            return context && createInferenceContextWorker(ts.map(context.inferences, cloneInferenceInfo), context.signature, context.flags | extraFlags, context.compareTypes);
+        }
+        function createInferenceContextWorker(inferences, signature, flags, compareTypes) {
+            var context = {
+                inferences: inferences,
+                signature: signature,
+                flags: flags,
+                compareTypes: compareTypes,
+                mapper: makeFunctionTypeMapper(function (t) { return mapToInferredType(context, t, /*fix*/ true); }),
+                nonFixingMapper: makeFunctionTypeMapper(function (t) { return mapToInferredType(context, t, /*fix*/ false); }),
+            };
+            return context;
+        }
+        function mapToInferredType(context, t, fix) {
+            var inferences = context.inferences;
+            for (var i = 0; i < inferences.length; i++) {
+                var inference = inferences[i];
+                if (t === inference.typeParameter) {
+                    if (fix && !inference.isFixed) {
+                        clearCachedInferences(inferences);
+                        inference.isFixed = true;
                     }
-                /* falls through */
-                case 197 /* NewExpression */:
-                    return getContextualTypeForArgument(parent, node);
-                case 199 /* TypeAssertionExpression */:
-                case 217 /* AsExpression */:
-                    return ts.isConstTypeReference(parent.type) ? undefined : getTypeFromTypeNode(parent.type);
-                case 209 /* BinaryExpression */:
-                    return getContextualTypeForBinaryOperand(node, contextFlags);
-                case 281 /* PropertyAssignment */:
-                case 282 /* ShorthandPropertyAssignment */:
-                    return getContextualTypeForObjectLiteralElement(parent, contextFlags);
-                case 283 /* SpreadAssignment */:
-                    return getApparentTypeOfContextualType(parent.parent, contextFlags);
-                case 192 /* ArrayLiteralExpression */: {
-                    var arrayLiteral = parent;
-                    var type = getApparentTypeOfContextualType(arrayLiteral, contextFlags);
-                    return getContextualTypeForElementExpression(type, ts.indexOfNode(arrayLiteral.elements, node));
+                    return getInferredType(context, i);
                 }
-                case 210 /* ConditionalExpression */:
-                    return getContextualTypeForConditionalOperand(node, contextFlags);
-                case 221 /* TemplateSpan */:
-                    ts.Debug.assert(parent.parent.kind === 211 /* TemplateExpression */);
-                    return getContextualTypeForSubstitutionExpression(parent.parent, node);
-                case 200 /* ParenthesizedExpression */: {
-                    // Like in `checkParenthesizedExpression`, an `/** @type {xyz} */` comment before a parenthesized expression acts as a type cast.
-                    var tag = ts.isInJSFile(parent) ? ts.getJSDocTypeTag(parent) : undefined;
-                    return tag ? getTypeFromTypeNode(tag.typeExpression.type) : getContextualType(parent, contextFlags);
+            }
+            return t;
+        }
+        function clearCachedInferences(inferences) {
+            for (var _i = 0, inferences_1 = inferences; _i < inferences_1.length; _i++) {
+                var inference = inferences_1[_i];
+                if (!inference.isFixed) {
+                    inference.inferredType = undefined;
                 }
-                case 276 /* JsxExpression */:
-                    return getContextualTypeForJsxExpression(parent);
-                case 273 /* JsxAttribute */:
-                case 275 /* JsxSpreadAttribute */:
-                    return getContextualTypeForJsxAttribute(parent);
-                case 268 /* JsxOpeningElement */:
-                case 267 /* JsxSelfClosingElement */:
-                    return getContextualJsxElementAttributesType(parent, contextFlags);
             }
-            return undefined;
         }
-        function getInferenceContext(node) {
-            var ancestor = ts.findAncestor(node, function (n) { return !!n.inferenceContext; });
-            return ancestor && ancestor.inferenceContext;
+        function createInferenceInfo(typeParameter) {
+            return {
+                typeParameter: typeParameter,
+                candidates: undefined,
+                contraCandidates: undefined,
+                inferredType: undefined,
+                priority: undefined,
+                topLevel: true,
+                isFixed: false,
+                impliedArity: undefined
+            };
         }
-        function getContextualJsxElementAttributesType(node, contextFlags) {
-            if (ts.isJsxOpeningElement(node) && node.parent.contextualType && contextFlags !== 4 /* Completions */) {
-                // Contextually applied type is moved from attributes up to the outer jsx attributes so when walking up from the children they get hit
-                // _However_ to hit them from the _attributes_ we must look for them here; otherwise we'll used the declared type
-                // (as below) instead!
-                return node.parent.contextualType;
-            }
-            return getContextualTypeForArgumentAtIndex(node, 0);
+        function cloneInferenceInfo(inference) {
+            return {
+                typeParameter: inference.typeParameter,
+                candidates: inference.candidates && inference.candidates.slice(),
+                contraCandidates: inference.contraCandidates && inference.contraCandidates.slice(),
+                inferredType: inference.inferredType,
+                priority: inference.priority,
+                topLevel: inference.topLevel,
+                isFixed: inference.isFixed,
+                impliedArity: inference.impliedArity
+            };
         }
-        function getEffectiveFirstArgumentForJsxSignature(signature, node) {
-            return getJsxReferenceKind(node) !== 0 /* Component */
-                ? getJsxPropsTypeFromCallSignature(signature, node)
-                : getJsxPropsTypeFromClassType(signature, node);
+        function cloneInferredPartOfContext(context) {
+            var inferences = ts.filter(context.inferences, hasInferenceCandidates);
+            return inferences.length ?
+                createInferenceContextWorker(ts.map(inferences, cloneInferenceInfo), context.signature, context.flags, context.compareTypes) :
+                undefined;
         }
-        function getJsxPropsTypeFromCallSignature(sig, context) {
-            var propsType = getTypeOfFirstParameterOfSignatureWithFallback(sig, unknownType);
-            propsType = getJsxManagedAttributesFromLocatedAttributes(context, getJsxNamespaceAt(context), propsType);
-            var intrinsicAttribs = getJsxType(JsxNames.IntrinsicAttributes, context);
-            if (intrinsicAttribs !== errorType) {
-                propsType = intersectTypes(intrinsicAttribs, propsType);
-            }
-            return propsType;
+        function getMapperFromContext(context) {
+            return context && context.mapper;
         }
-        function getJsxPropsTypeForSignatureFromMember(sig, forcedLookupLocation) {
-            if (sig.unionSignatures) {
-                // JSX Elements using the legacy `props`-field based lookup (eg, react class components) need to treat the `props` member as an input
-                // instead of an output position when resolving the signature. We need to go back to the input signatures of the composite signature,
-                // get the type of `props` on each return type individually, and then _intersect them_, rather than union them (as would normally occur
-                // for a union signature). It's an unfortunate quirk of looking in the output of the signature for the type we want to use for the input.
-                // The default behavior of `getTypeOfFirstParameterOfSignatureWithFallback` when no `props` member name is defined is much more sane.
-                var results = [];
-                for (var _i = 0, _a = sig.unionSignatures; _i < _a.length; _i++) {
-                    var signature = _a[_i];
-                    var instance = getReturnTypeOfSignature(signature);
-                    if (isTypeAny(instance)) {
-                        return instance;
-                    }
-                    var propType = getTypeOfPropertyOfType(instance, forcedLookupLocation);
-                    if (!propType) {
-                        return;
-                    }
-                    results.push(propType);
-                }
-                return getIntersectionType(results);
+        // Return true if the given type could possibly reference a type parameter for which
+        // we perform type inference (i.e. a type parameter of a generic function). We cache
+        // results for union and intersection types for performance reasons.
+        function couldContainTypeVariables(type) {
+            var objectFlags = ts.getObjectFlags(type);
+            if (objectFlags & 67108864 /* CouldContainTypeVariablesComputed */) {
+                return !!(objectFlags & 134217728 /* CouldContainTypeVariables */);
             }
-            var instanceType = getReturnTypeOfSignature(sig);
-            return isTypeAny(instanceType) ? instanceType : getTypeOfPropertyOfType(instanceType, forcedLookupLocation);
-        }
-        function getStaticTypeOfReferencedJsxConstructor(context) {
-            if (isJsxIntrinsicIdentifier(context.tagName)) {
-                var result = getIntrinsicAttributesTypeFromJsxOpeningLikeElement(context);
-                var fakeSignature = createSignatureForJSXIntrinsic(context, result);
-                return getOrCreateTypeFromSignature(fakeSignature);
+            var result = !!(type.flags & 465829888 /* Instantiable */ ||
+                type.flags & 524288 /* Object */ && !isNonGenericTopLevelType(type) && (objectFlags & 4 /* Reference */ && (type.node || ts.forEach(getTypeArguments(type), couldContainTypeVariables)) ||
+                    objectFlags & 16 /* Anonymous */ && type.symbol && type.symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 2048 /* TypeLiteral */ | 4096 /* ObjectLiteral */) && type.symbol.declarations ||
+                    objectFlags & (32 /* Mapped */ | 131072 /* ObjectRestType */)) ||
+                type.flags & 3145728 /* UnionOrIntersection */ && !(type.flags & 1024 /* EnumLiteral */) && !isNonGenericTopLevelType(type) && ts.some(type.types, couldContainTypeVariables));
+            if (type.flags & 3899393 /* ObjectFlagsType */) {
+                type.objectFlags |= 67108864 /* CouldContainTypeVariablesComputed */ | (result ? 134217728 /* CouldContainTypeVariables */ : 0);
             }
-            var tagType = checkExpressionCached(context.tagName);
-            if (tagType.flags & 128 /* StringLiteral */) {
-                var result = getIntrinsicAttributesTypeFromStringLiteralType(tagType, context);
-                if (!result) {
-                    return errorType;
-                }
-                var fakeSignature = createSignatureForJSXIntrinsic(context, result);
-                return getOrCreateTypeFromSignature(fakeSignature);
+            return result;
+        }
+        function isNonGenericTopLevelType(type) {
+            if (type.aliasSymbol && !type.aliasTypeArguments) {
+                var declaration = ts.getDeclarationOfKind(type.aliasSymbol, 254 /* TypeAliasDeclaration */);
+                return !!(declaration && ts.findAncestor(declaration.parent, function (n) { return n.kind === 297 /* SourceFile */ ? true : n.kind === 256 /* ModuleDeclaration */ ? false : "quit"; }));
             }
-            return tagType;
+            return false;
         }
-        function getJsxManagedAttributesFromLocatedAttributes(context, ns, attributesType) {
-            var managedSym = getJsxLibraryManagedAttributes(ns);
-            if (managedSym) {
-                var declaredManagedType = getDeclaredTypeOfSymbol(managedSym);
-                var ctorType = getStaticTypeOfReferencedJsxConstructor(context);
-                if (ts.length(declaredManagedType.typeParameters) >= 2) {
-                    var args = fillMissingTypeArguments([ctorType, attributesType], declaredManagedType.typeParameters, 2, ts.isInJSFile(context));
-                    return createTypeReference(declaredManagedType, args);
+        function isTypeParameterAtTopLevel(type, typeParameter) {
+            return !!(type === typeParameter ||
+                type.flags & 3145728 /* UnionOrIntersection */ && ts.some(type.types, function (t) { return isTypeParameterAtTopLevel(t, typeParameter); }) ||
+                type.flags & 16777216 /* Conditional */ && (getTrueTypeFromConditionalType(type) === typeParameter || getFalseTypeFromConditionalType(type) === typeParameter));
+        }
+        /** Create an object with properties named in the string literal type. Every property has type `any` */
+        function createEmptyObjectTypeFromStringLiteral(type) {
+            var members = ts.createSymbolTable();
+            forEachType(type, function (t) {
+                if (!(t.flags & 128 /* StringLiteral */)) {
+                    return;
                 }
-                else if (ts.length(declaredManagedType.aliasTypeArguments) >= 2) {
-                    var args = fillMissingTypeArguments([ctorType, attributesType], declaredManagedType.aliasTypeArguments, 2, ts.isInJSFile(context));
-                    return getTypeAliasInstantiation(declaredManagedType.aliasSymbol, args);
+                var name = ts.escapeLeadingUnderscores(t.value);
+                var literalProp = createSymbol(4 /* Property */, name);
+                literalProp.type = anyType;
+                if (t.symbol) {
+                    literalProp.declarations = t.symbol.declarations;
+                    literalProp.valueDeclaration = t.symbol.valueDeclaration;
                 }
+                members.set(name, literalProp);
+            });
+            var indexInfo = type.flags & 4 /* String */ ? createIndexInfo(emptyObjectType, /*isReadonly*/ false) : undefined;
+            return createAnonymousType(undefined, members, ts.emptyArray, ts.emptyArray, indexInfo, undefined);
+        }
+        /**
+         * Infer a suitable input type for a homomorphic mapped type { [P in keyof T]: X }. We construct
+         * an object type with the same set of properties as the source type, where the type of each
+         * property is computed by inferring from the source property type to X for the type
+         * variable T[P] (i.e. we treat the type T[P] as the type variable we're inferring for).
+         */
+        function inferTypeForHomomorphicMappedType(source, target, constraint) {
+            if (inInferTypeForHomomorphicMappedType) {
+                return undefined;
             }
-            return attributesType;
+            var key = source.id + "," + target.id + "," + constraint.id;
+            if (reverseMappedCache.has(key)) {
+                return reverseMappedCache.get(key);
+            }
+            inInferTypeForHomomorphicMappedType = true;
+            var type = createReverseMappedType(source, target, constraint);
+            inInferTypeForHomomorphicMappedType = false;
+            reverseMappedCache.set(key, type);
+            return type;
         }
-        function getJsxPropsTypeFromClassType(sig, context) {
-            var ns = getJsxNamespaceAt(context);
-            var forcedLookupLocation = getJsxElementPropertiesName(ns);
-            var attributesType = forcedLookupLocation === undefined
-                // If there is no type ElementAttributesProperty, return the type of the first parameter of the signature, which should be the props type
-                ? getTypeOfFirstParameterOfSignatureWithFallback(sig, unknownType)
-                : forcedLookupLocation === ""
-                    // If there is no e.g. 'props' member in ElementAttributesProperty, use the element class type instead
-                    ? getReturnTypeOfSignature(sig)
-                    // Otherwise get the type of the property on the signature return type
-                    : getJsxPropsTypeForSignatureFromMember(sig, forcedLookupLocation);
-            if (!attributesType) {
-                // There is no property named 'props' on this instance type
-                if (!!forcedLookupLocation && !!ts.length(context.attributes.properties)) {
-                    error(context, ts.Diagnostics.JSX_element_class_does_not_support_attributes_because_it_does_not_have_a_0_property, ts.unescapeLeadingUnderscores(forcedLookupLocation));
-                }
-                return unknownType;
+        // We consider a type to be partially inferable if it isn't marked non-inferable or if it is
+        // an object literal type with at least one property of an inferable type. For example, an object
+        // literal { a: 123, b: x => true } is marked non-inferable because it contains a context sensitive
+        // arrow function, but is considered partially inferable because property 'a' has an inferable type.
+        function isPartiallyInferableType(type) {
+            return !(ts.getObjectFlags(type) & 2097152 /* NonInferrableType */) ||
+                isObjectLiteralType(type) && ts.some(getPropertiesOfType(type), function (prop) { return isPartiallyInferableType(getTypeOfSymbol(prop)); }) ||
+                isTupleType(type) && ts.some(getTypeArguments(type), isPartiallyInferableType);
+        }
+        function createReverseMappedType(source, target, constraint) {
+            // We consider a source type reverse mappable if it has a string index signature or if
+            // it has one or more properties and is of a partially inferable type.
+            if (!(getIndexInfoOfType(source, 0 /* String */) || getPropertiesOfType(source).length !== 0 && isPartiallyInferableType(source))) {
+                return undefined;
             }
-            attributesType = getJsxManagedAttributesFromLocatedAttributes(context, ns, attributesType);
-            if (isTypeAny(attributesType)) {
-                // Props is of type 'any' or unknown
-                return attributesType;
+            // For arrays and tuples we infer new arrays and tuples where the reverse mapping has been
+            // applied to the element type(s).
+            if (isArrayType(source)) {
+                return createArrayType(inferReverseMappedType(getTypeArguments(source)[0], target, constraint), isReadonlyArrayType(source));
             }
-            else {
-                // Normal case -- add in IntrinsicClassElements<T> and IntrinsicElements
-                var apparentAttributesType = attributesType;
-                var intrinsicClassAttribs = getJsxType(JsxNames.IntrinsicClassAttributes, context);
-                if (intrinsicClassAttribs !== errorType) {
-                    var typeParams = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(intrinsicClassAttribs.symbol);
-                    var hostClassType = getReturnTypeOfSignature(sig);
-                    apparentAttributesType = intersectTypes(typeParams
-                        ? createTypeReference(intrinsicClassAttribs, fillMissingTypeArguments([hostClassType], typeParams, getMinTypeArgumentCount(typeParams), ts.isInJSFile(context)))
-                        : intrinsicClassAttribs, apparentAttributesType);
-                }
-                var intrinsicAttribs = getJsxType(JsxNames.IntrinsicAttributes, context);
-                if (intrinsicAttribs !== errorType) {
-                    apparentAttributesType = intersectTypes(intrinsicAttribs, apparentAttributesType);
-                }
-                return apparentAttributesType;
+            if (isTupleType(source)) {
+                var elementTypes = ts.map(getTypeArguments(source), function (t) { return inferReverseMappedType(t, target, constraint); });
+                var elementFlags = getMappedTypeModifiers(target) & 4 /* IncludeOptional */ ?
+                    ts.sameMap(source.target.elementFlags, function (f) { return f & 2 /* Optional */ ? 1 /* Required */ : f; }) :
+                    source.target.elementFlags;
+                return createTupleType(elementTypes, elementFlags, source.target.readonly, source.target.labeledElementDeclarations);
             }
+            // For all other object types we infer a new object type where the reverse mapping has been
+            // applied to the type of each property.
+            var reversed = createObjectType(2048 /* ReverseMapped */ | 16 /* Anonymous */, /*symbol*/ undefined);
+            reversed.source = source;
+            reversed.mappedType = target;
+            reversed.constraintType = constraint;
+            return reversed;
         }
-        // If the given type is an object or union type with a single signature, and if that signature has at
-        // least as many parameters as the given function, return the signature. Otherwise return undefined.
-        function getContextualCallSignature(type, node) {
-            var signatures = getSignaturesOfType(type, 0 /* Call */);
-            if (signatures.length === 1) {
-                var signature = signatures[0];
-                if (!isAritySmaller(signature, node)) {
-                    return signature;
-                }
-            }
+        function getTypeOfReverseMappedSymbol(symbol) {
+            return inferReverseMappedType(symbol.propertyType, symbol.mappedType, symbol.constraintType);
         }
-        /** If the contextual signature has fewer parameters than the function expression, do not use it */
-        function isAritySmaller(signature, target) {
-            var targetParameterCount = 0;
-            for (; targetParameterCount < target.parameters.length; targetParameterCount++) {
-                var param = target.parameters[targetParameterCount];
-                if (param.initializer || param.questionToken || param.dotDotDotToken || isJSDocOptionalParameter(param)) {
-                    break;
+        function inferReverseMappedType(sourceType, target, constraint) {
+            var typeParameter = getIndexedAccessType(constraint.type, getTypeParameterFromMappedType(target));
+            var templateType = getTemplateTypeFromMappedType(target);
+            var inference = createInferenceInfo(typeParameter);
+            inferTypes([inference], sourceType, templateType);
+            return getTypeFromInference(inference) || unknownType;
+        }
+        function getUnmatchedProperties(source, target, requireOptionalProperties, matchDiscriminantProperties) {
+            var properties, _i, properties_2, targetProp, sourceProp, targetType, sourceType;
+            return __generator(this, function (_a) {
+                switch (_a.label) {
+                    case 0:
+                        properties = getPropertiesOfType(target);
+                        _i = 0, properties_2 = properties;
+                        _a.label = 1;
+                    case 1:
+                        if (!(_i < properties_2.length)) return [3 /*break*/, 6];
+                        targetProp = properties_2[_i];
+                        // TODO: remove this when we support static private identifier fields and find other solutions to get privateNamesAndStaticFields test to pass
+                        if (isStaticPrivateIdentifierProperty(targetProp)) {
+                            return [3 /*break*/, 5];
+                        }
+                        if (!(requireOptionalProperties || !(targetProp.flags & 16777216 /* Optional */ || ts.getCheckFlags(targetProp) & 48 /* Partial */))) return [3 /*break*/, 5];
+                        sourceProp = getPropertyOfType(source, targetProp.escapedName);
+                        if (!!sourceProp) return [3 /*break*/, 3];
+                        return [4 /*yield*/, targetProp];
+                    case 2:
+                        _a.sent();
+                        return [3 /*break*/, 5];
+                    case 3:
+                        if (!matchDiscriminantProperties) return [3 /*break*/, 5];
+                        targetType = getTypeOfSymbol(targetProp);
+                        if (!(targetType.flags & 109440 /* Unit */)) return [3 /*break*/, 5];
+                        sourceType = getTypeOfSymbol(sourceProp);
+                        if (!!(sourceType.flags & 1 /* Any */ || getRegularTypeOfLiteralType(sourceType) === getRegularTypeOfLiteralType(targetType))) return [3 /*break*/, 5];
+                        return [4 /*yield*/, targetProp];
+                    case 4:
+                        _a.sent();
+                        _a.label = 5;
+                    case 5:
+                        _i++;
+                        return [3 /*break*/, 1];
+                    case 6: return [2 /*return*/];
                 }
-            }
-            if (target.parameters.length && ts.parameterIsThisKeyword(target.parameters[0])) {
-                targetParameterCount--;
-            }
-            return !hasEffectiveRestParameter(signature) && getParameterCount(signature) < targetParameterCount;
+            });
         }
-        function isFunctionExpressionOrArrowFunction(node) {
-            return node.kind === 201 /* FunctionExpression */ || node.kind === 202 /* ArrowFunction */;
+        function getUnmatchedProperty(source, target, requireOptionalProperties, matchDiscriminantProperties) {
+            var result = getUnmatchedProperties(source, target, requireOptionalProperties, matchDiscriminantProperties).next();
+            if (!result.done)
+                return result.value;
         }
-        function getContextualSignatureForFunctionLikeDeclaration(node) {
-            // Only function expressions, arrow functions, and object literal methods are contextually typed.
-            return isFunctionExpressionOrArrowFunction(node) || ts.isObjectLiteralMethod(node)
-                ? getContextualSignature(node)
-                : undefined;
+        function tupleTypesDefinitelyUnrelated(source, target) {
+            return !(target.target.combinedFlags & 8 /* Variadic */) && target.target.minLength > source.target.minLength ||
+                !target.target.hasRestElement && (source.target.hasRestElement || target.target.fixedLength < source.target.fixedLength);
         }
-        // Return the contextual signature for a given expression node. A contextual type provides a
-        // contextual signature if it has a single call signature and if that call signature is non-generic.
-        // If the contextual type is a union type, get the signature from each type possible and if they are
-        // all identical ignoring their return type, the result is same signature but with return type as
-        // union type of return types from these signatures
-        function getContextualSignature(node) {
-            ts.Debug.assert(node.kind !== 161 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
-            var typeTagSignature = getSignatureOfTypeTag(node);
-            if (typeTagSignature) {
-                return typeTagSignature;
-            }
-            var type = getApparentTypeOfContextualType(node, 1 /* Signature */);
-            if (!type) {
+        function typesDefinitelyUnrelated(source, target) {
+            // Two tuple types with incompatible arities are definitely unrelated.
+            // Two object types that each have a property that is unmatched in the other are definitely unrelated.
+            return isTupleType(source) && isTupleType(target) ? tupleTypesDefinitelyUnrelated(source, target) :
+                !!getUnmatchedProperty(source, target, /*requireOptionalProperties*/ false, /*matchDiscriminantProperties*/ true) &&
+                    !!getUnmatchedProperty(target, source, /*requireOptionalProperties*/ false, /*matchDiscriminantProperties*/ true);
+        }
+        function getTypeFromInference(inference) {
+            return inference.candidates ? getUnionType(inference.candidates, 2 /* Subtype */) :
+                inference.contraCandidates ? getIntersectionType(inference.contraCandidates) :
+                    undefined;
+        }
+        function hasSkipDirectInferenceFlag(node) {
+            return !!getNodeLinks(node).skipDirectInference;
+        }
+        function isFromInferenceBlockedSource(type) {
+            return !!(type.symbol && ts.some(type.symbol.declarations, hasSkipDirectInferenceFlag));
+        }
+        function isValidBigIntString(s) {
+            var scanner = ts.createScanner(99 /* ESNext */, /*skipTrivia*/ false);
+            var success = true;
+            scanner.setOnError(function () { return success = false; });
+            scanner.setText(s + "n");
+            var result = scanner.scan();
+            if (result === 40 /* MinusToken */) {
+                result = scanner.scan();
+            }
+            var flags = scanner.getTokenFlags();
+            // validate that
+            // * scanning proceeded without error
+            // * a bigint can be scanned, and that when it is scanned, it is
+            // * the full length of the input string (so the scanner is one character beyond the augmented input length)
+            // * it does not contain a numeric seperator (the `BigInt` constructor does not accept a numeric seperator in its input)
+            return success && result === 9 /* BigIntLiteral */ && scanner.getTextPos() === (s.length + 1) && !(flags & 512 /* ContainsSeparator */);
+        }
+        function isStringLiteralTypeValueParsableAsType(s, target) {
+            if (target.flags & 1048576 /* Union */) {
+                return !!forEachType(target, function (t) { return isStringLiteralTypeValueParsableAsType(s, t); });
+            }
+            switch (target) {
+                case stringType: return true;
+                case numberType: return s.value !== "" && isFinite(+(s.value));
+                case bigintType: return s.value !== "" && isValidBigIntString(s.value);
+                // the next 4 should be handled in `getTemplateLiteralType`, as they are all exactly one value, but are here for completeness, just in case
+                // this function is ever used on types which don't come from template literal holes
+                case trueType: return s.value === "true";
+                case falseType: return s.value === "false";
+                case undefinedType: return s.value === "undefined";
+                case nullType: return s.value === "null";
+                default: return !!(target.flags & 1 /* Any */);
+            }
+        }
+        function inferLiteralsFromTemplateLiteralType(source, target) {
+            var value = source.value;
+            var texts = target.texts;
+            var lastIndex = texts.length - 1;
+            var startText = texts[0];
+            var endText = texts[lastIndex];
+            if (!(value.startsWith(startText) && value.slice(startText.length).endsWith(endText)))
                 return undefined;
+            var matches = [];
+            var str = value.slice(startText.length, value.length - endText.length);
+            var pos = 0;
+            for (var i = 1; i < lastIndex; i++) {
+                var delim = texts[i];
+                var delimPos = delim.length > 0 ? str.indexOf(delim, pos) : pos < str.length ? pos + 1 : -1;
+                if (delimPos < 0)
+                    return undefined;
+                matches.push(getLiteralType(str.slice(pos, delimPos)));
+                pos = delimPos + delim.length;
             }
-            if (!(type.flags & 1048576 /* Union */)) {
-                return getContextualCallSignature(type, node);
-            }
-            var signatureList;
-            var types = type.types;
-            for (var _i = 0, types_17 = types; _i < types_17.length; _i++) {
-                var current = types_17[_i];
-                var signature = getContextualCallSignature(current, node);
-                if (signature) {
-                    if (!signatureList) {
-                        // This signature will contribute to contextual union signature
-                        signatureList = [signature];
+            matches.push(getLiteralType(str.slice(pos)));
+            return matches;
+        }
+        function inferTypes(inferences, originalSource, originalTarget, priority, contravariant) {
+            if (priority === void 0) { priority = 0; }
+            if (contravariant === void 0) { contravariant = false; }
+            var bivariant = false;
+            var propagationType;
+            var inferencePriority = 1024 /* MaxValue */;
+            var allowComplexConstraintInference = true;
+            var visited;
+            var sourceStack;
+            var targetStack;
+            var expandingFlags = 0 /* None */;
+            inferFromTypes(originalSource, originalTarget);
+            function inferFromTypes(source, target) {
+                if (!couldContainTypeVariables(target)) {
+                    return;
+                }
+                if (source === wildcardType) {
+                    // We are inferring from an 'any' type. We want to infer this type for every type parameter
+                    // referenced in the target type, so we record it as the propagation type and infer from the
+                    // target to itself. Then, as we find candidates we substitute the propagation type.
+                    var savePropagationType = propagationType;
+                    propagationType = source;
+                    inferFromTypes(target, target);
+                    propagationType = savePropagationType;
+                    return;
+                }
+                if (source.aliasSymbol && source.aliasTypeArguments && source.aliasSymbol === target.aliasSymbol) {
+                    // Source and target are types originating in the same generic type alias declaration.
+                    // Simply infer from source type arguments to target type arguments.
+                    inferFromTypeArguments(source.aliasTypeArguments, target.aliasTypeArguments, getAliasVariances(source.aliasSymbol));
+                    return;
+                }
+                if (source === target && source.flags & 3145728 /* UnionOrIntersection */) {
+                    // When source and target are the same union or intersection type, just relate each constituent
+                    // type to itself.
+                    for (var _i = 0, _a = source.types; _i < _a.length; _i++) {
+                        var t = _a[_i];
+                        inferFromTypes(t, t);
                     }
-                    else if (!compareSignaturesIdentical(signatureList[0], signature, /*partialMatch*/ false, /*ignoreThisTypes*/ true, /*ignoreReturnTypes*/ true, compareTypesIdentical)) {
-                        // Signatures aren't identical, do not use
-                        return undefined;
+                    return;
+                }
+                if (target.flags & 1048576 /* Union */) {
+                    // First, infer between identically matching source and target constituents and remove the
+                    // matching types.
+                    var _b = inferFromMatchingTypes(source.flags & 1048576 /* Union */ ? source.types : [source], target.types, isTypeOrBaseIdenticalTo), tempSources = _b[0], tempTargets = _b[1];
+                    // Next, infer between closely matching source and target constituents and remove
+                    // the matching types. Types closely match when they are instantiations of the same
+                    // object type or instantiations of the same type alias.
+                    var _c = inferFromMatchingTypes(tempSources, tempTargets, isTypeCloselyMatchedBy), sources = _c[0], targets = _c[1];
+                    if (targets.length === 0) {
+                        return;
                     }
-                    else {
-                        // Use this signature for contextual union signature
-                        signatureList.push(signature);
+                    target = getUnionType(targets);
+                    if (sources.length === 0) {
+                        // All source constituents have been matched and there is nothing further to infer from.
+                        // However, simply making no inferences is undesirable because it could ultimately mean
+                        // inferring a type parameter constraint. Instead, make a lower priority inference from
+                        // the full source to whatever remains in the target. For example, when inferring from
+                        // string to 'string | T', make a lower priority inference of string for T.
+                        inferWithPriority(source, target, 1 /* NakedTypeVariable */);
+                        return;
                     }
+                    source = getUnionType(sources);
                 }
-            }
-            // Result is union of signatures collected (return type is union of return types of this signature set)
-            if (signatureList) {
-                return signatureList.length === 1 ? signatureList[0] : createUnionSignature(signatureList[0], signatureList);
-            }
-        }
-        function checkSpreadExpression(node, checkMode) {
-            if (languageVersion < 2 /* ES2015 */) {
-                checkExternalEmitHelpers(node, compilerOptions.downlevelIteration ? 1536 /* SpreadIncludes */ : 2048 /* SpreadArrays */);
-            }
-            var arrayOrIterableType = checkExpression(node.expression, checkMode);
-            return checkIteratedTypeOrElementType(33 /* Spread */, arrayOrIterableType, undefinedType, node.expression);
-        }
-        function hasDefaultValue(node) {
-            return (node.kind === 191 /* BindingElement */ && !!node.initializer) ||
-                (node.kind === 209 /* BinaryExpression */ && node.operatorToken.kind === 62 /* EqualsToken */);
-        }
-        function checkArrayLiteral(node, checkMode, forceTuple) {
-            var elements = node.elements;
-            var elementCount = elements.length;
-            var elementTypes = [];
-            var hasEndingSpreadElement = false;
-            var hasNonEndingSpreadElement = false;
-            var contextualType = getApparentTypeOfContextualType(node);
-            var inDestructuringPattern = ts.isAssignmentTarget(node);
-            var inConstContext = isConstContext(node);
-            for (var i = 0; i < elementCount; i++) {
-                var e = elements[i];
-                var spread = e.kind === 213 /* SpreadElement */ && e.expression;
-                var spreadType = spread && checkExpression(spread, checkMode, forceTuple);
-                if (spreadType && isTupleType(spreadType)) {
-                    elementTypes.push.apply(elementTypes, getTypeArguments(spreadType));
-                    if (spreadType.target.hasRestElement) {
-                        if (i === elementCount - 1)
-                            hasEndingSpreadElement = true;
-                        else
-                            hasNonEndingSpreadElement = true;
+                else if (target.flags & 2097152 /* Intersection */ && ts.some(target.types, function (t) { return !!getInferenceInfoForType(t) || (isGenericMappedType(t) && !!getInferenceInfoForType(getHomomorphicTypeVariable(t) || neverType)); })) {
+                    // We reduce intersection types only when they contain naked type parameters. For example, when
+                    // inferring from 'string[] & { extra: any }' to 'string[] & T' we want to remove string[] and
+                    // infer { extra: any } for T. But when inferring to 'string[] & Iterable<T>' we want to keep the
+                    // string[] on the source side and infer string for T.
+                    // Likewise, we consider a homomorphic mapped type constrainted to the target type parameter as similar to a "naked type variable"
+                    // in such scenarios.
+                    if (!(source.flags & 1048576 /* Union */)) {
+                        // Infer between identically matching source and target constituents and remove the matching types.
+                        var _d = inferFromMatchingTypes(source.flags & 2097152 /* Intersection */ ? source.types : [source], target.types, isTypeIdenticalTo), sources = _d[0], targets = _d[1];
+                        if (sources.length === 0 || targets.length === 0) {
+                            return;
+                        }
+                        source = getIntersectionType(sources);
+                        target = getIntersectionType(targets);
                     }
                 }
-                else {
-                    if (inDestructuringPattern && spreadType) {
-                        // Given the following situation:
-                        //    var c: {};
-                        //    [...c] = ["", 0];
-                        //
-                        // c is represented in the tree as a spread element in an array literal.
-                        // But c really functions as a rest element, and its purpose is to provide
-                        // a contextual type for the right hand side of the assignment. Therefore,
-                        // instead of calling checkExpression on "...c", which will give an error
-                        // if c is not iterable/array-like, we need to act as if we are trying to
-                        // get the contextual element type from it. So we do something similar to
-                        // getContextualTypeForElementExpression, which will crucially not error
-                        // if there is no index type / iterated type.
-                        var restElementType = getIndexTypeOfType(spreadType, 1 /* Number */) ||
-                            getIteratedTypeOrElementType(65 /* Destructuring */, spreadType, undefinedType, /*errorNode*/ undefined, /*checkAssignability*/ false);
-                        if (restElementType) {
-                            elementTypes.push(restElementType);
+                else if (target.flags & (8388608 /* IndexedAccess */ | 33554432 /* Substitution */)) {
+                    target = getActualTypeVariable(target);
+                }
+                if (target.flags & 8650752 /* TypeVariable */) {
+                    // If target is a type parameter, make an inference, unless the source type contains
+                    // the anyFunctionType (the wildcard type that's used to avoid contextually typing functions).
+                    // Because the anyFunctionType is internal, it should not be exposed to the user by adding
+                    // it as an inference candidate. Hopefully, a better candidate will come along that does
+                    // not contain anyFunctionType when we come back to this argument for its second round
+                    // of inference. Also, we exclude inferences for silentNeverType (which is used as a wildcard
+                    // when constructing types from type parameters that had no inference candidates).
+                    if (ts.getObjectFlags(source) & 2097152 /* NonInferrableType */ || source === nonInferrableAnyType || source === silentNeverType ||
+                        (priority & 64 /* ReturnType */ && (source === autoType || source === autoArrayType)) || isFromInferenceBlockedSource(source)) {
+                        return;
+                    }
+                    var inference = getInferenceInfoForType(target);
+                    if (inference) {
+                        if (!inference.isFixed) {
+                            if (inference.priority === undefined || priority < inference.priority) {
+                                inference.candidates = undefined;
+                                inference.contraCandidates = undefined;
+                                inference.topLevel = true;
+                                inference.priority = priority;
+                            }
+                            if (priority === inference.priority) {
+                                var candidate = propagationType || source;
+                                // We make contravariant inferences only if we are in a pure contravariant position,
+                                // i.e. only if we have not descended into a bivariant position.
+                                if (contravariant && !bivariant) {
+                                    if (!ts.contains(inference.contraCandidates, candidate)) {
+                                        inference.contraCandidates = ts.append(inference.contraCandidates, candidate);
+                                        clearCachedInferences(inferences);
+                                    }
+                                }
+                                else if (!ts.contains(inference.candidates, candidate)) {
+                                    inference.candidates = ts.append(inference.candidates, candidate);
+                                    clearCachedInferences(inferences);
+                                }
+                            }
+                            if (!(priority & 64 /* ReturnType */) && target.flags & 262144 /* TypeParameter */ && inference.topLevel && !isTypeParameterAtTopLevel(originalTarget, target)) {
+                                inference.topLevel = false;
+                                clearCachedInferences(inferences);
+                            }
                         }
+                        inferencePriority = Math.min(inferencePriority, priority);
+                        return;
                     }
                     else {
-                        var elementContextualType = getContextualTypeForElementExpression(contextualType, elementTypes.length);
-                        var type = checkExpressionForMutableLocation(e, checkMode, elementContextualType, forceTuple);
-                        elementTypes.push(type);
+                        // Infer to the simplified version of an indexed access, if possible, to (hopefully) expose more bare type parameters to the inference engine
+                        var simplified = getSimplifiedType(target, /*writing*/ false);
+                        if (simplified !== target) {
+                            invokeOnce(source, simplified, inferFromTypes);
+                        }
+                        else if (target.flags & 8388608 /* IndexedAccess */) {
+                            var indexType = getSimplifiedType(target.indexType, /*writing*/ false);
+                            // Generally simplifications of instantiable indexes are avoided to keep relationship checking correct, however if our target is an access, we can consider
+                            // that key of that access to be "instantiated", since we're looking to find the infernce goal in any way we can.
+                            if (indexType.flags & 465829888 /* Instantiable */) {
+                                var simplified_1 = distributeIndexOverObjectType(getSimplifiedType(target.objectType, /*writing*/ false), indexType, /*writing*/ false);
+                                if (simplified_1 && simplified_1 !== target) {
+                                    invokeOnce(source, simplified_1, inferFromTypes);
+                                }
+                            }
+                        }
                     }
-                    if (spread) { // tuples are done above, so these are only arrays
-                        if (i === elementCount - 1)
-                            hasEndingSpreadElement = true;
-                        else
-                            hasNonEndingSpreadElement = true;
+                }
+                if (ts.getObjectFlags(source) & 4 /* Reference */ && ts.getObjectFlags(target) & 4 /* Reference */ && (source.target === target.target || isArrayType(source) && isArrayType(target)) &&
+                    !(source.node && target.node)) {
+                    // If source and target are references to the same generic type, infer from type arguments
+                    inferFromTypeArguments(getTypeArguments(source), getTypeArguments(target), getVariances(source.target));
+                }
+                else if (source.flags & 4194304 /* Index */ && target.flags & 4194304 /* Index */) {
+                    contravariant = !contravariant;
+                    inferFromTypes(source.type, target.type);
+                    contravariant = !contravariant;
+                }
+                else if ((isLiteralType(source) || source.flags & 4 /* String */) && target.flags & 4194304 /* Index */) {
+                    var empty = createEmptyObjectTypeFromStringLiteral(source);
+                    contravariant = !contravariant;
+                    inferWithPriority(empty, target.type, 128 /* LiteralKeyof */);
+                    contravariant = !contravariant;
+                }
+                else if (source.flags & 8388608 /* IndexedAccess */ && target.flags & 8388608 /* IndexedAccess */) {
+                    inferFromTypes(source.objectType, target.objectType);
+                    inferFromTypes(source.indexType, target.indexType);
+                }
+                else if (source.flags & 268435456 /* StringMapping */ && target.flags & 268435456 /* StringMapping */) {
+                    if (source.symbol === target.symbol) {
+                        inferFromTypes(source.type, target.type);
                     }
                 }
-            }
-            if (!hasNonEndingSpreadElement) {
-                var minLength = elementTypes.length - (hasEndingSpreadElement ? 1 : 0);
-                // If array literal is actually a destructuring pattern, mark it as an implied type. We do this such
-                // that we get the same behavior for "var [x, y] = []" and "[x, y] = []".
-                var tupleResult = void 0;
-                if (inDestructuringPattern && minLength > 0) {
-                    var type = cloneTypeReference(createTupleType(elementTypes, minLength, hasEndingSpreadElement));
-                    type.pattern = node;
-                    return type;
+                else if (target.flags & 16777216 /* Conditional */) {
+                    invokeOnce(source, target, inferToConditionalType);
+                }
+                else if (target.flags & 3145728 /* UnionOrIntersection */) {
+                    inferToMultipleTypes(source, target.types, target.flags);
+                }
+                else if (source.flags & 1048576 /* Union */) {
+                    // Source is a union or intersection type, infer from each constituent type
+                    var sourceTypes = source.types;
+                    for (var _e = 0, sourceTypes_2 = sourceTypes; _e < sourceTypes_2.length; _e++) {
+                        var sourceType = sourceTypes_2[_e];
+                        inferFromTypes(sourceType, target);
+                    }
                 }
-                else if (tupleResult = getArrayLiteralTupleTypeIfApplicable(elementTypes, contextualType, hasEndingSpreadElement, elementTypes.length, inConstContext)) {
-                    return createArrayLiteralType(tupleResult);
+                else if (target.flags & 134217728 /* TemplateLiteral */) {
+                    inferToTemplateLiteralType(source, target);
+                }
+                else {
+                    source = getReducedType(source);
+                    if (!(priority & 256 /* NoConstraints */ && source.flags & (2097152 /* Intersection */ | 465829888 /* Instantiable */))) {
+                        var apparentSource = getApparentType(source);
+                        // getApparentType can return _any_ type, since an indexed access or conditional may simplify to any other type.
+                        // If that occurs and it doesn't simplify to an object or intersection, we'll need to restart `inferFromTypes`
+                        // with the simplified source.
+                        if (apparentSource !== source && allowComplexConstraintInference && !(apparentSource.flags & (524288 /* Object */ | 2097152 /* Intersection */))) {
+                            // TODO: The `allowComplexConstraintInference` flag is a hack! This forbids inference from complex constraints within constraints!
+                            // This isn't required algorithmically, but rather is used to lower the memory burden caused by performing inference
+                            // that is _too good_ in projects with complicated constraints (eg, fp-ts). In such cases, if we did not limit ourselves
+                            // here, we might produce more valid inferences for types, causing us to do more checks and perform more instantiations
+                            // (in addition to the extra stack depth here) which, in turn, can push the already close process over its limit.
+                            // TL;DR: If we ever become generally more memory efficient (or our resource budget ever increases), we should just
+                            // remove this `allowComplexConstraintInference` flag.
+                            allowComplexConstraintInference = false;
+                            return inferFromTypes(apparentSource, target);
+                        }
+                        source = apparentSource;
+                    }
+                    if (source.flags & (524288 /* Object */ | 2097152 /* Intersection */)) {
+                        invokeOnce(source, target, inferFromObjectTypes);
+                    }
                 }
-                else if (forceTuple) {
-                    return createArrayLiteralType(createTupleType(elementTypes, minLength, hasEndingSpreadElement));
+                if (source.flags & 25165824 /* Simplifiable */) {
+                    var simplified = getSimplifiedType(source, contravariant);
+                    if (simplified !== source) {
+                        inferFromTypes(simplified, target);
+                    }
                 }
             }
-            return createArrayLiteralType(createArrayType(elementTypes.length ?
-                getUnionType(elementTypes, 2 /* Subtype */) :
-                strictNullChecks ? implicitNeverType : undefinedWideningType, inConstContext));
-        }
-        function createArrayLiteralType(type) {
-            if (!(ts.getObjectFlags(type) & 4 /* Reference */)) {
-                return type;
+            function inferWithPriority(source, target, newPriority) {
+                var savePriority = priority;
+                priority |= newPriority;
+                inferFromTypes(source, target);
+                priority = savePriority;
             }
-            var literalType = type.literalType;
-            if (!literalType) {
-                literalType = type.literalType = cloneTypeReference(type);
-                literalType.objectFlags |= 65536 /* ArrayLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
+            function invokeOnce(source, target, action) {
+                var key = source.id + "," + target.id;
+                var status = visited && visited.get(key);
+                if (status !== undefined) {
+                    inferencePriority = Math.min(inferencePriority, status);
+                    return;
+                }
+                (visited || (visited = new ts.Map())).set(key, -1 /* Circularity */);
+                var saveInferencePriority = inferencePriority;
+                inferencePriority = 1024 /* MaxValue */;
+                // We stop inferring and report a circularity if we encounter duplicate recursion identities on both
+                // the source side and the target side.
+                var saveExpandingFlags = expandingFlags;
+                var sourceIdentity = getRecursionIdentity(source) || source;
+                var targetIdentity = getRecursionIdentity(target) || target;
+                if (sourceIdentity && ts.contains(sourceStack, sourceIdentity))
+                    expandingFlags |= 1 /* Source */;
+                if (targetIdentity && ts.contains(targetStack, targetIdentity))
+                    expandingFlags |= 2 /* Target */;
+                if (expandingFlags !== 3 /* Both */) {
+                    if (sourceIdentity)
+                        (sourceStack || (sourceStack = [])).push(sourceIdentity);
+                    if (targetIdentity)
+                        (targetStack || (targetStack = [])).push(targetIdentity);
+                    action(source, target);
+                    if (targetIdentity)
+                        targetStack.pop();
+                    if (sourceIdentity)
+                        sourceStack.pop();
+                }
+                else {
+                    inferencePriority = -1 /* Circularity */;
+                }
+                expandingFlags = saveExpandingFlags;
+                visited.set(key, inferencePriority);
+                inferencePriority = Math.min(inferencePriority, saveInferencePriority);
             }
-            return literalType;
-        }
-        function getArrayLiteralTupleTypeIfApplicable(elementTypes, contextualType, hasRestElement, elementCount, readonly) {
-            if (elementCount === void 0) { elementCount = elementTypes.length; }
-            if (readonly === void 0) { readonly = false; }
-            // Infer a tuple type when the contextual type is or contains a tuple-like type
-            if (readonly || (contextualType && forEachType(contextualType, isTupleLikeType))) {
-                return createTupleType(elementTypes, elementCount - (hasRestElement ? 1 : 0), hasRestElement, readonly);
+            function inferFromMatchingTypes(sources, targets, matches) {
+                var matchedSources;
+                var matchedTargets;
+                for (var _i = 0, targets_1 = targets; _i < targets_1.length; _i++) {
+                    var t = targets_1[_i];
+                    for (var _a = 0, sources_1 = sources; _a < sources_1.length; _a++) {
+                        var s = sources_1[_a];
+                        if (matches(s, t)) {
+                            inferFromTypes(s, t);
+                            matchedSources = ts.appendIfUnique(matchedSources, s);
+                            matchedTargets = ts.appendIfUnique(matchedTargets, t);
+                        }
+                    }
+                }
+                return [
+                    matchedSources ? ts.filter(sources, function (t) { return !ts.contains(matchedSources, t); }) : sources,
+                    matchedTargets ? ts.filter(targets, function (t) { return !ts.contains(matchedTargets, t); }) : targets,
+                ];
             }
-        }
-        function isNumericName(name) {
-            switch (name.kind) {
-                case 154 /* ComputedPropertyName */:
-                    return isNumericComputedName(name);
-                case 75 /* Identifier */:
-                    return isNumericLiteralName(name.escapedText);
-                case 8 /* NumericLiteral */:
-                case 10 /* StringLiteral */:
-                    return isNumericLiteralName(name.text);
-                default:
-                    return false;
+            function inferFromTypeArguments(sourceTypes, targetTypes, variances) {
+                var count = sourceTypes.length < targetTypes.length ? sourceTypes.length : targetTypes.length;
+                for (var i = 0; i < count; i++) {
+                    if (i < variances.length && (variances[i] & 7 /* VarianceMask */) === 2 /* Contravariant */) {
+                        inferFromContravariantTypes(sourceTypes[i], targetTypes[i]);
+                    }
+                    else {
+                        inferFromTypes(sourceTypes[i], targetTypes[i]);
+                    }
+                }
             }
-        }
-        function isNumericComputedName(name) {
-            // It seems odd to consider an expression of type Any to result in a numeric name,
-            // but this behavior is consistent with checkIndexedAccess
-            return isTypeAssignableToKind(checkComputedPropertyName(name), 296 /* NumberLike */);
-        }
-        function isInfinityOrNaNString(name) {
-            return name === "Infinity" || name === "-Infinity" || name === "NaN";
-        }
-        function isNumericLiteralName(name) {
-            // The intent of numeric names is that
-            //     - they are names with text in a numeric form, and that
-            //     - setting properties/indexing with them is always equivalent to doing so with the numeric literal 'numLit',
-            //         acquired by applying the abstract 'ToNumber' operation on the name's text.
-            //
-            // The subtlety is in the latter portion, as we cannot reliably say that anything that looks like a numeric literal is a numeric name.
-            // In fact, it is the case that the text of the name must be equal to 'ToString(numLit)' for this to hold.
-            //
-            // Consider the property name '"0xF00D"'. When one indexes with '0xF00D', they are actually indexing with the value of 'ToString(0xF00D)'
-            // according to the ECMAScript specification, so it is actually as if the user indexed with the string '"61453"'.
-            // Thus, the text of all numeric literals equivalent to '61543' such as '0xF00D', '0xf00D', '0170015', etc. are not valid numeric names
-            // because their 'ToString' representation is not equal to their original text.
-            // This is motivated by ECMA-262 sections 9.3.1, 9.8.1, 11.1.5, and 11.2.1.
-            //
-            // Here, we test whether 'ToString(ToNumber(name))' is exactly equal to 'name'.
-            // The '+' prefix operator is equivalent here to applying the abstract ToNumber operation.
-            // Applying the 'toString()' method on a number gives us the abstract ToString operation on a number.
-            //
-            // Note that this accepts the values 'Infinity', '-Infinity', and 'NaN', and that this is intentional.
-            // This is desired behavior, because when indexing with them as numeric entities, you are indexing
-            // with the strings '"Infinity"', '"-Infinity"', and '"NaN"' respectively.
-            return (+name).toString() === name;
-        }
-        function checkComputedPropertyName(node) {
-            var links = getNodeLinks(node.expression);
-            if (!links.resolvedType) {
-                links.resolvedType = checkExpression(node.expression);
-                // This will allow types number, string, symbol or any. It will also allow enums, the unknown
-                // type, and any union of these types (like string | number).
-                if (links.resolvedType.flags & 98304 /* Nullable */ ||
-                    !isTypeAssignableToKind(links.resolvedType, 132 /* StringLike */ | 296 /* NumberLike */ | 12288 /* ESSymbolLike */) &&
-                        !isTypeAssignableTo(links.resolvedType, stringNumberSymbolType)) {
-                    error(node, ts.Diagnostics.A_computed_property_name_must_be_of_type_string_number_symbol_or_any);
+            function inferFromContravariantTypes(source, target) {
+                if (strictFunctionTypes || priority & 512 /* AlwaysStrict */) {
+                    contravariant = !contravariant;
+                    inferFromTypes(source, target);
+                    contravariant = !contravariant;
                 }
                 else {
-                    checkThatExpressionIsProperSymbolReference(node.expression, links.resolvedType, /*reportError*/ true);
+                    inferFromTypes(source, target);
                 }
             }
-            return links.resolvedType;
-        }
-        function getObjectLiteralIndexInfo(node, offset, properties, kind) {
-            var propTypes = [];
-            for (var i = 0; i < properties.length; i++) {
-                if (kind === 0 /* String */ || isNumericName(node.properties[i + offset].name)) {
-                    propTypes.push(getTypeOfSymbol(properties[i]));
+            function getInferenceInfoForType(type) {
+                if (type.flags & 8650752 /* TypeVariable */) {
+                    for (var _i = 0, inferences_2 = inferences; _i < inferences_2.length; _i++) {
+                        var inference = inferences_2[_i];
+                        if (type === inference.typeParameter) {
+                            return inference;
+                        }
+                    }
                 }
+                return undefined;
             }
-            var unionType = propTypes.length ? getUnionType(propTypes, 2 /* Subtype */) : undefinedType;
-            return createIndexInfo(unionType, isConstContext(node));
-        }
-        function getImmediateAliasedSymbol(symbol) {
-            ts.Debug.assert((symbol.flags & 2097152 /* Alias */) !== 0, "Should only get Alias here.");
-            var links = getSymbolLinks(symbol);
-            if (!links.immediateTarget) {
-                var node = getDeclarationOfAliasSymbol(symbol);
-                if (!node)
-                    return ts.Debug.fail();
-                links.immediateTarget = getTargetOfAliasDeclaration(node, /*dontRecursivelyResolve*/ true);
-            }
-            return links.immediateTarget;
-        }
-        function checkObjectLiteral(node, checkMode) {
-            var inDestructuringPattern = ts.isAssignmentTarget(node);
-            // Grammar checking
-            checkGrammarObjectLiteralExpression(node, inDestructuringPattern);
-            var allPropertiesTable = strictNullChecks ? ts.createSymbolTable() : undefined;
-            var propertiesTable = ts.createSymbolTable();
-            var propertiesArray = [];
-            var spread = emptyObjectType;
-            var contextualType = getApparentTypeOfContextualType(node);
-            var contextualTypeHasPattern = contextualType && contextualType.pattern &&
-                (contextualType.pattern.kind === 189 /* ObjectBindingPattern */ || contextualType.pattern.kind === 193 /* ObjectLiteralExpression */);
-            var inConstContext = isConstContext(node);
-            var checkFlags = inConstContext ? 8 /* Readonly */ : 0;
-            var isInJavascript = ts.isInJSFile(node) && !ts.isInJsonFile(node);
-            var enumTag = ts.getJSDocEnumTag(node);
-            var isJSObjectLiteral = !contextualType && isInJavascript && !enumTag;
-            var objectFlags = freshObjectLiteralFlag;
-            var patternWithComputedProperties = false;
-            var hasComputedStringProperty = false;
-            var hasComputedNumberProperty = false;
-            // Spreads may cause an early bail; ensure computed names are always checked (this is cached)
-            // As otherwise they may not be checked until exports for the type at this position are retrieved,
-            // which may never occur.
-            for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
-                var elem = _a[_i];
-                if (elem.name && ts.isComputedPropertyName(elem.name) && !ts.isWellKnownSymbolSyntactically(elem.name)) {
-                    checkComputedPropertyName(elem.name);
+            function getSingleTypeVariableFromIntersectionTypes(types) {
+                var typeVariable;
+                for (var _i = 0, types_15 = types; _i < types_15.length; _i++) {
+                    var type = types_15[_i];
+                    var t = type.flags & 2097152 /* Intersection */ && ts.find(type.types, function (t) { return !!getInferenceInfoForType(t); });
+                    if (!t || typeVariable && t !== typeVariable) {
+                        return undefined;
+                    }
+                    typeVariable = t;
                 }
+                return typeVariable;
             }
-            var offset = 0;
-            for (var i = 0; i < node.properties.length; i++) {
-                var memberDecl = node.properties[i];
-                var member = getSymbolOfNode(memberDecl);
-                var computedNameType = memberDecl.name && memberDecl.name.kind === 154 /* ComputedPropertyName */ && !ts.isWellKnownSymbolSyntactically(memberDecl.name.expression) ?
-                    checkComputedPropertyName(memberDecl.name) : undefined;
-                if (memberDecl.kind === 281 /* PropertyAssignment */ ||
-                    memberDecl.kind === 282 /* ShorthandPropertyAssignment */ ||
-                    ts.isObjectLiteralMethod(memberDecl)) {
-                    var type = memberDecl.kind === 281 /* PropertyAssignment */ ? checkPropertyAssignment(memberDecl, checkMode) :
-                        memberDecl.kind === 282 /* ShorthandPropertyAssignment */ ? checkExpressionForMutableLocation(memberDecl.name, checkMode) :
-                            checkObjectLiteralMethod(memberDecl, checkMode);
-                    if (isInJavascript) {
-                        var jsDocType = getTypeForDeclarationFromJSDocComment(memberDecl);
-                        if (jsDocType) {
-                            checkTypeAssignableTo(type, jsDocType, memberDecl);
-                            type = jsDocType;
+            function inferToMultipleTypes(source, targets, targetFlags) {
+                var typeVariableCount = 0;
+                if (targetFlags & 1048576 /* Union */) {
+                    var nakedTypeVariable = void 0;
+                    var sources = source.flags & 1048576 /* Union */ ? source.types : [source];
+                    var matched_1 = new Array(sources.length);
+                    var inferenceCircularity = false;
+                    // First infer to types that are not naked type variables. For each source type we
+                    // track whether inferences were made from that particular type to some target with
+                    // equal priority (i.e. of equal quality) to what we would infer for a naked type
+                    // parameter.
+                    for (var _i = 0, targets_2 = targets; _i < targets_2.length; _i++) {
+                        var t = targets_2[_i];
+                        if (getInferenceInfoForType(t)) {
+                            nakedTypeVariable = t;
+                            typeVariableCount++;
                         }
-                        else if (enumTag && enumTag.typeExpression) {
-                            checkTypeAssignableTo(type, getTypeFromTypeNode(enumTag.typeExpression), memberDecl);
+                        else {
+                            for (var i = 0; i < sources.length; i++) {
+                                var saveInferencePriority = inferencePriority;
+                                inferencePriority = 1024 /* MaxValue */;
+                                inferFromTypes(sources[i], t);
+                                if (inferencePriority === priority)
+                                    matched_1[i] = true;
+                                inferenceCircularity = inferenceCircularity || inferencePriority === -1 /* Circularity */;
+                                inferencePriority = Math.min(inferencePriority, saveInferencePriority);
+                            }
                         }
                     }
-                    objectFlags |= ts.getObjectFlags(type) & 3670016 /* PropagatingFlags */;
-                    var nameType = computedNameType && isTypeUsableAsPropertyName(computedNameType) ? computedNameType : undefined;
-                    var prop = nameType ?
-                        createSymbol(4 /* Property */ | member.flags, getPropertyNameFromType(nameType), checkFlags | 4096 /* Late */) :
-                        createSymbol(4 /* Property */ | member.flags, member.escapedName, checkFlags);
-                    if (nameType) {
-                        prop.nameType = nameType;
+                    if (typeVariableCount === 0) {
+                        // If every target is an intersection of types containing a single naked type variable,
+                        // make a lower priority inference to that type variable. This handles inferring from
+                        // 'A | B' to 'T & (X | Y)' where we want to infer 'A | B' for T.
+                        var intersectionTypeVariable = getSingleTypeVariableFromIntersectionTypes(targets);
+                        if (intersectionTypeVariable) {
+                            inferWithPriority(source, intersectionTypeVariable, 1 /* NakedTypeVariable */);
+                        }
+                        return;
                     }
-                    if (inDestructuringPattern) {
-                        // If object literal is an assignment pattern and if the assignment pattern specifies a default value
-                        // for the property, make the property optional.
-                        var isOptional = (memberDecl.kind === 281 /* PropertyAssignment */ && hasDefaultValue(memberDecl.initializer)) ||
-                            (memberDecl.kind === 282 /* ShorthandPropertyAssignment */ && memberDecl.objectAssignmentInitializer);
-                        if (isOptional) {
-                            prop.flags |= 16777216 /* Optional */;
+                    // If the target has a single naked type variable and no inference circularities were
+                    // encountered above (meaning we explored the types fully), create a union of the source
+                    // types from which no inferences have been made so far and infer from that union to the
+                    // naked type variable.
+                    if (typeVariableCount === 1 && !inferenceCircularity) {
+                        var unmatched = ts.flatMap(sources, function (s, i) { return matched_1[i] ? undefined : s; });
+                        if (unmatched.length) {
+                            inferFromTypes(getUnionType(unmatched), nakedTypeVariable);
+                            return;
                         }
                     }
-                    else if (contextualTypeHasPattern && !(ts.getObjectFlags(contextualType) & 512 /* ObjectLiteralPatternWithComputedProperties */)) {
-                        // If object literal is contextually typed by the implied type of a binding pattern, and if the
-                        // binding pattern specifies a default value for the property, make the property optional.
-                        var impliedProp = getPropertyOfType(contextualType, member.escapedName);
-                        if (impliedProp) {
-                            prop.flags |= impliedProp.flags & 16777216 /* Optional */;
+                }
+                else {
+                    // We infer from types that are not naked type variables first so that inferences we
+                    // make from nested naked type variables and given slightly higher priority by virtue
+                    // of being first in the candidates array.
+                    for (var _a = 0, targets_3 = targets; _a < targets_3.length; _a++) {
+                        var t = targets_3[_a];
+                        if (getInferenceInfoForType(t)) {
+                            typeVariableCount++;
                         }
-                        else if (!compilerOptions.suppressExcessPropertyErrors && !getIndexInfoOfType(contextualType, 0 /* String */)) {
-                            error(memberDecl.name, ts.Diagnostics.Object_literal_may_only_specify_known_properties_and_0_does_not_exist_in_type_1, symbolToString(member), typeToString(contextualType));
+                        else {
+                            inferFromTypes(source, t);
                         }
                     }
-                    prop.declarations = member.declarations;
-                    prop.parent = member.parent;
-                    if (member.valueDeclaration) {
-                        prop.valueDeclaration = member.valueDeclaration;
-                    }
-                    prop.type = type;
-                    prop.target = member;
-                    member = prop;
-                    allPropertiesTable === null || allPropertiesTable === void 0 ? void 0 : allPropertiesTable.set(prop.escapedName, prop);
                 }
-                else if (memberDecl.kind === 283 /* SpreadAssignment */) {
-                    if (languageVersion < 2 /* ES2015 */) {
-                        checkExternalEmitHelpers(memberDecl, 2 /* Assign */);
+                // Inferences directly to naked type variables are given lower priority as they are
+                // less specific. For example, when inferring from Promise<string> to T | Promise<T>,
+                // we want to infer string for T, not Promise<string> | string. For intersection types
+                // we only infer to single naked type variables.
+                if (targetFlags & 2097152 /* Intersection */ ? typeVariableCount === 1 : typeVariableCount > 0) {
+                    for (var _b = 0, targets_4 = targets; _b < targets_4.length; _b++) {
+                        var t = targets_4[_b];
+                        if (getInferenceInfoForType(t)) {
+                            inferWithPriority(source, t, 1 /* NakedTypeVariable */);
+                        }
                     }
-                    if (propertiesArray.length > 0) {
-                        spread = getSpreadType(spread, createObjectLiteralType(), node.symbol, objectFlags, inConstContext);
-                        propertiesArray = [];
-                        propertiesTable = ts.createSymbolTable();
-                        hasComputedStringProperty = false;
-                        hasComputedNumberProperty = false;
+                }
+            }
+            function inferToMappedType(source, target, constraintType) {
+                if (constraintType.flags & 1048576 /* Union */) {
+                    var result = false;
+                    for (var _i = 0, _a = constraintType.types; _i < _a.length; _i++) {
+                        var type = _a[_i];
+                        result = inferToMappedType(source, target, type) || result;
                     }
-                    var type = getReducedType(checkExpression(memberDecl.expression));
-                    if (!isValidSpreadType(type)) {
-                        error(memberDecl, ts.Diagnostics.Spread_types_may_only_be_created_from_object_types);
-                        return errorType;
+                    return result;
+                }
+                if (constraintType.flags & 4194304 /* Index */) {
+                    // We're inferring from some source type S to a homomorphic mapped type { [P in keyof T]: X },
+                    // where T is a type variable. Use inferTypeForHomomorphicMappedType to infer a suitable source
+                    // type and then make a secondary inference from that type to T. We make a secondary inference
+                    // such that direct inferences to T get priority over inferences to Partial<T>, for example.
+                    var inference = getInferenceInfoForType(constraintType.type);
+                    if (inference && !inference.isFixed && !isFromInferenceBlockedSource(source)) {
+                        var inferredType = inferTypeForHomomorphicMappedType(source, target, constraintType);
+                        if (inferredType) {
+                            // We assign a lower priority to inferences made from types containing non-inferrable
+                            // types because we may only have a partial result (i.e. we may have failed to make
+                            // reverse inferences for some properties).
+                            inferWithPriority(inferredType, inference.typeParameter, ts.getObjectFlags(source) & 2097152 /* NonInferrableType */ ?
+                                8 /* PartialHomomorphicMappedType */ :
+                                4 /* HomomorphicMappedType */);
+                        }
                     }
-                    if (allPropertiesTable) {
-                        checkSpreadPropOverrides(type, allPropertiesTable, memberDecl);
+                    return true;
+                }
+                if (constraintType.flags & 262144 /* TypeParameter */) {
+                    // We're inferring from some source type S to a mapped type { [P in K]: X }, where K is a type
+                    // parameter. First infer from 'keyof S' to K.
+                    inferWithPriority(getIndexType(source), constraintType, 16 /* MappedTypeConstraint */);
+                    // If K is constrained to a type C, also infer to C. Thus, for a mapped type { [P in K]: X },
+                    // where K extends keyof T, we make the same inferences as for a homomorphic mapped type
+                    // { [P in keyof T]: X }. This enables us to make meaningful inferences when the target is a
+                    // Pick<T, K>.
+                    var extendedConstraint = getConstraintOfType(constraintType);
+                    if (extendedConstraint && inferToMappedType(source, target, extendedConstraint)) {
+                        return true;
                     }
-                    spread = getSpreadType(spread, type, node.symbol, objectFlags, inConstContext);
-                    offset = i + 1;
-                    continue;
+                    // If no inferences can be made to K's constraint, infer from a union of the property types
+                    // in the source to the template type X.
+                    var propTypes = ts.map(getPropertiesOfType(source), getTypeOfSymbol);
+                    var stringIndexType = getIndexTypeOfType(source, 0 /* String */);
+                    var numberIndexInfo = getNonEnumNumberIndexInfo(source);
+                    var numberIndexType = numberIndexInfo && numberIndexInfo.type;
+                    inferFromTypes(getUnionType(ts.append(ts.append(propTypes, stringIndexType), numberIndexType)), getTemplateTypeFromMappedType(target));
+                    return true;
+                }
+                return false;
+            }
+            function inferToConditionalType(source, target) {
+                if (source.flags & 16777216 /* Conditional */) {
+                    inferFromTypes(source.checkType, target.checkType);
+                    inferFromTypes(source.extendsType, target.extendsType);
+                    inferFromTypes(getTrueTypeFromConditionalType(source), getTrueTypeFromConditionalType(target));
+                    inferFromTypes(getFalseTypeFromConditionalType(source), getFalseTypeFromConditionalType(target));
                 }
                 else {
-                    // TypeScript 1.0 spec (April 2014)
-                    // A get accessor declaration is processed in the same manner as
-                    // an ordinary function declaration(section 6.1) with no parameters.
-                    // A set accessor declaration is processed in the same manner
-                    // as an ordinary function declaration with a single parameter and a Void return type.
-                    ts.Debug.assert(memberDecl.kind === 163 /* GetAccessor */ || memberDecl.kind === 164 /* SetAccessor */);
-                    checkNodeDeferred(memberDecl);
+                    var savePriority = priority;
+                    priority |= contravariant ? 32 /* ContravariantConditional */ : 0;
+                    var targetTypes = [getTrueTypeFromConditionalType(target), getFalseTypeFromConditionalType(target)];
+                    inferToMultipleTypes(source, targetTypes, target.flags);
+                    priority = savePriority;
                 }
-                if (computedNameType && !(computedNameType.flags & 8576 /* StringOrNumberLiteralOrUnique */)) {
-                    if (isTypeAssignableTo(computedNameType, stringNumberSymbolType)) {
-                        if (isTypeAssignableTo(computedNameType, numberType)) {
-                            hasComputedNumberProperty = true;
-                        }
-                        else {
-                            hasComputedStringProperty = true;
+            }
+            function inferToTemplateLiteralType(source, target) {
+                var matches = source.flags & 128 /* StringLiteral */ ? inferLiteralsFromTemplateLiteralType(source, target) :
+                    source.flags & 134217728 /* TemplateLiteral */ && ts.arraysEqual(source.texts, target.texts) ? source.types :
+                        undefined;
+                var types = target.types;
+                for (var i = 0; i < types.length; i++) {
+                    inferFromTypes(matches ? matches[i] : neverType, types[i]);
+                }
+            }
+            function inferFromObjectTypes(source, target) {
+                if (ts.getObjectFlags(source) & 4 /* Reference */ && ts.getObjectFlags(target) & 4 /* Reference */ && (source.target === target.target || isArrayType(source) && isArrayType(target))) {
+                    // If source and target are references to the same generic type, infer from type arguments
+                    inferFromTypeArguments(getTypeArguments(source), getTypeArguments(target), getVariances(source.target));
+                    return;
+                }
+                if (isGenericMappedType(source) && isGenericMappedType(target)) {
+                    // The source and target types are generic types { [P in S]: X } and { [P in T]: Y }, so we infer
+                    // from S to T and from X to Y.
+                    inferFromTypes(getConstraintTypeFromMappedType(source), getConstraintTypeFromMappedType(target));
+                    inferFromTypes(getTemplateTypeFromMappedType(source), getTemplateTypeFromMappedType(target));
+                    var sourceNameType = getNameTypeFromMappedType(source);
+                    var targetNameType = getNameTypeFromMappedType(target);
+                    if (sourceNameType && targetNameType)
+                        inferFromTypes(sourceNameType, targetNameType);
+                }
+                if (ts.getObjectFlags(target) & 32 /* Mapped */ && !target.declaration.nameType) {
+                    var constraintType = getConstraintTypeFromMappedType(target);
+                    if (inferToMappedType(source, target, constraintType)) {
+                        return;
+                    }
+                }
+                // Infer from the members of source and target only if the two types are possibly related
+                if (!typesDefinitelyUnrelated(source, target)) {
+                    if (isArrayType(source) || isTupleType(source)) {
+                        if (isTupleType(target)) {
+                            var sourceArity = getTypeReferenceArity(source);
+                            var targetArity = getTypeReferenceArity(target);
+                            var elementTypes = getTypeArguments(target);
+                            var elementFlags = target.target.elementFlags;
+                            // When source and target are tuple types with the same structure (fixed, variadic, and rest are matched
+                            // to the same kind in each position), simply infer between the element types.
+                            if (isTupleType(source) && isTupleTypeStructureMatching(source, target)) {
+                                for (var i = 0; i < targetArity; i++) {
+                                    inferFromTypes(getTypeArguments(source)[i], elementTypes[i]);
+                                }
+                                return;
+                            }
+                            var startLength = isTupleType(source) ? Math.min(source.target.fixedLength, target.target.fixedLength) : 0;
+                            var sourceRestType = !isTupleType(source) || sourceArity > 0 && source.target.elementFlags[sourceArity - 1] & 4 /* Rest */ ?
+                                getTypeArguments(source)[sourceArity - 1] : undefined;
+                            var endLength = !(target.target.combinedFlags & 12 /* Variable */) ? 0 :
+                                sourceRestType ? getEndLengthOfType(target) :
+                                    Math.min(getEndLengthOfType(source), getEndLengthOfType(target));
+                            var sourceEndLength = sourceRestType ? 0 : endLength;
+                            // Infer between starting fixed elements.
+                            for (var i = 0; i < startLength; i++) {
+                                inferFromTypes(getTypeArguments(source)[i], elementTypes[i]);
+                            }
+                            if (sourceRestType && sourceArity - startLength === 1) {
+                                // Single rest element remains in source, infer from that to every element in target
+                                for (var i = startLength; i < targetArity - endLength; i++) {
+                                    inferFromTypes(elementFlags[i] & 8 /* Variadic */ ? createArrayType(sourceRestType) : sourceRestType, elementTypes[i]);
+                                }
+                            }
+                            else {
+                                var middleLength = targetArity - startLength - endLength;
+                                if (middleLength === 2 && elementFlags[startLength] & elementFlags[startLength + 1] & 8 /* Variadic */ && isTupleType(source)) {
+                                    // Middle of target is [...T, ...U] and source is tuple type
+                                    var targetInfo = getInferenceInfoForType(elementTypes[startLength]);
+                                    if (targetInfo && targetInfo.impliedArity !== undefined) {
+                                        // Infer slices from source based on implied arity of T.
+                                        inferFromTypes(sliceTupleType(source, startLength, sourceEndLength + sourceArity - targetInfo.impliedArity), elementTypes[startLength]);
+                                        inferFromTypes(sliceTupleType(source, startLength + targetInfo.impliedArity, sourceEndLength), elementTypes[startLength + 1]);
+                                    }
+                                }
+                                else if (middleLength === 1 && elementFlags[startLength] & 8 /* Variadic */) {
+                                    // Middle of target is exactly one variadic element. Infer the slice between the fixed parts in the source.
+                                    // If target ends in optional element(s), make a lower priority a speculative inference.
+                                    var endsInOptional = target.target.elementFlags[targetArity - 1] & 2 /* Optional */;
+                                    var sourceSlice = isTupleType(source) ? sliceTupleType(source, startLength, sourceEndLength) : createArrayType(sourceRestType);
+                                    inferWithPriority(sourceSlice, elementTypes[startLength], endsInOptional ? 2 /* SpeculativeTuple */ : 0);
+                                }
+                                else if (middleLength === 1 && elementFlags[startLength] & 4 /* Rest */) {
+                                    // Middle of target is exactly one rest element. If middle of source is not empty, infer union of middle element types.
+                                    var restType = isTupleType(source) ? getElementTypeOfSliceOfTupleType(source, startLength, sourceEndLength) : sourceRestType;
+                                    if (restType) {
+                                        inferFromTypes(restType, elementTypes[startLength]);
+                                    }
+                                }
+                            }
+                            // Infer between ending fixed elements
+                            for (var i = 0; i < endLength; i++) {
+                                inferFromTypes(sourceRestType || getTypeArguments(source)[sourceArity - i - 1], elementTypes[targetArity - i - 1]);
+                            }
+                            return;
                         }
-                        if (inDestructuringPattern) {
-                            patternWithComputedProperties = true;
+                        if (isArrayType(target)) {
+                            inferFromIndexTypes(source, target);
+                            return;
                         }
                     }
+                    inferFromProperties(source, target);
+                    inferFromSignatures(source, target, 0 /* Call */);
+                    inferFromSignatures(source, target, 1 /* Construct */);
+                    inferFromIndexTypes(source, target);
                 }
-                else {
-                    propertiesTable.set(member.escapedName, member);
-                }
-                propertiesArray.push(member);
             }
-            // If object literal is contextually typed by the implied type of a binding pattern, augment the result
-            // type with those properties for which the binding pattern specifies a default value.
-            // If the object literal is spread into another object literal, skip this step and let the top-level object
-            // literal handle it instead.
-            if (contextualTypeHasPattern && node.parent.kind !== 283 /* SpreadAssignment */) {
-                for (var _b = 0, _c = getPropertiesOfType(contextualType); _b < _c.length; _b++) {
-                    var prop = _c[_b];
-                    if (!propertiesTable.get(prop.escapedName) && !getPropertyOfType(spread, prop.escapedName)) {
-                        if (!(prop.flags & 16777216 /* Optional */)) {
-                            error(prop.valueDeclaration || prop.bindingElement, ts.Diagnostics.Initializer_provides_no_value_for_this_binding_element_and_the_binding_element_has_no_default_value);
-                        }
-                        propertiesTable.set(prop.escapedName, prop);
-                        propertiesArray.push(prop);
+            function inferFromProperties(source, target) {
+                var properties = getPropertiesOfObjectType(target);
+                for (var _i = 0, properties_3 = properties; _i < properties_3.length; _i++) {
+                    var targetProp = properties_3[_i];
+                    var sourceProp = getPropertyOfType(source, targetProp.escapedName);
+                    if (sourceProp) {
+                        inferFromTypes(getTypeOfSymbol(sourceProp), getTypeOfSymbol(targetProp));
                     }
                 }
             }
-            if (spread !== emptyObjectType) {
-                if (propertiesArray.length > 0) {
-                    spread = getSpreadType(spread, createObjectLiteralType(), node.symbol, objectFlags, inConstContext);
-                    propertiesArray = [];
-                    propertiesTable = ts.createSymbolTable();
-                    hasComputedStringProperty = false;
-                    hasComputedNumberProperty = false;
+            function inferFromSignatures(source, target, kind) {
+                var sourceSignatures = getSignaturesOfType(source, kind);
+                var targetSignatures = getSignaturesOfType(target, kind);
+                var sourceLen = sourceSignatures.length;
+                var targetLen = targetSignatures.length;
+                var len = sourceLen < targetLen ? sourceLen : targetLen;
+                var skipParameters = !!(ts.getObjectFlags(source) & 2097152 /* NonInferrableType */);
+                for (var i = 0; i < len; i++) {
+                    inferFromSignature(getBaseSignature(sourceSignatures[sourceLen - len + i]), getErasedSignature(targetSignatures[targetLen - len + i]), skipParameters);
                 }
-                // remap the raw emptyObjectType fed in at the top into a fresh empty object literal type, unique to this use site
-                return mapType(spread, function (t) { return t === emptyObjectType ? createObjectLiteralType() : t; });
             }
-            return createObjectLiteralType();
-            function createObjectLiteralType() {
-                var stringIndexInfo = hasComputedStringProperty ? getObjectLiteralIndexInfo(node, offset, propertiesArray, 0 /* String */) : undefined;
-                var numberIndexInfo = hasComputedNumberProperty ? getObjectLiteralIndexInfo(node, offset, propertiesArray, 1 /* Number */) : undefined;
-                var result = createAnonymousType(node.symbol, propertiesTable, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
-                result.objectFlags |= objectFlags | 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
-                if (isJSObjectLiteral) {
-                    result.objectFlags |= 16384 /* JSLiteral */;
+            function inferFromSignature(source, target, skipParameters) {
+                if (!skipParameters) {
+                    var saveBivariant = bivariant;
+                    var kind = target.declaration ? target.declaration.kind : 0 /* Unknown */;
+                    // Once we descend into a bivariant signature we remain bivariant for all nested inferences
+                    bivariant = bivariant || kind === 165 /* MethodDeclaration */ || kind === 164 /* MethodSignature */ || kind === 166 /* Constructor */;
+                    applyToParameterTypes(source, target, inferFromContravariantTypes);
+                    bivariant = saveBivariant;
                 }
-                if (patternWithComputedProperties) {
-                    result.objectFlags |= 512 /* ObjectLiteralPatternWithComputedProperties */;
+                applyToReturnTypes(source, target, inferFromTypes);
+            }
+            function inferFromIndexTypes(source, target) {
+                var targetStringIndexType = getIndexTypeOfType(target, 0 /* String */);
+                if (targetStringIndexType) {
+                    var sourceIndexType = getIndexTypeOfType(source, 0 /* String */) ||
+                        getImplicitIndexTypeOfType(source, 0 /* String */);
+                    if (sourceIndexType) {
+                        inferFromTypes(sourceIndexType, targetStringIndexType);
+                    }
                 }
-                if (inDestructuringPattern) {
-                    result.pattern = node;
+                var targetNumberIndexType = getIndexTypeOfType(target, 1 /* Number */);
+                if (targetNumberIndexType) {
+                    var sourceIndexType = getIndexTypeOfType(source, 1 /* Number */) ||
+                        getIndexTypeOfType(source, 0 /* String */) ||
+                        getImplicitIndexTypeOfType(source, 1 /* Number */);
+                    if (sourceIndexType) {
+                        inferFromTypes(sourceIndexType, targetNumberIndexType);
+                    }
                 }
-                return result;
             }
         }
-        function isValidSpreadType(type) {
-            if (type.flags & 63176704 /* Instantiable */) {
-                var constraint = getBaseConstraintOfType(type);
-                if (constraint !== undefined) {
-                    return isValidSpreadType(constraint);
-                }
-            }
-            return !!(type.flags & (1 /* Any */ | 67108864 /* NonPrimitive */ | 524288 /* Object */ | 58982400 /* InstantiableNonPrimitive */) ||
-                getFalsyFlags(type) & 117632 /* DefinitelyFalsy */ && isValidSpreadType(removeDefinitelyFalsyTypes(type)) ||
-                type.flags & 3145728 /* UnionOrIntersection */ && ts.every(type.types, isValidSpreadType));
+        function isTypeOrBaseIdenticalTo(s, t) {
+            return isTypeIdenticalTo(s, t) || !!(t.flags & 4 /* String */ && s.flags & 128 /* StringLiteral */ || t.flags & 8 /* Number */ && s.flags & 256 /* NumberLiteral */);
         }
-        function checkJsxSelfClosingElementDeferred(node) {
-            checkJsxOpeningLikeElementOrOpeningFragment(node);
-            resolveUntypedCall(node); // ensure type arguments and parameters are typechecked, even if there is an arity error
+        function isTypeCloselyMatchedBy(s, t) {
+            return !!(s.flags & 524288 /* Object */ && t.flags & 524288 /* Object */ && s.symbol && s.symbol === t.symbol ||
+                s.aliasSymbol && s.aliasTypeArguments && s.aliasSymbol === t.aliasSymbol);
         }
-        function checkJsxSelfClosingElement(node, _checkMode) {
-            checkNodeDeferred(node);
-            return getJsxElementTypeAt(node) || anyType;
+        function hasPrimitiveConstraint(type) {
+            var constraint = getConstraintOfTypeParameter(type);
+            return !!constraint && maybeTypeOfKind(constraint.flags & 16777216 /* Conditional */ ? getDefaultConstraintOfConditionalType(constraint) : constraint, 131068 /* Primitive */ | 4194304 /* Index */ | 134217728 /* TemplateLiteral */ | 268435456 /* StringMapping */);
         }
-        function checkJsxElementDeferred(node) {
-            // Check attributes
-            checkJsxOpeningLikeElementOrOpeningFragment(node.openingElement);
-            // Perform resolution on the closing tag so that rename/go to definition/etc work
-            if (isJsxIntrinsicIdentifier(node.closingElement.tagName)) {
-                getIntrinsicTagSymbol(node.closingElement);
-            }
-            else {
-                checkExpression(node.closingElement.tagName);
-            }
-            checkJsxChildren(node);
+        function isObjectLiteralType(type) {
+            return !!(ts.getObjectFlags(type) & 128 /* ObjectLiteral */);
         }
-        function checkJsxElement(node, _checkMode) {
-            checkNodeDeferred(node);
-            return getJsxElementTypeAt(node) || anyType;
+        function isObjectOrArrayLiteralType(type) {
+            return !!(ts.getObjectFlags(type) & (128 /* ObjectLiteral */ | 65536 /* ArrayLiteral */));
         }
-        function checkJsxFragment(node) {
-            checkJsxOpeningLikeElementOrOpeningFragment(node.openingFragment);
-            if (compilerOptions.jsx === 2 /* React */ && (compilerOptions.jsxFactory || ts.getSourceFileOfNode(node).pragmas.has("jsx"))) {
-                error(node, compilerOptions.jsxFactory
-                    ? ts.Diagnostics.JSX_fragment_is_not_supported_when_using_jsxFactory
-                    : ts.Diagnostics.JSX_fragment_is_not_supported_when_using_an_inline_JSX_factory_pragma);
+        function unionObjectAndArrayLiteralCandidates(candidates) {
+            if (candidates.length > 1) {
+                var objectLiterals = ts.filter(candidates, isObjectOrArrayLiteralType);
+                if (objectLiterals.length) {
+                    var literalsType = getUnionType(objectLiterals, 2 /* Subtype */);
+                    return ts.concatenate(ts.filter(candidates, function (t) { return !isObjectOrArrayLiteralType(t); }), [literalsType]);
+                }
             }
-            checkJsxChildren(node);
-            return getJsxElementTypeAt(node) || anyType;
-        }
-        /**
-         * Returns true iff the JSX element name would be a valid JS identifier, ignoring restrictions about keywords not being identifiers
-         */
-        function isUnhyphenatedJsxName(name) {
-            // - is the only character supported in JSX attribute names that isn't valid in JavaScript identifiers
-            return !ts.stringContains(name, "-");
+            return candidates;
         }
-        /**
-         * Returns true iff React would emit this tag name as a string rather than an identifier or qualified name
-         */
-        function isJsxIntrinsicIdentifier(tagName) {
-            return tagName.kind === 75 /* Identifier */ && ts.isIntrinsicJsxName(tagName.escapedText);
+        function getContravariantInference(inference) {
+            return inference.priority & 208 /* PriorityImpliesCombination */ ? getIntersectionType(inference.contraCandidates) : getCommonSubtype(inference.contraCandidates);
         }
-        function checkJsxAttribute(node, checkMode) {
-            return node.initializer
-                ? checkExpressionForMutableLocation(node.initializer, checkMode)
-                : trueType; // <Elem attr /> is sugar for <Elem attr={true} />
+        function getCovariantInference(inference, signature) {
+            // Extract all object and array literal types and replace them with a single widened and normalized type.
+            var candidates = unionObjectAndArrayLiteralCandidates(inference.candidates);
+            // We widen inferred literal types if
+            // all inferences were made to top-level occurrences of the type parameter, and
+            // the type parameter has no constraint or its constraint includes no primitive or literal types, and
+            // the type parameter was fixed during inference or does not occur at top-level in the return type.
+            var primitiveConstraint = hasPrimitiveConstraint(inference.typeParameter);
+            var widenLiteralTypes = !primitiveConstraint && inference.topLevel &&
+                (inference.isFixed || !isTypeParameterAtTopLevel(getReturnTypeOfSignature(signature), inference.typeParameter));
+            var baseCandidates = primitiveConstraint ? ts.sameMap(candidates, getRegularTypeOfLiteralType) :
+                widenLiteralTypes ? ts.sameMap(candidates, getWidenedLiteralType) :
+                    candidates;
+            // If all inferences were made from a position that implies a combined result, infer a union type.
+            // Otherwise, infer a common supertype.
+            var unwidenedType = inference.priority & 208 /* PriorityImpliesCombination */ ?
+                getUnionType(baseCandidates, 2 /* Subtype */) :
+                getCommonSupertype(baseCandidates);
+            return getWidenedType(unwidenedType);
         }
-        /**
-         * Get attributes type of the JSX opening-like element. The result is from resolving "attributes" property of the opening-like element.
-         *
-         * @param openingLikeElement a JSX opening-like element
-         * @param filter a function to remove attributes that will not participate in checking whether attributes are assignable
-         * @return an anonymous type (similar to the one returned by checkObjectLiteral) in which its properties are attributes property.
-         * @remarks Because this function calls getSpreadType, it needs to use the same checks as checkObjectLiteral,
-         * which also calls getSpreadType.
-         */
-        function createJsxAttributesTypeFromAttributesProperty(openingLikeElement, checkMode) {
-            var attributes = openingLikeElement.attributes;
-            var allAttributesTable = strictNullChecks ? ts.createSymbolTable() : undefined;
-            var attributesTable = ts.createSymbolTable();
-            var spread = emptyJsxObjectType;
-            var hasSpreadAnyType = false;
-            var typeToIntersect;
-            var explicitlySpecifyChildrenAttribute = false;
-            var objectFlags = 4096 /* JsxAttributes */;
-            var jsxChildrenPropertyName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(openingLikeElement));
-            for (var _i = 0, _a = attributes.properties; _i < _a.length; _i++) {
-                var attributeDecl = _a[_i];
-                var member = attributeDecl.symbol;
-                if (ts.isJsxAttribute(attributeDecl)) {
-                    var exprType = checkJsxAttribute(attributeDecl, checkMode);
-                    objectFlags |= ts.getObjectFlags(exprType) & 3670016 /* PropagatingFlags */;
-                    var attributeSymbol = createSymbol(4 /* Property */ | 33554432 /* Transient */ | member.flags, member.escapedName);
-                    attributeSymbol.declarations = member.declarations;
-                    attributeSymbol.parent = member.parent;
-                    if (member.valueDeclaration) {
-                        attributeSymbol.valueDeclaration = member.valueDeclaration;
-                    }
-                    attributeSymbol.type = exprType;
-                    attributeSymbol.target = member;
-                    attributesTable.set(attributeSymbol.escapedName, attributeSymbol);
-                    allAttributesTable === null || allAttributesTable === void 0 ? void 0 : allAttributesTable.set(attributeSymbol.escapedName, attributeSymbol);
-                    if (attributeDecl.name.escapedText === jsxChildrenPropertyName) {
-                        explicitlySpecifyChildrenAttribute = true;
-                    }
-                }
-                else {
-                    ts.Debug.assert(attributeDecl.kind === 275 /* JsxSpreadAttribute */);
-                    if (attributesTable.size > 0) {
-                        spread = getSpreadType(spread, createJsxAttributesType(), attributes.symbol, objectFlags, /*readonly*/ false);
-                        attributesTable = ts.createSymbolTable();
+        function getInferredType(context, index) {
+            var inference = context.inferences[index];
+            if (!inference.inferredType) {
+                var inferredType = void 0;
+                var signature = context.signature;
+                if (signature) {
+                    var inferredCovariantType = inference.candidates ? getCovariantInference(inference, signature) : undefined;
+                    if (inference.contraCandidates) {
+                        var inferredContravariantType = getContravariantInference(inference);
+                        // If we have both co- and contra-variant inferences, we prefer the contra-variant inference
+                        // unless the co-variant inference is a subtype and not 'never'.
+                        inferredType = inferredCovariantType && !(inferredCovariantType.flags & 131072 /* Never */) &&
+                            isTypeSubtypeOf(inferredCovariantType, inferredContravariantType) ?
+                            inferredCovariantType : inferredContravariantType;
                     }
-                    var exprType = getReducedType(checkExpressionCached(attributeDecl.expression, checkMode));
-                    if (isTypeAny(exprType)) {
-                        hasSpreadAnyType = true;
+                    else if (inferredCovariantType) {
+                        inferredType = inferredCovariantType;
                     }
-                    if (isValidSpreadType(exprType)) {
-                        spread = getSpreadType(spread, exprType, attributes.symbol, objectFlags, /*readonly*/ false);
-                        if (allAttributesTable) {
-                            checkSpreadPropOverrides(exprType, allAttributesTable, attributeDecl);
-                        }
+                    else if (context.flags & 1 /* NoDefault */) {
+                        // We use silentNeverType as the wildcard that signals no inferences.
+                        inferredType = silentNeverType;
                     }
                     else {
-                        typeToIntersect = typeToIntersect ? getIntersectionType([typeToIntersect, exprType]) : exprType;
+                        // Infer either the default or the empty object type when no inferences were
+                        // made. It is important to remember that in this case, inference still
+                        // succeeds, meaning there is no error for not having inference candidates. An
+                        // inference error only occurs when there are *conflicting* candidates, i.e.
+                        // candidates with no common supertype.
+                        var defaultType = getDefaultFromTypeParameter(inference.typeParameter);
+                        if (defaultType) {
+                            // Instantiate the default type. Any forward reference to a type
+                            // parameter should be instantiated to the empty object type.
+                            inferredType = instantiateType(defaultType, mergeTypeMappers(createBackreferenceMapper(context, index), context.nonFixingMapper));
+                        }
                     }
                 }
-            }
-            if (!hasSpreadAnyType) {
-                if (attributesTable.size > 0) {
-                    spread = getSpreadType(spread, createJsxAttributesType(), attributes.symbol, objectFlags, /*readonly*/ false);
+                else {
+                    inferredType = getTypeFromInference(inference);
                 }
-            }
-            // Handle children attribute
-            var parent = openingLikeElement.parent.kind === 266 /* JsxElement */ ? openingLikeElement.parent : undefined;
-            // We have to check that openingElement of the parent is the one we are visiting as this may not be true for selfClosingElement
-            if (parent && parent.openingElement === openingLikeElement && parent.children.length > 0) {
-                var childrenTypes = checkJsxChildren(parent, checkMode);
-                if (!hasSpreadAnyType && jsxChildrenPropertyName && jsxChildrenPropertyName !== "") {
-                    // Error if there is a attribute named "children" explicitly specified and children element.
-                    // This is because children element will overwrite the value from attributes.
-                    // Note: we will not warn "children" attribute overwritten if "children" attribute is specified in object spread.
-                    if (explicitlySpecifyChildrenAttribute) {
-                        error(attributes, ts.Diagnostics._0_are_specified_twice_The_attribute_named_0_will_be_overwritten, ts.unescapeLeadingUnderscores(jsxChildrenPropertyName));
+                inference.inferredType = inferredType || getDefaultTypeArgumentType(!!(context.flags & 2 /* AnyDefault */));
+                var constraint = getConstraintOfTypeParameter(inference.typeParameter);
+                if (constraint) {
+                    var instantiatedConstraint = instantiateType(constraint, context.nonFixingMapper);
+                    if (!inferredType || !context.compareTypes(inferredType, getTypeWithThisArgument(instantiatedConstraint, inferredType))) {
+                        inference.inferredType = inferredType = instantiatedConstraint;
                     }
-                    var contextualType = getApparentTypeOfContextualType(openingLikeElement.attributes);
-                    var childrenContextualType = contextualType && getTypeOfPropertyOfContextualType(contextualType, jsxChildrenPropertyName);
-                    // If there are children in the body of JSX element, create dummy attribute "children" with the union of children types so that it will pass the attribute checking process
-                    var childrenPropSymbol = createSymbol(4 /* Property */ | 33554432 /* Transient */, jsxChildrenPropertyName);
-                    childrenPropSymbol.type = childrenTypes.length === 1 ?
-                        childrenTypes[0] :
-                        (getArrayLiteralTupleTypeIfApplicable(childrenTypes, childrenContextualType, /*hasRestElement*/ false) || createArrayType(getUnionType(childrenTypes)));
-                    // Fake up a property declaration for the children
-                    childrenPropSymbol.valueDeclaration = ts.createPropertySignature(/*modifiers*/ undefined, ts.unescapeLeadingUnderscores(jsxChildrenPropertyName), /*questionToken*/ undefined, /*type*/ undefined, /*initializer*/ undefined);
-                    childrenPropSymbol.valueDeclaration.parent = attributes;
-                    childrenPropSymbol.valueDeclaration.symbol = childrenPropSymbol;
-                    var childPropMap = ts.createSymbolTable();
-                    childPropMap.set(jsxChildrenPropertyName, childrenPropSymbol);
-                    spread = getSpreadType(spread, createAnonymousType(attributes.symbol, childPropMap, ts.emptyArray, ts.emptyArray, /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined), attributes.symbol, objectFlags, /*readonly*/ false);
                 }
             }
-            if (hasSpreadAnyType) {
-                return anyType;
+            return inference.inferredType;
+        }
+        function getDefaultTypeArgumentType(isInJavaScriptFile) {
+            return isInJavaScriptFile ? anyType : unknownType;
+        }
+        function getInferredTypes(context) {
+            var result = [];
+            for (var i = 0; i < context.inferences.length; i++) {
+                result.push(getInferredType(context, i));
             }
-            if (typeToIntersect && spread !== emptyJsxObjectType) {
-                return getIntersectionType([typeToIntersect, spread]);
+            return result;
+        }
+        // EXPRESSION TYPE CHECKING
+        function getCannotFindNameDiagnosticForName(node) {
+            switch (node.escapedText) {
+                case "document":
+                case "console":
+                    return ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_include_dom;
+                case "$":
+                    return compilerOptions.types
+                        ? ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slashjquery_and_then_add_jquery_to_the_types_field_in_your_tsconfig
+                        : ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_jQuery_Try_npm_i_save_dev_types_Slashjquery;
+                case "describe":
+                case "suite":
+                case "it":
+                case "test":
+                    return compilerOptions.types
+                        ? ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_save_dev_types_Slashjest_or_npm_i_save_dev_types_Slashmocha_and_then_add_jest_or_mocha_to_the_types_field_in_your_tsconfig
+                        : ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_a_test_runner_Try_npm_i_save_dev_types_Slashjest_or_npm_i_save_dev_types_Slashmocha;
+                case "process":
+                case "require":
+                case "Buffer":
+                case "module":
+                    return compilerOptions.types
+                        ? ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_save_dev_types_Slashnode_and_then_add_node_to_the_types_field_in_your_tsconfig
+                        : ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_install_type_definitions_for_node_Try_npm_i_save_dev_types_Slashnode;
+                case "Map":
+                case "Set":
+                case "Promise":
+                case "Symbol":
+                case "WeakMap":
+                case "WeakSet":
+                case "Iterator":
+                case "AsyncIterator":
+                case "SharedArrayBuffer":
+                case "Atomics":
+                case "AsyncIterable":
+                case "AsyncIterableIterator":
+                case "AsyncGenerator":
+                case "AsyncGeneratorFunction":
+                case "BigInt":
+                case "Reflect":
+                case "BigInt64Array":
+                case "BigUint64Array":
+                    return ts.Diagnostics.Cannot_find_name_0_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_1_or_later;
+                default:
+                    if (node.parent.kind === 289 /* ShorthandPropertyAssignment */) {
+                        return ts.Diagnostics.No_value_exists_in_scope_for_the_shorthand_property_0_Either_declare_one_or_provide_an_initializer;
+                    }
+                    else {
+                        return ts.Diagnostics.Cannot_find_name_0;
+                    }
             }
-            return typeToIntersect || (spread === emptyJsxObjectType ? createJsxAttributesType() : spread);
-            /**
-             * Create anonymous type from given attributes symbol table.
-             * @param symbol a symbol of JsxAttributes containing attributes corresponding to attributesTable
-             * @param attributesTable a symbol table of attributes property
-             */
-            function createJsxAttributesType() {
-                objectFlags |= freshObjectLiteralFlag;
-                var result = createAnonymousType(attributes.symbol, attributesTable, ts.emptyArray, ts.emptyArray, /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined);
-                result.objectFlags |= objectFlags | 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
-                return result;
+        }
+        function getResolvedSymbol(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedSymbol) {
+                links.resolvedSymbol = !ts.nodeIsMissing(node) &&
+                    resolveName(node, node.escapedText, 111551 /* Value */ | 1048576 /* ExportValue */, getCannotFindNameDiagnosticForName(node), node, !ts.isWriteOnlyAccess(node), 
+                    /*excludeGlobals*/ false, ts.Diagnostics.Cannot_find_name_0_Did_you_mean_1) || unknownSymbol;
             }
+            return links.resolvedSymbol;
         }
-        function checkJsxChildren(node, checkMode) {
-            var childrenTypes = [];
-            for (var _i = 0, _a = node.children; _i < _a.length; _i++) {
-                var child = _a[_i];
-                // In React, JSX text that contains only whitespaces will be ignored so we don't want to type-check that
-                // because then type of children property will have constituent of string type.
-                if (child.kind === 11 /* JsxText */) {
-                    if (!child.containsOnlyTriviaWhiteSpaces) {
-                        childrenTypes.push(stringType);
+        function isInTypeQuery(node) {
+            // TypeScript 1.0 spec (April 2014): 3.6.3
+            // A type query consists of the keyword typeof followed by an expression.
+            // The expression is restricted to a single identifier or a sequence of identifiers separated by periods
+            return !!ts.findAncestor(node, function (n) { return n.kind === 176 /* TypeQuery */ ? true : n.kind === 78 /* Identifier */ || n.kind === 157 /* QualifiedName */ ? false : "quit"; });
+        }
+        // Return the flow cache key for a "dotted name" (i.e. a sequence of identifiers
+        // separated by dots). The key consists of the id of the symbol referenced by the
+        // leftmost identifier followed by zero or more property names separated by dots.
+        // The result is undefined if the reference isn't a dotted name. We prefix nodes
+        // occurring in an apparent type position with '@' because the control flow type
+        // of such nodes may be based on the apparent type instead of the declared type.
+        function getFlowCacheKey(node, declaredType, initialType, flowContainer) {
+            switch (node.kind) {
+                case 78 /* Identifier */:
+                    var symbol = getResolvedSymbol(node);
+                    return symbol !== unknownSymbol ? (flowContainer ? getNodeId(flowContainer) : "-1") + "|" + getTypeId(declaredType) + "|" + getTypeId(initialType) + "|" + (isConstraintPosition(node) ? "@" : "") + getSymbolId(symbol) : undefined;
+                case 107 /* ThisKeyword */:
+                    return "0|" + (flowContainer ? getNodeId(flowContainer) : "-1") + "|" + getTypeId(declaredType) + "|" + getTypeId(initialType);
+                case 225 /* NonNullExpression */:
+                case 207 /* ParenthesizedExpression */:
+                    return getFlowCacheKey(node.expression, declaredType, initialType, flowContainer);
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    var propName = getAccessedPropertyName(node);
+                    if (propName !== undefined) {
+                        var key = getFlowCacheKey(node.expression, declaredType, initialType, flowContainer);
+                        return key && key + "." + propName;
                     }
-                }
-                else {
-                    childrenTypes.push(checkExpressionForMutableLocation(child, checkMode));
-                }
             }
-            return childrenTypes;
+            return undefined;
         }
-        function checkSpreadPropOverrides(type, props, spread) {
-            for (var _i = 0, _a = getPropertiesOfType(type); _i < _a.length; _i++) {
-                var right = _a[_i];
-                var left = props.get(right.escapedName);
-                var rightType = getTypeOfSymbol(right);
-                if (left && !maybeTypeOfKind(rightType, 98304 /* Nullable */) && !(maybeTypeOfKind(rightType, 3 /* AnyOrUnknown */) && right.flags & 16777216 /* Optional */)) {
-                    var diagnostic = error(left.valueDeclaration, ts.Diagnostics._0_is_specified_more_than_once_so_this_usage_will_be_overwritten, ts.unescapeLeadingUnderscores(left.escapedName));
-                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(spread, ts.Diagnostics.This_spread_always_overwrites_this_property));
-                }
+        function isMatchingReference(source, target) {
+            switch (target.kind) {
+                case 207 /* ParenthesizedExpression */:
+                case 225 /* NonNullExpression */:
+                    return isMatchingReference(source, target.expression);
+                case 216 /* BinaryExpression */:
+                    return (ts.isAssignmentExpression(target) && isMatchingReference(source, target.left)) ||
+                        (ts.isBinaryExpression(target) && target.operatorToken.kind === 27 /* CommaToken */ && isMatchingReference(source, target.right));
+            }
+            switch (source.kind) {
+                case 78 /* Identifier */:
+                case 79 /* PrivateIdentifier */:
+                    return target.kind === 78 /* Identifier */ && getResolvedSymbol(source) === getResolvedSymbol(target) ||
+                        (target.kind === 249 /* VariableDeclaration */ || target.kind === 198 /* BindingElement */) &&
+                            getExportSymbolOfValueSymbolIfExported(getResolvedSymbol(source)) === getSymbolOfNode(target);
+                case 107 /* ThisKeyword */:
+                    return target.kind === 107 /* ThisKeyword */;
+                case 105 /* SuperKeyword */:
+                    return target.kind === 105 /* SuperKeyword */;
+                case 225 /* NonNullExpression */:
+                case 207 /* ParenthesizedExpression */:
+                    return isMatchingReference(source.expression, target);
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    return ts.isAccessExpression(target) &&
+                        getAccessedPropertyName(source) === getAccessedPropertyName(target) &&
+                        isMatchingReference(source.expression, target.expression);
             }
+            return false;
         }
-        /**
-         * Check attributes property of opening-like element. This function is called during chooseOverload to get call signature of a JSX opening-like element.
-         * (See "checkApplicableSignatureForJsxOpeningLikeElement" for how the function is used)
-         * @param node a JSXAttributes to be resolved of its type
-         */
-        function checkJsxAttributes(node, checkMode) {
-            return createJsxAttributesTypeFromAttributesProperty(node.parent, checkMode);
+        // Given a source x, check if target matches x or is an && operation with an operand that matches x.
+        function containsTruthyCheck(source, target) {
+            return isMatchingReference(source, target) ||
+                (target.kind === 216 /* BinaryExpression */ && target.operatorToken.kind === 55 /* AmpersandAmpersandToken */ &&
+                    (containsTruthyCheck(source, target.left) || containsTruthyCheck(source, target.right)));
         }
-        function getJsxType(name, location) {
-            var namespace = getJsxNamespaceAt(location);
-            var exports = namespace && getExportsOfSymbol(namespace);
-            var typeSymbol = exports && getSymbol(exports, name, 788968 /* Type */);
-            return typeSymbol ? getDeclaredTypeOfSymbol(typeSymbol) : errorType;
+        function getAccessedPropertyName(access) {
+            return access.kind === 201 /* PropertyAccessExpression */ ? access.name.escapedText :
+                ts.isStringOrNumericLiteralLike(access.argumentExpression) ? ts.escapeLeadingUnderscores(access.argumentExpression.text) :
+                    undefined;
         }
-        /**
-         * Looks up an intrinsic tag name and returns a symbol that either points to an intrinsic
-         * property (in which case nodeLinks.jsxFlags will be IntrinsicNamedElement) or an intrinsic
-         * string index signature (in which case nodeLinks.jsxFlags will be IntrinsicIndexedElement).
-         * May also return unknownSymbol if both of these lookups fail.
-         */
-        function getIntrinsicTagSymbol(node) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedSymbol) {
-                var intrinsicElementsType = getJsxType(JsxNames.IntrinsicElements, node);
-                if (intrinsicElementsType !== errorType) {
-                    // Property case
-                    if (!ts.isIdentifier(node.tagName))
-                        return ts.Debug.fail();
-                    var intrinsicProp = getPropertyOfType(intrinsicElementsType, node.tagName.escapedText);
-                    if (intrinsicProp) {
-                        links.jsxFlags |= 1 /* IntrinsicNamedElement */;
-                        return links.resolvedSymbol = intrinsicProp;
-                    }
-                    // Intrinsic string indexer case
-                    var indexSignatureType = getIndexTypeOfType(intrinsicElementsType, 0 /* String */);
-                    if (indexSignatureType) {
-                        links.jsxFlags |= 2 /* IntrinsicIndexedElement */;
-                        return links.resolvedSymbol = intrinsicElementsType.symbol;
-                    }
-                    // Wasn't found
-                    error(node, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.idText(node.tagName), "JSX." + JsxNames.IntrinsicElements);
-                    return links.resolvedSymbol = unknownSymbol;
-                }
-                else {
-                    if (noImplicitAny) {
-                        error(node, ts.Diagnostics.JSX_element_implicitly_has_type_any_because_no_interface_JSX_0_exists, ts.unescapeLeadingUnderscores(JsxNames.IntrinsicElements));
-                    }
-                    return links.resolvedSymbol = unknownSymbol;
+        function containsMatchingReference(source, target) {
+            while (ts.isAccessExpression(source)) {
+                source = source.expression;
+                if (isMatchingReference(source, target)) {
+                    return true;
                 }
             }
-            return links.resolvedSymbol;
+            return false;
         }
-        function getJsxNamespaceAt(location) {
-            var links = location && getNodeLinks(location);
-            if (links && links.jsxNamespace) {
-                return links.jsxNamespace;
+        function optionalChainContainsReference(source, target) {
+            while (ts.isOptionalChain(source)) {
+                source = source.expression;
+                if (isMatchingReference(source, target)) {
+                    return true;
+                }
             }
-            if (!links || links.jsxNamespace !== false) {
-                var namespaceName = getJsxNamespace(location);
-                var resolvedNamespace = resolveName(location, namespaceName, 1920 /* Namespace */, /*diagnosticMessage*/ undefined, namespaceName, /*isUse*/ false);
-                if (resolvedNamespace) {
-                    var candidate = resolveSymbol(getSymbol(getExportsOfSymbol(resolveSymbol(resolvedNamespace)), JsxNames.JSX, 1920 /* Namespace */));
-                    if (candidate) {
-                        if (links) {
-                            links.jsxNamespace = candidate;
-                        }
-                        return candidate;
-                    }
-                    if (links) {
-                        links.jsxNamespace = false;
+            return false;
+        }
+        function isDiscriminantProperty(type, name) {
+            if (type && type.flags & 1048576 /* Union */) {
+                var prop = getUnionOrIntersectionProperty(type, name);
+                if (prop && ts.getCheckFlags(prop) & 2 /* SyntheticProperty */) {
+                    if (prop.isDiscriminantProperty === undefined) {
+                        prop.isDiscriminantProperty =
+                            (prop.checkFlags & 192 /* Discriminant */) === 192 /* Discriminant */ &&
+                                !maybeTypeOfKind(getTypeOfSymbol(prop), 465829888 /* Instantiable */);
                     }
+                    return !!prop.isDiscriminantProperty;
                 }
             }
-            // JSX global fallback
-            return getGlobalSymbol(JsxNames.JSX, 1920 /* Namespace */, /*diagnosticMessage*/ undefined); // TODO: GH#18217
+            return false;
         }
-        /**
-         * Look into JSX namespace and then look for container with matching name as nameOfAttribPropContainer.
-         * Get a single property from that container if existed. Report an error if there are more than one property.
-         *
-         * @param nameOfAttribPropContainer a string of value JsxNames.ElementAttributesPropertyNameContainer or JsxNames.ElementChildrenAttributeNameContainer
-         *          if other string is given or the container doesn't exist, return undefined.
-         */
-        function getNameFromJsxElementAttributesContainer(nameOfAttribPropContainer, jsxNamespace) {
-            // JSX.ElementAttributesProperty | JSX.ElementChildrenAttribute [symbol]
-            var jsxElementAttribPropInterfaceSym = jsxNamespace && getSymbol(jsxNamespace.exports, nameOfAttribPropContainer, 788968 /* Type */);
-            // JSX.ElementAttributesProperty | JSX.ElementChildrenAttribute [type]
-            var jsxElementAttribPropInterfaceType = jsxElementAttribPropInterfaceSym && getDeclaredTypeOfSymbol(jsxElementAttribPropInterfaceSym);
-            // The properties of JSX.ElementAttributesProperty | JSX.ElementChildrenAttribute
-            var propertiesOfJsxElementAttribPropInterface = jsxElementAttribPropInterfaceType && getPropertiesOfType(jsxElementAttribPropInterfaceType);
-            if (propertiesOfJsxElementAttribPropInterface) {
-                // Element Attributes has zero properties, so the element attributes type will be the class instance type
-                if (propertiesOfJsxElementAttribPropInterface.length === 0) {
-                    return "";
-                }
-                // Element Attributes has one property, so the element attributes type will be the type of the corresponding
-                // property of the class instance type
-                else if (propertiesOfJsxElementAttribPropInterface.length === 1) {
-                    return propertiesOfJsxElementAttribPropInterface[0].escapedName;
-                }
-                else if (propertiesOfJsxElementAttribPropInterface.length > 1) {
-                    // More than one property on ElementAttributesProperty is an error
-                    error(jsxElementAttribPropInterfaceSym.declarations[0], ts.Diagnostics.The_global_type_JSX_0_may_not_have_more_than_one_property, ts.unescapeLeadingUnderscores(nameOfAttribPropContainer));
+        function findDiscriminantProperties(sourceProperties, target) {
+            var result;
+            for (var _i = 0, sourceProperties_2 = sourceProperties; _i < sourceProperties_2.length; _i++) {
+                var sourceProperty = sourceProperties_2[_i];
+                if (isDiscriminantProperty(target, sourceProperty.escapedName)) {
+                    if (result) {
+                        result.push(sourceProperty);
+                        continue;
+                    }
+                    result = [sourceProperty];
                 }
             }
-            return undefined;
-        }
-        function getJsxLibraryManagedAttributes(jsxNamespace) {
-            // JSX.LibraryManagedAttributes [symbol]
-            return jsxNamespace && getSymbol(jsxNamespace.exports, JsxNames.LibraryManagedAttributes, 788968 /* Type */);
-        }
-        /// e.g. "props" for React.d.ts,
-        /// or 'undefined' if ElementAttributesProperty doesn't exist (which means all
-        ///     non-intrinsic elements' attributes type is 'any'),
-        /// or '' if it has 0 properties (which means every
-        ///     non-intrinsic elements' attributes type is the element instance type)
-        function getJsxElementPropertiesName(jsxNamespace) {
-            return getNameFromJsxElementAttributesContainer(JsxNames.ElementAttributesPropertyNameContainer, jsxNamespace);
+            return result;
         }
-        function getJsxElementChildrenPropertyName(jsxNamespace) {
-            return getNameFromJsxElementAttributesContainer(JsxNames.ElementChildrenAttributeNameContainer, jsxNamespace);
+        function isOrContainsMatchingReference(source, target) {
+            return isMatchingReference(source, target) || containsMatchingReference(source, target);
         }
-        function getUninstantiatedJsxSignaturesOfType(elementType, caller) {
-            if (elementType.flags & 4 /* String */) {
-                return [anySignature];
-            }
-            else if (elementType.flags & 128 /* StringLiteral */) {
-                var intrinsicType = getIntrinsicAttributesTypeFromStringLiteralType(elementType, caller);
-                if (!intrinsicType) {
-                    error(caller, ts.Diagnostics.Property_0_does_not_exist_on_type_1, elementType.value, "JSX." + JsxNames.IntrinsicElements);
-                    return ts.emptyArray;
-                }
-                else {
-                    var fakeSignature = createSignatureForJSXIntrinsic(caller, intrinsicType);
-                    return [fakeSignature];
+        function hasMatchingArgument(callExpression, reference) {
+            if (callExpression.arguments) {
+                for (var _i = 0, _a = callExpression.arguments; _i < _a.length; _i++) {
+                    var argument = _a[_i];
+                    if (isOrContainsMatchingReference(reference, argument)) {
+                        return true;
+                    }
                 }
             }
-            var apparentElemType = getApparentType(elementType);
-            // Resolve the signatures, preferring constructor
-            var signatures = getSignaturesOfType(apparentElemType, 1 /* Construct */);
-            if (signatures.length === 0) {
-                // No construct signatures, try call signatures
-                signatures = getSignaturesOfType(apparentElemType, 0 /* Call */);
-            }
-            if (signatures.length === 0 && apparentElemType.flags & 1048576 /* Union */) {
-                // If each member has some combination of new/call signatures; make a union signature list for those
-                signatures = getUnionSignatures(ts.map(apparentElemType.types, function (t) { return getUninstantiatedJsxSignaturesOfType(t, caller); }));
+            if (callExpression.expression.kind === 201 /* PropertyAccessExpression */ &&
+                isOrContainsMatchingReference(reference, callExpression.expression.expression)) {
+                return true;
             }
-            return signatures;
+            return false;
         }
-        function getIntrinsicAttributesTypeFromStringLiteralType(type, location) {
-            // If the elemType is a stringLiteral type, we can then provide a check to make sure that the string literal type is one of the Jsx intrinsic element type
-            // For example:
-            //      var CustomTag: "h1" = "h1";
-            //      <CustomTag> Hello World </CustomTag>
-            var intrinsicElementsType = getJsxType(JsxNames.IntrinsicElements, location);
-            if (intrinsicElementsType !== errorType) {
-                var stringLiteralTypeName = type.value;
-                var intrinsicProp = getPropertyOfType(intrinsicElementsType, ts.escapeLeadingUnderscores(stringLiteralTypeName));
-                if (intrinsicProp) {
-                    return getTypeOfSymbol(intrinsicProp);
-                }
-                var indexSignatureType = getIndexTypeOfType(intrinsicElementsType, 0 /* String */);
-                if (indexSignatureType) {
-                    return indexSignatureType;
-                }
-                return undefined;
+        function getFlowNodeId(flow) {
+            if (!flow.id || flow.id < 0) {
+                flow.id = nextFlowId;
+                nextFlowId++;
             }
-            // If we need to report an error, we already done so here. So just return any to prevent any more error downstream
-            return anyType;
+            return flow.id;
         }
-        function checkJsxReturnAssignableToAppropriateBound(refKind, elemInstanceType, openingLikeElement) {
-            if (refKind === 1 /* Function */) {
-                var sfcReturnConstraint = getJsxStatelessElementTypeAt(openingLikeElement);
-                if (sfcReturnConstraint) {
-                    checkTypeRelatedTo(elemInstanceType, sfcReturnConstraint, assignableRelation, openingLikeElement.tagName, ts.Diagnostics.Its_return_type_0_is_not_a_valid_JSX_element, generateInitialErrorChain);
-                }
-            }
-            else if (refKind === 0 /* Component */) {
-                var classConstraint = getJsxElementClassTypeAt(openingLikeElement);
-                if (classConstraint) {
-                    // Issue an error if this return type isn't assignable to JSX.ElementClass, failing that
-                    checkTypeRelatedTo(elemInstanceType, classConstraint, assignableRelation, openingLikeElement.tagName, ts.Diagnostics.Its_instance_type_0_is_not_a_valid_JSX_element, generateInitialErrorChain);
-                }
+        function typeMaybeAssignableTo(source, target) {
+            if (!(source.flags & 1048576 /* Union */)) {
+                return isTypeAssignableTo(source, target);
             }
-            else { // Mixed
-                var sfcReturnConstraint = getJsxStatelessElementTypeAt(openingLikeElement);
-                var classConstraint = getJsxElementClassTypeAt(openingLikeElement);
-                if (!sfcReturnConstraint || !classConstraint) {
-                    return;
+            for (var _i = 0, _a = source.types; _i < _a.length; _i++) {
+                var t = _a[_i];
+                if (isTypeAssignableTo(t, target)) {
+                    return true;
                 }
-                var combined = getUnionType([sfcReturnConstraint, classConstraint]);
-                checkTypeRelatedTo(elemInstanceType, combined, assignableRelation, openingLikeElement.tagName, ts.Diagnostics.Its_element_type_0_is_not_a_valid_JSX_element, generateInitialErrorChain);
-            }
-            function generateInitialErrorChain() {
-                var componentName = ts.getTextOfNode(openingLikeElement.tagName);
-                return ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics._0_cannot_be_used_as_a_JSX_component, componentName);
             }
+            return false;
         }
-        /**
-         * Get attributes type of the given intrinsic opening-like Jsx element by resolving the tag name.
-         * The function is intended to be called from a function which has checked that the opening element is an intrinsic element.
-         * @param node an intrinsic JSX opening-like element
-         */
-        function getIntrinsicAttributesTypeFromJsxOpeningLikeElement(node) {
-            ts.Debug.assert(isJsxIntrinsicIdentifier(node.tagName));
-            var links = getNodeLinks(node);
-            if (!links.resolvedJsxElementAttributesType) {
-                var symbol = getIntrinsicTagSymbol(node);
-                if (links.jsxFlags & 1 /* IntrinsicNamedElement */) {
-                    return links.resolvedJsxElementAttributesType = getTypeOfSymbol(symbol);
+        // Remove those constituent types of declaredType to which no constituent type of assignedType is assignable.
+        // For example, when a variable of type number | string | boolean is assigned a value of type number | boolean,
+        // we remove type string.
+        function getAssignmentReducedType(declaredType, assignedType) {
+            if (declaredType !== assignedType) {
+                if (assignedType.flags & 131072 /* Never */) {
+                    return assignedType;
                 }
-                else if (links.jsxFlags & 2 /* IntrinsicIndexedElement */) {
-                    return links.resolvedJsxElementAttributesType =
-                        getIndexTypeOfType(getDeclaredTypeOfSymbol(symbol), 0 /* String */);
+                var reducedType = filterType(declaredType, function (t) { return typeMaybeAssignableTo(assignedType, t); });
+                if (assignedType.flags & 512 /* BooleanLiteral */ && isFreshLiteralType(assignedType)) {
+                    reducedType = mapType(reducedType, getFreshTypeOfLiteralType); // Ensure that if the assignment is a fresh type, that we narrow to fresh types
                 }
-                else {
-                    return links.resolvedJsxElementAttributesType = errorType;
+                // Our crude heuristic produces an invalid result in some cases: see GH#26130.
+                // For now, when that happens, we give up and don't narrow at all.  (This also
+                // means we'll never narrow for erroneous assignments where the assigned type
+                // is not assignable to the declared type.)
+                if (isTypeAssignableTo(assignedType, reducedType)) {
+                    return reducedType;
                 }
             }
-            return links.resolvedJsxElementAttributesType;
-        }
-        function getJsxElementClassTypeAt(location) {
-            var type = getJsxType(JsxNames.ElementClass, location);
-            if (type === errorType)
-                return undefined;
-            return type;
-        }
-        function getJsxElementTypeAt(location) {
-            return getJsxType(JsxNames.Element, location);
+            return declaredType;
         }
-        function getJsxStatelessElementTypeAt(location) {
-            var jsxElementType = getJsxElementTypeAt(location);
-            if (jsxElementType) {
-                return getUnionType([jsxElementType, nullType]);
+        function getTypeFactsOfTypes(types) {
+            var result = 0 /* None */;
+            for (var _i = 0, types_16 = types; _i < types_16.length; _i++) {
+                var t = types_16[_i];
+                result |= getTypeFacts(t);
             }
+            return result;
         }
-        /**
-         * Returns all the properties of the Jsx.IntrinsicElements interface
-         */
-        function getJsxIntrinsicTagNamesAt(location) {
-            var intrinsics = getJsxType(JsxNames.IntrinsicElements, location);
-            return intrinsics ? getPropertiesOfType(intrinsics) : ts.emptyArray;
+        function isFunctionObjectType(type) {
+            // We do a quick check for a "bind" property before performing the more expensive subtype
+            // check. This gives us a quicker out in the common case where an object type is not a function.
+            var resolved = resolveStructuredTypeMembers(type);
+            return !!(resolved.callSignatures.length || resolved.constructSignatures.length ||
+                resolved.members.get("bind") && isTypeSubtypeOf(type, globalFunctionType));
         }
-        function checkJsxPreconditions(errorNode) {
-            // Preconditions for using JSX
-            if ((compilerOptions.jsx || 0 /* None */) === 0 /* None */) {
-                error(errorNode, ts.Diagnostics.Cannot_use_JSX_unless_the_jsx_flag_is_provided);
+        function getTypeFacts(type) {
+            var flags = type.flags;
+            if (flags & 4 /* String */) {
+                return strictNullChecks ? 16317953 /* StringStrictFacts */ : 16776705 /* StringFacts */;
             }
-            if (getJsxElementTypeAt(errorNode) === undefined) {
-                if (noImplicitAny) {
-                    error(errorNode, ts.Diagnostics.JSX_element_implicitly_has_type_any_because_the_global_type_JSX_Element_does_not_exist);
-                }
+            if (flags & 128 /* StringLiteral */) {
+                var isEmpty = type.value === "";
+                return strictNullChecks ?
+                    isEmpty ? 12123649 /* EmptyStringStrictFacts */ : 7929345 /* NonEmptyStringStrictFacts */ :
+                    isEmpty ? 12582401 /* EmptyStringFacts */ : 16776705 /* NonEmptyStringFacts */;
             }
-        }
-        function checkJsxOpeningLikeElementOrOpeningFragment(node) {
-            var isNodeOpeningLikeElement = ts.isJsxOpeningLikeElement(node);
-            if (isNodeOpeningLikeElement) {
-                checkGrammarJsxElement(node);
+            if (flags & (8 /* Number */ | 32 /* Enum */)) {
+                return strictNullChecks ? 16317698 /* NumberStrictFacts */ : 16776450 /* NumberFacts */;
             }
-            checkJsxPreconditions(node);
-            // The reactNamespace/jsxFactory's root symbol should be marked as 'used' so we don't incorrectly elide its import.
-            // And if there is no reactNamespace/jsxFactory's symbol in scope when targeting React emit, we should issue an error.
-            var reactRefErr = diagnostics && compilerOptions.jsx === 2 /* React */ ? ts.Diagnostics.Cannot_find_name_0 : undefined;
-            var reactNamespace = getJsxNamespace(node);
-            var reactLocation = isNodeOpeningLikeElement ? node.tagName : node;
-            var reactSym = resolveName(reactLocation, reactNamespace, 111551 /* Value */, reactRefErr, reactNamespace, /*isUse*/ true);
-            if (reactSym) {
-                // Mark local symbol as referenced here because it might not have been marked
-                // if jsx emit was not react as there wont be error being emitted
-                reactSym.isReferenced = 67108863 /* All */;
-                // If react symbol is alias, mark it as refereced
-                if (reactSym.flags & 2097152 /* Alias */ && !getTypeOnlyAliasDeclaration(reactSym)) {
-                    markAliasSymbolAsReferenced(reactSym);
-                }
+            if (flags & 256 /* NumberLiteral */) {
+                var isZero = type.value === 0;
+                return strictNullChecks ?
+                    isZero ? 12123394 /* ZeroNumberStrictFacts */ : 7929090 /* NonZeroNumberStrictFacts */ :
+                    isZero ? 12582146 /* ZeroNumberFacts */ : 16776450 /* NonZeroNumberFacts */;
             }
-            if (isNodeOpeningLikeElement) {
-                var jsxOpeningLikeNode = node;
-                var sig = getResolvedSignature(jsxOpeningLikeNode);
-                checkJsxReturnAssignableToAppropriateBound(getJsxReferenceKind(jsxOpeningLikeNode), getReturnTypeOfSignature(sig), jsxOpeningLikeNode);
+            if (flags & 64 /* BigInt */) {
+                return strictNullChecks ? 16317188 /* BigIntStrictFacts */ : 16775940 /* BigIntFacts */;
             }
-        }
-        /**
-         * Check if a property with the given name is known anywhere in the given type. In an object type, a property
-         * is considered known if
-         * 1. the object type is empty and the check is for assignability, or
-         * 2. if the object type has index signatures, or
-         * 3. if the property is actually declared in the object type
-         *    (this means that 'toString', for example, is not usually a known property).
-         * 4. In a union or intersection type,
-         *    a property is considered known if it is known in any constituent type.
-         * @param targetType a type to search a given name in
-         * @param name a property name to search
-         * @param isComparingJsxAttributes a boolean flag indicating whether we are searching in JsxAttributesType
-         */
-        function isKnownProperty(targetType, name, isComparingJsxAttributes) {
-            if (targetType.flags & 524288 /* Object */) {
-                var resolved = resolveStructuredTypeMembers(targetType);
-                if (resolved.stringIndexInfo ||
-                    resolved.numberIndexInfo && isNumericLiteralName(name) ||
-                    getPropertyOfObjectType(targetType, name) ||
-                    isComparingJsxAttributes && !isUnhyphenatedJsxName(name)) {
-                    // For JSXAttributes, if the attribute has a hyphenated name, consider that the attribute to be known.
-                    return true;
-                }
+            if (flags & 2048 /* BigIntLiteral */) {
+                var isZero = isZeroBigInt(type);
+                return strictNullChecks ?
+                    isZero ? 12122884 /* ZeroBigIntStrictFacts */ : 7928580 /* NonZeroBigIntStrictFacts */ :
+                    isZero ? 12581636 /* ZeroBigIntFacts */ : 16775940 /* NonZeroBigIntFacts */;
             }
-            else if (targetType.flags & 3145728 /* UnionOrIntersection */ && isExcessPropertyCheckTarget(targetType)) {
-                for (var _i = 0, _a = targetType.types; _i < _a.length; _i++) {
-                    var t = _a[_i];
-                    if (isKnownProperty(t, name, isComparingJsxAttributes)) {
-                        return true;
-                    }
-                }
+            if (flags & 16 /* Boolean */) {
+                return strictNullChecks ? 16316168 /* BooleanStrictFacts */ : 16774920 /* BooleanFacts */;
             }
-            return false;
+            if (flags & 528 /* BooleanLike */) {
+                return strictNullChecks ?
+                    (type === falseType || type === regularFalseType) ? 12121864 /* FalseStrictFacts */ : 7927560 /* TrueStrictFacts */ :
+                    (type === falseType || type === regularFalseType) ? 12580616 /* FalseFacts */ : 16774920 /* TrueFacts */;
+            }
+            if (flags & 524288 /* Object */) {
+                return ts.getObjectFlags(type) & 16 /* Anonymous */ && isEmptyObjectType(type) ?
+                    strictNullChecks ? 16318463 /* EmptyObjectStrictFacts */ : 16777215 /* EmptyObjectFacts */ :
+                    isFunctionObjectType(type) ?
+                        strictNullChecks ? 7880640 /* FunctionStrictFacts */ : 16728000 /* FunctionFacts */ :
+                        strictNullChecks ? 7888800 /* ObjectStrictFacts */ : 16736160 /* ObjectFacts */;
+            }
+            if (flags & (16384 /* Void */ | 32768 /* Undefined */)) {
+                return 9830144 /* UndefinedFacts */;
+            }
+            if (flags & 65536 /* Null */) {
+                return 9363232 /* NullFacts */;
+            }
+            if (flags & 12288 /* ESSymbolLike */) {
+                return strictNullChecks ? 7925520 /* SymbolStrictFacts */ : 16772880 /* SymbolFacts */;
+            }
+            if (flags & 67108864 /* NonPrimitive */) {
+                return strictNullChecks ? 7888800 /* ObjectStrictFacts */ : 16736160 /* ObjectFacts */;
+            }
+            if (flags & 131072 /* Never */) {
+                return 0 /* None */;
+            }
+            if (flags & 465829888 /* Instantiable */) {
+                return getTypeFacts(getBaseConstraintOfType(type) || unknownType);
+            }
+            if (flags & 3145728 /* UnionOrIntersection */) {
+                return getTypeFactsOfTypes(type.types);
+            }
+            return 16777215 /* All */;
         }
-        function isExcessPropertyCheckTarget(type) {
-            return !!(type.flags & 524288 /* Object */ && !(ts.getObjectFlags(type) & 512 /* ObjectLiteralPatternWithComputedProperties */) ||
-                type.flags & 67108864 /* NonPrimitive */ ||
-                type.flags & 1048576 /* Union */ && ts.some(type.types, isExcessPropertyCheckTarget) ||
-                type.flags & 2097152 /* Intersection */ && ts.every(type.types, isExcessPropertyCheckTarget));
+        function getTypeWithFacts(type, include) {
+            return filterType(type, function (t) { return (getTypeFacts(t) & include) !== 0; });
         }
-        function checkJsxExpression(node, checkMode) {
-            checkGrammarJsxExpression(node);
-            if (node.expression) {
-                var type = checkExpression(node.expression, checkMode);
-                if (node.dotDotDotToken && type !== anyType && !isArrayType(type)) {
-                    error(node, ts.Diagnostics.JSX_spread_child_must_be_an_array_type);
-                }
-                return type;
+        function getTypeWithDefault(type, defaultExpression) {
+            if (defaultExpression) {
+                var defaultType = getTypeOfExpression(defaultExpression);
+                return getUnionType([getTypeWithFacts(type, 524288 /* NEUndefined */), defaultType]);
             }
-            else {
+            return type;
+        }
+        function getTypeOfDestructuredProperty(type, name) {
+            var nameType = getLiteralTypeFromPropertyName(name);
+            if (!isTypeUsableAsPropertyName(nameType))
                 return errorType;
+            var text = getPropertyNameFromType(nameType);
+            return getConstraintForLocation(getTypeOfPropertyOfType(type, text), name) ||
+                isNumericLiteralName(text) && includeUndefinedInIndexSignature(getIndexTypeOfType(type, 1 /* Number */)) ||
+                includeUndefinedInIndexSignature(getIndexTypeOfType(type, 0 /* String */)) ||
+                errorType;
+        }
+        function getTypeOfDestructuredArrayElement(type, index) {
+            return everyType(type, isTupleLikeType) && getTupleElementType(type, index) ||
+                includeUndefinedInIndexSignature(checkIteratedTypeOrElementType(65 /* Destructuring */, type, undefinedType, /*errorNode*/ undefined)) ||
+                errorType;
+        }
+        function includeUndefinedInIndexSignature(type) {
+            if (!type)
+                return type;
+            return compilerOptions.noUncheckedIndexedAccess ?
+                getUnionType([type, undefinedType]) :
+                type;
+        }
+        function getTypeOfDestructuredSpreadExpression(type) {
+            return createArrayType(checkIteratedTypeOrElementType(65 /* Destructuring */, type, undefinedType, /*errorNode*/ undefined) || errorType);
+        }
+        function getAssignedTypeOfBinaryExpression(node) {
+            var isDestructuringDefaultAssignment = node.parent.kind === 199 /* ArrayLiteralExpression */ && isDestructuringAssignmentTarget(node.parent) ||
+                node.parent.kind === 288 /* PropertyAssignment */ && isDestructuringAssignmentTarget(node.parent.parent);
+            return isDestructuringDefaultAssignment ?
+                getTypeWithDefault(getAssignedType(node), node.right) :
+                getTypeOfExpression(node.right);
+        }
+        function isDestructuringAssignmentTarget(parent) {
+            return parent.parent.kind === 216 /* BinaryExpression */ && parent.parent.left === parent ||
+                parent.parent.kind === 239 /* ForOfStatement */ && parent.parent.initializer === parent;
+        }
+        function getAssignedTypeOfArrayLiteralElement(node, element) {
+            return getTypeOfDestructuredArrayElement(getAssignedType(node), node.elements.indexOf(element));
+        }
+        function getAssignedTypeOfSpreadExpression(node) {
+            return getTypeOfDestructuredSpreadExpression(getAssignedType(node.parent));
+        }
+        function getAssignedTypeOfPropertyAssignment(node) {
+            return getTypeOfDestructuredProperty(getAssignedType(node.parent), node.name);
+        }
+        function getAssignedTypeOfShorthandPropertyAssignment(node) {
+            return getTypeWithDefault(getAssignedTypeOfPropertyAssignment(node), node.objectAssignmentInitializer);
+        }
+        function getAssignedType(node) {
+            var parent = node.parent;
+            switch (parent.kind) {
+                case 238 /* ForInStatement */:
+                    return stringType;
+                case 239 /* ForOfStatement */:
+                    return checkRightHandSideOfForOf(parent) || errorType;
+                case 216 /* BinaryExpression */:
+                    return getAssignedTypeOfBinaryExpression(parent);
+                case 210 /* DeleteExpression */:
+                    return undefinedType;
+                case 199 /* ArrayLiteralExpression */:
+                    return getAssignedTypeOfArrayLiteralElement(parent, node);
+                case 220 /* SpreadElement */:
+                    return getAssignedTypeOfSpreadExpression(parent);
+                case 288 /* PropertyAssignment */:
+                    return getAssignedTypeOfPropertyAssignment(parent);
+                case 289 /* ShorthandPropertyAssignment */:
+                    return getAssignedTypeOfShorthandPropertyAssignment(parent);
             }
+            return errorType;
         }
-        function getDeclarationNodeFlagsFromSymbol(s) {
-            return s.valueDeclaration ? ts.getCombinedNodeFlags(s.valueDeclaration) : 0;
+        function getInitialTypeOfBindingElement(node) {
+            var pattern = node.parent;
+            var parentType = getInitialType(pattern.parent);
+            var type = pattern.kind === 196 /* ObjectBindingPattern */ ?
+                getTypeOfDestructuredProperty(parentType, node.propertyName || node.name) :
+                !node.dotDotDotToken ?
+                    getTypeOfDestructuredArrayElement(parentType, pattern.elements.indexOf(node)) :
+                    getTypeOfDestructuredSpreadExpression(parentType);
+            return getTypeWithDefault(type, node.initializer);
         }
-        /**
-         * Return whether this symbol is a member of a prototype somewhere
-         * Note that this is not tracked well within the compiler, so the answer may be incorrect.
-         */
-        function isPrototypeProperty(symbol) {
-            if (symbol.flags & 8192 /* Method */ || ts.getCheckFlags(symbol) & 4 /* SyntheticMethod */) {
-                return true;
+        function getTypeOfInitializer(node) {
+            // Return the cached type if one is available. If the type of the variable was inferred
+            // from its initializer, we'll already have cached the type. Otherwise we compute it now
+            // without caching such that transient types are reflected.
+            var links = getNodeLinks(node);
+            return links.resolvedType || getTypeOfExpression(node);
+        }
+        function getInitialTypeOfVariableDeclaration(node) {
+            if (node.initializer) {
+                return getTypeOfInitializer(node.initializer);
             }
-            if (ts.isInJSFile(symbol.valueDeclaration)) {
-                var parent = symbol.valueDeclaration.parent;
-                return parent && ts.isBinaryExpression(parent) &&
-                    ts.getAssignmentDeclarationKind(parent) === 3 /* PrototypeProperty */;
+            if (node.parent.parent.kind === 238 /* ForInStatement */) {
+                return stringType;
+            }
+            if (node.parent.parent.kind === 239 /* ForOfStatement */) {
+                return checkRightHandSideOfForOf(node.parent.parent) || errorType;
             }
+            return errorType;
         }
-        /**
-         * Check whether the requested property access is valid.
-         * Returns true if node is a valid property access, and false otherwise.
-         * @param node The node to be checked.
-         * @param isSuper True if the access is from `super.`.
-         * @param type The type of the object whose property is being accessed. (Not the type of the property.)
-         * @param prop The symbol for the property being accessed.
-         */
-        function checkPropertyAccessibility(node, isSuper, type, prop) {
-            var flags = ts.getDeclarationModifierFlagsFromSymbol(prop);
-            var errorNode = node.kind === 153 /* QualifiedName */ ? node.right : node.kind === 188 /* ImportType */ ? node : node.name;
-            if (isSuper) {
-                // TS 1.0 spec (April 2014): 4.8.2
-                // - In a constructor, instance member function, instance member accessor, or
-                //   instance member variable initializer where this references a derived class instance,
-                //   a super property access is permitted and must specify a public instance member function of the base class.
-                // - In a static member function or static member accessor
-                //   where this references the constructor function object of a derived class,
-                //   a super property access is permitted and must specify a public static member function of the base class.
-                if (languageVersion < 2 /* ES2015 */) {
-                    if (symbolHasNonMethodDeclaration(prop)) {
-                        error(errorNode, ts.Diagnostics.Only_public_and_protected_methods_of_the_base_class_are_accessible_via_the_super_keyword);
-                        return false;
+        function getInitialType(node) {
+            return node.kind === 249 /* VariableDeclaration */ ?
+                getInitialTypeOfVariableDeclaration(node) :
+                getInitialTypeOfBindingElement(node);
+        }
+        function isEmptyArrayAssignment(node) {
+            return node.kind === 249 /* VariableDeclaration */ && node.initializer &&
+                isEmptyArrayLiteral(node.initializer) ||
+                node.kind !== 198 /* BindingElement */ && node.parent.kind === 216 /* BinaryExpression */ &&
+                    isEmptyArrayLiteral(node.parent.right);
+        }
+        function getReferenceCandidate(node) {
+            switch (node.kind) {
+                case 207 /* ParenthesizedExpression */:
+                    return getReferenceCandidate(node.expression);
+                case 216 /* BinaryExpression */:
+                    switch (node.operatorToken.kind) {
+                        case 62 /* EqualsToken */:
+                        case 74 /* BarBarEqualsToken */:
+                        case 75 /* AmpersandAmpersandEqualsToken */:
+                        case 76 /* QuestionQuestionEqualsToken */:
+                            return getReferenceCandidate(node.left);
+                        case 27 /* CommaToken */:
+                            return getReferenceCandidate(node.right);
                     }
-                }
-                if (flags & 128 /* Abstract */) {
-                    // A method cannot be accessed in a super property access if the method is abstract.
-                    // This error could mask a private property access error. But, a member
-                    // cannot simultaneously be private and abstract, so this will trigger an
-                    // additional error elsewhere.
-                    error(errorNode, ts.Diagnostics.Abstract_method_0_in_class_1_cannot_be_accessed_via_super_expression, symbolToString(prop), typeToString(getDeclaringClass(prop)));
-                    return false;
-                }
             }
-            // Referencing abstract properties within their own constructors is not allowed
-            if ((flags & 128 /* Abstract */) && ts.isThisProperty(node) && symbolHasNonMethodDeclaration(prop)) {
-                var declaringClassDeclaration = ts.getClassLikeDeclarationOfSymbol(getParentOfSymbol(prop));
-                if (declaringClassDeclaration && isNodeUsedDuringClassInitialization(node)) {
-                    error(errorNode, ts.Diagnostics.Abstract_property_0_in_class_1_cannot_be_accessed_in_the_constructor, symbolToString(prop), ts.getTextOfIdentifierOrLiteral(declaringClassDeclaration.name)); // TODO: GH#18217
-                    return false;
+            return node;
+        }
+        function getReferenceRoot(node) {
+            var parent = node.parent;
+            return parent.kind === 207 /* ParenthesizedExpression */ ||
+                parent.kind === 216 /* BinaryExpression */ && parent.operatorToken.kind === 62 /* EqualsToken */ && parent.left === node ||
+                parent.kind === 216 /* BinaryExpression */ && parent.operatorToken.kind === 27 /* CommaToken */ && parent.right === node ?
+                getReferenceRoot(parent) : node;
+        }
+        function getTypeOfSwitchClause(clause) {
+            if (clause.kind === 284 /* CaseClause */) {
+                return getRegularTypeOfLiteralType(getTypeOfExpression(clause.expression));
+            }
+            return neverType;
+        }
+        function getSwitchClauseTypes(switchStatement) {
+            var links = getNodeLinks(switchStatement);
+            if (!links.switchTypes) {
+                links.switchTypes = [];
+                for (var _i = 0, _a = switchStatement.caseBlock.clauses; _i < _a.length; _i++) {
+                    var clause = _a[_i];
+                    links.switchTypes.push(getTypeOfSwitchClause(clause));
                 }
             }
-            if (ts.isPropertyAccessExpression(node) && ts.isPrivateIdentifier(node.name)) {
-                if (!ts.getContainingClass(node)) {
-                    error(errorNode, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
-                    return false;
+            return links.switchTypes;
+        }
+        function getSwitchClauseTypeOfWitnesses(switchStatement, retainDefault) {
+            var witnesses = [];
+            for (var _i = 0, _a = switchStatement.caseBlock.clauses; _i < _a.length; _i++) {
+                var clause = _a[_i];
+                if (clause.kind === 284 /* CaseClause */) {
+                    if (ts.isStringLiteralLike(clause.expression)) {
+                        witnesses.push(clause.expression.text);
+                        continue;
+                    }
+                    return ts.emptyArray;
                 }
-                return true;
-            }
-            // Public properties are otherwise accessible.
-            if (!(flags & 24 /* NonPublicAccessibilityModifier */)) {
-                return true;
+                if (retainDefault)
+                    witnesses.push(/*explicitDefaultStatement*/ undefined);
             }
-            // Property is known to be private or protected at this point
-            // Private property is accessible if the property is within the declaring class
-            if (flags & 8 /* Private */) {
-                var declaringClassDeclaration = ts.getClassLikeDeclarationOfSymbol(getParentOfSymbol(prop));
-                if (!isNodeWithinClass(node, declaringClassDeclaration)) {
-                    error(errorNode, ts.Diagnostics.Property_0_is_private_and_only_accessible_within_class_1, symbolToString(prop), typeToString(getDeclaringClass(prop)));
-                    return false;
+            return witnesses;
+        }
+        function eachTypeContainedIn(source, types) {
+            return source.flags & 1048576 /* Union */ ? !ts.forEach(source.types, function (t) { return !ts.contains(types, t); }) : ts.contains(types, source);
+        }
+        function isTypeSubsetOf(source, target) {
+            return source === target || target.flags & 1048576 /* Union */ && isTypeSubsetOfUnion(source, target);
+        }
+        function isTypeSubsetOfUnion(source, target) {
+            if (source.flags & 1048576 /* Union */) {
+                for (var _i = 0, _a = source.types; _i < _a.length; _i++) {
+                    var t = _a[_i];
+                    if (!containsType(target.types, t)) {
+                        return false;
+                    }
                 }
                 return true;
             }
-            // Property is known to be protected at this point
-            // All protected properties of a supertype are accessible in a super access
-            if (isSuper) {
+            if (source.flags & 1024 /* EnumLiteral */ && getBaseTypeOfEnumLiteralType(source) === target) {
                 return true;
             }
-            // Find the first enclosing class that has the declaring classes of the protected constituents
-            // of the property as base classes
-            var enclosingClass = forEachEnclosingClass(node, function (enclosingDeclaration) {
-                var enclosingClass = getDeclaredTypeOfSymbol(getSymbolOfNode(enclosingDeclaration));
-                return isClassDerivedFromDeclaringClasses(enclosingClass, prop) ? enclosingClass : undefined;
-            });
-            // A protected property is accessible if the property is within the declaring class or classes derived from it
-            if (!enclosingClass) {
-                // allow PropertyAccessibility if context is in function with this parameter
-                // static member access is disallow
-                var thisParameter = void 0;
-                if (flags & 32 /* Static */ || !(thisParameter = getThisParameterFromNodeContext(node)) || !thisParameter.type) {
-                    error(errorNode, ts.Diagnostics.Property_0_is_protected_and_only_accessible_within_class_1_and_its_subclasses, symbolToString(prop), typeToString(getDeclaringClass(prop) || type));
-                    return false;
-                }
-                var thisType = getTypeFromTypeNode(thisParameter.type);
-                enclosingClass = ((thisType.flags & 262144 /* TypeParameter */) ? getConstraintOfTypeParameter(thisType) : thisType).target;
+            return containsType(target.types, source);
+        }
+        function forEachType(type, f) {
+            return type.flags & 1048576 /* Union */ ? ts.forEach(type.types, f) : f(type);
+        }
+        function everyType(type, f) {
+            return type.flags & 1048576 /* Union */ ? ts.every(type.types, f) : f(type);
+        }
+        function filterType(type, f) {
+            if (type.flags & 1048576 /* Union */) {
+                var types = type.types;
+                var filtered = ts.filter(types, f);
+                return filtered === types ? type : getUnionTypeFromSortedList(filtered, type.objectFlags);
             }
-            // No further restrictions for static properties
-            if (flags & 32 /* Static */) {
-                return true;
+            return type.flags & 131072 /* Never */ || f(type) ? type : neverType;
+        }
+        function countTypes(type) {
+            return type.flags & 1048576 /* Union */ ? type.types.length : 1;
+        }
+        function mapType(type, mapper, noReductions) {
+            if (type.flags & 131072 /* Never */) {
+                return type;
             }
-            if (type.flags & 262144 /* TypeParameter */) {
-                // get the original type -- represented as the type constraint of the 'this' type
-                type = type.isThisType ? getConstraintOfTypeParameter(type) : getBaseConstraintOfType(type); // TODO: GH#18217 Use a different variable that's allowed to be undefined
+            if (!(type.flags & 1048576 /* Union */)) {
+                return mapper(type);
             }
-            if (!type || !hasBaseType(type, enclosingClass)) {
-                error(errorNode, ts.Diagnostics.Property_0_is_protected_and_only_accessible_through_an_instance_of_class_1, symbolToString(prop), typeToString(enclosingClass));
-                return false;
+            var mappedTypes;
+            for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
+                var t = _a[_i];
+                var mapped = mapper(t);
+                if (mapped) {
+                    if (!mappedTypes) {
+                        mappedTypes = [mapped];
+                    }
+                    else {
+                        mappedTypes.push(mapped);
+                    }
+                }
             }
-            return true;
-        }
-        function getThisParameterFromNodeContext(node) {
-            var thisContainer = ts.getThisContainer(node, /* includeArrowFunctions */ false);
-            return thisContainer && ts.isFunctionLike(thisContainer) ? ts.getThisParameter(thisContainer) : undefined;
-        }
-        function symbolHasNonMethodDeclaration(symbol) {
-            return !!forEachProperty(symbol, function (prop) { return !(prop.flags & 8192 /* Method */); });
+            return mappedTypes && getUnionType(mappedTypes, noReductions ? 0 /* None */ : 1 /* Literal */);
         }
-        function checkNonNullExpression(node) {
-            return checkNonNullType(checkExpression(node), node);
+        function extractTypesOfKind(type, kind) {
+            return filterType(type, function (t) { return (t.flags & kind) !== 0; });
         }
-        function isNullableType(type) {
-            return !!((strictNullChecks ? getFalsyFlags(type) : type.flags) & 98304 /* Nullable */);
+        // Return a new type in which occurrences of the string and number primitive types in
+        // typeWithPrimitives have been replaced with occurrences of string literals and numeric
+        // literals in typeWithLiterals, respectively.
+        function replacePrimitivesWithLiterals(typeWithPrimitives, typeWithLiterals) {
+            if (isTypeSubsetOf(stringType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, 128 /* StringLiteral */) ||
+                isTypeSubsetOf(numberType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, 256 /* NumberLiteral */) ||
+                isTypeSubsetOf(bigintType, typeWithPrimitives) && maybeTypeOfKind(typeWithLiterals, 2048 /* BigIntLiteral */)) {
+                return mapType(typeWithPrimitives, function (t) {
+                    return t.flags & 4 /* String */ ? extractTypesOfKind(typeWithLiterals, 4 /* String */ | 128 /* StringLiteral */) :
+                        t.flags & 8 /* Number */ ? extractTypesOfKind(typeWithLiterals, 8 /* Number */ | 256 /* NumberLiteral */) :
+                            t.flags & 64 /* BigInt */ ? extractTypesOfKind(typeWithLiterals, 64 /* BigInt */ | 2048 /* BigIntLiteral */) : t;
+                });
+            }
+            return typeWithPrimitives;
         }
-        function getNonNullableTypeIfNeeded(type) {
-            return isNullableType(type) ? getNonNullableType(type) : type;
+        function isIncomplete(flowType) {
+            return flowType.flags === 0;
         }
-        function reportObjectPossiblyNullOrUndefinedError(node, flags) {
-            error(node, flags & 32768 /* Undefined */ ? flags & 65536 /* Null */ ?
-                ts.Diagnostics.Object_is_possibly_null_or_undefined :
-                ts.Diagnostics.Object_is_possibly_undefined :
-                ts.Diagnostics.Object_is_possibly_null);
+        function getTypeFromFlowType(flowType) {
+            return flowType.flags === 0 ? flowType.type : flowType;
         }
-        function reportCannotInvokePossiblyNullOrUndefinedError(node, flags) {
-            error(node, flags & 32768 /* Undefined */ ? flags & 65536 /* Null */ ?
-                ts.Diagnostics.Cannot_invoke_an_object_which_is_possibly_null_or_undefined :
-                ts.Diagnostics.Cannot_invoke_an_object_which_is_possibly_undefined :
-                ts.Diagnostics.Cannot_invoke_an_object_which_is_possibly_null);
+        function createFlowType(type, incomplete) {
+            return incomplete ? { flags: 0, type: type.flags & 131072 /* Never */ ? silentNeverType : type } : type;
         }
-        function checkNonNullTypeWithReporter(type, node, reportError) {
-            if (strictNullChecks && type.flags & 2 /* Unknown */) {
-                error(node, ts.Diagnostics.Object_is_of_type_unknown);
-                return errorType;
-            }
-            var kind = (strictNullChecks ? getFalsyFlags(type) : type.flags) & 98304 /* Nullable */;
-            if (kind) {
-                reportError(node, kind);
-                var t = getNonNullableType(type);
-                return t.flags & (98304 /* Nullable */ | 131072 /* Never */) ? errorType : t;
-            }
-            return type;
+        // An evolving array type tracks the element types that have so far been seen in an
+        // 'x.push(value)' or 'x[n] = value' operation along the control flow graph. Evolving
+        // array types are ultimately converted into manifest array types (using getFinalArrayType)
+        // and never escape the getFlowTypeOfReference function.
+        function createEvolvingArrayType(elementType) {
+            var result = createObjectType(256 /* EvolvingArray */);
+            result.elementType = elementType;
+            return result;
         }
-        function checkNonNullType(type, node) {
-            return checkNonNullTypeWithReporter(type, node, reportObjectPossiblyNullOrUndefinedError);
+        function getEvolvingArrayType(elementType) {
+            return evolvingArrayTypes[elementType.id] || (evolvingArrayTypes[elementType.id] = createEvolvingArrayType(elementType));
         }
-        function checkNonNullNonVoidType(type, node) {
-            var nonNullType = checkNonNullType(type, node);
-            if (nonNullType !== errorType && nonNullType.flags & 16384 /* Void */) {
-                error(node, ts.Diagnostics.Object_is_possibly_undefined);
-            }
-            return nonNullType;
+        // When adding evolving array element types we do not perform subtype reduction. Instead,
+        // we defer subtype reduction until the evolving array type is finalized into a manifest
+        // array type.
+        function addEvolvingArrayElementType(evolvingArrayType, node) {
+            var elementType = getRegularTypeOfObjectLiteral(getBaseTypeOfLiteralType(getContextFreeTypeOfExpression(node)));
+            return isTypeSubsetOf(elementType, evolvingArrayType.elementType) ? evolvingArrayType : getEvolvingArrayType(getUnionType([evolvingArrayType.elementType, elementType]));
         }
-        function checkPropertyAccessExpression(node) {
-            return node.flags & 32 /* OptionalChain */ ? checkPropertyAccessChain(node) :
-                checkPropertyAccessExpressionOrQualifiedName(node, node.expression, checkNonNullExpression(node.expression), node.name);
+        function createFinalArrayType(elementType) {
+            return elementType.flags & 131072 /* Never */ ?
+                autoArrayType :
+                createArrayType(elementType.flags & 1048576 /* Union */ ?
+                    getUnionType(elementType.types, 2 /* Subtype */) :
+                    elementType);
         }
-        function checkPropertyAccessChain(node) {
-            var leftType = checkExpression(node.expression);
-            var nonOptionalType = getOptionalExpressionType(leftType, node.expression);
-            return propagateOptionalTypeMarker(checkPropertyAccessExpressionOrQualifiedName(node, node.expression, checkNonNullType(nonOptionalType, node.expression), node.name), node, nonOptionalType !== leftType);
+        // We perform subtype reduction upon obtaining the final array type from an evolving array type.
+        function getFinalArrayType(evolvingArrayType) {
+            return evolvingArrayType.finalArrayType || (evolvingArrayType.finalArrayType = createFinalArrayType(evolvingArrayType.elementType));
         }
-        function checkQualifiedName(node) {
-            return checkPropertyAccessExpressionOrQualifiedName(node, node.left, checkNonNullExpression(node.left), node.right);
+        function finalizeEvolvingArrayType(type) {
+            return ts.getObjectFlags(type) & 256 /* EvolvingArray */ ? getFinalArrayType(type) : type;
         }
-        function isMethodAccessForCall(node) {
-            while (node.parent.kind === 200 /* ParenthesizedExpression */) {
-                node = node.parent;
-            }
-            return ts.isCallOrNewExpression(node.parent) && node.parent.expression === node;
+        function getElementTypeOfEvolvingArrayType(type) {
+            return ts.getObjectFlags(type) & 256 /* EvolvingArray */ ? type.elementType : neverType;
         }
-        // Lookup the private identifier lexically.
-        function lookupSymbolForPrivateIdentifierDeclaration(propName, location) {
-            for (var containingClass = ts.getContainingClass(location); !!containingClass; containingClass = ts.getContainingClass(containingClass)) {
-                var symbol = containingClass.symbol;
-                var name = ts.getSymbolNameForPrivateIdentifier(symbol, propName);
-                var prop = (symbol.members && symbol.members.get(name)) || (symbol.exports && symbol.exports.get(name));
-                if (prop) {
-                    return prop;
+        function isEvolvingArrayTypeList(types) {
+            var hasEvolvingArrayType = false;
+            for (var _i = 0, types_17 = types; _i < types_17.length; _i++) {
+                var t = types_17[_i];
+                if (!(t.flags & 131072 /* Never */)) {
+                    if (!(ts.getObjectFlags(t) & 256 /* EvolvingArray */)) {
+                        return false;
+                    }
+                    hasEvolvingArrayType = true;
                 }
             }
+            return hasEvolvingArrayType;
         }
-        function getPrivateIdentifierPropertyOfType(leftType, lexicallyScopedIdentifier) {
-            return getPropertyOfType(leftType, lexicallyScopedIdentifier.escapedName);
+        // At flow control branch or loop junctions, if the type along every antecedent code path
+        // is an evolving array type, we construct a combined evolving array type. Otherwise we
+        // finalize all evolving array types.
+        function getUnionOrEvolvingArrayType(types, subtypeReduction) {
+            return isEvolvingArrayTypeList(types) ?
+                getEvolvingArrayType(getUnionType(ts.map(types, getElementTypeOfEvolvingArrayType))) :
+                getUnionType(ts.sameMap(types, finalizeEvolvingArrayType), subtypeReduction);
         }
-        function checkPrivateIdentifierPropertyAccess(leftType, right, lexicallyScopedIdentifier) {
-            // Either the identifier could not be looked up in the lexical scope OR the lexically scoped identifier did not exist on the type.
-            // Find a private identifier with the same description on the type.
-            var propertyOnType;
-            var properties = getPropertiesOfType(leftType);
-            if (properties) {
-                ts.forEach(properties, function (symbol) {
-                    var decl = symbol.valueDeclaration;
-                    if (decl && ts.isNamedDeclaration(decl) && ts.isPrivateIdentifier(decl.name) && decl.name.escapedText === right.escapedText) {
-                        propertyOnType = symbol;
-                        return true;
-                    }
-                });
+        // Return true if the given node is 'x' in an 'x.length', x.push(value)', 'x.unshift(value)' or
+        // 'x[n] = value' operation, where 'n' is an expression of type any, undefined, or a number-like type.
+        function isEvolvingArrayOperationTarget(node) {
+            var root = getReferenceRoot(node);
+            var parent = root.parent;
+            var isLengthPushOrUnshift = ts.isPropertyAccessExpression(parent) && (parent.name.escapedText === "length" ||
+                parent.parent.kind === 203 /* CallExpression */
+                    && ts.isIdentifier(parent.name)
+                    && ts.isPushOrUnshiftIdentifier(parent.name));
+            var isElementAssignment = parent.kind === 202 /* ElementAccessExpression */ &&
+                parent.expression === root &&
+                parent.parent.kind === 216 /* BinaryExpression */ &&
+                parent.parent.operatorToken.kind === 62 /* EqualsToken */ &&
+                parent.parent.left === parent &&
+                !ts.isAssignmentTarget(parent.parent) &&
+                isTypeAssignableToKind(getTypeOfExpression(parent.argumentExpression), 296 /* NumberLike */);
+            return isLengthPushOrUnshift || isElementAssignment;
+        }
+        function isDeclarationWithExplicitTypeAnnotation(declaration) {
+            return (declaration.kind === 249 /* VariableDeclaration */ || declaration.kind === 160 /* Parameter */ ||
+                declaration.kind === 163 /* PropertyDeclaration */ || declaration.kind === 162 /* PropertySignature */) &&
+                !!ts.getEffectiveTypeAnnotationNode(declaration);
+        }
+        function getExplicitTypeOfSymbol(symbol, diagnostic) {
+            if (symbol.flags & (16 /* Function */ | 8192 /* Method */ | 32 /* Class */ | 512 /* ValueModule */)) {
+                return getTypeOfSymbol(symbol);
             }
-            var diagName = diagnosticName(right);
-            if (propertyOnType) {
-                var typeValueDecl = propertyOnType.valueDeclaration;
-                var typeClass_1 = ts.getContainingClass(typeValueDecl);
-                ts.Debug.assert(!!typeClass_1);
-                // We found a private identifier property with the same description.
-                // Either:
-                // - There is a lexically scoped private identifier AND it shadows the one we found on the type.
-                // - It is an attempt to access the private identifier outside of the class.
-                if (lexicallyScopedIdentifier) {
-                    var lexicalValueDecl = lexicallyScopedIdentifier.valueDeclaration;
-                    var lexicalClass = ts.getContainingClass(lexicalValueDecl);
-                    ts.Debug.assert(!!lexicalClass);
-                    if (ts.findAncestor(lexicalClass, function (n) { return typeClass_1 === n; })) {
-                        var diagnostic = error(right, ts.Diagnostics.The_property_0_cannot_be_accessed_on_type_1_within_this_class_because_it_is_shadowed_by_another_private_identifier_with_the_same_spelling, diagName, typeToString(leftType));
-                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(lexicalValueDecl, ts.Diagnostics.The_shadowing_declaration_of_0_is_defined_here, diagName), ts.createDiagnosticForNode(typeValueDecl, ts.Diagnostics.The_declaration_of_0_that_you_probably_intended_to_use_is_defined_here, diagName));
-                        return true;
+            if (symbol.flags & (3 /* Variable */ | 4 /* Property */)) {
+                if (ts.getCheckFlags(symbol) & 262144 /* Mapped */) {
+                    var origin = symbol.syntheticOrigin;
+                    if (origin && getExplicitTypeOfSymbol(origin)) {
+                        return getTypeOfSymbol(symbol);
                     }
                 }
-                error(right, ts.Diagnostics.Property_0_is_not_accessible_outside_class_1_because_it_has_a_private_identifier, diagName, diagnosticName(typeClass_1.name || anon));
-                return true;
-            }
-            return false;
-        }
-        function checkPropertyAccessExpressionOrQualifiedName(node, left, leftType, right) {
-            var parentSymbol = getNodeLinks(left).resolvedSymbol;
-            var assignmentKind = ts.getAssignmentTargetKind(node);
-            var apparentType = getApparentType(assignmentKind !== 0 /* None */ || isMethodAccessForCall(node) ? getWidenedType(leftType) : leftType);
-            if (ts.isPrivateIdentifier(right)) {
-                checkExternalEmitHelpers(node, 262144 /* ClassPrivateFieldGet */);
-            }
-            var isAnyLike = isTypeAny(apparentType) || apparentType === silentNeverType;
-            var prop;
-            if (ts.isPrivateIdentifier(right)) {
-                var lexicallyScopedSymbol = lookupSymbolForPrivateIdentifierDeclaration(right.escapedText, right);
-                if (isAnyLike) {
-                    if (lexicallyScopedSymbol) {
-                        return apparentType;
+                var declaration = symbol.valueDeclaration;
+                if (declaration) {
+                    if (isDeclarationWithExplicitTypeAnnotation(declaration)) {
+                        return getTypeOfSymbol(symbol);
                     }
-                    if (!ts.getContainingClass(right)) {
-                        grammarErrorOnNode(right, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
-                        return anyType;
+                    if (ts.isVariableDeclaration(declaration) && declaration.parent.parent.kind === 239 /* ForOfStatement */) {
+                        var statement = declaration.parent.parent;
+                        var expressionType = getTypeOfDottedName(statement.expression, /*diagnostic*/ undefined);
+                        if (expressionType) {
+                            var use = statement.awaitModifier ? 15 /* ForAwaitOf */ : 13 /* ForOf */;
+                            return checkIteratedTypeOrElementType(use, expressionType, undefinedType, /*errorNode*/ undefined);
+                        }
+                    }
+                    if (diagnostic) {
+                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(declaration, ts.Diagnostics._0_needs_an_explicit_type_annotation, symbolToString(symbol)));
                     }
                 }
-                prop = lexicallyScopedSymbol ? getPrivateIdentifierPropertyOfType(leftType, lexicallyScopedSymbol) : undefined;
-                // Check for private-identifier-specific shadowing and lexical-scoping errors.
-                if (!prop && checkPrivateIdentifierPropertyAccess(leftType, right, lexicallyScopedSymbol)) {
-                    return errorType;
+            }
+        }
+        // We require the dotted function name in an assertion expression to be comprised of identifiers
+        // that reference function, method, class or value module symbols; or variable, property or
+        // parameter symbols with declarations that have explicit type annotations. Such references are
+        // resolvable with no possibility of triggering circularities in control flow analysis.
+        function getTypeOfDottedName(node, diagnostic) {
+            if (!(node.flags & 16777216 /* InWithStatement */)) {
+                switch (node.kind) {
+                    case 78 /* Identifier */:
+                        var symbol = getExportSymbolOfValueSymbolIfExported(getResolvedSymbol(node));
+                        return getExplicitTypeOfSymbol(symbol.flags & 2097152 /* Alias */ ? resolveAlias(symbol) : symbol, diagnostic);
+                    case 107 /* ThisKeyword */:
+                        return getExplicitThisType(node);
+                    case 105 /* SuperKeyword */:
+                        return checkSuperExpression(node);
+                    case 201 /* PropertyAccessExpression */:
+                        var type = getTypeOfDottedName(node.expression, diagnostic);
+                        var prop = type && getPropertyOfType(type, node.name.escapedText);
+                        return prop && getExplicitTypeOfSymbol(prop, diagnostic);
+                    case 207 /* ParenthesizedExpression */:
+                        return getTypeOfDottedName(node.expression, diagnostic);
                 }
             }
-            else {
-                if (isAnyLike) {
-                    if (ts.isIdentifier(left) && parentSymbol) {
-                        markAliasReferenced(parentSymbol, node);
+        }
+        function getEffectsSignature(node) {
+            var links = getNodeLinks(node);
+            var signature = links.effectsSignature;
+            if (signature === undefined) {
+                // A call expression parented by an expression statement is a potential assertion. Other call
+                // expressions are potential type predicate function calls. In order to avoid triggering
+                // circularities in control flow analysis, we use getTypeOfDottedName when resolving the call
+                // target expression of an assertion.
+                var funcType = void 0;
+                if (node.parent.kind === 233 /* ExpressionStatement */) {
+                    funcType = getTypeOfDottedName(node.expression, /*diagnostic*/ undefined);
+                }
+                else if (node.expression.kind !== 105 /* SuperKeyword */) {
+                    if (ts.isOptionalChain(node)) {
+                        funcType = checkNonNullType(getOptionalExpressionType(checkExpression(node.expression), node.expression), node.expression);
+                    }
+                    else {
+                        funcType = checkNonNullExpression(node.expression);
                     }
-                    return apparentType;
                 }
-                prop = getPropertyOfType(apparentType, right.escapedText);
+                var signatures = getSignaturesOfType(funcType && getApparentType(funcType) || unknownType, 0 /* Call */);
+                var candidate = signatures.length === 1 && !signatures[0].typeParameters ? signatures[0] :
+                    ts.some(signatures, hasTypePredicateOrNeverReturnType) ? getResolvedSignature(node) :
+                        undefined;
+                signature = links.effectsSignature = candidate && hasTypePredicateOrNeverReturnType(candidate) ? candidate : unknownSignature;
             }
-            if (ts.isIdentifier(left) && parentSymbol && !(prop && isConstEnumOrConstEnumOnlyModule(prop))) {
-                markAliasReferenced(parentSymbol, node);
+            return signature === unknownSignature ? undefined : signature;
+        }
+        function hasTypePredicateOrNeverReturnType(signature) {
+            return !!(getTypePredicateOfSignature(signature) ||
+                signature.declaration && (getReturnTypeFromAnnotation(signature.declaration) || unknownType).flags & 131072 /* Never */);
+        }
+        function getTypePredicateArgument(predicate, callExpression) {
+            if (predicate.kind === 1 /* Identifier */ || predicate.kind === 3 /* AssertsIdentifier */) {
+                return callExpression.arguments[predicate.parameterIndex];
             }
-            var propType;
-            if (!prop) {
-                var indexInfo = !ts.isPrivateIdentifier(right) && (assignmentKind === 0 /* None */ || !isGenericObjectType(leftType) || isThisTypeParameter(leftType)) ? getIndexInfoOfType(apparentType, 0 /* String */) : undefined;
-                if (!(indexInfo && indexInfo.type)) {
-                    if (isJSLiteralType(leftType)) {
-                        return anyType;
+            var invokedExpression = ts.skipParentheses(callExpression.expression);
+            return ts.isAccessExpression(invokedExpression) ? ts.skipParentheses(invokedExpression.expression) : undefined;
+        }
+        function reportFlowControlError(node) {
+            var block = ts.findAncestor(node, ts.isFunctionOrModuleBlock);
+            var sourceFile = ts.getSourceFileOfNode(node);
+            var span = ts.getSpanOfTokenAtPosition(sourceFile, block.statements.pos);
+            diagnostics.add(ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.The_containing_function_or_module_body_is_too_large_for_control_flow_analysis));
+        }
+        function isReachableFlowNode(flow) {
+            var result = isReachableFlowNodeWorker(flow, /*noCacheCheck*/ false);
+            lastFlowNode = flow;
+            lastFlowNodeReachable = result;
+            return result;
+        }
+        function isFalseExpression(expr) {
+            var node = ts.skipParentheses(expr);
+            return node.kind === 94 /* FalseKeyword */ || node.kind === 216 /* BinaryExpression */ && (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */ && (isFalseExpression(node.left) || isFalseExpression(node.right)) ||
+                node.operatorToken.kind === 56 /* BarBarToken */ && isFalseExpression(node.left) && isFalseExpression(node.right));
+        }
+        function isReachableFlowNodeWorker(flow, noCacheCheck) {
+            while (true) {
+                if (flow === lastFlowNode) {
+                    return lastFlowNodeReachable;
+                }
+                var flags = flow.flags;
+                if (flags & 4096 /* Shared */) {
+                    if (!noCacheCheck) {
+                        var id = getFlowNodeId(flow);
+                        var reachable = flowNodeReachable[id];
+                        return reachable !== undefined ? reachable : (flowNodeReachable[id] = isReachableFlowNodeWorker(flow, /*noCacheCheck*/ true));
                     }
-                    if (leftType.symbol === globalThisSymbol) {
-                        if (globalThisSymbol.exports.has(right.escapedText) && (globalThisSymbol.exports.get(right.escapedText).flags & 418 /* BlockScoped */)) {
-                            error(right, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.unescapeLeadingUnderscores(right.escapedText), typeToString(leftType));
+                    noCacheCheck = false;
+                }
+                if (flags & (16 /* Assignment */ | 96 /* Condition */ | 256 /* ArrayMutation */)) {
+                    flow = flow.antecedent;
+                }
+                else if (flags & 512 /* Call */) {
+                    var signature = getEffectsSignature(flow.node);
+                    if (signature) {
+                        var predicate = getTypePredicateOfSignature(signature);
+                        if (predicate && predicate.kind === 3 /* AssertsIdentifier */ && !predicate.type) {
+                            var predicateArgument = flow.node.arguments[predicate.parameterIndex];
+                            if (predicateArgument && isFalseExpression(predicateArgument)) {
+                                return false;
+                            }
                         }
-                        else if (noImplicitAny) {
-                            error(right, ts.Diagnostics.Element_implicitly_has_an_any_type_because_type_0_has_no_index_signature, typeToString(leftType));
+                        if (getReturnTypeOfSignature(signature).flags & 131072 /* Never */) {
+                            return false;
                         }
-                        return anyType;
                     }
-                    if (right.escapedText && !checkAndReportErrorForExtendingInterface(node)) {
-                        reportNonexistentProperty(right, isThisTypeParameter(leftType) ? apparentType : leftType);
+                    flow = flow.antecedent;
+                }
+                else if (flags & 4 /* BranchLabel */) {
+                    // A branching point is reachable if any branch is reachable.
+                    return ts.some(flow.antecedents, function (f) { return isReachableFlowNodeWorker(f, /*noCacheCheck*/ false); });
+                }
+                else if (flags & 8 /* LoopLabel */) {
+                    // A loop is reachable if the control flow path that leads to the top is reachable.
+                    flow = flow.antecedents[0];
+                }
+                else if (flags & 128 /* SwitchClause */) {
+                    // The control flow path representing an unmatched value in a switch statement with
+                    // no default clause is unreachable if the switch statement is exhaustive.
+                    if (flow.clauseStart === flow.clauseEnd && isExhaustiveSwitchStatement(flow.switchStatement)) {
+                        return false;
                     }
-                    return errorType;
+                    flow = flow.antecedent;
                 }
-                if (indexInfo.isReadonly && (ts.isAssignmentTarget(node) || ts.isDeleteTarget(node))) {
-                    error(node, ts.Diagnostics.Index_signature_in_type_0_only_permits_reading, typeToString(apparentType));
+                else if (flags & 1024 /* ReduceLabel */) {
+                    // Cache is unreliable once we start adjusting labels
+                    lastFlowNode = undefined;
+                    var target = flow.target;
+                    var saveAntecedents = target.antecedents;
+                    target.antecedents = flow.antecedents;
+                    var result = isReachableFlowNodeWorker(flow.antecedent, /*noCacheCheck*/ false);
+                    target.antecedents = saveAntecedents;
+                    return result;
                 }
-                propType = indexInfo.type;
-            }
-            else {
-                checkPropertyNotUsedBeforeDeclaration(prop, node, right);
-                markPropertyAsReferenced(prop, node, left.kind === 104 /* ThisKeyword */);
-                getNodeLinks(node).resolvedSymbol = prop;
-                checkPropertyAccessibility(node, left.kind === 102 /* SuperKeyword */, apparentType, prop);
-                if (isAssignmentToReadonlyEntity(node, prop, assignmentKind)) {
-                    error(right, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_read_only_property, ts.idText(right));
-                    return errorType;
+                else {
+                    return !(flags & 1 /* Unreachable */);
                 }
-                propType = getConstraintForLocation(getTypeOfSymbol(prop), node);
             }
-            return getFlowTypeOfAccessExpression(node, prop, propType, right);
         }
-        function getFlowTypeOfAccessExpression(node, prop, propType, errorNode) {
-            // Only compute control flow type if this is a property access expression that isn't an
-            // assignment target, and the referenced property was declared as a variable, property,
-            // accessor, or optional method.
-            var assignmentKind = ts.getAssignmentTargetKind(node);
-            if (!ts.isAccessExpression(node) ||
-                assignmentKind === 1 /* Definite */ ||
-                prop && !(prop.flags & (3 /* Variable */ | 4 /* Property */ | 98304 /* Accessor */)) && !(prop.flags & 8192 /* Method */ && propType.flags & 1048576 /* Union */)) {
-                return propType;
-            }
-            // If strict null checks and strict property initialization checks are enabled, if we have
-            // a this.xxx property access, if the property is an instance property without an initializer,
-            // and if we are in a constructor of the same class as the property declaration, assume that
-            // the property is uninitialized at the top of the control flow.
-            var assumeUninitialized = false;
-            if (strictNullChecks && strictPropertyInitialization && node.expression.kind === 104 /* ThisKeyword */) {
-                var declaration = prop && prop.valueDeclaration;
-                if (declaration && isInstancePropertyWithoutInitializer(declaration)) {
-                    var flowContainer = getControlFlowContainer(node);
-                    if (flowContainer.kind === 162 /* Constructor */ && flowContainer.parent === declaration.parent && !(declaration.flags & 8388608 /* Ambient */)) {
-                        assumeUninitialized = true;
+        // Return true if the given flow node is preceded by a 'super(...)' call in every possible code path
+        // leading to the node.
+        function isPostSuperFlowNode(flow, noCacheCheck) {
+            while (true) {
+                var flags = flow.flags;
+                if (flags & 4096 /* Shared */) {
+                    if (!noCacheCheck) {
+                        var id = getFlowNodeId(flow);
+                        var postSuper = flowNodePostSuper[id];
+                        return postSuper !== undefined ? postSuper : (flowNodePostSuper[id] = isPostSuperFlowNode(flow, /*noCacheCheck*/ true));
+                    }
+                    noCacheCheck = false;
+                }
+                if (flags & (16 /* Assignment */ | 96 /* Condition */ | 256 /* ArrayMutation */ | 128 /* SwitchClause */)) {
+                    flow = flow.antecedent;
+                }
+                else if (flags & 512 /* Call */) {
+                    if (flow.node.expression.kind === 105 /* SuperKeyword */) {
+                        return true;
                     }
+                    flow = flow.antecedent;
+                }
+                else if (flags & 4 /* BranchLabel */) {
+                    // A branching point is post-super if every branch is post-super.
+                    return ts.every(flow.antecedents, function (f) { return isPostSuperFlowNode(f, /*noCacheCheck*/ false); });
+                }
+                else if (flags & 8 /* LoopLabel */) {
+                    // A loop is post-super if the control flow path that leads to the top is post-super.
+                    flow = flow.antecedents[0];
+                }
+                else if (flags & 1024 /* ReduceLabel */) {
+                    var target = flow.target;
+                    var saveAntecedents = target.antecedents;
+                    target.antecedents = flow.antecedents;
+                    var result = isPostSuperFlowNode(flow.antecedent, /*noCacheCheck*/ false);
+                    target.antecedents = saveAntecedents;
+                    return result;
+                }
+                else {
+                    // Unreachable nodes are considered post-super to silence errors
+                    return !!(flags & 1 /* Unreachable */);
                 }
             }
-            else if (strictNullChecks && prop && prop.valueDeclaration &&
-                ts.isPropertyAccessExpression(prop.valueDeclaration) &&
-                ts.getAssignmentDeclarationPropertyAccessKind(prop.valueDeclaration) &&
-                getControlFlowContainer(node) === getControlFlowContainer(prop.valueDeclaration)) {
-                assumeUninitialized = true;
+        }
+        function getFlowTypeOfReference(reference, declaredType, initialType, flowContainer, couldBeUninitialized) {
+            if (initialType === void 0) { initialType = declaredType; }
+            var key;
+            var isKeySet = false;
+            var flowDepth = 0;
+            if (flowAnalysisDisabled) {
+                return errorType;
             }
-            var flowType = getFlowTypeOfReference(node, propType, assumeUninitialized ? getOptionalType(propType) : propType);
-            if (assumeUninitialized && !(getFalsyFlags(propType) & 32768 /* Undefined */) && getFalsyFlags(flowType) & 32768 /* Undefined */) {
-                error(errorNode, ts.Diagnostics.Property_0_is_used_before_being_assigned, symbolToString(prop)); // TODO: GH#18217
-                // Return the declared type to reduce follow-on errors
-                return propType;
+            if (!reference.flowNode || !couldBeUninitialized && !(declaredType.flags & 536624127 /* Narrowable */)) {
+                return declaredType;
             }
-            return assignmentKind ? getBaseTypeOfLiteralType(flowType) : flowType;
-        }
-        function checkPropertyNotUsedBeforeDeclaration(prop, node, right) {
-            var valueDeclaration = prop.valueDeclaration;
-            if (!valueDeclaration || ts.getSourceFileOfNode(node).isDeclarationFile) {
-                return;
+            flowInvocationCount++;
+            var sharedFlowStart = sharedFlowCount;
+            var evolvedType = getTypeFromFlowType(getTypeAtFlowNode(reference.flowNode));
+            sharedFlowCount = sharedFlowStart;
+            // When the reference is 'x' in an 'x.length', 'x.push(value)', 'x.unshift(value)' or x[n] = value' operation,
+            // we give type 'any[]' to 'x' instead of using the type determined by control flow analysis such that operations
+            // on empty arrays are possible without implicit any errors and new element types can be inferred without
+            // type mismatch errors.
+            var resultType = ts.getObjectFlags(evolvedType) & 256 /* EvolvingArray */ && isEvolvingArrayOperationTarget(reference) ? autoArrayType : finalizeEvolvingArrayType(evolvedType);
+            if (resultType === unreachableNeverType || reference.parent && reference.parent.kind === 225 /* NonNullExpression */ && getTypeWithFacts(resultType, 2097152 /* NEUndefinedOrNull */).flags & 131072 /* Never */) {
+                return declaredType;
             }
-            var diagnosticMessage;
-            var declarationName = ts.idText(right);
-            if (isInPropertyInitializer(node)
-                && !(ts.isAccessExpression(node) && ts.isAccessExpression(node.expression))
-                && !isBlockScopedNameDeclaredBeforeUse(valueDeclaration, right)
-                && !isPropertyDeclaredInAncestorClass(prop)) {
-                diagnosticMessage = error(right, ts.Diagnostics.Property_0_is_used_before_its_initialization, declarationName);
+            return resultType;
+            function getOrSetCacheKey() {
+                if (isKeySet) {
+                    return key;
+                }
+                isKeySet = true;
+                return key = getFlowCacheKey(reference, declaredType, initialType, flowContainer);
             }
-            else if (valueDeclaration.kind === 245 /* ClassDeclaration */ &&
-                node.parent.kind !== 169 /* TypeReference */ &&
-                !(valueDeclaration.flags & 8388608 /* Ambient */) &&
-                !isBlockScopedNameDeclaredBeforeUse(valueDeclaration, right)) {
-                diagnosticMessage = error(right, ts.Diagnostics.Class_0_used_before_its_declaration, declarationName);
+            function getTypeAtFlowNode(flow) {
+                if (flowDepth === 2000) {
+                    // We have made 2000 recursive invocations. To avoid overflowing the call stack we report an error
+                    // and disable further control flow analysis in the containing function or module body.
+                    ts.tracing.instant("check" /* Check */, "getTypeAtFlowNode_DepthLimit", { flowId: flow.id });
+                    flowAnalysisDisabled = true;
+                    reportFlowControlError(reference);
+                    return errorType;
+                }
+                flowDepth++;
+                var sharedFlow;
+                while (true) {
+                    var flags = flow.flags;
+                    if (flags & 4096 /* Shared */) {
+                        // We cache results of flow type resolution for shared nodes that were previously visited in
+                        // the same getFlowTypeOfReference invocation. A node is considered shared when it is the
+                        // antecedent of more than one node.
+                        for (var i = sharedFlowStart; i < sharedFlowCount; i++) {
+                            if (sharedFlowNodes[i] === flow) {
+                                flowDepth--;
+                                return sharedFlowTypes[i];
+                            }
+                        }
+                        sharedFlow = flow;
+                    }
+                    var type = void 0;
+                    if (flags & 16 /* Assignment */) {
+                        type = getTypeAtFlowAssignment(flow);
+                        if (!type) {
+                            flow = flow.antecedent;
+                            continue;
+                        }
+                    }
+                    else if (flags & 512 /* Call */) {
+                        type = getTypeAtFlowCall(flow);
+                        if (!type) {
+                            flow = flow.antecedent;
+                            continue;
+                        }
+                    }
+                    else if (flags & 96 /* Condition */) {
+                        type = getTypeAtFlowCondition(flow);
+                    }
+                    else if (flags & 128 /* SwitchClause */) {
+                        type = getTypeAtSwitchClause(flow);
+                    }
+                    else if (flags & 12 /* Label */) {
+                        if (flow.antecedents.length === 1) {
+                            flow = flow.antecedents[0];
+                            continue;
+                        }
+                        type = flags & 4 /* BranchLabel */ ?
+                            getTypeAtFlowBranchLabel(flow) :
+                            getTypeAtFlowLoopLabel(flow);
+                    }
+                    else if (flags & 256 /* ArrayMutation */) {
+                        type = getTypeAtFlowArrayMutation(flow);
+                        if (!type) {
+                            flow = flow.antecedent;
+                            continue;
+                        }
+                    }
+                    else if (flags & 1024 /* ReduceLabel */) {
+                        var target = flow.target;
+                        var saveAntecedents = target.antecedents;
+                        target.antecedents = flow.antecedents;
+                        type = getTypeAtFlowNode(flow.antecedent);
+                        target.antecedents = saveAntecedents;
+                    }
+                    else if (flags & 2 /* Start */) {
+                        // Check if we should continue with the control flow of the containing function.
+                        var container = flow.node;
+                        if (container && container !== flowContainer &&
+                            reference.kind !== 201 /* PropertyAccessExpression */ &&
+                            reference.kind !== 202 /* ElementAccessExpression */ &&
+                            reference.kind !== 107 /* ThisKeyword */) {
+                            flow = container.flowNode;
+                            continue;
+                        }
+                        // At the top of the flow we have the initial type.
+                        type = initialType;
+                    }
+                    else {
+                        // Unreachable code errors are reported in the binding phase. Here we
+                        // simply return the non-auto declared type to reduce follow-on errors.
+                        type = convertAutoToAny(declaredType);
+                    }
+                    if (sharedFlow) {
+                        // Record visited node and the associated type in the cache.
+                        sharedFlowNodes[sharedFlowCount] = sharedFlow;
+                        sharedFlowTypes[sharedFlowCount] = type;
+                        sharedFlowCount++;
+                    }
+                    flowDepth--;
+                    return type;
+                }
             }
-            if (diagnosticMessage) {
-                ts.addRelatedInfo(diagnosticMessage, ts.createDiagnosticForNode(valueDeclaration, ts.Diagnostics._0_is_declared_here, declarationName));
+            function getInitialOrAssignedType(flow) {
+                var node = flow.node;
+                return getConstraintForLocation(node.kind === 249 /* VariableDeclaration */ || node.kind === 198 /* BindingElement */ ?
+                    getInitialType(node) :
+                    getAssignedType(node), reference);
             }
-        }
-        function isInPropertyInitializer(node) {
-            return !!ts.findAncestor(node, function (node) {
-                switch (node.kind) {
-                    case 159 /* PropertyDeclaration */:
-                        return true;
-                    case 281 /* PropertyAssignment */:
-                    case 161 /* MethodDeclaration */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                    case 283 /* SpreadAssignment */:
-                    case 154 /* ComputedPropertyName */:
-                    case 221 /* TemplateSpan */:
-                    case 276 /* JsxExpression */:
-                    case 273 /* JsxAttribute */:
-                    case 274 /* JsxAttributes */:
-                    case 275 /* JsxSpreadAttribute */:
-                    case 268 /* JsxOpeningElement */:
-                    case 216 /* ExpressionWithTypeArguments */:
-                    case 279 /* HeritageClause */:
-                        return false;
-                    default:
-                        return ts.isExpressionNode(node) ? false : "quit";
+            function getTypeAtFlowAssignment(flow) {
+                var node = flow.node;
+                // Assignments only narrow the computed type if the declared type is a union type. Thus, we
+                // only need to evaluate the assigned type if the declared type is a union type.
+                if (isMatchingReference(reference, node)) {
+                    if (!isReachableFlowNode(flow)) {
+                        return unreachableNeverType;
+                    }
+                    if (ts.getAssignmentTargetKind(node) === 2 /* Compound */) {
+                        var flowType = getTypeAtFlowNode(flow.antecedent);
+                        return createFlowType(getBaseTypeOfLiteralType(getTypeFromFlowType(flowType)), isIncomplete(flowType));
+                    }
+                    if (declaredType === autoType || declaredType === autoArrayType) {
+                        if (isEmptyArrayAssignment(node)) {
+                            return getEvolvingArrayType(neverType);
+                        }
+                        var assignedType = getWidenedLiteralType(getInitialOrAssignedType(flow));
+                        return isTypeAssignableTo(assignedType, declaredType) ? assignedType : anyArrayType;
+                    }
+                    if (declaredType.flags & 1048576 /* Union */) {
+                        return getAssignmentReducedType(declaredType, getInitialOrAssignedType(flow));
+                    }
+                    return declaredType;
                 }
-            });
-        }
-        /**
-         * It's possible that "prop.valueDeclaration" is a local declaration, but the property was also declared in a superclass.
-         * In that case we won't consider it used before its declaration, because it gets its value from the superclass' declaration.
-         */
-        function isPropertyDeclaredInAncestorClass(prop) {
-            if (!(prop.parent.flags & 32 /* Class */)) {
-                return false;
+                // We didn't have a direct match. However, if the reference is a dotted name, this
+                // may be an assignment to a left hand part of the reference. For example, for a
+                // reference 'x.y.z', we may be at an assignment to 'x.y' or 'x'. In that case,
+                // return the declared type.
+                if (containsMatchingReference(reference, node)) {
+                    if (!isReachableFlowNode(flow)) {
+                        return unreachableNeverType;
+                    }
+                    // A matching dotted name might also be an expando property on a function *expression*,
+                    // in which case we continue control flow analysis back to the function's declaration
+                    if (ts.isVariableDeclaration(node) && (ts.isInJSFile(node) || ts.isVarConst(node))) {
+                        var init = ts.getDeclaredExpandoInitializer(node);
+                        if (init && (init.kind === 208 /* FunctionExpression */ || init.kind === 209 /* ArrowFunction */)) {
+                            return getTypeAtFlowNode(flow.antecedent);
+                        }
+                    }
+                    return declaredType;
+                }
+                // for (const _ in ref) acts as a nonnull on ref
+                if (ts.isVariableDeclaration(node) && node.parent.parent.kind === 238 /* ForInStatement */ && isMatchingReference(reference, node.parent.parent.expression)) {
+                    return getNonNullableTypeIfNeeded(getTypeFromFlowType(getTypeAtFlowNode(flow.antecedent)));
+                }
+                // Assignment doesn't affect reference
+                return undefined;
             }
-            var classType = getTypeOfSymbol(prop.parent);
-            while (true) {
-                classType = classType.symbol && getSuperClass(classType);
-                if (!classType) {
-                    return false;
+            function narrowTypeByAssertion(type, expr) {
+                var node = ts.skipParentheses(expr);
+                if (node.kind === 94 /* FalseKeyword */) {
+                    return unreachableNeverType;
                 }
-                var superProperty = getPropertyOfType(classType, prop.escapedName);
-                if (superProperty && superProperty.valueDeclaration) {
-                    return true;
+                if (node.kind === 216 /* BinaryExpression */) {
+                    if (node.operatorToken.kind === 55 /* AmpersandAmpersandToken */) {
+                        return narrowTypeByAssertion(narrowTypeByAssertion(type, node.left), node.right);
+                    }
+                    if (node.operatorToken.kind === 56 /* BarBarToken */) {
+                        return getUnionType([narrowTypeByAssertion(type, node.left), narrowTypeByAssertion(type, node.right)]);
+                    }
                 }
+                return narrowType(type, node, /*assumeTrue*/ true);
             }
-        }
-        function getSuperClass(classType) {
-            var x = getBaseTypes(classType);
-            if (x.length === 0) {
+            function getTypeAtFlowCall(flow) {
+                var signature = getEffectsSignature(flow.node);
+                if (signature) {
+                    var predicate = getTypePredicateOfSignature(signature);
+                    if (predicate && (predicate.kind === 2 /* AssertsThis */ || predicate.kind === 3 /* AssertsIdentifier */)) {
+                        var flowType = getTypeAtFlowNode(flow.antecedent);
+                        var type = finalizeEvolvingArrayType(getTypeFromFlowType(flowType));
+                        var narrowedType = predicate.type ? narrowTypeByTypePredicate(type, predicate, flow.node, /*assumeTrue*/ true) :
+                            predicate.kind === 3 /* AssertsIdentifier */ && predicate.parameterIndex >= 0 && predicate.parameterIndex < flow.node.arguments.length ? narrowTypeByAssertion(type, flow.node.arguments[predicate.parameterIndex]) :
+                                type;
+                        return narrowedType === type ? flowType : createFlowType(narrowedType, isIncomplete(flowType));
+                    }
+                    if (getReturnTypeOfSignature(signature).flags & 131072 /* Never */) {
+                        return unreachableNeverType;
+                    }
+                }
                 return undefined;
             }
-            return getIntersectionType(x);
-        }
-        function reportNonexistentProperty(propNode, containingType) {
-            var errorInfo;
-            var relatedInfo;
-            if (!ts.isPrivateIdentifier(propNode) && containingType.flags & 1048576 /* Union */ && !(containingType.flags & 131068 /* Primitive */)) {
-                for (var _i = 0, _a = containingType.types; _i < _a.length; _i++) {
-                    var subtype = _a[_i];
-                    if (!getPropertyOfType(subtype, propNode.escapedText) && !getIndexInfoOfType(subtype, 0 /* String */)) {
-                        errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.declarationNameToString(propNode), typeToString(subtype));
-                        break;
+            function getTypeAtFlowArrayMutation(flow) {
+                if (declaredType === autoType || declaredType === autoArrayType) {
+                    var node = flow.node;
+                    var expr = node.kind === 203 /* CallExpression */ ?
+                        node.expression.expression :
+                        node.left.expression;
+                    if (isMatchingReference(reference, getReferenceCandidate(expr))) {
+                        var flowType = getTypeAtFlowNode(flow.antecedent);
+                        var type = getTypeFromFlowType(flowType);
+                        if (ts.getObjectFlags(type) & 256 /* EvolvingArray */) {
+                            var evolvedType_1 = type;
+                            if (node.kind === 203 /* CallExpression */) {
+                                for (var _i = 0, _a = node.arguments; _i < _a.length; _i++) {
+                                    var arg = _a[_i];
+                                    evolvedType_1 = addEvolvingArrayElementType(evolvedType_1, arg);
+                                }
+                            }
+                            else {
+                                // We must get the context free expression type so as to not recur in an uncached fashion on the LHS (which causes exponential blowup in compile time)
+                                var indexType = getContextFreeTypeOfExpression(node.left.argumentExpression);
+                                if (isTypeAssignableToKind(indexType, 296 /* NumberLike */)) {
+                                    evolvedType_1 = addEvolvingArrayElementType(evolvedType_1, node.right);
+                                }
+                            }
+                            return evolvedType_1 === type ? flowType : createFlowType(evolvedType_1, isIncomplete(flowType));
+                        }
+                        return flowType;
                     }
                 }
+                return undefined;
             }
-            if (typeHasStaticProperty(propNode.escapedText, containingType)) {
-                errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_is_a_static_member_of_type_1, ts.declarationNameToString(propNode), typeToString(containingType));
+            function getTypeAtFlowCondition(flow) {
+                var flowType = getTypeAtFlowNode(flow.antecedent);
+                var type = getTypeFromFlowType(flowType);
+                if (type.flags & 131072 /* Never */) {
+                    return flowType;
+                }
+                // If we have an antecedent type (meaning we're reachable in some way), we first
+                // attempt to narrow the antecedent type. If that produces the never type, and if
+                // the antecedent type is incomplete (i.e. a transient type in a loop), then we
+                // take the type guard as an indication that control *could* reach here once we
+                // have the complete type. We proceed by switching to the silent never type which
+                // doesn't report errors when operators are applied to it. Note that this is the
+                // *only* place a silent never type is ever generated.
+                var assumeTrue = (flow.flags & 32 /* TrueCondition */) !== 0;
+                var nonEvolvingType = finalizeEvolvingArrayType(type);
+                var narrowedType = narrowType(nonEvolvingType, flow.node, assumeTrue);
+                if (narrowedType === nonEvolvingType) {
+                    return flowType;
+                }
+                return createFlowType(narrowedType, isIncomplete(flowType));
             }
-            else {
-                var promisedType = getPromisedTypeOfPromise(containingType);
-                if (promisedType && getPropertyOfType(promisedType, propNode.escapedText)) {
-                    errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.declarationNameToString(propNode), typeToString(containingType));
-                    relatedInfo = ts.createDiagnosticForNode(propNode, ts.Diagnostics.Did_you_forget_to_use_await);
+            function getTypeAtSwitchClause(flow) {
+                var expr = flow.switchStatement.expression;
+                var flowType = getTypeAtFlowNode(flow.antecedent);
+                var type = getTypeFromFlowType(flowType);
+                if (isMatchingReference(reference, expr)) {
+                    type = narrowTypeBySwitchOnDiscriminant(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd);
+                }
+                else if (expr.kind === 211 /* TypeOfExpression */ && isMatchingReference(reference, expr.expression)) {
+                    type = narrowBySwitchOnTypeOf(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd);
                 }
                 else {
-                    var suggestion = getSuggestedSymbolForNonexistentProperty(propNode, containingType);
-                    if (suggestion !== undefined) {
-                        var suggestedName = ts.symbolName(suggestion);
-                        errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1_Did_you_mean_2, ts.declarationNameToString(propNode), typeToString(containingType), suggestedName);
-                        relatedInfo = suggestion.valueDeclaration && ts.createDiagnosticForNode(suggestion.valueDeclaration, ts.Diagnostics._0_is_declared_here, suggestedName);
+                    if (strictNullChecks) {
+                        if (optionalChainContainsReference(expr, reference)) {
+                            type = narrowTypeBySwitchOptionalChainContainment(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd, function (t) { return !(t.flags & (32768 /* Undefined */ | 131072 /* Never */)); });
+                        }
+                        else if (expr.kind === 211 /* TypeOfExpression */ && optionalChainContainsReference(expr.expression, reference)) {
+                            type = narrowTypeBySwitchOptionalChainContainment(type, flow.switchStatement, flow.clauseStart, flow.clauseEnd, function (t) { return !(t.flags & 131072 /* Never */ || t.flags & 128 /* StringLiteral */ && t.value === "undefined"); });
+                        }
                     }
-                    else {
-                        errorInfo = ts.chainDiagnosticMessages(elaborateNeverIntersection(errorInfo, containingType), ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.declarationNameToString(propNode), typeToString(containingType));
+                    if (isMatchingReferenceDiscriminant(expr, type)) {
+                        type = narrowTypeByDiscriminant(type, expr, function (t) { return narrowTypeBySwitchOnDiscriminant(t, flow.switchStatement, flow.clauseStart, flow.clauseEnd); });
                     }
                 }
+                return createFlowType(type, isIncomplete(flowType));
             }
-            var resultDiagnostic = ts.createDiagnosticForNodeFromMessageChain(propNode, errorInfo);
-            if (relatedInfo) {
-                ts.addRelatedInfo(resultDiagnostic, relatedInfo);
-            }
-            diagnostics.add(resultDiagnostic);
-        }
-        function typeHasStaticProperty(propName, containingType) {
-            var prop = containingType.symbol && getPropertyOfType(getTypeOfSymbol(containingType.symbol), propName);
-            return prop !== undefined && prop.valueDeclaration && ts.hasModifier(prop.valueDeclaration, 32 /* Static */);
-        }
-        function getSuggestedSymbolForNonexistentProperty(name, containingType) {
-            return getSpellingSuggestionForName(ts.isString(name) ? name : ts.idText(name), getPropertiesOfType(containingType), 111551 /* Value */);
-        }
-        function getSuggestionForNonexistentProperty(name, containingType) {
-            var suggestion = getSuggestedSymbolForNonexistentProperty(name, containingType);
-            return suggestion && ts.symbolName(suggestion);
-        }
-        function getSuggestedSymbolForNonexistentSymbol(location, outerName, meaning) {
-            ts.Debug.assert(outerName !== undefined, "outername should always be defined");
-            var result = resolveNameHelper(location, outerName, meaning, /*nameNotFoundMessage*/ undefined, outerName, /*isUse*/ false, /*excludeGlobals*/ false, function (symbols, name, meaning) {
-                ts.Debug.assertEqual(outerName, name, "name should equal outerName");
-                var symbol = getSymbol(symbols, name, meaning);
-                // Sometimes the symbol is found when location is a return type of a function: `typeof x` and `x` is declared in the body of the function
-                // So the table *contains* `x` but `x` isn't actually in scope.
-                // However, resolveNameHelper will continue and call this callback again, so we'll eventually get a correct suggestion.
-                return symbol || getSpellingSuggestionForName(ts.unescapeLeadingUnderscores(name), ts.arrayFrom(symbols.values()), meaning);
-            });
-            return result;
-        }
-        function getSuggestionForNonexistentSymbol(location, outerName, meaning) {
-            var symbolResult = getSuggestedSymbolForNonexistentSymbol(location, outerName, meaning);
-            return symbolResult && ts.symbolName(symbolResult);
-        }
-        function getSuggestedSymbolForNonexistentModule(name, targetModule) {
-            return targetModule.exports && getSpellingSuggestionForName(ts.idText(name), getExportsOfModuleAsArray(targetModule), 2623475 /* ModuleMember */);
-        }
-        function getSuggestionForNonexistentExport(name, targetModule) {
-            var suggestion = getSuggestedSymbolForNonexistentModule(name, targetModule);
-            return suggestion && ts.symbolName(suggestion);
-        }
-        function getSuggestionForNonexistentIndexSignature(objectType, expr, keyedType) {
-            // check if object type has setter or getter
-            function hasProp(name) {
-                var prop = getPropertyOfObjectType(objectType, name);
-                if (prop) {
-                    var s = getSingleCallSignature(getTypeOfSymbol(prop));
-                    return !!s && getMinArgumentCount(s) >= 1 && isTypeAssignableTo(keyedType, getTypeAtPosition(s, 0));
+            function getTypeAtFlowBranchLabel(flow) {
+                var antecedentTypes = [];
+                var subtypeReduction = false;
+                var seenIncomplete = false;
+                var bypassFlow;
+                for (var _i = 0, _a = flow.antecedents; _i < _a.length; _i++) {
+                    var antecedent = _a[_i];
+                    if (!bypassFlow && antecedent.flags & 128 /* SwitchClause */ && antecedent.clauseStart === antecedent.clauseEnd) {
+                        // The antecedent is the bypass branch of a potentially exhaustive switch statement.
+                        bypassFlow = antecedent;
+                        continue;
+                    }
+                    var flowType = getTypeAtFlowNode(antecedent);
+                    var type = getTypeFromFlowType(flowType);
+                    // If the type at a particular antecedent path is the declared type and the
+                    // reference is known to always be assigned (i.e. when declared and initial types
+                    // are the same), there is no reason to process more antecedents since the only
+                    // possible outcome is subtypes that will be removed in the final union type anyway.
+                    if (type === declaredType && declaredType === initialType) {
+                        return type;
+                    }
+                    ts.pushIfUnique(antecedentTypes, type);
+                    // If an antecedent type is not a subset of the declared type, we need to perform
+                    // subtype reduction. This happens when a "foreign" type is injected into the control
+                    // flow using the instanceof operator or a user defined type predicate.
+                    if (!isTypeSubsetOf(type, declaredType)) {
+                        subtypeReduction = true;
+                    }
+                    if (isIncomplete(flowType)) {
+                        seenIncomplete = true;
+                    }
                 }
-                return false;
+                if (bypassFlow) {
+                    var flowType = getTypeAtFlowNode(bypassFlow);
+                    var type = getTypeFromFlowType(flowType);
+                    // If the bypass flow contributes a type we haven't seen yet and the switch statement
+                    // isn't exhaustive, process the bypass flow type. Since exhaustiveness checks increase
+                    // the risk of circularities, we only want to perform them when they make a difference.
+                    if (!ts.contains(antecedentTypes, type) && !isExhaustiveSwitchStatement(bypassFlow.switchStatement)) {
+                        if (type === declaredType && declaredType === initialType) {
+                            return type;
+                        }
+                        antecedentTypes.push(type);
+                        if (!isTypeSubsetOf(type, declaredType)) {
+                            subtypeReduction = true;
+                        }
+                        if (isIncomplete(flowType)) {
+                            seenIncomplete = true;
+                        }
+                    }
+                }
+                return createFlowType(getUnionOrEvolvingArrayType(antecedentTypes, subtypeReduction ? 2 /* Subtype */ : 1 /* Literal */), seenIncomplete);
             }
-            ;
-            var suggestedMethod = ts.isAssignmentTarget(expr) ? "set" : "get";
-            if (!hasProp(suggestedMethod)) {
-                return undefined;
+            function getTypeAtFlowLoopLabel(flow) {
+                // If we have previously computed the control flow type for the reference at
+                // this flow loop junction, return the cached type.
+                var id = getFlowNodeId(flow);
+                var cache = flowLoopCaches[id] || (flowLoopCaches[id] = new ts.Map());
+                var key = getOrSetCacheKey();
+                if (!key) {
+                    // No cache key is generated when binding patterns are in unnarrowable situations
+                    return declaredType;
+                }
+                var cached = cache.get(key);
+                if (cached) {
+                    return cached;
+                }
+                // If this flow loop junction and reference are already being processed, return
+                // the union of the types computed for each branch so far, marked as incomplete.
+                // It is possible to see an empty array in cases where loops are nested and the
+                // back edge of the outer loop reaches an inner loop that is already being analyzed.
+                // In such cases we restart the analysis of the inner loop, which will then see
+                // a non-empty in-process array for the outer loop and eventually terminate because
+                // the first antecedent of a loop junction is always the non-looping control flow
+                // path that leads to the top.
+                for (var i = flowLoopStart; i < flowLoopCount; i++) {
+                    if (flowLoopNodes[i] === flow && flowLoopKeys[i] === key && flowLoopTypes[i].length) {
+                        return createFlowType(getUnionOrEvolvingArrayType(flowLoopTypes[i], 1 /* Literal */), /*incomplete*/ true);
+                    }
+                }
+                // Add the flow loop junction and reference to the in-process stack and analyze
+                // each antecedent code path.
+                var antecedentTypes = [];
+                var subtypeReduction = false;
+                var firstAntecedentType;
+                for (var _i = 0, _a = flow.antecedents; _i < _a.length; _i++) {
+                    var antecedent = _a[_i];
+                    var flowType = void 0;
+                    if (!firstAntecedentType) {
+                        // The first antecedent of a loop junction is always the non-looping control
+                        // flow path that leads to the top.
+                        flowType = firstAntecedentType = getTypeAtFlowNode(antecedent);
+                    }
+                    else {
+                        // All but the first antecedent are the looping control flow paths that lead
+                        // back to the loop junction. We track these on the flow loop stack.
+                        flowLoopNodes[flowLoopCount] = flow;
+                        flowLoopKeys[flowLoopCount] = key;
+                        flowLoopTypes[flowLoopCount] = antecedentTypes;
+                        flowLoopCount++;
+                        var saveFlowTypeCache = flowTypeCache;
+                        flowTypeCache = undefined;
+                        flowType = getTypeAtFlowNode(antecedent);
+                        flowTypeCache = saveFlowTypeCache;
+                        flowLoopCount--;
+                        // If we see a value appear in the cache it is a sign that control flow analysis
+                        // was restarted and completed by checkExpressionCached. We can simply pick up
+                        // the resulting type and bail out.
+                        var cached_1 = cache.get(key);
+                        if (cached_1) {
+                            return cached_1;
+                        }
+                    }
+                    var type = getTypeFromFlowType(flowType);
+                    ts.pushIfUnique(antecedentTypes, type);
+                    // If an antecedent type is not a subset of the declared type, we need to perform
+                    // subtype reduction. This happens when a "foreign" type is injected into the control
+                    // flow using the instanceof operator or a user defined type predicate.
+                    if (!isTypeSubsetOf(type, declaredType)) {
+                        subtypeReduction = true;
+                    }
+                    // If the type at a particular antecedent path is the declared type there is no
+                    // reason to process more antecedents since the only possible outcome is subtypes
+                    // that will be removed in the final union type anyway.
+                    if (type === declaredType) {
+                        break;
+                    }
+                }
+                // The result is incomplete if the first antecedent (the non-looping control flow path)
+                // is incomplete.
+                var result = getUnionOrEvolvingArrayType(antecedentTypes, subtypeReduction ? 2 /* Subtype */ : 1 /* Literal */);
+                if (isIncomplete(firstAntecedentType)) {
+                    return createFlowType(result, /*incomplete*/ true);
+                }
+                cache.set(key, result);
+                return result;
             }
-            var suggestion = ts.tryGetPropertyAccessOrIdentifierToString(expr.expression);
-            if (suggestion === undefined) {
-                suggestion = suggestedMethod;
+            function isMatchingReferenceDiscriminant(expr, computedType) {
+                var type = declaredType.flags & 1048576 /* Union */ ? declaredType : computedType;
+                if (!(type.flags & 1048576 /* Union */) || !ts.isAccessExpression(expr)) {
+                    return false;
+                }
+                var name = getAccessedPropertyName(expr);
+                if (name === undefined) {
+                    return false;
+                }
+                return isMatchingReference(reference, expr.expression) && isDiscriminantProperty(type, name);
             }
-            else {
-                suggestion += "." + suggestedMethod;
+            function narrowTypeByDiscriminant(type, access, narrowType) {
+                var propName = getAccessedPropertyName(access);
+                if (propName === undefined) {
+                    return type;
+                }
+                var propType = getTypeOfPropertyOfType(type, propName);
+                if (!propType) {
+                    return type;
+                }
+                var narrowedPropType = narrowType(propType);
+                return filterType(type, function (t) {
+                    var discriminantType = getTypeOfPropertyOrIndexSignature(t, propName);
+                    return !(discriminantType.flags & 131072 /* Never */) && isTypeComparableTo(discriminantType, narrowedPropType);
+                });
             }
-            return suggestion;
-        }
-        /**
-         * Given a name and a list of symbols whose names are *not* equal to the name, return a spelling suggestion if there is one that is close enough.
-         * Names less than length 3 only check for case-insensitive equality, not levenshtein distance.
-         *
-         * If there is a candidate that's the same except for case, return that.
-         * If there is a candidate that's within one edit of the name, return that.
-         * Otherwise, return the candidate with the smallest Levenshtein distance,
-         *    except for candidates:
-         *      * With no name
-         *      * Whose meaning doesn't match the `meaning` parameter.
-         *      * Whose length differs from the target name by more than 0.34 of the length of the name.
-         *      * Whose levenshtein distance is more than 0.4 of the length of the name
-         *        (0.4 allows 1 substitution/transposition for every 5 characters,
-         *         and 1 insertion/deletion at 3 characters)
-         */
-        function getSpellingSuggestionForName(name, symbols, meaning) {
-            return ts.getSpellingSuggestion(name, symbols, getCandidateName);
-            function getCandidateName(candidate) {
-                var candidateName = ts.symbolName(candidate);
-                if (ts.startsWith(candidateName, "\"")) {
-                    return undefined;
+            function narrowTypeByTruthiness(type, expr, assumeTrue) {
+                if (isMatchingReference(reference, expr)) {
+                    return getTypeWithFacts(type, assumeTrue ? 4194304 /* Truthy */ : 8388608 /* Falsy */);
                 }
-                if (candidate.flags & meaning) {
-                    return candidateName;
+                if (strictNullChecks && assumeTrue && optionalChainContainsReference(expr, reference)) {
+                    type = getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
                 }
-                if (candidate.flags & 2097152 /* Alias */) {
-                    var alias = tryResolveAlias(candidate);
-                    if (alias && alias.flags & meaning) {
-                        return candidateName;
-                    }
+                if (isMatchingReferenceDiscriminant(expr, type)) {
+                    return narrowTypeByDiscriminant(type, expr, function (t) { return getTypeWithFacts(t, assumeTrue ? 4194304 /* Truthy */ : 8388608 /* Falsy */); });
                 }
-                return undefined;
-            }
-        }
-        function markPropertyAsReferenced(prop, nodeForCheckWriteOnly, isThisAccess) {
-            var valueDeclaration = prop && (prop.flags & 106500 /* ClassMember */) && prop.valueDeclaration;
-            if (!valueDeclaration) {
-                return;
-            }
-            var hasPrivateModifier = ts.hasModifier(valueDeclaration, 8 /* Private */);
-            var hasPrivateIdentifier = ts.isNamedDeclaration(prop.valueDeclaration) && ts.isPrivateIdentifier(prop.valueDeclaration.name);
-            if (!hasPrivateModifier && !hasPrivateIdentifier) {
-                return;
-            }
-            if (nodeForCheckWriteOnly && ts.isWriteOnlyAccess(nodeForCheckWriteOnly) && !(prop.flags & 65536 /* SetAccessor */)) {
-                return;
+                return type;
             }
-            if (isThisAccess) {
-                // Find any FunctionLikeDeclaration because those create a new 'this' binding. But this should only matter for methods (or getters/setters).
-                var containingMethod = ts.findAncestor(nodeForCheckWriteOnly, ts.isFunctionLikeDeclaration);
-                if (containingMethod && containingMethod.symbol === prop) {
-                    return;
+            function isTypePresencePossible(type, propName, assumeTrue) {
+                if (getIndexInfoOfType(type, 0 /* String */)) {
+                    return true;
                 }
-            }
-            (ts.getCheckFlags(prop) & 1 /* Instantiated */ ? getSymbolLinks(prop).target : prop).isReferenced = 67108863 /* All */;
-        }
-        function isValidPropertyAccess(node, propertyName) {
-            switch (node.kind) {
-                case 194 /* PropertyAccessExpression */:
-                    return isValidPropertyAccessWithType(node, node.expression.kind === 102 /* SuperKeyword */, propertyName, getWidenedType(checkExpression(node.expression)));
-                case 153 /* QualifiedName */:
-                    return isValidPropertyAccessWithType(node, /*isSuper*/ false, propertyName, getWidenedType(checkExpression(node.left)));
-                case 188 /* ImportType */:
-                    return isValidPropertyAccessWithType(node, /*isSuper*/ false, propertyName, getTypeFromTypeNode(node));
-            }
-        }
-        function isValidPropertyAccessForCompletions(node, type, property) {
-            return isValidPropertyAccessWithType(node, node.kind === 194 /* PropertyAccessExpression */ && node.expression.kind === 102 /* SuperKeyword */, property.escapedName, type);
-            // Previously we validated the 'this' type of methods but this adversely affected performance. See #31377 for more context.
-        }
-        function isValidPropertyAccessWithType(node, isSuper, propertyName, type) {
-            if (type === errorType || isTypeAny(type)) {
-                return true;
-            }
-            var prop = getPropertyOfType(type, propertyName);
-            if (prop) {
-                if (ts.isPropertyAccessExpression(node) && prop.valueDeclaration && ts.isPrivateIdentifierPropertyDeclaration(prop.valueDeclaration)) {
-                    var declClass_1 = ts.getContainingClass(prop.valueDeclaration);
-                    return !ts.isOptionalChain(node) && !!ts.findAncestor(node, function (parent) { return parent === declClass_1; });
+                var prop = getPropertyOfType(type, propName);
+                if (prop) {
+                    return prop.flags & 16777216 /* Optional */ ? true : assumeTrue;
                 }
-                return checkPropertyAccessibility(node, isSuper, type, prop);
+                return !assumeTrue;
             }
-            // In js files properties of unions are allowed in completion
-            return ts.isInJSFile(node) && (type.flags & 1048576 /* Union */) !== 0 && type.types.some(function (elementType) { return isValidPropertyAccessWithType(node, isSuper, propertyName, elementType); });
-        }
-        /**
-         * Return the symbol of the for-in variable declared or referenced by the given for-in statement.
-         */
-        function getForInVariableSymbol(node) {
-            var initializer = node.initializer;
-            if (initializer.kind === 243 /* VariableDeclarationList */) {
-                var variable = initializer.declarations[0];
-                if (variable && !ts.isBindingPattern(variable.name)) {
-                    return getSymbolOfNode(variable);
+            function narrowByInKeyword(type, literal, assumeTrue) {
+                if (type.flags & (1048576 /* Union */ | 524288 /* Object */)
+                    || isThisTypeParameter(type)
+                    || type.flags & 2097152 /* Intersection */ && ts.every(type.types, function (t) { return t.symbol !== globalThisSymbol; })) {
+                    var propName_1 = ts.escapeLeadingUnderscores(literal.text);
+                    return filterType(type, function (t) { return isTypePresencePossible(t, propName_1, assumeTrue); });
                 }
+                return type;
             }
-            else if (initializer.kind === 75 /* Identifier */) {
-                return getResolvedSymbol(initializer);
-            }
-            return undefined;
-        }
-        /**
-         * Return true if the given type is considered to have numeric property names.
-         */
-        function hasNumericPropertyNames(type) {
-            return getIndexTypeOfType(type, 1 /* Number */) && !getIndexTypeOfType(type, 0 /* String */);
-        }
-        /**
-         * Return true if given node is an expression consisting of an identifier (possibly parenthesized)
-         * that references a for-in variable for an object with numeric property names.
-         */
-        function isForInVariableForNumericPropertyNames(expr) {
-            var e = ts.skipParentheses(expr);
-            if (e.kind === 75 /* Identifier */) {
-                var symbol = getResolvedSymbol(e);
-                if (symbol.flags & 3 /* Variable */) {
-                    var child = expr;
-                    var node = expr.parent;
-                    while (node) {
-                        if (node.kind === 231 /* ForInStatement */ &&
-                            child === node.statement &&
-                            getForInVariableSymbol(node) === symbol &&
-                            hasNumericPropertyNames(getTypeOfExpression(node.expression))) {
-                            return true;
+            function narrowTypeByBinaryExpression(type, expr, assumeTrue) {
+                switch (expr.operatorToken.kind) {
+                    case 62 /* EqualsToken */:
+                    case 74 /* BarBarEqualsToken */:
+                    case 75 /* AmpersandAmpersandEqualsToken */:
+                    case 76 /* QuestionQuestionEqualsToken */:
+                        return narrowTypeByTruthiness(narrowType(type, expr.right, assumeTrue), expr.left, assumeTrue);
+                    case 34 /* EqualsEqualsToken */:
+                    case 35 /* ExclamationEqualsToken */:
+                    case 36 /* EqualsEqualsEqualsToken */:
+                    case 37 /* ExclamationEqualsEqualsToken */:
+                        var operator_1 = expr.operatorToken.kind;
+                        var left_1 = getReferenceCandidate(expr.left);
+                        var right_1 = getReferenceCandidate(expr.right);
+                        if (left_1.kind === 211 /* TypeOfExpression */ && ts.isStringLiteralLike(right_1)) {
+                            return narrowTypeByTypeof(type, left_1, operator_1, right_1, assumeTrue);
                         }
-                        child = node;
-                        node = node.parent;
-                    }
+                        if (right_1.kind === 211 /* TypeOfExpression */ && ts.isStringLiteralLike(left_1)) {
+                            return narrowTypeByTypeof(type, right_1, operator_1, left_1, assumeTrue);
+                        }
+                        if (isMatchingReference(reference, left_1)) {
+                            return narrowTypeByEquality(type, operator_1, right_1, assumeTrue);
+                        }
+                        if (isMatchingReference(reference, right_1)) {
+                            return narrowTypeByEquality(type, operator_1, left_1, assumeTrue);
+                        }
+                        if (strictNullChecks) {
+                            if (optionalChainContainsReference(left_1, reference)) {
+                                type = narrowTypeByOptionalChainContainment(type, operator_1, right_1, assumeTrue);
+                            }
+                            else if (optionalChainContainsReference(right_1, reference)) {
+                                type = narrowTypeByOptionalChainContainment(type, operator_1, left_1, assumeTrue);
+                            }
+                        }
+                        if (isMatchingReferenceDiscriminant(left_1, type)) {
+                            return narrowTypeByDiscriminant(type, left_1, function (t) { return narrowTypeByEquality(t, operator_1, right_1, assumeTrue); });
+                        }
+                        if (isMatchingReferenceDiscriminant(right_1, type)) {
+                            return narrowTypeByDiscriminant(type, right_1, function (t) { return narrowTypeByEquality(t, operator_1, left_1, assumeTrue); });
+                        }
+                        if (isMatchingConstructorReference(left_1)) {
+                            return narrowTypeByConstructor(type, operator_1, right_1, assumeTrue);
+                        }
+                        if (isMatchingConstructorReference(right_1)) {
+                            return narrowTypeByConstructor(type, operator_1, left_1, assumeTrue);
+                        }
+                        break;
+                    case 101 /* InstanceOfKeyword */:
+                        return narrowTypeByInstanceof(type, expr, assumeTrue);
+                    case 100 /* InKeyword */:
+                        var target = getReferenceCandidate(expr.right);
+                        if (ts.isStringLiteralLike(expr.left) && isMatchingReference(reference, target)) {
+                            return narrowByInKeyword(type, expr.left, assumeTrue);
+                        }
+                        break;
+                    case 27 /* CommaToken */:
+                        return narrowType(type, expr.right, assumeTrue);
                 }
+                return type;
             }
-            return false;
-        }
-        function checkIndexedAccess(node) {
-            return node.flags & 32 /* OptionalChain */ ? checkElementAccessChain(node) :
-                checkElementAccessExpression(node, checkNonNullExpression(node.expression));
-        }
-        function checkElementAccessChain(node) {
-            var exprType = checkExpression(node.expression);
-            var nonOptionalType = getOptionalExpressionType(exprType, node.expression);
-            return propagateOptionalTypeMarker(checkElementAccessExpression(node, checkNonNullType(nonOptionalType, node.expression)), node, nonOptionalType !== exprType);
-        }
-        function checkElementAccessExpression(node, exprType) {
-            var objectType = ts.getAssignmentTargetKind(node) !== 0 /* None */ || isMethodAccessForCall(node) ? getWidenedType(exprType) : exprType;
-            var indexExpression = node.argumentExpression;
-            var indexType = checkExpression(indexExpression);
-            if (objectType === errorType || objectType === silentNeverType) {
-                return objectType;
-            }
-            if (isConstEnumObjectType(objectType) && !ts.isStringLiteralLike(indexExpression)) {
-                error(indexExpression, ts.Diagnostics.A_const_enum_member_can_only_be_accessed_using_a_string_literal);
-                return errorType;
-            }
-            var effectiveIndexType = isForInVariableForNumericPropertyNames(indexExpression) ? numberType : indexType;
-            var accessFlags = ts.isAssignmentTarget(node) ?
-                2 /* Writing */ | (isGenericObjectType(objectType) && !isThisTypeParameter(objectType) ? 1 /* NoIndexSignatures */ : 0) :
-                0 /* None */;
-            var indexedAccessType = getIndexedAccessTypeOrUndefined(objectType, effectiveIndexType, node, accessFlags) || errorType;
-            return checkIndexedAccessIndexType(getFlowTypeOfAccessExpression(node, indexedAccessType.symbol, indexedAccessType, indexExpression), node);
-        }
-        function checkThatExpressionIsProperSymbolReference(expression, expressionType, reportError) {
-            if (expressionType === errorType) {
-                // There is already an error, so no need to report one.
-                return false;
-            }
-            if (!ts.isWellKnownSymbolSyntactically(expression)) {
-                return false;
+            function narrowTypeByOptionalChainContainment(type, operator, value, assumeTrue) {
+                // We are in a branch of obj?.foo === value (or any one of the other equality operators). We narrow obj as follows:
+                // When operator is === and type of value excludes undefined, null and undefined is removed from type of obj in true branch.
+                // When operator is !== and type of value excludes undefined, null and undefined is removed from type of obj in false branch.
+                // When operator is == and type of value excludes null and undefined, null and undefined is removed from type of obj in true branch.
+                // When operator is != and type of value excludes null and undefined, null and undefined is removed from type of obj in false branch.
+                // When operator is === and type of value is undefined, null and undefined is removed from type of obj in false branch.
+                // When operator is !== and type of value is undefined, null and undefined is removed from type of obj in true branch.
+                // When operator is == and type of value is null or undefined, null and undefined is removed from type of obj in false branch.
+                // When operator is != and type of value is null or undefined, null and undefined is removed from type of obj in true branch.
+                var equalsOperator = operator === 34 /* EqualsEqualsToken */ || operator === 36 /* EqualsEqualsEqualsToken */;
+                var nullableFlags = operator === 34 /* EqualsEqualsToken */ || operator === 35 /* ExclamationEqualsToken */ ? 98304 /* Nullable */ : 32768 /* Undefined */;
+                var valueType = getTypeOfExpression(value);
+                // Note that we include any and unknown in the exclusion test because their domain includes null and undefined.
+                var removeNullable = equalsOperator !== assumeTrue && everyType(valueType, function (t) { return !!(t.flags & nullableFlags); }) ||
+                    equalsOperator === assumeTrue && everyType(valueType, function (t) { return !(t.flags & (3 /* AnyOrUnknown */ | nullableFlags)); });
+                return removeNullable ? getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */) : type;
             }
-            // Make sure the property type is the primitive symbol type
-            if ((expressionType.flags & 12288 /* ESSymbolLike */) === 0) {
-                if (reportError) {
-                    error(expression, ts.Diagnostics.A_computed_property_name_of_the_form_0_must_be_of_type_symbol, ts.getTextOfNode(expression));
+            function narrowTypeByEquality(type, operator, value, assumeTrue) {
+                if (type.flags & 1 /* Any */) {
+                    return type;
                 }
-                return false;
-            }
-            // The name is Symbol.<someName>, so make sure Symbol actually resolves to the
-            // global Symbol object
-            var leftHandSide = expression.expression;
-            var leftHandSideSymbol = getResolvedSymbol(leftHandSide);
-            if (!leftHandSideSymbol) {
-                return false;
-            }
-            var globalESSymbol = getGlobalESSymbolConstructorSymbol(/*reportErrors*/ true);
-            if (!globalESSymbol) {
-                // Already errored when we tried to look up the symbol
-                return false;
-            }
-            if (leftHandSideSymbol !== globalESSymbol) {
-                if (reportError) {
-                    error(leftHandSide, ts.Diagnostics.Symbol_reference_does_not_refer_to_the_global_Symbol_constructor_object);
+                if (operator === 35 /* ExclamationEqualsToken */ || operator === 37 /* ExclamationEqualsEqualsToken */) {
+                    assumeTrue = !assumeTrue;
                 }
-                return false;
-            }
-            return true;
-        }
-        function callLikeExpressionMayHaveTypeArguments(node) {
-            return ts.isCallOrNewExpression(node) || ts.isTaggedTemplateExpression(node) || ts.isJsxOpeningLikeElement(node);
-        }
-        function resolveUntypedCall(node) {
-            if (callLikeExpressionMayHaveTypeArguments(node)) {
-                // Check type arguments even though we will give an error that untyped calls may not accept type arguments.
-                // This gets us diagnostics for the type arguments and marks them as referenced.
-                ts.forEach(node.typeArguments, checkSourceElement);
-            }
-            if (node.kind === 198 /* TaggedTemplateExpression */) {
-                checkExpression(node.template);
-            }
-            else if (ts.isJsxOpeningLikeElement(node)) {
-                checkExpression(node.attributes);
-            }
-            else if (node.kind !== 157 /* Decorator */) {
-                ts.forEach(node.arguments, function (argument) {
-                    checkExpression(argument);
-                });
-            }
-            return anySignature;
-        }
-        function resolveErrorCall(node) {
-            resolveUntypedCall(node);
-            return unknownSignature;
-        }
-        // Re-order candidate signatures into the result array. Assumes the result array to be empty.
-        // The candidate list orders groups in reverse, but within a group signatures are kept in declaration order
-        // A nit here is that we reorder only signatures that belong to the same symbol,
-        // so order how inherited signatures are processed is still preserved.
-        // interface A { (x: string): void }
-        // interface B extends A { (x: 'foo'): string }
-        // const b: B;
-        // b('foo') // <- here overloads should be processed as [(x:'foo'): string, (x: string): void]
-        function reorderCandidates(signatures, result, callChainFlags) {
-            var lastParent;
-            var lastSymbol;
-            var cutoffIndex = 0;
-            var index;
-            var specializedIndex = -1;
-            var spliceIndex;
-            ts.Debug.assert(!result.length);
-            for (var _i = 0, signatures_7 = signatures; _i < signatures_7.length; _i++) {
-                var signature = signatures_7[_i];
-                var symbol = signature.declaration && getSymbolOfNode(signature.declaration);
-                var parent = signature.declaration && signature.declaration.parent;
-                if (!lastSymbol || symbol === lastSymbol) {
-                    if (lastParent && parent === lastParent) {
-                        index = index + 1;
+                var valueType = getTypeOfExpression(value);
+                if ((type.flags & 2 /* Unknown */) && assumeTrue && (operator === 36 /* EqualsEqualsEqualsToken */ || operator === 37 /* ExclamationEqualsEqualsToken */)) {
+                    if (valueType.flags & (131068 /* Primitive */ | 67108864 /* NonPrimitive */)) {
+                        return valueType;
                     }
-                    else {
-                        lastParent = parent;
-                        index = cutoffIndex;
+                    if (valueType.flags & 524288 /* Object */) {
+                        return nonPrimitiveType;
                     }
+                    return type;
                 }
-                else {
-                    // current declaration belongs to a different symbol
-                    // set cutoffIndex so re-orderings in the future won't change result set from 0 to cutoffIndex
-                    index = cutoffIndex = result.length;
-                    lastParent = parent;
+                if (valueType.flags & 98304 /* Nullable */) {
+                    if (!strictNullChecks) {
+                        return type;
+                    }
+                    var doubleEquals = operator === 34 /* EqualsEqualsToken */ || operator === 35 /* ExclamationEqualsToken */;
+                    var facts = doubleEquals ?
+                        assumeTrue ? 262144 /* EQUndefinedOrNull */ : 2097152 /* NEUndefinedOrNull */ :
+                        valueType.flags & 65536 /* Null */ ?
+                            assumeTrue ? 131072 /* EQNull */ : 1048576 /* NENull */ :
+                            assumeTrue ? 65536 /* EQUndefined */ : 524288 /* NEUndefined */;
+                    return getTypeWithFacts(type, facts);
                 }
-                lastSymbol = symbol;
-                // specialized signatures always need to be placed before non-specialized signatures regardless
-                // of the cutoff position; see GH#1133
-                if (signatureHasLiteralTypes(signature)) {
-                    specializedIndex++;
-                    spliceIndex = specializedIndex;
-                    // The cutoff index always needs to be greater than or equal to the specialized signature index
-                    // in order to prevent non-specialized signatures from being added before a specialized
-                    // signature.
-                    cutoffIndex++;
+                if (assumeTrue) {
+                    var filterFn = operator === 34 /* EqualsEqualsToken */ ?
+                        (function (t) { return areTypesComparable(t, valueType) || isCoercibleUnderDoubleEquals(t, valueType); }) :
+                        function (t) { return areTypesComparable(t, valueType); };
+                    return replacePrimitivesWithLiterals(filterType(type, filterFn), valueType);
                 }
-                else {
-                    spliceIndex = index;
+                if (isUnitType(valueType)) {
+                    var regularType_1 = getRegularTypeOfLiteralType(valueType);
+                    return filterType(type, function (t) { return isUnitType(t) ? !areTypesComparable(t, valueType) : getRegularTypeOfLiteralType(t) !== regularType_1; });
                 }
-                result.splice(spliceIndex, 0, callChainFlags ? getOptionalCallSignature(signature, callChainFlags) : signature);
+                return type;
             }
-        }
-        function isSpreadArgument(arg) {
-            return !!arg && (arg.kind === 213 /* SpreadElement */ || arg.kind === 220 /* SyntheticExpression */ && arg.isSpread);
-        }
-        function getSpreadArgumentIndex(args) {
-            return ts.findIndex(args, isSpreadArgument);
-        }
-        function acceptsVoid(t) {
-            return !!(t.flags & 16384 /* Void */);
-        }
-        function hasCorrectArity(node, args, signature, signatureHelpTrailingComma) {
-            if (signatureHelpTrailingComma === void 0) { signatureHelpTrailingComma = false; }
-            var argCount;
-            var callIsIncomplete = false; // In incomplete call we want to be lenient when we have too few arguments
-            var effectiveParameterCount = getParameterCount(signature);
-            var effectiveMinimumArguments = getMinArgumentCount(signature);
-            if (node.kind === 198 /* TaggedTemplateExpression */) {
-                argCount = args.length;
-                if (node.template.kind === 211 /* TemplateExpression */) {
-                    // If a tagged template expression lacks a tail literal, the call is incomplete.
-                    // Specifically, a template only can end in a TemplateTail or a Missing literal.
-                    var lastSpan = ts.last(node.template.templateSpans); // we should always have at least one span.
-                    callIsIncomplete = ts.nodeIsMissing(lastSpan.literal) || !!lastSpan.literal.isUnterminated;
-                }
-                else {
-                    // If the template didn't end in a backtick, or its beginning occurred right prior to EOF,
-                    // then this might actually turn out to be a TemplateHead in the future;
-                    // so we consider the call to be incomplete.
-                    var templateLiteral = node.template;
-                    ts.Debug.assert(templateLiteral.kind === 14 /* NoSubstitutionTemplateLiteral */);
-                    callIsIncomplete = !!templateLiteral.isUnterminated;
+            function narrowTypeByTypeof(type, typeOfExpr, operator, literal, assumeTrue) {
+                // We have '==', '!=', '===', or !==' operator with 'typeof xxx' and string literal operands
+                if (operator === 35 /* ExclamationEqualsToken */ || operator === 37 /* ExclamationEqualsEqualsToken */) {
+                    assumeTrue = !assumeTrue;
                 }
-            }
-            else if (node.kind === 157 /* Decorator */) {
-                argCount = getDecoratorArgumentCount(node, signature);
-            }
-            else if (ts.isJsxOpeningLikeElement(node)) {
-                callIsIncomplete = node.attributes.end === node.end;
-                if (callIsIncomplete) {
-                    return true;
+                var target = getReferenceCandidate(typeOfExpr.expression);
+                if (!isMatchingReference(reference, target)) {
+                    if (strictNullChecks && optionalChainContainsReference(target, reference) && assumeTrue === (literal.text !== "undefined")) {
+                        return getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
+                    }
+                    return type;
                 }
-                argCount = effectiveMinimumArguments === 0 ? args.length : 1;
-                effectiveParameterCount = args.length === 0 ? effectiveParameterCount : 1; // class may have argumentless ctor functions - still resolve ctor and compare vs props member type
-                effectiveMinimumArguments = Math.min(effectiveMinimumArguments, 1); // sfc may specify context argument - handled by framework and not typechecked
-            }
-            else {
-                if (!node.arguments) {
-                    // This only happens when we have something of the form: 'new C'
-                    ts.Debug.assert(node.kind === 197 /* NewExpression */);
-                    return getMinArgumentCount(signature) === 0;
+                if (type.flags & 1 /* Any */ && literal.text === "function") {
+                    return type;
                 }
-                argCount = signatureHelpTrailingComma ? args.length + 1 : args.length;
-                // If we are missing the close parenthesis, the call is incomplete.
-                callIsIncomplete = node.arguments.end === node.end;
-                // If a spread argument is present, check that it corresponds to a rest parameter or at least that it's in the valid range.
-                var spreadArgIndex = getSpreadArgumentIndex(args);
-                if (spreadArgIndex >= 0) {
-                    return spreadArgIndex >= getMinArgumentCount(signature) && (hasEffectiveRestParameter(signature) || spreadArgIndex < getParameterCount(signature));
+                if (assumeTrue && type.flags & 2 /* Unknown */ && literal.text === "object") {
+                    // The pattern x && typeof x === 'object', where x is of type unknown, narrows x to type object. We don't
+                    // need to check for the reverse typeof x === 'object' && x since that already narrows correctly.
+                    if (typeOfExpr.parent.parent.kind === 216 /* BinaryExpression */) {
+                        var expr = typeOfExpr.parent.parent;
+                        if (expr.operatorToken.kind === 55 /* AmpersandAmpersandToken */ && expr.right === typeOfExpr.parent && containsTruthyCheck(reference, expr.left)) {
+                            return nonPrimitiveType;
+                        }
+                    }
+                    return getUnionType([nonPrimitiveType, nullType]);
                 }
+                var facts = assumeTrue ?
+                    typeofEQFacts.get(literal.text) || 128 /* TypeofEQHostObject */ :
+                    typeofNEFacts.get(literal.text) || 32768 /* TypeofNEHostObject */;
+                var impliedType = getImpliedTypeFromTypeofGuard(type, literal.text);
+                return getTypeWithFacts(assumeTrue && impliedType ? mapType(type, narrowUnionMemberByTypeof(impliedType)) : type, facts);
             }
-            // Too many arguments implies incorrect arity.
-            if (!hasEffectiveRestParameter(signature) && argCount > effectiveParameterCount) {
-                return false;
-            }
-            // If the call is incomplete, we should skip the lower bound check.
-            // JSX signatures can have extra parameters provided by the library which we don't check
-            if (callIsIncomplete || argCount >= effectiveMinimumArguments) {
-                return true;
+            function narrowTypeBySwitchOptionalChainContainment(type, switchStatement, clauseStart, clauseEnd, clauseCheck) {
+                var everyClauseChecks = clauseStart !== clauseEnd && ts.every(getSwitchClauseTypes(switchStatement).slice(clauseStart, clauseEnd), clauseCheck);
+                return everyClauseChecks ? getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */) : type;
             }
-            for (var i = argCount; i < effectiveMinimumArguments; i++) {
-                var type = getTypeAtPosition(signature, i);
-                if (filterType(type, acceptsVoid).flags & 131072 /* Never */) {
-                    return false;
+            function narrowTypeBySwitchOnDiscriminant(type, switchStatement, clauseStart, clauseEnd) {
+                // We only narrow if all case expressions specify
+                // values with unit types, except for the case where
+                // `type` is unknown. In this instance we map object
+                // types to the nonPrimitive type and narrow with that.
+                var switchTypes = getSwitchClauseTypes(switchStatement);
+                if (!switchTypes.length) {
+                    return type;
                 }
-            }
-            return true;
-        }
-        function hasCorrectTypeArgumentArity(signature, typeArguments) {
-            // If the user supplied type arguments, but the number of type arguments does not match
-            // the declared number of type parameters, the call has an incorrect arity.
-            var numTypeParameters = ts.length(signature.typeParameters);
-            var minTypeArgumentCount = getMinTypeArgumentCount(signature.typeParameters);
-            return !ts.some(typeArguments) ||
-                (typeArguments.length >= minTypeArgumentCount && typeArguments.length <= numTypeParameters);
-        }
-        // If type has a single call signature and no other members, return that signature. Otherwise, return undefined.
-        function getSingleCallSignature(type) {
-            return getSingleSignature(type, 0 /* Call */, /*allowMembers*/ false);
-        }
-        function getSingleCallOrConstructSignature(type) {
-            return getSingleSignature(type, 0 /* Call */, /*allowMembers*/ false) ||
-                getSingleSignature(type, 1 /* Construct */, /*allowMembers*/ false);
-        }
-        function getSingleSignature(type, kind, allowMembers) {
-            if (type.flags & 524288 /* Object */) {
-                var resolved = resolveStructuredTypeMembers(type);
-                if (allowMembers || resolved.properties.length === 0 && !resolved.stringIndexInfo && !resolved.numberIndexInfo) {
-                    if (kind === 0 /* Call */ && resolved.callSignatures.length === 1 && resolved.constructSignatures.length === 0) {
-                        return resolved.callSignatures[0];
-                    }
-                    if (kind === 1 /* Construct */ && resolved.constructSignatures.length === 1 && resolved.callSignatures.length === 0) {
-                        return resolved.constructSignatures[0];
+                var clauseTypes = switchTypes.slice(clauseStart, clauseEnd);
+                var hasDefaultClause = clauseStart === clauseEnd || ts.contains(clauseTypes, neverType);
+                if ((type.flags & 2 /* Unknown */) && !hasDefaultClause) {
+                    var groundClauseTypes = void 0;
+                    for (var i = 0; i < clauseTypes.length; i += 1) {
+                        var t = clauseTypes[i];
+                        if (t.flags & (131068 /* Primitive */ | 67108864 /* NonPrimitive */)) {
+                            if (groundClauseTypes !== undefined) {
+                                groundClauseTypes.push(t);
+                            }
+                        }
+                        else if (t.flags & 524288 /* Object */) {
+                            if (groundClauseTypes === undefined) {
+                                groundClauseTypes = clauseTypes.slice(0, i);
+                            }
+                            groundClauseTypes.push(nonPrimitiveType);
+                        }
+                        else {
+                            return type;
+                        }
                     }
+                    return getUnionType(groundClauseTypes === undefined ? clauseTypes : groundClauseTypes);
                 }
-            }
-            return undefined;
-        }
-        // Instantiate a generic signature in the context of a non-generic signature (section 3.8.5 in TypeScript spec)
-        function instantiateSignatureInContextOf(signature, contextualSignature, inferenceContext, compareTypes) {
-            var context = createInferenceContext(signature.typeParameters, signature, 0 /* None */, compareTypes);
-            // We clone the inferenceContext to avoid fixing. For example, when the source signature is <T>(x: T) => T[] and
-            // the contextual signature is (...args: A) => B, we want to infer the element type of A's constraint (say 'any')
-            // for T but leave it possible to later infer '[any]' back to A.
-            var restType = getEffectiveRestType(contextualSignature);
-            var mapper = inferenceContext && (restType && restType.flags & 262144 /* TypeParameter */ ? inferenceContext.nonFixingMapper : inferenceContext.mapper);
-            var sourceSignature = mapper ? instantiateSignature(contextualSignature, mapper) : contextualSignature;
-            applyToParameterTypes(sourceSignature, signature, function (source, target) {
-                // Type parameters from outer context referenced by source type are fixed by instantiation of the source type
-                inferTypes(context.inferences, source, target);
-            });
-            if (!inferenceContext) {
-                applyToReturnTypes(contextualSignature, signature, function (source, target) {
-                    inferTypes(context.inferences, source, target, 32 /* ReturnType */);
-                });
-            }
-            return getSignatureInstantiation(signature, getInferredTypes(context), ts.isInJSFile(contextualSignature.declaration));
-        }
-        function inferJsxTypeArguments(node, signature, checkMode, context) {
-            var paramType = getEffectiveFirstArgumentForJsxSignature(signature, node);
-            var checkAttrType = checkExpressionWithContextualType(node.attributes, paramType, context, checkMode);
-            inferTypes(context.inferences, checkAttrType, paramType);
-            return getInferredTypes(context);
-        }
-        function inferTypeArguments(node, signature, args, checkMode, context) {
-            if (ts.isJsxOpeningLikeElement(node)) {
-                return inferJsxTypeArguments(node, signature, checkMode, context);
-            }
-            // If a contextual type is available, infer from that type to the return type of the call expression. For
-            // example, given a 'function wrap<T, U>(cb: (x: T) => U): (x: T) => U' and a call expression
-            // 'let f: (x: string) => number = wrap(s => s.length)', we infer from the declared type of 'f' to the
-            // return type of 'wrap'.
-            if (node.kind !== 157 /* Decorator */) {
-                var contextualType = getContextualType(node);
-                if (contextualType) {
-                    // We clone the inference context to avoid disturbing a resolution in progress for an
-                    // outer call expression. Effectively we just want a snapshot of whatever has been
-                    // inferred for any outer call expression so far.
-                    var outerContext = getInferenceContext(node);
-                    var outerMapper = getMapperFromContext(cloneInferenceContext(outerContext, 1 /* NoDefault */));
-                    var instantiatedType = instantiateType(contextualType, outerMapper);
-                    // If the contextual type is a generic function type with a single call signature, we
-                    // instantiate the type with its own type parameters and type arguments. This ensures that
-                    // the type parameters are not erased to type any during type inference such that they can
-                    // be inferred as actual types from the contextual type. For example:
-                    //   declare function arrayMap<T, U>(f: (x: T) => U): (a: T[]) => U[];
-                    //   const boxElements: <A>(a: A[]) => { value: A }[] = arrayMap(value => ({ value }));
-                    // Above, the type of the 'value' parameter is inferred to be 'A'.
-                    var contextualSignature = getSingleCallSignature(instantiatedType);
-                    var inferenceSourceType = contextualSignature && contextualSignature.typeParameters ?
-                        getOrCreateTypeFromSignature(getSignatureInstantiationWithoutFillingInTypeArguments(contextualSignature, contextualSignature.typeParameters)) :
-                        instantiatedType;
-                    var inferenceTargetType = getReturnTypeOfSignature(signature);
-                    // Inferences made from return types have lower priority than all other inferences.
-                    inferTypes(context.inferences, inferenceSourceType, inferenceTargetType, 32 /* ReturnType */);
-                    // Create a type mapper for instantiating generic contextual types using the inferences made
-                    // from the return type. We need a separate inference pass here because (a) instantiation of
-                    // the source type uses the outer context's return mapper (which excludes inferences made from
-                    // outer arguments), and (b) we don't want any further inferences going into this context.
-                    var returnContext = createInferenceContext(signature.typeParameters, signature, context.flags);
-                    var returnSourceType = instantiateType(contextualType, outerContext && outerContext.returnMapper);
-                    inferTypes(returnContext.inferences, returnSourceType, inferenceTargetType);
-                    context.returnMapper = ts.some(returnContext.inferences, hasInferenceCandidates) ? getMapperFromContext(cloneInferredPartOfContext(returnContext)) : undefined;
+                var discriminantType = getUnionType(clauseTypes);
+                var caseType = discriminantType.flags & 131072 /* Never */ ? neverType :
+                    replacePrimitivesWithLiterals(filterType(type, function (t) { return areTypesComparable(discriminantType, t); }), discriminantType);
+                if (!hasDefaultClause) {
+                    return caseType;
                 }
+                var defaultType = filterType(type, function (t) { return !(isUnitType(t) && ts.contains(switchTypes, getRegularTypeOfLiteralType(t))); });
+                return caseType.flags & 131072 /* Never */ ? defaultType : getUnionType([caseType, defaultType]);
             }
-            var thisType = getThisTypeOfSignature(signature);
-            if (thisType) {
-                var thisArgumentNode = getThisArgumentOfCall(node);
-                var thisArgumentType = thisArgumentNode ? checkExpression(thisArgumentNode) : voidType;
-                inferTypes(context.inferences, thisArgumentType, thisType);
-            }
-            var restType = getNonArrayRestType(signature);
-            var argCount = restType ? Math.min(getParameterCount(signature) - 1, args.length) : args.length;
-            for (var i = 0; i < argCount; i++) {
-                var arg = args[i];
-                if (arg.kind !== 215 /* OmittedExpression */) {
-                    var paramType = getTypeAtPosition(signature, i);
-                    var argType = checkExpressionWithContextualType(arg, paramType, context, checkMode);
-                    inferTypes(context.inferences, argType, paramType);
+            function getImpliedTypeFromTypeofGuard(type, text) {
+                switch (text) {
+                    case "function":
+                        return type.flags & 1 /* Any */ ? type : globalFunctionType;
+                    case "object":
+                        return type.flags & 2 /* Unknown */ ? getUnionType([nonPrimitiveType, nullType]) : type;
+                    default:
+                        return typeofTypesByName.get(text);
                 }
             }
-            if (restType) {
-                var spreadType = getSpreadArgumentType(args, argCount, args.length, restType, context);
-                inferTypes(context.inferences, spreadType, restType);
+            // When narrowing a union type by a `typeof` guard using type-facts alone, constituent types that are
+            // super-types of the implied guard will be retained in the final type: this is because type-facts only
+            // filter. Instead, we would like to replace those union constituents with the more precise type implied by
+            // the guard. For example: narrowing `{} | undefined` by `"boolean"` should produce the type `boolean`, not
+            // the filtered type `{}`. For this reason we narrow constituents of the union individually, in addition to
+            // filtering by type-facts.
+            function narrowUnionMemberByTypeof(candidate) {
+                return function (type) {
+                    if (isTypeSubtypeOf(type, candidate)) {
+                        return type;
+                    }
+                    if (isTypeSubtypeOf(candidate, type)) {
+                        return candidate;
+                    }
+                    if (type.flags & 465829888 /* Instantiable */) {
+                        var constraint = getBaseConstraintOfType(type) || anyType;
+                        if (isTypeSubtypeOf(candidate, constraint)) {
+                            return getIntersectionType([type, candidate]);
+                        }
+                    }
+                    return type;
+                };
             }
-            return getInferredTypes(context);
-        }
-        function getArrayifiedType(type) {
-            return type.flags & 1048576 /* Union */ ? mapType(type, getArrayifiedType) :
-                type.flags & (1 /* Any */ | 63176704 /* Instantiable */) || isMutableArrayOrTuple(type) ? type :
-                    isTupleType(type) ? createTupleType(getTypeArguments(type), type.target.minLength, type.target.hasRestElement, /*readonly*/ false, type.target.associatedNames) :
-                        createArrayType(getIndexedAccessType(type, numberType));
-        }
-        function getSpreadArgumentType(args, index, argCount, restType, context) {
-            if (index >= argCount - 1) {
-                var arg = args[argCount - 1];
-                if (isSpreadArgument(arg)) {
-                    // We are inferring from a spread expression in the last argument position, i.e. both the parameter
-                    // and the argument are ...x forms.
-                    return arg.kind === 220 /* SyntheticExpression */ ?
-                        createArrayType(arg.type) :
-                        getArrayifiedType(checkExpressionWithContextualType(arg.expression, restType, context, 0 /* Normal */));
+            function narrowBySwitchOnTypeOf(type, switchStatement, clauseStart, clauseEnd) {
+                var switchWitnesses = getSwitchClauseTypeOfWitnesses(switchStatement, /*retainDefault*/ true);
+                if (!switchWitnesses.length) {
+                    return type;
                 }
-            }
-            var types = [];
-            var spreadIndex = -1;
-            for (var i = index; i < argCount; i++) {
-                var contextualType = getIndexedAccessType(restType, getLiteralType(i - index));
-                var argType = checkExpressionWithContextualType(args[i], contextualType, context, 0 /* Normal */);
-                if (spreadIndex < 0 && isSpreadArgument(args[i])) {
-                    spreadIndex = i - index;
+                //  Equal start and end denotes implicit fallthrough; undefined marks explicit default clause
+                var defaultCaseLocation = ts.findIndex(switchWitnesses, function (elem) { return elem === undefined; });
+                var hasDefaultClause = clauseStart === clauseEnd || (defaultCaseLocation >= clauseStart && defaultCaseLocation < clauseEnd);
+                var clauseWitnesses;
+                var switchFacts;
+                if (defaultCaseLocation > -1) {
+                    // We no longer need the undefined denoting an explicit default case. Remove the undefined and
+                    // fix-up clauseStart and clauseEnd.  This means that we don't have to worry about undefined in the
+                    // witness array.
+                    var witnesses = switchWitnesses.filter(function (witness) { return witness !== undefined; });
+                    // The adjusted clause start and end after removing the `default` statement.
+                    var fixedClauseStart = defaultCaseLocation < clauseStart ? clauseStart - 1 : clauseStart;
+                    var fixedClauseEnd = defaultCaseLocation < clauseEnd ? clauseEnd - 1 : clauseEnd;
+                    clauseWitnesses = witnesses.slice(fixedClauseStart, fixedClauseEnd);
+                    switchFacts = getFactsFromTypeofSwitch(fixedClauseStart, fixedClauseEnd, witnesses, hasDefaultClause);
                 }
-                var hasPrimitiveContextualType = maybeTypeOfKind(contextualType, 131068 /* Primitive */ | 4194304 /* Index */);
-                types.push(hasPrimitiveContextualType ? getRegularTypeOfLiteralType(argType) : getWidenedLiteralType(argType));
-            }
-            return spreadIndex < 0 ?
-                createTupleType(types) :
-                createTupleType(ts.append(types.slice(0, spreadIndex), getUnionType(types.slice(spreadIndex))), spreadIndex, /*hasRestElement*/ true);
-        }
-        function checkTypeArguments(signature, typeArgumentNodes, reportErrors, headMessage) {
-            var isJavascript = ts.isInJSFile(signature.declaration);
-            var typeParameters = signature.typeParameters;
-            var typeArgumentTypes = fillMissingTypeArguments(ts.map(typeArgumentNodes, getTypeFromTypeNode), typeParameters, getMinTypeArgumentCount(typeParameters), isJavascript);
-            var mapper;
-            for (var i = 0; i < typeArgumentNodes.length; i++) {
-                ts.Debug.assert(typeParameters[i] !== undefined, "Should not call checkTypeArguments with too many type arguments");
-                var constraint = getConstraintOfTypeParameter(typeParameters[i]);
-                if (constraint) {
-                    var errorInfo = reportErrors && headMessage ? (function () { return ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1); }) : undefined;
-                    var typeArgumentHeadMessage = headMessage || ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1;
-                    if (!mapper) {
-                        mapper = createTypeMapper(typeParameters, typeArgumentTypes);
-                    }
-                    var typeArgument = typeArgumentTypes[i];
-                    if (!checkTypeAssignableTo(typeArgument, getTypeWithThisArgument(instantiateType(constraint, mapper), typeArgument), reportErrors ? typeArgumentNodes[i] : undefined, typeArgumentHeadMessage, errorInfo)) {
-                        return undefined;
-                    }
+                else {
+                    clauseWitnesses = switchWitnesses.slice(clauseStart, clauseEnd);
+                    switchFacts = getFactsFromTypeofSwitch(clauseStart, clauseEnd, switchWitnesses, hasDefaultClause);
                 }
+                if (hasDefaultClause) {
+                    return filterType(type, function (t) { return (getTypeFacts(t) & switchFacts) === switchFacts; });
+                }
+                /*
+                  The implied type is the raw type suggested by a
+                  value being caught in this clause.
+
+                  When the clause contains a default case we ignore
+                  the implied type and try to narrow using any facts
+                  we can learn: see `switchFacts`.
+
+                  Example:
+                  switch (typeof x) {
+                      case 'number':
+                      case 'string': break;
+                      default: break;
+                      case 'number':
+                      case 'boolean': break
+                  }
+
+                  In the first clause (case `number` and `string`) the
+                  implied type is number | string.
+
+                  In the default clause we de not compute an implied type.
+
+                  In the third clause (case `number` and `boolean`)
+                  the naive implied type is number | boolean, however
+                  we use the type facts to narrow the implied type to
+                  boolean. We know that number cannot be selected
+                  because it is caught in the first clause.
+                */
+                var impliedType = getTypeWithFacts(getUnionType(clauseWitnesses.map(function (text) { return getImpliedTypeFromTypeofGuard(type, text) || type; })), switchFacts);
+                return getTypeWithFacts(mapType(type, narrowUnionMemberByTypeof(impliedType)), switchFacts);
             }
-            return typeArgumentTypes;
-        }
-        function getJsxReferenceKind(node) {
-            if (isJsxIntrinsicIdentifier(node.tagName)) {
-                return 2 /* Mixed */;
-            }
-            var tagType = getApparentType(checkExpression(node.tagName));
-            if (ts.length(getSignaturesOfType(tagType, 1 /* Construct */))) {
-                return 0 /* Component */;
-            }
-            if (ts.length(getSignaturesOfType(tagType, 0 /* Call */))) {
-                return 1 /* Function */;
+            function isMatchingConstructorReference(expr) {
+                return (ts.isPropertyAccessExpression(expr) && ts.idText(expr.name) === "constructor" ||
+                    ts.isElementAccessExpression(expr) && ts.isStringLiteralLike(expr.argumentExpression) && expr.argumentExpression.text === "constructor") &&
+                    isMatchingReference(reference, expr.expression);
             }
-            return 2 /* Mixed */;
-        }
-        /**
-         * Check if the given signature can possibly be a signature called by the JSX opening-like element.
-         * @param node a JSX opening-like element we are trying to figure its call signature
-         * @param signature a candidate signature we are trying whether it is a call signature
-         * @param relation a relationship to check parameter and argument type
-         */
-        function checkApplicableSignatureForJsxOpeningLikeElement(node, signature, relation, checkMode, reportErrors, containingMessageChain, errorOutputContainer) {
-            // Stateless function components can have maximum of three arguments: "props", "context", and "updater".
-            // However "context" and "updater" are implicit and can't be specify by users. Only the first parameter, props,
-            // can be specified by users through attributes property.
-            var paramType = getEffectiveFirstArgumentForJsxSignature(signature, node);
-            var attributesType = checkExpressionWithContextualType(node.attributes, paramType, /*inferenceContext*/ undefined, checkMode);
-            return checkTagNameDoesNotExpectTooManyArguments() && checkTypeRelatedToAndOptionallyElaborate(attributesType, paramType, relation, reportErrors ? node.tagName : undefined, node.attributes, 
-            /*headMessage*/ undefined, containingMessageChain, errorOutputContainer);
-            function checkTagNameDoesNotExpectTooManyArguments() {
-                var _a;
-                var tagType = ts.isJsxOpeningElement(node) || ts.isJsxSelfClosingElement(node) && !isJsxIntrinsicIdentifier(node.tagName) ? checkExpression(node.tagName) : undefined;
-                if (!tagType) {
-                    return true;
+            function narrowTypeByConstructor(type, operator, identifier, assumeTrue) {
+                // Do not narrow when checking inequality.
+                if (assumeTrue ? (operator !== 34 /* EqualsEqualsToken */ && operator !== 36 /* EqualsEqualsEqualsToken */) : (operator !== 35 /* ExclamationEqualsToken */ && operator !== 37 /* ExclamationEqualsEqualsToken */)) {
+                    return type;
                 }
-                var tagCallSignatures = getSignaturesOfType(tagType, 0 /* Call */);
-                if (!ts.length(tagCallSignatures)) {
-                    return true;
+                // Get the type of the constructor identifier expression, if it is not a function then do not narrow.
+                var identifierType = getTypeOfExpression(identifier);
+                if (!isFunctionType(identifierType) && !isConstructorType(identifierType)) {
+                    return type;
                 }
-                var factory = getJsxFactoryEntity(node);
-                if (!factory) {
-                    return true;
+                // Get the prototype property of the type identifier so we can find out its type.
+                var prototypeProperty = getPropertyOfType(identifierType, "prototype");
+                if (!prototypeProperty) {
+                    return type;
                 }
-                var factorySymbol = resolveEntityName(factory, 111551 /* Value */, /*ignoreErrors*/ true, /*dontResolveAlias*/ false, node);
-                if (!factorySymbol) {
-                    return true;
+                // Get the type of the prototype, if it is undefined, or the global `Object` or `Function` types then do not narrow.
+                var prototypeType = getTypeOfSymbol(prototypeProperty);
+                var candidate = !isTypeAny(prototypeType) ? prototypeType : undefined;
+                if (!candidate || candidate === globalObjectType || candidate === globalFunctionType) {
+                    return type;
                 }
-                var factoryType = getTypeOfSymbol(factorySymbol);
-                var callSignatures = getSignaturesOfType(factoryType, 0 /* Call */);
-                if (!ts.length(callSignatures)) {
-                    return true;
+                // If the type that is being narrowed is `any` then just return the `candidate` type since every type is a subtype of `any`.
+                if (isTypeAny(type)) {
+                    return candidate;
                 }
-                var hasFirstParamSignatures = false;
-                var maxParamCount = 0;
-                // Check that _some_ first parameter expects a FC-like thing, and that some overload of the SFC expects an acceptable number of arguments
-                for (var _i = 0, callSignatures_1 = callSignatures; _i < callSignatures_1.length; _i++) {
-                    var sig = callSignatures_1[_i];
-                    var firstparam = getTypeAtPosition(sig, 0);
-                    var signaturesOfParam = getSignaturesOfType(firstparam, 0 /* Call */);
-                    if (!ts.length(signaturesOfParam))
-                        continue;
-                    for (var _b = 0, signaturesOfParam_1 = signaturesOfParam; _b < signaturesOfParam_1.length; _b++) {
-                        var paramSig = signaturesOfParam_1[_b];
-                        hasFirstParamSignatures = true;
-                        if (hasEffectiveRestParameter(paramSig)) {
-                            return true; // some signature has a rest param, so function components can have an arbitrary number of arguments
-                        }
-                        var paramCount = getParameterCount(paramSig);
-                        if (paramCount > maxParamCount) {
-                            maxParamCount = paramCount;
-                        }
+                // Filter out types that are not considered to be "constructed by" the `candidate` type.
+                return filterType(type, function (t) { return isConstructedBy(t, candidate); });
+                function isConstructedBy(source, target) {
+                    // If either the source or target type are a class type then we need to check that they are the same exact type.
+                    // This is because you may have a class `A` that defines some set of properties, and another class `B`
+                    // that defines the same set of properties as class `A`, in that case they are structurally the same
+                    // type, but when you do something like `instanceOfA.constructor === B` it will return false.
+                    if (source.flags & 524288 /* Object */ && ts.getObjectFlags(source) & 1 /* Class */ ||
+                        target.flags & 524288 /* Object */ && ts.getObjectFlags(target) & 1 /* Class */) {
+                        return source.symbol === target.symbol;
                     }
+                    // For all other types just check that the `source` type is a subtype of the `target` type.
+                    return isTypeSubtypeOf(source, target);
                 }
-                if (!hasFirstParamSignatures) {
-                    // Not a single signature had a first parameter which expected a signature - for back compat, and
-                    // to guard against generic factories which won't have signatures directly, do not error
-                    return true;
-                }
-                var absoluteMinArgCount = Infinity;
-                for (var _c = 0, tagCallSignatures_1 = tagCallSignatures; _c < tagCallSignatures_1.length; _c++) {
-                    var tagSig = tagCallSignatures_1[_c];
-                    var tagRequiredArgCount = getMinArgumentCount(tagSig);
-                    if (tagRequiredArgCount < absoluteMinArgCount) {
-                        absoluteMinArgCount = tagRequiredArgCount;
+            }
+            function narrowTypeByInstanceof(type, expr, assumeTrue) {
+                var left = getReferenceCandidate(expr.left);
+                if (!isMatchingReference(reference, left)) {
+                    if (assumeTrue && strictNullChecks && optionalChainContainsReference(left, reference)) {
+                        return getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
                     }
+                    return type;
                 }
-                if (absoluteMinArgCount <= maxParamCount) {
-                    return true; // some signature accepts the number of arguments the function component provides
+                // Check that right operand is a function type with a prototype property
+                var rightType = getTypeOfExpression(expr.right);
+                if (!isTypeDerivedFrom(rightType, globalFunctionType)) {
+                    return type;
                 }
-                if (reportErrors) {
-                    var diag = ts.createDiagnosticForNode(node.tagName, ts.Diagnostics.Tag_0_expects_at_least_1_arguments_but_the_JSX_factory_2_provides_at_most_3, ts.entityNameToString(node.tagName), absoluteMinArgCount, ts.entityNameToString(factory), maxParamCount);
-                    var tagNameDeclaration = (_a = getSymbolAtLocation(node.tagName)) === null || _a === void 0 ? void 0 : _a.valueDeclaration;
-                    if (tagNameDeclaration) {
-                        ts.addRelatedInfo(diag, ts.createDiagnosticForNode(tagNameDeclaration, ts.Diagnostics._0_is_declared_here, ts.entityNameToString(node.tagName)));
-                    }
-                    if (errorOutputContainer && errorOutputContainer.skipLogging) {
-                        (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
-                    }
-                    if (!errorOutputContainer.skipLogging) {
-                        diagnostics.add(diag);
+                var targetType;
+                var prototypeProperty = getPropertyOfType(rightType, "prototype");
+                if (prototypeProperty) {
+                    // Target type is type of the prototype property
+                    var prototypePropertyType = getTypeOfSymbol(prototypeProperty);
+                    if (!isTypeAny(prototypePropertyType)) {
+                        targetType = prototypePropertyType;
                     }
                 }
-                return false;
+                // Don't narrow from 'any' if the target type is exactly 'Object' or 'Function'
+                if (isTypeAny(type) && (targetType === globalObjectType || targetType === globalFunctionType)) {
+                    return type;
+                }
+                if (!targetType) {
+                    var constructSignatures = getSignaturesOfType(rightType, 1 /* Construct */);
+                    targetType = constructSignatures.length ?
+                        getUnionType(ts.map(constructSignatures, function (signature) { return getReturnTypeOfSignature(getErasedSignature(signature)); })) :
+                        emptyObjectType;
+                }
+                // We can't narrow a union based off instanceof without negated types see #31576 for more info
+                if (!assumeTrue && rightType.flags & 1048576 /* Union */) {
+                    var nonConstructorTypeInUnion = ts.find(rightType.types, function (t) { return !isConstructorType(t); });
+                    if (!nonConstructorTypeInUnion)
+                        return type;
+                }
+                return getNarrowedType(type, targetType, assumeTrue, isTypeDerivedFrom);
             }
-        }
-        function getSignatureApplicabilityError(node, args, signature, relation, checkMode, reportErrors, containingMessageChain) {
-            var errorOutputContainer = { errors: undefined, skipLogging: true };
-            if (ts.isJsxOpeningLikeElement(node)) {
-                if (!checkApplicableSignatureForJsxOpeningLikeElement(node, signature, relation, checkMode, reportErrors, containingMessageChain, errorOutputContainer)) {
-                    ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "jsx should have errors when reporting errors");
-                    return errorOutputContainer.errors || ts.emptyArray;
+            function getNarrowedType(type, candidate, assumeTrue, isRelated) {
+                if (!assumeTrue) {
+                    return filterType(type, function (t) { return !isRelated(t, candidate); });
                 }
-                return undefined;
+                // If the current type is a union type, remove all constituents that couldn't be instances of
+                // the candidate type. If one or more constituents remain, return a union of those.
+                if (type.flags & 1048576 /* Union */) {
+                    var assignableType = filterType(type, function (t) { return isRelated(t, candidate); });
+                    if (!(assignableType.flags & 131072 /* Never */)) {
+                        return assignableType;
+                    }
+                }
+                // If the candidate type is a subtype of the target type, narrow to the candidate type,
+                // if the target type is a subtype of the candidate type, narrow to the target type,
+                // otherwise, narrow to an intersection of the two types.
+                return isTypeSubtypeOf(candidate, type) ? candidate : isTypeSubtypeOf(type, candidate) ? type : getIntersectionType([type, candidate]);
             }
-            var thisType = getThisTypeOfSignature(signature);
-            if (thisType && thisType !== voidType && node.kind !== 197 /* NewExpression */) {
-                // If the called expression is not of the form `x.f` or `x["f"]`, then sourceType = voidType
-                // If the signature's 'this' type is voidType, then the check is skipped -- anything is compatible.
-                // If the expression is a new expression, then the check is skipped.
-                var thisArgumentNode = getThisArgumentOfCall(node);
-                var thisArgumentType = void 0;
-                if (thisArgumentNode) {
-                    thisArgumentType = checkExpression(thisArgumentNode);
-                    if (ts.isOptionalChainRoot(thisArgumentNode.parent)) {
-                        thisArgumentType = getNonNullableType(thisArgumentType);
+            function narrowTypeByCallExpression(type, callExpression, assumeTrue) {
+                if (hasMatchingArgument(callExpression, reference)) {
+                    var signature = assumeTrue || !ts.isCallChain(callExpression) ? getEffectsSignature(callExpression) : undefined;
+                    var predicate = signature && getTypePredicateOfSignature(signature);
+                    if (predicate && (predicate.kind === 0 /* This */ || predicate.kind === 1 /* Identifier */)) {
+                        return narrowTypeByTypePredicate(type, predicate, callExpression, assumeTrue);
                     }
-                    else if (ts.isOptionalChain(thisArgumentNode.parent)) {
-                        thisArgumentType = removeOptionalTypeMarker(thisArgumentType);
+                }
+                return type;
+            }
+            function narrowTypeByTypePredicate(type, predicate, callExpression, assumeTrue) {
+                // Don't narrow from 'any' if the predicate type is exactly 'Object' or 'Function'
+                if (predicate.type && !(isTypeAny(type) && (predicate.type === globalObjectType || predicate.type === globalFunctionType))) {
+                    var predicateArgument = getTypePredicateArgument(predicate, callExpression);
+                    if (predicateArgument) {
+                        if (isMatchingReference(reference, predicateArgument)) {
+                            return getNarrowedType(type, predicate.type, assumeTrue, isTypeSubtypeOf);
+                        }
+                        if (strictNullChecks && assumeTrue && optionalChainContainsReference(predicateArgument, reference) &&
+                            !(getTypeFacts(predicate.type) & 65536 /* EQUndefined */)) {
+                            type = getTypeWithFacts(type, 2097152 /* NEUndefinedOrNull */);
+                        }
+                        if (isMatchingReferenceDiscriminant(predicateArgument, type)) {
+                            return narrowTypeByDiscriminant(type, predicateArgument, function (t) { return getNarrowedType(t, predicate.type, assumeTrue, isTypeSubtypeOf); });
+                        }
                     }
                 }
-                else {
-                    thisArgumentType = voidType;
+                return type;
+            }
+            // Narrow the given type based on the given expression having the assumed boolean value. The returned type
+            // will be a subtype or the same type as the argument.
+            function narrowType(type, expr, assumeTrue) {
+                // for `a?.b`, we emulate a synthetic `a !== null && a !== undefined` condition for `a`
+                if (ts.isExpressionOfOptionalChainRoot(expr) ||
+                    ts.isBinaryExpression(expr.parent) && expr.parent.operatorToken.kind === 60 /* QuestionQuestionToken */ && expr.parent.left === expr) {
+                    return narrowTypeByOptionality(type, expr, assumeTrue);
                 }
-                var errorNode = reportErrors ? (thisArgumentNode || node) : undefined;
-                var headMessage_1 = ts.Diagnostics.The_this_context_of_type_0_is_not_assignable_to_method_s_this_of_type_1;
-                if (!checkTypeRelatedTo(thisArgumentType, thisType, relation, errorNode, headMessage_1, containingMessageChain, errorOutputContainer)) {
-                    ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "this parameter should have errors when reporting errors");
-                    return errorOutputContainer.errors || ts.emptyArray;
+                switch (expr.kind) {
+                    case 78 /* Identifier */:
+                    case 107 /* ThisKeyword */:
+                    case 105 /* SuperKeyword */:
+                    case 201 /* PropertyAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
+                        return narrowTypeByTruthiness(type, expr, assumeTrue);
+                    case 203 /* CallExpression */:
+                        return narrowTypeByCallExpression(type, expr, assumeTrue);
+                    case 207 /* ParenthesizedExpression */:
+                    case 225 /* NonNullExpression */:
+                        return narrowType(type, expr.expression, assumeTrue);
+                    case 216 /* BinaryExpression */:
+                        return narrowTypeByBinaryExpression(type, expr, assumeTrue);
+                    case 214 /* PrefixUnaryExpression */:
+                        if (expr.operator === 53 /* ExclamationToken */) {
+                            return narrowType(type, expr.operand, !assumeTrue);
+                        }
+                        break;
                 }
+                return type;
             }
-            var headMessage = ts.Diagnostics.Argument_of_type_0_is_not_assignable_to_parameter_of_type_1;
-            var restType = getNonArrayRestType(signature);
-            var argCount = restType ? Math.min(getParameterCount(signature) - 1, args.length) : args.length;
-            for (var i = 0; i < argCount; i++) {
-                var arg = args[i];
-                if (arg.kind !== 215 /* OmittedExpression */) {
-                    var paramType = getTypeAtPosition(signature, i);
-                    var argType = checkExpressionWithContextualType(arg, paramType, /*inferenceContext*/ undefined, checkMode);
-                    // If one or more arguments are still excluded (as indicated by CheckMode.SkipContextSensitive),
-                    // we obtain the regular type of any object literal arguments because we may not have inferred complete
-                    // parameter types yet and therefore excess property checks may yield false positives (see #17041).
-                    var checkArgType = checkMode & 4 /* SkipContextSensitive */ ? getRegularTypeOfObjectLiteral(argType) : argType;
-                    if (!checkTypeRelatedToAndOptionallyElaborate(checkArgType, paramType, relation, reportErrors ? arg : undefined, arg, headMessage, containingMessageChain, errorOutputContainer)) {
-                        ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "parameter should have errors when reporting errors");
-                        maybeAddMissingAwaitInfo(arg, checkArgType, paramType);
-                        return errorOutputContainer.errors || ts.emptyArray;
-                    }
+            function narrowTypeByOptionality(type, expr, assumePresent) {
+                if (isMatchingReference(reference, expr)) {
+                    return getTypeWithFacts(type, assumePresent ? 2097152 /* NEUndefinedOrNull */ : 262144 /* EQUndefinedOrNull */);
                 }
-            }
-            if (restType) {
-                var spreadType = getSpreadArgumentType(args, argCount, args.length, restType, /*context*/ undefined);
-                var errorNode = reportErrors ? argCount < args.length ? args[argCount] : node : undefined;
-                if (!checkTypeRelatedTo(spreadType, restType, relation, errorNode, headMessage, /*containingMessageChain*/ undefined, errorOutputContainer)) {
-                    ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "rest parameter should have errors when reporting errors");
-                    maybeAddMissingAwaitInfo(errorNode, spreadType, restType);
-                    return errorOutputContainer.errors || ts.emptyArray;
+                if (isMatchingReferenceDiscriminant(expr, type)) {
+                    return narrowTypeByDiscriminant(type, expr, function (t) { return getTypeWithFacts(t, assumePresent ? 2097152 /* NEUndefinedOrNull */ : 262144 /* EQUndefinedOrNull */); });
                 }
+                return type;
             }
-            return undefined;
-            function maybeAddMissingAwaitInfo(errorNode, source, target) {
-                if (errorNode && reportErrors && errorOutputContainer.errors && errorOutputContainer.errors.length) {
-                    // Bail if target is Promise-like---something else is wrong
-                    if (getAwaitedTypeOfPromise(target)) {
-                        return;
-                    }
-                    var awaitedTypeOfSource = getAwaitedTypeOfPromise(source);
-                    if (awaitedTypeOfSource && isTypeRelatedTo(awaitedTypeOfSource, target, relation)) {
-                        ts.addRelatedInfo(errorOutputContainer.errors[0], ts.createDiagnosticForNode(errorNode, ts.Diagnostics.Did_you_forget_to_use_await));
+        }
+        function getTypeOfSymbolAtLocation(symbol, location) {
+            symbol = symbol.exportSymbol || symbol;
+            // If we have an identifier or a property access at the given location, if the location is
+            // an dotted name expression, and if the location is not an assignment target, obtain the type
+            // of the expression (which will reflect control flow analysis). If the expression indeed
+            // resolved to the given symbol, return the narrowed type.
+            if (location.kind === 78 /* Identifier */) {
+                if (ts.isRightSideOfQualifiedNameOrPropertyAccess(location)) {
+                    location = location.parent;
+                }
+                if (ts.isExpressionNode(location) && !ts.isAssignmentTarget(location)) {
+                    var type = getTypeOfExpression(location);
+                    if (getExportSymbolOfValueSymbolIfExported(getNodeLinks(location).resolvedSymbol) === symbol) {
+                        return type;
                     }
                 }
             }
+            // The location isn't a reference to the given symbol, meaning we're being asked
+            // a hypothetical question of what type the symbol would have if there was a reference
+            // to it at the given location. Since we have no control flow information for the
+            // hypothetical reference (control flow information is created and attached by the
+            // binder), we simply return the declared type of the symbol.
+            return getTypeOfSymbol(symbol);
         }
-        /**
-         * Returns the this argument in calls like x.f(...) and x[f](...). Undefined otherwise.
-         */
-        function getThisArgumentOfCall(node) {
-            if (node.kind === 196 /* CallExpression */) {
-                var callee = ts.skipOuterExpressions(node.expression);
-                if (ts.isAccessExpression(callee)) {
-                    return callee.expression;
+        function getControlFlowContainer(node) {
+            return ts.findAncestor(node.parent, function (node) {
+                return ts.isFunctionLike(node) && !ts.getImmediatelyInvokedFunctionExpression(node) ||
+                    node.kind === 257 /* ModuleBlock */ ||
+                    node.kind === 297 /* SourceFile */ ||
+                    node.kind === 163 /* PropertyDeclaration */;
+            });
+        }
+        // Check if a parameter is assigned anywhere within its declaring function.
+        function isParameterAssigned(symbol) {
+            var func = ts.getRootDeclaration(symbol.valueDeclaration).parent;
+            var links = getNodeLinks(func);
+            if (!(links.flags & 8388608 /* AssignmentsMarked */)) {
+                links.flags |= 8388608 /* AssignmentsMarked */;
+                if (!hasParentWithAssignmentsMarked(func)) {
+                    markParameterAssignments(func);
                 }
             }
+            return symbol.isAssigned || false;
         }
-        function createSyntheticExpression(parent, type, isSpread) {
-            var result = ts.createNode(220 /* SyntheticExpression */, parent.pos, parent.end);
-            result.parent = parent;
-            result.type = type;
-            result.isSpread = isSpread || false;
-            return result;
+        function hasParentWithAssignmentsMarked(node) {
+            return !!ts.findAncestor(node.parent, function (node) { return ts.isFunctionLike(node) && !!(getNodeLinks(node).flags & 8388608 /* AssignmentsMarked */); });
         }
-        /**
-         * Returns the effective arguments for an expression that works like a function invocation.
-         */
-        function getEffectiveCallArguments(node) {
-            if (node.kind === 198 /* TaggedTemplateExpression */) {
-                var template = node.template;
-                var args_3 = [createSyntheticExpression(template, getGlobalTemplateStringsArrayType())];
-                if (template.kind === 211 /* TemplateExpression */) {
-                    ts.forEach(template.templateSpans, function (span) {
-                        args_3.push(span.expression);
-                    });
+        function markParameterAssignments(node) {
+            if (node.kind === 78 /* Identifier */) {
+                if (ts.isAssignmentTarget(node)) {
+                    var symbol = getResolvedSymbol(node);
+                    if (symbol.valueDeclaration && ts.getRootDeclaration(symbol.valueDeclaration).kind === 160 /* Parameter */) {
+                        symbol.isAssigned = true;
+                    }
                 }
-                return args_3;
             }
-            if (node.kind === 157 /* Decorator */) {
-                return getEffectiveDecoratorArguments(node);
+            else {
+                ts.forEachChild(node, markParameterAssignments);
             }
-            if (ts.isJsxOpeningLikeElement(node)) {
-                return node.attributes.properties.length > 0 || (ts.isJsxOpeningElement(node) && node.parent.children.length > 0) ? [node.attributes] : ts.emptyArray;
+        }
+        function isConstVariable(symbol) {
+            return symbol.flags & 3 /* Variable */ && (getDeclarationNodeFlagsFromSymbol(symbol) & 2 /* Const */) !== 0 && getTypeOfSymbol(symbol) !== autoArrayType;
+        }
+        /** remove undefined from the annotated type of a parameter when there is an initializer (that doesn't include undefined) */
+        function removeOptionalityFromDeclaredType(declaredType, declaration) {
+            if (pushTypeResolution(declaration.symbol, 2 /* DeclaredType */)) {
+                var annotationIncludesUndefined = strictNullChecks &&
+                    declaration.kind === 160 /* Parameter */ &&
+                    declaration.initializer &&
+                    getFalsyFlags(declaredType) & 32768 /* Undefined */ &&
+                    !(getFalsyFlags(checkExpression(declaration.initializer)) & 32768 /* Undefined */);
+                popTypeResolution();
+                return annotationIncludesUndefined ? getTypeWithFacts(declaredType, 524288 /* NEUndefined */) : declaredType;
             }
-            var args = node.arguments || ts.emptyArray;
-            var length = args.length;
-            if (length && isSpreadArgument(args[length - 1]) && getSpreadArgumentIndex(args) === length - 1) {
-                // We have a spread argument in the last position and no other spread arguments. If the type
-                // of the argument is a tuple type, spread the tuple elements into the argument list. We can
-                // call checkExpressionCached because spread expressions never have a contextual type.
-                var spreadArgument_1 = args[length - 1];
-                var type = flowLoopCount ? checkExpression(spreadArgument_1.expression) : checkExpressionCached(spreadArgument_1.expression);
-                if (isTupleType(type)) {
-                    var typeArguments = getTypeArguments(type);
-                    var restIndex_2 = type.target.hasRestElement ? typeArguments.length - 1 : -1;
-                    var syntheticArgs = ts.map(typeArguments, function (t, i) { return createSyntheticExpression(spreadArgument_1, t, /*isSpread*/ i === restIndex_2); });
-                    return ts.concatenate(args.slice(0, length - 1), syntheticArgs);
-                }
+            else {
+                reportCircularityError(declaration.symbol);
+                return declaredType;
             }
-            return args;
         }
-        /**
-         * Returns the synthetic argument list for a decorator invocation.
-         */
-        function getEffectiveDecoratorArguments(node) {
+        function isConstraintPosition(node) {
             var parent = node.parent;
-            var expr = node.expression;
-            switch (parent.kind) {
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                    // For a class decorator, the `target` is the type of the class (e.g. the
-                    // "static" or "constructor" side of the class).
-                    return [
-                        createSyntheticExpression(expr, getTypeOfSymbol(getSymbolOfNode(parent)))
-                    ];
-                case 156 /* Parameter */:
-                    // A parameter declaration decorator will have three arguments (see
-                    // `ParameterDecorator` in core.d.ts).
-                    var func = parent.parent;
-                    return [
-                        createSyntheticExpression(expr, parent.parent.kind === 162 /* Constructor */ ? getTypeOfSymbol(getSymbolOfNode(func)) : errorType),
-                        createSyntheticExpression(expr, anyType),
-                        createSyntheticExpression(expr, numberType)
-                    ];
-                case 159 /* PropertyDeclaration */:
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    // A method or accessor declaration decorator will have two or three arguments (see
-                    // `PropertyDecorator` and `MethodDecorator` in core.d.ts). If we are emitting decorators
-                    // for ES3, we will only pass two arguments.
-                    var hasPropDesc = parent.kind !== 159 /* PropertyDeclaration */ && languageVersion !== 0 /* ES3 */;
-                    return [
-                        createSyntheticExpression(expr, getParentTypeOfClassElement(parent)),
-                        createSyntheticExpression(expr, getClassElementPropertyKeyType(parent)),
-                        createSyntheticExpression(expr, hasPropDesc ? createTypedPropertyDescriptorType(getTypeOfNode(parent)) : anyType)
-                    ];
-            }
-            return ts.Debug.fail();
+            return parent.kind === 201 /* PropertyAccessExpression */ ||
+                parent.kind === 203 /* CallExpression */ && parent.expression === node ||
+                parent.kind === 202 /* ElementAccessExpression */ && parent.expression === node ||
+                parent.kind === 198 /* BindingElement */ && parent.name === node && !!parent.initializer;
         }
-        /**
-         * Returns the argument count for a decorator node that works like a function invocation.
-         */
-        function getDecoratorArgumentCount(node, signature) {
-            switch (node.parent.kind) {
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                    return 1;
-                case 159 /* PropertyDeclaration */:
-                    return 2;
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    // For ES3 or decorators with only two parameters we supply only two arguments
-                    return languageVersion === 0 /* ES3 */ || signature.parameters.length <= 2 ? 2 : 3;
-                case 156 /* Parameter */:
-                    return 3;
-                default:
-                    return ts.Debug.fail();
-            }
+        function typeHasNullableConstraint(type) {
+            return type.flags & 58982400 /* InstantiableNonPrimitive */ && maybeTypeOfKind(getBaseConstraintOfType(type) || unknownType, 98304 /* Nullable */);
         }
-        function getDiagnosticSpanForCallNode(node, doNotIncludeArguments) {
-            var start;
-            var length;
-            var sourceFile = ts.getSourceFileOfNode(node);
-            if (ts.isPropertyAccessExpression(node.expression)) {
-                var nameSpan = ts.getErrorSpanForNode(sourceFile, node.expression.name);
-                start = nameSpan.start;
-                length = doNotIncludeArguments ? nameSpan.length : node.end - start;
-            }
-            else {
-                var expressionSpan = ts.getErrorSpanForNode(sourceFile, node.expression);
-                start = expressionSpan.start;
-                length = doNotIncludeArguments ? expressionSpan.length : node.end - start;
+        function getConstraintForLocation(type, node) {
+            // When a node is the left hand expression of a property access, element access, or call expression,
+            // and the type of the node includes type variables with constraints that are nullable, we fetch the
+            // apparent type of the node *before* performing control flow analysis such that narrowings apply to
+            // the constraint type.
+            if (type && isConstraintPosition(node) && forEachType(type, typeHasNullableConstraint)) {
+                return mapType(getWidenedType(type), getBaseConstraintOrType);
             }
-            return { start: start, length: length, sourceFile: sourceFile };
+            return type;
         }
-        function getDiagnosticForCallNode(node, message, arg0, arg1, arg2, arg3) {
-            if (ts.isCallExpression(node)) {
-                var _a = getDiagnosticSpanForCallNode(node), sourceFile = _a.sourceFile, start = _a.start, length_5 = _a.length;
-                return ts.createFileDiagnostic(sourceFile, start, length_5, message, arg0, arg1, arg2, arg3);
-            }
-            else {
-                return ts.createDiagnosticForNode(node, message, arg0, arg1, arg2, arg3);
-            }
+        function isExportOrExportExpression(location) {
+            return !!ts.findAncestor(location, function (e) { return e.parent && ts.isExportAssignment(e.parent) && e.parent.expression === e && ts.isEntityNameExpression(e); });
         }
-        function getArgumentArityError(node, signatures, args) {
-            var min = Number.POSITIVE_INFINITY;
-            var max = Number.NEGATIVE_INFINITY;
-            var belowArgCount = Number.NEGATIVE_INFINITY;
-            var aboveArgCount = Number.POSITIVE_INFINITY;
-            var argCount = args.length;
-            var closestSignature;
-            for (var _i = 0, signatures_8 = signatures; _i < signatures_8.length; _i++) {
-                var sig = signatures_8[_i];
-                var minCount = getMinArgumentCount(sig);
-                var maxCount = getParameterCount(sig);
-                if (minCount < argCount && minCount > belowArgCount)
-                    belowArgCount = minCount;
-                if (argCount < maxCount && maxCount < aboveArgCount)
-                    aboveArgCount = maxCount;
-                if (minCount < min) {
-                    min = minCount;
-                    closestSignature = sig;
+        function markAliasReferenced(symbol, location) {
+            if (isNonLocalAlias(symbol, /*excludes*/ 111551 /* Value */) && !isInTypeQuery(location) && !getTypeOnlyAliasDeclaration(symbol)) {
+                if (compilerOptions.preserveConstEnums && isExportOrExportExpression(location) || !isConstEnumOrConstEnumOnlyModule(resolveAlias(symbol))) {
+                    markAliasSymbolAsReferenced(symbol);
+                }
+                else {
+                    markConstEnumAliasAsReferenced(symbol);
                 }
-                max = Math.max(max, maxCount);
             }
-            var hasRestParameter = ts.some(signatures, hasEffectiveRestParameter);
-            var paramRange = hasRestParameter ? min :
-                min < max ? min + "-" + max :
-                    min;
-            var hasSpreadArgument = getSpreadArgumentIndex(args) > -1;
-            if (argCount <= max && hasSpreadArgument) {
-                argCount--;
+        }
+        function checkIdentifier(node) {
+            var symbol = getResolvedSymbol(node);
+            if (symbol === unknownSymbol) {
+                return errorType;
             }
-            var spanArray;
-            var related;
-            var error = hasRestParameter || hasSpreadArgument ? hasRestParameter && hasSpreadArgument ? ts.Diagnostics.Expected_at_least_0_arguments_but_got_1_or_more :
-                hasRestParameter ? ts.Diagnostics.Expected_at_least_0_arguments_but_got_1 :
-                    ts.Diagnostics.Expected_0_arguments_but_got_1_or_more : ts.Diagnostics.Expected_0_arguments_but_got_1;
-            if (closestSignature && getMinArgumentCount(closestSignature) > argCount && closestSignature.declaration) {
-                var paramDecl = closestSignature.declaration.parameters[closestSignature.thisParameter ? argCount + 1 : argCount];
-                if (paramDecl) {
-                    related = ts.createDiagnosticForNode(paramDecl, ts.isBindingPattern(paramDecl.name) ? ts.Diagnostics.An_argument_matching_this_binding_pattern_was_not_provided : ts.Diagnostics.An_argument_for_0_was_not_provided, !paramDecl.name ? argCount : !ts.isBindingPattern(paramDecl.name) ? ts.idText(ts.getFirstIdentifier(paramDecl.name)) : undefined);
+            // As noted in ECMAScript 6 language spec, arrow functions never have an arguments objects.
+            // Although in down-level emit of arrow function, we emit it using function expression which means that
+            // arguments objects will be bound to the inner object; emitting arrow function natively in ES6, arguments objects
+            // will be bound to non-arrow function that contain this arrow function. This results in inconsistent behavior.
+            // To avoid that we will give an error to users if they use arguments objects in arrow function so that they
+            // can explicitly bound arguments objects
+            if (symbol === argumentsSymbol) {
+                var container = ts.getContainingFunction(node);
+                if (languageVersion < 2 /* ES2015 */) {
+                    if (container.kind === 209 /* ArrowFunction */) {
+                        error(node, ts.Diagnostics.The_arguments_object_cannot_be_referenced_in_an_arrow_function_in_ES3_and_ES5_Consider_using_a_standard_function_expression);
+                    }
+                    else if (ts.hasSyntacticModifier(container, 256 /* Async */)) {
+                        error(node, ts.Diagnostics.The_arguments_object_cannot_be_referenced_in_an_async_function_or_method_in_ES3_and_ES5_Consider_using_a_standard_function_or_method);
+                    }
                 }
+                getNodeLinks(container).flags |= 8192 /* CaptureArguments */;
+                return getTypeOfSymbol(symbol);
             }
-            if (min < argCount && argCount < max) {
-                return getDiagnosticForCallNode(node, ts.Diagnostics.No_overload_expects_0_arguments_but_overloads_do_exist_that_expect_either_1_or_2_arguments, argCount, belowArgCount, aboveArgCount);
+            // We should only mark aliases as referenced if there isn't a local value declaration
+            // for the symbol. Also, don't mark any property access expression LHS - checkPropertyAccessExpression will handle that
+            if (!(node.parent && ts.isPropertyAccessExpression(node.parent) && node.parent.expression === node)) {
+                markAliasReferenced(symbol, node);
             }
-            if (!hasSpreadArgument && argCount < min) {
-                var diagnostic_1 = getDiagnosticForCallNode(node, error, paramRange, argCount);
-                return related ? ts.addRelatedInfo(diagnostic_1, related) : diagnostic_1;
+            var localOrExportSymbol = getExportSymbolOfValueSymbolIfExported(symbol);
+            var sourceSymbol = localOrExportSymbol.flags & 2097152 /* Alias */ ? resolveAlias(localOrExportSymbol) : localOrExportSymbol;
+            if (getDeclarationNodeFlagsFromSymbol(sourceSymbol) & 134217728 /* Deprecated */ && isUncalledFunctionReference(node.parent, sourceSymbol)) {
+                errorOrSuggestion(/* isError */ false, node, ts.Diagnostics._0_is_deprecated, node.escapedText);
             }
-            if (hasRestParameter || hasSpreadArgument) {
-                spanArray = ts.createNodeArray(args);
-                if (hasSpreadArgument && argCount) {
-                    var nextArg = ts.elementAt(args, getSpreadArgumentIndex(args) + 1) || undefined;
-                    spanArray = ts.createNodeArray(args.slice(max > argCount && nextArg ? args.indexOf(nextArg) : Math.min(max, args.length - 1)));
+            var declaration = localOrExportSymbol.valueDeclaration;
+            if (localOrExportSymbol.flags & 32 /* Class */) {
+                // Due to the emit for class decorators, any reference to the class from inside of the class body
+                // must instead be rewritten to point to a temporary variable to avoid issues with the double-bind
+                // behavior of class names in ES6.
+                if (declaration.kind === 252 /* ClassDeclaration */
+                    && ts.nodeIsDecorated(declaration)) {
+                    var container = ts.getContainingClass(node);
+                    while (container !== undefined) {
+                        if (container === declaration && container.name !== node) {
+                            getNodeLinks(declaration).flags |= 16777216 /* ClassWithConstructorReference */;
+                            getNodeLinks(node).flags |= 33554432 /* ConstructorReferenceInClass */;
+                            break;
+                        }
+                        container = ts.getContainingClass(container);
+                    }
+                }
+                else if (declaration.kind === 221 /* ClassExpression */) {
+                    // When we emit a class expression with static members that contain a reference
+                    // to the constructor in the initializer, we will need to substitute that
+                    // binding with an alias as the class name is not in scope.
+                    var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
+                    while (container.kind !== 297 /* SourceFile */) {
+                        if (container.parent === declaration) {
+                            if (container.kind === 163 /* PropertyDeclaration */ && ts.hasSyntacticModifier(container, 32 /* Static */)) {
+                                getNodeLinks(declaration).flags |= 16777216 /* ClassWithConstructorReference */;
+                                getNodeLinks(node).flags |= 33554432 /* ConstructorReferenceInClass */;
+                            }
+                            break;
+                        }
+                        container = ts.getThisContainer(container, /*includeArrowFunctions*/ false);
+                    }
                 }
             }
-            else {
-                spanArray = ts.createNodeArray(args.slice(max));
+            checkNestedBlockScopedBinding(node, symbol);
+            var type = getConstraintForLocation(getTypeOfSymbol(localOrExportSymbol), node);
+            var assignmentKind = ts.getAssignmentTargetKind(node);
+            if (assignmentKind) {
+                if (!(localOrExportSymbol.flags & 3 /* Variable */) &&
+                    !(ts.isInJSFile(node) && localOrExportSymbol.flags & 512 /* ValueModule */)) {
+                    error(node, ts.Diagnostics.Cannot_assign_to_0_because_it_is_not_a_variable, symbolToString(symbol));
+                    return errorType;
+                }
+                if (isReadonlySymbol(localOrExportSymbol)) {
+                    if (localOrExportSymbol.flags & 3 /* Variable */) {
+                        error(node, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_constant, symbolToString(symbol));
+                    }
+                    else {
+                        error(node, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_read_only_property, symbolToString(symbol));
+                    }
+                    return errorType;
+                }
             }
-            spanArray.pos = ts.first(spanArray).pos;
-            spanArray.end = ts.last(spanArray).end;
-            if (spanArray.end === spanArray.pos) {
-                spanArray.end++;
+            var isAlias = localOrExportSymbol.flags & 2097152 /* Alias */;
+            // We only narrow variables and parameters occurring in a non-assignment position. For all other
+            // entities we simply return the declared type.
+            if (localOrExportSymbol.flags & 3 /* Variable */) {
+                if (assignmentKind === 1 /* Definite */) {
+                    return type;
+                }
             }
-            var diagnostic = ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), spanArray, error, paramRange, argCount);
-            return related ? ts.addRelatedInfo(diagnostic, related) : diagnostic;
-        }
-        function getTypeArgumentArityError(node, signatures, typeArguments) {
-            var argCount = typeArguments.length;
-            // No overloads exist
-            if (signatures.length === 1) {
-                var sig = signatures[0];
-                var min_1 = getMinTypeArgumentCount(sig.typeParameters);
-                var max = ts.length(sig.typeParameters);
-                return ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), typeArguments, ts.Diagnostics.Expected_0_type_arguments_but_got_1, min_1 < max ? min_1 + "-" + max : min_1, argCount);
+            else if (isAlias) {
+                declaration = ts.find(symbol.declarations, isSomeImportDeclaration);
             }
-            // Overloads exist
-            var belowArgCount = -Infinity;
-            var aboveArgCount = Infinity;
-            for (var _i = 0, signatures_9 = signatures; _i < signatures_9.length; _i++) {
-                var sig = signatures_9[_i];
-                var min_2 = getMinTypeArgumentCount(sig.typeParameters);
-                var max = ts.length(sig.typeParameters);
-                if (min_2 > argCount) {
-                    aboveArgCount = Math.min(aboveArgCount, min_2);
-                }
-                else if (max < argCount) {
-                    belowArgCount = Math.max(belowArgCount, max);
-                }
+            else {
+                return type;
             }
-            if (belowArgCount !== -Infinity && aboveArgCount !== Infinity) {
-                return ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), typeArguments, ts.Diagnostics.No_overload_expects_0_type_arguments_but_overloads_do_exist_that_expect_either_1_or_2_type_arguments, argCount, belowArgCount, aboveArgCount);
+            if (!declaration) {
+                return type;
             }
-            return ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), typeArguments, ts.Diagnostics.Expected_0_type_arguments_but_got_1, belowArgCount === -Infinity ? aboveArgCount : belowArgCount, argCount);
-        }
-        function resolveCall(node, signatures, candidatesOutArray, checkMode, callChainFlags, fallbackError) {
-            var isTaggedTemplate = node.kind === 198 /* TaggedTemplateExpression */;
-            var isDecorator = node.kind === 157 /* Decorator */;
-            var isJsxOpeningOrSelfClosingElement = ts.isJsxOpeningLikeElement(node);
-            var reportErrors = !candidatesOutArray;
-            var typeArguments;
-            if (!isDecorator) {
-                typeArguments = node.typeArguments;
-                // We already perform checking on the type arguments on the class declaration itself.
-                if (isTaggedTemplate || isJsxOpeningOrSelfClosingElement || node.expression.kind !== 102 /* SuperKeyword */) {
-                    ts.forEach(typeArguments, checkSourceElement);
-                }
+            // The declaration container is the innermost function that encloses the declaration of the variable
+            // or parameter. The flow container is the innermost function starting with which we analyze the control
+            // flow graph to determine the control flow based type.
+            var isParameter = ts.getRootDeclaration(declaration).kind === 160 /* Parameter */;
+            var declarationContainer = getControlFlowContainer(declaration);
+            var flowContainer = getControlFlowContainer(node);
+            var isOuterVariable = flowContainer !== declarationContainer;
+            var isSpreadDestructuringAssignmentTarget = node.parent && node.parent.parent && ts.isSpreadAssignment(node.parent) && isDestructuringAssignmentTarget(node.parent.parent);
+            var isModuleExports = symbol.flags & 134217728 /* ModuleExports */;
+            // When the control flow originates in a function expression or arrow function and we are referencing
+            // a const variable or parameter from an outer function, we extend the origin of the control flow
+            // analysis to include the immediately enclosing function.
+            while (flowContainer !== declarationContainer && (flowContainer.kind === 208 /* FunctionExpression */ ||
+                flowContainer.kind === 209 /* ArrowFunction */ || ts.isObjectLiteralOrClassExpressionMethod(flowContainer)) &&
+                (isConstVariable(localOrExportSymbol) || isParameter && !isParameterAssigned(localOrExportSymbol))) {
+                flowContainer = getControlFlowContainer(flowContainer);
             }
-            var candidates = candidatesOutArray || [];
-            // reorderCandidates fills up the candidates array directly
-            reorderCandidates(signatures, candidates, callChainFlags);
-            if (!candidates.length) {
-                if (reportErrors) {
-                    diagnostics.add(getDiagnosticForCallNode(node, ts.Diagnostics.Call_target_does_not_contain_any_signatures));
+            // We only look for uninitialized variables in strict null checking mode, and only when we can analyze
+            // the entire control flow graph from the variable's declaration (i.e. when the flow container and
+            // declaration container are the same).
+            var assumeInitialized = isParameter || isAlias || isOuterVariable || isSpreadDestructuringAssignmentTarget || isModuleExports || ts.isBindingElement(declaration) ||
+                type !== autoType && type !== autoArrayType && (!strictNullChecks || (type.flags & (3 /* AnyOrUnknown */ | 16384 /* Void */)) !== 0 ||
+                    isInTypeQuery(node) || node.parent.kind === 270 /* ExportSpecifier */) ||
+                node.parent.kind === 225 /* NonNullExpression */ ||
+                declaration.kind === 249 /* VariableDeclaration */ && declaration.exclamationToken ||
+                declaration.flags & 8388608 /* Ambient */;
+            var initialType = assumeInitialized ? (isParameter ? removeOptionalityFromDeclaredType(type, declaration) : type) :
+                type === autoType || type === autoArrayType ? undefinedType :
+                    getOptionalType(type);
+            var flowType = getFlowTypeOfReference(node, type, initialType, flowContainer, !assumeInitialized);
+            // A variable is considered uninitialized when it is possible to analyze the entire control flow graph
+            // from declaration to use, and when the variable's declared type doesn't include undefined but the
+            // control flow based type does include undefined.
+            if (!isEvolvingArrayOperationTarget(node) && (type === autoType || type === autoArrayType)) {
+                if (flowType === autoType || flowType === autoArrayType) {
+                    if (noImplicitAny) {
+                        error(ts.getNameOfDeclaration(declaration), ts.Diagnostics.Variable_0_implicitly_has_type_1_in_some_locations_where_its_type_cannot_be_determined, symbolToString(symbol), typeToString(flowType));
+                        error(node, ts.Diagnostics.Variable_0_implicitly_has_an_1_type, symbolToString(symbol), typeToString(flowType));
+                    }
+                    return convertAutoToAny(flowType);
                 }
-                return resolveErrorCall(node);
             }
-            var args = getEffectiveCallArguments(node);
-            // The excludeArgument array contains true for each context sensitive argument (an argument
-            // is context sensitive it is susceptible to a one-time permanent contextual typing).
-            //
-            // The idea is that we will perform type argument inference & assignability checking once
-            // without using the susceptible parameters that are functions, and once more for those
-            // parameters, contextually typing each as we go along.
-            //
-            // For a tagged template, then the first argument be 'undefined' if necessary because it
-            // represents a TemplateStringsArray.
-            //
-            // For a decorator, no arguments are susceptible to contextual typing due to the fact
-            // decorators are applied to a declaration by the emitter, and not to an expression.
-            var isSingleNonGenericCandidate = candidates.length === 1 && !candidates[0].typeParameters;
-            var argCheckMode = !isDecorator && !isSingleNonGenericCandidate && ts.some(args, isContextSensitive) ? 4 /* SkipContextSensitive */ : 0 /* Normal */;
-            // The following variables are captured and modified by calls to chooseOverload.
-            // If overload resolution or type argument inference fails, we want to report the
-            // best error possible. The best error is one which says that an argument was not
-            // assignable to a parameter. This implies that everything else about the overload
-            // was fine. So if there is any overload that is only incorrect because of an
-            // argument, we will report an error on that one.
-            //
-            //     function foo(s: string): void;
-            //     function foo(n: number): void; // Report argument error on this overload
-            //     function foo(): void;
-            //     foo(true);
-            //
-            // If none of the overloads even made it that far, there are two possibilities.
-            // There was a problem with type arguments for some overload, in which case
-            // report an error on that. Or none of the overloads even had correct arity,
-            // in which case give an arity error.
-            //
-            //     function foo<T extends string>(x: T): void; // Report type argument error
-            //     function foo(): void;
-            //     foo<number>(0);
-            //
-            var candidatesForArgumentError;
-            var candidateForArgumentArityError;
-            var candidateForTypeArgumentError;
-            var result;
-            // If we are in signature help, a trailing comma indicates that we intend to provide another argument,
-            // so we will only accept overloads with arity at least 1 higher than the current number of provided arguments.
-            var signatureHelpTrailingComma = !!(checkMode & 16 /* IsForSignatureHelp */) && node.kind === 196 /* CallExpression */ && node.arguments.hasTrailingComma;
-            // Section 4.12.1:
-            // if the candidate list contains one or more signatures for which the type of each argument
-            // expression is a subtype of each corresponding parameter type, the return type of the first
-            // of those signatures becomes the return type of the function call.
-            // Otherwise, the return type of the first signature in the candidate list becomes the return
-            // type of the function call.
-            //
-            // Whether the call is an error is determined by assignability of the arguments. The subtype pass
-            // is just important for choosing the best signature. So in the case where there is only one
-            // signature, the subtype pass is useless. So skipping it is an optimization.
-            if (candidates.length > 1) {
-                result = chooseOverload(candidates, subtypeRelation, signatureHelpTrailingComma);
+            else if (!assumeInitialized && !(getFalsyFlags(type) & 32768 /* Undefined */) && getFalsyFlags(flowType) & 32768 /* Undefined */) {
+                error(node, ts.Diagnostics.Variable_0_is_used_before_being_assigned, symbolToString(symbol));
+                // Return the declared type to reduce follow-on errors
+                return type;
             }
-            if (!result) {
-                result = chooseOverload(candidates, assignableRelation, signatureHelpTrailingComma);
+            return assignmentKind ? getBaseTypeOfLiteralType(flowType) : flowType;
+        }
+        function isInsideFunction(node, threshold) {
+            return !!ts.findAncestor(node, function (n) { return n === threshold ? "quit" : ts.isFunctionLike(n); });
+        }
+        function getPartOfForStatementContainingNode(node, container) {
+            return ts.findAncestor(node, function (n) { return n === container ? "quit" : n === container.initializer || n === container.condition || n === container.incrementor || n === container.statement; });
+        }
+        function checkNestedBlockScopedBinding(node, symbol) {
+            if (languageVersion >= 2 /* ES2015 */ ||
+                (symbol.flags & (2 /* BlockScopedVariable */ | 32 /* Class */)) === 0 ||
+                ts.isSourceFile(symbol.valueDeclaration) ||
+                symbol.valueDeclaration.parent.kind === 287 /* CatchClause */) {
+                return;
             }
-            if (result) {
-                return result;
+            // 1. walk from the use site up to the declaration and check
+            // if there is anything function like between declaration and use-site (is binding/class is captured in function).
+            // 2. walk from the declaration up to the boundary of lexical environment and check
+            // if there is an iteration statement in between declaration and boundary (is binding/class declared inside iteration statement)
+            var container = ts.getEnclosingBlockScopeContainer(symbol.valueDeclaration);
+            var usedInFunction = isInsideFunction(node.parent, container);
+            var current = container;
+            var containedInIterationStatement = false;
+            while (current && !ts.nodeStartsNewLexicalEnvironment(current)) {
+                if (ts.isIterationStatement(current, /*lookInLabeledStatements*/ false)) {
+                    containedInIterationStatement = true;
+                    break;
+                }
+                current = current.parent;
             }
-            // No signatures were applicable. Now report errors based on the last applicable signature with
-            // no arguments excluded from assignability checks.
-            // If candidate is undefined, it means that no candidates had a suitable arity. In that case,
-            // skip the checkApplicableSignature check.
-            if (reportErrors) {
-                if (candidatesForArgumentError) {
-                    if (candidatesForArgumentError.length === 1 || candidatesForArgumentError.length > 3) {
-                        var last_2 = candidatesForArgumentError[candidatesForArgumentError.length - 1];
-                        var chain_1;
-                        if (candidatesForArgumentError.length > 3) {
-                            chain_1 = ts.chainDiagnosticMessages(chain_1, ts.Diagnostics.The_last_overload_gave_the_following_error);
-                            chain_1 = ts.chainDiagnosticMessages(chain_1, ts.Diagnostics.No_overload_matches_this_call);
-                        }
-                        var diags = getSignatureApplicabilityError(node, args, last_2, assignableRelation, 0 /* Normal */, /*reportErrors*/ true, function () { return chain_1; });
-                        if (diags) {
-                            for (var _i = 0, diags_1 = diags; _i < diags_1.length; _i++) {
-                                var d = diags_1[_i];
-                                if (last_2.declaration && candidatesForArgumentError.length > 3) {
-                                    ts.addRelatedInfo(d, ts.createDiagnosticForNode(last_2.declaration, ts.Diagnostics.The_last_overload_is_declared_here));
+            if (containedInIterationStatement) {
+                if (usedInFunction) {
+                    // mark iteration statement as containing block-scoped binding captured in some function
+                    var capturesBlockScopeBindingInLoopBody = true;
+                    if (ts.isForStatement(container)) {
+                        var varDeclList = ts.getAncestor(symbol.valueDeclaration, 250 /* VariableDeclarationList */);
+                        if (varDeclList && varDeclList.parent === container) {
+                            var part = getPartOfForStatementContainingNode(node.parent, container);
+                            if (part) {
+                                var links = getNodeLinks(part);
+                                links.flags |= 131072 /* ContainsCapturedBlockScopeBinding */;
+                                var capturedBindings = links.capturedBlockScopeBindings || (links.capturedBlockScopeBindings = []);
+                                ts.pushIfUnique(capturedBindings, symbol);
+                                if (part === container.initializer) {
+                                    capturesBlockScopeBindingInLoopBody = false; // Initializer is outside of loop body
                                 }
-                                diagnostics.add(d);
                             }
                         }
-                        else {
-                            ts.Debug.fail("No error for last overload signature");
-                        }
                     }
-                    else {
-                        var allDiagnostics = [];
-                        var max = 0;
-                        var min_3 = Number.MAX_VALUE;
-                        var minIndex = 0;
-                        var i_1 = 0;
-                        var _loop_17 = function (c) {
-                            var chain_2 = function () { return ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.Overload_0_of_1_2_gave_the_following_error, i_1 + 1, candidates.length, signatureToString(c)); };
-                            var diags_2 = getSignatureApplicabilityError(node, args, c, assignableRelation, 0 /* Normal */, /*reportErrors*/ true, chain_2);
-                            if (diags_2) {
-                                if (diags_2.length <= min_3) {
-                                    min_3 = diags_2.length;
-                                    minIndex = i_1;
-                                }
-                                max = Math.max(max, diags_2.length);
-                                allDiagnostics.push(diags_2);
-                            }
-                            else {
-                                ts.Debug.fail("No error for 3 or fewer overload signatures");
-                            }
-                            i_1++;
-                        };
-                        for (var _a = 0, candidatesForArgumentError_1 = candidatesForArgumentError; _a < candidatesForArgumentError_1.length; _a++) {
-                            var c = candidatesForArgumentError_1[_a];
-                            _loop_17(c);
-                        }
-                        var diags_3 = max > 1 ? allDiagnostics[minIndex] : ts.flatten(allDiagnostics);
-                        ts.Debug.assert(diags_3.length > 0, "No errors reported for 3 or fewer overload signatures");
-                        var chain = ts.chainDiagnosticMessages(ts.map(diags_3, function (d) { return typeof d.messageText === "string" ? d : d.messageText; }), ts.Diagnostics.No_overload_matches_this_call);
-                        var related = ts.flatMap(diags_3, function (d) { return d.relatedInformation; });
-                        if (ts.every(diags_3, function (d) { return d.start === diags_3[0].start && d.length === diags_3[0].length && d.file === diags_3[0].file; })) {
-                            var _b = diags_3[0], file = _b.file, start = _b.start, length_6 = _b.length;
-                            diagnostics.add({ file: file, start: start, length: length_6, code: chain.code, category: chain.category, messageText: chain, relatedInformation: related });
-                        }
-                        else {
-                            diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(node, chain, related));
-                        }
+                    if (capturesBlockScopeBindingInLoopBody) {
+                        getNodeLinks(current).flags |= 65536 /* LoopWithCapturedBlockScopedBinding */;
                     }
                 }
-                else if (candidateForArgumentArityError) {
-                    diagnostics.add(getArgumentArityError(node, [candidateForArgumentArityError], args));
-                }
-                else if (candidateForTypeArgumentError) {
-                    checkTypeArguments(candidateForTypeArgumentError, node.typeArguments, /*reportErrors*/ true, fallbackError);
-                }
-                else {
-                    var signaturesWithCorrectTypeArgumentArity = ts.filter(signatures, function (s) { return hasCorrectTypeArgumentArity(s, typeArguments); });
-                    if (signaturesWithCorrectTypeArgumentArity.length === 0) {
-                        diagnostics.add(getTypeArgumentArityError(node, signatures, typeArguments));
-                    }
-                    else if (!isDecorator) {
-                        diagnostics.add(getArgumentArityError(node, signaturesWithCorrectTypeArgumentArity, args));
-                    }
-                    else if (fallbackError) {
-                        diagnostics.add(getDiagnosticForCallNode(node, fallbackError));
+                // mark variables that are declared in loop initializer and reassigned inside the body of ForStatement.
+                // if body of ForStatement will be converted to function then we'll need a extra machinery to propagate reassigned values back.
+                if (ts.isForStatement(container)) {
+                    var varDeclList = ts.getAncestor(symbol.valueDeclaration, 250 /* VariableDeclarationList */);
+                    if (varDeclList && varDeclList.parent === container && isAssignedInBodyOfForStatement(node, container)) {
+                        getNodeLinks(symbol.valueDeclaration).flags |= 4194304 /* NeedsLoopOutParameter */;
                     }
                 }
+                // set 'declared inside loop' bit on the block-scoped binding
+                getNodeLinks(symbol.valueDeclaration).flags |= 524288 /* BlockScopedBindingInLoop */;
             }
-            return getCandidateForOverloadFailure(node, candidates, args, !!candidatesOutArray);
-            function chooseOverload(candidates, relation, signatureHelpTrailingComma) {
-                if (signatureHelpTrailingComma === void 0) { signatureHelpTrailingComma = false; }
-                candidatesForArgumentError = undefined;
-                candidateForArgumentArityError = undefined;
-                candidateForTypeArgumentError = undefined;
-                if (isSingleNonGenericCandidate) {
-                    var candidate = candidates[0];
-                    if (ts.some(typeArguments) || !hasCorrectArity(node, args, candidate, signatureHelpTrailingComma)) {
-                        return undefined;
-                    }
-                    if (getSignatureApplicabilityError(node, args, candidate, relation, 0 /* Normal */, /*reportErrors*/ false, /*containingMessageChain*/ undefined)) {
-                        candidatesForArgumentError = [candidate];
-                        return undefined;
-                    }
-                    return candidate;
-                }
-                for (var candidateIndex = 0; candidateIndex < candidates.length; candidateIndex++) {
-                    var candidate = candidates[candidateIndex];
-                    if (!hasCorrectTypeArgumentArity(candidate, typeArguments) || !hasCorrectArity(node, args, candidate, signatureHelpTrailingComma)) {
-                        continue;
-                    }
-                    var checkCandidate = void 0;
-                    var inferenceContext = void 0;
-                    if (candidate.typeParameters) {
-                        var typeArgumentTypes = void 0;
-                        if (ts.some(typeArguments)) {
-                            typeArgumentTypes = checkTypeArguments(candidate, typeArguments, /*reportErrors*/ false);
-                            if (!typeArgumentTypes) {
-                                candidateForTypeArgumentError = candidate;
-                                continue;
-                            }
-                        }
-                        else {
-                            inferenceContext = createInferenceContext(candidate.typeParameters, candidate, /*flags*/ ts.isInJSFile(node) ? 2 /* AnyDefault */ : 0 /* None */);
-                            typeArgumentTypes = inferTypeArguments(node, candidate, args, argCheckMode | 8 /* SkipGenericFunctions */, inferenceContext);
-                            argCheckMode |= inferenceContext.flags & 4 /* SkippedGenericFunction */ ? 8 /* SkipGenericFunctions */ : 0 /* Normal */;
-                        }
-                        checkCandidate = getSignatureInstantiation(candidate, typeArgumentTypes, ts.isInJSFile(candidate.declaration), inferenceContext && inferenceContext.inferredTypeParameters);
-                        // If the original signature has a generic rest type, instantiation may produce a
-                        // signature with different arity and we need to perform another arity check.
-                        if (getNonArrayRestType(candidate) && !hasCorrectArity(node, args, checkCandidate, signatureHelpTrailingComma)) {
-                            candidateForArgumentArityError = checkCandidate;
-                            continue;
-                        }
-                    }
-                    else {
-                        checkCandidate = candidate;
-                    }
-                    if (getSignatureApplicabilityError(node, args, checkCandidate, relation, argCheckMode, /*reportErrors*/ false, /*containingMessageChain*/ undefined)) {
-                        // Give preference to error candidates that have no rest parameters (as they are more specific)
-                        (candidatesForArgumentError || (candidatesForArgumentError = [])).push(checkCandidate);
-                        continue;
-                    }
-                    if (argCheckMode) {
-                        // If one or more context sensitive arguments were excluded, we start including
-                        // them now (and keeping do so for any subsequent candidates) and perform a second
-                        // round of type inference and applicability checking for this particular candidate.
-                        argCheckMode = 0 /* Normal */;
-                        if (inferenceContext) {
-                            var typeArgumentTypes = inferTypeArguments(node, candidate, args, argCheckMode, inferenceContext);
-                            checkCandidate = getSignatureInstantiation(candidate, typeArgumentTypes, ts.isInJSFile(candidate.declaration), inferenceContext && inferenceContext.inferredTypeParameters);
-                            // If the original signature has a generic rest type, instantiation may produce a
-                            // signature with different arity and we need to perform another arity check.
-                            if (getNonArrayRestType(candidate) && !hasCorrectArity(node, args, checkCandidate, signatureHelpTrailingComma)) {
-                                candidateForArgumentArityError = checkCandidate;
-                                continue;
-                            }
-                        }
-                        if (getSignatureApplicabilityError(node, args, checkCandidate, relation, argCheckMode, /*reportErrors*/ false, /*containingMessageChain*/ undefined)) {
-                            // Give preference to error candidates that have no rest parameters (as they are more specific)
-                            (candidatesForArgumentError || (candidatesForArgumentError = [])).push(checkCandidate);
-                            continue;
-                        }
-                    }
-                    candidates[candidateIndex] = checkCandidate;
-                    return checkCandidate;
-                }
-                return undefined;
+            if (usedInFunction) {
+                getNodeLinks(symbol.valueDeclaration).flags |= 262144 /* CapturedBlockScopedBinding */;
             }
         }
-        // No signature was applicable. We have already reported the errors for the invalid signature.
-        function getCandidateForOverloadFailure(node, candidates, args, hasCandidatesOutArray) {
-            ts.Debug.assert(candidates.length > 0); // Else should not have called this.
-            checkNodeDeferred(node);
-            // Normally we will combine overloads. Skip this if they have type parameters since that's hard to combine.
-            // Don't do this if there is a `candidatesOutArray`,
-            // because then we want the chosen best candidate to be one of the overloads, not a combination.
-            return hasCandidatesOutArray || candidates.length === 1 || candidates.some(function (c) { return !!c.typeParameters; })
-                ? pickLongestCandidateSignature(node, candidates, args)
-                : createUnionOfSignaturesForOverloadFailure(candidates);
+        function isBindingCapturedByNode(node, decl) {
+            var links = getNodeLinks(node);
+            return !!links && ts.contains(links.capturedBlockScopeBindings, getSymbolOfNode(decl));
         }
-        function createUnionOfSignaturesForOverloadFailure(candidates) {
-            var thisParameters = ts.mapDefined(candidates, function (c) { return c.thisParameter; });
-            var thisParameter;
-            if (thisParameters.length) {
-                thisParameter = createCombinedSymbolFromTypes(thisParameters, thisParameters.map(getTypeOfParameter));
+        function isAssignedInBodyOfForStatement(node, container) {
+            // skip parenthesized nodes
+            var current = node;
+            while (current.parent.kind === 207 /* ParenthesizedExpression */) {
+                current = current.parent;
             }
-            var _a = ts.minAndMax(candidates, getNumNonRestParameters), minArgumentCount = _a.min, maxNonRestParam = _a.max;
-            var parameters = [];
-            var _loop_18 = function (i) {
-                var symbols = ts.mapDefined(candidates, function (s) { return signatureHasRestParameter(s) ?
-                    i < s.parameters.length - 1 ? s.parameters[i] : ts.last(s.parameters) :
-                    i < s.parameters.length ? s.parameters[i] : undefined; });
-                ts.Debug.assert(symbols.length !== 0);
-                parameters.push(createCombinedSymbolFromTypes(symbols, ts.mapDefined(candidates, function (candidate) { return tryGetTypeAtPosition(candidate, i); })));
-            };
-            for (var i = 0; i < maxNonRestParam; i++) {
-                _loop_18(i);
+            // check if node is used as LHS in some assignment expression
+            var isAssigned = false;
+            if (ts.isAssignmentTarget(current)) {
+                isAssigned = true;
             }
-            var restParameterSymbols = ts.mapDefined(candidates, function (c) { return signatureHasRestParameter(c) ? ts.last(c.parameters) : undefined; });
-            var flags = 0 /* None */;
-            if (restParameterSymbols.length !== 0) {
-                var type = createArrayType(getUnionType(ts.mapDefined(candidates, tryGetRestTypeOfSignature), 2 /* Subtype */));
-                parameters.push(createCombinedSymbolForOverloadFailure(restParameterSymbols, type));
-                flags |= 1 /* HasRestParameter */;
+            else if ((current.parent.kind === 214 /* PrefixUnaryExpression */ || current.parent.kind === 215 /* PostfixUnaryExpression */)) {
+                var expr = current.parent;
+                isAssigned = expr.operator === 45 /* PlusPlusToken */ || expr.operator === 46 /* MinusMinusToken */;
             }
-            if (candidates.some(signatureHasLiteralTypes)) {
-                flags |= 2 /* HasLiteralTypes */;
+            if (!isAssigned) {
+                return false;
             }
-            return createSignature(candidates[0].declaration, 
-            /*typeParameters*/ undefined, // Before calling this we tested for `!candidates.some(c => !!c.typeParameters)`.
-            thisParameter, parameters, 
-            /*resolvedReturnType*/ getIntersectionType(candidates.map(getReturnTypeOfSignature)), 
-            /*typePredicate*/ undefined, minArgumentCount, flags);
+            // at this point we know that node is the target of assignment
+            // now check that modification happens inside the statement part of the ForStatement
+            return !!ts.findAncestor(current, function (n) { return n === container ? "quit" : n === container.statement; });
         }
-        function getNumNonRestParameters(signature) {
-            var numParams = signature.parameters.length;
-            return signatureHasRestParameter(signature) ? numParams - 1 : numParams;
+        function captureLexicalThis(node, container) {
+            getNodeLinks(node).flags |= 2 /* LexicalThis */;
+            if (container.kind === 163 /* PropertyDeclaration */ || container.kind === 166 /* Constructor */) {
+                var classNode = container.parent;
+                getNodeLinks(classNode).flags |= 4 /* CaptureThis */;
+            }
+            else {
+                getNodeLinks(container).flags |= 4 /* CaptureThis */;
+            }
         }
-        function createCombinedSymbolFromTypes(sources, types) {
-            return createCombinedSymbolForOverloadFailure(sources, getUnionType(types, 2 /* Subtype */));
+        function findFirstSuperCall(node) {
+            return ts.isSuperCall(node) ? node :
+                ts.isFunctionLike(node) ? undefined :
+                    ts.forEachChild(node, findFirstSuperCall);
         }
-        function createCombinedSymbolForOverloadFailure(sources, type) {
-            // This function is currently only used for erroneous overloads, so it's good enough to just use the first source.
-            return createSymbolWithType(ts.first(sources), type);
+        /**
+         * Check if the given class-declaration extends null then return true.
+         * Otherwise, return false
+         * @param classDecl a class declaration to check if it extends null
+         */
+        function classDeclarationExtendsNull(classDecl) {
+            var classSymbol = getSymbolOfNode(classDecl);
+            var classInstanceType = getDeclaredTypeOfSymbol(classSymbol);
+            var baseConstructorType = getBaseConstructorTypeOfClass(classInstanceType);
+            return baseConstructorType === nullWideningType;
         }
-        function pickLongestCandidateSignature(node, candidates, args) {
-            // Pick the longest signature. This way we can get a contextual type for cases like:
-            //     declare function f(a: { xa: number; xb: number; }, b: number);
-            //     f({ |
-            // Also, use explicitly-supplied type arguments if they are provided, so we can get a contextual signature in cases like:
-            //     declare function f<T>(k: keyof T);
-            //     f<Foo>("
-            var bestIndex = getLongestCandidateIndex(candidates, apparentArgumentCount === undefined ? args.length : apparentArgumentCount);
-            var candidate = candidates[bestIndex];
-            var typeParameters = candidate.typeParameters;
-            if (!typeParameters) {
-                return candidate;
+        function checkThisBeforeSuper(node, container, diagnosticMessage) {
+            var containingClassDecl = container.parent;
+            var baseTypeNode = ts.getClassExtendsHeritageElement(containingClassDecl);
+            // If a containing class does not have extends clause or the class extends null
+            // skip checking whether super statement is called before "this" accessing.
+            if (baseTypeNode && !classDeclarationExtendsNull(containingClassDecl)) {
+                if (node.flowNode && !isPostSuperFlowNode(node.flowNode, /*noCacheCheck*/ false)) {
+                    error(node, diagnosticMessage);
+                }
             }
-            var typeArgumentNodes = callLikeExpressionMayHaveTypeArguments(node) ? node.typeArguments : undefined;
-            var instantiated = typeArgumentNodes
-                ? createSignatureInstantiation(candidate, getTypeArgumentsFromNodes(typeArgumentNodes, typeParameters, ts.isInJSFile(node)))
-                : inferSignatureInstantiationForOverloadFailure(node, typeParameters, candidate, args);
-            candidates[bestIndex] = instantiated;
-            return instantiated;
         }
-        function getTypeArgumentsFromNodes(typeArgumentNodes, typeParameters, isJs) {
-            var typeArguments = typeArgumentNodes.map(getTypeOfNode);
-            while (typeArguments.length > typeParameters.length) {
-                typeArguments.pop();
+        function checkThisExpression(node) {
+            // Stop at the first arrow function so that we can
+            // tell whether 'this' needs to be captured.
+            var container = ts.getThisContainer(node, /* includeArrowFunctions */ true);
+            var capturedByArrowFunction = false;
+            if (container.kind === 166 /* Constructor */) {
+                checkThisBeforeSuper(node, container, ts.Diagnostics.super_must_be_called_before_accessing_this_in_the_constructor_of_a_derived_class);
             }
-            while (typeArguments.length < typeParameters.length) {
-                typeArguments.push(getConstraintOfTypeParameter(typeParameters[typeArguments.length]) || getDefaultTypeArgumentType(isJs));
+            // Now skip arrow functions to get the "real" owner of 'this'.
+            if (container.kind === 209 /* ArrowFunction */) {
+                container = ts.getThisContainer(container, /* includeArrowFunctions */ false);
+                capturedByArrowFunction = true;
             }
-            return typeArguments;
-        }
-        function inferSignatureInstantiationForOverloadFailure(node, typeParameters, candidate, args) {
-            var inferenceContext = createInferenceContext(typeParameters, candidate, /*flags*/ ts.isInJSFile(node) ? 2 /* AnyDefault */ : 0 /* None */);
-            var typeArgumentTypes = inferTypeArguments(node, candidate, args, 4 /* SkipContextSensitive */ | 8 /* SkipGenericFunctions */, inferenceContext);
-            return createSignatureInstantiation(candidate, typeArgumentTypes);
-        }
-        function getLongestCandidateIndex(candidates, argsCount) {
-            var maxParamsIndex = -1;
-            var maxParams = -1;
-            for (var i = 0; i < candidates.length; i++) {
-                var candidate = candidates[i];
-                var paramCount = getParameterCount(candidate);
-                if (hasEffectiveRestParameter(candidate) || paramCount >= argsCount) {
-                    return i;
+            switch (container.kind) {
+                case 256 /* ModuleDeclaration */:
+                    error(node, ts.Diagnostics.this_cannot_be_referenced_in_a_module_or_namespace_body);
+                    // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
+                    break;
+                case 255 /* EnumDeclaration */:
+                    error(node, ts.Diagnostics.this_cannot_be_referenced_in_current_location);
+                    // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
+                    break;
+                case 166 /* Constructor */:
+                    if (isInConstructorArgumentInitializer(node, container)) {
+                        error(node, ts.Diagnostics.this_cannot_be_referenced_in_constructor_arguments);
+                        // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
+                    }
+                    break;
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                    if (ts.hasSyntacticModifier(container, 32 /* Static */) && !(compilerOptions.target === 99 /* ESNext */ && compilerOptions.useDefineForClassFields)) {
+                        error(node, ts.Diagnostics.this_cannot_be_referenced_in_a_static_property_initializer);
+                        // do not return here so in case if lexical this is captured - it will be reflected in flags on NodeLinks
+                    }
+                    break;
+                case 158 /* ComputedPropertyName */:
+                    error(node, ts.Diagnostics.this_cannot_be_referenced_in_a_computed_property_name);
+                    break;
+            }
+            // When targeting es6, mark that we'll need to capture `this` in its lexically bound scope.
+            if (capturedByArrowFunction && languageVersion < 2 /* ES2015 */) {
+                captureLexicalThis(node, container);
+            }
+            var type = tryGetThisTypeAt(node, /*includeGlobalThis*/ true, container);
+            if (noImplicitThis) {
+                var globalThisType_1 = getTypeOfSymbol(globalThisSymbol);
+                if (type === globalThisType_1 && capturedByArrowFunction) {
+                    error(node, ts.Diagnostics.The_containing_arrow_function_captures_the_global_value_of_this);
                 }
-                if (paramCount > maxParams) {
-                    maxParams = paramCount;
-                    maxParamsIndex = i;
+                else if (!type) {
+                    // With noImplicitThis, functions may not reference 'this' if it has type 'any'
+                    var diag = error(node, ts.Diagnostics.this_implicitly_has_type_any_because_it_does_not_have_a_type_annotation);
+                    if (!ts.isSourceFile(container)) {
+                        var outsideThis = tryGetThisTypeAt(container);
+                        if (outsideThis && outsideThis !== globalThisType_1) {
+                            ts.addRelatedInfo(diag, ts.createDiagnosticForNode(container, ts.Diagnostics.An_outer_value_of_this_is_shadowed_by_this_container));
+                        }
+                    }
                 }
             }
-            return maxParamsIndex;
+            return type || anyType;
         }
-        function resolveCallExpression(node, candidatesOutArray, checkMode) {
-            if (node.expression.kind === 102 /* SuperKeyword */) {
-                var superType = checkSuperExpression(node.expression);
-                if (isTypeAny(superType)) {
-                    for (var _i = 0, _a = node.arguments; _i < _a.length; _i++) {
-                        var arg = _a[_i];
-                        checkExpression(arg); // Still visit arguments so they get marked for visibility, etc
+        function tryGetThisTypeAt(node, includeGlobalThis, container) {
+            if (includeGlobalThis === void 0) { includeGlobalThis = true; }
+            if (container === void 0) { container = ts.getThisContainer(node, /*includeArrowFunctions*/ false); }
+            var isInJS = ts.isInJSFile(node);
+            if (ts.isFunctionLike(container) &&
+                (!isInParameterInitializerBeforeContainingFunction(node) || ts.getThisParameter(container))) {
+                var thisType = getThisTypeOfDeclaration(container) || isInJS && getTypeForThisExpressionFromJSDoc(container);
+                // Note: a parameter initializer should refer to class-this unless function-this is explicitly annotated.
+                // If this is a function in a JS file, it might be a class method.
+                if (!thisType) {
+                    var className = getClassNameFromPrototypeMethod(container);
+                    if (isInJS && className) {
+                        var classSymbol = checkExpression(className).symbol;
+                        if (classSymbol && classSymbol.members && (classSymbol.flags & 16 /* Function */)) {
+                            thisType = getDeclaredTypeOfSymbol(classSymbol).thisType;
+                        }
                     }
-                    return anySignature;
-                }
-                if (superType !== errorType) {
-                    // In super call, the candidate signatures are the matching arity signatures of the base constructor function instantiated
-                    // with the type arguments specified in the extends clause.
-                    var baseTypeNode = ts.getEffectiveBaseTypeNode(ts.getContainingClass(node));
-                    if (baseTypeNode) {
-                        var baseConstructors = getInstantiatedConstructorsForTypeArguments(superType, baseTypeNode.typeArguments, baseTypeNode);
-                        return resolveCall(node, baseConstructors, candidatesOutArray, checkMode, 0 /* None */);
+                    else if (isJSConstructor(container)) {
+                        thisType = getDeclaredTypeOfSymbol(getMergedSymbol(container.symbol)).thisType;
                     }
+                    thisType || (thisType = getContextualThisParameterType(container));
+                }
+                if (thisType) {
+                    return getFlowTypeOfReference(node, thisType);
                 }
-                return resolveUntypedCall(node);
             }
-            var callChainFlags;
-            var funcType = checkExpression(node.expression);
-            if (ts.isCallChain(node)) {
-                var nonOptionalType = getOptionalExpressionType(funcType, node.expression);
-                callChainFlags = nonOptionalType === funcType ? 0 /* None */ :
-                    ts.isOutermostOptionalChain(node) ? 8 /* IsOuterCallChain */ :
-                        4 /* IsInnerCallChain */;
-                funcType = nonOptionalType;
+            if (ts.isClassLike(container.parent)) {
+                var symbol = getSymbolOfNode(container.parent);
+                var type = ts.hasSyntacticModifier(container, 32 /* Static */) ? getTypeOfSymbol(symbol) : getDeclaredTypeOfSymbol(symbol).thisType;
+                return getFlowTypeOfReference(node, type);
             }
-            else {
-                callChainFlags = 0 /* None */;
+            if (ts.isSourceFile(container)) {
+                // look up in the source file's locals or exports
+                if (container.commonJsModuleIndicator) {
+                    var fileSymbol = getSymbolOfNode(container);
+                    return fileSymbol && getTypeOfSymbol(fileSymbol);
+                }
+                else if (container.externalModuleIndicator) {
+                    // TODO: Maybe issue a better error than 'object is possibly undefined'
+                    return undefinedType;
+                }
+                else if (includeGlobalThis) {
+                    return getTypeOfSymbol(globalThisSymbol);
+                }
             }
-            funcType = checkNonNullTypeWithReporter(funcType, node.expression, reportCannotInvokePossiblyNullOrUndefinedError);
-            if (funcType === silentNeverType) {
-                return silentNeverSignature;
+        }
+        function getExplicitThisType(node) {
+            var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
+            if (ts.isFunctionLike(container)) {
+                var signature = getSignatureFromDeclaration(container);
+                if (signature.thisParameter) {
+                    return getExplicitTypeOfSymbol(signature.thisParameter);
+                }
             }
-            var apparentType = getApparentType(funcType);
-            if (apparentType === errorType) {
-                // Another error has already been reported
-                return resolveErrorCall(node);
+            if (ts.isClassLike(container.parent)) {
+                var symbol = getSymbolOfNode(container.parent);
+                return ts.hasSyntacticModifier(container, 32 /* Static */) ? getTypeOfSymbol(symbol) : getDeclaredTypeOfSymbol(symbol).thisType;
             }
-            // Technically, this signatures list may be incomplete. We are taking the apparent type,
-            // but we are not including call signatures that may have been added to the Object or
-            // Function interface, since they have none by default. This is a bit of a leap of faith
-            // that the user will not add any.
-            var callSignatures = getSignaturesOfType(apparentType, 0 /* Call */);
-            var numConstructSignatures = getSignaturesOfType(apparentType, 1 /* Construct */).length;
-            // TS 1.0 Spec: 4.12
-            // In an untyped function call no TypeArgs are permitted, Args can be any argument list, no contextual
-            // types are provided for the argument expressions, and the result is always of type Any.
-            if (isUntypedFunctionCall(funcType, apparentType, callSignatures.length, numConstructSignatures)) {
-                // The unknownType indicates that an error already occurred (and was reported).  No
-                // need to report another error in this case.
-                if (funcType !== errorType && node.typeArguments) {
-                    error(node, ts.Diagnostics.Untyped_function_calls_may_not_accept_type_arguments);
+        }
+        function getClassNameFromPrototypeMethod(container) {
+            // Check if it's the RHS of a x.prototype.y = function [name]() { .... }
+            if (container.kind === 208 /* FunctionExpression */ &&
+                ts.isBinaryExpression(container.parent) &&
+                ts.getAssignmentDeclarationKind(container.parent) === 3 /* PrototypeProperty */) {
+                // Get the 'x' of 'x.prototype.y = container'
+                return container.parent // x.prototype.y = container
+                    .left // x.prototype.y
+                    .expression // x.prototype
+                    .expression; // x
+            }
+            // x.prototype = { method() { } }
+            else if (container.kind === 165 /* MethodDeclaration */ &&
+                container.parent.kind === 200 /* ObjectLiteralExpression */ &&
+                ts.isBinaryExpression(container.parent.parent) &&
+                ts.getAssignmentDeclarationKind(container.parent.parent) === 6 /* Prototype */) {
+                return container.parent.parent.left.expression;
+            }
+            // x.prototype = { method: function() { } }
+            else if (container.kind === 208 /* FunctionExpression */ &&
+                container.parent.kind === 288 /* PropertyAssignment */ &&
+                container.parent.parent.kind === 200 /* ObjectLiteralExpression */ &&
+                ts.isBinaryExpression(container.parent.parent.parent) &&
+                ts.getAssignmentDeclarationKind(container.parent.parent.parent) === 6 /* Prototype */) {
+                return container.parent.parent.parent.left.expression;
+            }
+            // Object.defineProperty(x, "method", { value: function() { } });
+            // Object.defineProperty(x, "method", { set: (x: () => void) => void });
+            // Object.defineProperty(x, "method", { get: () => function() { }) });
+            else if (container.kind === 208 /* FunctionExpression */ &&
+                ts.isPropertyAssignment(container.parent) &&
+                ts.isIdentifier(container.parent.name) &&
+                (container.parent.name.escapedText === "value" || container.parent.name.escapedText === "get" || container.parent.name.escapedText === "set") &&
+                ts.isObjectLiteralExpression(container.parent.parent) &&
+                ts.isCallExpression(container.parent.parent.parent) &&
+                container.parent.parent.parent.arguments[2] === container.parent.parent &&
+                ts.getAssignmentDeclarationKind(container.parent.parent.parent) === 9 /* ObjectDefinePrototypeProperty */) {
+                return container.parent.parent.parent.arguments[0].expression;
+            }
+            // Object.defineProperty(x, "method", { value() { } });
+            // Object.defineProperty(x, "method", { set(x: () => void) {} });
+            // Object.defineProperty(x, "method", { get() { return () => {} } });
+            else if (ts.isMethodDeclaration(container) &&
+                ts.isIdentifier(container.name) &&
+                (container.name.escapedText === "value" || container.name.escapedText === "get" || container.name.escapedText === "set") &&
+                ts.isObjectLiteralExpression(container.parent) &&
+                ts.isCallExpression(container.parent.parent) &&
+                container.parent.parent.arguments[2] === container.parent &&
+                ts.getAssignmentDeclarationKind(container.parent.parent) === 9 /* ObjectDefinePrototypeProperty */) {
+                return container.parent.parent.arguments[0].expression;
+            }
+        }
+        function getTypeForThisExpressionFromJSDoc(node) {
+            var jsdocType = ts.getJSDocType(node);
+            if (jsdocType && jsdocType.kind === 308 /* JSDocFunctionType */) {
+                var jsDocFunctionType = jsdocType;
+                if (jsDocFunctionType.parameters.length > 0 &&
+                    jsDocFunctionType.parameters[0].name &&
+                    jsDocFunctionType.parameters[0].name.escapedText === "this" /* This */) {
+                    return getTypeFromTypeNode(jsDocFunctionType.parameters[0].type);
                 }
-                return resolveUntypedCall(node);
             }
-            // If FuncExpr's apparent type(section 3.8.1) is a function type, the call is a typed function call.
-            // TypeScript employs overload resolution in typed function calls in order to support functions
-            // with multiple call signatures.
-            if (!callSignatures.length) {
-                if (numConstructSignatures) {
-                    error(node, ts.Diagnostics.Value_of_type_0_is_not_callable_Did_you_mean_to_include_new, typeToString(funcType));
+            var thisTag = ts.getJSDocThisTag(node);
+            if (thisTag && thisTag.typeExpression) {
+                return getTypeFromTypeNode(thisTag.typeExpression);
+            }
+        }
+        function isInConstructorArgumentInitializer(node, constructorDecl) {
+            return !!ts.findAncestor(node, function (n) { return ts.isFunctionLikeDeclaration(n) ? "quit" : n.kind === 160 /* Parameter */ && n.parent === constructorDecl; });
+        }
+        function checkSuperExpression(node) {
+            var isCallExpression = node.parent.kind === 203 /* CallExpression */ && node.parent.expression === node;
+            var immediateContainer = ts.getSuperContainer(node, /*stopOnFunctions*/ true);
+            var container = immediateContainer;
+            var needToCaptureLexicalThis = false;
+            // adjust the container reference in case if super is used inside arrow functions with arbitrarily deep nesting
+            if (!isCallExpression) {
+                while (container && container.kind === 209 /* ArrowFunction */) {
+                    container = ts.getSuperContainer(container, /*stopOnFunctions*/ true);
+                    needToCaptureLexicalThis = languageVersion < 2 /* ES2015 */;
+                }
+            }
+            var canUseSuperExpression = isLegalUsageOfSuperExpression(container);
+            var nodeCheckFlag = 0;
+            if (!canUseSuperExpression) {
+                // issue more specific error if super is used in computed property name
+                // class A { foo() { return "1" }}
+                // class B {
+                //     [super.foo()]() {}
+                // }
+                var current = ts.findAncestor(node, function (n) { return n === container ? "quit" : n.kind === 158 /* ComputedPropertyName */; });
+                if (current && current.kind === 158 /* ComputedPropertyName */) {
+                    error(node, ts.Diagnostics.super_cannot_be_referenced_in_a_computed_property_name);
+                }
+                else if (isCallExpression) {
+                    error(node, ts.Diagnostics.Super_calls_are_not_permitted_outside_constructors_or_in_nested_functions_inside_constructors);
+                }
+                else if (!container || !container.parent || !(ts.isClassLike(container.parent) || container.parent.kind === 200 /* ObjectLiteralExpression */)) {
+                    error(node, ts.Diagnostics.super_can_only_be_referenced_in_members_of_derived_classes_or_object_literal_expressions);
                 }
                 else {
-                    var relatedInformation = void 0;
-                    if (node.arguments.length === 1) {
-                        var text = ts.getSourceFileOfNode(node).text;
-                        if (ts.isLineBreak(text.charCodeAt(ts.skipTrivia(text, node.expression.end, /* stopAfterLineBreak */ true) - 1))) {
-                            relatedInformation = ts.createDiagnosticForNode(node.expression, ts.Diagnostics.Are_you_missing_a_semicolon);
-                        }
-                    }
-                    invocationError(node.expression, apparentType, 0 /* Call */, relatedInformation);
+                    error(node, ts.Diagnostics.super_property_access_is_permitted_only_in_a_constructor_member_function_or_member_accessor_of_a_derived_class);
+                }
+                return errorType;
+            }
+            if (!isCallExpression && immediateContainer.kind === 166 /* Constructor */) {
+                checkThisBeforeSuper(node, container, ts.Diagnostics.super_must_be_called_before_accessing_a_property_of_super_in_the_constructor_of_a_derived_class);
+            }
+            if (ts.hasSyntacticModifier(container, 32 /* Static */) || isCallExpression) {
+                nodeCheckFlag = 512 /* SuperStatic */;
+            }
+            else {
+                nodeCheckFlag = 256 /* SuperInstance */;
+            }
+            getNodeLinks(node).flags |= nodeCheckFlag;
+            // Due to how we emit async functions, we need to specialize the emit for an async method that contains a `super` reference.
+            // This is due to the fact that we emit the body of an async function inside of a generator function. As generator
+            // functions cannot reference `super`, we emit a helper inside of the method body, but outside of the generator. This helper
+            // uses an arrow function, which is permitted to reference `super`.
+            //
+            // There are two primary ways we can access `super` from within an async method. The first is getting the value of a property
+            // or indexed access on super, either as part of a right-hand-side expression or call expression. The second is when setting the value
+            // of a property or indexed access, either as part of an assignment expression or destructuring assignment.
+            //
+            // The simplest case is reading a value, in which case we will emit something like the following:
+            //
+            //  // ts
+            //  ...
+            //  async asyncMethod() {
+            //    let x = await super.asyncMethod();
+            //    return x;
+            //  }
+            //  ...
+            //
+            //  // js
+            //  ...
+            //  asyncMethod() {
+            //      const _super = Object.create(null, {
+            //        asyncMethod: { get: () => super.asyncMethod },
+            //      });
+            //      return __awaiter(this, arguments, Promise, function *() {
+            //          let x = yield _super.asyncMethod.call(this);
+            //          return x;
+            //      });
+            //  }
+            //  ...
+            //
+            // The more complex case is when we wish to assign a value, especially as part of a destructuring assignment. As both cases
+            // are legal in ES6, but also likely less frequent, we only emit setters if there is an assignment:
+            //
+            //  // ts
+            //  ...
+            //  async asyncMethod(ar: Promise<any[]>) {
+            //      [super.a, super.b] = await ar;
+            //  }
+            //  ...
+            //
+            //  // js
+            //  ...
+            //  asyncMethod(ar) {
+            //      const _super = Object.create(null, {
+            //        a: { get: () => super.a, set: (v) => super.a = v },
+            //        b: { get: () => super.b, set: (v) => super.b = v }
+            //      };
+            //      return __awaiter(this, arguments, Promise, function *() {
+            //          [_super.a, _super.b] = yield ar;
+            //      });
+            //  }
+            //  ...
+            //
+            // Creating an object that has getter and setters instead of just an accessor function is required for destructuring assignments
+            // as a call expression cannot be used as the target of a destructuring assignment while a property access can.
+            //
+            // For element access expressions (`super[x]`), we emit a generic helper that forwards the element access in both situations.
+            if (container.kind === 165 /* MethodDeclaration */ && ts.hasSyntacticModifier(container, 256 /* Async */)) {
+                if (ts.isSuperProperty(node.parent) && ts.isAssignmentTarget(node.parent)) {
+                    getNodeLinks(container).flags |= 4096 /* AsyncMethodWithSuperBinding */;
+                }
+                else {
+                    getNodeLinks(container).flags |= 2048 /* AsyncMethodWithSuper */;
                 }
-                return resolveErrorCall(node);
-            }
-            // When a call to a generic function is an argument to an outer call to a generic function for which
-            // inference is in process, we have a choice to make. If the inner call relies on inferences made from
-            // its contextual type to its return type, deferring the inner call processing allows the best possible
-            // contextual type to accumulate. But if the outer call relies on inferences made from the return type of
-            // the inner call, the inner call should be processed early. There's no sure way to know which choice is
-            // right (only a full unification algorithm can determine that), so we resort to the following heuristic:
-            // If no type arguments are specified in the inner call and at least one call signature is generic and
-            // returns a function type, we choose to defer processing. This narrowly permits function composition
-            // operators to flow inferences through return types, but otherwise processes calls right away. We
-            // use the resolvingSignature singleton to indicate that we deferred processing. This result will be
-            // propagated out and eventually turned into nonInferrableType (a type that is assignable to anything and
-            // from which we never make inferences).
-            if (checkMode & 8 /* SkipGenericFunctions */ && !node.typeArguments && callSignatures.some(isGenericFunctionReturningFunction)) {
-                skippedGenericFunction(node, checkMode);
-                return resolvingSignature;
             }
-            // If the function is explicitly marked with `@class`, then it must be constructed.
-            if (callSignatures.some(function (sig) { return ts.isInJSFile(sig.declaration) && !!ts.getJSDocClassTag(sig.declaration); })) {
-                error(node, ts.Diagnostics.Value_of_type_0_is_not_callable_Did_you_mean_to_include_new, typeToString(funcType));
-                return resolveErrorCall(node);
+            if (needToCaptureLexicalThis) {
+                // call expressions are allowed only in constructors so they should always capture correct 'this'
+                // super property access expressions can also appear in arrow functions -
+                // in this case they should also use correct lexical this
+                captureLexicalThis(node.parent, container);
             }
-            return resolveCall(node, callSignatures, candidatesOutArray, checkMode, callChainFlags);
-        }
-        function isGenericFunctionReturningFunction(signature) {
-            return !!(signature.typeParameters && isFunctionType(getReturnTypeOfSignature(signature)));
-        }
-        /**
-         * TS 1.0 spec: 4.12
-         * If FuncExpr is of type Any, or of an object type that has no call or construct signatures
-         * but is a subtype of the Function interface, the call is an untyped function call.
-         */
-        function isUntypedFunctionCall(funcType, apparentFuncType, numCallSignatures, numConstructSignatures) {
-            // We exclude union types because we may have a union of function types that happen to have no common signatures.
-            return isTypeAny(funcType) || isTypeAny(apparentFuncType) && !!(funcType.flags & 262144 /* TypeParameter */) ||
-                !numCallSignatures && !numConstructSignatures && !(apparentFuncType.flags & (1048576 /* Union */ | 131072 /* Never */)) && isTypeAssignableTo(funcType, globalFunctionType);
-        }
-        function resolveNewExpression(node, candidatesOutArray, checkMode) {
-            if (node.arguments && languageVersion < 1 /* ES5 */) {
-                var spreadIndex = getSpreadArgumentIndex(node.arguments);
-                if (spreadIndex >= 0) {
-                    error(node.arguments[spreadIndex], ts.Diagnostics.Spread_operator_in_new_expressions_is_only_available_when_targeting_ECMAScript_5_and_higher);
+            if (container.parent.kind === 200 /* ObjectLiteralExpression */) {
+                if (languageVersion < 2 /* ES2015 */) {
+                    error(node, ts.Diagnostics.super_is_only_allowed_in_members_of_object_literal_expressions_when_option_target_is_ES2015_or_higher);
+                    return errorType;
+                }
+                else {
+                    // for object literal assume that type of 'super' is 'any'
+                    return anyType;
                 }
             }
-            var expressionType = checkNonNullExpression(node.expression);
-            if (expressionType === silentNeverType) {
-                return silentNeverSignature;
+            // at this point the only legal case for parent is ClassLikeDeclaration
+            var classLikeDeclaration = container.parent;
+            if (!ts.getClassExtendsHeritageElement(classLikeDeclaration)) {
+                error(node, ts.Diagnostics.super_can_only_be_referenced_in_a_derived_class);
+                return errorType;
             }
-            // If expressionType's apparent type(section 3.8.1) is an object type with one or
-            // more construct signatures, the expression is processed in the same manner as a
-            // function call, but using the construct signatures as the initial set of candidate
-            // signatures for overload resolution. The result type of the function call becomes
-            // the result type of the operation.
-            expressionType = getApparentType(expressionType);
-            if (expressionType === errorType) {
-                // Another error has already been reported
-                return resolveErrorCall(node);
+            var classType = getDeclaredTypeOfSymbol(getSymbolOfNode(classLikeDeclaration));
+            var baseClassType = classType && getBaseTypes(classType)[0];
+            if (!baseClassType) {
+                return errorType;
             }
-            // TS 1.0 spec: 4.11
-            // If expressionType is of type Any, Args can be any argument
-            // list and the result of the operation is of type Any.
-            if (isTypeAny(expressionType)) {
-                if (node.typeArguments) {
-                    error(node, ts.Diagnostics.Untyped_function_calls_may_not_accept_type_arguments);
-                }
-                return resolveUntypedCall(node);
+            if (container.kind === 166 /* Constructor */ && isInConstructorArgumentInitializer(node, container)) {
+                // issue custom error message for super property access in constructor arguments (to be aligned with old compiler)
+                error(node, ts.Diagnostics.super_cannot_be_referenced_in_constructor_arguments);
+                return errorType;
             }
-            // Technically, this signatures list may be incomplete. We are taking the apparent type,
-            // but we are not including construct signatures that may have been added to the Object or
-            // Function interface, since they have none by default. This is a bit of a leap of faith
-            // that the user will not add any.
-            var constructSignatures = getSignaturesOfType(expressionType, 1 /* Construct */);
-            if (constructSignatures.length) {
-                if (!isConstructorAccessible(node, constructSignatures[0])) {
-                    return resolveErrorCall(node);
-                }
-                // If the expression is a class of abstract type, then it cannot be instantiated.
-                // Note, only class declarations can be declared abstract.
-                // In the case of a merged class-module or class-interface declaration,
-                // only the class declaration node will have the Abstract flag set.
-                var valueDecl = expressionType.symbol && ts.getClassLikeDeclarationOfSymbol(expressionType.symbol);
-                if (valueDecl && ts.hasModifier(valueDecl, 128 /* Abstract */)) {
-                    error(node, ts.Diagnostics.Cannot_create_an_instance_of_an_abstract_class);
-                    return resolveErrorCall(node);
+            return nodeCheckFlag === 512 /* SuperStatic */
+                ? getBaseConstructorTypeOfClass(classType)
+                : getTypeWithThisArgument(baseClassType, classType.thisType);
+            function isLegalUsageOfSuperExpression(container) {
+                if (!container) {
+                    return false;
                 }
-                return resolveCall(node, constructSignatures, candidatesOutArray, checkMode, 0 /* None */);
-            }
-            // If expressionType's apparent type is an object type with no construct signatures but
-            // one or more call signatures, the expression is processed as a function call. A compile-time
-            // error occurs if the result of the function call is not Void. The type of the result of the
-            // operation is Any. It is an error to have a Void this type.
-            var callSignatures = getSignaturesOfType(expressionType, 0 /* Call */);
-            if (callSignatures.length) {
-                var signature = resolveCall(node, callSignatures, candidatesOutArray, checkMode, 0 /* None */);
-                if (!noImplicitAny) {
-                    if (signature.declaration && !isJSConstructor(signature.declaration) && getReturnTypeOfSignature(signature) !== voidType) {
-                        error(node, ts.Diagnostics.Only_a_void_function_can_be_called_with_the_new_keyword);
-                    }
-                    if (getThisTypeOfSignature(signature) === voidType) {
-                        error(node, ts.Diagnostics.A_function_that_is_called_with_the_new_keyword_cannot_have_a_this_type_that_is_void);
-                    }
+                if (isCallExpression) {
+                    // TS 1.0 SPEC (April 2014): 4.8.1
+                    // Super calls are only permitted in constructors of derived classes
+                    return container.kind === 166 /* Constructor */;
                 }
-                return signature;
-            }
-            invocationError(node.expression, expressionType, 1 /* Construct */);
-            return resolveErrorCall(node);
-        }
-        function typeHasProtectedAccessibleBase(target, type) {
-            var baseTypes = getBaseTypes(type);
-            if (!ts.length(baseTypes)) {
-                return false;
-            }
-            var firstBase = baseTypes[0];
-            if (firstBase.flags & 2097152 /* Intersection */) {
-                var types = firstBase.types;
-                var mixinFlags = findMixins(types);
-                var i = 0;
-                for (var _i = 0, _a = firstBase.types; _i < _a.length; _i++) {
-                    var intersectionMember = _a[_i];
-                    // We want to ignore mixin ctors
-                    if (!mixinFlags[i]) {
-                        if (ts.getObjectFlags(intersectionMember) & (1 /* Class */ | 2 /* Interface */)) {
-                            if (intersectionMember.symbol === target) {
-                                return true;
-                            }
-                            if (typeHasProtectedAccessibleBase(target, intersectionMember)) {
-                                return true;
-                            }
+                else {
+                    // TS 1.0 SPEC (April 2014)
+                    // 'super' property access is allowed
+                    // - In a constructor, instance member function, instance member accessor, or instance member variable initializer where this references a derived class instance
+                    // - In a static member function or static member accessor
+                    // topmost container must be something that is directly nested in the class declaration\object literal expression
+                    if (ts.isClassLike(container.parent) || container.parent.kind === 200 /* ObjectLiteralExpression */) {
+                        if (ts.hasSyntacticModifier(container, 32 /* Static */)) {
+                            return container.kind === 165 /* MethodDeclaration */ ||
+                                container.kind === 164 /* MethodSignature */ ||
+                                container.kind === 167 /* GetAccessor */ ||
+                                container.kind === 168 /* SetAccessor */;
+                        }
+                        else {
+                            return container.kind === 165 /* MethodDeclaration */ ||
+                                container.kind === 164 /* MethodSignature */ ||
+                                container.kind === 167 /* GetAccessor */ ||
+                                container.kind === 168 /* SetAccessor */ ||
+                                container.kind === 163 /* PropertyDeclaration */ ||
+                                container.kind === 162 /* PropertySignature */ ||
+                                container.kind === 166 /* Constructor */;
                         }
                     }
-                    i++;
                 }
                 return false;
             }
-            if (firstBase.symbol === target) {
-                return true;
-            }
-            return typeHasProtectedAccessibleBase(target, firstBase);
         }
-        function isConstructorAccessible(node, signature) {
-            if (!signature || !signature.declaration) {
-                return true;
-            }
-            var declaration = signature.declaration;
-            var modifiers = ts.getSelectedModifierFlags(declaration, 24 /* NonPublicAccessibilityModifier */);
-            // (1) Public constructors and (2) constructor functions are always accessible.
-            if (!modifiers || declaration.kind !== 162 /* Constructor */) {
-                return true;
+        function getContainingObjectLiteral(func) {
+            return (func.kind === 165 /* MethodDeclaration */ ||
+                func.kind === 167 /* GetAccessor */ ||
+                func.kind === 168 /* SetAccessor */) && func.parent.kind === 200 /* ObjectLiteralExpression */ ? func.parent :
+                func.kind === 208 /* FunctionExpression */ && func.parent.kind === 288 /* PropertyAssignment */ ? func.parent.parent :
+                    undefined;
+        }
+        function getThisTypeArgument(type) {
+            return ts.getObjectFlags(type) & 4 /* Reference */ && type.target === globalThisType ? getTypeArguments(type)[0] : undefined;
+        }
+        function getThisTypeFromContextualType(type) {
+            return mapType(type, function (t) {
+                return t.flags & 2097152 /* Intersection */ ? ts.forEach(t.types, getThisTypeArgument) : getThisTypeArgument(t);
+            });
+        }
+        function getContextualThisParameterType(func) {
+            if (func.kind === 209 /* ArrowFunction */) {
+                return undefined;
             }
-            var declaringClassDeclaration = ts.getClassLikeDeclarationOfSymbol(declaration.parent.symbol);
-            var declaringClass = getDeclaredTypeOfSymbol(declaration.parent.symbol);
-            // A private or protected constructor can only be instantiated within its own class (or a subclass, for protected)
-            if (!isNodeWithinClass(node, declaringClassDeclaration)) {
-                var containingClass = ts.getContainingClass(node);
-                if (containingClass && modifiers & 16 /* Protected */) {
-                    var containingType = getTypeOfNode(containingClass);
-                    if (typeHasProtectedAccessibleBase(declaration.parent.symbol, containingType)) {
-                        return true;
+            if (isContextSensitiveFunctionOrObjectLiteralMethod(func)) {
+                var contextualSignature = getContextualSignature(func);
+                if (contextualSignature) {
+                    var thisParameter = contextualSignature.thisParameter;
+                    if (thisParameter) {
+                        return getTypeOfSymbol(thisParameter);
                     }
                 }
-                if (modifiers & 8 /* Private */) {
-                    error(node, ts.Diagnostics.Constructor_of_class_0_is_private_and_only_accessible_within_the_class_declaration, typeToString(declaringClass));
-                }
-                if (modifiers & 16 /* Protected */) {
-                    error(node, ts.Diagnostics.Constructor_of_class_0_is_protected_and_only_accessible_within_the_class_declaration, typeToString(declaringClass));
-                }
-                return false;
             }
-            return true;
-        }
-        function invocationErrorDetails(apparentType, kind) {
-            var errorInfo;
-            var isCall = kind === 0 /* Call */;
-            var awaitedType = getAwaitedType(apparentType);
-            var maybeMissingAwait = awaitedType && getSignaturesOfType(awaitedType, kind).length > 0;
-            if (apparentType.flags & 1048576 /* Union */) {
-                var types = apparentType.types;
-                var hasSignatures = false;
-                for (var _i = 0, types_18 = types; _i < types_18.length; _i++) {
-                    var constituent = types_18[_i];
-                    var signatures = getSignaturesOfType(constituent, kind);
-                    if (signatures.length !== 0) {
-                        hasSignatures = true;
-                        if (errorInfo) {
-                            // Bail early if we already have an error, no chance of "No constituent of type is callable"
-                            break;
-                        }
-                    }
-                    else {
-                        // Error on the first non callable constituent only
-                        if (!errorInfo) {
-                            errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
-                                ts.Diagnostics.Type_0_has_no_call_signatures :
-                                ts.Diagnostics.Type_0_has_no_construct_signatures, typeToString(constituent));
-                            errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
-                                ts.Diagnostics.Not_all_constituents_of_type_0_are_callable :
-                                ts.Diagnostics.Not_all_constituents_of_type_0_are_constructable, typeToString(apparentType));
+            var inJs = ts.isInJSFile(func);
+            if (noImplicitThis || inJs) {
+                var containingLiteral = getContainingObjectLiteral(func);
+                if (containingLiteral) {
+                    // We have an object literal method. Check if the containing object literal has a contextual type
+                    // that includes a ThisType<T>. If so, T is the contextual type for 'this'. We continue looking in
+                    // any directly enclosing object literals.
+                    var contextualType = getApparentTypeOfContextualType(containingLiteral);
+                    var literal = containingLiteral;
+                    var type = contextualType;
+                    while (type) {
+                        var thisType = getThisTypeFromContextualType(type);
+                        if (thisType) {
+                            return instantiateType(thisType, getMapperFromContext(getInferenceContext(containingLiteral)));
                         }
-                        if (hasSignatures) {
-                            // Bail early if we already found a siganture, no chance of "No constituent of type is callable"
+                        if (literal.parent.kind !== 288 /* PropertyAssignment */) {
                             break;
                         }
+                        literal = literal.parent.parent;
+                        type = getApparentTypeOfContextualType(literal);
                     }
+                    // There was no contextual ThisType<T> for the containing object literal, so the contextual type
+                    // for 'this' is the non-null form of the contextual type for the containing object literal or
+                    // the type of the object literal itself.
+                    return getWidenedType(contextualType ? getNonNullableType(contextualType) : checkExpressionCached(containingLiteral));
                 }
-                if (!hasSignatures) {
-                    errorInfo = ts.chainDiagnosticMessages(
-                    /* detials */ undefined, isCall ?
-                        ts.Diagnostics.No_constituent_of_type_0_is_callable :
-                        ts.Diagnostics.No_constituent_of_type_0_is_constructable, typeToString(apparentType));
-                }
-                if (!errorInfo) {
-                    errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
-                        ts.Diagnostics.Each_member_of_the_union_type_0_has_signatures_but_none_of_those_signatures_are_compatible_with_each_other :
-                        ts.Diagnostics.Each_member_of_the_union_type_0_has_construct_signatures_but_none_of_those_signatures_are_compatible_with_each_other, typeToString(apparentType));
+                // In an assignment of the form 'obj.xxx = function(...)' or 'obj[xxx] = function(...)', the
+                // contextual type for 'this' is 'obj'.
+                var parent = ts.walkUpParenthesizedExpressions(func.parent);
+                if (parent.kind === 216 /* BinaryExpression */ && parent.operatorToken.kind === 62 /* EqualsToken */) {
+                    var target = parent.left;
+                    if (ts.isAccessExpression(target)) {
+                        var expression = target.expression;
+                        // Don't contextually type `this` as `exports` in `exports.Point = function(x, y) { this.x = x; this.y = y; }`
+                        if (inJs && ts.isIdentifier(expression)) {
+                            var sourceFile = ts.getSourceFileOfNode(parent);
+                            if (sourceFile.commonJsModuleIndicator && getResolvedSymbol(expression) === sourceFile.symbol) {
+                                return undefined;
+                            }
+                        }
+                        return getWidenedType(checkExpressionCached(expression));
+                    }
                 }
             }
-            else {
-                errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
-                    ts.Diagnostics.Type_0_has_no_call_signatures :
-                    ts.Diagnostics.Type_0_has_no_construct_signatures, typeToString(apparentType));
-            }
-            return {
-                messageChain: ts.chainDiagnosticMessages(errorInfo, isCall ? ts.Diagnostics.This_expression_is_not_callable : ts.Diagnostics.This_expression_is_not_constructable),
-                relatedMessage: maybeMissingAwait ? ts.Diagnostics.Did_you_forget_to_use_await : undefined,
-            };
+            return undefined;
         }
-        function invocationError(errorTarget, apparentType, kind, relatedInformation) {
-            var _a = invocationErrorDetails(apparentType, kind), messageChain = _a.messageChain, relatedInfo = _a.relatedMessage;
-            var diagnostic = ts.createDiagnosticForNodeFromMessageChain(errorTarget, messageChain);
-            if (relatedInfo) {
-                ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(errorTarget, relatedInfo));
+        // Return contextual type of parameter or undefined if no contextual type is available
+        function getContextuallyTypedParameterType(parameter) {
+            var func = parameter.parent;
+            if (!isContextSensitiveFunctionOrObjectLiteralMethod(func)) {
+                return undefined;
             }
-            if (ts.isCallExpression(errorTarget.parent)) {
-                var _b = getDiagnosticSpanForCallNode(errorTarget.parent, /* doNotIncludeArguments */ true), start = _b.start, length_7 = _b.length;
-                diagnostic.start = start;
-                diagnostic.length = length_7;
+            var iife = ts.getImmediatelyInvokedFunctionExpression(func);
+            if (iife && iife.arguments) {
+                var args = getEffectiveCallArguments(iife);
+                var indexOfParameter = func.parameters.indexOf(parameter);
+                if (parameter.dotDotDotToken) {
+                    return getSpreadArgumentType(args, indexOfParameter, args.length, anyType, /*context*/ undefined, 0 /* Normal */);
+                }
+                var links = getNodeLinks(iife);
+                var cached = links.resolvedSignature;
+                links.resolvedSignature = anySignature;
+                var type = indexOfParameter < args.length ?
+                    getWidenedLiteralType(checkExpression(args[indexOfParameter])) :
+                    parameter.initializer ? undefined : undefinedWideningType;
+                links.resolvedSignature = cached;
+                return type;
+            }
+            var contextualSignature = getContextualSignature(func);
+            if (contextualSignature) {
+                var index = func.parameters.indexOf(parameter) - (ts.getThisParameter(func) ? 1 : 0);
+                return parameter.dotDotDotToken && ts.lastOrUndefined(func.parameters) === parameter ?
+                    getRestTypeAtPosition(contextualSignature, index) :
+                    tryGetTypeAtPosition(contextualSignature, index);
             }
-            diagnostics.add(diagnostic);
-            invocationErrorRecovery(apparentType, kind, relatedInformation ? ts.addRelatedInfo(diagnostic, relatedInformation) : diagnostic);
         }
-        function invocationErrorRecovery(apparentType, kind, diagnostic) {
-            if (!apparentType.symbol) {
-                return;
+        function getContextualTypeForVariableLikeDeclaration(declaration) {
+            var typeNode = ts.getEffectiveTypeAnnotationNode(declaration);
+            if (typeNode) {
+                return getTypeFromTypeNode(typeNode);
             }
-            var importNode = getSymbolLinks(apparentType.symbol).originatingImport;
-            // Create a diagnostic on the originating import if possible onto which we can attach a quickfix
-            //  An import call expression cannot be rewritten into another form to correct the error - the only solution is to use `.default` at the use-site
-            if (importNode && !ts.isImportCall(importNode)) {
-                var sigs = getSignaturesOfType(getTypeOfSymbol(getSymbolLinks(apparentType.symbol).target), kind);
-                if (!sigs || !sigs.length)
-                    return;
-                ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(importNode, ts.Diagnostics.Type_originates_at_this_import_A_namespace_style_import_cannot_be_called_or_constructed_and_will_cause_a_failure_at_runtime_Consider_using_a_default_import_or_import_require_here_instead));
+            switch (declaration.kind) {
+                case 160 /* Parameter */:
+                    return getContextuallyTypedParameterType(declaration);
+                case 198 /* BindingElement */:
+                    return getContextualTypeForBindingElement(declaration);
+                case 163 /* PropertyDeclaration */:
+                    if (ts.hasSyntacticModifier(declaration, 32 /* Static */)) {
+                        return getContextualTypeForStaticPropertyDeclaration(declaration);
+                    }
+                // By default, do nothing and return undefined - only the above cases have context implied by a parent
             }
         }
-        function resolveTaggedTemplateExpression(node, candidatesOutArray, checkMode) {
-            var tagType = checkExpression(node.tag);
-            var apparentType = getApparentType(tagType);
-            if (apparentType === errorType) {
-                // Another error has already been reported
-                return resolveErrorCall(node);
-            }
-            var callSignatures = getSignaturesOfType(apparentType, 0 /* Call */);
-            var numConstructSignatures = getSignaturesOfType(apparentType, 1 /* Construct */).length;
-            if (isUntypedFunctionCall(tagType, apparentType, callSignatures.length, numConstructSignatures)) {
-                return resolveUntypedCall(node);
+        function getContextualTypeForBindingElement(declaration) {
+            var parent = declaration.parent.parent;
+            var name = declaration.propertyName || declaration.name;
+            var parentType = getContextualTypeForVariableLikeDeclaration(parent) ||
+                parent.kind !== 198 /* BindingElement */ && parent.initializer && checkDeclarationInitializer(parent);
+            if (!parentType || ts.isBindingPattern(name) || ts.isComputedNonLiteralName(name))
+                return undefined;
+            if (parent.name.kind === 197 /* ArrayBindingPattern */) {
+                var index = ts.indexOfNode(declaration.parent.elements, declaration);
+                if (index < 0)
+                    return undefined;
+                return getContextualTypeForElementExpression(parentType, index);
             }
-            if (!callSignatures.length) {
-                invocationError(node.tag, apparentType, 0 /* Call */);
-                return resolveErrorCall(node);
+            var nameType = getLiteralTypeFromPropertyName(name);
+            if (isTypeUsableAsPropertyName(nameType)) {
+                var text = getPropertyNameFromType(nameType);
+                return getTypeOfPropertyOfType(parentType, text);
             }
-            return resolveCall(node, callSignatures, candidatesOutArray, checkMode, 0 /* None */);
         }
-        /**
-         * Gets the localized diagnostic head message to use for errors when resolving a decorator as a call expression.
-         */
-        function getDiagnosticHeadMessageForDecoratorResolution(node) {
-            switch (node.parent.kind) {
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                    return ts.Diagnostics.Unable_to_resolve_signature_of_class_decorator_when_called_as_an_expression;
-                case 156 /* Parameter */:
-                    return ts.Diagnostics.Unable_to_resolve_signature_of_parameter_decorator_when_called_as_an_expression;
-                case 159 /* PropertyDeclaration */:
-                    return ts.Diagnostics.Unable_to_resolve_signature_of_property_decorator_when_called_as_an_expression;
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    return ts.Diagnostics.Unable_to_resolve_signature_of_method_decorator_when_called_as_an_expression;
-                default:
-                    return ts.Debug.fail();
-            }
+        function getContextualTypeForStaticPropertyDeclaration(declaration) {
+            var parentType = ts.isExpression(declaration.parent) && getContextualType(declaration.parent);
+            if (!parentType)
+                return undefined;
+            return getTypeOfPropertyOfContextualType(parentType, getSymbolOfNode(declaration).escapedName);
         }
-        /**
-         * Resolves a decorator as if it were a call expression.
-         */
-        function resolveDecorator(node, candidatesOutArray, checkMode) {
-            var funcType = checkExpression(node.expression);
-            var apparentType = getApparentType(funcType);
-            if (apparentType === errorType) {
-                return resolveErrorCall(node);
+        // In a variable, parameter or property declaration with a type annotation,
+        //   the contextual type of an initializer expression is the type of the variable, parameter or property.
+        // Otherwise, in a parameter declaration of a contextually typed function expression,
+        //   the contextual type of an initializer expression is the contextual type of the parameter.
+        // Otherwise, in a variable or parameter declaration with a binding pattern name,
+        //   the contextual type of an initializer expression is the type implied by the binding pattern.
+        // Otherwise, in a binding pattern inside a variable or parameter declaration,
+        //   the contextual type of an initializer expression is the type annotation of the containing declaration, if present.
+        function getContextualTypeForInitializerExpression(node, contextFlags) {
+            var declaration = node.parent;
+            if (ts.hasInitializer(declaration) && node === declaration.initializer) {
+                var result = getContextualTypeForVariableLikeDeclaration(declaration);
+                if (result) {
+                    return result;
+                }
+                if (!(contextFlags & 8 /* SkipBindingPatterns */) && ts.isBindingPattern(declaration.name)) { // This is less a contextual type and more an implied shape - in some cases, this may be undesirable
+                    return getTypeFromBindingPattern(declaration.name, /*includePatternInType*/ true, /*reportErrors*/ false);
+                }
             }
-            var callSignatures = getSignaturesOfType(apparentType, 0 /* Call */);
-            var numConstructSignatures = getSignaturesOfType(apparentType, 1 /* Construct */).length;
-            if (isUntypedFunctionCall(funcType, apparentType, callSignatures.length, numConstructSignatures)) {
-                return resolveUntypedCall(node);
+            return undefined;
+        }
+        function getContextualTypeForReturnExpression(node) {
+            var func = ts.getContainingFunction(node);
+            if (func) {
+                var contextualReturnType = getContextualReturnType(func);
+                if (contextualReturnType) {
+                    var functionFlags = ts.getFunctionFlags(func);
+                    if (functionFlags & 1 /* Generator */) { // Generator or AsyncGenerator function
+                        var use = functionFlags & 2 /* Async */ ? 2 /* AsyncGeneratorReturnType */ : 1 /* GeneratorReturnType */;
+                        var iterationTypes = getIterationTypesOfIterable(contextualReturnType, use, /*errorNode*/ undefined);
+                        if (!iterationTypes) {
+                            return undefined;
+                        }
+                        contextualReturnType = iterationTypes.returnType;
+                        // falls through to unwrap Promise for AsyncGenerators
+                    }
+                    if (functionFlags & 2 /* Async */) { // Async function or AsyncGenerator function
+                        var contextualAwaitedType = mapType(contextualReturnType, getAwaitedType);
+                        return contextualAwaitedType && getUnionType([contextualAwaitedType, createPromiseLikeType(contextualAwaitedType)]);
+                    }
+                    return contextualReturnType; // Regular function or Generator function
+                }
             }
-            if (isPotentiallyUncalledDecorator(node, callSignatures)) {
-                var nodeStr = ts.getTextOfNode(node.expression, /*includeTrivia*/ false);
-                error(node, ts.Diagnostics._0_accepts_too_few_arguments_to_be_used_as_a_decorator_here_Did_you_mean_to_call_it_first_and_write_0, nodeStr);
-                return resolveErrorCall(node);
+            return undefined;
+        }
+        function getContextualTypeForAwaitOperand(node, contextFlags) {
+            var contextualType = getContextualType(node, contextFlags);
+            if (contextualType) {
+                var contextualAwaitedType = getAwaitedType(contextualType);
+                return contextualAwaitedType && getUnionType([contextualAwaitedType, createPromiseLikeType(contextualAwaitedType)]);
             }
-            var headMessage = getDiagnosticHeadMessageForDecoratorResolution(node);
-            if (!callSignatures.length) {
-                var errorDetails = invocationErrorDetails(apparentType, 0 /* Call */);
-                var messageChain = ts.chainDiagnosticMessages(errorDetails.messageChain, headMessage);
-                var diag = ts.createDiagnosticForNodeFromMessageChain(node.expression, messageChain);
-                if (errorDetails.relatedMessage) {
-                    ts.addRelatedInfo(diag, ts.createDiagnosticForNode(node.expression, errorDetails.relatedMessage));
+            return undefined;
+        }
+        function getContextualTypeForYieldOperand(node) {
+            var func = ts.getContainingFunction(node);
+            if (func) {
+                var functionFlags = ts.getFunctionFlags(func);
+                var contextualReturnType = getContextualReturnType(func);
+                if (contextualReturnType) {
+                    return node.asteriskToken
+                        ? contextualReturnType
+                        : getIterationTypeOfGeneratorFunctionReturnType(0 /* Yield */, contextualReturnType, (functionFlags & 2 /* Async */) !== 0);
                 }
-                diagnostics.add(diag);
-                invocationErrorRecovery(apparentType, 0 /* Call */, diag);
-                return resolveErrorCall(node);
             }
-            return resolveCall(node, callSignatures, candidatesOutArray, checkMode, 0 /* None */, headMessage);
-        }
-        function createSignatureForJSXIntrinsic(node, result) {
-            var namespace = getJsxNamespaceAt(node);
-            var exports = namespace && getExportsOfSymbol(namespace);
-            // We fake up a SFC signature for each intrinsic, however a more specific per-element signature drawn from the JSX declaration
-            // file would probably be preferable.
-            var typeSymbol = exports && getSymbol(exports, JsxNames.Element, 788968 /* Type */);
-            var returnNode = typeSymbol && nodeBuilder.symbolToEntityName(typeSymbol, 788968 /* Type */, node);
-            var declaration = ts.createFunctionTypeNode(/*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotdotdot*/ undefined, "props", /*questionMark*/ undefined, nodeBuilder.typeToTypeNode(result, node))], returnNode ? ts.createTypeReferenceNode(returnNode, /*typeArguments*/ undefined) : ts.createKeywordTypeNode(125 /* AnyKeyword */));
-            var parameterSymbol = createSymbol(1 /* FunctionScopedVariable */, "props");
-            parameterSymbol.type = result;
-            return createSignature(declaration, 
-            /*typeParameters*/ undefined, 
-            /*thisParameter*/ undefined, [parameterSymbol], typeSymbol ? getDeclaredTypeOfSymbol(typeSymbol) : errorType, 
-            /*returnTypePredicate*/ undefined, 1, 0 /* None */);
+            return undefined;
         }
-        function resolveJsxOpeningLikeElement(node, candidatesOutArray, checkMode) {
-            if (isJsxIntrinsicIdentifier(node.tagName)) {
-                var result = getIntrinsicAttributesTypeFromJsxOpeningLikeElement(node);
-                var fakeSignature = createSignatureForJSXIntrinsic(node, result);
-                checkTypeAssignableToAndOptionallyElaborate(checkExpressionWithContextualType(node.attributes, getEffectiveFirstArgumentForJsxSignature(fakeSignature, node), /*mapper*/ undefined, 0 /* Normal */), result, node.tagName, node.attributes);
-                return fakeSignature;
+        function isInParameterInitializerBeforeContainingFunction(node) {
+            var inBindingInitializer = false;
+            while (node.parent && !ts.isFunctionLike(node.parent)) {
+                if (ts.isParameter(node.parent) && (inBindingInitializer || node.parent.initializer === node)) {
+                    return true;
+                }
+                if (ts.isBindingElement(node.parent) && node.parent.initializer === node) {
+                    inBindingInitializer = true;
+                }
+                node = node.parent;
             }
-            var exprTypes = checkExpression(node.tagName);
-            var apparentType = getApparentType(exprTypes);
-            if (apparentType === errorType) {
-                return resolveErrorCall(node);
+            return false;
+        }
+        function getContextualIterationType(kind, functionDecl) {
+            var isAsync = !!(ts.getFunctionFlags(functionDecl) & 2 /* Async */);
+            var contextualReturnType = getContextualReturnType(functionDecl);
+            if (contextualReturnType) {
+                return getIterationTypeOfGeneratorFunctionReturnType(kind, contextualReturnType, isAsync)
+                    || undefined;
             }
-            var signatures = getUninstantiatedJsxSignaturesOfType(exprTypes, node);
-            if (isUntypedFunctionCall(exprTypes, apparentType, signatures.length, /*constructSignatures*/ 0)) {
-                return resolveUntypedCall(node);
+            return undefined;
+        }
+        function getContextualReturnType(functionDecl) {
+            // If the containing function has a return type annotation, is a constructor, or is a get accessor whose
+            // corresponding set accessor has a type annotation, return statements in the function are contextually typed
+            var returnType = getReturnTypeFromAnnotation(functionDecl);
+            if (returnType) {
+                return returnType;
             }
-            if (signatures.length === 0) {
-                // We found no signatures at all, which is an error
-                error(node.tagName, ts.Diagnostics.JSX_element_type_0_does_not_have_any_construct_or_call_signatures, ts.getTextOfNode(node.tagName));
-                return resolveErrorCall(node);
+            // Otherwise, if the containing function is contextually typed by a function type with exactly one call signature
+            // and that call signature is non-generic, return statements are contextually typed by the return type of the signature
+            var signature = getContextualSignatureForFunctionLikeDeclaration(functionDecl);
+            if (signature && !isResolvingReturnTypeOfSignature(signature)) {
+                return getReturnTypeOfSignature(signature);
             }
-            return resolveCall(node, signatures, candidatesOutArray, checkMode, 0 /* None */);
+            return undefined;
         }
-        /**
-         * Sometimes, we have a decorator that could accept zero arguments,
-         * but is receiving too many arguments as part of the decorator invocation.
-         * In those cases, a user may have meant to *call* the expression before using it as a decorator.
-         */
-        function isPotentiallyUncalledDecorator(decorator, signatures) {
-            return signatures.length && ts.every(signatures, function (signature) {
-                return signature.minArgumentCount === 0 &&
-                    !signatureHasRestParameter(signature) &&
-                    signature.parameters.length < getDecoratorArgumentCount(decorator, signature);
-            });
+        // In a typed function call, an argument or substitution expression is contextually typed by the type of the corresponding parameter.
+        function getContextualTypeForArgument(callTarget, arg) {
+            var args = getEffectiveCallArguments(callTarget);
+            var argIndex = args.indexOf(arg); // -1 for e.g. the expression of a CallExpression, or the tag of a TaggedTemplateExpression
+            return argIndex === -1 ? undefined : getContextualTypeForArgumentAtIndex(callTarget, argIndex);
         }
-        function resolveSignature(node, candidatesOutArray, checkMode) {
-            switch (node.kind) {
-                case 196 /* CallExpression */:
-                    return resolveCallExpression(node, candidatesOutArray, checkMode);
-                case 197 /* NewExpression */:
-                    return resolveNewExpression(node, candidatesOutArray, checkMode);
-                case 198 /* TaggedTemplateExpression */:
-                    return resolveTaggedTemplateExpression(node, candidatesOutArray, checkMode);
-                case 157 /* Decorator */:
-                    return resolveDecorator(node, candidatesOutArray, checkMode);
-                case 268 /* JsxOpeningElement */:
-                case 267 /* JsxSelfClosingElement */:
-                    return resolveJsxOpeningLikeElement(node, candidatesOutArray, checkMode);
+        function getContextualTypeForArgumentAtIndex(callTarget, argIndex) {
+            // If we're already in the process of resolving the given signature, don't resolve again as
+            // that could cause infinite recursion. Instead, return anySignature.
+            var signature = getNodeLinks(callTarget).resolvedSignature === resolvingSignature ? resolvingSignature : getResolvedSignature(callTarget);
+            if (ts.isJsxOpeningLikeElement(callTarget) && argIndex === 0) {
+                return getEffectiveFirstArgumentForJsxSignature(signature, callTarget);
             }
-            throw ts.Debug.assertNever(node, "Branch in 'resolveSignature' should be unreachable.");
+            return getTypeAtPosition(signature, argIndex);
         }
-        /**
-         * Resolve a signature of a given call-like expression.
-         * @param node a call-like expression to try resolve a signature for
-         * @param candidatesOutArray an array of signature to be filled in by the function. It is passed by signature help in the language service;
-         *                           the function will fill it up with appropriate candidate signatures
-         * @return a signature of the call-like expression or undefined if one can't be found
-         */
-        function getResolvedSignature(node, candidatesOutArray, checkMode) {
-            var links = getNodeLinks(node);
-            // If getResolvedSignature has already been called, we will have cached the resolvedSignature.
-            // However, it is possible that either candidatesOutArray was not passed in the first time,
-            // or that a different candidatesOutArray was passed in. Therefore, we need to redo the work
-            // to correctly fill the candidatesOutArray.
-            var cached = links.resolvedSignature;
-            if (cached && cached !== resolvingSignature && !candidatesOutArray) {
-                return cached;
-            }
-            links.resolvedSignature = resolvingSignature;
-            var result = resolveSignature(node, candidatesOutArray, checkMode || 0 /* Normal */);
-            // When CheckMode.SkipGenericFunctions is set we use resolvingSignature to indicate that call
-            // resolution should be deferred.
-            if (result !== resolvingSignature) {
-                // If signature resolution originated in control flow type analysis (for example to compute the
-                // assigned type in a flow assignment) we don't cache the result as it may be based on temporary
-                // types from the control flow analysis.
-                links.resolvedSignature = flowLoopStart === flowLoopCount ? result : cached;
+        function getContextualTypeForSubstitutionExpression(template, substitutionExpression) {
+            if (template.parent.kind === 205 /* TaggedTemplateExpression */) {
+                return getContextualTypeForArgument(template.parent, substitutionExpression);
             }
-            return result;
+            return undefined;
         }
-        /**
-         * Indicates whether a declaration can be treated as a constructor in a JavaScript
-         * file.
-         */
-        function isJSConstructor(node) {
-            if (!node || !ts.isInJSFile(node)) {
-                return false;
-            }
-            var func = ts.isFunctionDeclaration(node) || ts.isFunctionExpression(node) ? node :
-                ts.isVariableDeclaration(node) && node.initializer && ts.isFunctionExpression(node.initializer) ? node.initializer :
-                    undefined;
-            if (func) {
-                // If the node has a @class tag, treat it like a constructor.
-                if (ts.getJSDocClassTag(node))
-                    return true;
-                // If the symbol of the node has members, treat it like a constructor.
-                var symbol = getSymbolOfNode(func);
-                return !!symbol && ts.hasEntries(symbol.members);
+        function getContextualTypeForBinaryOperand(node, contextFlags) {
+            var binaryExpression = node.parent;
+            var left = binaryExpression.left, operatorToken = binaryExpression.operatorToken, right = binaryExpression.right;
+            switch (operatorToken.kind) {
+                case 62 /* EqualsToken */:
+                case 75 /* AmpersandAmpersandEqualsToken */:
+                case 74 /* BarBarEqualsToken */:
+                case 76 /* QuestionQuestionEqualsToken */:
+                    return node === right ? getContextualTypeForAssignmentDeclaration(binaryExpression) : undefined;
+                case 56 /* BarBarToken */:
+                case 60 /* QuestionQuestionToken */:
+                    // When an || expression has a contextual type, the operands are contextually typed by that type, except
+                    // when that type originates in a binding pattern, the right operand is contextually typed by the type of
+                    // the left operand. When an || expression has no contextual type, the right operand is contextually typed
+                    // by the type of the left operand, except for the special case of Javascript declarations of the form
+                    // `namespace.prop = namespace.prop || {}`.
+                    var type = getContextualType(binaryExpression, contextFlags);
+                    return node === right && (type && type.pattern || !type && !ts.isDefaultedExpandoInitializer(binaryExpression)) ?
+                        getTypeOfExpression(left) : type;
+                case 55 /* AmpersandAmpersandToken */:
+                case 27 /* CommaToken */:
+                    return node === right ? getContextualType(binaryExpression, contextFlags) : undefined;
+                default:
+                    return undefined;
             }
-            return false;
         }
-        function mergeJSSymbols(target, source) {
-            if (source) {
-                var links = getSymbolLinks(source);
-                if (!links.inferredClassSymbol || !links.inferredClassSymbol.has("" + getSymbolId(target))) {
-                    var inferred = ts.isTransientSymbol(target) ? target : cloneSymbol(target);
-                    inferred.exports = inferred.exports || ts.createSymbolTable();
-                    inferred.members = inferred.members || ts.createSymbolTable();
-                    inferred.flags |= source.flags & 32 /* Class */;
-                    if (ts.hasEntries(source.exports)) {
-                        mergeSymbolTable(inferred.exports, source.exports);
+        // In an assignment expression, the right operand is contextually typed by the type of the left operand.
+        // Don't do this for assignment declarations unless there is a type tag on the assignment, to avoid circularity from checking the right operand.
+        function getContextualTypeForAssignmentDeclaration(binaryExpression) {
+            var kind = ts.getAssignmentDeclarationKind(binaryExpression);
+            switch (kind) {
+                case 0 /* None */:
+                    return getTypeOfExpression(binaryExpression.left);
+                case 5 /* Property */:
+                case 1 /* ExportsProperty */:
+                case 6 /* Prototype */:
+                case 3 /* PrototypeProperty */:
+                    if (isPossiblyAliasedThisProperty(binaryExpression, kind)) {
+                        return getContextualTypeForThisPropertyAssignment(binaryExpression, kind);
                     }
-                    if (ts.hasEntries(source.members)) {
-                        mergeSymbolTable(inferred.members, source.members);
+                    // If `binaryExpression.left` was assigned a symbol, then this is a new declaration; otherwise it is an assignment to an existing declaration.
+                    // See `bindStaticPropertyAssignment` in `binder.ts`.
+                    else if (!binaryExpression.left.symbol) {
+                        return getTypeOfExpression(binaryExpression.left);
                     }
-                    (links.inferredClassSymbol || (links.inferredClassSymbol = ts.createMap())).set("" + getSymbolId(inferred), inferred);
-                    return inferred;
-                }
-                return links.inferredClassSymbol.get("" + getSymbolId(target));
-            }
-        }
-        function getAssignedClassSymbol(decl) {
-            var assignmentSymbol = decl && decl.parent &&
-                (ts.isFunctionDeclaration(decl) && getSymbolOfNode(decl) ||
-                    ts.isBinaryExpression(decl.parent) && getSymbolOfNode(decl.parent.left) ||
-                    ts.isVariableDeclaration(decl.parent) && getSymbolOfNode(decl.parent));
-            var prototype = assignmentSymbol && assignmentSymbol.exports && assignmentSymbol.exports.get("prototype");
-            var init = prototype && prototype.valueDeclaration && getAssignedJSPrototype(prototype.valueDeclaration);
-            return init ? getSymbolOfNode(init) : undefined;
-        }
-        function getAssignedJSPrototype(node) {
-            if (!node.parent) {
-                return false;
-            }
-            var parent = node.parent;
-            while (parent && parent.kind === 194 /* PropertyAccessExpression */) {
-                parent = parent.parent;
-            }
-            if (parent && ts.isBinaryExpression(parent) && ts.isPrototypeAccess(parent.left) && parent.operatorToken.kind === 62 /* EqualsToken */) {
-                var right = ts.getInitializerOfBinaryExpression(parent);
-                return ts.isObjectLiteralExpression(right) && right;
+                    else {
+                        var decl = binaryExpression.left.symbol.valueDeclaration;
+                        if (!decl) {
+                            return undefined;
+                        }
+                        var lhs = ts.cast(binaryExpression.left, ts.isAccessExpression);
+                        var overallAnnotation = ts.getEffectiveTypeAnnotationNode(decl);
+                        if (overallAnnotation) {
+                            return getTypeFromTypeNode(overallAnnotation);
+                        }
+                        else if (ts.isIdentifier(lhs.expression)) {
+                            var id = lhs.expression;
+                            var parentSymbol = resolveName(id, id.escapedText, 111551 /* Value */, undefined, id.escapedText, /*isUse*/ true);
+                            if (parentSymbol) {
+                                var annotated = parentSymbol.valueDeclaration && ts.getEffectiveTypeAnnotationNode(parentSymbol.valueDeclaration);
+                                if (annotated) {
+                                    var nameStr = ts.getElementOrPropertyAccessName(lhs);
+                                    if (nameStr !== undefined) {
+                                        return getTypeOfPropertyOfContextualType(getTypeFromTypeNode(annotated), nameStr);
+                                    }
+                                }
+                                return undefined;
+                            }
+                        }
+                        return ts.isInJSFile(decl) ? undefined : getTypeOfExpression(binaryExpression.left);
+                    }
+                case 2 /* ModuleExports */:
+                case 4 /* ThisProperty */:
+                    return getContextualTypeForThisPropertyAssignment(binaryExpression, kind);
+                case 7 /* ObjectDefinePropertyValue */:
+                case 8 /* ObjectDefinePropertyExports */:
+                case 9 /* ObjectDefinePrototypeProperty */:
+                    return ts.Debug.fail("Does not apply");
+                default:
+                    return ts.Debug.assertNever(kind);
             }
         }
-        /**
-         * Syntactically and semantically checks a call or new expression.
-         * @param node The call/new expression to be checked.
-         * @returns On success, the expression's signature's return type. On failure, anyType.
-         */
-        function checkCallExpression(node, checkMode) {
-            if (!checkGrammarTypeArguments(node, node.typeArguments))
-                checkGrammarArguments(node.arguments);
-            var signature = getResolvedSignature(node, /*candidatesOutArray*/ undefined, checkMode);
-            if (signature === resolvingSignature) {
-                // CheckMode.SkipGenericFunctions is enabled and this is a call to a generic function that
-                // returns a function type. We defer checking and return nonInferrableType.
-                return nonInferrableType;
+        function isPossiblyAliasedThisProperty(declaration, kind) {
+            if (kind === void 0) { kind = ts.getAssignmentDeclarationKind(declaration); }
+            if (kind === 4 /* ThisProperty */) {
+                return true;
             }
-            if (node.expression.kind === 102 /* SuperKeyword */) {
-                return voidType;
+            if (!ts.isInJSFile(declaration) || kind !== 5 /* Property */ || !ts.isIdentifier(declaration.left.expression)) {
+                return false;
             }
-            if (node.kind === 197 /* NewExpression */) {
-                var declaration = signature.declaration;
-                if (declaration &&
-                    declaration.kind !== 162 /* Constructor */ &&
-                    declaration.kind !== 166 /* ConstructSignature */ &&
-                    declaration.kind !== 171 /* ConstructorType */ &&
-                    !ts.isJSDocConstructSignature(declaration) &&
-                    !isJSConstructor(declaration)) {
-                    // When resolved signature is a call signature (and not a construct signature) the result type is any
-                    if (noImplicitAny) {
-                        error(node, ts.Diagnostics.new_expression_whose_target_lacks_a_construct_signature_implicitly_has_an_any_type);
+            var name = declaration.left.expression.escapedText;
+            var symbol = resolveName(declaration.left, name, 111551 /* Value */, undefined, undefined, /*isUse*/ true, /*excludeGlobals*/ true);
+            return ts.isThisInitializedDeclaration(symbol === null || symbol === void 0 ? void 0 : symbol.valueDeclaration);
+        }
+        function getContextualTypeForThisPropertyAssignment(binaryExpression, kind) {
+            if (!binaryExpression.symbol)
+                return getTypeOfExpression(binaryExpression.left);
+            if (binaryExpression.symbol.valueDeclaration) {
+                var annotated = ts.getEffectiveTypeAnnotationNode(binaryExpression.symbol.valueDeclaration);
+                if (annotated) {
+                    var type = getTypeFromTypeNode(annotated);
+                    if (type) {
+                        return type;
                     }
-                    return anyType;
                 }
             }
-            // In JavaScript files, calls to any identifier 'require' are treated as external module imports
-            if (ts.isInJSFile(node) && isCommonJsRequire(node)) {
-                return resolveExternalModuleTypeByLiteral(node.arguments[0]);
-            }
-            var returnType = getReturnTypeOfSignature(signature);
-            // Treat any call to the global 'Symbol' function that is part of a const variable or readonly property
-            // as a fresh unique symbol literal type.
-            if (returnType.flags & 12288 /* ESSymbolLike */ && isSymbolOrSymbolForCall(node)) {
-                return getESSymbolLikeTypeForNode(ts.walkUpParenthesizedExpressions(node.parent));
+            if (kind === 2 /* ModuleExports */)
+                return undefined;
+            var thisAccess = ts.cast(binaryExpression.left, ts.isAccessExpression);
+            if (!ts.isObjectLiteralMethod(ts.getThisContainer(thisAccess.expression, /*includeArrowFunctions*/ false))) {
+                return undefined;
             }
-            if (node.kind === 196 /* CallExpression */ && node.parent.kind === 226 /* ExpressionStatement */ &&
-                returnType.flags & 16384 /* Void */ && getTypePredicateOfSignature(signature)) {
-                if (!ts.isDottedName(node.expression)) {
-                    error(node.expression, ts.Diagnostics.Assertions_require_the_call_target_to_be_an_identifier_or_qualified_name);
+            var thisType = checkThisExpression(thisAccess.expression);
+            var nameStr = ts.getElementOrPropertyAccessName(thisAccess);
+            return nameStr !== undefined && getTypeOfPropertyOfContextualType(thisType, nameStr) || undefined;
+        }
+        function isCircularMappedProperty(symbol) {
+            return !!(ts.getCheckFlags(symbol) & 262144 /* Mapped */ && !symbol.type && findResolutionCycleStartIndex(symbol, 0 /* Type */) >= 0);
+        }
+        function getTypeOfPropertyOfContextualType(type, name) {
+            return mapType(type, function (t) {
+                if (isGenericMappedType(t)) {
+                    var constraint = getConstraintTypeFromMappedType(t);
+                    var constraintOfConstraint = getBaseConstraintOfType(constraint) || constraint;
+                    var propertyNameType = getLiteralType(ts.unescapeLeadingUnderscores(name));
+                    if (isTypeAssignableTo(propertyNameType, constraintOfConstraint)) {
+                        return substituteIndexedMappedType(t, propertyNameType);
+                    }
                 }
-                else if (!getEffectsSignature(node)) {
-                    var diagnostic = error(node.expression, ts.Diagnostics.Assertions_require_every_name_in_the_call_target_to_be_declared_with_an_explicit_type_annotation);
-                    getTypeOfDottedName(node.expression, diagnostic);
+                else if (t.flags & 3670016 /* StructuredType */) {
+                    var prop = getPropertyOfType(t, name);
+                    if (prop) {
+                        return isCircularMappedProperty(prop) ? undefined : getTypeOfSymbol(prop);
+                    }
+                    if (isTupleType(t)) {
+                        var restType = getRestTypeOfTupleType(t);
+                        if (restType && isNumericLiteralName(name) && +name >= 0) {
+                            return restType;
+                        }
+                    }
+                    return isNumericLiteralName(name) && getIndexTypeOfContextualType(t, 1 /* Number */) ||
+                        getIndexTypeOfContextualType(t, 0 /* String */);
                 }
+                return undefined;
+            }, /*noReductions*/ true);
+        }
+        function getIndexTypeOfContextualType(type, kind) {
+            return mapType(type, function (t) { return getIndexTypeOfStructuredType(t, kind); }, /*noReductions*/ true);
+        }
+        // In an object literal contextually typed by a type T, the contextual type of a property assignment is the type of
+        // the matching property in T, if one exists. Otherwise, it is the type of the numeric index signature in T, if one
+        // exists. Otherwise, it is the type of the string index signature in T, if one exists.
+        function getContextualTypeForObjectLiteralMethod(node, contextFlags) {
+            ts.Debug.assert(ts.isObjectLiteralMethod(node));
+            if (node.flags & 16777216 /* InWithStatement */) {
+                // We cannot answer semantic questions within a with block, do not proceed any further
+                return undefined;
             }
-            if (ts.isInJSFile(node)) {
-                var decl = ts.getDeclarationOfExpando(node);
-                if (decl) {
-                    var jsSymbol = getSymbolOfNode(decl);
-                    if (jsSymbol && ts.hasEntries(jsSymbol.exports)) {
-                        var jsAssignmentType = createAnonymousType(jsSymbol, jsSymbol.exports, ts.emptyArray, ts.emptyArray, undefined, undefined);
-                        jsAssignmentType.objectFlags |= 16384 /* JSLiteral */;
-                        return getIntersectionType([returnType, jsAssignmentType]);
+            return getContextualTypeForObjectLiteralElement(node, contextFlags);
+        }
+        function getContextualTypeForObjectLiteralElement(element, contextFlags) {
+            var objectLiteral = element.parent;
+            var type = getApparentTypeOfContextualType(objectLiteral, contextFlags);
+            if (type) {
+                if (!hasNonBindableDynamicName(element)) {
+                    // For a (non-symbol) computed property, there is no reason to look up the name
+                    // in the type. It will just be "__computed", which does not appear in any
+                    // SymbolTable.
+                    var symbolName_3 = getSymbolOfNode(element).escapedName;
+                    var propertyType = getTypeOfPropertyOfContextualType(type, symbolName_3);
+                    if (propertyType) {
+                        return propertyType;
                     }
                 }
+                return isNumericName(element.name) && getIndexTypeOfContextualType(type, 1 /* Number */) ||
+                    getIndexTypeOfContextualType(type, 0 /* String */);
             }
-            return returnType;
+            return undefined;
         }
-        function isSymbolOrSymbolForCall(node) {
-            if (!ts.isCallExpression(node))
-                return false;
-            var left = node.expression;
-            if (ts.isPropertyAccessExpression(left) && left.name.escapedText === "for") {
-                left = left.expression;
-            }
-            if (!ts.isIdentifier(left) || left.escapedText !== "Symbol") {
-                return false;
-            }
-            // make sure `Symbol` is the global symbol
-            var globalESSymbol = getGlobalESSymbolConstructorSymbol(/*reportErrors*/ false);
-            if (!globalESSymbol) {
-                return false;
+        // In an array literal contextually typed by a type T, the contextual type of an element expression at index N is
+        // the type of the property with the numeric name N in T, if one exists. Otherwise, if T has a numeric index signature,
+        // it is the type of the numeric index signature in T. Otherwise, in ES6 and higher, the contextual type is the iterated
+        // type of T.
+        function getContextualTypeForElementExpression(arrayContextualType, index) {
+            return arrayContextualType && (getTypeOfPropertyOfContextualType(arrayContextualType, "" + index)
+                || mapType(arrayContextualType, function (t) { return getIteratedTypeOrElementType(1 /* Element */, t, undefinedType, /*errorNode*/ undefined, /*checkAssignability*/ false); }, 
+                /*noReductions*/ true));
+        }
+        // In a contextually typed conditional expression, the true/false expressions are contextually typed by the same type.
+        function getContextualTypeForConditionalOperand(node, contextFlags) {
+            var conditional = node.parent;
+            return node === conditional.whenTrue || node === conditional.whenFalse ? getContextualType(conditional, contextFlags) : undefined;
+        }
+        function getContextualTypeForChildJsxExpression(node, child) {
+            var attributesType = getApparentTypeOfContextualType(node.openingElement.tagName);
+            // JSX expression is in children of JSX Element, we will look for an "children" atttribute (we get the name from JSX.ElementAttributesProperty)
+            var jsxChildrenPropertyName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(node));
+            if (!(attributesType && !isTypeAny(attributesType) && jsxChildrenPropertyName && jsxChildrenPropertyName !== "")) {
+                return undefined;
             }
-            return globalESSymbol === resolveName(left, "Symbol", 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
+            var realChildren = ts.getSemanticJsxChildren(node.children);
+            var childIndex = realChildren.indexOf(child);
+            var childFieldType = getTypeOfPropertyOfContextualType(attributesType, jsxChildrenPropertyName);
+            return childFieldType && (realChildren.length === 1 ? childFieldType : mapType(childFieldType, function (t) {
+                if (isArrayLikeType(t)) {
+                    return getIndexedAccessType(t, getLiteralType(childIndex));
+                }
+                else {
+                    return t;
+                }
+            }, /*noReductions*/ true));
         }
-        function checkImportCallExpression(node) {
-            // Check grammar of dynamic import
-            if (!checkGrammarArguments(node.arguments))
-                checkGrammarImportCallExpression(node);
-            if (node.arguments.length === 0) {
-                return createPromiseReturnType(node, anyType);
+        function getContextualTypeForJsxExpression(node) {
+            var exprParent = node.parent;
+            return ts.isJsxAttributeLike(exprParent)
+                ? getContextualType(node)
+                : ts.isJsxElement(exprParent)
+                    ? getContextualTypeForChildJsxExpression(exprParent, node)
+                    : undefined;
+        }
+        function getContextualTypeForJsxAttribute(attribute) {
+            // When we trying to resolve JsxOpeningLikeElement as a stateless function element, we will already give its attributes a contextual type
+            // which is a type of the parameter of the signature we are trying out.
+            // If there is no contextual type (e.g. we are trying to resolve stateful component), get attributes type from resolving element's tagName
+            if (ts.isJsxAttribute(attribute)) {
+                var attributesType = getApparentTypeOfContextualType(attribute.parent);
+                if (!attributesType || isTypeAny(attributesType)) {
+                    return undefined;
+                }
+                return getTypeOfPropertyOfContextualType(attributesType, attribute.name.escapedText);
             }
-            var specifier = node.arguments[0];
-            var specifierType = checkExpressionCached(specifier);
-            // Even though multiple arguments is grammatically incorrect, type-check extra arguments for completion
-            for (var i = 1; i < node.arguments.length; ++i) {
-                checkExpressionCached(node.arguments[i]);
+            else {
+                return getContextualType(attribute.parent);
             }
-            if (specifierType.flags & 32768 /* Undefined */ || specifierType.flags & 65536 /* Null */ || !isTypeAssignableTo(specifierType, stringType)) {
-                error(specifier, ts.Diagnostics.Dynamic_import_s_specifier_must_be_of_type_string_but_here_has_type_0, typeToString(specifierType));
+        }
+        // Return true if the given expression is possibly a discriminant value. We limit the kinds of
+        // expressions we check to those that don't depend on their contextual type in order not to cause
+        // recursive (and possibly infinite) invocations of getContextualType.
+        function isPossiblyDiscriminantValue(node) {
+            switch (node.kind) {
+                case 10 /* StringLiteral */:
+                case 8 /* NumericLiteral */:
+                case 9 /* BigIntLiteral */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                case 103 /* NullKeyword */:
+                case 78 /* Identifier */:
+                case 150 /* UndefinedKeyword */:
+                    return true;
+                case 201 /* PropertyAccessExpression */:
+                case 207 /* ParenthesizedExpression */:
+                    return isPossiblyDiscriminantValue(node.expression);
+                case 283 /* JsxExpression */:
+                    return !node.expression || isPossiblyDiscriminantValue(node.expression);
             }
-            // resolveExternalModuleName will return undefined if the moduleReferenceExpression is not a string literal
-            var moduleSymbol = resolveExternalModuleName(node, specifier);
-            if (moduleSymbol) {
-                var esModuleSymbol = resolveESModuleSymbol(moduleSymbol, specifier, /*dontRecursivelyResolve*/ true, /*suppressUsageError*/ false);
-                if (esModuleSymbol) {
-                    return createPromiseReturnType(node, getTypeWithSyntheticDefaultImportType(getTypeOfSymbol(esModuleSymbol), esModuleSymbol, moduleSymbol));
+            return false;
+        }
+        function discriminateContextualTypeByObjectMembers(node, contextualType) {
+            return discriminateTypeByDiscriminableItems(contextualType, ts.map(ts.filter(node.properties, function (p) { return !!p.symbol && p.kind === 288 /* PropertyAssignment */ && isPossiblyDiscriminantValue(p.initializer) && isDiscriminantProperty(contextualType, p.symbol.escapedName); }), function (prop) { return [function () { return checkExpression(prop.initializer); }, prop.symbol.escapedName]; }), isTypeAssignableTo, contextualType);
+        }
+        function discriminateContextualTypeByJSXAttributes(node, contextualType) {
+            return discriminateTypeByDiscriminableItems(contextualType, ts.map(ts.filter(node.properties, function (p) { return !!p.symbol && p.kind === 280 /* JsxAttribute */ && isDiscriminantProperty(contextualType, p.symbol.escapedName) && (!p.initializer || isPossiblyDiscriminantValue(p.initializer)); }), function (prop) { return [!prop.initializer ? (function () { return trueType; }) : (function () { return checkExpression(prop.initializer); }), prop.symbol.escapedName]; }), isTypeAssignableTo, contextualType);
+        }
+        // Return the contextual type for a given expression node. During overload resolution, a contextual type may temporarily
+        // be "pushed" onto a node using the contextualType property.
+        function getApparentTypeOfContextualType(node, contextFlags) {
+            var contextualType = ts.isObjectLiteralMethod(node) ?
+                getContextualTypeForObjectLiteralMethod(node, contextFlags) :
+                getContextualType(node, contextFlags);
+            var instantiatedType = instantiateContextualType(contextualType, node, contextFlags);
+            if (instantiatedType && !(contextFlags && contextFlags & 2 /* NoConstraints */ && instantiatedType.flags & 8650752 /* TypeVariable */)) {
+                var apparentType = mapType(instantiatedType, getApparentType, /*noReductions*/ true);
+                if (apparentType.flags & 1048576 /* Union */) {
+                    if (ts.isObjectLiteralExpression(node)) {
+                        return discriminateContextualTypeByObjectMembers(node, apparentType);
+                    }
+                    else if (ts.isJsxAttributes(node)) {
+                        return discriminateContextualTypeByJSXAttributes(node, apparentType);
+                    }
                 }
+                return apparentType;
             }
-            return createPromiseReturnType(node, anyType);
         }
-        function getTypeWithSyntheticDefaultImportType(type, symbol, originalSymbol) {
-            if (allowSyntheticDefaultImports && type && type !== errorType) {
-                var synthType = type;
-                if (!synthType.syntheticType) {
-                    var file = ts.find(originalSymbol.declarations, ts.isSourceFile);
-                    var hasSyntheticDefault = canHaveSyntheticDefault(file, originalSymbol, /*dontResolveAlias*/ false);
-                    if (hasSyntheticDefault) {
-                        var memberTable = ts.createSymbolTable();
-                        var newSymbol = createSymbol(2097152 /* Alias */, "default" /* Default */);
-                        newSymbol.nameType = getLiteralType("default");
-                        newSymbol.target = resolveSymbol(symbol);
-                        memberTable.set("default" /* Default */, newSymbol);
-                        var anonymousSymbol = createSymbol(2048 /* TypeLiteral */, "__type" /* Type */);
-                        var defaultContainingObject = createAnonymousType(anonymousSymbol, memberTable, ts.emptyArray, ts.emptyArray, /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined);
-                        anonymousSymbol.type = defaultContainingObject;
-                        synthType.syntheticType = isValidSpreadType(type) ? getSpreadType(type, defaultContainingObject, anonymousSymbol, /*objectFlags*/ 0, /*readonly*/ false) : defaultContainingObject;
+        // If the given contextual type contains instantiable types and if a mapper representing
+        // return type inferences is available, instantiate those types using that mapper.
+        function instantiateContextualType(contextualType, node, contextFlags) {
+            if (contextualType && maybeTypeOfKind(contextualType, 465829888 /* Instantiable */)) {
+                var inferenceContext = getInferenceContext(node);
+                // If no inferences have been made, nothing is gained from instantiating as type parameters
+                // would just be replaced with their defaults similar to the apparent type.
+                if (inferenceContext && ts.some(inferenceContext.inferences, hasInferenceCandidates)) {
+                    // For contextual signatures we incorporate all inferences made so far, e.g. from return
+                    // types as well as arguments to the left in a function call.
+                    if (contextFlags && contextFlags & 1 /* Signature */) {
+                        return instantiateInstantiableTypes(contextualType, inferenceContext.nonFixingMapper);
                     }
-                    else {
-                        synthType.syntheticType = type;
+                    // For other purposes (e.g. determining whether to produce literal types) we only
+                    // incorporate inferences made from the return type in a function call.
+                    if (inferenceContext.returnMapper) {
+                        return instantiateInstantiableTypes(contextualType, inferenceContext.returnMapper);
                     }
                 }
-                return synthType.syntheticType;
             }
-            return type;
+            return contextualType;
         }
-        function isCommonJsRequire(node) {
-            if (!ts.isRequireCall(node, /*checkArgumentIsStringLiteralLike*/ true)) {
-                return false;
-            }
-            // Make sure require is not a local function
-            if (!ts.isIdentifier(node.expression))
-                return ts.Debug.fail();
-            var resolvedRequire = resolveName(node.expression, node.expression.escapedText, 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true); // TODO: GH#18217
-            if (resolvedRequire === requireSymbol) {
-                return true;
+        // This function is similar to instantiateType, except that (a) it only instantiates types that
+        // are classified as instantiable (i.e. it doesn't instantiate object types), and (b) it performs
+        // no reductions on instantiated union types.
+        function instantiateInstantiableTypes(type, mapper) {
+            if (type.flags & 465829888 /* Instantiable */) {
+                return instantiateType(type, mapper);
             }
-            // project includes symbol named 'require' - make sure that it is ambient and local non-alias
-            if (resolvedRequire.flags & 2097152 /* Alias */) {
-                return false;
+            if (type.flags & 1048576 /* Union */) {
+                return getUnionType(ts.map(type.types, function (t) { return instantiateInstantiableTypes(t, mapper); }), 0 /* None */);
             }
-            var targetDeclarationKind = resolvedRequire.flags & 16 /* Function */
-                ? 244 /* FunctionDeclaration */
-                : resolvedRequire.flags & 3 /* Variable */
-                    ? 242 /* VariableDeclaration */
-                    : 0 /* Unknown */;
-            if (targetDeclarationKind !== 0 /* Unknown */) {
-                var decl = ts.getDeclarationOfKind(resolvedRequire, targetDeclarationKind);
-                // function/variable declaration should be ambient
-                return !!decl && !!(decl.flags & 8388608 /* Ambient */);
+            if (type.flags & 2097152 /* Intersection */) {
+                return getIntersectionType(ts.map(type.types, function (t) { return instantiateInstantiableTypes(t, mapper); }));
             }
-            return false;
+            return type;
         }
-        function checkTaggedTemplateExpression(node) {
-            if (!checkGrammarTaggedTemplateChain(node))
-                checkGrammarTypeArguments(node, node.typeArguments);
-            if (languageVersion < 2 /* ES2015 */) {
-                checkExternalEmitHelpers(node, 131072 /* MakeTemplateObject */);
+        /**
+         * Whoa! Do you really want to use this function?
+         *
+         * Unless you're trying to get the *non-apparent* type for a
+         * value-literal type or you're authoring relevant portions of this algorithm,
+         * you probably meant to use 'getApparentTypeOfContextualType'.
+         * Otherwise this may not be very useful.
+         *
+         * In cases where you *are* working on this function, you should understand
+         * when it is appropriate to use 'getContextualType' and 'getApparentTypeOfContextualType'.
+         *
+         *   - Use 'getContextualType' when you are simply going to propagate the result to the expression.
+         *   - Use 'getApparentTypeOfContextualType' when you're going to need the members of the type.
+         *
+         * @param node the expression whose contextual type will be returned.
+         * @returns the contextual type of an expression.
+         */
+        function getContextualType(node, contextFlags) {
+            if (node.flags & 16777216 /* InWithStatement */) {
+                // We cannot answer semantic questions within a with block, do not proceed any further
+                return undefined;
             }
-            return getReturnTypeOfSignature(getResolvedSignature(node));
-        }
-        function checkAssertion(node) {
-            return checkAssertionWorker(node, node.type, node.expression);
-        }
-        function isValidConstAssertionArgument(node) {
-            switch (node.kind) {
-                case 10 /* StringLiteral */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 8 /* NumericLiteral */:
-                case 9 /* BigIntLiteral */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
-                case 192 /* ArrayLiteralExpression */:
-                case 193 /* ObjectLiteralExpression */:
-                    return true;
-                case 200 /* ParenthesizedExpression */:
-                    return isValidConstAssertionArgument(node.expression);
-                case 207 /* PrefixUnaryExpression */:
-                    var op = node.operator;
-                    var arg = node.operand;
-                    return op === 40 /* MinusToken */ && (arg.kind === 8 /* NumericLiteral */ || arg.kind === 9 /* BigIntLiteral */) ||
-                        op === 39 /* PlusToken */ && arg.kind === 8 /* NumericLiteral */;
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    var expr = node.expression;
-                    if (ts.isIdentifier(expr)) {
-                        var symbol = getSymbolAtLocation(expr);
-                        if (symbol && symbol.flags & 2097152 /* Alias */) {
-                            symbol = resolveAlias(symbol);
-                        }
-                        return !!(symbol && (symbol.flags & 384 /* Enum */) && getEnumKind(symbol) === 1 /* Literal */);
+            if (node.contextualType) {
+                return node.contextualType;
+            }
+            var parent = node.parent;
+            switch (parent.kind) {
+                case 249 /* VariableDeclaration */:
+                case 160 /* Parameter */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                case 198 /* BindingElement */:
+                    return getContextualTypeForInitializerExpression(node, contextFlags);
+                case 209 /* ArrowFunction */:
+                case 242 /* ReturnStatement */:
+                    return getContextualTypeForReturnExpression(node);
+                case 219 /* YieldExpression */:
+                    return getContextualTypeForYieldOperand(parent);
+                case 213 /* AwaitExpression */:
+                    return getContextualTypeForAwaitOperand(parent, contextFlags);
+                case 203 /* CallExpression */:
+                    if (parent.expression.kind === 99 /* ImportKeyword */) {
+                        return stringType;
                     }
-            }
-            return false;
-        }
-        function checkAssertionWorker(errNode, type, expression, checkMode) {
-            var exprType = checkExpression(expression, checkMode);
-            if (ts.isConstTypeReference(type)) {
-                if (!isValidConstAssertionArgument(expression)) {
-                    error(expression, ts.Diagnostics.A_const_assertions_can_only_be_applied_to_references_to_enum_members_or_string_number_boolean_array_or_object_literals);
+                /* falls through */
+                case 204 /* NewExpression */:
+                    return getContextualTypeForArgument(parent, node);
+                case 206 /* TypeAssertionExpression */:
+                case 224 /* AsExpression */:
+                    return ts.isConstTypeReference(parent.type) ? tryFindWhenConstTypeReference(parent) : getTypeFromTypeNode(parent.type);
+                case 216 /* BinaryExpression */:
+                    return getContextualTypeForBinaryOperand(node, contextFlags);
+                case 288 /* PropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
+                    return getContextualTypeForObjectLiteralElement(parent, contextFlags);
+                case 290 /* SpreadAssignment */:
+                    return getApparentTypeOfContextualType(parent.parent, contextFlags);
+                case 199 /* ArrayLiteralExpression */: {
+                    var arrayLiteral = parent;
+                    var type = getApparentTypeOfContextualType(arrayLiteral, contextFlags);
+                    return getContextualTypeForElementExpression(type, ts.indexOfNode(arrayLiteral.elements, node));
                 }
-                return getRegularTypeOfLiteralType(exprType);
+                case 217 /* ConditionalExpression */:
+                    return getContextualTypeForConditionalOperand(node, contextFlags);
+                case 228 /* TemplateSpan */:
+                    ts.Debug.assert(parent.parent.kind === 218 /* TemplateExpression */);
+                    return getContextualTypeForSubstitutionExpression(parent.parent, node);
+                case 207 /* ParenthesizedExpression */: {
+                    // Like in `checkParenthesizedExpression`, an `/** @type {xyz} */` comment before a parenthesized expression acts as a type cast.
+                    var tag = ts.isInJSFile(parent) ? ts.getJSDocTypeTag(parent) : undefined;
+                    return tag ? getTypeFromTypeNode(tag.typeExpression.type) : getContextualType(parent, contextFlags);
+                }
+                case 283 /* JsxExpression */:
+                    return getContextualTypeForJsxExpression(parent);
+                case 280 /* JsxAttribute */:
+                case 282 /* JsxSpreadAttribute */:
+                    return getContextualTypeForJsxAttribute(parent);
+                case 275 /* JsxOpeningElement */:
+                case 274 /* JsxSelfClosingElement */:
+                    return getContextualJsxElementAttributesType(parent, contextFlags);
             }
-            checkSourceElement(type);
-            exprType = getRegularTypeOfObjectLiteral(getBaseTypeOfLiteralType(exprType));
-            var targetType = getTypeFromTypeNode(type);
-            if (produceDiagnostics && targetType !== errorType) {
-                var widenedType = getWidenedType(exprType);
-                if (!isTypeComparableTo(targetType, widenedType)) {
-                    checkTypeComparableTo(exprType, targetType, errNode, ts.Diagnostics.Conversion_of_type_0_to_type_1_may_be_a_mistake_because_neither_type_sufficiently_overlaps_with_the_other_If_this_was_intentional_convert_the_expression_to_unknown_first);
+            return undefined;
+            function tryFindWhenConstTypeReference(node) {
+                if (ts.isCallLikeExpression(node.parent)) {
+                    return getContextualTypeForArgument(node.parent, node);
                 }
+                return undefined;
             }
-            return targetType;
-        }
-        function checkNonNullChain(node) {
-            var leftType = checkExpression(node.expression);
-            var nonOptionalType = getOptionalExpressionType(leftType, node.expression);
-            return propagateOptionalTypeMarker(getNonNullableType(nonOptionalType), node, nonOptionalType !== leftType);
         }
-        function checkNonNullAssertion(node) {
-            return node.flags & 32 /* OptionalChain */ ? checkNonNullChain(node) :
-                getNonNullableType(checkExpression(node.expression));
+        function getInferenceContext(node) {
+            var ancestor = ts.findAncestor(node, function (n) { return !!n.inferenceContext; });
+            return ancestor && ancestor.inferenceContext;
         }
-        function checkMetaProperty(node) {
-            checkGrammarMetaProperty(node);
-            if (node.keywordToken === 99 /* NewKeyword */) {
-                return checkNewTargetMetaProperty(node);
-            }
-            if (node.keywordToken === 96 /* ImportKeyword */) {
-                return checkImportMetaProperty(node);
+        function getContextualJsxElementAttributesType(node, contextFlags) {
+            if (ts.isJsxOpeningElement(node) && node.parent.contextualType && contextFlags !== 4 /* Completions */) {
+                // Contextually applied type is moved from attributes up to the outer jsx attributes so when walking up from the children they get hit
+                // _However_ to hit them from the _attributes_ we must look for them here; otherwise we'll used the declared type
+                // (as below) instead!
+                return node.parent.contextualType;
             }
-            return ts.Debug.assertNever(node.keywordToken);
+            return getContextualTypeForArgumentAtIndex(node, 0);
         }
-        function checkNewTargetMetaProperty(node) {
-            var container = ts.getNewTargetContainer(node);
-            if (!container) {
-                error(node, ts.Diagnostics.Meta_property_0_is_only_allowed_in_the_body_of_a_function_declaration_function_expression_or_constructor, "new.target");
-                return errorType;
-            }
-            else if (container.kind === 162 /* Constructor */) {
-                var symbol = getSymbolOfNode(container.parent);
-                return getTypeOfSymbol(symbol);
-            }
-            else {
-                var symbol = getSymbolOfNode(container);
-                return getTypeOfSymbol(symbol);
-            }
+        function getEffectiveFirstArgumentForJsxSignature(signature, node) {
+            return getJsxReferenceKind(node) !== 0 /* Component */
+                ? getJsxPropsTypeFromCallSignature(signature, node)
+                : getJsxPropsTypeFromClassType(signature, node);
         }
-        function checkImportMetaProperty(node) {
-            if (moduleKind !== ts.ModuleKind.ESNext && moduleKind !== ts.ModuleKind.System) {
-                error(node, ts.Diagnostics.The_import_meta_meta_property_is_only_allowed_when_the_module_option_is_esnext_or_system);
+        function getJsxPropsTypeFromCallSignature(sig, context) {
+            var propsType = getTypeOfFirstParameterOfSignatureWithFallback(sig, unknownType);
+            propsType = getJsxManagedAttributesFromLocatedAttributes(context, getJsxNamespaceAt(context), propsType);
+            var intrinsicAttribs = getJsxType(JsxNames.IntrinsicAttributes, context);
+            if (intrinsicAttribs !== errorType) {
+                propsType = intersectTypes(intrinsicAttribs, propsType);
             }
-            var file = ts.getSourceFileOfNode(node);
-            ts.Debug.assert(!!(file.flags & 2097152 /* PossiblyContainsImportMeta */), "Containing file is missing import meta node flag.");
-            ts.Debug.assert(!!file.externalModuleIndicator, "Containing file should be a module.");
-            return node.name.escapedText === "meta" ? getGlobalImportMetaType() : errorType;
+            return propsType;
         }
-        function getTypeOfParameter(symbol) {
-            var type = getTypeOfSymbol(symbol);
-            if (strictNullChecks) {
-                var declaration = symbol.valueDeclaration;
-                if (declaration && ts.hasInitializer(declaration)) {
-                    return getOptionalType(type);
+        function getJsxPropsTypeForSignatureFromMember(sig, forcedLookupLocation) {
+            if (sig.unionSignatures) {
+                // JSX Elements using the legacy `props`-field based lookup (eg, react class components) need to treat the `props` member as an input
+                // instead of an output position when resolving the signature. We need to go back to the input signatures of the composite signature,
+                // get the type of `props` on each return type individually, and then _intersect them_, rather than union them (as would normally occur
+                // for a union signature). It's an unfortunate quirk of looking in the output of the signature for the type we want to use for the input.
+                // The default behavior of `getTypeOfFirstParameterOfSignatureWithFallback` when no `props` member name is defined is much more sane.
+                var results = [];
+                for (var _i = 0, _a = sig.unionSignatures; _i < _a.length; _i++) {
+                    var signature = _a[_i];
+                    var instance = getReturnTypeOfSignature(signature);
+                    if (isTypeAny(instance)) {
+                        return instance;
+                    }
+                    var propType = getTypeOfPropertyOfType(instance, forcedLookupLocation);
+                    if (!propType) {
+                        return;
+                    }
+                    results.push(propType);
                 }
+                return getIntersectionType(results);
             }
-            return type;
+            var instanceType = getReturnTypeOfSignature(sig);
+            return isTypeAny(instanceType) ? instanceType : getTypeOfPropertyOfType(instanceType, forcedLookupLocation);
         }
-        function getParameterNameAtPosition(signature, pos) {
-            var paramCount = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
-            if (pos < paramCount) {
-                return signature.parameters[pos].escapedName;
+        function getStaticTypeOfReferencedJsxConstructor(context) {
+            if (isJsxIntrinsicIdentifier(context.tagName)) {
+                var result = getIntrinsicAttributesTypeFromJsxOpeningLikeElement(context);
+                var fakeSignature = createSignatureForJSXIntrinsic(context, result);
+                return getOrCreateTypeFromSignature(fakeSignature);
             }
-            var restParameter = signature.parameters[paramCount] || unknownSymbol;
-            var restType = getTypeOfSymbol(restParameter);
-            if (isTupleType(restType)) {
-                var associatedNames = restType.target.associatedNames;
-                var index = pos - paramCount;
-                return associatedNames && associatedNames[index] || restParameter.escapedName + "_" + index;
+            var tagType = checkExpressionCached(context.tagName);
+            if (tagType.flags & 128 /* StringLiteral */) {
+                var result = getIntrinsicAttributesTypeFromStringLiteralType(tagType, context);
+                if (!result) {
+                    return errorType;
+                }
+                var fakeSignature = createSignatureForJSXIntrinsic(context, result);
+                return getOrCreateTypeFromSignature(fakeSignature);
             }
-            return restParameter.escapedName;
-        }
-        function getTypeAtPosition(signature, pos) {
-            return tryGetTypeAtPosition(signature, pos) || anyType;
+            return tagType;
         }
-        function tryGetTypeAtPosition(signature, pos) {
-            var paramCount = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
-            if (pos < paramCount) {
-                return getTypeOfParameter(signature.parameters[pos]);
-            }
-            if (signatureHasRestParameter(signature)) {
-                // We want to return the value undefined for an out of bounds parameter position,
-                // so we need to check bounds here before calling getIndexedAccessType (which
-                // otherwise would return the type 'undefined').
-                var restType = getTypeOfSymbol(signature.parameters[paramCount]);
-                var index = pos - paramCount;
-                if (!isTupleType(restType) || restType.target.hasRestElement || index < getTypeArguments(restType).length) {
-                    return getIndexedAccessType(restType, getLiteralType(index));
+        function getJsxManagedAttributesFromLocatedAttributes(context, ns, attributesType) {
+            var managedSym = getJsxLibraryManagedAttributes(ns);
+            if (managedSym) {
+                var declaredManagedType = getDeclaredTypeOfSymbol(managedSym);
+                var ctorType = getStaticTypeOfReferencedJsxConstructor(context);
+                if (ts.length(declaredManagedType.typeParameters) >= 2) {
+                    var args = fillMissingTypeArguments([ctorType, attributesType], declaredManagedType.typeParameters, 2, ts.isInJSFile(context));
+                    return createTypeReference(declaredManagedType, args);
+                }
+                else if (ts.length(declaredManagedType.aliasTypeArguments) >= 2) {
+                    var args = fillMissingTypeArguments([ctorType, attributesType], declaredManagedType.aliasTypeArguments, 2, ts.isInJSFile(context));
+                    return getTypeAliasInstantiation(declaredManagedType.aliasSymbol, args);
                 }
             }
-            return undefined;
+            return attributesType;
         }
-        function getRestTypeAtPosition(source, pos) {
-            var paramCount = getParameterCount(source);
-            var restType = getEffectiveRestType(source);
-            var nonRestCount = paramCount - (restType ? 1 : 0);
-            if (restType && pos === nonRestCount) {
-                return restType;
-            }
-            var types = [];
-            var names = [];
-            for (var i = pos; i < nonRestCount; i++) {
-                types.push(getTypeAtPosition(source, i));
-                names.push(getParameterNameAtPosition(source, i));
+        function getJsxPropsTypeFromClassType(sig, context) {
+            var ns = getJsxNamespaceAt(context);
+            var forcedLookupLocation = getJsxElementPropertiesName(ns);
+            var attributesType = forcedLookupLocation === undefined
+                // If there is no type ElementAttributesProperty, return the type of the first parameter of the signature, which should be the props type
+                ? getTypeOfFirstParameterOfSignatureWithFallback(sig, unknownType)
+                : forcedLookupLocation === ""
+                    // If there is no e.g. 'props' member in ElementAttributesProperty, use the element class type instead
+                    ? getReturnTypeOfSignature(sig)
+                    // Otherwise get the type of the property on the signature return type
+                    : getJsxPropsTypeForSignatureFromMember(sig, forcedLookupLocation);
+            if (!attributesType) {
+                // There is no property named 'props' on this instance type
+                if (!!forcedLookupLocation && !!ts.length(context.attributes.properties)) {
+                    error(context, ts.Diagnostics.JSX_element_class_does_not_support_attributes_because_it_does_not_have_a_0_property, ts.unescapeLeadingUnderscores(forcedLookupLocation));
+                }
+                return unknownType;
             }
-            if (restType) {
-                types.push(getIndexedAccessType(restType, numberType));
-                names.push(getParameterNameAtPosition(source, nonRestCount));
+            attributesType = getJsxManagedAttributesFromLocatedAttributes(context, ns, attributesType);
+            if (isTypeAny(attributesType)) {
+                // Props is of type 'any' or unknown
+                return attributesType;
             }
-            var minArgumentCount = getMinArgumentCount(source);
-            var minLength = minArgumentCount < pos ? 0 : minArgumentCount - pos;
-            return createTupleType(types, minLength, !!restType, /*readonly*/ false, names);
-        }
-        function getParameterCount(signature) {
-            var length = signature.parameters.length;
-            if (signatureHasRestParameter(signature)) {
-                var restType = getTypeOfSymbol(signature.parameters[length - 1]);
-                if (isTupleType(restType)) {
-                    return length + getTypeArguments(restType).length - 1;
+            else {
+                // Normal case -- add in IntrinsicClassElements<T> and IntrinsicElements
+                var apparentAttributesType = attributesType;
+                var intrinsicClassAttribs = getJsxType(JsxNames.IntrinsicClassAttributes, context);
+                if (intrinsicClassAttribs !== errorType) {
+                    var typeParams = getLocalTypeParametersOfClassOrInterfaceOrTypeAlias(intrinsicClassAttribs.symbol);
+                    var hostClassType = getReturnTypeOfSignature(sig);
+                    apparentAttributesType = intersectTypes(typeParams
+                        ? createTypeReference(intrinsicClassAttribs, fillMissingTypeArguments([hostClassType], typeParams, getMinTypeArgumentCount(typeParams), ts.isInJSFile(context)))
+                        : intrinsicClassAttribs, apparentAttributesType);
+                }
+                var intrinsicAttribs = getJsxType(JsxNames.IntrinsicAttributes, context);
+                if (intrinsicAttribs !== errorType) {
+                    apparentAttributesType = intersectTypes(intrinsicAttribs, apparentAttributesType);
                 }
+                return apparentAttributesType;
             }
-            return length;
         }
-        function getMinArgumentCount(signature, strongArityForUntypedJS) {
-            if (signatureHasRestParameter(signature)) {
-                var restType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
-                if (isTupleType(restType)) {
-                    var minLength = restType.target.minLength;
-                    if (minLength > 0) {
-                        return signature.parameters.length - 1 + minLength;
-                    }
+        // If the given type is an object or union type with a single signature, and if that signature has at
+        // least as many parameters as the given function, return the signature. Otherwise return undefined.
+        function getContextualCallSignature(type, node) {
+            var signatures = getSignaturesOfType(type, 0 /* Call */);
+            if (signatures.length === 1) {
+                var signature = signatures[0];
+                if (!isAritySmaller(signature, node)) {
+                    return signature;
                 }
             }
-            if (!strongArityForUntypedJS && signature.flags & 16 /* IsUntypedSignatureInJSFile */) {
-                return 0;
-            }
-            return signature.minArgumentCount;
         }
-        function hasEffectiveRestParameter(signature) {
-            if (signatureHasRestParameter(signature)) {
-                var restType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
-                return !isTupleType(restType) || restType.target.hasRestElement;
+        /** If the contextual signature has fewer parameters than the function expression, do not use it */
+        function isAritySmaller(signature, target) {
+            var targetParameterCount = 0;
+            for (; targetParameterCount < target.parameters.length; targetParameterCount++) {
+                var param = target.parameters[targetParameterCount];
+                if (param.initializer || param.questionToken || param.dotDotDotToken || isJSDocOptionalParameter(param)) {
+                    break;
+                }
             }
-            return false;
-        }
-        function getEffectiveRestType(signature) {
-            if (signatureHasRestParameter(signature)) {
-                var restType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
-                return isTupleType(restType) ? getRestArrayTypeOfTupleType(restType) : restType;
+            if (target.parameters.length && ts.parameterIsThisKeyword(target.parameters[0])) {
+                targetParameterCount--;
             }
-            return undefined;
-        }
-        function getNonArrayRestType(signature) {
-            var restType = getEffectiveRestType(signature);
-            return restType && !isArrayType(restType) && !isTypeAny(restType) && (getReducedType(restType).flags & 131072 /* Never */) === 0 ? restType : undefined;
+            return !hasEffectiveRestParameter(signature) && getParameterCount(signature) < targetParameterCount;
         }
-        function getTypeOfFirstParameterOfSignature(signature) {
-            return getTypeOfFirstParameterOfSignatureWithFallback(signature, neverType);
+        function isFunctionExpressionOrArrowFunction(node) {
+            return node.kind === 208 /* FunctionExpression */ || node.kind === 209 /* ArrowFunction */;
         }
-        function getTypeOfFirstParameterOfSignatureWithFallback(signature, fallbackType) {
-            return signature.parameters.length > 0 ? getTypeAtPosition(signature, 0) : fallbackType;
+        function getContextualSignatureForFunctionLikeDeclaration(node) {
+            // Only function expressions, arrow functions, and object literal methods are contextually typed.
+            return isFunctionExpressionOrArrowFunction(node) || ts.isObjectLiteralMethod(node)
+                ? getContextualSignature(node)
+                : undefined;
         }
-        function inferFromAnnotatedParameters(signature, context, inferenceContext) {
-            var len = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
-            for (var i = 0; i < len; i++) {
-                var declaration = signature.parameters[i].valueDeclaration;
-                if (declaration.type) {
-                    var typeNode = ts.getEffectiveTypeAnnotationNode(declaration);
-                    if (typeNode) {
-                        inferTypes(inferenceContext.inferences, getTypeFromTypeNode(typeNode), getTypeAtPosition(context, i));
+        // Return the contextual signature for a given expression node. A contextual type provides a
+        // contextual signature if it has a single call signature and if that call signature is non-generic.
+        // If the contextual type is a union type, get the signature from each type possible and if they are
+        // all identical ignoring their return type, the result is same signature but with return type as
+        // union type of return types from these signatures
+        function getContextualSignature(node) {
+            ts.Debug.assert(node.kind !== 165 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
+            var typeTagSignature = getSignatureOfTypeTag(node);
+            if (typeTagSignature) {
+                return typeTagSignature;
+            }
+            var type = getApparentTypeOfContextualType(node, 1 /* Signature */);
+            if (!type) {
+                return undefined;
+            }
+            if (!(type.flags & 1048576 /* Union */)) {
+                return getContextualCallSignature(type, node);
+            }
+            var signatureList;
+            var types = type.types;
+            for (var _i = 0, types_18 = types; _i < types_18.length; _i++) {
+                var current = types_18[_i];
+                var signature = getContextualCallSignature(current, node);
+                if (signature) {
+                    if (!signatureList) {
+                        // This signature will contribute to contextual union signature
+                        signatureList = [signature];
+                    }
+                    else if (!compareSignaturesIdentical(signatureList[0], signature, /*partialMatch*/ false, /*ignoreThisTypes*/ true, /*ignoreReturnTypes*/ true, compareTypesIdentical)) {
+                        // Signatures aren't identical, do not use
+                        return undefined;
+                    }
+                    else {
+                        // Use this signature for contextual union signature
+                        signatureList.push(signature);
                     }
                 }
             }
-            var restType = getEffectiveRestType(context);
-            if (restType && restType.flags & 262144 /* TypeParameter */) {
-                // The contextual signature has a generic rest parameter. We first instantiate the contextual
-                // signature (without fixing type parameters) and assign types to contextually typed parameters.
-                var instantiatedContext = instantiateSignature(context, inferenceContext.nonFixingMapper);
-                assignContextualParameterTypes(signature, instantiatedContext);
-                // We then infer from a tuple type representing the parameters that correspond to the contextual
-                // rest parameter.
-                var restPos = getParameterCount(context) - 1;
-                inferTypes(inferenceContext.inferences, getRestTypeAtPosition(signature, restPos), restType);
+            // Result is union of signatures collected (return type is union of return types of this signature set)
+            if (signatureList) {
+                return signatureList.length === 1 ? signatureList[0] : createUnionSignature(signatureList[0], signatureList);
             }
         }
-        function assignContextualParameterTypes(signature, context) {
-            signature.typeParameters = context.typeParameters;
-            if (context.thisParameter) {
-                var parameter = signature.thisParameter;
-                if (!parameter || parameter.valueDeclaration && !parameter.valueDeclaration.type) {
-                    if (!parameter) {
-                        signature.thisParameter = createSymbolWithType(context.thisParameter, /*type*/ undefined);
+        function checkSpreadExpression(node, checkMode) {
+            if (languageVersion < 2 /* ES2015 */) {
+                checkExternalEmitHelpers(node, compilerOptions.downlevelIteration ? 1536 /* SpreadIncludes */ : 2048 /* SpreadArrays */);
+            }
+            var arrayOrIterableType = checkExpression(node.expression, checkMode);
+            return checkIteratedTypeOrElementType(33 /* Spread */, arrayOrIterableType, undefinedType, node.expression);
+        }
+        function checkSyntheticExpression(node) {
+            return node.isSpread ? getIndexedAccessType(node.type, numberType) : node.type;
+        }
+        function hasDefaultValue(node) {
+            return (node.kind === 198 /* BindingElement */ && !!node.initializer) ||
+                (node.kind === 216 /* BinaryExpression */ && node.operatorToken.kind === 62 /* EqualsToken */);
+        }
+        function checkArrayLiteral(node, checkMode, forceTuple) {
+            var elements = node.elements;
+            var elementCount = elements.length;
+            var elementTypes = [];
+            var elementFlags = [];
+            var contextualType = getApparentTypeOfContextualType(node);
+            var inDestructuringPattern = ts.isAssignmentTarget(node);
+            var inConstContext = isConstContext(node);
+            for (var i = 0; i < elementCount; i++) {
+                var e = elements[i];
+                if (e.kind === 220 /* SpreadElement */) {
+                    if (languageVersion < 2 /* ES2015 */) {
+                        checkExternalEmitHelpers(e, compilerOptions.downlevelIteration ? 1536 /* SpreadIncludes */ : 2048 /* SpreadArrays */);
+                    }
+                    var spreadType = checkExpression(e.expression, checkMode, forceTuple);
+                    if (isArrayLikeType(spreadType)) {
+                        elementTypes.push(spreadType);
+                        elementFlags.push(8 /* Variadic */);
+                    }
+                    else if (inDestructuringPattern) {
+                        // Given the following situation:
+                        //    var c: {};
+                        //    [...c] = ["", 0];
+                        //
+                        // c is represented in the tree as a spread element in an array literal.
+                        // But c really functions as a rest element, and its purpose is to provide
+                        // a contextual type for the right hand side of the assignment. Therefore,
+                        // instead of calling checkExpression on "...c", which will give an error
+                        // if c is not iterable/array-like, we need to act as if we are trying to
+                        // get the contextual element type from it. So we do something similar to
+                        // getContextualTypeForElementExpression, which will crucially not error
+                        // if there is no index type / iterated type.
+                        var restElementType = getIndexTypeOfType(spreadType, 1 /* Number */) ||
+                            getIteratedTypeOrElementType(65 /* Destructuring */, spreadType, undefinedType, /*errorNode*/ undefined, /*checkAssignability*/ false) ||
+                            unknownType;
+                        elementTypes.push(restElementType);
+                        elementFlags.push(4 /* Rest */);
+                    }
+                    else {
+                        elementTypes.push(checkIteratedTypeOrElementType(33 /* Spread */, spreadType, undefinedType, e.expression));
+                        elementFlags.push(4 /* Rest */);
                     }
-                    assignParameterType(signature.thisParameter, getTypeOfSymbol(context.thisParameter));
                 }
-            }
-            var len = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
-            for (var i = 0; i < len; i++) {
-                var parameter = signature.parameters[i];
-                if (!ts.getEffectiveTypeAnnotationNode(parameter.valueDeclaration)) {
-                    var contextualParameterType = tryGetTypeAtPosition(context, i);
-                    assignParameterType(parameter, contextualParameterType);
+                else {
+                    var elementContextualType = getContextualTypeForElementExpression(contextualType, elementTypes.length);
+                    var type = checkExpressionForMutableLocation(e, checkMode, elementContextualType, forceTuple);
+                    elementTypes.push(type);
+                    elementFlags.push(1 /* Required */);
                 }
             }
-            if (signatureHasRestParameter(signature)) {
-                // parameter might be a transient symbol generated by use of `arguments` in the function body.
-                var parameter = ts.last(signature.parameters);
-                if (ts.isTransientSymbol(parameter) || !ts.getEffectiveTypeAnnotationNode(parameter.valueDeclaration)) {
-                    var contextualParameterType = getRestTypeAtPosition(context, len);
-                    assignParameterType(parameter, contextualParameterType);
-                }
+            if (inDestructuringPattern) {
+                return createTupleType(elementTypes, elementFlags);
+            }
+            if (forceTuple || inConstContext || contextualType && forEachType(contextualType, isTupleLikeType)) {
+                return createArrayLiteralType(createTupleType(elementTypes, elementFlags, /*readonly*/ inConstContext));
             }
+            return createArrayLiteralType(createArrayType(elementTypes.length ?
+                getUnionType(ts.sameMap(elementTypes, function (t, i) { return elementFlags[i] & 8 /* Variadic */ ? getIndexedAccessTypeOrUndefined(t, numberType) || anyType : t; }), 2 /* Subtype */) :
+                strictNullChecks ? implicitNeverType : undefinedWideningType, inConstContext));
         }
-        function assignNonContextualParameterTypes(signature) {
-            if (signature.thisParameter) {
-                assignParameterType(signature.thisParameter);
+        function createArrayLiteralType(type) {
+            if (!(ts.getObjectFlags(type) & 4 /* Reference */)) {
+                return type;
             }
-            for (var _i = 0, _a = signature.parameters; _i < _a.length; _i++) {
-                var parameter = _a[_i];
-                assignParameterType(parameter);
+            var literalType = type.literalType;
+            if (!literalType) {
+                literalType = type.literalType = cloneTypeReference(type);
+                literalType.objectFlags |= 65536 /* ArrayLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
             }
+            return literalType;
         }
-        function assignParameterType(parameter, type) {
-            var links = getSymbolLinks(parameter);
-            if (!links.type) {
-                var declaration = parameter.valueDeclaration;
-                links.type = type || getWidenedTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ true);
-                if (declaration.name.kind !== 75 /* Identifier */) {
-                    // if inference didn't come up with anything but unknown, fall back to the binding pattern if present.
-                    if (links.type === unknownType) {
-                        links.type = getTypeFromBindingPattern(declaration.name);
-                    }
-                    assignBindingElementTypes(declaration.name);
-                }
+        function isNumericName(name) {
+            switch (name.kind) {
+                case 158 /* ComputedPropertyName */:
+                    return isNumericComputedName(name);
+                case 78 /* Identifier */:
+                    return isNumericLiteralName(name.escapedText);
+                case 8 /* NumericLiteral */:
+                case 10 /* StringLiteral */:
+                    return isNumericLiteralName(name.text);
+                default:
+                    return false;
             }
         }
-        // When contextual typing assigns a type to a parameter that contains a binding pattern, we also need to push
-        // the destructured type into the contained binding elements.
-        function assignBindingElementTypes(pattern) {
-            for (var _i = 0, _a = pattern.elements; _i < _a.length; _i++) {
-                var element = _a[_i];
-                if (!ts.isOmittedExpression(element)) {
-                    if (element.name.kind === 75 /* Identifier */) {
-                        getSymbolLinks(getSymbolOfNode(element)).type = getTypeForBindingElement(element);
-                    }
-                    else {
-                        assignBindingElementTypes(element.name);
-                    }
+        function isNumericComputedName(name) {
+            // It seems odd to consider an expression of type Any to result in a numeric name,
+            // but this behavior is consistent with checkIndexedAccess
+            return isTypeAssignableToKind(checkComputedPropertyName(name), 296 /* NumberLike */);
+        }
+        function isInfinityOrNaNString(name) {
+            return name === "Infinity" || name === "-Infinity" || name === "NaN";
+        }
+        function isNumericLiteralName(name) {
+            // The intent of numeric names is that
+            //     - they are names with text in a numeric form, and that
+            //     - setting properties/indexing with them is always equivalent to doing so with the numeric literal 'numLit',
+            //         acquired by applying the abstract 'ToNumber' operation on the name's text.
+            //
+            // The subtlety is in the latter portion, as we cannot reliably say that anything that looks like a numeric literal is a numeric name.
+            // In fact, it is the case that the text of the name must be equal to 'ToString(numLit)' for this to hold.
+            //
+            // Consider the property name '"0xF00D"'. When one indexes with '0xF00D', they are actually indexing with the value of 'ToString(0xF00D)'
+            // according to the ECMAScript specification, so it is actually as if the user indexed with the string '"61453"'.
+            // Thus, the text of all numeric literals equivalent to '61543' such as '0xF00D', '0xf00D', '0170015', etc. are not valid numeric names
+            // because their 'ToString' representation is not equal to their original text.
+            // This is motivated by ECMA-262 sections 9.3.1, 9.8.1, 11.1.5, and 11.2.1.
+            //
+            // Here, we test whether 'ToString(ToNumber(name))' is exactly equal to 'name'.
+            // The '+' prefix operator is equivalent here to applying the abstract ToNumber operation.
+            // Applying the 'toString()' method on a number gives us the abstract ToString operation on a number.
+            //
+            // Note that this accepts the values 'Infinity', '-Infinity', and 'NaN', and that this is intentional.
+            // This is desired behavior, because when indexing with them as numeric entities, you are indexing
+            // with the strings '"Infinity"', '"-Infinity"', and '"NaN"' respectively.
+            return (+name).toString() === name;
+        }
+        function checkComputedPropertyName(node) {
+            var links = getNodeLinks(node.expression);
+            if (!links.resolvedType) {
+                links.resolvedType = checkExpression(node.expression);
+                // This will allow types number, string, symbol or any. It will also allow enums, the unknown
+                // type, and any union of these types (like string | number).
+                if (links.resolvedType.flags & 98304 /* Nullable */ ||
+                    !isTypeAssignableToKind(links.resolvedType, 402653316 /* StringLike */ | 296 /* NumberLike */ | 12288 /* ESSymbolLike */) &&
+                        !isTypeAssignableTo(links.resolvedType, stringNumberSymbolType)) {
+                    error(node, ts.Diagnostics.A_computed_property_name_must_be_of_type_string_number_symbol_or_any);
+                }
+                else {
+                    checkThatExpressionIsProperSymbolReference(node.expression, links.resolvedType, /*reportError*/ true);
                 }
             }
+            return links.resolvedType;
         }
-        function createPromiseType(promisedType) {
-            // creates a `Promise<T>` type where `T` is the promisedType argument
-            var globalPromiseType = getGlobalPromiseType(/*reportErrors*/ true);
-            if (globalPromiseType !== emptyGenericType) {
-                // if the promised type is itself a promise, get the underlying type; otherwise, fallback to the promised type
-                promisedType = getAwaitedType(promisedType) || unknownType;
-                return createTypeReference(globalPromiseType, [promisedType]);
-            }
-            return unknownType;
+        function isSymbolWithNumericName(symbol) {
+            var _a;
+            var firstDecl = (_a = symbol.declarations) === null || _a === void 0 ? void 0 : _a[0];
+            return isNumericLiteralName(symbol.escapedName) || (firstDecl && ts.isNamedDeclaration(firstDecl) && isNumericName(firstDecl.name));
         }
-        function createPromiseLikeType(promisedType) {
-            // creates a `PromiseLike<T>` type where `T` is the promisedType argument
-            var globalPromiseLikeType = getGlobalPromiseLikeType(/*reportErrors*/ true);
-            if (globalPromiseLikeType !== emptyGenericType) {
-                // if the promised type is itself a promise, get the underlying type; otherwise, fallback to the promised type
-                promisedType = getAwaitedType(promisedType) || unknownType;
-                return createTypeReference(globalPromiseLikeType, [promisedType]);
+        function getObjectLiteralIndexInfo(node, offset, properties, kind) {
+            var propTypes = [];
+            for (var i = offset; i < properties.length; i++) {
+                if (kind === 0 /* String */ || isSymbolWithNumericName(properties[i])) {
+                    propTypes.push(getTypeOfSymbol(properties[i]));
+                }
             }
-            return unknownType;
+            var unionType = propTypes.length ? getUnionType(propTypes, 2 /* Subtype */) : undefinedType;
+            return createIndexInfo(unionType, isConstContext(node));
         }
-        function createPromiseReturnType(func, promisedType) {
-            var promiseType = createPromiseType(promisedType);
-            if (promiseType === unknownType) {
-                error(func, ts.isImportCall(func) ?
-                    ts.Diagnostics.A_dynamic_import_call_returns_a_Promise_Make_sure_you_have_a_declaration_for_Promise_or_include_ES2015_in_your_lib_option :
-                    ts.Diagnostics.An_async_function_or_method_must_return_a_Promise_Make_sure_you_have_a_declaration_for_Promise_or_include_ES2015_in_your_lib_option);
-                return errorType;
-            }
-            else if (!getGlobalPromiseConstructorSymbol(/*reportErrors*/ true)) {
-                error(func, ts.isImportCall(func) ?
-                    ts.Diagnostics.A_dynamic_import_call_in_ES5_SlashES3_requires_the_Promise_constructor_Make_sure_you_have_a_declaration_for_the_Promise_constructor_or_include_ES2015_in_your_lib_option :
-                    ts.Diagnostics.An_async_function_or_method_in_ES5_SlashES3_requires_the_Promise_constructor_Make_sure_you_have_a_declaration_for_the_Promise_constructor_or_include_ES2015_in_your_lib_option);
+        function getImmediateAliasedSymbol(symbol) {
+            ts.Debug.assert((symbol.flags & 2097152 /* Alias */) !== 0, "Should only get Alias here.");
+            var links = getSymbolLinks(symbol);
+            if (!links.immediateTarget) {
+                var node = getDeclarationOfAliasSymbol(symbol);
+                if (!node)
+                    return ts.Debug.fail();
+                links.immediateTarget = getTargetOfAliasDeclaration(node, /*dontRecursivelyResolve*/ true);
             }
-            return promiseType;
+            return links.immediateTarget;
         }
-        function getReturnTypeFromBody(func, checkMode) {
-            if (!func.body) {
-                return errorType;
+        function checkObjectLiteral(node, checkMode) {
+            var inDestructuringPattern = ts.isAssignmentTarget(node);
+            // Grammar checking
+            checkGrammarObjectLiteralExpression(node, inDestructuringPattern);
+            var allPropertiesTable = strictNullChecks ? ts.createSymbolTable() : undefined;
+            var propertiesTable = ts.createSymbolTable();
+            var propertiesArray = [];
+            var spread = emptyObjectType;
+            var contextualType = getApparentTypeOfContextualType(node);
+            var contextualTypeHasPattern = contextualType && contextualType.pattern &&
+                (contextualType.pattern.kind === 196 /* ObjectBindingPattern */ || contextualType.pattern.kind === 200 /* ObjectLiteralExpression */);
+            var inConstContext = isConstContext(node);
+            var checkFlags = inConstContext ? 8 /* Readonly */ : 0;
+            var isInJavascript = ts.isInJSFile(node) && !ts.isInJsonFile(node);
+            var enumTag = ts.getJSDocEnumTag(node);
+            var isJSObjectLiteral = !contextualType && isInJavascript && !enumTag;
+            var objectFlags = freshObjectLiteralFlag;
+            var patternWithComputedProperties = false;
+            var hasComputedStringProperty = false;
+            var hasComputedNumberProperty = false;
+            // Spreads may cause an early bail; ensure computed names are always checked (this is cached)
+            // As otherwise they may not be checked until exports for the type at this position are retrieved,
+            // which may never occur.
+            for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
+                var elem = _a[_i];
+                if (elem.name && ts.isComputedPropertyName(elem.name) && !ts.isWellKnownSymbolSyntactically(elem.name)) {
+                    checkComputedPropertyName(elem.name);
+                }
             }
-            var functionFlags = ts.getFunctionFlags(func);
-            var isAsync = (functionFlags & 2 /* Async */) !== 0;
-            var isGenerator = (functionFlags & 1 /* Generator */) !== 0;
-            var returnType;
-            var yieldType;
-            var nextType;
-            var fallbackReturnType = voidType;
-            if (func.body.kind !== 223 /* Block */) { // Async or normal arrow function
-                returnType = checkExpressionCached(func.body, checkMode && checkMode & ~8 /* SkipGenericFunctions */);
-                if (isAsync) {
-                    // From within an async function you can return either a non-promise value or a promise. Any
-                    // Promise/A+ compatible implementation will always assimilate any foreign promise, so the
-                    // return type of the body should be unwrapped to its awaited type, which we will wrap in
-                    // the native Promise<T> type later in this function.
-                    returnType = checkAwaitedType(returnType, /*errorNode*/ func, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+            var offset = 0;
+            for (var _b = 0, _c = node.properties; _b < _c.length; _b++) {
+                var memberDecl = _c[_b];
+                var member = getSymbolOfNode(memberDecl);
+                var computedNameType = memberDecl.name && memberDecl.name.kind === 158 /* ComputedPropertyName */ && !ts.isWellKnownSymbolSyntactically(memberDecl.name.expression) ?
+                    checkComputedPropertyName(memberDecl.name) : undefined;
+                if (memberDecl.kind === 288 /* PropertyAssignment */ ||
+                    memberDecl.kind === 289 /* ShorthandPropertyAssignment */ ||
+                    ts.isObjectLiteralMethod(memberDecl)) {
+                    var type = memberDecl.kind === 288 /* PropertyAssignment */ ? checkPropertyAssignment(memberDecl, checkMode) :
+                        // avoid resolving the left side of the ShorthandPropertyAssignment outside of the destructuring
+                        // for error recovery purposes. For example, if a user wrote `{ a = 100 }` instead of `{ a: 100 }`.
+                        // we don't want to say "could not find 'a'".
+                        memberDecl.kind === 289 /* ShorthandPropertyAssignment */ ? checkExpressionForMutableLocation(!inDestructuringPattern && memberDecl.objectAssignmentInitializer ? memberDecl.objectAssignmentInitializer : memberDecl.name, checkMode) :
+                            checkObjectLiteralMethod(memberDecl, checkMode);
+                    if (isInJavascript) {
+                        var jsDocType = getTypeForDeclarationFromJSDocComment(memberDecl);
+                        if (jsDocType) {
+                            checkTypeAssignableTo(type, jsDocType, memberDecl);
+                            type = jsDocType;
+                        }
+                        else if (enumTag && enumTag.typeExpression) {
+                            checkTypeAssignableTo(type, getTypeFromTypeNode(enumTag.typeExpression), memberDecl);
+                        }
+                    }
+                    objectFlags |= ts.getObjectFlags(type) & 3670016 /* PropagatingFlags */;
+                    var nameType = computedNameType && isTypeUsableAsPropertyName(computedNameType) ? computedNameType : undefined;
+                    var prop = nameType ?
+                        createSymbol(4 /* Property */ | member.flags, getPropertyNameFromType(nameType), checkFlags | 4096 /* Late */) :
+                        createSymbol(4 /* Property */ | member.flags, member.escapedName, checkFlags);
+                    if (nameType) {
+                        prop.nameType = nameType;
+                    }
+                    if (inDestructuringPattern) {
+                        // If object literal is an assignment pattern and if the assignment pattern specifies a default value
+                        // for the property, make the property optional.
+                        var isOptional = (memberDecl.kind === 288 /* PropertyAssignment */ && hasDefaultValue(memberDecl.initializer)) ||
+                            (memberDecl.kind === 289 /* ShorthandPropertyAssignment */ && memberDecl.objectAssignmentInitializer);
+                        if (isOptional) {
+                            prop.flags |= 16777216 /* Optional */;
+                        }
+                    }
+                    else if (contextualTypeHasPattern && !(ts.getObjectFlags(contextualType) & 512 /* ObjectLiteralPatternWithComputedProperties */)) {
+                        // If object literal is contextually typed by the implied type of a binding pattern, and if the
+                        // binding pattern specifies a default value for the property, make the property optional.
+                        var impliedProp = getPropertyOfType(contextualType, member.escapedName);
+                        if (impliedProp) {
+                            prop.flags |= impliedProp.flags & 16777216 /* Optional */;
+                        }
+                        else if (!compilerOptions.suppressExcessPropertyErrors && !getIndexInfoOfType(contextualType, 0 /* String */)) {
+                            error(memberDecl.name, ts.Diagnostics.Object_literal_may_only_specify_known_properties_and_0_does_not_exist_in_type_1, symbolToString(member), typeToString(contextualType));
+                        }
+                    }
+                    prop.declarations = member.declarations;
+                    prop.parent = member.parent;
+                    if (member.valueDeclaration) {
+                        prop.valueDeclaration = member.valueDeclaration;
+                    }
+                    prop.type = type;
+                    prop.target = member;
+                    member = prop;
+                    allPropertiesTable === null || allPropertiesTable === void 0 ? void 0 : allPropertiesTable.set(prop.escapedName, prop);
+                }
+                else if (memberDecl.kind === 290 /* SpreadAssignment */) {
+                    if (languageVersion < 2 /* ES2015 */) {
+                        checkExternalEmitHelpers(memberDecl, 2 /* Assign */);
+                    }
+                    if (propertiesArray.length > 0) {
+                        spread = getSpreadType(spread, createObjectLiteralType(), node.symbol, objectFlags, inConstContext);
+                        propertiesArray = [];
+                        propertiesTable = ts.createSymbolTable();
+                        hasComputedStringProperty = false;
+                        hasComputedNumberProperty = false;
+                    }
+                    var type = getReducedType(checkExpression(memberDecl.expression));
+                    if (!isValidSpreadType(type)) {
+                        error(memberDecl, ts.Diagnostics.Spread_types_may_only_be_created_from_object_types);
+                        return errorType;
+                    }
+                    if (allPropertiesTable) {
+                        checkSpreadPropOverrides(type, allPropertiesTable, memberDecl);
+                    }
+                    spread = getSpreadType(spread, type, node.symbol, objectFlags, inConstContext);
+                    offset = propertiesArray.length;
+                    continue;
+                }
+                else {
+                    // TypeScript 1.0 spec (April 2014)
+                    // A get accessor declaration is processed in the same manner as
+                    // an ordinary function declaration(section 6.1) with no parameters.
+                    // A set accessor declaration is processed in the same manner
+                    // as an ordinary function declaration with a single parameter and a Void return type.
+                    ts.Debug.assert(memberDecl.kind === 167 /* GetAccessor */ || memberDecl.kind === 168 /* SetAccessor */);
+                    checkNodeDeferred(memberDecl);
+                }
+                if (computedNameType && !(computedNameType.flags & 8576 /* StringOrNumberLiteralOrUnique */)) {
+                    if (isTypeAssignableTo(computedNameType, stringNumberSymbolType)) {
+                        if (isTypeAssignableTo(computedNameType, numberType)) {
+                            hasComputedNumberProperty = true;
+                        }
+                        else {
+                            hasComputedStringProperty = true;
+                        }
+                        if (inDestructuringPattern) {
+                            patternWithComputedProperties = true;
+                        }
+                    }
+                }
+                else {
+                    propertiesTable.set(member.escapedName, member);
                 }
+                propertiesArray.push(member);
             }
-            else if (isGenerator) { // Generator or AsyncGenerator function
-                var returnTypes = checkAndAggregateReturnExpressionTypes(func, checkMode);
-                if (!returnTypes) {
-                    fallbackReturnType = neverType;
+            // If object literal is contextually typed by the implied type of a binding pattern, augment the result
+            // type with those properties for which the binding pattern specifies a default value.
+            // If the object literal is spread into another object literal, skip this step and let the top-level object
+            // literal handle it instead.
+            if (contextualTypeHasPattern && node.parent.kind !== 290 /* SpreadAssignment */) {
+                for (var _d = 0, _e = getPropertiesOfType(contextualType); _d < _e.length; _d++) {
+                    var prop = _e[_d];
+                    if (!propertiesTable.get(prop.escapedName) && !getPropertyOfType(spread, prop.escapedName)) {
+                        if (!(prop.flags & 16777216 /* Optional */)) {
+                            error(prop.valueDeclaration || prop.bindingElement, ts.Diagnostics.Initializer_provides_no_value_for_this_binding_element_and_the_binding_element_has_no_default_value);
+                        }
+                        propertiesTable.set(prop.escapedName, prop);
+                        propertiesArray.push(prop);
+                    }
                 }
-                else if (returnTypes.length > 0) {
-                    returnType = getUnionType(returnTypes, 2 /* Subtype */);
+            }
+            if (spread !== emptyObjectType) {
+                if (propertiesArray.length > 0) {
+                    spread = getSpreadType(spread, createObjectLiteralType(), node.symbol, objectFlags, inConstContext);
+                    propertiesArray = [];
+                    propertiesTable = ts.createSymbolTable();
+                    hasComputedStringProperty = false;
+                    hasComputedNumberProperty = false;
                 }
-                var _a = checkAndAggregateYieldOperandTypes(func, checkMode), yieldTypes = _a.yieldTypes, nextTypes = _a.nextTypes;
-                yieldType = ts.some(yieldTypes) ? getUnionType(yieldTypes, 2 /* Subtype */) : undefined;
-                nextType = ts.some(nextTypes) ? getIntersectionType(nextTypes) : undefined;
+                // remap the raw emptyObjectType fed in at the top into a fresh empty object literal type, unique to this use site
+                return mapType(spread, function (t) { return t === emptyObjectType ? createObjectLiteralType() : t; });
             }
-            else { // Async or normal function
-                var types = checkAndAggregateReturnExpressionTypes(func, checkMode);
-                if (!types) {
-                    // For an async function, the return type will not be never, but rather a Promise for never.
-                    return functionFlags & 2 /* Async */
-                        ? createPromiseReturnType(func, neverType) // Async function
-                        : neverType; // Normal function
+            return createObjectLiteralType();
+            function createObjectLiteralType() {
+                var stringIndexInfo = hasComputedStringProperty ? getObjectLiteralIndexInfo(node, offset, propertiesArray, 0 /* String */) : undefined;
+                var numberIndexInfo = hasComputedNumberProperty ? getObjectLiteralIndexInfo(node, offset, propertiesArray, 1 /* Number */) : undefined;
+                var result = createAnonymousType(node.symbol, propertiesTable, ts.emptyArray, ts.emptyArray, stringIndexInfo, numberIndexInfo);
+                result.objectFlags |= objectFlags | 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
+                if (isJSObjectLiteral) {
+                    result.objectFlags |= 16384 /* JSLiteral */;
                 }
-                if (types.length === 0) {
-                    // For an async function, the return type will not be void, but rather a Promise for void.
-                    return functionFlags & 2 /* Async */
-                        ? createPromiseReturnType(func, voidType) // Async function
-                        : voidType; // Normal function
+                if (patternWithComputedProperties) {
+                    result.objectFlags |= 512 /* ObjectLiteralPatternWithComputedProperties */;
                 }
-                // Return a union of the return expression types.
-                returnType = getUnionType(types, 2 /* Subtype */);
+                if (inDestructuringPattern) {
+                    result.pattern = node;
+                }
+                return result;
             }
-            if (returnType || yieldType || nextType) {
-                if (yieldType)
-                    reportErrorsFromWidening(func, yieldType, 3 /* GeneratorYield */);
-                if (returnType)
-                    reportErrorsFromWidening(func, returnType, 1 /* FunctionReturn */);
-                if (nextType)
-                    reportErrorsFromWidening(func, nextType, 2 /* GeneratorNext */);
-                if (returnType && isUnitType(returnType) ||
-                    yieldType && isUnitType(yieldType) ||
-                    nextType && isUnitType(nextType)) {
-                    var contextualSignature = getContextualSignatureForFunctionLikeDeclaration(func);
-                    var contextualType = !contextualSignature ? undefined :
-                        contextualSignature === getSignatureFromDeclaration(func) ? isGenerator ? undefined : returnType :
-                            instantiateContextualType(getReturnTypeOfSignature(contextualSignature), func);
-                    if (isGenerator) {
-                        yieldType = getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(yieldType, contextualType, 0 /* Yield */, isAsync);
-                        returnType = getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(returnType, contextualType, 1 /* Return */, isAsync);
-                        nextType = getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(nextType, contextualType, 2 /* Next */, isAsync);
-                    }
-                    else {
-                        returnType = getWidenedLiteralLikeTypeForContextualReturnTypeIfNeeded(returnType, contextualType, isAsync);
-                    }
+        }
+        function isValidSpreadType(type) {
+            if (type.flags & 465829888 /* Instantiable */) {
+                var constraint = getBaseConstraintOfType(type);
+                if (constraint !== undefined) {
+                    return isValidSpreadType(constraint);
                 }
-                if (yieldType)
-                    yieldType = getWidenedType(yieldType);
-                if (returnType)
-                    returnType = getWidenedType(returnType);
-                if (nextType)
-                    nextType = getWidenedType(nextType);
             }
-            if (isGenerator) {
-                return createGeneratorReturnType(yieldType || neverType, returnType || fallbackReturnType, nextType || getContextualIterationType(2 /* Next */, func) || unknownType, isAsync);
+            return !!(type.flags & (1 /* Any */ | 67108864 /* NonPrimitive */ | 524288 /* Object */ | 58982400 /* InstantiableNonPrimitive */) ||
+                getFalsyFlags(type) & 117632 /* DefinitelyFalsy */ && isValidSpreadType(removeDefinitelyFalsyTypes(type)) ||
+                type.flags & 3145728 /* UnionOrIntersection */ && ts.every(type.types, isValidSpreadType));
+        }
+        function checkJsxSelfClosingElementDeferred(node) {
+            checkJsxOpeningLikeElementOrOpeningFragment(node);
+        }
+        function checkJsxSelfClosingElement(node, _checkMode) {
+            checkNodeDeferred(node);
+            return getJsxElementTypeAt(node) || anyType;
+        }
+        function checkJsxElementDeferred(node) {
+            // Check attributes
+            checkJsxOpeningLikeElementOrOpeningFragment(node.openingElement);
+            // Perform resolution on the closing tag so that rename/go to definition/etc work
+            if (isJsxIntrinsicIdentifier(node.closingElement.tagName)) {
+                getIntrinsicTagSymbol(node.closingElement);
             }
             else {
-                // From within an async function you can return either a non-promise value or a promise. Any
-                // Promise/A+ compatible implementation will always assimilate any foreign promise, so the
-                // return type of the body is awaited type of the body, wrapped in a native Promise<T> type.
-                return isAsync
-                    ? createPromiseType(returnType || fallbackReturnType)
-                    : returnType || fallbackReturnType;
+                checkExpression(node.closingElement.tagName);
             }
+            checkJsxChildren(node);
         }
-        function createGeneratorReturnType(yieldType, returnType, nextType, isAsyncGenerator) {
-            var resolver = isAsyncGenerator ? asyncIterationTypesResolver : syncIterationTypesResolver;
-            var globalGeneratorType = resolver.getGlobalGeneratorType(/*reportErrors*/ false);
-            yieldType = resolver.resolveIterationType(yieldType, /*errorNode*/ undefined) || unknownType;
-            returnType = resolver.resolveIterationType(returnType, /*errorNode*/ undefined) || unknownType;
-            nextType = resolver.resolveIterationType(nextType, /*errorNode*/ undefined) || unknownType;
-            if (globalGeneratorType === emptyGenericType) {
-                // Fall back to the global IterableIterator if returnType is assignable to the expected return iteration
-                // type of IterableIterator, and the expected next iteration type of IterableIterator is assignable to
-                // nextType.
-                var globalType = resolver.getGlobalIterableIteratorType(/*reportErrors*/ false);
-                var iterationTypes = globalType !== emptyGenericType ? getIterationTypesOfGlobalIterableType(globalType, resolver) : undefined;
-                var iterableIteratorReturnType = iterationTypes ? iterationTypes.returnType : anyType;
-                var iterableIteratorNextType = iterationTypes ? iterationTypes.nextType : undefinedType;
-                if (isTypeAssignableTo(returnType, iterableIteratorReturnType) &&
-                    isTypeAssignableTo(iterableIteratorNextType, nextType)) {
-                    if (globalType !== emptyGenericType) {
-                        return createTypeFromGenericGlobalType(globalType, [yieldType]);
-                    }
-                    // The global IterableIterator type doesn't exist, so report an error
-                    resolver.getGlobalIterableIteratorType(/*reportErrors*/ true);
-                    return emptyObjectType;
-                }
-                // The global Generator type doesn't exist, so report an error
-                resolver.getGlobalGeneratorType(/*reportErrors*/ true);
-                return emptyObjectType;
+        function checkJsxElement(node, _checkMode) {
+            checkNodeDeferred(node);
+            return getJsxElementTypeAt(node) || anyType;
+        }
+        function checkJsxFragment(node) {
+            checkJsxOpeningLikeElementOrOpeningFragment(node.openingFragment);
+            // by default, jsx:'react' will use jsxFactory = React.createElement and jsxFragmentFactory = React.Fragment
+            // if jsxFactory compiler option is provided, ensure jsxFragmentFactory compiler option or @jsxFrag pragma is provided too
+            var nodeSourceFile = ts.getSourceFileOfNode(node);
+            if (ts.getJSXTransformEnabled(compilerOptions) && (compilerOptions.jsxFactory || nodeSourceFile.pragmas.has("jsx"))
+                && !compilerOptions.jsxFragmentFactory && !nodeSourceFile.pragmas.has("jsxfrag")) {
+                error(node, compilerOptions.jsxFactory
+                    ? ts.Diagnostics.The_jsxFragmentFactory_compiler_option_must_be_provided_to_use_JSX_fragments_with_the_jsxFactory_compiler_option
+                    : ts.Diagnostics.An_jsxFrag_pragma_is_required_when_using_an_jsx_pragma_with_JSX_fragments);
             }
-            return createTypeFromGenericGlobalType(globalGeneratorType, [yieldType, returnType, nextType]);
+            checkJsxChildren(node);
+            return getJsxElementTypeAt(node) || anyType;
         }
-        function checkAndAggregateYieldOperandTypes(func, checkMode) {
-            var yieldTypes = [];
-            var nextTypes = [];
-            var isAsync = (ts.getFunctionFlags(func) & 2 /* Async */) !== 0;
-            ts.forEachYieldExpression(func.body, function (yieldExpression) {
-                var yieldExpressionType = yieldExpression.expression ? checkExpression(yieldExpression.expression, checkMode) : undefinedWideningType;
-                ts.pushIfUnique(yieldTypes, getYieldedTypeOfYieldExpression(yieldExpression, yieldExpressionType, anyType, isAsync));
-                var nextType;
-                if (yieldExpression.asteriskToken) {
-                    var iterationTypes = getIterationTypesOfIterable(yieldExpressionType, isAsync ? 19 /* AsyncYieldStar */ : 17 /* YieldStar */, yieldExpression.expression);
-                    nextType = iterationTypes && iterationTypes.nextType;
-                }
-                else {
-                    nextType = getContextualType(yieldExpression);
-                }
-                if (nextType)
-                    ts.pushIfUnique(nextTypes, nextType);
-            });
-            return { yieldTypes: yieldTypes, nextTypes: nextTypes };
+        /**
+         * Returns true iff the JSX element name would be a valid JS identifier, ignoring restrictions about keywords not being identifiers
+         */
+        function isUnhyphenatedJsxName(name) {
+            // - is the only character supported in JSX attribute names that isn't valid in JavaScript identifiers
+            return !ts.stringContains(name, "-");
         }
-        function getYieldedTypeOfYieldExpression(node, expressionType, sentType, isAsync) {
-            var errorNode = node.expression || node;
-            // A `yield*` expression effectively yields everything that its operand yields
-            var yieldedType = node.asteriskToken ? checkIteratedTypeOrElementType(isAsync ? 19 /* AsyncYieldStar */ : 17 /* YieldStar */, expressionType, sentType, errorNode) : expressionType;
-            return !isAsync ? yieldedType : getAwaitedType(yieldedType, errorNode, node.asteriskToken
-                ? ts.Diagnostics.Type_of_iterated_elements_of_a_yield_Asterisk_operand_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member
-                : ts.Diagnostics.Type_of_yield_operand_in_an_async_generator_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+        /**
+         * Returns true iff React would emit this tag name as a string rather than an identifier or qualified name
+         */
+        function isJsxIntrinsicIdentifier(tagName) {
+            return tagName.kind === 78 /* Identifier */ && ts.isIntrinsicJsxName(tagName.escapedText);
+        }
+        function checkJsxAttribute(node, checkMode) {
+            return node.initializer
+                ? checkExpressionForMutableLocation(node.initializer, checkMode)
+                : trueType; // <Elem attr /> is sugar for <Elem attr={true} />
         }
         /**
-         * Collect the TypeFacts learned from a typeof switch with
-         * total clauses `witnesses`, and the active clause ranging
-         * from `start` to `end`. Parameter `hasDefault` denotes
-         * whether the active clause contains a default clause.
+         * Get attributes type of the JSX opening-like element. The result is from resolving "attributes" property of the opening-like element.
+         *
+         * @param openingLikeElement a JSX opening-like element
+         * @param filter a function to remove attributes that will not participate in checking whether attributes are assignable
+         * @return an anonymous type (similar to the one returned by checkObjectLiteral) in which its properties are attributes property.
+         * @remarks Because this function calls getSpreadType, it needs to use the same checks as checkObjectLiteral,
+         * which also calls getSpreadType.
          */
-        function getFactsFromTypeofSwitch(start, end, witnesses, hasDefault) {
-            var facts = 0 /* None */;
-            // When in the default we only collect inequality facts
-            // because default is 'in theory' a set of infinite
-            // equalities.
-            if (hasDefault) {
-                // Value is not equal to any types after the active clause.
-                for (var i = end; i < witnesses.length; i++) {
-                    facts |= typeofNEFacts.get(witnesses[i]) || 32768 /* TypeofNEHostObject */;
-                }
-                // Remove inequalities for types that appear in the
-                // active clause because they appear before other
-                // types collected so far.
-                for (var i = start; i < end; i++) {
-                    facts &= ~(typeofNEFacts.get(witnesses[i]) || 0);
+        function createJsxAttributesTypeFromAttributesProperty(openingLikeElement, checkMode) {
+            var attributes = openingLikeElement.attributes;
+            var allAttributesTable = strictNullChecks ? ts.createSymbolTable() : undefined;
+            var attributesTable = ts.createSymbolTable();
+            var spread = emptyJsxObjectType;
+            var hasSpreadAnyType = false;
+            var typeToIntersect;
+            var explicitlySpecifyChildrenAttribute = false;
+            var objectFlags = 4096 /* JsxAttributes */;
+            var jsxChildrenPropertyName = getJsxElementChildrenPropertyName(getJsxNamespaceAt(openingLikeElement));
+            for (var _i = 0, _a = attributes.properties; _i < _a.length; _i++) {
+                var attributeDecl = _a[_i];
+                var member = attributeDecl.symbol;
+                if (ts.isJsxAttribute(attributeDecl)) {
+                    var exprType = checkJsxAttribute(attributeDecl, checkMode);
+                    objectFlags |= ts.getObjectFlags(exprType) & 3670016 /* PropagatingFlags */;
+                    var attributeSymbol = createSymbol(4 /* Property */ | member.flags, member.escapedName);
+                    attributeSymbol.declarations = member.declarations;
+                    attributeSymbol.parent = member.parent;
+                    if (member.valueDeclaration) {
+                        attributeSymbol.valueDeclaration = member.valueDeclaration;
+                    }
+                    attributeSymbol.type = exprType;
+                    attributeSymbol.target = member;
+                    attributesTable.set(attributeSymbol.escapedName, attributeSymbol);
+                    allAttributesTable === null || allAttributesTable === void 0 ? void 0 : allAttributesTable.set(attributeSymbol.escapedName, attributeSymbol);
+                    if (attributeDecl.name.escapedText === jsxChildrenPropertyName) {
+                        explicitlySpecifyChildrenAttribute = true;
+                    }
                 }
-                // Add inequalities for types before the active clause unconditionally.
-                for (var i = 0; i < start; i++) {
-                    facts |= typeofNEFacts.get(witnesses[i]) || 32768 /* TypeofNEHostObject */;
+                else {
+                    ts.Debug.assert(attributeDecl.kind === 282 /* JsxSpreadAttribute */);
+                    if (attributesTable.size > 0) {
+                        spread = getSpreadType(spread, createJsxAttributesType(), attributes.symbol, objectFlags, /*readonly*/ false);
+                        attributesTable = ts.createSymbolTable();
+                    }
+                    var exprType = getReducedType(checkExpressionCached(attributeDecl.expression, checkMode));
+                    if (isTypeAny(exprType)) {
+                        hasSpreadAnyType = true;
+                    }
+                    if (isValidSpreadType(exprType)) {
+                        spread = getSpreadType(spread, exprType, attributes.symbol, objectFlags, /*readonly*/ false);
+                        if (allAttributesTable) {
+                            checkSpreadPropOverrides(exprType, allAttributesTable, attributeDecl);
+                        }
+                    }
+                    else {
+                        typeToIntersect = typeToIntersect ? getIntersectionType([typeToIntersect, exprType]) : exprType;
+                    }
                 }
             }
-            // When in an active clause without default the set of
-            // equalities is finite.
-            else {
-                // Add equalities for all types in the active clause.
-                for (var i = start; i < end; i++) {
-                    facts |= typeofEQFacts.get(witnesses[i]) || 128 /* TypeofEQHostObject */;
+            if (!hasSpreadAnyType) {
+                if (attributesTable.size > 0) {
+                    spread = getSpreadType(spread, createJsxAttributesType(), attributes.symbol, objectFlags, /*readonly*/ false);
                 }
-                // Remove equalities for types that appear before the
-                // active clause.
-                for (var i = 0; i < start; i++) {
-                    facts &= ~(typeofEQFacts.get(witnesses[i]) || 0);
+            }
+            // Handle children attribute
+            var parent = openingLikeElement.parent.kind === 273 /* JsxElement */ ? openingLikeElement.parent : undefined;
+            // We have to check that openingElement of the parent is the one we are visiting as this may not be true for selfClosingElement
+            if (parent && parent.openingElement === openingLikeElement && parent.children.length > 0) {
+                var childrenTypes = checkJsxChildren(parent, checkMode);
+                if (!hasSpreadAnyType && jsxChildrenPropertyName && jsxChildrenPropertyName !== "") {
+                    // Error if there is a attribute named "children" explicitly specified and children element.
+                    // This is because children element will overwrite the value from attributes.
+                    // Note: we will not warn "children" attribute overwritten if "children" attribute is specified in object spread.
+                    if (explicitlySpecifyChildrenAttribute) {
+                        error(attributes, ts.Diagnostics._0_are_specified_twice_The_attribute_named_0_will_be_overwritten, ts.unescapeLeadingUnderscores(jsxChildrenPropertyName));
+                    }
+                    var contextualType = getApparentTypeOfContextualType(openingLikeElement.attributes);
+                    var childrenContextualType = contextualType && getTypeOfPropertyOfContextualType(contextualType, jsxChildrenPropertyName);
+                    // If there are children in the body of JSX element, create dummy attribute "children" with the union of children types so that it will pass the attribute checking process
+                    var childrenPropSymbol = createSymbol(4 /* Property */, jsxChildrenPropertyName);
+                    childrenPropSymbol.type = childrenTypes.length === 1 ? childrenTypes[0] :
+                        childrenContextualType && forEachType(childrenContextualType, isTupleLikeType) ? createTupleType(childrenTypes) :
+                            createArrayType(getUnionType(childrenTypes));
+                    // Fake up a property declaration for the children
+                    childrenPropSymbol.valueDeclaration = ts.factory.createPropertySignature(/*modifiers*/ undefined, ts.unescapeLeadingUnderscores(jsxChildrenPropertyName), /*questionToken*/ undefined, /*type*/ undefined);
+                    ts.setParent(childrenPropSymbol.valueDeclaration, attributes);
+                    childrenPropSymbol.valueDeclaration.symbol = childrenPropSymbol;
+                    var childPropMap = ts.createSymbolTable();
+                    childPropMap.set(jsxChildrenPropertyName, childrenPropSymbol);
+                    spread = getSpreadType(spread, createAnonymousType(attributes.symbol, childPropMap, ts.emptyArray, ts.emptyArray, /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined), attributes.symbol, objectFlags, /*readonly*/ false);
                 }
             }
-            return facts;
-        }
-        function isExhaustiveSwitchStatement(node) {
-            var links = getNodeLinks(node);
-            return links.isExhaustive !== undefined ? links.isExhaustive : (links.isExhaustive = computeExhaustiveSwitchStatement(node));
-        }
-        function computeExhaustiveSwitchStatement(node) {
-            if (node.expression.kind === 204 /* TypeOfExpression */) {
-                var operandType = getTypeOfExpression(node.expression.expression);
-                var witnesses = getSwitchClauseTypeOfWitnesses(node, /*retainDefault*/ false);
-                // notEqualFacts states that the type of the switched value is not equal to every type in the switch.
-                var notEqualFacts_1 = getFactsFromTypeofSwitch(0, 0, witnesses, /*hasDefault*/ true);
-                var type_3 = getBaseConstraintOfType(operandType) || operandType;
-                return !!(filterType(type_3, function (t) { return (getTypeFacts(t) & notEqualFacts_1) === notEqualFacts_1; }).flags & 131072 /* Never */);
+            if (hasSpreadAnyType) {
+                return anyType;
             }
-            var type = getTypeOfExpression(node.expression);
-            if (!isLiteralType(type)) {
-                return false;
+            if (typeToIntersect && spread !== emptyJsxObjectType) {
+                return getIntersectionType([typeToIntersect, spread]);
             }
-            var switchTypes = getSwitchClauseTypes(node);
-            if (!switchTypes.length || ts.some(switchTypes, isNeitherUnitTypeNorNever)) {
-                return false;
+            return typeToIntersect || (spread === emptyJsxObjectType ? createJsxAttributesType() : spread);
+            /**
+             * Create anonymous type from given attributes symbol table.
+             * @param symbol a symbol of JsxAttributes containing attributes corresponding to attributesTable
+             * @param attributesTable a symbol table of attributes property
+             */
+            function createJsxAttributesType() {
+                objectFlags |= freshObjectLiteralFlag;
+                var result = createAnonymousType(attributes.symbol, attributesTable, ts.emptyArray, ts.emptyArray, /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined);
+                result.objectFlags |= objectFlags | 128 /* ObjectLiteral */ | 1048576 /* ContainsObjectOrArrayLiteral */;
+                return result;
+            }
+        }
+        function checkJsxChildren(node, checkMode) {
+            var childrenTypes = [];
+            for (var _i = 0, _a = node.children; _i < _a.length; _i++) {
+                var child = _a[_i];
+                // In React, JSX text that contains only whitespaces will be ignored so we don't want to type-check that
+                // because then type of children property will have constituent of string type.
+                if (child.kind === 11 /* JsxText */) {
+                    if (!child.containsOnlyTriviaWhiteSpaces) {
+                        childrenTypes.push(stringType);
+                    }
+                }
+                else if (child.kind === 283 /* JsxExpression */ && !child.expression) {
+                    continue; // empty jsx expressions don't *really* count as present children
+                }
+                else {
+                    childrenTypes.push(checkExpressionForMutableLocation(child, checkMode));
+                }
+            }
+            return childrenTypes;
+        }
+        function checkSpreadPropOverrides(type, props, spread) {
+            for (var _i = 0, _a = getPropertiesOfType(type); _i < _a.length; _i++) {
+                var right = _a[_i];
+                var left = props.get(right.escapedName);
+                var rightType = getTypeOfSymbol(right);
+                if (left && !maybeTypeOfKind(rightType, 98304 /* Nullable */) && !(maybeTypeOfKind(rightType, 3 /* AnyOrUnknown */) && right.flags & 16777216 /* Optional */)) {
+                    var diagnostic = error(left.valueDeclaration, ts.Diagnostics._0_is_specified_more_than_once_so_this_usage_will_be_overwritten, ts.unescapeLeadingUnderscores(left.escapedName));
+                    ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(spread, ts.Diagnostics.This_spread_always_overwrites_this_property));
+                }
             }
-            return eachTypeContainedIn(mapType(type, getRegularTypeOfLiteralType), switchTypes);
         }
-        function functionHasImplicitReturn(func) {
-            return func.endFlowNode && isReachableFlowNode(func.endFlowNode);
+        /**
+         * Check attributes property of opening-like element. This function is called during chooseOverload to get call signature of a JSX opening-like element.
+         * (See "checkApplicableSignatureForJsxOpeningLikeElement" for how the function is used)
+         * @param node a JSXAttributes to be resolved of its type
+         */
+        function checkJsxAttributes(node, checkMode) {
+            return createJsxAttributesTypeFromAttributesProperty(node.parent, checkMode);
         }
-        /** NOTE: Return value of `[]` means a different thing than `undefined`. `[]` means func returns `void`, `undefined` means it returns `never`. */
-        function checkAndAggregateReturnExpressionTypes(func, checkMode) {
-            var functionFlags = ts.getFunctionFlags(func);
-            var aggregatedTypes = [];
-            var hasReturnWithNoExpression = functionHasImplicitReturn(func);
-            var hasReturnOfTypeNever = false;
-            ts.forEachReturnStatement(func.body, function (returnStatement) {
-                var expr = returnStatement.expression;
-                if (expr) {
-                    var type = checkExpressionCached(expr, checkMode && checkMode & ~8 /* SkipGenericFunctions */);
-                    if (functionFlags & 2 /* Async */) {
-                        // From within an async function you can return either a non-promise value or a promise. Any
-                        // Promise/A+ compatible implementation will always assimilate any foreign promise, so the
-                        // return type of the body should be unwrapped to its awaited type, which should be wrapped in
-                        // the native Promise<T> type by the caller.
-                        type = checkAwaitedType(type, func, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+        function getJsxType(name, location) {
+            var namespace = getJsxNamespaceAt(location);
+            var exports = namespace && getExportsOfSymbol(namespace);
+            var typeSymbol = exports && getSymbol(exports, name, 788968 /* Type */);
+            return typeSymbol ? getDeclaredTypeOfSymbol(typeSymbol) : errorType;
+        }
+        /**
+         * Looks up an intrinsic tag name and returns a symbol that either points to an intrinsic
+         * property (in which case nodeLinks.jsxFlags will be IntrinsicNamedElement) or an intrinsic
+         * string index signature (in which case nodeLinks.jsxFlags will be IntrinsicIndexedElement).
+         * May also return unknownSymbol if both of these lookups fail.
+         */
+        function getIntrinsicTagSymbol(node) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedSymbol) {
+                var intrinsicElementsType = getJsxType(JsxNames.IntrinsicElements, node);
+                if (intrinsicElementsType !== errorType) {
+                    // Property case
+                    if (!ts.isIdentifier(node.tagName))
+                        return ts.Debug.fail();
+                    var intrinsicProp = getPropertyOfType(intrinsicElementsType, node.tagName.escapedText);
+                    if (intrinsicProp) {
+                        links.jsxFlags |= 1 /* IntrinsicNamedElement */;
+                        return links.resolvedSymbol = intrinsicProp;
                     }
-                    if (type.flags & 131072 /* Never */) {
-                        hasReturnOfTypeNever = true;
+                    // Intrinsic string indexer case
+                    var indexSignatureType = getIndexTypeOfType(intrinsicElementsType, 0 /* String */);
+                    if (indexSignatureType) {
+                        links.jsxFlags |= 2 /* IntrinsicIndexedElement */;
+                        return links.resolvedSymbol = intrinsicElementsType.symbol;
                     }
-                    ts.pushIfUnique(aggregatedTypes, type);
+                    // Wasn't found
+                    error(node, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.idText(node.tagName), "JSX." + JsxNames.IntrinsicElements);
+                    return links.resolvedSymbol = unknownSymbol;
                 }
                 else {
-                    hasReturnWithNoExpression = true;
+                    if (noImplicitAny) {
+                        error(node, ts.Diagnostics.JSX_element_implicitly_has_type_any_because_no_interface_JSX_0_exists, ts.unescapeLeadingUnderscores(JsxNames.IntrinsicElements));
+                    }
+                    return links.resolvedSymbol = unknownSymbol;
                 }
-            });
-            if (aggregatedTypes.length === 0 && !hasReturnWithNoExpression && (hasReturnOfTypeNever || mayReturnNever(func))) {
+            }
+            return links.resolvedSymbol;
+        }
+        function getJsxNamespaceContainerForImplicitImport(location) {
+            var file = location && ts.getSourceFileOfNode(location);
+            var links = file && getNodeLinks(file);
+            if (links && links.jsxImplicitImportContainer === false) {
                 return undefined;
             }
-            if (strictNullChecks && aggregatedTypes.length && hasReturnWithNoExpression &&
-                !(isJSConstructor(func) && aggregatedTypes.some(function (t) { return t.symbol === func.symbol; }))) {
-                // Javascript "callable constructors", containing eg `if (!(this instanceof A)) return new A()` should not add undefined
-                ts.pushIfUnique(aggregatedTypes, undefinedType);
+            if (links && links.jsxImplicitImportContainer) {
+                return links.jsxImplicitImportContainer;
             }
-            return aggregatedTypes;
+            var runtimeImportSpecifier = ts.getJSXRuntimeImport(ts.getJSXImplicitImportBase(compilerOptions, file), compilerOptions);
+            if (!runtimeImportSpecifier) {
+                return undefined;
+            }
+            var isClassic = ts.getEmitModuleResolutionKind(compilerOptions) === ts.ModuleResolutionKind.Classic;
+            var errorMessage = isClassic
+                ? ts.Diagnostics.Cannot_find_module_0_Did_you_mean_to_set_the_moduleResolution_option_to_node_or_to_add_aliases_to_the_paths_option
+                : ts.Diagnostics.Cannot_find_module_0_or_its_corresponding_type_declarations;
+            var mod = resolveExternalModule(location, runtimeImportSpecifier, errorMessage, location);
+            var result = mod && mod !== unknownSymbol ? getMergedSymbol(resolveSymbol(mod)) : undefined;
+            if (links) {
+                links.jsxImplicitImportContainer = result || false;
+            }
+            return result;
         }
-        function mayReturnNever(func) {
-            switch (func.kind) {
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                    return true;
-                case 161 /* MethodDeclaration */:
-                    return func.parent.kind === 193 /* ObjectLiteralExpression */;
-                default:
-                    return false;
+        function getJsxNamespaceAt(location) {
+            var links = location && getNodeLinks(location);
+            if (links && links.jsxNamespace) {
+                return links.jsxNamespace;
+            }
+            if (!links || links.jsxNamespace !== false) {
+                var resolvedNamespace = getJsxNamespaceContainerForImplicitImport(location);
+                if (!resolvedNamespace || resolvedNamespace === unknownSymbol) {
+                    var namespaceName = getJsxNamespace(location);
+                    resolvedNamespace = resolveName(location, namespaceName, 1920 /* Namespace */, /*diagnosticMessage*/ undefined, namespaceName, /*isUse*/ false);
+                }
+                if (resolvedNamespace) {
+                    var candidate = resolveSymbol(getSymbol(getExportsOfSymbol(resolveSymbol(resolvedNamespace)), JsxNames.JSX, 1920 /* Namespace */));
+                    if (candidate && candidate !== unknownSymbol) {
+                        if (links) {
+                            links.jsxNamespace = candidate;
+                        }
+                        return candidate;
+                    }
+                }
+                if (links) {
+                    links.jsxNamespace = false;
+                }
+            }
+            // JSX global fallback
+            var s = resolveSymbol(getGlobalSymbol(JsxNames.JSX, 1920 /* Namespace */, /*diagnosticMessage*/ undefined));
+            if (s === unknownSymbol) {
+                return undefined; // TODO: GH#18217
             }
+            return s; // TODO: GH#18217
         }
         /**
-         * TypeScript Specification 1.0 (6.3) - July 2014
-         *   An explicitly typed function whose return type isn't the Void type,
-         *   the Any type, or a union type containing the Void or Any type as a constituent
-         *   must have at least one return statement somewhere in its body.
-         *   An exception to this rule is if the function implementation consists of a single 'throw' statement.
+         * Look into JSX namespace and then look for container with matching name as nameOfAttribPropContainer.
+         * Get a single property from that container if existed. Report an error if there are more than one property.
          *
-         * @param returnType - return type of the function, can be undefined if return type is not explicitly specified
+         * @param nameOfAttribPropContainer a string of value JsxNames.ElementAttributesPropertyNameContainer or JsxNames.ElementChildrenAttributeNameContainer
+         *          if other string is given or the container doesn't exist, return undefined.
          */
-        function checkAllCodePathsInNonVoidFunctionReturnOrThrow(func, returnType) {
-            if (!produceDiagnostics) {
-                return;
-            }
-            var functionFlags = ts.getFunctionFlags(func);
-            var type = returnType && unwrapReturnType(returnType, functionFlags);
-            // Functions with with an explicitly specified 'void' or 'any' return type don't need any return expressions.
-            if (type && maybeTypeOfKind(type, 1 /* Any */ | 16384 /* Void */)) {
-                return;
+        function getNameFromJsxElementAttributesContainer(nameOfAttribPropContainer, jsxNamespace) {
+            // JSX.ElementAttributesProperty | JSX.ElementChildrenAttribute [symbol]
+            var jsxElementAttribPropInterfaceSym = jsxNamespace && getSymbol(jsxNamespace.exports, nameOfAttribPropContainer, 788968 /* Type */);
+            // JSX.ElementAttributesProperty | JSX.ElementChildrenAttribute [type]
+            var jsxElementAttribPropInterfaceType = jsxElementAttribPropInterfaceSym && getDeclaredTypeOfSymbol(jsxElementAttribPropInterfaceSym);
+            // The properties of JSX.ElementAttributesProperty | JSX.ElementChildrenAttribute
+            var propertiesOfJsxElementAttribPropInterface = jsxElementAttribPropInterfaceType && getPropertiesOfType(jsxElementAttribPropInterfaceType);
+            if (propertiesOfJsxElementAttribPropInterface) {
+                // Element Attributes has zero properties, so the element attributes type will be the class instance type
+                if (propertiesOfJsxElementAttribPropInterface.length === 0) {
+                    return "";
+                }
+                // Element Attributes has one property, so the element attributes type will be the type of the corresponding
+                // property of the class instance type
+                else if (propertiesOfJsxElementAttribPropInterface.length === 1) {
+                    return propertiesOfJsxElementAttribPropInterface[0].escapedName;
+                }
+                else if (propertiesOfJsxElementAttribPropInterface.length > 1) {
+                    // More than one property on ElementAttributesProperty is an error
+                    error(jsxElementAttribPropInterfaceSym.declarations[0], ts.Diagnostics.The_global_type_JSX_0_may_not_have_more_than_one_property, ts.unescapeLeadingUnderscores(nameOfAttribPropContainer));
+                }
             }
-            // If all we have is a function signature, or an arrow function with an expression body, then there is nothing to check.
-            // also if HasImplicitReturn flag is not set this means that all codepaths in function body end with return or throw
-            if (func.kind === 160 /* MethodSignature */ || ts.nodeIsMissing(func.body) || func.body.kind !== 223 /* Block */ || !functionHasImplicitReturn(func)) {
-                return;
+            return undefined;
+        }
+        function getJsxLibraryManagedAttributes(jsxNamespace) {
+            // JSX.LibraryManagedAttributes [symbol]
+            return jsxNamespace && getSymbol(jsxNamespace.exports, JsxNames.LibraryManagedAttributes, 788968 /* Type */);
+        }
+        /// e.g. "props" for React.d.ts,
+        /// or 'undefined' if ElementAttributesProperty doesn't exist (which means all
+        ///     non-intrinsic elements' attributes type is 'any'),
+        /// or '' if it has 0 properties (which means every
+        ///     non-intrinsic elements' attributes type is the element instance type)
+        function getJsxElementPropertiesName(jsxNamespace) {
+            return getNameFromJsxElementAttributesContainer(JsxNames.ElementAttributesPropertyNameContainer, jsxNamespace);
+        }
+        function getJsxElementChildrenPropertyName(jsxNamespace) {
+            return getNameFromJsxElementAttributesContainer(JsxNames.ElementChildrenAttributeNameContainer, jsxNamespace);
+        }
+        function getUninstantiatedJsxSignaturesOfType(elementType, caller) {
+            if (elementType.flags & 4 /* String */) {
+                return [anySignature];
             }
-            var hasExplicitReturn = func.flags & 512 /* HasExplicitReturn */;
-            if (type && type.flags & 131072 /* Never */) {
-                error(ts.getEffectiveReturnTypeNode(func), ts.Diagnostics.A_function_returning_never_cannot_have_a_reachable_end_point);
+            else if (elementType.flags & 128 /* StringLiteral */) {
+                var intrinsicType = getIntrinsicAttributesTypeFromStringLiteralType(elementType, caller);
+                if (!intrinsicType) {
+                    error(caller, ts.Diagnostics.Property_0_does_not_exist_on_type_1, elementType.value, "JSX." + JsxNames.IntrinsicElements);
+                    return ts.emptyArray;
+                }
+                else {
+                    var fakeSignature = createSignatureForJSXIntrinsic(caller, intrinsicType);
+                    return [fakeSignature];
+                }
             }
-            else if (type && !hasExplicitReturn) {
-                // minimal check: function has syntactic return type annotation and no explicit return statements in the body
-                // this function does not conform to the specification.
-                // NOTE: having returnType !== undefined is a precondition for entering this branch so func.type will always be present
-                error(ts.getEffectiveReturnTypeNode(func), ts.Diagnostics.A_function_whose_declared_type_is_neither_void_nor_any_must_return_a_value);
+            var apparentElemType = getApparentType(elementType);
+            // Resolve the signatures, preferring constructor
+            var signatures = getSignaturesOfType(apparentElemType, 1 /* Construct */);
+            if (signatures.length === 0) {
+                // No construct signatures, try call signatures
+                signatures = getSignaturesOfType(apparentElemType, 0 /* Call */);
             }
-            else if (type && strictNullChecks && !isTypeAssignableTo(undefinedType, type)) {
-                error(ts.getEffectiveReturnTypeNode(func) || func, ts.Diagnostics.Function_lacks_ending_return_statement_and_return_type_does_not_include_undefined);
+            if (signatures.length === 0 && apparentElemType.flags & 1048576 /* Union */) {
+                // If each member has some combination of new/call signatures; make a union signature list for those
+                signatures = getUnionSignatures(ts.map(apparentElemType.types, function (t) { return getUninstantiatedJsxSignaturesOfType(t, caller); }));
             }
-            else if (compilerOptions.noImplicitReturns) {
-                if (!type) {
-                    // If return type annotation is omitted check if function has any explicit return statements.
-                    // If it does not have any - its inferred return type is void - don't do any checks.
-                    // Otherwise get inferred return type from function body and report error only if it is not void / anytype
-                    if (!hasExplicitReturn) {
-                        return;
-                    }
-                    var inferredReturnType = getReturnTypeOfSignature(getSignatureFromDeclaration(func));
-                    if (isUnwrappedReturnTypeVoidOrAny(func, inferredReturnType)) {
-                        return;
-                    }
+            return signatures;
+        }
+        function getIntrinsicAttributesTypeFromStringLiteralType(type, location) {
+            // If the elemType is a stringLiteral type, we can then provide a check to make sure that the string literal type is one of the Jsx intrinsic element type
+            // For example:
+            //      var CustomTag: "h1" = "h1";
+            //      <CustomTag> Hello World </CustomTag>
+            var intrinsicElementsType = getJsxType(JsxNames.IntrinsicElements, location);
+            if (intrinsicElementsType !== errorType) {
+                var stringLiteralTypeName = type.value;
+                var intrinsicProp = getPropertyOfType(intrinsicElementsType, ts.escapeLeadingUnderscores(stringLiteralTypeName));
+                if (intrinsicProp) {
+                    return getTypeOfSymbol(intrinsicProp);
                 }
-                error(ts.getEffectiveReturnTypeNode(func) || func, ts.Diagnostics.Not_all_code_paths_return_a_value);
+                var indexSignatureType = getIndexTypeOfType(intrinsicElementsType, 0 /* String */);
+                if (indexSignatureType) {
+                    return indexSignatureType;
+                }
+                return undefined;
             }
+            // If we need to report an error, we already done so here. So just return any to prevent any more error downstream
+            return anyType;
         }
-        function checkFunctionExpressionOrObjectLiteralMethod(node, checkMode) {
-            ts.Debug.assert(node.kind !== 161 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
-            checkNodeDeferred(node);
-            // The identityMapper object is used to indicate that function expressions are wildcards
-            if (checkMode && checkMode & 4 /* SkipContextSensitive */ && isContextSensitive(node)) {
-                // Skip parameters, return signature with return type that retains noncontextual parts so inferences can still be drawn in an early stage
-                if (!ts.getEffectiveReturnTypeNode(node) && !hasContextSensitiveParameters(node)) {
-                    // Return plain anyFunctionType if there is no possibility we'll make inferences from the return type
-                    var contextualSignature = getContextualSignature(node);
-                    if (contextualSignature && couldContainTypeVariables(getReturnTypeOfSignature(contextualSignature))) {
-                        var links = getNodeLinks(node);
-                        if (links.contextFreeType) {
-                            return links.contextFreeType;
-                        }
-                        var returnType = getReturnTypeFromBody(node, checkMode);
-                        var returnOnlySignature = createSignature(undefined, undefined, undefined, ts.emptyArray, returnType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
-                        var returnOnlyType = createAnonymousType(node.symbol, emptySymbols, [returnOnlySignature], ts.emptyArray, undefined, undefined);
-                        returnOnlyType.objectFlags |= 2097152 /* NonInferrableType */;
-                        return links.contextFreeType = returnOnlyType;
-                    }
+        function checkJsxReturnAssignableToAppropriateBound(refKind, elemInstanceType, openingLikeElement) {
+            if (refKind === 1 /* Function */) {
+                var sfcReturnConstraint = getJsxStatelessElementTypeAt(openingLikeElement);
+                if (sfcReturnConstraint) {
+                    checkTypeRelatedTo(elemInstanceType, sfcReturnConstraint, assignableRelation, openingLikeElement.tagName, ts.Diagnostics.Its_return_type_0_is_not_a_valid_JSX_element, generateInitialErrorChain);
                 }
-                return anyFunctionType;
             }
-            // Grammar checking
-            var hasGrammarError = checkGrammarFunctionLikeDeclaration(node);
-            if (!hasGrammarError && node.kind === 201 /* FunctionExpression */) {
-                checkGrammarForGenerator(node);
+            else if (refKind === 0 /* Component */) {
+                var classConstraint = getJsxElementClassTypeAt(openingLikeElement);
+                if (classConstraint) {
+                    // Issue an error if this return type isn't assignable to JSX.ElementClass, failing that
+                    checkTypeRelatedTo(elemInstanceType, classConstraint, assignableRelation, openingLikeElement.tagName, ts.Diagnostics.Its_instance_type_0_is_not_a_valid_JSX_element, generateInitialErrorChain);
+                }
             }
-            contextuallyCheckFunctionExpressionOrObjectLiteralMethod(node, checkMode);
-            return getTypeOfSymbol(getSymbolOfNode(node));
-        }
-        function contextuallyCheckFunctionExpressionOrObjectLiteralMethod(node, checkMode) {
-            var links = getNodeLinks(node);
-            // Check if function expression is contextually typed and assign parameter types if so.
-            if (!(links.flags & 1024 /* ContextChecked */)) {
-                var contextualSignature = getContextualSignature(node);
-                // If a type check is started at a function expression that is an argument of a function call, obtaining the
-                // contextual type may recursively get back to here during overload resolution of the call. If so, we will have
-                // already assigned contextual types.
-                if (!(links.flags & 1024 /* ContextChecked */)) {
-                    links.flags |= 1024 /* ContextChecked */;
-                    var signature = ts.firstOrUndefined(getSignaturesOfType(getTypeOfSymbol(getSymbolOfNode(node)), 0 /* Call */));
-                    if (!signature) {
-                        return;
-                    }
-                    if (isContextSensitive(node)) {
-                        if (contextualSignature) {
-                            var inferenceContext = getInferenceContext(node);
-                            if (checkMode && checkMode & 2 /* Inferential */) {
-                                inferFromAnnotatedParameters(signature, contextualSignature, inferenceContext);
-                            }
-                            var instantiatedContextualSignature = inferenceContext ?
-                                instantiateSignature(contextualSignature, inferenceContext.mapper) : contextualSignature;
-                            assignContextualParameterTypes(signature, instantiatedContextualSignature);
-                        }
-                        else {
-                            // Force resolution of all parameter types such that the absence of a contextual type is consistently reflected.
-                            assignNonContextualParameterTypes(signature);
-                        }
-                    }
-                    if (contextualSignature && !getReturnTypeFromAnnotation(node) && !signature.resolvedReturnType) {
-                        var returnType = getReturnTypeFromBody(node, checkMode);
-                        if (!signature.resolvedReturnType) {
-                            signature.resolvedReturnType = returnType;
-                        }
-                    }
-                    checkSignatureDeclaration(node);
+            else { // Mixed
+                var sfcReturnConstraint = getJsxStatelessElementTypeAt(openingLikeElement);
+                var classConstraint = getJsxElementClassTypeAt(openingLikeElement);
+                if (!sfcReturnConstraint || !classConstraint) {
+                    return;
                 }
+                var combined = getUnionType([sfcReturnConstraint, classConstraint]);
+                checkTypeRelatedTo(elemInstanceType, combined, assignableRelation, openingLikeElement.tagName, ts.Diagnostics.Its_element_type_0_is_not_a_valid_JSX_element, generateInitialErrorChain);
+            }
+            function generateInitialErrorChain() {
+                var componentName = ts.getTextOfNode(openingLikeElement.tagName);
+                return ts.chainDiagnosticMessages(/* details */ undefined, ts.Diagnostics._0_cannot_be_used_as_a_JSX_component, componentName);
             }
         }
-        function checkFunctionExpressionOrObjectLiteralMethodDeferred(node) {
-            ts.Debug.assert(node.kind !== 161 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
-            var functionFlags = ts.getFunctionFlags(node);
-            var returnType = getReturnTypeFromAnnotation(node);
-            checkAllCodePathsInNonVoidFunctionReturnOrThrow(node, returnType);
-            if (node.body) {
-                if (!ts.getEffectiveReturnTypeNode(node)) {
-                    // There are some checks that are only performed in getReturnTypeFromBody, that may produce errors
-                    // we need. An example is the noImplicitAny errors resulting from widening the return expression
-                    // of a function. Because checking of function expression bodies is deferred, there was never an
-                    // appropriate time to do this during the main walk of the file (see the comment at the top of
-                    // checkFunctionExpressionBodies). So it must be done now.
-                    getReturnTypeOfSignature(getSignatureFromDeclaration(node));
+        /**
+         * Get attributes type of the given intrinsic opening-like Jsx element by resolving the tag name.
+         * The function is intended to be called from a function which has checked that the opening element is an intrinsic element.
+         * @param node an intrinsic JSX opening-like element
+         */
+        function getIntrinsicAttributesTypeFromJsxOpeningLikeElement(node) {
+            ts.Debug.assert(isJsxIntrinsicIdentifier(node.tagName));
+            var links = getNodeLinks(node);
+            if (!links.resolvedJsxElementAttributesType) {
+                var symbol = getIntrinsicTagSymbol(node);
+                if (links.jsxFlags & 1 /* IntrinsicNamedElement */) {
+                    return links.resolvedJsxElementAttributesType = getTypeOfSymbol(symbol);
                 }
-                if (node.body.kind === 223 /* Block */) {
-                    checkSourceElement(node.body);
+                else if (links.jsxFlags & 2 /* IntrinsicIndexedElement */) {
+                    return links.resolvedJsxElementAttributesType =
+                        getIndexTypeOfType(getDeclaredTypeOfSymbol(symbol), 0 /* String */);
                 }
                 else {
-                    // From within an async function you can return either a non-promise value or a promise. Any
-                    // Promise/A+ compatible implementation will always assimilate any foreign promise, so we
-                    // should not be checking assignability of a promise to the return type. Instead, we need to
-                    // check assignability of the awaited type of the expression body against the promised type of
-                    // its return type annotation.
-                    var exprType = checkExpression(node.body);
-                    var returnOrPromisedType = returnType && unwrapReturnType(returnType, functionFlags);
-                    if (returnOrPromisedType) {
-                        if ((functionFlags & 3 /* AsyncGenerator */) === 2 /* Async */) { // Async function
-                            var awaitedType = checkAwaitedType(exprType, node.body, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
-                            checkTypeAssignableToAndOptionallyElaborate(awaitedType, returnOrPromisedType, node.body, node.body);
-                        }
-                        else { // Normal function
-                            checkTypeAssignableToAndOptionallyElaborate(exprType, returnOrPromisedType, node.body, node.body);
-                        }
-                    }
+                    return links.resolvedJsxElementAttributesType = errorType;
                 }
             }
+            return links.resolvedJsxElementAttributesType;
         }
-        function checkArithmeticOperandType(operand, type, diagnostic, isAwaitValid) {
-            if (isAwaitValid === void 0) { isAwaitValid = false; }
-            if (!isTypeAssignableTo(type, numberOrBigIntType)) {
-                var awaitedType = isAwaitValid && getAwaitedTypeOfPromise(type);
-                errorAndMaybeSuggestAwait(operand, !!awaitedType && isTypeAssignableTo(awaitedType, numberOrBigIntType), diagnostic);
-                return false;
+        function getJsxElementClassTypeAt(location) {
+            var type = getJsxType(JsxNames.ElementClass, location);
+            if (type === errorType)
+                return undefined;
+            return type;
+        }
+        function getJsxElementTypeAt(location) {
+            return getJsxType(JsxNames.Element, location);
+        }
+        function getJsxStatelessElementTypeAt(location) {
+            var jsxElementType = getJsxElementTypeAt(location);
+            if (jsxElementType) {
+                return getUnionType([jsxElementType, nullType]);
             }
-            return true;
         }
-        function isReadonlyAssignmentDeclaration(d) {
-            if (!ts.isCallExpression(d)) {
-                return false;
+        /**
+         * Returns all the properties of the Jsx.IntrinsicElements interface
+         */
+        function getJsxIntrinsicTagNamesAt(location) {
+            var intrinsics = getJsxType(JsxNames.IntrinsicElements, location);
+            return intrinsics ? getPropertiesOfType(intrinsics) : ts.emptyArray;
+        }
+        function checkJsxPreconditions(errorNode) {
+            // Preconditions for using JSX
+            if ((compilerOptions.jsx || 0 /* None */) === 0 /* None */) {
+                error(errorNode, ts.Diagnostics.Cannot_use_JSX_unless_the_jsx_flag_is_provided);
             }
-            if (!ts.isBindableObjectDefinePropertyCall(d)) {
-                return false;
+            if (getJsxElementTypeAt(errorNode) === undefined) {
+                if (noImplicitAny) {
+                    error(errorNode, ts.Diagnostics.JSX_element_implicitly_has_type_any_because_the_global_type_JSX_Element_does_not_exist);
+                }
             }
-            var objectLitType = checkExpressionCached(d.arguments[2]);
-            var valueType = getTypeOfPropertyOfType(objectLitType, "value");
-            if (valueType) {
-                var writableProp = getPropertyOfType(objectLitType, "writable");
-                var writableType = writableProp && getTypeOfSymbol(writableProp);
-                if (!writableType || writableType === falseType || writableType === regularFalseType) {
+        }
+        function checkJsxOpeningLikeElementOrOpeningFragment(node) {
+            var isNodeOpeningLikeElement = ts.isJsxOpeningLikeElement(node);
+            if (isNodeOpeningLikeElement) {
+                checkGrammarJsxElement(node);
+            }
+            checkJsxPreconditions(node);
+            if (!getJsxNamespaceContainerForImplicitImport(node)) {
+                // The reactNamespace/jsxFactory's root symbol should be marked as 'used' so we don't incorrectly elide its import.
+                // And if there is no reactNamespace/jsxFactory's symbol in scope when targeting React emit, we should issue an error.
+                var jsxFactoryRefErr = diagnostics && compilerOptions.jsx === 2 /* React */ ? ts.Diagnostics.Cannot_find_name_0 : undefined;
+                var jsxFactoryNamespace = getJsxNamespace(node);
+                var jsxFactoryLocation = isNodeOpeningLikeElement ? node.tagName : node;
+                // allow null as jsxFragmentFactory
+                var jsxFactorySym = void 0;
+                if (!(ts.isJsxOpeningFragment(node) && jsxFactoryNamespace === "null")) {
+                    jsxFactorySym = resolveName(jsxFactoryLocation, jsxFactoryNamespace, 111551 /* Value */, jsxFactoryRefErr, jsxFactoryNamespace, /*isUse*/ true);
+                }
+                if (jsxFactorySym) {
+                    // Mark local symbol as referenced here because it might not have been marked
+                    // if jsx emit was not jsxFactory as there wont be error being emitted
+                    jsxFactorySym.isReferenced = 67108863 /* All */;
+                    // If react/jsxFactory symbol is alias, mark it as refereced
+                    if (jsxFactorySym.flags & 2097152 /* Alias */ && !getTypeOnlyAliasDeclaration(jsxFactorySym)) {
+                        markAliasSymbolAsReferenced(jsxFactorySym);
+                    }
+                }
+            }
+            if (isNodeOpeningLikeElement) {
+                var jsxOpeningLikeNode = node;
+                var sig = getResolvedSignature(jsxOpeningLikeNode);
+                checkDeprecatedSignature(sig, node);
+                checkJsxReturnAssignableToAppropriateBound(getJsxReferenceKind(jsxOpeningLikeNode), getReturnTypeOfSignature(sig), jsxOpeningLikeNode);
+            }
+        }
+        /**
+         * Check if a property with the given name is known anywhere in the given type. In an object type, a property
+         * is considered known if
+         * 1. the object type is empty and the check is for assignability, or
+         * 2. if the object type has index signatures, or
+         * 3. if the property is actually declared in the object type
+         *    (this means that 'toString', for example, is not usually a known property).
+         * 4. In a union or intersection type,
+         *    a property is considered known if it is known in any constituent type.
+         * @param targetType a type to search a given name in
+         * @param name a property name to search
+         * @param isComparingJsxAttributes a boolean flag indicating whether we are searching in JsxAttributesType
+         */
+        function isKnownProperty(targetType, name, isComparingJsxAttributes) {
+            if (targetType.flags & 524288 /* Object */) {
+                var resolved = resolveStructuredTypeMembers(targetType);
+                if (resolved.stringIndexInfo ||
+                    resolved.numberIndexInfo && isNumericLiteralName(name) ||
+                    getPropertyOfObjectType(targetType, name) ||
+                    isComparingJsxAttributes && !isUnhyphenatedJsxName(name)) {
+                    // For JSXAttributes, if the attribute has a hyphenated name, consider that the attribute to be known.
                     return true;
                 }
-                // We include this definition whereupon we walk back and check the type at the declaration because
-                // The usual definition of `Object.defineProperty` will _not_ cause literal types to be preserved in the
-                // argument types, should the type be contextualized by the call itself.
-                if (writableProp && writableProp.valueDeclaration && ts.isPropertyAssignment(writableProp.valueDeclaration)) {
-                    var initializer = writableProp.valueDeclaration.initializer;
-                    var rawOriginalType = checkExpression(initializer);
-                    if (rawOriginalType === falseType || rawOriginalType === regularFalseType) {
+            }
+            else if (targetType.flags & 3145728 /* UnionOrIntersection */ && isExcessPropertyCheckTarget(targetType)) {
+                for (var _i = 0, _a = targetType.types; _i < _a.length; _i++) {
+                    var t = _a[_i];
+                    if (isKnownProperty(t, name, isComparingJsxAttributes)) {
                         return true;
                     }
                 }
-                return false;
             }
-            var setProp = getPropertyOfType(objectLitType, "set");
-            return !setProp;
+            return false;
         }
-        function isReadonlySymbol(symbol) {
-            // The following symbols are considered read-only:
-            // Properties with a 'readonly' modifier
-            // Variables declared with 'const'
-            // Get accessors without matching set accessors
-            // Enum members
-            // Object.defineProperty assignments with writable false or no setter
-            // Unions and intersections of the above (unions and intersections eagerly set isReadonly on creation)
-            return !!(ts.getCheckFlags(symbol) & 8 /* Readonly */ ||
-                symbol.flags & 4 /* Property */ && ts.getDeclarationModifierFlagsFromSymbol(symbol) & 64 /* Readonly */ ||
-                symbol.flags & 3 /* Variable */ && getDeclarationNodeFlagsFromSymbol(symbol) & 2 /* Const */ ||
-                symbol.flags & 98304 /* Accessor */ && !(symbol.flags & 65536 /* SetAccessor */) ||
-                symbol.flags & 8 /* EnumMember */ ||
-                ts.some(symbol.declarations, isReadonlyAssignmentDeclaration));
+        function isExcessPropertyCheckTarget(type) {
+            return !!(type.flags & 524288 /* Object */ && !(ts.getObjectFlags(type) & 512 /* ObjectLiteralPatternWithComputedProperties */) ||
+                type.flags & 67108864 /* NonPrimitive */ ||
+                type.flags & 1048576 /* Union */ && ts.some(type.types, isExcessPropertyCheckTarget) ||
+                type.flags & 2097152 /* Intersection */ && ts.every(type.types, isExcessPropertyCheckTarget));
         }
-        function isAssignmentToReadonlyEntity(expr, symbol, assignmentKind) {
-            var _a, _b;
-            if (assignmentKind === 0 /* None */) {
-                // no assigment means it doesn't matter whether the entity is readonly
-                return false;
+        function checkJsxExpression(node, checkMode) {
+            checkGrammarJsxExpression(node);
+            if (node.expression) {
+                var type = checkExpression(node.expression, checkMode);
+                if (node.dotDotDotToken && type !== anyType && !isArrayType(type)) {
+                    error(node, ts.Diagnostics.JSX_spread_child_must_be_an_array_type);
+                }
+                return type;
             }
-            if (isReadonlySymbol(symbol)) {
-                // Allow assignments to readonly properties within constructors of the same class declaration.
-                if (symbol.flags & 4 /* Property */ &&
-                    ts.isAccessExpression(expr) &&
-                    expr.expression.kind === 104 /* ThisKeyword */) {
-                    // Look for if this is the constructor for the class that `symbol` is a property of.
-                    var ctor = ts.getContainingFunction(expr);
-                    if (!(ctor && ctor.kind === 162 /* Constructor */)) {
-                        return true;
-                    }
-                    if (symbol.valueDeclaration) {
-                        var isAssignmentDeclaration_1 = ts.isBinaryExpression(symbol.valueDeclaration);
-                        var isLocalPropertyDeclaration = ctor.parent === symbol.valueDeclaration.parent;
-                        var isLocalParameterProperty = ctor === symbol.valueDeclaration.parent;
-                        var isLocalThisPropertyAssignment = isAssignmentDeclaration_1 && ((_a = symbol.parent) === null || _a === void 0 ? void 0 : _a.valueDeclaration) === ctor.parent;
-                        var isLocalThisPropertyAssignmentConstructorFunction = isAssignmentDeclaration_1 && ((_b = symbol.parent) === null || _b === void 0 ? void 0 : _b.valueDeclaration) === ctor;
-                        var isWriteableSymbol = isLocalPropertyDeclaration
-                            || isLocalParameterProperty
-                            || isLocalThisPropertyAssignment
-                            || isLocalThisPropertyAssignmentConstructorFunction;
-                        return !isWriteableSymbol;
+            else {
+                return errorType;
+            }
+        }
+        function getDeclarationNodeFlagsFromSymbol(s) {
+            return s.valueDeclaration ? ts.getCombinedNodeFlags(s.valueDeclaration) : 0;
+        }
+        /**
+         * Return whether this symbol is a member of a prototype somewhere
+         * Note that this is not tracked well within the compiler, so the answer may be incorrect.
+         */
+        function isPrototypeProperty(symbol) {
+            if (symbol.flags & 8192 /* Method */ || ts.getCheckFlags(symbol) & 4 /* SyntheticMethod */) {
+                return true;
+            }
+            if (ts.isInJSFile(symbol.valueDeclaration)) {
+                var parent = symbol.valueDeclaration.parent;
+                return parent && ts.isBinaryExpression(parent) &&
+                    ts.getAssignmentDeclarationKind(parent) === 3 /* PrototypeProperty */;
+            }
+        }
+        /**
+         * Check whether the requested property access is valid.
+         * Returns true if node is a valid property access, and false otherwise.
+         * @param node The node to be checked.
+         * @param isSuper True if the access is from `super.`.
+         * @param type The type of the object whose property is being accessed. (Not the type of the property.)
+         * @param prop The symbol for the property being accessed.
+         */
+        function checkPropertyAccessibility(node, isSuper, type, prop) {
+            var flags = ts.getDeclarationModifierFlagsFromSymbol(prop);
+            var errorNode = node.kind === 157 /* QualifiedName */ ? node.right : node.kind === 195 /* ImportType */ ? node : node.name;
+            if (isSuper) {
+                // TS 1.0 spec (April 2014): 4.8.2
+                // - In a constructor, instance member function, instance member accessor, or
+                //   instance member variable initializer where this references a derived class instance,
+                //   a super property access is permitted and must specify a public instance member function of the base class.
+                // - In a static member function or static member accessor
+                //   where this references the constructor function object of a derived class,
+                //   a super property access is permitted and must specify a public static member function of the base class.
+                if (languageVersion < 2 /* ES2015 */) {
+                    if (symbolHasNonMethodDeclaration(prop)) {
+                        error(errorNode, ts.Diagnostics.Only_public_and_protected_methods_of_the_base_class_are_accessible_via_the_super_keyword);
+                        return false;
                     }
                 }
+                if (flags & 128 /* Abstract */) {
+                    // A method cannot be accessed in a super property access if the method is abstract.
+                    // This error could mask a private property access error. But, a member
+                    // cannot simultaneously be private and abstract, so this will trigger an
+                    // additional error elsewhere.
+                    error(errorNode, ts.Diagnostics.Abstract_method_0_in_class_1_cannot_be_accessed_via_super_expression, symbolToString(prop), typeToString(getDeclaringClass(prop)));
+                    return false;
+                }
+            }
+            // Referencing abstract properties within their own constructors is not allowed
+            if ((flags & 128 /* Abstract */) && ts.isThisProperty(node) && symbolHasNonMethodDeclaration(prop)) {
+                var declaringClassDeclaration = ts.getClassLikeDeclarationOfSymbol(getParentOfSymbol(prop));
+                if (declaringClassDeclaration && isNodeUsedDuringClassInitialization(node)) {
+                    error(errorNode, ts.Diagnostics.Abstract_property_0_in_class_1_cannot_be_accessed_in_the_constructor, symbolToString(prop), ts.getTextOfIdentifierOrLiteral(declaringClassDeclaration.name)); // TODO: GH#18217
+                    return false;
+                }
+            }
+            if (ts.isPropertyAccessExpression(node) && ts.isPrivateIdentifier(node.name)) {
+                if (!ts.getContainingClass(node)) {
+                    error(errorNode, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
+                    return false;
+                }
                 return true;
             }
-            if (ts.isAccessExpression(expr)) {
-                // references through namespace import should be readonly
-                var node = ts.skipParentheses(expr.expression);
-                if (node.kind === 75 /* Identifier */) {
-                    var symbol_2 = getNodeLinks(node).resolvedSymbol;
-                    if (symbol_2.flags & 2097152 /* Alias */) {
-                        var declaration = getDeclarationOfAliasSymbol(symbol_2);
-                        return !!declaration && declaration.kind === 256 /* NamespaceImport */;
-                    }
+            // Public properties are otherwise accessible.
+            if (!(flags & 24 /* NonPublicAccessibilityModifier */)) {
+                return true;
+            }
+            // Property is known to be private or protected at this point
+            // Private property is accessible if the property is within the declaring class
+            if (flags & 8 /* Private */) {
+                var declaringClassDeclaration = ts.getClassLikeDeclarationOfSymbol(getParentOfSymbol(prop));
+                if (!isNodeWithinClass(node, declaringClassDeclaration)) {
+                    error(errorNode, ts.Diagnostics.Property_0_is_private_and_only_accessible_within_class_1, symbolToString(prop), typeToString(getDeclaringClass(prop)));
+                    return false;
                 }
+                return true;
             }
-            return false;
-        }
-        function checkReferenceExpression(expr, invalidReferenceMessage, invalidOptionalChainMessage) {
-            // References are combinations of identifiers, parentheses, and property accesses.
-            var node = ts.skipOuterExpressions(expr, 6 /* Assertions */ | 1 /* Parentheses */);
-            if (node.kind !== 75 /* Identifier */ && !ts.isAccessExpression(node)) {
-                error(expr, invalidReferenceMessage);
-                return false;
+            // Property is known to be protected at this point
+            // All protected properties of a supertype are accessible in a super access
+            if (isSuper) {
+                return true;
             }
-            if (node.flags & 32 /* OptionalChain */) {
-                error(expr, invalidOptionalChainMessage);
+            // Find the first enclosing class that has the declaring classes of the protected constituents
+            // of the property as base classes
+            var enclosingClass = forEachEnclosingClass(node, function (enclosingDeclaration) {
+                var enclosingClass = getDeclaredTypeOfSymbol(getSymbolOfNode(enclosingDeclaration));
+                return isClassDerivedFromDeclaringClasses(enclosingClass, prop) ? enclosingClass : undefined;
+            });
+            // A protected property is accessible if the property is within the declaring class or classes derived from it
+            if (!enclosingClass) {
+                // allow PropertyAccessibility if context is in function with this parameter
+                // static member access is disallow
+                var thisParameter = void 0;
+                if (flags & 32 /* Static */ || !(thisParameter = getThisParameterFromNodeContext(node)) || !thisParameter.type) {
+                    error(errorNode, ts.Diagnostics.Property_0_is_protected_and_only_accessible_within_class_1_and_its_subclasses, symbolToString(prop), typeToString(getDeclaringClass(prop) || type));
+                    return false;
+                }
+                var thisType = getTypeFromTypeNode(thisParameter.type);
+                enclosingClass = ((thisType.flags & 262144 /* TypeParameter */) ? getConstraintOfTypeParameter(thisType) : thisType).target;
+            }
+            // No further restrictions for static properties
+            if (flags & 32 /* Static */) {
+                return true;
+            }
+            if (type.flags & 262144 /* TypeParameter */) {
+                // get the original type -- represented as the type constraint of the 'this' type
+                type = type.isThisType ? getConstraintOfTypeParameter(type) : getBaseConstraintOfType(type); // TODO: GH#18217 Use a different variable that's allowed to be undefined
+            }
+            if (!type || !hasBaseType(type, enclosingClass)) {
+                error(errorNode, ts.Diagnostics.Property_0_is_protected_and_only_accessible_through_an_instance_of_class_1, symbolToString(prop), typeToString(enclosingClass));
                 return false;
             }
             return true;
         }
-        function checkDeleteExpression(node) {
-            checkExpression(node.expression);
-            var expr = ts.skipParentheses(node.expression);
-            if (!ts.isAccessExpression(expr)) {
-                error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_must_be_a_property_reference);
-                return booleanType;
+        function getThisParameterFromNodeContext(node) {
+            var thisContainer = ts.getThisContainer(node, /* includeArrowFunctions */ false);
+            return thisContainer && ts.isFunctionLike(thisContainer) ? ts.getThisParameter(thisContainer) : undefined;
+        }
+        function symbolHasNonMethodDeclaration(symbol) {
+            return !!forEachProperty(symbol, function (prop) { return !(prop.flags & 8192 /* Method */); });
+        }
+        function checkNonNullExpression(node) {
+            return checkNonNullType(checkExpression(node), node);
+        }
+        function isNullableType(type) {
+            return !!((strictNullChecks ? getFalsyFlags(type) : type.flags) & 98304 /* Nullable */);
+        }
+        function getNonNullableTypeIfNeeded(type) {
+            return isNullableType(type) ? getNonNullableType(type) : type;
+        }
+        function reportObjectPossiblyNullOrUndefinedError(node, flags) {
+            error(node, flags & 32768 /* Undefined */ ? flags & 65536 /* Null */ ?
+                ts.Diagnostics.Object_is_possibly_null_or_undefined :
+                ts.Diagnostics.Object_is_possibly_undefined :
+                ts.Diagnostics.Object_is_possibly_null);
+        }
+        function reportCannotInvokePossiblyNullOrUndefinedError(node, flags) {
+            error(node, flags & 32768 /* Undefined */ ? flags & 65536 /* Null */ ?
+                ts.Diagnostics.Cannot_invoke_an_object_which_is_possibly_null_or_undefined :
+                ts.Diagnostics.Cannot_invoke_an_object_which_is_possibly_undefined :
+                ts.Diagnostics.Cannot_invoke_an_object_which_is_possibly_null);
+        }
+        function checkNonNullTypeWithReporter(type, node, reportError) {
+            if (strictNullChecks && type.flags & 2 /* Unknown */) {
+                error(node, ts.Diagnostics.Object_is_of_type_unknown);
+                return errorType;
             }
-            // eslint-disable-next-line
-            if (expr.kind === 194 /* PropertyAccessExpression */ && ts.isPrivateIdentifier(expr.name)) {
-                error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_cannot_be_a_private_identifier);
+            var kind = (strictNullChecks ? getFalsyFlags(type) : type.flags) & 98304 /* Nullable */;
+            if (kind) {
+                reportError(node, kind);
+                var t = getNonNullableType(type);
+                return t.flags & (98304 /* Nullable */ | 131072 /* Never */) ? errorType : t;
             }
-            var links = getNodeLinks(expr);
-            var symbol = getExportSymbolOfValueSymbolIfExported(links.resolvedSymbol);
-            if (symbol && isReadonlySymbol(symbol)) {
-                error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_cannot_be_a_read_only_property);
+            return type;
+        }
+        function checkNonNullType(type, node) {
+            return checkNonNullTypeWithReporter(type, node, reportObjectPossiblyNullOrUndefinedError);
+        }
+        function checkNonNullNonVoidType(type, node) {
+            var nonNullType = checkNonNullType(type, node);
+            if (nonNullType !== errorType && nonNullType.flags & 16384 /* Void */) {
+                error(node, ts.Diagnostics.Object_is_possibly_undefined);
             }
-            return booleanType;
+            return nonNullType;
         }
-        function checkTypeOfExpression(node) {
-            checkExpression(node.expression);
-            return typeofType;
+        function checkPropertyAccessExpression(node) {
+            return node.flags & 32 /* OptionalChain */ ? checkPropertyAccessChain(node) :
+                checkPropertyAccessExpressionOrQualifiedName(node, node.expression, checkNonNullExpression(node.expression), node.name);
         }
-        function checkVoidExpression(node) {
-            checkExpression(node.expression);
-            return undefinedWideningType;
+        function checkPropertyAccessChain(node) {
+            var leftType = checkExpression(node.expression);
+            var nonOptionalType = getOptionalExpressionType(leftType, node.expression);
+            return propagateOptionalTypeMarker(checkPropertyAccessExpressionOrQualifiedName(node, node.expression, checkNonNullType(nonOptionalType, node.expression), node.name), node, nonOptionalType !== leftType);
         }
-        function isTopLevelAwait(node) {
-            var container = ts.getThisContainer(node, /*includeArrowFunctions*/ true);
-            return ts.isSourceFile(container);
+        function checkQualifiedName(node) {
+            return checkPropertyAccessExpressionOrQualifiedName(node, node.left, checkNonNullExpression(node.left), node.right);
         }
-        function checkAwaitExpression(node) {
-            // Grammar checking
-            if (produceDiagnostics) {
-                if (!(node.flags & 32768 /* AwaitContext */)) {
-                    if (isTopLevelAwait(node)) {
-                        var sourceFile = ts.getSourceFileOfNode(node);
-                        if (!hasParseDiagnostics(sourceFile)) {
-                            var span = void 0;
-                            if (!ts.isEffectiveExternalModule(sourceFile, compilerOptions)) {
-                                if (!span)
-                                    span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
-                                var diagnostic = ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.await_expressions_are_only_allowed_at_the_top_level_of_a_file_when_that_file_is_a_module_but_this_file_has_no_imports_or_exports_Consider_adding_an_empty_export_to_make_this_file_a_module);
-                                diagnostics.add(diagnostic);
-                            }
-                            if ((moduleKind !== ts.ModuleKind.ESNext && moduleKind !== ts.ModuleKind.System) || languageVersion < 4 /* ES2017 */) {
-                                span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
-                                var diagnostic = ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.Top_level_await_expressions_are_only_allowed_when_the_module_option_is_set_to_esnext_or_system_and_the_target_option_is_set_to_es2017_or_higher);
-                                diagnostics.add(diagnostic);
-                            }
-                        }
+        function isMethodAccessForCall(node) {
+            while (node.parent.kind === 207 /* ParenthesizedExpression */) {
+                node = node.parent;
+            }
+            return ts.isCallOrNewExpression(node.parent) && node.parent.expression === node;
+        }
+        // Lookup the private identifier lexically.
+        function lookupSymbolForPrivateIdentifierDeclaration(propName, location) {
+            for (var containingClass = ts.getContainingClass(location); !!containingClass; containingClass = ts.getContainingClass(containingClass)) {
+                var symbol = containingClass.symbol;
+                var name = ts.getSymbolNameForPrivateIdentifier(symbol, propName);
+                var prop = (symbol.members && symbol.members.get(name)) || (symbol.exports && symbol.exports.get(name));
+                if (prop) {
+                    return prop;
+                }
+            }
+        }
+        function getPrivateIdentifierPropertyOfType(leftType, lexicallyScopedIdentifier) {
+            return getPropertyOfType(leftType, lexicallyScopedIdentifier.escapedName);
+        }
+        function checkPrivateIdentifierPropertyAccess(leftType, right, lexicallyScopedIdentifier) {
+            // Either the identifier could not be looked up in the lexical scope OR the lexically scoped identifier did not exist on the type.
+            // Find a private identifier with the same description on the type.
+            var propertyOnType;
+            var properties = getPropertiesOfType(leftType);
+            if (properties) {
+                ts.forEach(properties, function (symbol) {
+                    var decl = symbol.valueDeclaration;
+                    if (decl && ts.isNamedDeclaration(decl) && ts.isPrivateIdentifier(decl.name) && decl.name.escapedText === right.escapedText) {
+                        propertyOnType = symbol;
+                        return true;
                     }
-                    else {
-                        // use of 'await' in non-async function
-                        var sourceFile = ts.getSourceFileOfNode(node);
-                        if (!hasParseDiagnostics(sourceFile)) {
-                            var span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
-                            var diagnostic = ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.await_expressions_are_only_allowed_within_async_functions_and_at_the_top_levels_of_modules);
-                            var func = ts.getContainingFunction(node);
-                            if (func && func.kind !== 162 /* Constructor */ && (ts.getFunctionFlags(func) & 2 /* Async */) === 0) {
-                                var relatedInfo = ts.createDiagnosticForNode(func, ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async);
-                                ts.addRelatedInfo(diagnostic, relatedInfo);
-                            }
-                            diagnostics.add(diagnostic);
-                        }
+                });
+            }
+            var diagName = diagnosticName(right);
+            if (propertyOnType) {
+                var typeValueDecl = propertyOnType.valueDeclaration;
+                var typeClass_1 = ts.getContainingClass(typeValueDecl);
+                ts.Debug.assert(!!typeClass_1);
+                // We found a private identifier property with the same description.
+                // Either:
+                // - There is a lexically scoped private identifier AND it shadows the one we found on the type.
+                // - It is an attempt to access the private identifier outside of the class.
+                if (lexicallyScopedIdentifier) {
+                    var lexicalValueDecl = lexicallyScopedIdentifier.valueDeclaration;
+                    var lexicalClass = ts.getContainingClass(lexicalValueDecl);
+                    ts.Debug.assert(!!lexicalClass);
+                    if (ts.findAncestor(lexicalClass, function (n) { return typeClass_1 === n; })) {
+                        var diagnostic = error(right, ts.Diagnostics.The_property_0_cannot_be_accessed_on_type_1_within_this_class_because_it_is_shadowed_by_another_private_identifier_with_the_same_spelling, diagName, typeToString(leftType));
+                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(lexicalValueDecl, ts.Diagnostics.The_shadowing_declaration_of_0_is_defined_here, diagName), ts.createDiagnosticForNode(typeValueDecl, ts.Diagnostics.The_declaration_of_0_that_you_probably_intended_to_use_is_defined_here, diagName));
+                        return true;
                     }
                 }
-                if (isInParameterInitializerBeforeContainingFunction(node)) {
-                    error(node, ts.Diagnostics.await_expressions_cannot_be_used_in_a_parameter_initializer);
-                }
-            }
-            var operandType = checkExpression(node.expression);
-            var awaitedType = checkAwaitedType(operandType, node, ts.Diagnostics.Type_of_await_operand_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
-            if (awaitedType === operandType && awaitedType !== errorType && !(operandType.flags & 3 /* AnyOrUnknown */)) {
-                addErrorOrSuggestion(/*isError*/ false, ts.createDiagnosticForNode(node, ts.Diagnostics.await_has_no_effect_on_the_type_of_this_expression));
+                error(right, ts.Diagnostics.Property_0_is_not_accessible_outside_class_1_because_it_has_a_private_identifier, diagName, diagnosticName(typeClass_1.name || anon));
+                return true;
             }
-            return awaitedType;
+            return false;
         }
-        function checkPrefixUnaryExpression(node) {
-            var operandType = checkExpression(node.operand);
-            if (operandType === silentNeverType) {
-                return silentNeverType;
+        function isThisPropertyAccessInConstructor(node, prop) {
+            return (isConstructorDeclaredProperty(prop) || ts.isThisProperty(node) && isAutoTypedProperty(prop))
+                && ts.getThisContainer(node, /*includeArrowFunctions*/ true) === getDeclaringConstructor(prop);
+        }
+        function checkPropertyAccessExpressionOrQualifiedName(node, left, leftType, right) {
+            var parentSymbol = getNodeLinks(left).resolvedSymbol;
+            var assignmentKind = ts.getAssignmentTargetKind(node);
+            var apparentType = getApparentType(assignmentKind !== 0 /* None */ || isMethodAccessForCall(node) ? getWidenedType(leftType) : leftType);
+            if (ts.isPrivateIdentifier(right)) {
+                checkExternalEmitHelpers(node, 1048576 /* ClassPrivateFieldGet */);
             }
-            switch (node.operand.kind) {
-                case 8 /* NumericLiteral */:
-                    switch (node.operator) {
-                        case 40 /* MinusToken */:
-                            return getFreshTypeOfLiteralType(getLiteralType(-node.operand.text));
-                        case 39 /* PlusToken */:
-                            return getFreshTypeOfLiteralType(getLiteralType(+node.operand.text));
+            var isAnyLike = isTypeAny(apparentType) || apparentType === silentNeverType;
+            var prop;
+            if (ts.isPrivateIdentifier(right)) {
+                var lexicallyScopedSymbol = lookupSymbolForPrivateIdentifierDeclaration(right.escapedText, right);
+                if (isAnyLike) {
+                    if (lexicallyScopedSymbol) {
+                        return apparentType;
                     }
-                    break;
-                case 9 /* BigIntLiteral */:
-                    if (node.operator === 40 /* MinusToken */) {
-                        return getFreshTypeOfLiteralType(getLiteralType({
-                            negative: true,
-                            base10Value: ts.parsePseudoBigInt(node.operand.text)
-                        }));
+                    if (!ts.getContainingClass(right)) {
+                        grammarErrorOnNode(right, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
+                        return anyType;
                     }
+                }
+                prop = lexicallyScopedSymbol ? getPrivateIdentifierPropertyOfType(leftType, lexicallyScopedSymbol) : undefined;
+                // Check for private-identifier-specific shadowing and lexical-scoping errors.
+                if (!prop && checkPrivateIdentifierPropertyAccess(leftType, right, lexicallyScopedSymbol)) {
+                    return errorType;
+                }
             }
-            switch (node.operator) {
-                case 39 /* PlusToken */:
-                case 40 /* MinusToken */:
-                case 54 /* TildeToken */:
-                    checkNonNullType(operandType, node.operand);
-                    if (maybeTypeOfKind(operandType, 12288 /* ESSymbolLike */)) {
-                        error(node.operand, ts.Diagnostics.The_0_operator_cannot_be_applied_to_type_symbol, ts.tokenToString(node.operator));
+            else {
+                if (isAnyLike) {
+                    if (ts.isIdentifier(left) && parentSymbol) {
+                        markAliasReferenced(parentSymbol, node);
                     }
-                    if (node.operator === 39 /* PlusToken */) {
-                        if (maybeTypeOfKind(operandType, 2112 /* BigIntLike */)) {
-                            error(node.operand, ts.Diagnostics.Operator_0_cannot_be_applied_to_type_1, ts.tokenToString(node.operator), typeToString(getBaseTypeOfLiteralType(operandType)));
+                    return apparentType;
+                }
+                prop = getPropertyOfType(apparentType, right.escapedText);
+            }
+            if (ts.isIdentifier(left) && parentSymbol && !(prop && isConstEnumOrConstEnumOnlyModule(prop))) {
+                markAliasReferenced(parentSymbol, node);
+            }
+            var propType;
+            if (!prop) {
+                var indexInfo = !ts.isPrivateIdentifier(right) && (assignmentKind === 0 /* None */ || !isGenericObjectType(leftType) || isThisTypeParameter(leftType)) ? getIndexInfoOfType(apparentType, 0 /* String */) : undefined;
+                if (!(indexInfo && indexInfo.type)) {
+                    if (isJSLiteralType(leftType)) {
+                        return anyType;
+                    }
+                    if (leftType.symbol === globalThisSymbol) {
+                        if (globalThisSymbol.exports.has(right.escapedText) && (globalThisSymbol.exports.get(right.escapedText).flags & 418 /* BlockScoped */)) {
+                            error(right, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.unescapeLeadingUnderscores(right.escapedText), typeToString(leftType));
                         }
-                        return numberType;
+                        else if (noImplicitAny) {
+                            error(right, ts.Diagnostics.Element_implicitly_has_an_any_type_because_type_0_has_no_index_signature, typeToString(leftType));
+                        }
+                        return anyType;
                     }
-                    return getUnaryResultType(operandType);
-                case 53 /* ExclamationToken */:
-                    checkTruthinessExpression(node.operand);
-                    var facts = getTypeFacts(operandType) & (4194304 /* Truthy */ | 8388608 /* Falsy */);
-                    return facts === 4194304 /* Truthy */ ? falseType :
-                        facts === 8388608 /* Falsy */ ? trueType :
-                            booleanType;
-                case 45 /* PlusPlusToken */:
-                case 46 /* MinusMinusToken */:
-                    var ok = checkArithmeticOperandType(node.operand, checkNonNullType(operandType, node.operand), ts.Diagnostics.An_arithmetic_operand_must_be_of_type_any_number_bigint_or_an_enum_type);
-                    if (ok) {
-                        // run check only if former checks succeeded to avoid reporting cascading errors
-                        checkReferenceExpression(node.operand, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_may_not_be_an_optional_property_access);
+                    if (right.escapedText && !checkAndReportErrorForExtendingInterface(node)) {
+                        reportNonexistentProperty(right, isThisTypeParameter(leftType) ? apparentType : leftType);
                     }
-                    return getUnaryResultType(operandType);
-            }
-            return errorType;
-        }
-        function checkPostfixUnaryExpression(node) {
-            var operandType = checkExpression(node.operand);
-            if (operandType === silentNeverType) {
-                return silentNeverType;
+                    return errorType;
+                }
+                if (indexInfo.isReadonly && (ts.isAssignmentTarget(node) || ts.isDeleteTarget(node))) {
+                    error(node, ts.Diagnostics.Index_signature_in_type_0_only_permits_reading, typeToString(apparentType));
+                }
+                propType = (compilerOptions.noUncheckedIndexedAccess && !ts.isAssignmentTarget(node)) ? getUnionType([indexInfo.type, undefinedType]) : indexInfo.type;
             }
-            var ok = checkArithmeticOperandType(node.operand, checkNonNullType(operandType, node.operand), ts.Diagnostics.An_arithmetic_operand_must_be_of_type_any_number_bigint_or_an_enum_type);
-            if (ok) {
-                // run check only if former checks succeeded to avoid reporting cascading errors
-                checkReferenceExpression(node.operand, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_may_not_be_an_optional_property_access);
+            else {
+                if (getDeclarationNodeFlagsFromSymbol(prop) & 134217728 /* Deprecated */ && isUncalledFunctionReference(node, prop)) {
+                    errorOrSuggestion(/* isError */ false, right, ts.Diagnostics._0_is_deprecated, right.escapedText);
+                }
+                checkPropertyNotUsedBeforeDeclaration(prop, node, right);
+                markPropertyAsReferenced(prop, node, left.kind === 107 /* ThisKeyword */);
+                getNodeLinks(node).resolvedSymbol = prop;
+                checkPropertyAccessibility(node, left.kind === 105 /* SuperKeyword */, apparentType, prop);
+                if (isAssignmentToReadonlyEntity(node, prop, assignmentKind)) {
+                    error(right, ts.Diagnostics.Cannot_assign_to_0_because_it_is_a_read_only_property, ts.idText(right));
+                    return errorType;
+                }
+                propType = isThisPropertyAccessInConstructor(node, prop) ? autoType : getConstraintForLocation(getTypeOfSymbol(prop), node);
             }
-            return getUnaryResultType(operandType);
+            return getFlowTypeOfAccessExpression(node, prop, propType, right);
         }
-        function getUnaryResultType(operandType) {
-            if (maybeTypeOfKind(operandType, 2112 /* BigIntLike */)) {
-                return isTypeAssignableToKind(operandType, 3 /* AnyOrUnknown */) || maybeTypeOfKind(operandType, 296 /* NumberLike */)
-                    ? numberOrBigIntType
-                    : bigintType;
+        function getFlowTypeOfAccessExpression(node, prop, propType, errorNode) {
+            // Only compute control flow type if this is a property access expression that isn't an
+            // assignment target, and the referenced property was declared as a variable, property,
+            // accessor, or optional method.
+            var assignmentKind = ts.getAssignmentTargetKind(node);
+            if (!ts.isAccessExpression(node) ||
+                assignmentKind === 1 /* Definite */ ||
+                prop && !(prop.flags & (3 /* Variable */ | 4 /* Property */ | 98304 /* Accessor */)) && !(prop.flags & 8192 /* Method */ && propType.flags & 1048576 /* Union */)) {
+                return propType;
             }
-            // If it's not a bigint type, implicit coercion will result in a number
-            return numberType;
-        }
-        // Return true if type might be of the given kind. A union or intersection type might be of a given
-        // kind if at least one constituent type is of the given kind.
-        function maybeTypeOfKind(type, kind) {
-            if (type.flags & kind) {
-                return true;
+            if (propType === autoType) {
+                return getFlowTypeOfProperty(node, prop);
             }
-            if (type.flags & 3145728 /* UnionOrIntersection */) {
-                var types = type.types;
-                for (var _i = 0, types_19 = types; _i < types_19.length; _i++) {
-                    var t = types_19[_i];
-                    if (maybeTypeOfKind(t, kind)) {
-                        return true;
+            // If strict null checks and strict property initialization checks are enabled, if we have
+            // a this.xxx property access, if the property is an instance property without an initializer,
+            // and if we are in a constructor of the same class as the property declaration, assume that
+            // the property is uninitialized at the top of the control flow.
+            var assumeUninitialized = false;
+            if (strictNullChecks && strictPropertyInitialization && node.expression.kind === 107 /* ThisKeyword */) {
+                var declaration = prop && prop.valueDeclaration;
+                if (declaration && isInstancePropertyWithoutInitializer(declaration)) {
+                    var flowContainer = getControlFlowContainer(node);
+                    if (flowContainer.kind === 166 /* Constructor */ && flowContainer.parent === declaration.parent && !(declaration.flags & 8388608 /* Ambient */)) {
+                        assumeUninitialized = true;
                     }
                 }
             }
-            return false;
-        }
-        function isTypeAssignableToKind(source, kind, strict) {
-            if (source.flags & kind) {
-                return true;
+            else if (strictNullChecks && prop && prop.valueDeclaration &&
+                ts.isPropertyAccessExpression(prop.valueDeclaration) &&
+                ts.getAssignmentDeclarationPropertyAccessKind(prop.valueDeclaration) &&
+                getControlFlowContainer(node) === getControlFlowContainer(prop.valueDeclaration)) {
+                assumeUninitialized = true;
             }
-            if (strict && source.flags & (3 /* AnyOrUnknown */ | 16384 /* Void */ | 32768 /* Undefined */ | 65536 /* Null */)) {
-                return false;
+            var flowType = getFlowTypeOfReference(node, propType, assumeUninitialized ? getOptionalType(propType) : propType);
+            if (assumeUninitialized && !(getFalsyFlags(propType) & 32768 /* Undefined */) && getFalsyFlags(flowType) & 32768 /* Undefined */) {
+                error(errorNode, ts.Diagnostics.Property_0_is_used_before_being_assigned, symbolToString(prop)); // TODO: GH#18217
+                // Return the declared type to reduce follow-on errors
+                return propType;
             }
-            return !!(kind & 296 /* NumberLike */) && isTypeAssignableTo(source, numberType) ||
-                !!(kind & 2112 /* BigIntLike */) && isTypeAssignableTo(source, bigintType) ||
-                !!(kind & 132 /* StringLike */) && isTypeAssignableTo(source, stringType) ||
-                !!(kind & 528 /* BooleanLike */) && isTypeAssignableTo(source, booleanType) ||
-                !!(kind & 16384 /* Void */) && isTypeAssignableTo(source, voidType) ||
-                !!(kind & 131072 /* Never */) && isTypeAssignableTo(source, neverType) ||
-                !!(kind & 65536 /* Null */) && isTypeAssignableTo(source, nullType) ||
-                !!(kind & 32768 /* Undefined */) && isTypeAssignableTo(source, undefinedType) ||
-                !!(kind & 4096 /* ESSymbol */) && isTypeAssignableTo(source, esSymbolType) ||
-                !!(kind & 67108864 /* NonPrimitive */) && isTypeAssignableTo(source, nonPrimitiveType);
-        }
-        function allTypesAssignableToKind(source, kind, strict) {
-            return source.flags & 1048576 /* Union */ ?
-                ts.every(source.types, function (subType) { return allTypesAssignableToKind(subType, kind, strict); }) :
-                isTypeAssignableToKind(source, kind, strict);
-        }
-        function isConstEnumObjectType(type) {
-            return !!(ts.getObjectFlags(type) & 16 /* Anonymous */) && !!type.symbol && isConstEnumSymbol(type.symbol);
-        }
-        function isConstEnumSymbol(symbol) {
-            return (symbol.flags & 128 /* ConstEnum */) !== 0;
+            return assignmentKind ? getBaseTypeOfLiteralType(flowType) : flowType;
         }
-        function checkInstanceOfExpression(left, right, leftType, rightType) {
-            if (leftType === silentNeverType || rightType === silentNeverType) {
-                return silentNeverType;
-            }
-            // TypeScript 1.0 spec (April 2014): 4.15.4
-            // The instanceof operator requires the left operand to be of type Any, an object type, or a type parameter type,
-            // and the right operand to be of type Any, a subtype of the 'Function' interface type, or have a call or construct signature.
-            // The result is always of the Boolean primitive type.
-            // NOTE: do not raise error if leftType is unknown as related error was already reported
-            if (!isTypeAny(leftType) &&
-                allTypesAssignableToKind(leftType, 131068 /* Primitive */)) {
-                error(left, ts.Diagnostics.The_left_hand_side_of_an_instanceof_expression_must_be_of_type_any_an_object_type_or_a_type_parameter);
-            }
-            // NOTE: do not raise error if right is unknown as related error was already reported
-            if (!(isTypeAny(rightType) || typeHasCallOrConstructSignatures(rightType) || isTypeSubtypeOf(rightType, globalFunctionType))) {
-                error(right, ts.Diagnostics.The_right_hand_side_of_an_instanceof_expression_must_be_of_type_any_or_of_a_type_assignable_to_the_Function_interface_type);
+        function checkPropertyNotUsedBeforeDeclaration(prop, node, right) {
+            var valueDeclaration = prop.valueDeclaration;
+            if (!valueDeclaration || ts.getSourceFileOfNode(node).isDeclarationFile) {
+                return;
             }
-            return booleanType;
-        }
-        function checkInExpression(left, right, leftType, rightType) {
-            if (leftType === silentNeverType || rightType === silentNeverType) {
-                return silentNeverType;
+            var diagnosticMessage;
+            var declarationName = ts.idText(right);
+            if (isInPropertyInitializer(node)
+                && !(ts.isAccessExpression(node) && ts.isAccessExpression(node.expression))
+                && !isBlockScopedNameDeclaredBeforeUse(valueDeclaration, right)
+                && !isPropertyDeclaredInAncestorClass(prop)) {
+                diagnosticMessage = error(right, ts.Diagnostics.Property_0_is_used_before_its_initialization, declarationName);
             }
-            leftType = checkNonNullType(leftType, left);
-            rightType = checkNonNullType(rightType, right);
-            // TypeScript 1.0 spec (April 2014): 4.15.5
-            // The in operator requires the left operand to be of type Any, the String primitive type, or the Number primitive type,
-            // and the right operand to be of type Any, an object type, or a type parameter type.
-            // The result is always of the Boolean primitive type.
-            if (!(isTypeComparableTo(leftType, stringType) || isTypeAssignableToKind(leftType, 296 /* NumberLike */ | 12288 /* ESSymbolLike */))) {
-                error(left, ts.Diagnostics.The_left_hand_side_of_an_in_expression_must_be_of_type_any_string_number_or_symbol);
+            else if (valueDeclaration.kind === 252 /* ClassDeclaration */ &&
+                node.parent.kind !== 173 /* TypeReference */ &&
+                !(valueDeclaration.flags & 8388608 /* Ambient */) &&
+                !isBlockScopedNameDeclaredBeforeUse(valueDeclaration, right)) {
+                diagnosticMessage = error(right, ts.Diagnostics.Class_0_used_before_its_declaration, declarationName);
             }
-            if (!allTypesAssignableToKind(rightType, 67108864 /* NonPrimitive */ | 58982400 /* InstantiableNonPrimitive */)) {
-                error(right, ts.Diagnostics.The_right_hand_side_of_an_in_expression_must_be_of_type_any_an_object_type_or_a_type_parameter);
+            if (diagnosticMessage) {
+                ts.addRelatedInfo(diagnosticMessage, ts.createDiagnosticForNode(valueDeclaration, ts.Diagnostics._0_is_declared_here, declarationName));
             }
-            return booleanType;
         }
-        function checkObjectLiteralAssignment(node, sourceType, rightIsThis) {
-            var properties = node.properties;
-            if (strictNullChecks && properties.length === 0) {
-                return checkNonNullType(sourceType, node);
+        function isInPropertyInitializer(node) {
+            return !!ts.findAncestor(node, function (node) {
+                switch (node.kind) {
+                    case 163 /* PropertyDeclaration */:
+                        return true;
+                    case 288 /* PropertyAssignment */:
+                    case 165 /* MethodDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 290 /* SpreadAssignment */:
+                    case 158 /* ComputedPropertyName */:
+                    case 228 /* TemplateSpan */:
+                    case 283 /* JsxExpression */:
+                    case 280 /* JsxAttribute */:
+                    case 281 /* JsxAttributes */:
+                    case 282 /* JsxSpreadAttribute */:
+                    case 275 /* JsxOpeningElement */:
+                    case 223 /* ExpressionWithTypeArguments */:
+                    case 286 /* HeritageClause */:
+                        return false;
+                    default:
+                        return ts.isExpressionNode(node) ? false : "quit";
+                }
+            });
+        }
+        /**
+         * It's possible that "prop.valueDeclaration" is a local declaration, but the property was also declared in a superclass.
+         * In that case we won't consider it used before its declaration, because it gets its value from the superclass' declaration.
+         */
+        function isPropertyDeclaredInAncestorClass(prop) {
+            if (!(prop.parent.flags & 32 /* Class */)) {
+                return false;
             }
-            for (var i = 0; i < properties.length; i++) {
-                checkObjectLiteralDestructuringPropertyAssignment(node, sourceType, i, properties, rightIsThis);
+            var classType = getTypeOfSymbol(prop.parent);
+            while (true) {
+                classType = classType.symbol && getSuperClass(classType);
+                if (!classType) {
+                    return false;
+                }
+                var superProperty = getPropertyOfType(classType, prop.escapedName);
+                if (superProperty && superProperty.valueDeclaration) {
+                    return true;
+                }
             }
-            return sourceType;
         }
-        /** Note: If property cannot be a SpreadAssignment, then allProperties does not need to be provided */
-        function checkObjectLiteralDestructuringPropertyAssignment(node, objectLiteralType, propertyIndex, allProperties, rightIsThis) {
-            if (rightIsThis === void 0) { rightIsThis = false; }
-            var properties = node.properties;
-            var property = properties[propertyIndex];
-            if (property.kind === 281 /* PropertyAssignment */ || property.kind === 282 /* ShorthandPropertyAssignment */) {
-                var name = property.name;
-                var exprType = getLiteralTypeFromPropertyName(name);
-                if (isTypeUsableAsPropertyName(exprType)) {
-                    var text = getPropertyNameFromType(exprType);
-                    var prop = getPropertyOfType(objectLiteralType, text);
-                    if (prop) {
-                        markPropertyAsReferenced(prop, property, rightIsThis);
-                        checkPropertyAccessibility(property, /*isSuper*/ false, objectLiteralType, prop);
+        function getSuperClass(classType) {
+            var x = getBaseTypes(classType);
+            if (x.length === 0) {
+                return undefined;
+            }
+            return getIntersectionType(x);
+        }
+        function reportNonexistentProperty(propNode, containingType) {
+            var errorInfo;
+            var relatedInfo;
+            if (!ts.isPrivateIdentifier(propNode) && containingType.flags & 1048576 /* Union */ && !(containingType.flags & 131068 /* Primitive */)) {
+                for (var _i = 0, _a = containingType.types; _i < _a.length; _i++) {
+                    var subtype = _a[_i];
+                    if (!getPropertyOfType(subtype, propNode.escapedText) && !getIndexInfoOfType(subtype, 0 /* String */)) {
+                        errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.declarationNameToString(propNode), typeToString(subtype));
+                        break;
                     }
                 }
-                var elementType = getIndexedAccessType(objectLiteralType, exprType, name);
-                var type = getFlowTypeOfDestructuring(property, elementType);
-                return checkDestructuringAssignment(property.kind === 282 /* ShorthandPropertyAssignment */ ? property : property.initializer, type);
             }
-            else if (property.kind === 283 /* SpreadAssignment */) {
-                if (propertyIndex < properties.length - 1) {
-                    error(property, ts.Diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern);
+            if (typeHasStaticProperty(propNode.escapedText, containingType)) {
+                errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_is_a_static_member_of_type_1, ts.declarationNameToString(propNode), typeToString(containingType));
+            }
+            else {
+                var promisedType = getPromisedTypeOfPromise(containingType);
+                if (promisedType && getPropertyOfType(promisedType, propNode.escapedText)) {
+                    errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1, ts.declarationNameToString(propNode), typeToString(containingType));
+                    relatedInfo = ts.createDiagnosticForNode(propNode, ts.Diagnostics.Did_you_forget_to_use_await);
                 }
                 else {
-                    if (languageVersion < 99 /* ESNext */) {
-                        checkExternalEmitHelpers(property, 4 /* Rest */);
+                    var missingProperty = ts.declarationNameToString(propNode);
+                    var container = typeToString(containingType);
+                    var libSuggestion = getSuggestedLibForNonExistentProperty(missingProperty, containingType);
+                    if (libSuggestion !== undefined) {
+                        errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1_Do_you_need_to_change_your_target_library_Try_changing_the_lib_compiler_option_to_2_or_later, missingProperty, container, libSuggestion);
                     }
-                    var nonRestNames = [];
-                    if (allProperties) {
-                        for (var _i = 0, allProperties_1 = allProperties; _i < allProperties_1.length; _i++) {
-                            var otherProperty = allProperties_1[_i];
-                            if (!ts.isSpreadAssignment(otherProperty)) {
-                                nonRestNames.push(otherProperty.name);
-                            }
+                    else {
+                        var suggestion = getSuggestedSymbolForNonexistentProperty(propNode, containingType);
+                        if (suggestion !== undefined) {
+                            var suggestedName = ts.symbolName(suggestion);
+                            errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Property_0_does_not_exist_on_type_1_Did_you_mean_2, missingProperty, container, suggestedName);
+                            relatedInfo = suggestion.valueDeclaration && ts.createDiagnosticForNode(suggestion.valueDeclaration, ts.Diagnostics._0_is_declared_here, suggestedName);
+                        }
+                        else {
+                            errorInfo = ts.chainDiagnosticMessages(elaborateNeverIntersection(errorInfo, containingType), ts.Diagnostics.Property_0_does_not_exist_on_type_1, missingProperty, container);
                         }
                     }
-                    var type = getRestType(objectLiteralType, nonRestNames, objectLiteralType.symbol);
-                    checkGrammarForDisallowedTrailingComma(allProperties, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
-                    return checkDestructuringAssignment(property.expression, type);
                 }
             }
-            else {
-                error(property, ts.Diagnostics.Property_assignment_expected);
+            var resultDiagnostic = ts.createDiagnosticForNodeFromMessageChain(propNode, errorInfo);
+            if (relatedInfo) {
+                ts.addRelatedInfo(resultDiagnostic, relatedInfo);
             }
+            diagnostics.add(resultDiagnostic);
         }
-        function checkArrayLiteralAssignment(node, sourceType, checkMode) {
-            var elements = node.elements;
-            if (languageVersion < 2 /* ES2015 */ && compilerOptions.downlevelIteration) {
-                checkExternalEmitHelpers(node, 512 /* Read */);
+        function typeHasStaticProperty(propName, containingType) {
+            var prop = containingType.symbol && getPropertyOfType(getTypeOfSymbol(containingType.symbol), propName);
+            return prop !== undefined && prop.valueDeclaration && ts.hasSyntacticModifier(prop.valueDeclaration, 32 /* Static */);
+        }
+        function getSuggestedLibForNonExistentName(name) {
+            var missingName = diagnosticName(name);
+            var allFeatures = ts.getScriptTargetFeatures();
+            var libTargets = ts.getOwnKeys(allFeatures);
+            for (var _i = 0, libTargets_1 = libTargets; _i < libTargets_1.length; _i++) {
+                var libTarget = libTargets_1[_i];
+                var containingTypes = ts.getOwnKeys(allFeatures[libTarget]);
+                if (containingTypes !== undefined && ts.contains(containingTypes, missingName)) {
+                    return libTarget;
+                }
             }
-            // This elementType will be used if the specific property corresponding to this index is not
-            // present (aka the tuple element property). This call also checks that the parentType is in
-            // fact an iterable or array (depending on target language).
-            var elementType = checkIteratedTypeOrElementType(65 /* Destructuring */, sourceType, undefinedType, node) || errorType;
-            for (var i = 0; i < elements.length; i++) {
-                checkArrayLiteralDestructuringElementAssignment(node, sourceType, i, elementType, checkMode);
+        }
+        function getSuggestedLibForNonExistentProperty(missingProperty, containingType) {
+            var container = getApparentType(containingType).symbol;
+            if (!container) {
+                return undefined;
+            }
+            var allFeatures = ts.getScriptTargetFeatures();
+            var libTargets = ts.getOwnKeys(allFeatures);
+            for (var _i = 0, libTargets_2 = libTargets; _i < libTargets_2.length; _i++) {
+                var libTarget = libTargets_2[_i];
+                var featuresOfLib = allFeatures[libTarget];
+                var featuresOfContainingType = featuresOfLib[ts.symbolName(container)];
+                if (featuresOfContainingType !== undefined && ts.contains(featuresOfContainingType, missingProperty)) {
+                    return libTarget;
+                }
             }
-            return sourceType;
         }
-        function checkArrayLiteralDestructuringElementAssignment(node, sourceType, elementIndex, elementType, checkMode) {
-            var elements = node.elements;
-            var element = elements[elementIndex];
-            if (element.kind !== 215 /* OmittedExpression */) {
-                if (element.kind !== 213 /* SpreadElement */) {
-                    var indexType = getLiteralType(elementIndex);
-                    if (isArrayLikeType(sourceType)) {
-                        // We create a synthetic expression so that getIndexedAccessType doesn't get confused
-                        // when the element is a SyntaxKind.ElementAccessExpression.
-                        var accessFlags = hasDefaultValue(element) ? 8 /* NoTupleBoundsCheck */ : 0;
-                        var elementType_2 = getIndexedAccessTypeOrUndefined(sourceType, indexType, createSyntheticExpression(element, indexType), accessFlags) || errorType;
-                        var assignedType = hasDefaultValue(element) ? getTypeWithFacts(elementType_2, 524288 /* NEUndefined */) : elementType_2;
-                        var type = getFlowTypeOfDestructuring(element, assignedType);
-                        return checkDestructuringAssignment(element, type, checkMode);
-                    }
-                    return checkDestructuringAssignment(element, elementType, checkMode);
+        function getSuggestedSymbolForNonexistentProperty(name, containingType) {
+            return getSpellingSuggestionForName(ts.isString(name) ? name : ts.idText(name), getPropertiesOfType(containingType), 111551 /* Value */);
+        }
+        function getSuggestedSymbolForNonexistentJSXAttribute(name, containingType) {
+            var strName = ts.isString(name) ? name : ts.idText(name);
+            var properties = getPropertiesOfType(containingType);
+            var jsxSpecific = strName === "for" ? ts.find(properties, function (x) { return ts.symbolName(x) === "htmlFor"; })
+                : strName === "class" ? ts.find(properties, function (x) { return ts.symbolName(x) === "className"; })
+                    : undefined;
+            return jsxSpecific !== null && jsxSpecific !== void 0 ? jsxSpecific : getSpellingSuggestionForName(strName, properties, 111551 /* Value */);
+        }
+        function getSuggestionForNonexistentProperty(name, containingType) {
+            var suggestion = getSuggestedSymbolForNonexistentProperty(name, containingType);
+            return suggestion && ts.symbolName(suggestion);
+        }
+        function getSuggestedSymbolForNonexistentSymbol(location, outerName, meaning) {
+            ts.Debug.assert(outerName !== undefined, "outername should always be defined");
+            var result = resolveNameHelper(location, outerName, meaning, /*nameNotFoundMessage*/ undefined, outerName, /*isUse*/ false, /*excludeGlobals*/ false, function (symbols, name, meaning) {
+                ts.Debug.assertEqual(outerName, name, "name should equal outerName");
+                var symbol = getSymbol(symbols, name, meaning);
+                // Sometimes the symbol is found when location is a return type of a function: `typeof x` and `x` is declared in the body of the function
+                // So the table *contains* `x` but `x` isn't actually in scope.
+                // However, resolveNameHelper will continue and call this callback again, so we'll eventually get a correct suggestion.
+                return symbol || getSpellingSuggestionForName(ts.unescapeLeadingUnderscores(name), ts.arrayFrom(symbols.values()), meaning);
+            });
+            return result;
+        }
+        function getSuggestionForNonexistentSymbol(location, outerName, meaning) {
+            var symbolResult = getSuggestedSymbolForNonexistentSymbol(location, outerName, meaning);
+            return symbolResult && ts.symbolName(symbolResult);
+        }
+        function getSuggestedSymbolForNonexistentModule(name, targetModule) {
+            return targetModule.exports && getSpellingSuggestionForName(ts.idText(name), getExportsOfModuleAsArray(targetModule), 2623475 /* ModuleMember */);
+        }
+        function getSuggestionForNonexistentExport(name, targetModule) {
+            var suggestion = getSuggestedSymbolForNonexistentModule(name, targetModule);
+            return suggestion && ts.symbolName(suggestion);
+        }
+        function getSuggestionForNonexistentIndexSignature(objectType, expr, keyedType) {
+            // check if object type has setter or getter
+            function hasProp(name) {
+                var prop = getPropertyOfObjectType(objectType, name);
+                if (prop) {
+                    var s = getSingleCallSignature(getTypeOfSymbol(prop));
+                    return !!s && getMinArgumentCount(s) >= 1 && isTypeAssignableTo(keyedType, getTypeAtPosition(s, 0));
                 }
-                if (elementIndex < elements.length - 1) {
-                    error(element, ts.Diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern);
+                return false;
+            }
+            ;
+            var suggestedMethod = ts.isAssignmentTarget(expr) ? "set" : "get";
+            if (!hasProp(suggestedMethod)) {
+                return undefined;
+            }
+            var suggestion = ts.tryGetPropertyAccessOrIdentifierToString(expr.expression);
+            if (suggestion === undefined) {
+                suggestion = suggestedMethod;
+            }
+            else {
+                suggestion += "." + suggestedMethod;
+            }
+            return suggestion;
+        }
+        /**
+         * Given a name and a list of symbols whose names are *not* equal to the name, return a spelling suggestion if there is one that is close enough.
+         * Names less than length 3 only check for case-insensitive equality, not levenshtein distance.
+         *
+         * If there is a candidate that's the same except for case, return that.
+         * If there is a candidate that's within one edit of the name, return that.
+         * Otherwise, return the candidate with the smallest Levenshtein distance,
+         *    except for candidates:
+         *      * With no name
+         *      * Whose meaning doesn't match the `meaning` parameter.
+         *      * Whose length differs from the target name by more than 0.34 of the length of the name.
+         *      * Whose levenshtein distance is more than 0.4 of the length of the name
+         *        (0.4 allows 1 substitution/transposition for every 5 characters,
+         *         and 1 insertion/deletion at 3 characters)
+         */
+        function getSpellingSuggestionForName(name, symbols, meaning) {
+            return ts.getSpellingSuggestion(name, symbols, getCandidateName);
+            function getCandidateName(candidate) {
+                var candidateName = ts.symbolName(candidate);
+                if (ts.startsWith(candidateName, "\"")) {
+                    return undefined;
                 }
-                else {
-                    var restExpression = element.expression;
-                    if (restExpression.kind === 209 /* BinaryExpression */ && restExpression.operatorToken.kind === 62 /* EqualsToken */) {
-                        error(restExpression.operatorToken, ts.Diagnostics.A_rest_element_cannot_have_an_initializer);
-                    }
-                    else {
-                        checkGrammarForDisallowedTrailingComma(node.elements, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
-                        var type = everyType(sourceType, isTupleType) ?
-                            mapType(sourceType, function (t) { return sliceTupleType(t, elementIndex); }) :
-                            createArrayType(elementType);
-                        return checkDestructuringAssignment(restExpression, type, checkMode);
+                if (candidate.flags & meaning) {
+                    return candidateName;
+                }
+                if (candidate.flags & 2097152 /* Alias */) {
+                    var alias = tryResolveAlias(candidate);
+                    if (alias && alias.flags & meaning) {
+                        return candidateName;
                     }
                 }
+                return undefined;
             }
-            return undefined;
         }
-        function checkDestructuringAssignment(exprOrAssignment, sourceType, checkMode, rightIsThis) {
-            var target;
-            if (exprOrAssignment.kind === 282 /* ShorthandPropertyAssignment */) {
-                var prop = exprOrAssignment;
-                if (prop.objectAssignmentInitializer) {
-                    // In strict null checking mode, if a default value of a non-undefined type is specified, remove
-                    // undefined from the final type.
-                    if (strictNullChecks &&
-                        !(getFalsyFlags(checkExpression(prop.objectAssignmentInitializer)) & 32768 /* Undefined */)) {
-                        sourceType = getTypeWithFacts(sourceType, 524288 /* NEUndefined */);
-                    }
-                    checkBinaryLikeExpression(prop.name, prop.equalsToken, prop.objectAssignmentInitializer, checkMode);
-                }
-                target = exprOrAssignment.name;
+        function markPropertyAsReferenced(prop, nodeForCheckWriteOnly, isThisAccess) {
+            var valueDeclaration = prop && (prop.flags & 106500 /* ClassMember */) && prop.valueDeclaration;
+            if (!valueDeclaration) {
+                return;
             }
-            else {
-                target = exprOrAssignment;
+            var hasPrivateModifier = ts.hasEffectiveModifier(valueDeclaration, 8 /* Private */);
+            var hasPrivateIdentifier = ts.isNamedDeclaration(prop.valueDeclaration) && ts.isPrivateIdentifier(prop.valueDeclaration.name);
+            if (!hasPrivateModifier && !hasPrivateIdentifier) {
+                return;
             }
-            if (target.kind === 209 /* BinaryExpression */ && target.operatorToken.kind === 62 /* EqualsToken */) {
-                checkBinaryExpression(target, checkMode);
-                target = target.left;
+            if (nodeForCheckWriteOnly && ts.isWriteOnlyAccess(nodeForCheckWriteOnly) && !(prop.flags & 65536 /* SetAccessor */)) {
+                return;
             }
-            if (target.kind === 193 /* ObjectLiteralExpression */) {
-                return checkObjectLiteralAssignment(target, sourceType, rightIsThis);
+            if (isThisAccess) {
+                // Find any FunctionLikeDeclaration because those create a new 'this' binding. But this should only matter for methods (or getters/setters).
+                var containingMethod = ts.findAncestor(nodeForCheckWriteOnly, ts.isFunctionLikeDeclaration);
+                if (containingMethod && containingMethod.symbol === prop) {
+                    return;
+                }
             }
-            if (target.kind === 192 /* ArrayLiteralExpression */) {
-                return checkArrayLiteralAssignment(target, sourceType, checkMode);
+            (ts.getCheckFlags(prop) & 1 /* Instantiated */ ? getSymbolLinks(prop).target : prop).isReferenced = 67108863 /* All */;
+        }
+        function isValidPropertyAccess(node, propertyName) {
+            switch (node.kind) {
+                case 201 /* PropertyAccessExpression */:
+                    return isValidPropertyAccessWithType(node, node.expression.kind === 105 /* SuperKeyword */, propertyName, getWidenedType(checkExpression(node.expression)));
+                case 157 /* QualifiedName */:
+                    return isValidPropertyAccessWithType(node, /*isSuper*/ false, propertyName, getWidenedType(checkExpression(node.left)));
+                case 195 /* ImportType */:
+                    return isValidPropertyAccessWithType(node, /*isSuper*/ false, propertyName, getTypeFromTypeNode(node));
             }
-            return checkReferenceAssignment(target, sourceType, checkMode);
         }
-        function checkReferenceAssignment(target, sourceType, checkMode) {
-            var targetType = checkExpression(target, checkMode);
-            var error = target.parent.kind === 283 /* SpreadAssignment */ ?
-                ts.Diagnostics.The_target_of_an_object_rest_assignment_must_be_a_variable_or_a_property_access :
-                ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_must_be_a_variable_or_a_property_access;
-            var optionalError = target.parent.kind === 283 /* SpreadAssignment */ ?
-                ts.Diagnostics.The_target_of_an_object_rest_assignment_may_not_be_an_optional_property_access :
-                ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_may_not_be_an_optional_property_access;
-            if (checkReferenceExpression(target, error, optionalError)) {
-                checkTypeAssignableToAndOptionallyElaborate(sourceType, targetType, target, target);
+        function isValidPropertyAccessForCompletions(node, type, property) {
+            return isValidPropertyAccessWithType(node, node.kind === 201 /* PropertyAccessExpression */ && node.expression.kind === 105 /* SuperKeyword */, property.escapedName, type);
+            // Previously we validated the 'this' type of methods but this adversely affected performance. See #31377 for more context.
+        }
+        function isValidPropertyAccessWithType(node, isSuper, propertyName, type) {
+            if (type === errorType || isTypeAny(type)) {
+                return true;
             }
-            if (ts.isPrivateIdentifierPropertyAccessExpression(target)) {
-                checkExternalEmitHelpers(target.parent, 524288 /* ClassPrivateFieldSet */);
+            var prop = getPropertyOfType(type, propertyName);
+            if (prop) {
+                if (ts.isPropertyAccessExpression(node) && prop.valueDeclaration && ts.isPrivateIdentifierPropertyDeclaration(prop.valueDeclaration)) {
+                    var declClass_1 = ts.getContainingClass(prop.valueDeclaration);
+                    return !ts.isOptionalChain(node) && !!ts.findAncestor(node, function (parent) { return parent === declClass_1; });
+                }
+                return checkPropertyAccessibility(node, isSuper, type, prop);
             }
-            return sourceType;
+            // In js files properties of unions are allowed in completion
+            return ts.isInJSFile(node) && (type.flags & 1048576 /* Union */) !== 0 && type.types.some(function (elementType) { return isValidPropertyAccessWithType(node, isSuper, propertyName, elementType); });
         }
         /**
-         * This is a *shallow* check: An expression is side-effect-free if the
-         * evaluation of the expression *itself* cannot produce side effects.
-         * For example, x++ / 3 is side-effect free because the / operator
-         * does not have side effects.
-         * The intent is to "smell test" an expression for correctness in positions where
-         * its value is discarded (e.g. the left side of the comma operator).
+         * Return the symbol of the for-in variable declared or referenced by the given for-in statement.
          */
-        function isSideEffectFree(node) {
-            node = ts.skipParentheses(node);
-            switch (node.kind) {
-                case 75 /* Identifier */:
-                case 10 /* StringLiteral */:
-                case 13 /* RegularExpressionLiteral */:
-                case 198 /* TaggedTemplateExpression */:
-                case 211 /* TemplateExpression */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 8 /* NumericLiteral */:
-                case 9 /* BigIntLiteral */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
-                case 100 /* NullKeyword */:
-                case 146 /* UndefinedKeyword */:
-                case 201 /* FunctionExpression */:
-                case 214 /* ClassExpression */:
-                case 202 /* ArrowFunction */:
-                case 192 /* ArrayLiteralExpression */:
-                case 193 /* ObjectLiteralExpression */:
-                case 204 /* TypeOfExpression */:
-                case 218 /* NonNullExpression */:
-                case 267 /* JsxSelfClosingElement */:
-                case 266 /* JsxElement */:
-                    return true;
-                case 210 /* ConditionalExpression */:
-                    return isSideEffectFree(node.whenTrue) &&
-                        isSideEffectFree(node.whenFalse);
-                case 209 /* BinaryExpression */:
-                    if (ts.isAssignmentOperator(node.operatorToken.kind)) {
-                        return false;
-                    }
-                    return isSideEffectFree(node.left) &&
-                        isSideEffectFree(node.right);
-                case 207 /* PrefixUnaryExpression */:
-                case 208 /* PostfixUnaryExpression */:
-                    // Unary operators ~, !, +, and - have no side effects.
-                    // The rest do.
-                    switch (node.operator) {
-                        case 53 /* ExclamationToken */:
-                        case 39 /* PlusToken */:
-                        case 40 /* MinusToken */:
-                        case 54 /* TildeToken */:
-                            return true;
-                    }
-                    return false;
-                // Some forms listed here for clarity
-                case 205 /* VoidExpression */: // Explicit opt-out
-                case 199 /* TypeAssertionExpression */: // Not SEF, but can produce useful type warnings
-                case 217 /* AsExpression */: // Not SEF, but can produce useful type warnings
-                default:
-                    return false;
+        function getForInVariableSymbol(node) {
+            var initializer = node.initializer;
+            if (initializer.kind === 250 /* VariableDeclarationList */) {
+                var variable = initializer.declarations[0];
+                if (variable && !ts.isBindingPattern(variable.name)) {
+                    return getSymbolOfNode(variable);
+                }
+            }
+            else if (initializer.kind === 78 /* Identifier */) {
+                return getResolvedSymbol(initializer);
             }
+            return undefined;
         }
-        function isTypeEqualityComparableTo(source, target) {
-            return (target.flags & 98304 /* Nullable */) !== 0 || isTypeComparableTo(source, target);
+        /**
+         * Return true if the given type is considered to have numeric property names.
+         */
+        function hasNumericPropertyNames(type) {
+            return getIndexTypeOfType(type, 1 /* Number */) && !getIndexTypeOfType(type, 0 /* String */);
         }
-        var CheckBinaryExpressionState;
-        (function (CheckBinaryExpressionState) {
-            CheckBinaryExpressionState[CheckBinaryExpressionState["MaybeCheckLeft"] = 0] = "MaybeCheckLeft";
-            CheckBinaryExpressionState[CheckBinaryExpressionState["CheckRight"] = 1] = "CheckRight";
-            CheckBinaryExpressionState[CheckBinaryExpressionState["FinishCheck"] = 2] = "FinishCheck";
-        })(CheckBinaryExpressionState || (CheckBinaryExpressionState = {}));
-        function checkBinaryExpression(node, checkMode) {
-            var workStacks = {
-                expr: [node],
-                state: [0 /* MaybeCheckLeft */],
-                leftType: [undefined]
-            };
-            var stackIndex = 0;
-            var lastResult;
-            while (stackIndex >= 0) {
-                node = workStacks.expr[stackIndex];
-                switch (workStacks.state[stackIndex]) {
-                    case 0 /* MaybeCheckLeft */: {
-                        if (ts.isInJSFile(node) && ts.getAssignedExpandoInitializer(node)) {
-                            finishInvocation(checkExpression(node.right, checkMode));
-                            break;
-                        }
-                        checkGrammarNullishCoalesceWithLogicalExpression(node);
-                        var operator = node.operatorToken.kind;
-                        if (operator === 62 /* EqualsToken */ && (node.left.kind === 193 /* ObjectLiteralExpression */ || node.left.kind === 192 /* ArrayLiteralExpression */)) {
-                            finishInvocation(checkDestructuringAssignment(node.left, checkExpression(node.right, checkMode), checkMode, node.right.kind === 104 /* ThisKeyword */));
-                            break;
-                        }
-                        advanceState(1 /* CheckRight */);
-                        maybeCheckExpression(node.left);
-                        break;
-                    }
-                    case 1 /* CheckRight */: {
-                        var leftType = lastResult;
-                        workStacks.leftType[stackIndex] = leftType;
-                        var operator = node.operatorToken.kind;
-                        if (operator === 55 /* AmpersandAmpersandToken */ || operator === 56 /* BarBarToken */ || operator === 60 /* QuestionQuestionToken */) {
-                            checkTruthinessOfType(leftType, node.left);
+        /**
+         * Return true if given node is an expression consisting of an identifier (possibly parenthesized)
+         * that references a for-in variable for an object with numeric property names.
+         */
+        function isForInVariableForNumericPropertyNames(expr) {
+            var e = ts.skipParentheses(expr);
+            if (e.kind === 78 /* Identifier */) {
+                var symbol = getResolvedSymbol(e);
+                if (symbol.flags & 3 /* Variable */) {
+                    var child = expr;
+                    var node = expr.parent;
+                    while (node) {
+                        if (node.kind === 238 /* ForInStatement */ &&
+                            child === node.statement &&
+                            getForInVariableSymbol(node) === symbol &&
+                            hasNumericPropertyNames(getTypeOfExpression(node.expression))) {
+                            return true;
                         }
-                        advanceState(2 /* FinishCheck */);
-                        maybeCheckExpression(node.right);
-                        break;
-                    }
-                    case 2 /* FinishCheck */: {
-                        var leftType = workStacks.leftType[stackIndex];
-                        var rightType = lastResult;
-                        finishInvocation(checkBinaryLikeExpressionWorker(node.left, node.operatorToken, node.right, leftType, rightType, node));
-                        break;
+                        child = node;
+                        node = node.parent;
                     }
-                    default: return ts.Debug.fail("Invalid state " + workStacks.state[stackIndex] + " for checkBinaryExpression");
                 }
             }
-            return lastResult;
-            function finishInvocation(result) {
-                lastResult = result;
-                stackIndex--;
-            }
-            /**
-             * Note that `advanceState` sets the _current_ head state, and that `maybeCheckExpression` potentially pushes on a new
-             * head state; so `advanceState` must be called before any `maybeCheckExpression` during a state's execution.
-             */
-            function advanceState(nextState) {
-                workStacks.state[stackIndex] = nextState;
+            return false;
+        }
+        function checkIndexedAccess(node) {
+            return node.flags & 32 /* OptionalChain */ ? checkElementAccessChain(node) :
+                checkElementAccessExpression(node, checkNonNullExpression(node.expression));
+        }
+        function checkElementAccessChain(node) {
+            var exprType = checkExpression(node.expression);
+            var nonOptionalType = getOptionalExpressionType(exprType, node.expression);
+            return propagateOptionalTypeMarker(checkElementAccessExpression(node, checkNonNullType(nonOptionalType, node.expression)), node, nonOptionalType !== exprType);
+        }
+        function checkElementAccessExpression(node, exprType) {
+            var objectType = ts.getAssignmentTargetKind(node) !== 0 /* None */ || isMethodAccessForCall(node) ? getWidenedType(exprType) : exprType;
+            var indexExpression = node.argumentExpression;
+            var indexType = checkExpression(indexExpression);
+            if (objectType === errorType || objectType === silentNeverType) {
+                return objectType;
             }
-            function maybeCheckExpression(node) {
-                if (ts.isBinaryExpression(node)) {
-                    stackIndex++;
-                    workStacks.expr[stackIndex] = node;
-                    workStacks.state[stackIndex] = 0 /* MaybeCheckLeft */;
-                    workStacks.leftType[stackIndex] = undefined;
-                }
-                else {
-                    lastResult = checkExpression(node, checkMode);
-                }
+            if (isConstEnumObjectType(objectType) && !ts.isStringLiteralLike(indexExpression)) {
+                error(indexExpression, ts.Diagnostics.A_const_enum_member_can_only_be_accessed_using_a_string_literal);
+                return errorType;
             }
+            var effectiveIndexType = isForInVariableForNumericPropertyNames(indexExpression) ? numberType : indexType;
+            var accessFlags = ts.isAssignmentTarget(node) ?
+                2 /* Writing */ | (isGenericObjectType(objectType) && !isThisTypeParameter(objectType) ? 1 /* NoIndexSignatures */ : 0) :
+                0 /* None */;
+            var indexedAccessType = getIndexedAccessTypeOrUndefined(objectType, effectiveIndexType, /*noUncheckedIndexedAccessCandidate*/ undefined, node, accessFlags | 16 /* ExpressionPosition */) || errorType;
+            return checkIndexedAccessIndexType(getFlowTypeOfAccessExpression(node, indexedAccessType.symbol, indexedAccessType, indexExpression), node);
         }
-        function checkGrammarNullishCoalesceWithLogicalExpression(node) {
-            var left = node.left, operatorToken = node.operatorToken, right = node.right;
-            if (operatorToken.kind === 60 /* QuestionQuestionToken */) {
-                if (ts.isBinaryExpression(left) && (left.operatorToken.kind === 56 /* BarBarToken */ || left.operatorToken.kind === 55 /* AmpersandAmpersandToken */)) {
-                    grammarErrorOnNode(left, ts.Diagnostics._0_and_1_operations_cannot_be_mixed_without_parentheses, ts.tokenToString(left.operatorToken.kind), ts.tokenToString(operatorToken.kind));
+        function checkThatExpressionIsProperSymbolReference(expression, expressionType, reportError) {
+            if (expressionType === errorType) {
+                // There is already an error, so no need to report one.
+                return false;
+            }
+            if (!ts.isWellKnownSymbolSyntactically(expression)) {
+                return false;
+            }
+            // Make sure the property type is the primitive symbol type
+            if ((expressionType.flags & 12288 /* ESSymbolLike */) === 0) {
+                if (reportError) {
+                    error(expression, ts.Diagnostics.A_computed_property_name_of_the_form_0_must_be_of_type_symbol, ts.getTextOfNode(expression));
                 }
-                if (ts.isBinaryExpression(right) && (right.operatorToken.kind === 56 /* BarBarToken */ || right.operatorToken.kind === 55 /* AmpersandAmpersandToken */)) {
-                    grammarErrorOnNode(right, ts.Diagnostics._0_and_1_operations_cannot_be_mixed_without_parentheses, ts.tokenToString(right.operatorToken.kind), ts.tokenToString(operatorToken.kind));
+                return false;
+            }
+            // The name is Symbol.<someName>, so make sure Symbol actually resolves to the
+            // global Symbol object
+            var leftHandSide = expression.expression;
+            var leftHandSideSymbol = getResolvedSymbol(leftHandSide);
+            if (!leftHandSideSymbol) {
+                return false;
+            }
+            var globalESSymbol = getGlobalESSymbolConstructorSymbol(/*reportErrors*/ true);
+            if (!globalESSymbol) {
+                // Already errored when we tried to look up the symbol
+                return false;
+            }
+            if (leftHandSideSymbol !== globalESSymbol) {
+                if (reportError) {
+                    error(leftHandSide, ts.Diagnostics.Symbol_reference_does_not_refer_to_the_global_Symbol_constructor_object);
                 }
+                return false;
             }
+            return true;
         }
-        // Note that this and `checkBinaryExpression` above should behave mostly the same, except this elides some
-        // expression-wide checks and does not use a work stack to fold nested binary expressions into the same callstack frame
-        function checkBinaryLikeExpression(left, operatorToken, right, checkMode, errorNode) {
-            var operator = operatorToken.kind;
-            if (operator === 62 /* EqualsToken */ && (left.kind === 193 /* ObjectLiteralExpression */ || left.kind === 192 /* ArrayLiteralExpression */)) {
-                return checkDestructuringAssignment(left, checkExpression(right, checkMode), checkMode, right.kind === 104 /* ThisKeyword */);
+        function callLikeExpressionMayHaveTypeArguments(node) {
+            return ts.isCallOrNewExpression(node) || ts.isTaggedTemplateExpression(node) || ts.isJsxOpeningLikeElement(node);
+        }
+        function resolveUntypedCall(node) {
+            if (callLikeExpressionMayHaveTypeArguments(node)) {
+                // Check type arguments even though we will give an error that untyped calls may not accept type arguments.
+                // This gets us diagnostics for the type arguments and marks them as referenced.
+                ts.forEach(node.typeArguments, checkSourceElement);
             }
-            var leftType;
-            if (operator === 55 /* AmpersandAmpersandToken */ || operator === 56 /* BarBarToken */ || operator === 60 /* QuestionQuestionToken */) {
-                leftType = checkTruthinessExpression(left, checkMode);
+            if (node.kind === 205 /* TaggedTemplateExpression */) {
+                checkExpression(node.template);
             }
-            else {
-                leftType = checkExpression(left, checkMode);
+            else if (ts.isJsxOpeningLikeElement(node)) {
+                checkExpression(node.attributes);
             }
-            var rightType = checkExpression(right, checkMode);
-            return checkBinaryLikeExpressionWorker(left, operatorToken, right, leftType, rightType, errorNode);
+            else if (node.kind !== 161 /* Decorator */) {
+                ts.forEach(node.arguments, function (argument) {
+                    checkExpression(argument);
+                });
+            }
+            return anySignature;
         }
-        function checkBinaryLikeExpressionWorker(left, operatorToken, right, leftType, rightType, errorNode) {
-            var operator = operatorToken.kind;
-            switch (operator) {
-                case 41 /* AsteriskToken */:
-                case 42 /* AsteriskAsteriskToken */:
-                case 65 /* AsteriskEqualsToken */:
-                case 66 /* AsteriskAsteriskEqualsToken */:
-                case 43 /* SlashToken */:
-                case 67 /* SlashEqualsToken */:
-                case 44 /* PercentToken */:
-                case 68 /* PercentEqualsToken */:
-                case 40 /* MinusToken */:
-                case 64 /* MinusEqualsToken */:
-                case 47 /* LessThanLessThanToken */:
-                case 69 /* LessThanLessThanEqualsToken */:
-                case 48 /* GreaterThanGreaterThanToken */:
-                case 70 /* GreaterThanGreaterThanEqualsToken */:
-                case 49 /* GreaterThanGreaterThanGreaterThanToken */:
-                case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
-                case 51 /* BarToken */:
-                case 73 /* BarEqualsToken */:
-                case 52 /* CaretToken */:
-                case 74 /* CaretEqualsToken */:
-                case 50 /* AmpersandToken */:
-                case 72 /* AmpersandEqualsToken */:
-                    if (leftType === silentNeverType || rightType === silentNeverType) {
-                        return silentNeverType;
-                    }
-                    leftType = checkNonNullType(leftType, left);
-                    rightType = checkNonNullType(rightType, right);
-                    var suggestedOperator = void 0;
-                    // if a user tries to apply a bitwise operator to 2 boolean operands
-                    // try and return them a helpful suggestion
-                    if ((leftType.flags & 528 /* BooleanLike */) &&
-                        (rightType.flags & 528 /* BooleanLike */) &&
-                        (suggestedOperator = getSuggestedBooleanOperator(operatorToken.kind)) !== undefined) {
-                        error(errorNode || operatorToken, ts.Diagnostics.The_0_operator_is_not_allowed_for_boolean_types_Consider_using_1_instead, ts.tokenToString(operatorToken.kind), ts.tokenToString(suggestedOperator));
-                        return numberType;
-                    }
-                    else {
-                        // otherwise just check each operand separately and report errors as normal
-                        var leftOk = checkArithmeticOperandType(left, leftType, ts.Diagnostics.The_left_hand_side_of_an_arithmetic_operation_must_be_of_type_any_number_bigint_or_an_enum_type, /*isAwaitValid*/ true);
-                        var rightOk = checkArithmeticOperandType(right, rightType, ts.Diagnostics.The_right_hand_side_of_an_arithmetic_operation_must_be_of_type_any_number_bigint_or_an_enum_type, /*isAwaitValid*/ true);
-                        var resultType_1;
-                        // If both are any or unknown, allow operation; assume it will resolve to number
-                        if ((isTypeAssignableToKind(leftType, 3 /* AnyOrUnknown */) && isTypeAssignableToKind(rightType, 3 /* AnyOrUnknown */)) ||
-                            // Or, if neither could be bigint, implicit coercion results in a number result
-                            !(maybeTypeOfKind(leftType, 2112 /* BigIntLike */) || maybeTypeOfKind(rightType, 2112 /* BigIntLike */))) {
-                            resultType_1 = numberType;
-                        }
-                        // At least one is assignable to bigint, so check that both are
-                        else if (bothAreBigIntLike(leftType, rightType)) {
-                            switch (operator) {
-                                case 49 /* GreaterThanGreaterThanGreaterThanToken */:
-                                case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
-                                    reportOperatorError();
-                            }
-                            resultType_1 = bigintType;
-                        }
-                        // Exactly one of leftType/rightType is assignable to bigint
-                        else {
-                            reportOperatorError(bothAreBigIntLike);
-                            resultType_1 = errorType;
-                        }
-                        if (leftOk && rightOk) {
-                            checkAssignmentOperator(resultType_1);
-                        }
-                        return resultType_1;
-                    }
-                case 39 /* PlusToken */:
-                case 63 /* PlusEqualsToken */:
-                    if (leftType === silentNeverType || rightType === silentNeverType) {
-                        return silentNeverType;
-                    }
-                    if (!isTypeAssignableToKind(leftType, 132 /* StringLike */) && !isTypeAssignableToKind(rightType, 132 /* StringLike */)) {
-                        leftType = checkNonNullType(leftType, left);
-                        rightType = checkNonNullType(rightType, right);
-                    }
-                    var resultType = void 0;
-                    if (isTypeAssignableToKind(leftType, 296 /* NumberLike */, /*strict*/ true) && isTypeAssignableToKind(rightType, 296 /* NumberLike */, /*strict*/ true)) {
-                        // Operands of an enum type are treated as having the primitive type Number.
-                        // If both operands are of the Number primitive type, the result is of the Number primitive type.
-                        resultType = numberType;
-                    }
-                    else if (isTypeAssignableToKind(leftType, 2112 /* BigIntLike */, /*strict*/ true) && isTypeAssignableToKind(rightType, 2112 /* BigIntLike */, /*strict*/ true)) {
-                        // If both operands are of the BigInt primitive type, the result is of the BigInt primitive type.
-                        resultType = bigintType;
-                    }
-                    else if (isTypeAssignableToKind(leftType, 132 /* StringLike */, /*strict*/ true) || isTypeAssignableToKind(rightType, 132 /* StringLike */, /*strict*/ true)) {
-                        // If one or both operands are of the String primitive type, the result is of the String primitive type.
-                        resultType = stringType;
-                    }
-                    else if (isTypeAny(leftType) || isTypeAny(rightType)) {
-                        // Otherwise, the result is of type Any.
-                        // NOTE: unknown type here denotes error type. Old compiler treated this case as any type so do we.
-                        resultType = leftType === errorType || rightType === errorType ? errorType : anyType;
-                    }
-                    // Symbols are not allowed at all in arithmetic expressions
-                    if (resultType && !checkForDisallowedESSymbolOperand(operator)) {
-                        return resultType;
-                    }
-                    if (!resultType) {
-                        // Types that have a reasonably good chance of being a valid operand type.
-                        // If both types have an awaited type of one of these, we'll assume the user
-                        // might be missing an await without doing an exhaustive check that inserting
-                        // await(s) will actually be a completely valid binary expression.
-                        var closeEnoughKind_1 = 296 /* NumberLike */ | 2112 /* BigIntLike */ | 132 /* StringLike */ | 3 /* AnyOrUnknown */;
-                        reportOperatorError(function (left, right) {
-                            return isTypeAssignableToKind(left, closeEnoughKind_1) &&
-                                isTypeAssignableToKind(right, closeEnoughKind_1);
-                        });
-                        return anyType;
-                    }
-                    if (operator === 63 /* PlusEqualsToken */) {
-                        checkAssignmentOperator(resultType);
-                    }
-                    return resultType;
-                case 29 /* LessThanToken */:
-                case 31 /* GreaterThanToken */:
-                case 32 /* LessThanEqualsToken */:
-                case 33 /* GreaterThanEqualsToken */:
-                    if (checkForDisallowedESSymbolOperand(operator)) {
-                        leftType = getBaseTypeOfLiteralType(checkNonNullType(leftType, left));
-                        rightType = getBaseTypeOfLiteralType(checkNonNullType(rightType, right));
-                        reportOperatorErrorUnless(function (left, right) {
-                            return isTypeComparableTo(left, right) || isTypeComparableTo(right, left) || (isTypeAssignableTo(left, numberOrBigIntType) && isTypeAssignableTo(right, numberOrBigIntType));
-                        });
-                    }
-                    return booleanType;
-                case 34 /* EqualsEqualsToken */:
-                case 35 /* ExclamationEqualsToken */:
-                case 36 /* EqualsEqualsEqualsToken */:
-                case 37 /* ExclamationEqualsEqualsToken */:
-                    reportOperatorErrorUnless(function (left, right) { return isTypeEqualityComparableTo(left, right) || isTypeEqualityComparableTo(right, left); });
-                    return booleanType;
-                case 98 /* InstanceOfKeyword */:
-                    return checkInstanceOfExpression(left, right, leftType, rightType);
-                case 97 /* InKeyword */:
-                    return checkInExpression(left, right, leftType, rightType);
-                case 55 /* AmpersandAmpersandToken */:
-                    return getTypeFacts(leftType) & 4194304 /* Truthy */ ?
-                        getUnionType([extractDefinitelyFalsyTypes(strictNullChecks ? leftType : getBaseTypeOfLiteralType(rightType)), rightType]) :
-                        leftType;
-                case 56 /* BarBarToken */:
-                    return getTypeFacts(leftType) & 8388608 /* Falsy */ ?
-                        getUnionType([removeDefinitelyFalsyTypes(leftType), rightType], 2 /* Subtype */) :
-                        leftType;
-                case 60 /* QuestionQuestionToken */:
-                    return getTypeFacts(leftType) & 262144 /* EQUndefinedOrNull */ ?
-                        getUnionType([getNonNullableType(leftType), rightType], 2 /* Subtype */) :
-                        leftType;
-                case 62 /* EqualsToken */:
-                    var declKind = ts.isBinaryExpression(left.parent) ? ts.getAssignmentDeclarationKind(left.parent) : 0 /* None */;
-                    checkAssignmentDeclaration(declKind, rightType);
-                    if (isAssignmentDeclaration(declKind)) {
-                        if (!(rightType.flags & 524288 /* Object */) ||
-                            declKind !== 2 /* ModuleExports */ &&
-                                declKind !== 6 /* Prototype */ &&
-                                !isEmptyObjectType(rightType) &&
-                                !isFunctionObjectType(rightType) &&
-                                !(ts.getObjectFlags(rightType) & 1 /* Class */)) {
-                            // don't check assignability of module.exports=, C.prototype=, or expando types because they will necessarily be incomplete
-                            checkAssignmentOperator(rightType);
-                        }
-                        return leftType;
+        function resolveErrorCall(node) {
+            resolveUntypedCall(node);
+            return unknownSignature;
+        }
+        // Re-order candidate signatures into the result array. Assumes the result array to be empty.
+        // The candidate list orders groups in reverse, but within a group signatures are kept in declaration order
+        // A nit here is that we reorder only signatures that belong to the same symbol,
+        // so order how inherited signatures are processed is still preserved.
+        // interface A { (x: string): void }
+        // interface B extends A { (x: 'foo'): string }
+        // const b: B;
+        // b('foo') // <- here overloads should be processed as [(x:'foo'): string, (x: string): void]
+        function reorderCandidates(signatures, result, callChainFlags) {
+            var lastParent;
+            var lastSymbol;
+            var cutoffIndex = 0;
+            var index;
+            var specializedIndex = -1;
+            var spliceIndex;
+            ts.Debug.assert(!result.length);
+            for (var _i = 0, signatures_7 = signatures; _i < signatures_7.length; _i++) {
+                var signature = signatures_7[_i];
+                var symbol = signature.declaration && getSymbolOfNode(signature.declaration);
+                var parent = signature.declaration && signature.declaration.parent;
+                if (!lastSymbol || symbol === lastSymbol) {
+                    if (lastParent && parent === lastParent) {
+                        index = index + 1;
                     }
                     else {
-                        checkAssignmentOperator(rightType);
-                        return getRegularTypeOfObjectLiteral(rightType);
-                    }
-                case 27 /* CommaToken */:
-                    if (!compilerOptions.allowUnreachableCode && isSideEffectFree(left) && !isEvalNode(right)) {
-                        error(left, ts.Diagnostics.Left_side_of_comma_operator_is_unused_and_has_no_side_effects);
-                    }
-                    return rightType;
-                default:
-                    return ts.Debug.fail();
-            }
-            function bothAreBigIntLike(left, right) {
-                return isTypeAssignableToKind(left, 2112 /* BigIntLike */) && isTypeAssignableToKind(right, 2112 /* BigIntLike */);
-            }
-            function checkAssignmentDeclaration(kind, rightType) {
-                if (kind === 2 /* ModuleExports */) {
-                    for (var _i = 0, _a = getPropertiesOfObjectType(rightType); _i < _a.length; _i++) {
-                        var prop = _a[_i];
-                        var propType = getTypeOfSymbol(prop);
-                        if (propType.symbol && propType.symbol.flags & 32 /* Class */) {
-                            var name = prop.escapedName;
-                            var symbol = resolveName(prop.valueDeclaration, name, 788968 /* Type */, undefined, name, /*isUse*/ false);
-                            if (symbol && symbol.declarations.some(ts.isJSDocTypedefTag)) {
-                                addDuplicateDeclarationErrorsForSymbols(symbol, ts.Diagnostics.Duplicate_identifier_0, ts.unescapeLeadingUnderscores(name), prop);
-                                addDuplicateDeclarationErrorsForSymbols(prop, ts.Diagnostics.Duplicate_identifier_0, ts.unescapeLeadingUnderscores(name), symbol);
-                            }
-                        }
+                        lastParent = parent;
+                        index = cutoffIndex;
                     }
                 }
-            }
-            function isEvalNode(node) {
-                return node.kind === 75 /* Identifier */ && node.escapedText === "eval";
-            }
-            // Return true if there was no error, false if there was an error.
-            function checkForDisallowedESSymbolOperand(operator) {
-                var offendingSymbolOperand = maybeTypeOfKind(leftType, 12288 /* ESSymbolLike */) ? left :
-                    maybeTypeOfKind(rightType, 12288 /* ESSymbolLike */) ? right :
-                        undefined;
-                if (offendingSymbolOperand) {
-                    error(offendingSymbolOperand, ts.Diagnostics.The_0_operator_cannot_be_applied_to_type_symbol, ts.tokenToString(operator));
-                    return false;
+                else {
+                    // current declaration belongs to a different symbol
+                    // set cutoffIndex so re-orderings in the future won't change result set from 0 to cutoffIndex
+                    index = cutoffIndex = result.length;
+                    lastParent = parent;
                 }
-                return true;
-            }
-            function getSuggestedBooleanOperator(operator) {
-                switch (operator) {
-                    case 51 /* BarToken */:
-                    case 73 /* BarEqualsToken */:
-                        return 56 /* BarBarToken */;
-                    case 52 /* CaretToken */:
-                    case 74 /* CaretEqualsToken */:
-                        return 37 /* ExclamationEqualsEqualsToken */;
-                    case 50 /* AmpersandToken */:
-                    case 72 /* AmpersandEqualsToken */:
-                        return 55 /* AmpersandAmpersandToken */;
-                    default:
-                        return undefined;
+                lastSymbol = symbol;
+                // specialized signatures always need to be placed before non-specialized signatures regardless
+                // of the cutoff position; see GH#1133
+                if (signatureHasLiteralTypes(signature)) {
+                    specializedIndex++;
+                    spliceIndex = specializedIndex;
+                    // The cutoff index always needs to be greater than or equal to the specialized signature index
+                    // in order to prevent non-specialized signatures from being added before a specialized
+                    // signature.
+                    cutoffIndex++;
                 }
-            }
-            function checkAssignmentOperator(valueType) {
-                if (produceDiagnostics && ts.isAssignmentOperator(operator)) {
-                    // TypeScript 1.0 spec (April 2014): 4.17
-                    // An assignment of the form
-                    //    VarExpr = ValueExpr
-                    // requires VarExpr to be classified as a reference
-                    // A compound assignment furthermore requires VarExpr to be classified as a reference (section 4.1)
-                    // and the type of the non-compound operation to be assignable to the type of VarExpr.
-                    if (checkReferenceExpression(left, ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_may_not_be_an_optional_property_access)
-                        && (!ts.isIdentifier(left) || ts.unescapeLeadingUnderscores(left.escapedText) !== "exports")) {
-                        // to avoid cascading errors check assignability only if 'isReference' check succeeded and no errors were reported
-                        checkTypeAssignableToAndOptionallyElaborate(valueType, leftType, left, right);
-                    }
+                else {
+                    spliceIndex = index;
                 }
+                result.splice(spliceIndex, 0, callChainFlags ? getOptionalCallSignature(signature, callChainFlags) : signature);
             }
-            function isAssignmentDeclaration(kind) {
-                switch (kind) {
-                    case 2 /* ModuleExports */:
-                        return true;
-                    case 1 /* ExportsProperty */:
-                    case 5 /* Property */:
-                    case 6 /* Prototype */:
-                    case 3 /* PrototypeProperty */:
-                    case 4 /* ThisProperty */:
-                        var symbol = getSymbolOfNode(left);
-                        var init = ts.getAssignedExpandoInitializer(right);
-                        return init && ts.isObjectLiteralExpression(init) &&
-                            symbol && ts.hasEntries(symbol.exports);
-                    default:
-                        return false;
+        }
+        function isSpreadArgument(arg) {
+            return !!arg && (arg.kind === 220 /* SpreadElement */ || arg.kind === 227 /* SyntheticExpression */ && arg.isSpread);
+        }
+        function getSpreadArgumentIndex(args) {
+            return ts.findIndex(args, isSpreadArgument);
+        }
+        function acceptsVoid(t) {
+            return !!(t.flags & 16384 /* Void */);
+        }
+        function acceptsVoidUndefinedUnknownOrAny(t) {
+            return !!(t.flags & (16384 /* Void */ | 32768 /* Undefined */ | 2 /* Unknown */ | 1 /* Any */));
+        }
+        function hasCorrectArity(node, args, signature, signatureHelpTrailingComma) {
+            if (signatureHelpTrailingComma === void 0) { signatureHelpTrailingComma = false; }
+            var argCount;
+            var callIsIncomplete = false; // In incomplete call we want to be lenient when we have too few arguments
+            var effectiveParameterCount = getParameterCount(signature);
+            var effectiveMinimumArguments = getMinArgumentCount(signature);
+            if (node.kind === 205 /* TaggedTemplateExpression */) {
+                argCount = args.length;
+                if (node.template.kind === 218 /* TemplateExpression */) {
+                    // If a tagged template expression lacks a tail literal, the call is incomplete.
+                    // Specifically, a template only can end in a TemplateTail or a Missing literal.
+                    var lastSpan = ts.last(node.template.templateSpans); // we should always have at least one span.
+                    callIsIncomplete = ts.nodeIsMissing(lastSpan.literal) || !!lastSpan.literal.isUnterminated;
+                }
+                else {
+                    // If the template didn't end in a backtick, or its beginning occurred right prior to EOF,
+                    // then this might actually turn out to be a TemplateHead in the future;
+                    // so we consider the call to be incomplete.
+                    var templateLiteral = node.template;
+                    ts.Debug.assert(templateLiteral.kind === 14 /* NoSubstitutionTemplateLiteral */);
+                    callIsIncomplete = !!templateLiteral.isUnterminated;
                 }
             }
-            /**
-             * Returns true if an error is reported
-             */
-            function reportOperatorErrorUnless(typesAreCompatible) {
-                if (!typesAreCompatible(leftType, rightType)) {
-                    reportOperatorError(typesAreCompatible);
+            else if (node.kind === 161 /* Decorator */) {
+                argCount = getDecoratorArgumentCount(node, signature);
+            }
+            else if (ts.isJsxOpeningLikeElement(node)) {
+                callIsIncomplete = node.attributes.end === node.end;
+                if (callIsIncomplete) {
                     return true;
                 }
-                return false;
+                argCount = effectiveMinimumArguments === 0 ? args.length : 1;
+                effectiveParameterCount = args.length === 0 ? effectiveParameterCount : 1; // class may have argumentless ctor functions - still resolve ctor and compare vs props member type
+                effectiveMinimumArguments = Math.min(effectiveMinimumArguments, 1); // sfc may specify context argument - handled by framework and not typechecked
             }
-            function reportOperatorError(isRelated) {
-                var _a;
-                var wouldWorkWithAwait = false;
-                var errNode = errorNode || operatorToken;
-                if (isRelated) {
-                    var awaitedLeftType = getAwaitedType(leftType);
-                    var awaitedRightType = getAwaitedType(rightType);
-                    wouldWorkWithAwait = !(awaitedLeftType === leftType && awaitedRightType === rightType)
-                        && !!(awaitedLeftType && awaitedRightType)
-                        && isRelated(awaitedLeftType, awaitedRightType);
-                }
-                var effectiveLeft = leftType;
-                var effectiveRight = rightType;
-                if (!wouldWorkWithAwait && isRelated) {
-                    _a = getBaseTypesIfUnrelated(leftType, rightType, isRelated), effectiveLeft = _a[0], effectiveRight = _a[1];
-                }
-                var _b = getTypeNamesForErrorDisplay(effectiveLeft, effectiveRight), leftStr = _b[0], rightStr = _b[1];
-                if (!tryGiveBetterPrimaryError(errNode, wouldWorkWithAwait, leftStr, rightStr)) {
-                    errorAndMaybeSuggestAwait(errNode, wouldWorkWithAwait, ts.Diagnostics.Operator_0_cannot_be_applied_to_types_1_and_2, ts.tokenToString(operatorToken.kind), leftStr, rightStr);
-                }
+            else if (!node.arguments) {
+                // This only happens when we have something of the form: 'new C'
+                ts.Debug.assert(node.kind === 204 /* NewExpression */);
+                return getMinArgumentCount(signature) === 0;
             }
-            function tryGiveBetterPrimaryError(errNode, maybeMissingAwait, leftStr, rightStr) {
-                var typeName;
-                switch (operatorToken.kind) {
-                    case 36 /* EqualsEqualsEqualsToken */:
-                    case 34 /* EqualsEqualsToken */:
-                        typeName = "false";
-                        break;
-                    case 37 /* ExclamationEqualsEqualsToken */:
-                    case 35 /* ExclamationEqualsToken */:
-                        typeName = "true";
+            else {
+                argCount = signatureHelpTrailingComma ? args.length + 1 : args.length;
+                // If we are missing the close parenthesis, the call is incomplete.
+                callIsIncomplete = node.arguments.end === node.end;
+                // If a spread argument is present, check that it corresponds to a rest parameter or at least that it's in the valid range.
+                var spreadArgIndex = getSpreadArgumentIndex(args);
+                if (spreadArgIndex >= 0) {
+                    return spreadArgIndex >= getMinArgumentCount(signature) && (hasEffectiveRestParameter(signature) || spreadArgIndex < getParameterCount(signature));
                 }
-                if (typeName) {
-                    return errorAndMaybeSuggestAwait(errNode, maybeMissingAwait, ts.Diagnostics.This_condition_will_always_return_0_since_the_types_1_and_2_have_no_overlap, typeName, leftStr, rightStr);
+            }
+            // Too many arguments implies incorrect arity.
+            if (!hasEffectiveRestParameter(signature) && argCount > effectiveParameterCount) {
+                return false;
+            }
+            // If the call is incomplete, we should skip the lower bound check.
+            // JSX signatures can have extra parameters provided by the library which we don't check
+            if (callIsIncomplete || argCount >= effectiveMinimumArguments) {
+                return true;
+            }
+            for (var i = argCount; i < effectiveMinimumArguments; i++) {
+                var type = getTypeAtPosition(signature, i);
+                if (filterType(type, ts.isInJSFile(node) && !strictNullChecks ? acceptsVoidUndefinedUnknownOrAny : acceptsVoid).flags & 131072 /* Never */) {
+                    return false;
                 }
-                return undefined;
             }
+            return true;
         }
-        function getBaseTypesIfUnrelated(leftType, rightType, isRelated) {
-            var effectiveLeft = leftType;
-            var effectiveRight = rightType;
-            var leftBase = getBaseTypeOfLiteralType(leftType);
-            var rightBase = getBaseTypeOfLiteralType(rightType);
-            if (!isRelated(leftBase, rightBase)) {
-                effectiveLeft = leftBase;
-                effectiveRight = rightBase;
-            }
-            return [effectiveLeft, effectiveRight];
+        function hasCorrectTypeArgumentArity(signature, typeArguments) {
+            // If the user supplied type arguments, but the number of type arguments does not match
+            // the declared number of type parameters, the call has an incorrect arity.
+            var numTypeParameters = ts.length(signature.typeParameters);
+            var minTypeArgumentCount = getMinTypeArgumentCount(signature.typeParameters);
+            return !ts.some(typeArguments) ||
+                (typeArguments.length >= minTypeArgumentCount && typeArguments.length <= numTypeParameters);
         }
-        function checkYieldExpression(node) {
-            // Grammar checking
-            if (produceDiagnostics) {
-                if (!(node.flags & 8192 /* YieldContext */)) {
-                    grammarErrorOnFirstToken(node, ts.Diagnostics.A_yield_expression_is_only_allowed_in_a_generator_body);
-                }
-                if (isInParameterInitializerBeforeContainingFunction(node)) {
-                    error(node, ts.Diagnostics.yield_expressions_cannot_be_used_in_a_parameter_initializer);
+        // If type has a single call signature and no other members, return that signature. Otherwise, return undefined.
+        function getSingleCallSignature(type) {
+            return getSingleSignature(type, 0 /* Call */, /*allowMembers*/ false);
+        }
+        function getSingleCallOrConstructSignature(type) {
+            return getSingleSignature(type, 0 /* Call */, /*allowMembers*/ false) ||
+                getSingleSignature(type, 1 /* Construct */, /*allowMembers*/ false);
+        }
+        function getSingleSignature(type, kind, allowMembers) {
+            if (type.flags & 524288 /* Object */) {
+                var resolved = resolveStructuredTypeMembers(type);
+                if (allowMembers || resolved.properties.length === 0 && !resolved.stringIndexInfo && !resolved.numberIndexInfo) {
+                    if (kind === 0 /* Call */ && resolved.callSignatures.length === 1 && resolved.constructSignatures.length === 0) {
+                        return resolved.callSignatures[0];
+                    }
+                    if (kind === 1 /* Construct */ && resolved.constructSignatures.length === 1 && resolved.callSignatures.length === 0) {
+                        return resolved.constructSignatures[0];
+                    }
                 }
             }
-            var func = ts.getContainingFunction(node);
-            if (!func)
-                return anyType;
-            var functionFlags = ts.getFunctionFlags(func);
-            if (!(functionFlags & 1 /* Generator */)) {
-                // If the user's code is syntactically correct, the func should always have a star. After all, we are in a yield context.
-                return anyType;
+            return undefined;
+        }
+        // Instantiate a generic signature in the context of a non-generic signature (section 3.8.5 in TypeScript spec)
+        function instantiateSignatureInContextOf(signature, contextualSignature, inferenceContext, compareTypes) {
+            var context = createInferenceContext(signature.typeParameters, signature, 0 /* None */, compareTypes);
+            // We clone the inferenceContext to avoid fixing. For example, when the source signature is <T>(x: T) => T[] and
+            // the contextual signature is (...args: A) => B, we want to infer the element type of A's constraint (say 'any')
+            // for T but leave it possible to later infer '[any]' back to A.
+            var restType = getEffectiveRestType(contextualSignature);
+            var mapper = inferenceContext && (restType && restType.flags & 262144 /* TypeParameter */ ? inferenceContext.nonFixingMapper : inferenceContext.mapper);
+            var sourceSignature = mapper ? instantiateSignature(contextualSignature, mapper) : contextualSignature;
+            applyToParameterTypes(sourceSignature, signature, function (source, target) {
+                // Type parameters from outer context referenced by source type are fixed by instantiation of the source type
+                inferTypes(context.inferences, source, target);
+            });
+            if (!inferenceContext) {
+                applyToReturnTypes(contextualSignature, signature, function (source, target) {
+                    inferTypes(context.inferences, source, target, 64 /* ReturnType */);
+                });
             }
-            var isAsync = (functionFlags & 2 /* Async */) !== 0;
-            if (node.asteriskToken) {
-                // Async generator functions prior to ESNext require the __await, __asyncDelegator,
-                // and __asyncValues helpers
-                if (isAsync && languageVersion < 99 /* ESNext */) {
-                    checkExternalEmitHelpers(node, 53248 /* AsyncDelegatorIncludes */);
-                }
-                // Generator functions prior to ES2015 require the __values helper
-                if (!isAsync && languageVersion < 2 /* ES2015 */ && compilerOptions.downlevelIteration) {
-                    checkExternalEmitHelpers(node, 256 /* Values */);
+            return getSignatureInstantiation(signature, getInferredTypes(context), ts.isInJSFile(contextualSignature.declaration));
+        }
+        function inferJsxTypeArguments(node, signature, checkMode, context) {
+            var paramType = getEffectiveFirstArgumentForJsxSignature(signature, node);
+            var checkAttrType = checkExpressionWithContextualType(node.attributes, paramType, context, checkMode);
+            inferTypes(context.inferences, checkAttrType, paramType);
+            return getInferredTypes(context);
+        }
+        function inferTypeArguments(node, signature, args, checkMode, context) {
+            if (ts.isJsxOpeningLikeElement(node)) {
+                return inferJsxTypeArguments(node, signature, checkMode, context);
+            }
+            // If a contextual type is available, infer from that type to the return type of the call expression. For
+            // example, given a 'function wrap<T, U>(cb: (x: T) => U): (x: T) => U' and a call expression
+            // 'let f: (x: string) => number = wrap(s => s.length)', we infer from the declared type of 'f' to the
+            // return type of 'wrap'.
+            if (node.kind !== 161 /* Decorator */) {
+                var contextualType = getContextualType(node, ts.every(signature.typeParameters, function (p) { return !!getDefaultFromTypeParameter(p); }) ? 8 /* SkipBindingPatterns */ : 0 /* None */);
+                if (contextualType) {
+                    // We clone the inference context to avoid disturbing a resolution in progress for an
+                    // outer call expression. Effectively we just want a snapshot of whatever has been
+                    // inferred for any outer call expression so far.
+                    var outerContext = getInferenceContext(node);
+                    var outerMapper = getMapperFromContext(cloneInferenceContext(outerContext, 1 /* NoDefault */));
+                    var instantiatedType = instantiateType(contextualType, outerMapper);
+                    // If the contextual type is a generic function type with a single call signature, we
+                    // instantiate the type with its own type parameters and type arguments. This ensures that
+                    // the type parameters are not erased to type any during type inference such that they can
+                    // be inferred as actual types from the contextual type. For example:
+                    //   declare function arrayMap<T, U>(f: (x: T) => U): (a: T[]) => U[];
+                    //   const boxElements: <A>(a: A[]) => { value: A }[] = arrayMap(value => ({ value }));
+                    // Above, the type of the 'value' parameter is inferred to be 'A'.
+                    var contextualSignature = getSingleCallSignature(instantiatedType);
+                    var inferenceSourceType = contextualSignature && contextualSignature.typeParameters ?
+                        getOrCreateTypeFromSignature(getSignatureInstantiationWithoutFillingInTypeArguments(contextualSignature, contextualSignature.typeParameters)) :
+                        instantiatedType;
+                    var inferenceTargetType = getReturnTypeOfSignature(signature);
+                    // Inferences made from return types have lower priority than all other inferences.
+                    inferTypes(context.inferences, inferenceSourceType, inferenceTargetType, 64 /* ReturnType */);
+                    // Create a type mapper for instantiating generic contextual types using the inferences made
+                    // from the return type. We need a separate inference pass here because (a) instantiation of
+                    // the source type uses the outer context's return mapper (which excludes inferences made from
+                    // outer arguments), and (b) we don't want any further inferences going into this context.
+                    var returnContext = createInferenceContext(signature.typeParameters, signature, context.flags);
+                    var returnSourceType = instantiateType(contextualType, outerContext && outerContext.returnMapper);
+                    inferTypes(returnContext.inferences, returnSourceType, inferenceTargetType);
+                    context.returnMapper = ts.some(returnContext.inferences, hasInferenceCandidates) ? getMapperFromContext(cloneInferredPartOfContext(returnContext)) : undefined;
                 }
             }
-            // There is no point in doing an assignability check if the function
-            // has no explicit return type because the return type is directly computed
-            // from the yield expressions.
-            var returnType = getReturnTypeFromAnnotation(func);
-            var iterationTypes = returnType && getIterationTypesOfGeneratorFunctionReturnType(returnType, isAsync);
-            var signatureYieldType = iterationTypes && iterationTypes.yieldType || anyType;
-            var signatureNextType = iterationTypes && iterationTypes.nextType || anyType;
-            var resolvedSignatureNextType = isAsync ? getAwaitedType(signatureNextType) || anyType : signatureNextType;
-            var yieldExpressionType = node.expression ? checkExpression(node.expression) : undefinedWideningType;
-            var yieldedType = getYieldedTypeOfYieldExpression(node, yieldExpressionType, resolvedSignatureNextType, isAsync);
-            if (returnType && yieldedType) {
-                checkTypeAssignableToAndOptionallyElaborate(yieldedType, signatureYieldType, node.expression || node, node.expression);
-            }
-            if (node.asteriskToken) {
-                var use = isAsync ? 19 /* AsyncYieldStar */ : 17 /* YieldStar */;
-                return getIterationTypeOfIterable(use, 1 /* Return */, yieldExpressionType, node.expression)
-                    || anyType;
+            var restType = getNonArrayRestType(signature);
+            var argCount = restType ? Math.min(getParameterCount(signature) - 1, args.length) : args.length;
+            if (restType && restType.flags & 262144 /* TypeParameter */) {
+                var info = ts.find(context.inferences, function (info) { return info.typeParameter === restType; });
+                if (info) {
+                    info.impliedArity = ts.findIndex(args, isSpreadArgument, argCount) < 0 ? args.length - argCount : undefined;
+                }
             }
-            else if (returnType) {
-                return getIterationTypeOfGeneratorFunctionReturnType(2 /* Next */, returnType, isAsync)
-                    || anyType;
+            var thisType = getThisTypeOfSignature(signature);
+            if (thisType) {
+                var thisArgumentNode = getThisArgumentOfCall(node);
+                var thisArgumentType = thisArgumentNode ? checkExpression(thisArgumentNode) : voidType;
+                inferTypes(context.inferences, thisArgumentType, thisType);
             }
-            return getContextualIterationType(2 /* Next */, func) || anyType;
-        }
-        function checkConditionalExpression(node, checkMode) {
-            var type = checkTruthinessExpression(node.condition);
-            checkTestingKnownTruthyCallableType(node.condition, node.whenTrue, type);
-            var type1 = checkExpression(node.whenTrue, checkMode);
-            var type2 = checkExpression(node.whenFalse, checkMode);
-            return getUnionType([type1, type2], 2 /* Subtype */);
-        }
-        function checkTemplateExpression(node) {
-            // We just want to check each expressions, but we are unconcerned with
-            // the type of each expression, as any value may be coerced into a string.
-            // It is worth asking whether this is what we really want though.
-            // A place where we actually *are* concerned with the expressions' types are
-            // in tagged templates.
-            ts.forEach(node.templateSpans, function (templateSpan) {
-                if (maybeTypeOfKind(checkExpression(templateSpan.expression), 12288 /* ESSymbolLike */)) {
-                    error(templateSpan.expression, ts.Diagnostics.Implicit_conversion_of_a_symbol_to_a_string_will_fail_at_runtime_Consider_wrapping_this_expression_in_String);
+            for (var i = 0; i < argCount; i++) {
+                var arg = args[i];
+                if (arg.kind !== 222 /* OmittedExpression */) {
+                    var paramType = getTypeAtPosition(signature, i);
+                    var argType = checkExpressionWithContextualType(arg, paramType, context, checkMode);
+                    inferTypes(context.inferences, argType, paramType);
                 }
-            });
-            return stringType;
-        }
-        function getContextNode(node) {
-            if (node.kind === 274 /* JsxAttributes */ && !ts.isJsxSelfClosingElement(node.parent)) {
-                return node.parent.parent; // Needs to be the root JsxElement, so it encompasses the attributes _and_ the children (which are essentially part of the attributes)
-            }
-            return node;
-        }
-        function checkExpressionWithContextualType(node, contextualType, inferenceContext, checkMode) {
-            var context = getContextNode(node);
-            var saveContextualType = context.contextualType;
-            var saveInferenceContext = context.inferenceContext;
-            try {
-                context.contextualType = contextualType;
-                context.inferenceContext = inferenceContext;
-                var type = checkExpression(node, checkMode | 1 /* Contextual */ | (inferenceContext ? 2 /* Inferential */ : 0));
-                // We strip literal freshness when an appropriate contextual type is present such that contextually typed
-                // literals always preserve their literal types (otherwise they might widen during type inference). An alternative
-                // here would be to not mark contextually typed literals as fresh in the first place.
-                var result = maybeTypeOfKind(type, 2944 /* Literal */) && isLiteralOfContextualType(type, instantiateContextualType(contextualType, node)) ?
-                    getRegularTypeOfLiteralType(type) : type;
-                return result;
             }
-            finally {
-                // In the event our operation is canceled or some other exception occurs, reset the contextual type
-                // so that we do not accidentally hold onto an instance of the checker, as a Type created in the services layer
-                // may hold onto the checker that created it.
-                context.contextualType = saveContextualType;
-                context.inferenceContext = saveInferenceContext;
+            if (restType) {
+                var spreadType = getSpreadArgumentType(args, argCount, args.length, restType, context, checkMode);
+                inferTypes(context.inferences, spreadType, restType);
             }
+            return getInferredTypes(context);
         }
-        function checkExpressionCached(node, checkMode) {
-            var links = getNodeLinks(node);
-            if (!links.resolvedType) {
-                if (checkMode && checkMode !== 0 /* Normal */) {
-                    return checkExpression(node, checkMode);
+        function getMutableArrayOrTupleType(type) {
+            return type.flags & 1048576 /* Union */ ? mapType(type, getMutableArrayOrTupleType) :
+                type.flags & 1 /* Any */ || isMutableArrayOrTuple(getBaseConstraintOfType(type) || type) ? type :
+                    isTupleType(type) ? createTupleType(getTypeArguments(type), type.target.elementFlags, /*readonly*/ false, type.target.labeledElementDeclarations) :
+                        createTupleType([type], [8 /* Variadic */]);
+        }
+        function getSpreadArgumentType(args, index, argCount, restType, context, checkMode) {
+            if (index >= argCount - 1) {
+                var arg = args[argCount - 1];
+                if (isSpreadArgument(arg)) {
+                    // We are inferring from a spread expression in the last argument position, i.e. both the parameter
+                    // and the argument are ...x forms.
+                    return getMutableArrayOrTupleType(arg.kind === 227 /* SyntheticExpression */ ? arg.type :
+                        checkExpressionWithContextualType(arg.expression, restType, context, checkMode));
                 }
-                // When computing a type that we're going to cache, we need to ignore any ongoing control flow
-                // analysis because variables may have transient types in indeterminable states. Moving flowLoopStart
-                // to the top of the stack ensures all transient types are computed from a known point.
-                var saveFlowLoopStart = flowLoopStart;
-                var saveFlowTypeCache = flowTypeCache;
-                flowLoopStart = flowLoopCount;
-                flowTypeCache = undefined;
-                links.resolvedType = checkExpression(node, checkMode);
-                flowTypeCache = saveFlowTypeCache;
-                flowLoopStart = saveFlowLoopStart;
             }
-            return links.resolvedType;
-        }
-        function isTypeAssertion(node) {
-            node = ts.skipParentheses(node);
-            return node.kind === 199 /* TypeAssertionExpression */ || node.kind === 217 /* AsExpression */;
-        }
-        function checkDeclarationInitializer(declaration, contextualType) {
-            var initializer = ts.getEffectiveInitializer(declaration);
-            var type = getQuickTypeOfExpression(initializer) ||
-                (contextualType ? checkExpressionWithContextualType(initializer, contextualType, /*inferenceContext*/ undefined, 0 /* Normal */) : checkExpressionCached(initializer));
-            return ts.isParameter(declaration) && declaration.name.kind === 190 /* ArrayBindingPattern */ &&
-                isTupleType(type) && !type.target.hasRestElement && getTypeReferenceArity(type) < declaration.name.elements.length ?
-                padTupleType(type, declaration.name) : type;
-        }
-        function padTupleType(type, pattern) {
-            var patternElements = pattern.elements;
-            var arity = getTypeReferenceArity(type);
-            var elementTypes = arity ? getTypeArguments(type).slice() : [];
-            for (var i = arity; i < patternElements.length; i++) {
-                var e = patternElements[i];
-                if (i < patternElements.length - 1 || !(e.kind === 191 /* BindingElement */ && e.dotDotDotToken)) {
-                    elementTypes.push(!ts.isOmittedExpression(e) && hasDefaultValue(e) ? getTypeFromBindingElement(e, /*includePatternInType*/ false, /*reportErrors*/ false) : anyType);
-                    if (!ts.isOmittedExpression(e) && !hasDefaultValue(e)) {
-                        reportImplicitAny(e, anyType);
+            var types = [];
+            var flags = [];
+            var names = [];
+            for (var i = index; i < argCount; i++) {
+                var arg = args[i];
+                if (isSpreadArgument(arg)) {
+                    var spreadType = arg.kind === 227 /* SyntheticExpression */ ? arg.type : checkExpression(arg.expression);
+                    if (isArrayLikeType(spreadType)) {
+                        types.push(spreadType);
+                        flags.push(8 /* Variadic */);
+                    }
+                    else {
+                        types.push(checkIteratedTypeOrElementType(33 /* Spread */, spreadType, undefinedType, arg.kind === 220 /* SpreadElement */ ? arg.expression : arg));
+                        flags.push(4 /* Rest */);
                     }
                 }
-            }
-            return createTupleType(elementTypes, type.target.minLength, /*hasRestElement*/ false, type.target.readonly);
-        }
-        function widenTypeInferredFromInitializer(declaration, type) {
-            var widened = ts.getCombinedNodeFlags(declaration) & 2 /* Const */ || ts.isDeclarationReadonly(declaration) ? type : getWidenedLiteralType(type);
-            if (ts.isInJSFile(declaration)) {
-                if (widened.flags & 98304 /* Nullable */) {
-                    reportImplicitAny(declaration, anyType);
-                    return anyType;
+                else {
+                    var contextualType = getIndexedAccessType(restType, getLiteralType(i - index));
+                    var argType = checkExpressionWithContextualType(arg, contextualType, context, checkMode);
+                    var hasPrimitiveContextualType = maybeTypeOfKind(contextualType, 131068 /* Primitive */ | 4194304 /* Index */ | 134217728 /* TemplateLiteral */ | 268435456 /* StringMapping */);
+                    types.push(hasPrimitiveContextualType ? getRegularTypeOfLiteralType(argType) : getWidenedLiteralType(argType));
+                    flags.push(1 /* Required */);
                 }
-                else if (isEmptyArrayLiteralType(widened)) {
-                    reportImplicitAny(declaration, anyArrayType);
-                    return anyArrayType;
+                if (arg.kind === 227 /* SyntheticExpression */ && arg.tupleNameSource) {
+                    names.push(arg.tupleNameSource);
                 }
             }
-            return widened;
+            return createTupleType(types, flags, /*readonly*/ false, ts.length(names) === ts.length(types) ? names : undefined);
         }
-        function isLiteralOfContextualType(candidateType, contextualType) {
-            if (contextualType) {
-                if (contextualType.flags & 3145728 /* UnionOrIntersection */) {
-                    var types = contextualType.types;
-                    return ts.some(types, function (t) { return isLiteralOfContextualType(candidateType, t); });
-                }
-                if (contextualType.flags & 58982400 /* InstantiableNonPrimitive */) {
-                    // If the contextual type is a type variable constrained to a primitive type, consider
-                    // this a literal context for literals of that primitive type. For example, given a
-                    // type parameter 'T extends string', infer string literal types for T.
-                    var constraint = getBaseConstraintOfType(contextualType) || unknownType;
-                    return maybeTypeOfKind(constraint, 4 /* String */) && maybeTypeOfKind(candidateType, 128 /* StringLiteral */) ||
-                        maybeTypeOfKind(constraint, 8 /* Number */) && maybeTypeOfKind(candidateType, 256 /* NumberLiteral */) ||
-                        maybeTypeOfKind(constraint, 64 /* BigInt */) && maybeTypeOfKind(candidateType, 2048 /* BigIntLiteral */) ||
-                        maybeTypeOfKind(constraint, 4096 /* ESSymbol */) && maybeTypeOfKind(candidateType, 8192 /* UniqueESSymbol */) ||
-                        isLiteralOfContextualType(candidateType, constraint);
+        function checkTypeArguments(signature, typeArgumentNodes, reportErrors, headMessage) {
+            var isJavascript = ts.isInJSFile(signature.declaration);
+            var typeParameters = signature.typeParameters;
+            var typeArgumentTypes = fillMissingTypeArguments(ts.map(typeArgumentNodes, getTypeFromTypeNode), typeParameters, getMinTypeArgumentCount(typeParameters), isJavascript);
+            var mapper;
+            for (var i = 0; i < typeArgumentNodes.length; i++) {
+                ts.Debug.assert(typeParameters[i] !== undefined, "Should not call checkTypeArguments with too many type arguments");
+                var constraint = getConstraintOfTypeParameter(typeParameters[i]);
+                if (constraint) {
+                    var errorInfo = reportErrors && headMessage ? (function () { return ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1); }) : undefined;
+                    var typeArgumentHeadMessage = headMessage || ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1;
+                    if (!mapper) {
+                        mapper = createTypeMapper(typeParameters, typeArgumentTypes);
+                    }
+                    var typeArgument = typeArgumentTypes[i];
+                    if (!checkTypeAssignableTo(typeArgument, getTypeWithThisArgument(instantiateType(constraint, mapper), typeArgument), reportErrors ? typeArgumentNodes[i] : undefined, typeArgumentHeadMessage, errorInfo)) {
+                        return undefined;
+                    }
                 }
-                // If the contextual type is a literal of a particular primitive type, we consider this a
-                // literal context for all literals of that primitive type.
-                return !!(contextualType.flags & (128 /* StringLiteral */ | 4194304 /* Index */) && maybeTypeOfKind(candidateType, 128 /* StringLiteral */) ||
-                    contextualType.flags & 256 /* NumberLiteral */ && maybeTypeOfKind(candidateType, 256 /* NumberLiteral */) ||
-                    contextualType.flags & 2048 /* BigIntLiteral */ && maybeTypeOfKind(candidateType, 2048 /* BigIntLiteral */) ||
-                    contextualType.flags & 512 /* BooleanLiteral */ && maybeTypeOfKind(candidateType, 512 /* BooleanLiteral */) ||
-                    contextualType.flags & 8192 /* UniqueESSymbol */ && maybeTypeOfKind(candidateType, 8192 /* UniqueESSymbol */));
             }
-            return false;
-        }
-        function isConstContext(node) {
-            var parent = node.parent;
-            return ts.isAssertionExpression(parent) && ts.isConstTypeReference(parent.type) ||
-                (ts.isParenthesizedExpression(parent) || ts.isArrayLiteralExpression(parent) || ts.isSpreadElement(parent)) && isConstContext(parent) ||
-                (ts.isPropertyAssignment(parent) || ts.isShorthandPropertyAssignment(parent)) && isConstContext(parent.parent);
-        }
-        function checkExpressionForMutableLocation(node, checkMode, contextualType, forceTuple) {
-            var type = checkExpression(node, checkMode, forceTuple);
-            return isConstContext(node) ? getRegularTypeOfLiteralType(type) :
-                isTypeAssertion(node) ? type :
-                    getWidenedLiteralLikeTypeForContextualType(type, instantiateContextualType(arguments.length === 2 ? getContextualType(node) : contextualType, node));
+            return typeArgumentTypes;
         }
-        function checkPropertyAssignment(node, checkMode) {
-            // Do not use hasDynamicName here, because that returns false for well known symbols.
-            // We want to perform checkComputedPropertyName for all computed properties, including
-            // well known symbols.
-            if (node.name.kind === 154 /* ComputedPropertyName */) {
-                checkComputedPropertyName(node.name);
+        function getJsxReferenceKind(node) {
+            if (isJsxIntrinsicIdentifier(node.tagName)) {
+                return 2 /* Mixed */;
             }
-            return checkExpressionForMutableLocation(node.initializer, checkMode);
-        }
-        function checkObjectLiteralMethod(node, checkMode) {
-            // Grammar checking
-            checkGrammarMethod(node);
-            // Do not use hasDynamicName here, because that returns false for well known symbols.
-            // We want to perform checkComputedPropertyName for all computed properties, including
-            // well known symbols.
-            if (node.name.kind === 154 /* ComputedPropertyName */) {
-                checkComputedPropertyName(node.name);
+            var tagType = getApparentType(checkExpression(node.tagName));
+            if (ts.length(getSignaturesOfType(tagType, 1 /* Construct */))) {
+                return 0 /* Component */;
             }
-            var uninstantiatedType = checkFunctionExpressionOrObjectLiteralMethod(node, checkMode);
-            return instantiateTypeWithSingleGenericCallSignature(node, uninstantiatedType, checkMode);
+            if (ts.length(getSignaturesOfType(tagType, 0 /* Call */))) {
+                return 1 /* Function */;
+            }
+            return 2 /* Mixed */;
         }
-        function instantiateTypeWithSingleGenericCallSignature(node, type, checkMode) {
-            if (checkMode && checkMode & (2 /* Inferential */ | 8 /* SkipGenericFunctions */)) {
-                var callSignature = getSingleSignature(type, 0 /* Call */, /*allowMembers*/ true);
-                var constructSignature = getSingleSignature(type, 1 /* Construct */, /*allowMembers*/ true);
-                var signature = callSignature || constructSignature;
-                if (signature && signature.typeParameters) {
-                    var contextualType = getApparentTypeOfContextualType(node, 2 /* NoConstraints */);
-                    if (contextualType) {
-                        var contextualSignature = getSingleSignature(getNonNullableType(contextualType), callSignature ? 0 /* Call */ : 1 /* Construct */, /*allowMembers*/ false);
-                        if (contextualSignature && !contextualSignature.typeParameters) {
-                            if (checkMode & 8 /* SkipGenericFunctions */) {
-                                skippedGenericFunction(node, checkMode);
-                                return anyFunctionType;
-                            }
-                            var context = getInferenceContext(node);
-                            // We have an expression that is an argument of a generic function for which we are performing
-                            // type argument inference. The expression is of a function type with a single generic call
-                            // signature and a contextual function type with a single non-generic call signature. Now check
-                            // if the outer function returns a function type with a single non-generic call signature and
-                            // if some of the outer function type parameters have no inferences so far. If so, we can
-                            // potentially add inferred type parameters to the outer function return type.
-                            var returnType = context.signature && getReturnTypeOfSignature(context.signature);
-                            var returnSignature = returnType && getSingleCallOrConstructSignature(returnType);
-                            if (returnSignature && !returnSignature.typeParameters && !ts.every(context.inferences, hasInferenceCandidates)) {
-                                // Instantiate the signature with its own type parameters as type arguments, possibly
-                                // renaming the type parameters to ensure they have unique names.
-                                var uniqueTypeParameters = getUniqueTypeParameters(context, signature.typeParameters);
-                                var instantiatedSignature = getSignatureInstantiationWithoutFillingInTypeArguments(signature, uniqueTypeParameters);
-                                // Infer from the parameters of the instantiated signature to the parameters of the
-                                // contextual signature starting with an empty set of inference candidates.
-                                var inferences_3 = ts.map(context.inferences, function (info) { return createInferenceInfo(info.typeParameter); });
-                                applyToParameterTypes(instantiatedSignature, contextualSignature, function (source, target) {
-                                    inferTypes(inferences_3, source, target, /*priority*/ 0, /*contravariant*/ true);
-                                });
-                                if (ts.some(inferences_3, hasInferenceCandidates)) {
-                                    // We have inference candidates, indicating that one or more type parameters are referenced
-                                    // in the parameter types of the contextual signature. Now also infer from the return type.
-                                    applyToReturnTypes(instantiatedSignature, contextualSignature, function (source, target) {
-                                        inferTypes(inferences_3, source, target);
-                                    });
-                                    // If the type parameters for which we produced candidates do not have any inferences yet,
-                                    // we adopt the new inference candidates and add the type parameters of the expression type
-                                    // to the set of inferred type parameters for the outer function return type.
-                                    if (!hasOverlappingInferences(context.inferences, inferences_3)) {
-                                        mergeInferences(context.inferences, inferences_3);
-                                        context.inferredTypeParameters = ts.concatenate(context.inferredTypeParameters, uniqueTypeParameters);
-                                        return getOrCreateTypeFromSignature(instantiatedSignature);
-                                    }
-                                }
-                            }
-                            return getOrCreateTypeFromSignature(instantiateSignatureInContextOf(signature, contextualSignature, context));
+        /**
+         * Check if the given signature can possibly be a signature called by the JSX opening-like element.
+         * @param node a JSX opening-like element we are trying to figure its call signature
+         * @param signature a candidate signature we are trying whether it is a call signature
+         * @param relation a relationship to check parameter and argument type
+         */
+        function checkApplicableSignatureForJsxOpeningLikeElement(node, signature, relation, checkMode, reportErrors, containingMessageChain, errorOutputContainer) {
+            // Stateless function components can have maximum of three arguments: "props", "context", and "updater".
+            // However "context" and "updater" are implicit and can't be specify by users. Only the first parameter, props,
+            // can be specified by users through attributes property.
+            var paramType = getEffectiveFirstArgumentForJsxSignature(signature, node);
+            var attributesType = checkExpressionWithContextualType(node.attributes, paramType, /*inferenceContext*/ undefined, checkMode);
+            return checkTagNameDoesNotExpectTooManyArguments() && checkTypeRelatedToAndOptionallyElaborate(attributesType, paramType, relation, reportErrors ? node.tagName : undefined, node.attributes, 
+            /*headMessage*/ undefined, containingMessageChain, errorOutputContainer);
+            function checkTagNameDoesNotExpectTooManyArguments() {
+                var _a;
+                var tagType = ts.isJsxOpeningElement(node) || ts.isJsxSelfClosingElement(node) && !isJsxIntrinsicIdentifier(node.tagName) ? checkExpression(node.tagName) : undefined;
+                if (!tagType) {
+                    return true;
+                }
+                var tagCallSignatures = getSignaturesOfType(tagType, 0 /* Call */);
+                if (!ts.length(tagCallSignatures)) {
+                    return true;
+                }
+                var factory = getJsxFactoryEntity(node);
+                if (!factory) {
+                    return true;
+                }
+                var factorySymbol = resolveEntityName(factory, 111551 /* Value */, /*ignoreErrors*/ true, /*dontResolveAlias*/ false, node);
+                if (!factorySymbol) {
+                    return true;
+                }
+                var factoryType = getTypeOfSymbol(factorySymbol);
+                var callSignatures = getSignaturesOfType(factoryType, 0 /* Call */);
+                if (!ts.length(callSignatures)) {
+                    return true;
+                }
+                var hasFirstParamSignatures = false;
+                var maxParamCount = 0;
+                // Check that _some_ first parameter expects a FC-like thing, and that some overload of the SFC expects an acceptable number of arguments
+                for (var _i = 0, callSignatures_1 = callSignatures; _i < callSignatures_1.length; _i++) {
+                    var sig = callSignatures_1[_i];
+                    var firstparam = getTypeAtPosition(sig, 0);
+                    var signaturesOfParam = getSignaturesOfType(firstparam, 0 /* Call */);
+                    if (!ts.length(signaturesOfParam))
+                        continue;
+                    for (var _b = 0, signaturesOfParam_1 = signaturesOfParam; _b < signaturesOfParam_1.length; _b++) {
+                        var paramSig = signaturesOfParam_1[_b];
+                        hasFirstParamSignatures = true;
+                        if (hasEffectiveRestParameter(paramSig)) {
+                            return true; // some signature has a rest param, so function components can have an arbitrary number of arguments
+                        }
+                        var paramCount = getParameterCount(paramSig);
+                        if (paramCount > maxParamCount) {
+                            maxParamCount = paramCount;
                         }
                     }
                 }
-            }
-            return type;
-        }
-        function skippedGenericFunction(node, checkMode) {
-            if (checkMode & 2 /* Inferential */) {
-                // We have skipped a generic function during inferential typing. Obtain the inference context and
-                // indicate this has occurred such that we know a second pass of inference is be needed.
-                var context = getInferenceContext(node);
-                context.flags |= 4 /* SkippedGenericFunction */;
-            }
-        }
-        function hasInferenceCandidates(info) {
-            return !!(info.candidates || info.contraCandidates);
-        }
-        function hasOverlappingInferences(a, b) {
-            for (var i = 0; i < a.length; i++) {
-                if (hasInferenceCandidates(a[i]) && hasInferenceCandidates(b[i])) {
+                if (!hasFirstParamSignatures) {
+                    // Not a single signature had a first parameter which expected a signature - for back compat, and
+                    // to guard against generic factories which won't have signatures directly, do not error
                     return true;
                 }
+                var absoluteMinArgCount = Infinity;
+                for (var _c = 0, tagCallSignatures_1 = tagCallSignatures; _c < tagCallSignatures_1.length; _c++) {
+                    var tagSig = tagCallSignatures_1[_c];
+                    var tagRequiredArgCount = getMinArgumentCount(tagSig);
+                    if (tagRequiredArgCount < absoluteMinArgCount) {
+                        absoluteMinArgCount = tagRequiredArgCount;
+                    }
+                }
+                if (absoluteMinArgCount <= maxParamCount) {
+                    return true; // some signature accepts the number of arguments the function component provides
+                }
+                if (reportErrors) {
+                    var diag = ts.createDiagnosticForNode(node.tagName, ts.Diagnostics.Tag_0_expects_at_least_1_arguments_but_the_JSX_factory_2_provides_at_most_3, ts.entityNameToString(node.tagName), absoluteMinArgCount, ts.entityNameToString(factory), maxParamCount);
+                    var tagNameDeclaration = (_a = getSymbolAtLocation(node.tagName)) === null || _a === void 0 ? void 0 : _a.valueDeclaration;
+                    if (tagNameDeclaration) {
+                        ts.addRelatedInfo(diag, ts.createDiagnosticForNode(tagNameDeclaration, ts.Diagnostics._0_is_declared_here, ts.entityNameToString(node.tagName)));
+                    }
+                    if (errorOutputContainer && errorOutputContainer.skipLogging) {
+                        (errorOutputContainer.errors || (errorOutputContainer.errors = [])).push(diag);
+                    }
+                    if (!errorOutputContainer.skipLogging) {
+                        diagnostics.add(diag);
+                    }
+                }
+                return false;
             }
-            return false;
         }
-        function mergeInferences(target, source) {
-            for (var i = 0; i < target.length; i++) {
-                if (!hasInferenceCandidates(target[i]) && hasInferenceCandidates(source[i])) {
-                    target[i] = source[i];
+        function getSignatureApplicabilityError(node, args, signature, relation, checkMode, reportErrors, containingMessageChain) {
+            var errorOutputContainer = { errors: undefined, skipLogging: true };
+            if (ts.isJsxOpeningLikeElement(node)) {
+                if (!checkApplicableSignatureForJsxOpeningLikeElement(node, signature, relation, checkMode, reportErrors, containingMessageChain, errorOutputContainer)) {
+                    ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "jsx should have errors when reporting errors");
+                    return errorOutputContainer.errors || ts.emptyArray;
                 }
+                return undefined;
             }
-        }
-        function getUniqueTypeParameters(context, typeParameters) {
-            var result = [];
-            var oldTypeParameters;
-            var newTypeParameters;
-            for (var _i = 0, typeParameters_2 = typeParameters; _i < typeParameters_2.length; _i++) {
-                var tp = typeParameters_2[_i];
-                var name = tp.symbol.escapedName;
-                if (hasTypeParameterByName(context.inferredTypeParameters, name) || hasTypeParameterByName(result, name)) {
-                    var newName = getUniqueTypeParameterName(ts.concatenate(context.inferredTypeParameters, result), name);
-                    var symbol = createSymbol(262144 /* TypeParameter */, newName);
-                    var newTypeParameter = createTypeParameter(symbol);
-                    newTypeParameter.target = tp;
-                    oldTypeParameters = ts.append(oldTypeParameters, tp);
-                    newTypeParameters = ts.append(newTypeParameters, newTypeParameter);
-                    result.push(newTypeParameter);
+            var thisType = getThisTypeOfSignature(signature);
+            if (thisType && thisType !== voidType && node.kind !== 204 /* NewExpression */) {
+                // If the called expression is not of the form `x.f` or `x["f"]`, then sourceType = voidType
+                // If the signature's 'this' type is voidType, then the check is skipped -- anything is compatible.
+                // If the expression is a new expression, then the check is skipped.
+                var thisArgumentNode = getThisArgumentOfCall(node);
+                var thisArgumentType = void 0;
+                if (thisArgumentNode) {
+                    thisArgumentType = checkExpression(thisArgumentNode);
+                    if (ts.isOptionalChainRoot(thisArgumentNode.parent)) {
+                        thisArgumentType = getNonNullableType(thisArgumentType);
+                    }
+                    else if (ts.isOptionalChain(thisArgumentNode.parent)) {
+                        thisArgumentType = removeOptionalTypeMarker(thisArgumentType);
+                    }
                 }
                 else {
-                    result.push(tp);
+                    thisArgumentType = voidType;
+                }
+                var errorNode = reportErrors ? (thisArgumentNode || node) : undefined;
+                var headMessage_1 = ts.Diagnostics.The_this_context_of_type_0_is_not_assignable_to_method_s_this_of_type_1;
+                if (!checkTypeRelatedTo(thisArgumentType, thisType, relation, errorNode, headMessage_1, containingMessageChain, errorOutputContainer)) {
+                    ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "this parameter should have errors when reporting errors");
+                    return errorOutputContainer.errors || ts.emptyArray;
                 }
             }
-            if (newTypeParameters) {
-                var mapper = createTypeMapper(oldTypeParameters, newTypeParameters);
-                for (var _a = 0, newTypeParameters_1 = newTypeParameters; _a < newTypeParameters_1.length; _a++) {
-                    var tp = newTypeParameters_1[_a];
-                    tp.mapper = mapper;
+            var headMessage = ts.Diagnostics.Argument_of_type_0_is_not_assignable_to_parameter_of_type_1;
+            var restType = getNonArrayRestType(signature);
+            var argCount = restType ? Math.min(getParameterCount(signature) - 1, args.length) : args.length;
+            for (var i = 0; i < argCount; i++) {
+                var arg = args[i];
+                if (arg.kind !== 222 /* OmittedExpression */) {
+                    var paramType = getTypeAtPosition(signature, i);
+                    var argType = checkExpressionWithContextualType(arg, paramType, /*inferenceContext*/ undefined, checkMode);
+                    // If one or more arguments are still excluded (as indicated by CheckMode.SkipContextSensitive),
+                    // we obtain the regular type of any object literal arguments because we may not have inferred complete
+                    // parameter types yet and therefore excess property checks may yield false positives (see #17041).
+                    var checkArgType = checkMode & 4 /* SkipContextSensitive */ ? getRegularTypeOfObjectLiteral(argType) : argType;
+                    if (!checkTypeRelatedToAndOptionallyElaborate(checkArgType, paramType, relation, reportErrors ? arg : undefined, arg, headMessage, containingMessageChain, errorOutputContainer)) {
+                        ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "parameter should have errors when reporting errors");
+                        maybeAddMissingAwaitInfo(arg, checkArgType, paramType);
+                        return errorOutputContainer.errors || ts.emptyArray;
+                    }
                 }
             }
-            return result;
-        }
-        function hasTypeParameterByName(typeParameters, name) {
-            return ts.some(typeParameters, function (tp) { return tp.symbol.escapedName === name; });
-        }
-        function getUniqueTypeParameterName(typeParameters, baseName) {
-            var len = baseName.length;
-            while (len > 1 && baseName.charCodeAt(len - 1) >= 48 /* _0 */ && baseName.charCodeAt(len - 1) <= 57 /* _9 */)
-                len--;
-            var s = baseName.slice(0, len);
-            for (var index = 1; true; index++) {
-                var augmentedName = (s + index);
-                if (!hasTypeParameterByName(typeParameters, augmentedName)) {
-                    return augmentedName;
+            if (restType) {
+                var spreadType = getSpreadArgumentType(args, argCount, args.length, restType, /*context*/ undefined, checkMode);
+                var errorNode = reportErrors ? argCount < args.length ? args[argCount] : node : undefined;
+                if (!checkTypeRelatedTo(spreadType, restType, relation, errorNode, headMessage, /*containingMessageChain*/ undefined, errorOutputContainer)) {
+                    ts.Debug.assert(!reportErrors || !!errorOutputContainer.errors, "rest parameter should have errors when reporting errors");
+                    maybeAddMissingAwaitInfo(errorNode, spreadType, restType);
+                    return errorOutputContainer.errors || ts.emptyArray;
+                }
+            }
+            return undefined;
+            function maybeAddMissingAwaitInfo(errorNode, source, target) {
+                if (errorNode && reportErrors && errorOutputContainer.errors && errorOutputContainer.errors.length) {
+                    // Bail if target is Promise-like---something else is wrong
+                    if (getAwaitedTypeOfPromise(target)) {
+                        return;
+                    }
+                    var awaitedTypeOfSource = getAwaitedTypeOfPromise(source);
+                    if (awaitedTypeOfSource && isTypeRelatedTo(awaitedTypeOfSource, target, relation)) {
+                        ts.addRelatedInfo(errorOutputContainer.errors[0], ts.createDiagnosticForNode(errorNode, ts.Diagnostics.Did_you_forget_to_use_await));
+                    }
                 }
             }
         }
-        function getReturnTypeOfSingleNonGenericCallSignature(funcType) {
-            var signature = getSingleCallSignature(funcType);
-            if (signature && !signature.typeParameters) {
-                return getReturnTypeOfSignature(signature);
+        /**
+         * Returns the this argument in calls like x.f(...) and x[f](...). Undefined otherwise.
+         */
+        function getThisArgumentOfCall(node) {
+            if (node.kind === 203 /* CallExpression */) {
+                var callee = ts.skipOuterExpressions(node.expression);
+                if (ts.isAccessExpression(callee)) {
+                    return callee.expression;
+                }
             }
         }
-        function getReturnTypeOfSingleNonGenericSignatureOfCallChain(expr) {
-            var funcType = checkExpression(expr.expression);
-            var nonOptionalType = getOptionalExpressionType(funcType, expr.expression);
-            var returnType = getReturnTypeOfSingleNonGenericCallSignature(funcType);
-            return returnType && propagateOptionalTypeMarker(returnType, expr, nonOptionalType !== funcType);
+        function createSyntheticExpression(parent, type, isSpread, tupleNameSource) {
+            var result = ts.parseNodeFactory.createSyntheticExpression(type, isSpread, tupleNameSource);
+            ts.setTextRange(result, parent);
+            ts.setParent(result, parent);
+            return result;
         }
         /**
-         * Returns the type of an expression. Unlike checkExpression, this function is simply concerned
-         * with computing the type and may not fully check all contained sub-expressions for errors.
+         * Returns the effective arguments for an expression that works like a function invocation.
          */
-        function getTypeOfExpression(node) {
-            // Don't bother caching types that require no flow analysis and are quick to compute.
-            var quickType = getQuickTypeOfExpression(node);
-            if (quickType) {
-                return quickType;
-            }
-            // If a type has been cached for the node, return it.
-            if (node.flags & 67108864 /* TypeCached */ && flowTypeCache) {
-                var cachedType = flowTypeCache[getNodeId(node)];
-                if (cachedType) {
-                    return cachedType;
+        function getEffectiveCallArguments(node) {
+            if (node.kind === 205 /* TaggedTemplateExpression */) {
+                var template = node.template;
+                var args_3 = [createSyntheticExpression(template, getGlobalTemplateStringsArrayType())];
+                if (template.kind === 218 /* TemplateExpression */) {
+                    ts.forEach(template.templateSpans, function (span) {
+                        args_3.push(span.expression);
+                    });
                 }
+                return args_3;
             }
-            var startInvocationCount = flowInvocationCount;
-            var type = checkExpression(node);
-            // If control flow analysis was required to determine the type, it is worth caching.
-            if (flowInvocationCount !== startInvocationCount) {
-                var cache = flowTypeCache || (flowTypeCache = []);
-                cache[getNodeId(node)] = type;
-                node.flags |= 67108864 /* TypeCached */;
+            if (node.kind === 161 /* Decorator */) {
+                return getEffectiveDecoratorArguments(node);
             }
-            return type;
-        }
-        function getQuickTypeOfExpression(node) {
-            var expr = ts.skipParentheses(node);
-            // Optimize for the common case of a call to a function with a single non-generic call
-            // signature where we can just fetch the return type without checking the arguments.
-            if (ts.isCallExpression(expr) && expr.expression.kind !== 102 /* SuperKeyword */ && !ts.isRequireCall(expr, /*checkArgumentIsStringLiteralLike*/ true) && !isSymbolOrSymbolForCall(expr)) {
-                var type = ts.isCallChain(expr) ? getReturnTypeOfSingleNonGenericSignatureOfCallChain(expr) :
-                    getReturnTypeOfSingleNonGenericCallSignature(checkNonNullExpression(expr.expression));
-                if (type) {
-                    return type;
-                }
+            if (ts.isJsxOpeningLikeElement(node)) {
+                return node.attributes.properties.length > 0 || (ts.isJsxOpeningElement(node) && node.parent.children.length > 0) ? [node.attributes] : ts.emptyArray;
             }
-            else if (ts.isAssertionExpression(expr) && !ts.isConstTypeReference(expr.type)) {
-                return getTypeFromTypeNode(expr.type);
+            var args = node.arguments || ts.emptyArray;
+            var spreadIndex = getSpreadArgumentIndex(args);
+            if (spreadIndex >= 0) {
+                // Create synthetic arguments from spreads of tuple types.
+                var effectiveArgs_1 = args.slice(0, spreadIndex);
+                var _loop_20 = function (i) {
+                    var arg = args[i];
+                    // We can call checkExpressionCached because spread expressions never have a contextual type.
+                    var spreadType = arg.kind === 220 /* SpreadElement */ && (flowLoopCount ? checkExpression(arg.expression) : checkExpressionCached(arg.expression));
+                    if (spreadType && isTupleType(spreadType)) {
+                        ts.forEach(getTypeArguments(spreadType), function (t, i) {
+                            var _a;
+                            var flags = spreadType.target.elementFlags[i];
+                            var syntheticArg = createSyntheticExpression(arg, flags & 4 /* Rest */ ? createArrayType(t) : t, !!(flags & 12 /* Variable */), (_a = spreadType.target.labeledElementDeclarations) === null || _a === void 0 ? void 0 : _a[i]);
+                            effectiveArgs_1.push(syntheticArg);
+                        });
+                    }
+                    else {
+                        effectiveArgs_1.push(arg);
+                    }
+                };
+                for (var i = spreadIndex; i < args.length; i++) {
+                    _loop_20(i);
+                }
+                return effectiveArgs_1;
             }
-            else if (node.kind === 8 /* NumericLiteral */ || node.kind === 10 /* StringLiteral */ ||
-                node.kind === 106 /* TrueKeyword */ || node.kind === 91 /* FalseKeyword */) {
-                return checkExpression(node);
+            return args;
+        }
+        /**
+         * Returns the synthetic argument list for a decorator invocation.
+         */
+        function getEffectiveDecoratorArguments(node) {
+            var parent = node.parent;
+            var expr = node.expression;
+            switch (parent.kind) {
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                    // For a class decorator, the `target` is the type of the class (e.g. the
+                    // "static" or "constructor" side of the class).
+                    return [
+                        createSyntheticExpression(expr, getTypeOfSymbol(getSymbolOfNode(parent)))
+                    ];
+                case 160 /* Parameter */:
+                    // A parameter declaration decorator will have three arguments (see
+                    // `ParameterDecorator` in core.d.ts).
+                    var func = parent.parent;
+                    return [
+                        createSyntheticExpression(expr, parent.parent.kind === 166 /* Constructor */ ? getTypeOfSymbol(getSymbolOfNode(func)) : errorType),
+                        createSyntheticExpression(expr, anyType),
+                        createSyntheticExpression(expr, numberType)
+                    ];
+                case 163 /* PropertyDeclaration */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    // A method or accessor declaration decorator will have two or three arguments (see
+                    // `PropertyDecorator` and `MethodDecorator` in core.d.ts). If we are emitting decorators
+                    // for ES3, we will only pass two arguments.
+                    var hasPropDesc = parent.kind !== 163 /* PropertyDeclaration */ && languageVersion !== 0 /* ES3 */;
+                    return [
+                        createSyntheticExpression(expr, getParentTypeOfClassElement(parent)),
+                        createSyntheticExpression(expr, getClassElementPropertyKeyType(parent)),
+                        createSyntheticExpression(expr, hasPropDesc ? createTypedPropertyDescriptorType(getTypeOfNode(parent)) : anyType)
+                    ];
             }
-            return undefined;
+            return ts.Debug.fail();
         }
         /**
-         * Returns the type of an expression. Unlike checkExpression, this function is simply concerned
-         * with computing the type and may not fully check all contained sub-expressions for errors.
-         * It is intended for uses where you know there is no contextual type,
-         * and requesting the contextual type might cause a circularity or other bad behaviour.
-         * It sets the contextual type of the node to any before calling getTypeOfExpression.
+         * Returns the argument count for a decorator node that works like a function invocation.
          */
-        function getContextFreeTypeOfExpression(node) {
-            var links = getNodeLinks(node);
-            if (links.contextFreeType) {
-                return links.contextFreeType;
+        function getDecoratorArgumentCount(node, signature) {
+            switch (node.parent.kind) {
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                    return 1;
+                case 163 /* PropertyDeclaration */:
+                    return 2;
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    // For ES3 or decorators with only two parameters we supply only two arguments
+                    return languageVersion === 0 /* ES3 */ || signature.parameters.length <= 2 ? 2 : 3;
+                case 160 /* Parameter */:
+                    return 3;
+                default:
+                    return ts.Debug.fail();
             }
-            var saveContextualType = node.contextualType;
-            node.contextualType = anyType;
-            try {
-                var type = links.contextFreeType = checkExpression(node, 4 /* SkipContextSensitive */);
-                return type;
+        }
+        function getDiagnosticSpanForCallNode(node, doNotIncludeArguments) {
+            var start;
+            var length;
+            var sourceFile = ts.getSourceFileOfNode(node);
+            if (ts.isPropertyAccessExpression(node.expression)) {
+                var nameSpan = ts.getErrorSpanForNode(sourceFile, node.expression.name);
+                start = nameSpan.start;
+                length = doNotIncludeArguments ? nameSpan.length : node.end - start;
             }
-            finally {
-                // In the event our operation is canceled or some other exception occurs, reset the contextual type
-                // so that we do not accidentally hold onto an instance of the checker, as a Type created in the services layer
-                // may hold onto the checker that created it.
-                node.contextualType = saveContextualType;
+            else {
+                var expressionSpan = ts.getErrorSpanForNode(sourceFile, node.expression);
+                start = expressionSpan.start;
+                length = doNotIncludeArguments ? expressionSpan.length : node.end - start;
             }
+            return { start: start, length: length, sourceFile: sourceFile };
         }
-        function checkExpression(node, checkMode, forceTuple) {
-            var saveCurrentNode = currentNode;
-            currentNode = node;
-            instantiationCount = 0;
-            var uninstantiatedType = checkExpressionWorker(node, checkMode, forceTuple);
-            var type = instantiateTypeWithSingleGenericCallSignature(node, uninstantiatedType, checkMode);
-            if (isConstEnumObjectType(type)) {
-                checkConstEnumAccess(node, type);
+        function getDiagnosticForCallNode(node, message, arg0, arg1, arg2, arg3) {
+            if (ts.isCallExpression(node)) {
+                var _a = getDiagnosticSpanForCallNode(node), sourceFile = _a.sourceFile, start = _a.start, length_5 = _a.length;
+                return ts.createFileDiagnostic(sourceFile, start, length_5, message, arg0, arg1, arg2, arg3);
+            }
+            else {
+                return ts.createDiagnosticForNode(node, message, arg0, arg1, arg2, arg3);
             }
-            currentNode = saveCurrentNode;
-            return type;
         }
-        function checkConstEnumAccess(node, type) {
-            // enum object type for const enums are only permitted in:
-            // - 'left' in property access
-            // - 'object' in indexed access
-            // - target in rhs of import statement
-            var ok = (node.parent.kind === 194 /* PropertyAccessExpression */ && node.parent.expression === node) ||
-                (node.parent.kind === 195 /* ElementAccessExpression */ && node.parent.expression === node) ||
-                ((node.kind === 75 /* Identifier */ || node.kind === 153 /* QualifiedName */) && isInRightSideOfImportOrExportAssignment(node) ||
-                    (node.parent.kind === 172 /* TypeQuery */ && node.parent.exprName === node)) ||
-                (node.parent.kind === 263 /* ExportSpecifier */); // We allow reexporting const enums
-            if (!ok) {
-                error(node, ts.Diagnostics.const_enums_can_only_be_used_in_property_or_index_access_expressions_or_the_right_hand_side_of_an_import_declaration_or_export_assignment_or_type_query);
+        function isPromiseResolveArityError(node) {
+            if (!ts.isCallExpression(node) || !ts.isIdentifier(node.expression))
+                return false;
+            var symbol = resolveName(node.expression, node.expression.escapedText, 111551 /* Value */, undefined, undefined, false);
+            var decl = symbol === null || symbol === void 0 ? void 0 : symbol.valueDeclaration;
+            if (!decl || !ts.isParameter(decl) || !isFunctionExpressionOrArrowFunction(decl.parent) || !ts.isNewExpression(decl.parent.parent) || !ts.isIdentifier(decl.parent.parent.expression)) {
+                return false;
             }
-            if (compilerOptions.isolatedModules) {
-                ts.Debug.assert(!!(type.symbol.flags & 128 /* ConstEnum */));
-                var constEnumDeclaration = type.symbol.valueDeclaration;
-                if (constEnumDeclaration.flags & 8388608 /* Ambient */) {
-                    error(node, ts.Diagnostics.Cannot_access_ambient_const_enums_when_the_isolatedModules_flag_is_provided);
+            var globalPromiseSymbol = getGlobalPromiseConstructorSymbol(/*reportErrors*/ false);
+            if (!globalPromiseSymbol)
+                return false;
+            var constructorSymbol = getSymbolAtLocation(decl.parent.parent.expression, /*ignoreErrors*/ true);
+            return constructorSymbol === globalPromiseSymbol;
+        }
+        function getArgumentArityError(node, signatures, args) {
+            var min = Number.POSITIVE_INFINITY;
+            var max = Number.NEGATIVE_INFINITY;
+            var belowArgCount = Number.NEGATIVE_INFINITY;
+            var aboveArgCount = Number.POSITIVE_INFINITY;
+            var argCount = args.length;
+            var closestSignature;
+            for (var _i = 0, signatures_8 = signatures; _i < signatures_8.length; _i++) {
+                var sig = signatures_8[_i];
+                var minCount = getMinArgumentCount(sig);
+                var maxCount = getParameterCount(sig);
+                if (minCount < argCount && minCount > belowArgCount)
+                    belowArgCount = minCount;
+                if (argCount < maxCount && maxCount < aboveArgCount)
+                    aboveArgCount = maxCount;
+                if (minCount < min) {
+                    min = minCount;
+                    closestSignature = sig;
+                }
+                max = Math.max(max, maxCount);
+            }
+            var hasRestParameter = ts.some(signatures, hasEffectiveRestParameter);
+            var paramRange = hasRestParameter ? min :
+                min < max ? min + "-" + max :
+                    min;
+            var hasSpreadArgument = getSpreadArgumentIndex(args) > -1;
+            if (argCount <= max && hasSpreadArgument) {
+                argCount--;
+            }
+            var spanArray;
+            var related;
+            var error = hasRestParameter || hasSpreadArgument ?
+                hasRestParameter && hasSpreadArgument ?
+                    ts.Diagnostics.Expected_at_least_0_arguments_but_got_1_or_more :
+                    hasRestParameter ?
+                        ts.Diagnostics.Expected_at_least_0_arguments_but_got_1 :
+                        ts.Diagnostics.Expected_0_arguments_but_got_1_or_more :
+                paramRange === 1 && argCount === 0 && isPromiseResolveArityError(node) ?
+                    ts.Diagnostics.Expected_0_arguments_but_got_1_Did_you_forget_to_include_void_in_your_type_argument_to_Promise :
+                    ts.Diagnostics.Expected_0_arguments_but_got_1;
+            if (closestSignature && getMinArgumentCount(closestSignature) > argCount && closestSignature.declaration) {
+                var paramDecl = closestSignature.declaration.parameters[closestSignature.thisParameter ? argCount + 1 : argCount];
+                if (paramDecl) {
+                    related = ts.createDiagnosticForNode(paramDecl, ts.isBindingPattern(paramDecl.name) ? ts.Diagnostics.An_argument_matching_this_binding_pattern_was_not_provided :
+                        ts.isRestParameter(paramDecl) ? ts.Diagnostics.Arguments_for_the_rest_parameter_0_were_not_provided : ts.Diagnostics.An_argument_for_0_was_not_provided, !paramDecl.name ? argCount : !ts.isBindingPattern(paramDecl.name) ? ts.idText(ts.getFirstIdentifier(paramDecl.name)) : undefined);
+                }
+            }
+            if (min < argCount && argCount < max) {
+                return getDiagnosticForCallNode(node, ts.Diagnostics.No_overload_expects_0_arguments_but_overloads_do_exist_that_expect_either_1_or_2_arguments, argCount, belowArgCount, aboveArgCount);
+            }
+            if (!hasSpreadArgument && argCount < min) {
+                var diagnostic_1 = getDiagnosticForCallNode(node, error, paramRange, argCount);
+                return related ? ts.addRelatedInfo(diagnostic_1, related) : diagnostic_1;
+            }
+            if (hasRestParameter || hasSpreadArgument) {
+                spanArray = ts.factory.createNodeArray(args);
+                if (hasSpreadArgument && argCount) {
+                    var nextArg = ts.elementAt(args, getSpreadArgumentIndex(args) + 1) || undefined;
+                    spanArray = ts.factory.createNodeArray(args.slice(max > argCount && nextArg ? args.indexOf(nextArg) : Math.min(max, args.length - 1)));
                 }
             }
+            else {
+                spanArray = ts.factory.createNodeArray(args.slice(max));
+            }
+            var pos = ts.first(spanArray).pos;
+            var end = ts.last(spanArray).end;
+            if (end === pos) {
+                end++;
+            }
+            ts.setTextRangePosEnd(spanArray, pos, end);
+            var diagnostic = ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), spanArray, error, paramRange, argCount);
+            return related ? ts.addRelatedInfo(diagnostic, related) : diagnostic;
         }
-        function checkParenthesizedExpression(node, checkMode) {
-            var tag = ts.isInJSFile(node) ? ts.getJSDocTypeTag(node) : undefined;
-            if (tag) {
-                return checkAssertionWorker(tag, tag.typeExpression.type, node.expression, checkMode);
+        function getTypeArgumentArityError(node, signatures, typeArguments) {
+            var argCount = typeArguments.length;
+            // No overloads exist
+            if (signatures.length === 1) {
+                var sig = signatures[0];
+                var min_1 = getMinTypeArgumentCount(sig.typeParameters);
+                var max = ts.length(sig.typeParameters);
+                return ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), typeArguments, ts.Diagnostics.Expected_0_type_arguments_but_got_1, min_1 < max ? min_1 + "-" + max : min_1, argCount);
             }
-            return checkExpression(node.expression, checkMode);
+            // Overloads exist
+            var belowArgCount = -Infinity;
+            var aboveArgCount = Infinity;
+            for (var _i = 0, signatures_9 = signatures; _i < signatures_9.length; _i++) {
+                var sig = signatures_9[_i];
+                var min_2 = getMinTypeArgumentCount(sig.typeParameters);
+                var max = ts.length(sig.typeParameters);
+                if (min_2 > argCount) {
+                    aboveArgCount = Math.min(aboveArgCount, min_2);
+                }
+                else if (max < argCount) {
+                    belowArgCount = Math.max(belowArgCount, max);
+                }
+            }
+            if (belowArgCount !== -Infinity && aboveArgCount !== Infinity) {
+                return ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), typeArguments, ts.Diagnostics.No_overload_expects_0_type_arguments_but_overloads_do_exist_that_expect_either_1_or_2_type_arguments, argCount, belowArgCount, aboveArgCount);
+            }
+            return ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), typeArguments, ts.Diagnostics.Expected_0_type_arguments_but_got_1, belowArgCount === -Infinity ? aboveArgCount : belowArgCount, argCount);
         }
-        function checkExpressionWorker(node, checkMode, forceTuple) {
-            var kind = node.kind;
-            if (cancellationToken) {
-                // Only bother checking on a few construct kinds.  We don't want to be excessively
-                // hitting the cancellation token on every node we check.
-                switch (kind) {
-                    case 214 /* ClassExpression */:
-                    case 201 /* FunctionExpression */:
-                    case 202 /* ArrowFunction */:
-                        cancellationToken.throwIfCancellationRequested();
+        function resolveCall(node, signatures, candidatesOutArray, checkMode, callChainFlags, fallbackError) {
+            var isTaggedTemplate = node.kind === 205 /* TaggedTemplateExpression */;
+            var isDecorator = node.kind === 161 /* Decorator */;
+            var isJsxOpeningOrSelfClosingElement = ts.isJsxOpeningLikeElement(node);
+            var reportErrors = !candidatesOutArray && produceDiagnostics;
+            var typeArguments;
+            if (!isDecorator) {
+                typeArguments = node.typeArguments;
+                // We already perform checking on the type arguments on the class declaration itself.
+                if (isTaggedTemplate || isJsxOpeningOrSelfClosingElement || node.expression.kind !== 105 /* SuperKeyword */) {
+                    ts.forEach(typeArguments, checkSourceElement);
                 }
             }
-            switch (kind) {
-                case 75 /* Identifier */:
-                    return checkIdentifier(node);
-                case 104 /* ThisKeyword */:
-                    return checkThisExpression(node);
-                case 102 /* SuperKeyword */:
-                    return checkSuperExpression(node);
-                case 100 /* NullKeyword */:
-                    return nullWideningType;
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                case 10 /* StringLiteral */:
-                    return getFreshTypeOfLiteralType(getLiteralType(node.text));
-                case 8 /* NumericLiteral */:
-                    checkGrammarNumericLiteral(node);
-                    return getFreshTypeOfLiteralType(getLiteralType(+node.text));
-                case 9 /* BigIntLiteral */:
-                    checkGrammarBigIntLiteral(node);
-                    return getFreshTypeOfLiteralType(getBigIntLiteralType(node));
-                case 106 /* TrueKeyword */:
-                    return trueType;
-                case 91 /* FalseKeyword */:
-                    return falseType;
-                case 211 /* TemplateExpression */:
-                    return checkTemplateExpression(node);
-                case 13 /* RegularExpressionLiteral */:
-                    return globalRegExpType;
-                case 192 /* ArrayLiteralExpression */:
-                    return checkArrayLiteral(node, checkMode, forceTuple);
-                case 193 /* ObjectLiteralExpression */:
-                    return checkObjectLiteral(node, checkMode);
-                case 194 /* PropertyAccessExpression */:
-                    return checkPropertyAccessExpression(node);
-                case 153 /* QualifiedName */:
-                    return checkQualifiedName(node);
-                case 195 /* ElementAccessExpression */:
-                    return checkIndexedAccess(node);
-                case 196 /* CallExpression */:
-                    if (node.expression.kind === 96 /* ImportKeyword */) {
-                        return checkImportCallExpression(node);
+            var candidates = candidatesOutArray || [];
+            // reorderCandidates fills up the candidates array directly
+            reorderCandidates(signatures, candidates, callChainFlags);
+            if (!candidates.length) {
+                if (reportErrors) {
+                    diagnostics.add(getDiagnosticForCallNode(node, ts.Diagnostics.Call_target_does_not_contain_any_signatures));
+                }
+                return resolveErrorCall(node);
+            }
+            var args = getEffectiveCallArguments(node);
+            // The excludeArgument array contains true for each context sensitive argument (an argument
+            // is context sensitive it is susceptible to a one-time permanent contextual typing).
+            //
+            // The idea is that we will perform type argument inference & assignability checking once
+            // without using the susceptible parameters that are functions, and once more for those
+            // parameters, contextually typing each as we go along.
+            //
+            // For a tagged template, then the first argument be 'undefined' if necessary because it
+            // represents a TemplateStringsArray.
+            //
+            // For a decorator, no arguments are susceptible to contextual typing due to the fact
+            // decorators are applied to a declaration by the emitter, and not to an expression.
+            var isSingleNonGenericCandidate = candidates.length === 1 && !candidates[0].typeParameters;
+            var argCheckMode = !isDecorator && !isSingleNonGenericCandidate && ts.some(args, isContextSensitive) ? 4 /* SkipContextSensitive */ : 0 /* Normal */;
+            // The following variables are captured and modified by calls to chooseOverload.
+            // If overload resolution or type argument inference fails, we want to report the
+            // best error possible. The best error is one which says that an argument was not
+            // assignable to a parameter. This implies that everything else about the overload
+            // was fine. So if there is any overload that is only incorrect because of an
+            // argument, we will report an error on that one.
+            //
+            //     function foo(s: string): void;
+            //     function foo(n: number): void; // Report argument error on this overload
+            //     function foo(): void;
+            //     foo(true);
+            //
+            // If none of the overloads even made it that far, there are two possibilities.
+            // There was a problem with type arguments for some overload, in which case
+            // report an error on that. Or none of the overloads even had correct arity,
+            // in which case give an arity error.
+            //
+            //     function foo<T extends string>(x: T): void; // Report type argument error
+            //     function foo(): void;
+            //     foo<number>(0);
+            //
+            var candidatesForArgumentError;
+            var candidateForArgumentArityError;
+            var candidateForTypeArgumentError;
+            var result;
+            // If we are in signature help, a trailing comma indicates that we intend to provide another argument,
+            // so we will only accept overloads with arity at least 1 higher than the current number of provided arguments.
+            var signatureHelpTrailingComma = !!(checkMode & 16 /* IsForSignatureHelp */) && node.kind === 203 /* CallExpression */ && node.arguments.hasTrailingComma;
+            // Section 4.12.1:
+            // if the candidate list contains one or more signatures for which the type of each argument
+            // expression is a subtype of each corresponding parameter type, the return type of the first
+            // of those signatures becomes the return type of the function call.
+            // Otherwise, the return type of the first signature in the candidate list becomes the return
+            // type of the function call.
+            //
+            // Whether the call is an error is determined by assignability of the arguments. The subtype pass
+            // is just important for choosing the best signature. So in the case where there is only one
+            // signature, the subtype pass is useless. So skipping it is an optimization.
+            if (candidates.length > 1) {
+                result = chooseOverload(candidates, subtypeRelation, isSingleNonGenericCandidate, signatureHelpTrailingComma);
+            }
+            if (!result) {
+                result = chooseOverload(candidates, assignableRelation, isSingleNonGenericCandidate, signatureHelpTrailingComma);
+            }
+            if (result) {
+                return result;
+            }
+            // No signatures were applicable. Now report errors based on the last applicable signature with
+            // no arguments excluded from assignability checks.
+            // If candidate is undefined, it means that no candidates had a suitable arity. In that case,
+            // skip the checkApplicableSignature check.
+            if (reportErrors) {
+                if (candidatesForArgumentError) {
+                    if (candidatesForArgumentError.length === 1 || candidatesForArgumentError.length > 3) {
+                        var last_2 = candidatesForArgumentError[candidatesForArgumentError.length - 1];
+                        var chain_1;
+                        if (candidatesForArgumentError.length > 3) {
+                            chain_1 = ts.chainDiagnosticMessages(chain_1, ts.Diagnostics.The_last_overload_gave_the_following_error);
+                            chain_1 = ts.chainDiagnosticMessages(chain_1, ts.Diagnostics.No_overload_matches_this_call);
+                        }
+                        var diags = getSignatureApplicabilityError(node, args, last_2, assignableRelation, 0 /* Normal */, /*reportErrors*/ true, function () { return chain_1; });
+                        if (diags) {
+                            for (var _i = 0, diags_1 = diags; _i < diags_1.length; _i++) {
+                                var d = diags_1[_i];
+                                if (last_2.declaration && candidatesForArgumentError.length > 3) {
+                                    ts.addRelatedInfo(d, ts.createDiagnosticForNode(last_2.declaration, ts.Diagnostics.The_last_overload_is_declared_here));
+                                }
+                                addImplementationSuccessElaboration(last_2, d);
+                                diagnostics.add(d);
+                            }
+                        }
+                        else {
+                            ts.Debug.fail("No error for last overload signature");
+                        }
                     }
-                // falls through
-                case 197 /* NewExpression */:
-                    return checkCallExpression(node, checkMode);
-                case 198 /* TaggedTemplateExpression */:
-                    return checkTaggedTemplateExpression(node);
-                case 200 /* ParenthesizedExpression */:
-                    return checkParenthesizedExpression(node, checkMode);
-                case 214 /* ClassExpression */:
-                    return checkClassExpression(node);
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                    return checkFunctionExpressionOrObjectLiteralMethod(node, checkMode);
-                case 204 /* TypeOfExpression */:
-                    return checkTypeOfExpression(node);
-                case 199 /* TypeAssertionExpression */:
-                case 217 /* AsExpression */:
-                    return checkAssertion(node);
-                case 218 /* NonNullExpression */:
-                    return checkNonNullAssertion(node);
-                case 219 /* MetaProperty */:
-                    return checkMetaProperty(node);
-                case 203 /* DeleteExpression */:
-                    return checkDeleteExpression(node);
-                case 205 /* VoidExpression */:
-                    return checkVoidExpression(node);
-                case 206 /* AwaitExpression */:
-                    return checkAwaitExpression(node);
-                case 207 /* PrefixUnaryExpression */:
-                    return checkPrefixUnaryExpression(node);
-                case 208 /* PostfixUnaryExpression */:
-                    return checkPostfixUnaryExpression(node);
-                case 209 /* BinaryExpression */:
-                    return checkBinaryExpression(node, checkMode);
-                case 210 /* ConditionalExpression */:
-                    return checkConditionalExpression(node, checkMode);
-                case 213 /* SpreadElement */:
-                    return checkSpreadExpression(node, checkMode);
-                case 215 /* OmittedExpression */:
-                    return undefinedWideningType;
-                case 212 /* YieldExpression */:
-                    return checkYieldExpression(node);
-                case 220 /* SyntheticExpression */:
-                    return node.type;
-                case 276 /* JsxExpression */:
-                    return checkJsxExpression(node, checkMode);
-                case 266 /* JsxElement */:
-                    return checkJsxElement(node, checkMode);
-                case 267 /* JsxSelfClosingElement */:
-                    return checkJsxSelfClosingElement(node, checkMode);
-                case 270 /* JsxFragment */:
-                    return checkJsxFragment(node);
-                case 274 /* JsxAttributes */:
-                    return checkJsxAttributes(node, checkMode);
-                case 268 /* JsxOpeningElement */:
-                    ts.Debug.fail("Shouldn't ever directly check a JsxOpeningElement");
+                    else {
+                        var allDiagnostics = [];
+                        var max = 0;
+                        var min_3 = Number.MAX_VALUE;
+                        var minIndex = 0;
+                        var i_1 = 0;
+                        var _loop_21 = function (c) {
+                            var chain_2 = function () { return ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.Overload_0_of_1_2_gave_the_following_error, i_1 + 1, candidates.length, signatureToString(c)); };
+                            var diags_2 = getSignatureApplicabilityError(node, args, c, assignableRelation, 0 /* Normal */, /*reportErrors*/ true, chain_2);
+                            if (diags_2) {
+                                if (diags_2.length <= min_3) {
+                                    min_3 = diags_2.length;
+                                    minIndex = i_1;
+                                }
+                                max = Math.max(max, diags_2.length);
+                                allDiagnostics.push(diags_2);
+                            }
+                            else {
+                                ts.Debug.fail("No error for 3 or fewer overload signatures");
+                            }
+                            i_1++;
+                        };
+                        for (var _a = 0, candidatesForArgumentError_1 = candidatesForArgumentError; _a < candidatesForArgumentError_1.length; _a++) {
+                            var c = candidatesForArgumentError_1[_a];
+                            _loop_21(c);
+                        }
+                        var diags_3 = max > 1 ? allDiagnostics[minIndex] : ts.flatten(allDiagnostics);
+                        ts.Debug.assert(diags_3.length > 0, "No errors reported for 3 or fewer overload signatures");
+                        var chain = ts.chainDiagnosticMessages(ts.map(diags_3, function (d) { return typeof d.messageText === "string" ? d : d.messageText; }), ts.Diagnostics.No_overload_matches_this_call);
+                        // The below is a spread to guarantee we get a new (mutable) array - our `flatMap` helper tries to do "smart" optimizations where it reuses input
+                        // arrays and the emptyArray singleton where possible, which is decidedly not what we want while we're still constructing this diagnostic
+                        var related = __spreadArrays(ts.flatMap(diags_3, function (d) { return d.relatedInformation; }));
+                        var diag = void 0;
+                        if (ts.every(diags_3, function (d) { return d.start === diags_3[0].start && d.length === diags_3[0].length && d.file === diags_3[0].file; })) {
+                            var _b = diags_3[0], file = _b.file, start = _b.start, length_6 = _b.length;
+                            diag = { file: file, start: start, length: length_6, code: chain.code, category: chain.category, messageText: chain, relatedInformation: related };
+                        }
+                        else {
+                            diag = ts.createDiagnosticForNodeFromMessageChain(node, chain, related);
+                        }
+                        addImplementationSuccessElaboration(candidatesForArgumentError[0], diag);
+                        diagnostics.add(diag);
+                    }
+                }
+                else if (candidateForArgumentArityError) {
+                    diagnostics.add(getArgumentArityError(node, [candidateForArgumentArityError], args));
+                }
+                else if (candidateForTypeArgumentError) {
+                    checkTypeArguments(candidateForTypeArgumentError, node.typeArguments, /*reportErrors*/ true, fallbackError);
+                }
+                else {
+                    var signaturesWithCorrectTypeArgumentArity = ts.filter(signatures, function (s) { return hasCorrectTypeArgumentArity(s, typeArguments); });
+                    if (signaturesWithCorrectTypeArgumentArity.length === 0) {
+                        diagnostics.add(getTypeArgumentArityError(node, signatures, typeArguments));
+                    }
+                    else if (!isDecorator) {
+                        diagnostics.add(getArgumentArityError(node, signaturesWithCorrectTypeArgumentArity, args));
+                    }
+                    else if (fallbackError) {
+                        diagnostics.add(getDiagnosticForCallNode(node, fallbackError));
+                    }
+                }
+            }
+            return getCandidateForOverloadFailure(node, candidates, args, !!candidatesOutArray);
+            function addImplementationSuccessElaboration(failed, diagnostic) {
+                var _a, _b;
+                var oldCandidatesForArgumentError = candidatesForArgumentError;
+                var oldCandidateForArgumentArityError = candidateForArgumentArityError;
+                var oldCandidateForTypeArgumentError = candidateForTypeArgumentError;
+                var failedSignatureDeclarations = ((_b = (_a = failed.declaration) === null || _a === void 0 ? void 0 : _a.symbol) === null || _b === void 0 ? void 0 : _b.declarations) || ts.emptyArray;
+                var isOverload = failedSignatureDeclarations.length > 1;
+                var implDecl = isOverload ? ts.find(failedSignatureDeclarations, function (d) { return ts.isFunctionLikeDeclaration(d) && ts.nodeIsPresent(d.body); }) : undefined;
+                if (implDecl) {
+                    var candidate = getSignatureFromDeclaration(implDecl);
+                    var isSingleNonGenericCandidate_1 = !candidate.typeParameters;
+                    if (chooseOverload([candidate], assignableRelation, isSingleNonGenericCandidate_1)) {
+                        ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(implDecl, ts.Diagnostics.The_call_would_have_succeeded_against_this_implementation_but_implementation_signatures_of_overloads_are_not_externally_visible));
+                    }
+                }
+                candidatesForArgumentError = oldCandidatesForArgumentError;
+                candidateForArgumentArityError = oldCandidateForArgumentArityError;
+                candidateForTypeArgumentError = oldCandidateForTypeArgumentError;
+            }
+            function chooseOverload(candidates, relation, isSingleNonGenericCandidate, signatureHelpTrailingComma) {
+                if (signatureHelpTrailingComma === void 0) { signatureHelpTrailingComma = false; }
+                candidatesForArgumentError = undefined;
+                candidateForArgumentArityError = undefined;
+                candidateForTypeArgumentError = undefined;
+                if (isSingleNonGenericCandidate) {
+                    var candidate = candidates[0];
+                    if (ts.some(typeArguments) || !hasCorrectArity(node, args, candidate, signatureHelpTrailingComma)) {
+                        return undefined;
+                    }
+                    if (getSignatureApplicabilityError(node, args, candidate, relation, 0 /* Normal */, /*reportErrors*/ false, /*containingMessageChain*/ undefined)) {
+                        candidatesForArgumentError = [candidate];
+                        return undefined;
+                    }
+                    return candidate;
+                }
+                for (var candidateIndex = 0; candidateIndex < candidates.length; candidateIndex++) {
+                    var candidate = candidates[candidateIndex];
+                    if (!hasCorrectTypeArgumentArity(candidate, typeArguments) || !hasCorrectArity(node, args, candidate, signatureHelpTrailingComma)) {
+                        continue;
+                    }
+                    var checkCandidate = void 0;
+                    var inferenceContext = void 0;
+                    if (candidate.typeParameters) {
+                        var typeArgumentTypes = void 0;
+                        if (ts.some(typeArguments)) {
+                            typeArgumentTypes = checkTypeArguments(candidate, typeArguments, /*reportErrors*/ false);
+                            if (!typeArgumentTypes) {
+                                candidateForTypeArgumentError = candidate;
+                                continue;
+                            }
+                        }
+                        else {
+                            inferenceContext = createInferenceContext(candidate.typeParameters, candidate, /*flags*/ ts.isInJSFile(node) ? 2 /* AnyDefault */ : 0 /* None */);
+                            typeArgumentTypes = inferTypeArguments(node, candidate, args, argCheckMode | 8 /* SkipGenericFunctions */, inferenceContext);
+                            argCheckMode |= inferenceContext.flags & 4 /* SkippedGenericFunction */ ? 8 /* SkipGenericFunctions */ : 0 /* Normal */;
+                        }
+                        checkCandidate = getSignatureInstantiation(candidate, typeArgumentTypes, ts.isInJSFile(candidate.declaration), inferenceContext && inferenceContext.inferredTypeParameters);
+                        // If the original signature has a generic rest type, instantiation may produce a
+                        // signature with different arity and we need to perform another arity check.
+                        if (getNonArrayRestType(candidate) && !hasCorrectArity(node, args, checkCandidate, signatureHelpTrailingComma)) {
+                            candidateForArgumentArityError = checkCandidate;
+                            continue;
+                        }
+                    }
+                    else {
+                        checkCandidate = candidate;
+                    }
+                    if (getSignatureApplicabilityError(node, args, checkCandidate, relation, argCheckMode, /*reportErrors*/ false, /*containingMessageChain*/ undefined)) {
+                        // Give preference to error candidates that have no rest parameters (as they are more specific)
+                        (candidatesForArgumentError || (candidatesForArgumentError = [])).push(checkCandidate);
+                        continue;
+                    }
+                    if (argCheckMode) {
+                        // If one or more context sensitive arguments were excluded, we start including
+                        // them now (and keeping do so for any subsequent candidates) and perform a second
+                        // round of type inference and applicability checking for this particular candidate.
+                        argCheckMode = 0 /* Normal */;
+                        if (inferenceContext) {
+                            var typeArgumentTypes = inferTypeArguments(node, candidate, args, argCheckMode, inferenceContext);
+                            checkCandidate = getSignatureInstantiation(candidate, typeArgumentTypes, ts.isInJSFile(candidate.declaration), inferenceContext && inferenceContext.inferredTypeParameters);
+                            // If the original signature has a generic rest type, instantiation may produce a
+                            // signature with different arity and we need to perform another arity check.
+                            if (getNonArrayRestType(candidate) && !hasCorrectArity(node, args, checkCandidate, signatureHelpTrailingComma)) {
+                                candidateForArgumentArityError = checkCandidate;
+                                continue;
+                            }
+                        }
+                        if (getSignatureApplicabilityError(node, args, checkCandidate, relation, argCheckMode, /*reportErrors*/ false, /*containingMessageChain*/ undefined)) {
+                            // Give preference to error candidates that have no rest parameters (as they are more specific)
+                            (candidatesForArgumentError || (candidatesForArgumentError = [])).push(checkCandidate);
+                            continue;
+                        }
+                    }
+                    candidates[candidateIndex] = checkCandidate;
+                    return checkCandidate;
+                }
+                return undefined;
             }
-            return errorType;
         }
-        // DECLARATION AND STATEMENT TYPE CHECKING
-        function checkTypeParameter(node) {
-            // Grammar Checking
-            if (node.expression) {
-                grammarErrorOnFirstToken(node.expression, ts.Diagnostics.Type_expected);
+        // No signature was applicable. We have already reported the errors for the invalid signature.
+        function getCandidateForOverloadFailure(node, candidates, args, hasCandidatesOutArray) {
+            ts.Debug.assert(candidates.length > 0); // Else should not have called this.
+            checkNodeDeferred(node);
+            // Normally we will combine overloads. Skip this if they have type parameters since that's hard to combine.
+            // Don't do this if there is a `candidatesOutArray`,
+            // because then we want the chosen best candidate to be one of the overloads, not a combination.
+            return hasCandidatesOutArray || candidates.length === 1 || candidates.some(function (c) { return !!c.typeParameters; })
+                ? pickLongestCandidateSignature(node, candidates, args)
+                : createUnionOfSignaturesForOverloadFailure(candidates);
+        }
+        function createUnionOfSignaturesForOverloadFailure(candidates) {
+            var thisParameters = ts.mapDefined(candidates, function (c) { return c.thisParameter; });
+            var thisParameter;
+            if (thisParameters.length) {
+                thisParameter = createCombinedSymbolFromTypes(thisParameters, thisParameters.map(getTypeOfParameter));
             }
-            checkSourceElement(node.constraint);
-            checkSourceElement(node.default);
-            var typeParameter = getDeclaredTypeOfTypeParameter(getSymbolOfNode(node));
-            // Resolve base constraint to reveal circularity errors
-            getBaseConstraintOfType(typeParameter);
-            if (!hasNonCircularTypeParameterDefault(typeParameter)) {
-                error(node.default, ts.Diagnostics.Type_parameter_0_has_a_circular_default, typeToString(typeParameter));
+            var _a = ts.minAndMax(candidates, getNumNonRestParameters), minArgumentCount = _a.min, maxNonRestParam = _a.max;
+            var parameters = [];
+            var _loop_22 = function (i) {
+                var symbols = ts.mapDefined(candidates, function (s) { return signatureHasRestParameter(s) ?
+                    i < s.parameters.length - 1 ? s.parameters[i] : ts.last(s.parameters) :
+                    i < s.parameters.length ? s.parameters[i] : undefined; });
+                ts.Debug.assert(symbols.length !== 0);
+                parameters.push(createCombinedSymbolFromTypes(symbols, ts.mapDefined(candidates, function (candidate) { return tryGetTypeAtPosition(candidate, i); })));
+            };
+            for (var i = 0; i < maxNonRestParam; i++) {
+                _loop_22(i);
             }
-            var constraintType = getConstraintOfTypeParameter(typeParameter);
-            var defaultType = getDefaultFromTypeParameter(typeParameter);
-            if (constraintType && defaultType) {
-                checkTypeAssignableTo(defaultType, getTypeWithThisArgument(instantiateType(constraintType, makeUnaryTypeMapper(typeParameter, defaultType)), defaultType), node.default, ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1);
+            var restParameterSymbols = ts.mapDefined(candidates, function (c) { return signatureHasRestParameter(c) ? ts.last(c.parameters) : undefined; });
+            var flags = 0 /* None */;
+            if (restParameterSymbols.length !== 0) {
+                var type = createArrayType(getUnionType(ts.mapDefined(candidates, tryGetRestTypeOfSignature), 2 /* Subtype */));
+                parameters.push(createCombinedSymbolForOverloadFailure(restParameterSymbols, type));
+                flags |= 1 /* HasRestParameter */;
             }
-            if (produceDiagnostics) {
-                checkTypeNameIsReserved(node.name, ts.Diagnostics.Type_parameter_name_cannot_be_0);
+            if (candidates.some(signatureHasLiteralTypes)) {
+                flags |= 2 /* HasLiteralTypes */;
             }
+            return createSignature(candidates[0].declaration, 
+            /*typeParameters*/ undefined, // Before calling this we tested for `!candidates.some(c => !!c.typeParameters)`.
+            thisParameter, parameters, 
+            /*resolvedReturnType*/ getIntersectionType(candidates.map(getReturnTypeOfSignature)), 
+            /*typePredicate*/ undefined, minArgumentCount, flags);
         }
-        function checkParameter(node) {
-            // Grammar checking
-            // It is a SyntaxError if the Identifier "eval" or the Identifier "arguments" occurs as the
-            // Identifier in a PropertySetParameterList of a PropertyAssignment that is contained in strict code
-            // or if its FunctionBody is strict code(11.1.5).
-            checkGrammarDecoratorsAndModifiers(node);
-            checkVariableLikeDeclaration(node);
-            var func = ts.getContainingFunction(node);
-            if (ts.hasModifier(node, 92 /* ParameterPropertyModifier */)) {
-                if (!(func.kind === 162 /* Constructor */ && ts.nodeIsPresent(func.body))) {
-                    error(node, ts.Diagnostics.A_parameter_property_is_only_allowed_in_a_constructor_implementation);
-                }
-                if (func.kind === 162 /* Constructor */ && ts.isIdentifier(node.name) && node.name.escapedText === "constructor") {
-                    error(node.name, ts.Diagnostics.constructor_cannot_be_used_as_a_parameter_property_name);
-                }
+        function getNumNonRestParameters(signature) {
+            var numParams = signature.parameters.length;
+            return signatureHasRestParameter(signature) ? numParams - 1 : numParams;
+        }
+        function createCombinedSymbolFromTypes(sources, types) {
+            return createCombinedSymbolForOverloadFailure(sources, getUnionType(types, 2 /* Subtype */));
+        }
+        function createCombinedSymbolForOverloadFailure(sources, type) {
+            // This function is currently only used for erroneous overloads, so it's good enough to just use the first source.
+            return createSymbolWithType(ts.first(sources), type);
+        }
+        function pickLongestCandidateSignature(node, candidates, args) {
+            // Pick the longest signature. This way we can get a contextual type for cases like:
+            //     declare function f(a: { xa: number; xb: number; }, b: number);
+            //     f({ |
+            // Also, use explicitly-supplied type arguments if they are provided, so we can get a contextual signature in cases like:
+            //     declare function f<T>(k: keyof T);
+            //     f<Foo>("
+            var bestIndex = getLongestCandidateIndex(candidates, apparentArgumentCount === undefined ? args.length : apparentArgumentCount);
+            var candidate = candidates[bestIndex];
+            var typeParameters = candidate.typeParameters;
+            if (!typeParameters) {
+                return candidate;
             }
-            if (node.questionToken && ts.isBindingPattern(node.name) && func.body) {
-                error(node, ts.Diagnostics.A_binding_pattern_parameter_cannot_be_optional_in_an_implementation_signature);
+            var typeArgumentNodes = callLikeExpressionMayHaveTypeArguments(node) ? node.typeArguments : undefined;
+            var instantiated = typeArgumentNodes
+                ? createSignatureInstantiation(candidate, getTypeArgumentsFromNodes(typeArgumentNodes, typeParameters, ts.isInJSFile(node)))
+                : inferSignatureInstantiationForOverloadFailure(node, typeParameters, candidate, args);
+            candidates[bestIndex] = instantiated;
+            return instantiated;
+        }
+        function getTypeArgumentsFromNodes(typeArgumentNodes, typeParameters, isJs) {
+            var typeArguments = typeArgumentNodes.map(getTypeOfNode);
+            while (typeArguments.length > typeParameters.length) {
+                typeArguments.pop();
             }
-            if (node.name && ts.isIdentifier(node.name) && (node.name.escapedText === "this" || node.name.escapedText === "new")) {
-                if (func.parameters.indexOf(node) !== 0) {
-                    error(node, ts.Diagnostics.A_0_parameter_must_be_the_first_parameter, node.name.escapedText);
+            while (typeArguments.length < typeParameters.length) {
+                typeArguments.push(getConstraintOfTypeParameter(typeParameters[typeArguments.length]) || getDefaultTypeArgumentType(isJs));
+            }
+            return typeArguments;
+        }
+        function inferSignatureInstantiationForOverloadFailure(node, typeParameters, candidate, args) {
+            var inferenceContext = createInferenceContext(typeParameters, candidate, /*flags*/ ts.isInJSFile(node) ? 2 /* AnyDefault */ : 0 /* None */);
+            var typeArgumentTypes = inferTypeArguments(node, candidate, args, 4 /* SkipContextSensitive */ | 8 /* SkipGenericFunctions */, inferenceContext);
+            return createSignatureInstantiation(candidate, typeArgumentTypes);
+        }
+        function getLongestCandidateIndex(candidates, argsCount) {
+            var maxParamsIndex = -1;
+            var maxParams = -1;
+            for (var i = 0; i < candidates.length; i++) {
+                var candidate = candidates[i];
+                var paramCount = getParameterCount(candidate);
+                if (hasEffectiveRestParameter(candidate) || paramCount >= argsCount) {
+                    return i;
                 }
-                if (func.kind === 162 /* Constructor */ || func.kind === 166 /* ConstructSignature */ || func.kind === 171 /* ConstructorType */) {
-                    error(node, ts.Diagnostics.A_constructor_cannot_have_a_this_parameter);
+                if (paramCount > maxParams) {
+                    maxParams = paramCount;
+                    maxParamsIndex = i;
                 }
-                if (func.kind === 202 /* ArrowFunction */) {
-                    error(node, ts.Diagnostics.An_arrow_function_cannot_have_a_this_parameter);
+            }
+            return maxParamsIndex;
+        }
+        function resolveCallExpression(node, candidatesOutArray, checkMode) {
+            if (node.expression.kind === 105 /* SuperKeyword */) {
+                var superType = checkSuperExpression(node.expression);
+                if (isTypeAny(superType)) {
+                    for (var _i = 0, _a = node.arguments; _i < _a.length; _i++) {
+                        var arg = _a[_i];
+                        checkExpression(arg); // Still visit arguments so they get marked for visibility, etc
+                    }
+                    return anySignature;
                 }
-                if (func.kind === 163 /* GetAccessor */ || func.kind === 164 /* SetAccessor */) {
-                    error(node, ts.Diagnostics.get_and_set_accessors_cannot_declare_this_parameters);
+                if (superType !== errorType) {
+                    // In super call, the candidate signatures are the matching arity signatures of the base constructor function instantiated
+                    // with the type arguments specified in the extends clause.
+                    var baseTypeNode = ts.getEffectiveBaseTypeNode(ts.getContainingClass(node));
+                    if (baseTypeNode) {
+                        var baseConstructors = getInstantiatedConstructorsForTypeArguments(superType, baseTypeNode.typeArguments, baseTypeNode);
+                        return resolveCall(node, baseConstructors, candidatesOutArray, checkMode, 0 /* None */);
+                    }
                 }
+                return resolveUntypedCall(node);
             }
-            // Only check rest parameter type if it's not a binding pattern. Since binding patterns are
-            // not allowed in a rest parameter, we already have an error from checkGrammarParameterList.
-            if (node.dotDotDotToken && !ts.isBindingPattern(node.name) && !isTypeAssignableTo(getReducedType(getTypeOfSymbol(node.symbol)), anyReadonlyArrayType)) {
-                error(node, ts.Diagnostics.A_rest_parameter_must_be_of_an_array_type);
+            var callChainFlags;
+            var funcType = checkExpression(node.expression);
+            if (ts.isCallChain(node)) {
+                var nonOptionalType = getOptionalExpressionType(funcType, node.expression);
+                callChainFlags = nonOptionalType === funcType ? 0 /* None */ :
+                    ts.isOutermostOptionalChain(node) ? 8 /* IsOuterCallChain */ :
+                        4 /* IsInnerCallChain */;
+                funcType = nonOptionalType;
             }
-        }
-        function checkTypePredicate(node) {
-            var parent = getTypePredicateParent(node);
-            if (!parent) {
-                // The parent must not be valid.
-                error(node, ts.Diagnostics.A_type_predicate_is_only_allowed_in_return_type_position_for_functions_and_methods);
-                return;
+            else {
+                callChainFlags = 0 /* None */;
             }
-            var signature = getSignatureFromDeclaration(parent);
-            var typePredicate = getTypePredicateOfSignature(signature);
-            if (!typePredicate) {
-                return;
+            funcType = checkNonNullTypeWithReporter(funcType, node.expression, reportCannotInvokePossiblyNullOrUndefinedError);
+            if (funcType === silentNeverType) {
+                return silentNeverSignature;
             }
-            checkSourceElement(node.type);
-            var parameterName = node.parameterName;
-            if (typePredicate.kind === 0 /* This */ || typePredicate.kind === 2 /* AssertsThis */) {
-                getTypeFromThisTypeNode(parameterName);
+            var apparentType = getApparentType(funcType);
+            if (apparentType === errorType) {
+                // Another error has already been reported
+                return resolveErrorCall(node);
             }
-            else {
-                if (typePredicate.parameterIndex >= 0) {
-                    if (signatureHasRestParameter(signature) && typePredicate.parameterIndex === signature.parameters.length - 1) {
-                        error(parameterName, ts.Diagnostics.A_type_predicate_cannot_reference_a_rest_parameter);
-                    }
-                    else {
-                        if (typePredicate.type) {
-                            var leadingError = function () { return ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.A_type_predicate_s_type_must_be_assignable_to_its_parameter_s_type); };
-                            checkTypeAssignableTo(typePredicate.type, getTypeOfSymbol(signature.parameters[typePredicate.parameterIndex]), node.type, 
-                            /*headMessage*/ undefined, leadingError);
-                        }
-                    }
+            // Technically, this signatures list may be incomplete. We are taking the apparent type,
+            // but we are not including call signatures that may have been added to the Object or
+            // Function interface, since they have none by default. This is a bit of a leap of faith
+            // that the user will not add any.
+            var callSignatures = getSignaturesOfType(apparentType, 0 /* Call */);
+            var numConstructSignatures = getSignaturesOfType(apparentType, 1 /* Construct */).length;
+            // TS 1.0 Spec: 4.12
+            // In an untyped function call no TypeArgs are permitted, Args can be any argument list, no contextual
+            // types are provided for the argument expressions, and the result is always of type Any.
+            if (isUntypedFunctionCall(funcType, apparentType, callSignatures.length, numConstructSignatures)) {
+                // The unknownType indicates that an error already occurred (and was reported).  No
+                // need to report another error in this case.
+                if (funcType !== errorType && node.typeArguments) {
+                    error(node, ts.Diagnostics.Untyped_function_calls_may_not_accept_type_arguments);
                 }
-                else if (parameterName) {
-                    var hasReportedError = false;
-                    for (var _i = 0, _a = parent.parameters; _i < _a.length; _i++) {
-                        var name = _a[_i].name;
-                        if (ts.isBindingPattern(name) &&
-                            checkIfTypePredicateVariableIsDeclaredInBindingPattern(name, parameterName, typePredicate.parameterName)) {
-                            hasReportedError = true;
-                            break;
+                return resolveUntypedCall(node);
+            }
+            // If FuncExpr's apparent type(section 3.8.1) is a function type, the call is a typed function call.
+            // TypeScript employs overload resolution in typed function calls in order to support functions
+            // with multiple call signatures.
+            if (!callSignatures.length) {
+                if (numConstructSignatures) {
+                    error(node, ts.Diagnostics.Value_of_type_0_is_not_callable_Did_you_mean_to_include_new, typeToString(funcType));
+                }
+                else {
+                    var relatedInformation = void 0;
+                    if (node.arguments.length === 1) {
+                        var text = ts.getSourceFileOfNode(node).text;
+                        if (ts.isLineBreak(text.charCodeAt(ts.skipTrivia(text, node.expression.end, /* stopAfterLineBreak */ true) - 1))) {
+                            relatedInformation = ts.createDiagnosticForNode(node.expression, ts.Diagnostics.Are_you_missing_a_semicolon);
                         }
                     }
-                    if (!hasReportedError) {
-                        error(node.parameterName, ts.Diagnostics.Cannot_find_parameter_0, typePredicate.parameterName);
-                    }
+                    invocationError(node.expression, apparentType, 0 /* Call */, relatedInformation);
                 }
+                return resolveErrorCall(node);
             }
-        }
-        function getTypePredicateParent(node) {
-            switch (node.parent.kind) {
-                case 202 /* ArrowFunction */:
-                case 165 /* CallSignature */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 170 /* FunctionType */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    var parent = node.parent;
-                    if (node === parent.type) {
-                        return parent;
-                    }
+            // When a call to a generic function is an argument to an outer call to a generic function for which
+            // inference is in process, we have a choice to make. If the inner call relies on inferences made from
+            // its contextual type to its return type, deferring the inner call processing allows the best possible
+            // contextual type to accumulate. But if the outer call relies on inferences made from the return type of
+            // the inner call, the inner call should be processed early. There's no sure way to know which choice is
+            // right (only a full unification algorithm can determine that), so we resort to the following heuristic:
+            // If no type arguments are specified in the inner call and at least one call signature is generic and
+            // returns a function type, we choose to defer processing. This narrowly permits function composition
+            // operators to flow inferences through return types, but otherwise processes calls right away. We
+            // use the resolvingSignature singleton to indicate that we deferred processing. This result will be
+            // propagated out and eventually turned into nonInferrableType (a type that is assignable to anything and
+            // from which we never make inferences).
+            if (checkMode & 8 /* SkipGenericFunctions */ && !node.typeArguments && callSignatures.some(isGenericFunctionReturningFunction)) {
+                skippedGenericFunction(node, checkMode);
+                return resolvingSignature;
+            }
+            // If the function is explicitly marked with `@class`, then it must be constructed.
+            if (callSignatures.some(function (sig) { return ts.isInJSFile(sig.declaration) && !!ts.getJSDocClassTag(sig.declaration); })) {
+                error(node, ts.Diagnostics.Value_of_type_0_is_not_callable_Did_you_mean_to_include_new, typeToString(funcType));
+                return resolveErrorCall(node);
             }
+            return resolveCall(node, callSignatures, candidatesOutArray, checkMode, callChainFlags);
         }
-        function checkIfTypePredicateVariableIsDeclaredInBindingPattern(pattern, predicateVariableNode, predicateVariableName) {
-            for (var _i = 0, _a = pattern.elements; _i < _a.length; _i++) {
-                var element = _a[_i];
-                if (ts.isOmittedExpression(element)) {
-                    continue;
-                }
-                var name = element.name;
-                if (name.kind === 75 /* Identifier */ && name.escapedText === predicateVariableName) {
-                    error(predicateVariableNode, ts.Diagnostics.A_type_predicate_cannot_reference_element_0_in_a_binding_pattern, predicateVariableName);
-                    return true;
-                }
-                else if (name.kind === 190 /* ArrayBindingPattern */ || name.kind === 189 /* ObjectBindingPattern */) {
-                    if (checkIfTypePredicateVariableIsDeclaredInBindingPattern(name, predicateVariableNode, predicateVariableName)) {
-                        return true;
-                    }
+        function isGenericFunctionReturningFunction(signature) {
+            return !!(signature.typeParameters && isFunctionType(getReturnTypeOfSignature(signature)));
+        }
+        /**
+         * TS 1.0 spec: 4.12
+         * If FuncExpr is of type Any, or of an object type that has no call or construct signatures
+         * but is a subtype of the Function interface, the call is an untyped function call.
+         */
+        function isUntypedFunctionCall(funcType, apparentFuncType, numCallSignatures, numConstructSignatures) {
+            // We exclude union types because we may have a union of function types that happen to have no common signatures.
+            return isTypeAny(funcType) || isTypeAny(apparentFuncType) && !!(funcType.flags & 262144 /* TypeParameter */) ||
+                !numCallSignatures && !numConstructSignatures && !(apparentFuncType.flags & (1048576 /* Union */ | 131072 /* Never */)) && isTypeAssignableTo(funcType, globalFunctionType);
+        }
+        function resolveNewExpression(node, candidatesOutArray, checkMode) {
+            if (node.arguments && languageVersion < 1 /* ES5 */) {
+                var spreadIndex = getSpreadArgumentIndex(node.arguments);
+                if (spreadIndex >= 0) {
+                    error(node.arguments[spreadIndex], ts.Diagnostics.Spread_operator_in_new_expressions_is_only_available_when_targeting_ECMAScript_5_and_higher);
                 }
             }
-        }
-        function checkSignatureDeclaration(node) {
-            // Grammar checking
-            if (node.kind === 167 /* IndexSignature */) {
-                checkGrammarIndexSignature(node);
+            var expressionType = checkNonNullExpression(node.expression);
+            if (expressionType === silentNeverType) {
+                return silentNeverSignature;
             }
-            // TODO (yuisu): Remove this check in else-if when SyntaxKind.Construct is moved and ambient context is handled
-            else if (node.kind === 170 /* FunctionType */ || node.kind === 244 /* FunctionDeclaration */ || node.kind === 171 /* ConstructorType */ ||
-                node.kind === 165 /* CallSignature */ || node.kind === 162 /* Constructor */ ||
-                node.kind === 166 /* ConstructSignature */) {
-                checkGrammarFunctionLikeDeclaration(node);
+            // If expressionType's apparent type(section 3.8.1) is an object type with one or
+            // more construct signatures, the expression is processed in the same manner as a
+            // function call, but using the construct signatures as the initial set of candidate
+            // signatures for overload resolution. The result type of the function call becomes
+            // the result type of the operation.
+            expressionType = getApparentType(expressionType);
+            if (expressionType === errorType) {
+                // Another error has already been reported
+                return resolveErrorCall(node);
             }
-            var functionFlags = ts.getFunctionFlags(node);
-            if (!(functionFlags & 4 /* Invalid */)) {
-                // Async generators prior to ESNext require the __await and __asyncGenerator helpers
-                if ((functionFlags & 3 /* AsyncGenerator */) === 3 /* AsyncGenerator */ && languageVersion < 99 /* ESNext */) {
-                    checkExternalEmitHelpers(node, 12288 /* AsyncGeneratorIncludes */);
+            // TS 1.0 spec: 4.11
+            // If expressionType is of type Any, Args can be any argument
+            // list and the result of the operation is of type Any.
+            if (isTypeAny(expressionType)) {
+                if (node.typeArguments) {
+                    error(node, ts.Diagnostics.Untyped_function_calls_may_not_accept_type_arguments);
                 }
-                // Async functions prior to ES2017 require the __awaiter helper
-                if ((functionFlags & 3 /* AsyncGenerator */) === 2 /* Async */ && languageVersion < 4 /* ES2017 */) {
-                    checkExternalEmitHelpers(node, 64 /* Awaiter */);
+                return resolveUntypedCall(node);
+            }
+            // Technically, this signatures list may be incomplete. We are taking the apparent type,
+            // but we are not including construct signatures that may have been added to the Object or
+            // Function interface, since they have none by default. This is a bit of a leap of faith
+            // that the user will not add any.
+            var constructSignatures = getSignaturesOfType(expressionType, 1 /* Construct */);
+            if (constructSignatures.length) {
+                if (!isConstructorAccessible(node, constructSignatures[0])) {
+                    return resolveErrorCall(node);
                 }
-                // Generator functions, Async functions, and Async Generator functions prior to
-                // ES2015 require the __generator helper
-                if ((functionFlags & 3 /* AsyncGenerator */) !== 0 /* Normal */ && languageVersion < 2 /* ES2015 */) {
-                    checkExternalEmitHelpers(node, 128 /* Generator */);
+                // If the expression is a class of abstract type, then it cannot be instantiated.
+                // Note, only class declarations can be declared abstract.
+                // In the case of a merged class-module or class-interface declaration,
+                // only the class declaration node will have the Abstract flag set.
+                var valueDecl = expressionType.symbol && ts.getClassLikeDeclarationOfSymbol(expressionType.symbol);
+                if (valueDecl && ts.hasSyntacticModifier(valueDecl, 128 /* Abstract */)) {
+                    error(node, ts.Diagnostics.Cannot_create_an_instance_of_an_abstract_class);
+                    return resolveErrorCall(node);
                 }
+                return resolveCall(node, constructSignatures, candidatesOutArray, checkMode, 0 /* None */);
             }
-            checkTypeParameters(node.typeParameters);
-            ts.forEach(node.parameters, checkParameter);
-            // TODO(rbuckton): Should we start checking JSDoc types?
-            if (node.type) {
-                checkSourceElement(node.type);
-            }
-            if (produceDiagnostics) {
-                checkCollisionWithArgumentsInGeneratedCode(node);
-                var returnTypeNode = ts.getEffectiveReturnTypeNode(node);
-                if (noImplicitAny && !returnTypeNode) {
-                    switch (node.kind) {
-                        case 166 /* ConstructSignature */:
-                            error(node, ts.Diagnostics.Construct_signature_which_lacks_return_type_annotation_implicitly_has_an_any_return_type);
-                            break;
-                        case 165 /* CallSignature */:
-                            error(node, ts.Diagnostics.Call_signature_which_lacks_return_type_annotation_implicitly_has_an_any_return_type);
-                            break;
-                    }
-                }
-                if (returnTypeNode) {
-                    var functionFlags_1 = ts.getFunctionFlags(node);
-                    if ((functionFlags_1 & (4 /* Invalid */ | 1 /* Generator */)) === 1 /* Generator */) {
-                        var returnType = getTypeFromTypeNode(returnTypeNode);
-                        if (returnType === voidType) {
-                            error(returnTypeNode, ts.Diagnostics.A_generator_cannot_have_a_void_type_annotation);
-                        }
-                        else {
-                            // Naively, one could check that Generator<any, any, any> is assignable to the return type annotation.
-                            // However, that would not catch the error in the following case.
-                            //
-                            //    interface BadGenerator extends Iterable<number>, Iterator<string> { }
-                            //    function* g(): BadGenerator { } // Iterable and Iterator have different types!
-                            //
-                            var generatorYieldType = getIterationTypeOfGeneratorFunctionReturnType(0 /* Yield */, returnType, (functionFlags_1 & 2 /* Async */) !== 0) || anyType;
-                            var generatorReturnType = getIterationTypeOfGeneratorFunctionReturnType(1 /* Return */, returnType, (functionFlags_1 & 2 /* Async */) !== 0) || generatorYieldType;
-                            var generatorNextType = getIterationTypeOfGeneratorFunctionReturnType(2 /* Next */, returnType, (functionFlags_1 & 2 /* Async */) !== 0) || unknownType;
-                            var generatorInstantiation = createGeneratorReturnType(generatorYieldType, generatorReturnType, generatorNextType, !!(functionFlags_1 & 2 /* Async */));
-                            checkTypeAssignableTo(generatorInstantiation, returnType, returnTypeNode);
-                        }
+            // If expressionType's apparent type is an object type with no construct signatures but
+            // one or more call signatures, the expression is processed as a function call. A compile-time
+            // error occurs if the result of the function call is not Void. The type of the result of the
+            // operation is Any. It is an error to have a Void this type.
+            var callSignatures = getSignaturesOfType(expressionType, 0 /* Call */);
+            if (callSignatures.length) {
+                var signature = resolveCall(node, callSignatures, candidatesOutArray, checkMode, 0 /* None */);
+                if (!noImplicitAny) {
+                    if (signature.declaration && !isJSConstructor(signature.declaration) && getReturnTypeOfSignature(signature) !== voidType) {
+                        error(node, ts.Diagnostics.Only_a_void_function_can_be_called_with_the_new_keyword);
                     }
-                    else if ((functionFlags_1 & 3 /* AsyncGenerator */) === 2 /* Async */) {
-                        checkAsyncFunctionReturnType(node, returnTypeNode);
+                    if (getThisTypeOfSignature(signature) === voidType) {
+                        error(node, ts.Diagnostics.A_function_that_is_called_with_the_new_keyword_cannot_have_a_this_type_that_is_void);
                     }
                 }
-                if (node.kind !== 167 /* IndexSignature */ && node.kind !== 300 /* JSDocFunctionType */) {
-                    registerForUnusedIdentifiersCheck(node);
-                }
+                return signature;
             }
+            invocationError(node.expression, expressionType, 1 /* Construct */);
+            return resolveErrorCall(node);
         }
-        function checkClassForDuplicateDeclarations(node) {
-            var instanceNames = ts.createUnderscoreEscapedMap();
-            var staticNames = ts.createUnderscoreEscapedMap();
-            // instance and static private identifiers share the same scope
-            var privateIdentifiers = ts.createUnderscoreEscapedMap();
-            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                var member = _a[_i];
-                if (member.kind === 162 /* Constructor */) {
-                    for (var _b = 0, _c = member.parameters; _b < _c.length; _b++) {
-                        var param = _c[_b];
-                        if (ts.isParameterPropertyDeclaration(param, member) && !ts.isBindingPattern(param.name)) {
-                            addName(instanceNames, param.name, param.name.escapedText, 3 /* GetOrSetAccessor */);
+        function typeHasProtectedAccessibleBase(target, type) {
+            var baseTypes = getBaseTypes(type);
+            if (!ts.length(baseTypes)) {
+                return false;
+            }
+            var firstBase = baseTypes[0];
+            if (firstBase.flags & 2097152 /* Intersection */) {
+                var types = firstBase.types;
+                var mixinFlags = findMixins(types);
+                var i = 0;
+                for (var _i = 0, _a = firstBase.types; _i < _a.length; _i++) {
+                    var intersectionMember = _a[_i];
+                    // We want to ignore mixin ctors
+                    if (!mixinFlags[i]) {
+                        if (ts.getObjectFlags(intersectionMember) & (1 /* Class */ | 2 /* Interface */)) {
+                            if (intersectionMember.symbol === target) {
+                                return true;
+                            }
+                            if (typeHasProtectedAccessibleBase(target, intersectionMember)) {
+                                return true;
+                            }
                         }
                     }
+                    i++;
                 }
-                else {
-                    var isStatic = ts.hasModifier(member, 32 /* Static */);
-                    var name = member.name;
-                    if (!name) {
-                        return;
-                    }
-                    var names = ts.isPrivateIdentifier(name) ? privateIdentifiers :
-                        isStatic ? staticNames :
-                            instanceNames;
-                    var memberName = name && ts.getPropertyNameForPropertyNameNode(name);
-                    if (memberName) {
-                        switch (member.kind) {
-                            case 163 /* GetAccessor */:
-                                addName(names, name, memberName, 1 /* GetAccessor */);
-                                break;
-                            case 164 /* SetAccessor */:
-                                addName(names, name, memberName, 2 /* SetAccessor */);
-                                break;
-                            case 159 /* PropertyDeclaration */:
-                                addName(names, name, memberName, 3 /* GetOrSetAccessor */);
-                                break;
-                            case 161 /* MethodDeclaration */:
-                                addName(names, name, memberName, 8 /* Method */);
-                                break;
-                        }
+                return false;
+            }
+            if (firstBase.symbol === target) {
+                return true;
+            }
+            return typeHasProtectedAccessibleBase(target, firstBase);
+        }
+        function isConstructorAccessible(node, signature) {
+            if (!signature || !signature.declaration) {
+                return true;
+            }
+            var declaration = signature.declaration;
+            var modifiers = ts.getSelectedEffectiveModifierFlags(declaration, 24 /* NonPublicAccessibilityModifier */);
+            // (1) Public constructors and (2) constructor functions are always accessible.
+            if (!modifiers || declaration.kind !== 166 /* Constructor */) {
+                return true;
+            }
+            var declaringClassDeclaration = ts.getClassLikeDeclarationOfSymbol(declaration.parent.symbol);
+            var declaringClass = getDeclaredTypeOfSymbol(declaration.parent.symbol);
+            // A private or protected constructor can only be instantiated within its own class (or a subclass, for protected)
+            if (!isNodeWithinClass(node, declaringClassDeclaration)) {
+                var containingClass = ts.getContainingClass(node);
+                if (containingClass && modifiers & 16 /* Protected */) {
+                    var containingType = getTypeOfNode(containingClass);
+                    if (typeHasProtectedAccessibleBase(declaration.parent.symbol, containingType)) {
+                        return true;
                     }
                 }
+                if (modifiers & 8 /* Private */) {
+                    error(node, ts.Diagnostics.Constructor_of_class_0_is_private_and_only_accessible_within_the_class_declaration, typeToString(declaringClass));
+                }
+                if (modifiers & 16 /* Protected */) {
+                    error(node, ts.Diagnostics.Constructor_of_class_0_is_protected_and_only_accessible_within_the_class_declaration, typeToString(declaringClass));
+                }
+                return false;
             }
-            function addName(names, location, name, meaning) {
-                var prev = names.get(name);
-                if (prev) {
-                    if (prev & 8 /* Method */) {
-                        if (meaning !== 8 /* Method */) {
-                            error(location, ts.Diagnostics.Duplicate_identifier_0, ts.getTextOfNode(location));
+            return true;
+        }
+        function invocationErrorDetails(errorTarget, apparentType, kind) {
+            var errorInfo;
+            var isCall = kind === 0 /* Call */;
+            var awaitedType = getAwaitedType(apparentType);
+            var maybeMissingAwait = awaitedType && getSignaturesOfType(awaitedType, kind).length > 0;
+            if (apparentType.flags & 1048576 /* Union */) {
+                var types = apparentType.types;
+                var hasSignatures = false;
+                for (var _i = 0, types_19 = types; _i < types_19.length; _i++) {
+                    var constituent = types_19[_i];
+                    var signatures = getSignaturesOfType(constituent, kind);
+                    if (signatures.length !== 0) {
+                        hasSignatures = true;
+                        if (errorInfo) {
+                            // Bail early if we already have an error, no chance of "No constituent of type is callable"
+                            break;
                         }
                     }
-                    else if (prev & meaning) {
-                        error(location, ts.Diagnostics.Duplicate_identifier_0, ts.getTextOfNode(location));
-                    }
                     else {
-                        names.set(name, prev | meaning);
+                        // Error on the first non callable constituent only
+                        if (!errorInfo) {
+                            errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
+                                ts.Diagnostics.Type_0_has_no_call_signatures :
+                                ts.Diagnostics.Type_0_has_no_construct_signatures, typeToString(constituent));
+                            errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
+                                ts.Diagnostics.Not_all_constituents_of_type_0_are_callable :
+                                ts.Diagnostics.Not_all_constituents_of_type_0_are_constructable, typeToString(apparentType));
+                        }
+                        if (hasSignatures) {
+                            // Bail early if we already found a siganture, no chance of "No constituent of type is callable"
+                            break;
+                        }
                     }
                 }
-                else {
-                    names.set(name, meaning);
+                if (!hasSignatures) {
+                    errorInfo = ts.chainDiagnosticMessages(
+                    /* detials */ undefined, isCall ?
+                        ts.Diagnostics.No_constituent_of_type_0_is_callable :
+                        ts.Diagnostics.No_constituent_of_type_0_is_constructable, typeToString(apparentType));
+                }
+                if (!errorInfo) {
+                    errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
+                        ts.Diagnostics.Each_member_of_the_union_type_0_has_signatures_but_none_of_those_signatures_are_compatible_with_each_other :
+                        ts.Diagnostics.Each_member_of_the_union_type_0_has_construct_signatures_but_none_of_those_signatures_are_compatible_with_each_other, typeToString(apparentType));
                 }
             }
-        }
-        /**
-         * Static members being set on a constructor function may conflict with built-in properties
-         * of Function. Esp. in ECMAScript 5 there are non-configurable and non-writable
-         * built-in properties. This check issues a transpile error when a class has a static
-         * member with the same name as a non-writable built-in property.
-         *
-         * @see http://www.ecma-international.org/ecma-262/5.1/#sec-15.3.3
-         * @see http://www.ecma-international.org/ecma-262/5.1/#sec-15.3.5
-         * @see http://www.ecma-international.org/ecma-262/6.0/#sec-properties-of-the-function-constructor
-         * @see http://www.ecma-international.org/ecma-262/6.0/#sec-function-instances
-         */
-        function checkClassForStaticPropertyNameConflicts(node) {
-            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                var member = _a[_i];
-                var memberNameNode = member.name;
-                var isStatic = ts.hasModifier(member, 32 /* Static */);
-                if (isStatic && memberNameNode) {
-                    var memberName = ts.getPropertyNameForPropertyNameNode(memberNameNode);
-                    switch (memberName) {
-                        case "name":
-                        case "length":
-                        case "caller":
-                        case "arguments":
-                        case "prototype":
-                            var message = ts.Diagnostics.Static_property_0_conflicts_with_built_in_property_Function_0_of_constructor_function_1;
-                            var className = getNameOfSymbolAsWritten(getSymbolOfNode(node));
-                            error(memberNameNode, message, memberName, className);
-                            break;
-                    }
+            else {
+                errorInfo = ts.chainDiagnosticMessages(errorInfo, isCall ?
+                    ts.Diagnostics.Type_0_has_no_call_signatures :
+                    ts.Diagnostics.Type_0_has_no_construct_signatures, typeToString(apparentType));
+            }
+            var headMessage = isCall ? ts.Diagnostics.This_expression_is_not_callable : ts.Diagnostics.This_expression_is_not_constructable;
+            // Diagnose get accessors incorrectly called as functions
+            if (ts.isCallExpression(errorTarget.parent) && errorTarget.parent.arguments.length === 0) {
+                var resolvedSymbol = getNodeLinks(errorTarget).resolvedSymbol;
+                if (resolvedSymbol && resolvedSymbol.flags & 32768 /* GetAccessor */) {
+                    headMessage = ts.Diagnostics.This_expression_is_not_callable_because_it_is_a_get_accessor_Did_you_mean_to_use_it_without;
                 }
             }
+            return {
+                messageChain: ts.chainDiagnosticMessages(errorInfo, headMessage),
+                relatedMessage: maybeMissingAwait ? ts.Diagnostics.Did_you_forget_to_use_await : undefined,
+            };
         }
-        function checkObjectTypeForDuplicateDeclarations(node) {
-            var names = ts.createMap();
-            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                var member = _a[_i];
-                if (member.kind === 158 /* PropertySignature */) {
-                    var memberName = void 0;
-                    var name = member.name;
-                    switch (name.kind) {
-                        case 10 /* StringLiteral */:
-                        case 8 /* NumericLiteral */:
-                            memberName = name.text;
-                            break;
-                        case 75 /* Identifier */:
-                            memberName = ts.idText(name);
-                            break;
-                        default:
-                            continue;
-                    }
-                    if (names.get(memberName)) {
-                        error(ts.getNameOfDeclaration(member.symbol.valueDeclaration), ts.Diagnostics.Duplicate_identifier_0, memberName);
-                        error(member.name, ts.Diagnostics.Duplicate_identifier_0, memberName);
-                    }
-                    else {
-                        names.set(memberName, true);
-                    }
-                }
+        function invocationError(errorTarget, apparentType, kind, relatedInformation) {
+            var _a = invocationErrorDetails(errorTarget, apparentType, kind), messageChain = _a.messageChain, relatedInfo = _a.relatedMessage;
+            var diagnostic = ts.createDiagnosticForNodeFromMessageChain(errorTarget, messageChain);
+            if (relatedInfo) {
+                ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(errorTarget, relatedInfo));
+            }
+            if (ts.isCallExpression(errorTarget.parent)) {
+                var _b = getDiagnosticSpanForCallNode(errorTarget.parent, /* doNotIncludeArguments */ true), start = _b.start, length_7 = _b.length;
+                diagnostic.start = start;
+                diagnostic.length = length_7;
             }
+            diagnostics.add(diagnostic);
+            invocationErrorRecovery(apparentType, kind, relatedInformation ? ts.addRelatedInfo(diagnostic, relatedInformation) : diagnostic);
         }
-        function checkTypeForDuplicateIndexSignatures(node) {
-            if (node.kind === 246 /* InterfaceDeclaration */) {
-                var nodeSymbol = getSymbolOfNode(node);
-                // in case of merging interface declaration it is possible that we'll enter this check procedure several times for every declaration
-                // to prevent this run check only for the first declaration of a given kind
-                if (nodeSymbol.declarations.length > 0 && nodeSymbol.declarations[0] !== node) {
+        function invocationErrorRecovery(apparentType, kind, diagnostic) {
+            if (!apparentType.symbol) {
+                return;
+            }
+            var importNode = getSymbolLinks(apparentType.symbol).originatingImport;
+            // Create a diagnostic on the originating import if possible onto which we can attach a quickfix
+            //  An import call expression cannot be rewritten into another form to correct the error - the only solution is to use `.default` at the use-site
+            if (importNode && !ts.isImportCall(importNode)) {
+                var sigs = getSignaturesOfType(getTypeOfSymbol(getSymbolLinks(apparentType.symbol).target), kind);
+                if (!sigs || !sigs.length)
                     return;
+                ts.addRelatedInfo(diagnostic, ts.createDiagnosticForNode(importNode, ts.Diagnostics.Type_originates_at_this_import_A_namespace_style_import_cannot_be_called_or_constructed_and_will_cause_a_failure_at_runtime_Consider_using_a_default_import_or_import_require_here_instead));
+            }
+        }
+        function resolveTaggedTemplateExpression(node, candidatesOutArray, checkMode) {
+            var tagType = checkExpression(node.tag);
+            var apparentType = getApparentType(tagType);
+            if (apparentType === errorType) {
+                // Another error has already been reported
+                return resolveErrorCall(node);
+            }
+            var callSignatures = getSignaturesOfType(apparentType, 0 /* Call */);
+            var numConstructSignatures = getSignaturesOfType(apparentType, 1 /* Construct */).length;
+            if (isUntypedFunctionCall(tagType, apparentType, callSignatures.length, numConstructSignatures)) {
+                return resolveUntypedCall(node);
+            }
+            if (!callSignatures.length) {
+                if (ts.isArrayLiteralExpression(node.parent)) {
+                    var diagnostic = ts.createDiagnosticForNode(node.tag, ts.Diagnostics.It_is_likely_that_you_are_missing_a_comma_to_separate_these_two_template_expressions_They_form_a_tagged_template_expression_which_cannot_be_invoked);
+                    diagnostics.add(diagnostic);
+                    return resolveErrorCall(node);
                 }
+                invocationError(node.tag, apparentType, 0 /* Call */);
+                return resolveErrorCall(node);
             }
-            // TypeScript 1.0 spec (April 2014)
-            // 3.7.4: An object type can contain at most one string index signature and one numeric index signature.
-            // 8.5: A class declaration can have at most one string index member declaration and one numeric index member declaration
-            var indexSymbol = getIndexSymbol(getSymbolOfNode(node));
-            if (indexSymbol) {
-                var seenNumericIndexer = false;
-                var seenStringIndexer = false;
-                for (var _i = 0, _a = indexSymbol.declarations; _i < _a.length; _i++) {
-                    var decl = _a[_i];
-                    var declaration = decl;
-                    if (declaration.parameters.length === 1 && declaration.parameters[0].type) {
-                        switch (declaration.parameters[0].type.kind) {
-                            case 143 /* StringKeyword */:
-                                if (!seenStringIndexer) {
-                                    seenStringIndexer = true;
-                                }
-                                else {
-                                    error(declaration, ts.Diagnostics.Duplicate_string_index_signature);
-                                }
-                                break;
-                            case 140 /* NumberKeyword */:
-                                if (!seenNumericIndexer) {
-                                    seenNumericIndexer = true;
-                                }
-                                else {
-                                    error(declaration, ts.Diagnostics.Duplicate_number_index_signature);
-                                }
-                                break;
-                        }
-                    }
+            return resolveCall(node, callSignatures, candidatesOutArray, checkMode, 0 /* None */);
+        }
+        /**
+         * Gets the localized diagnostic head message to use for errors when resolving a decorator as a call expression.
+         */
+        function getDiagnosticHeadMessageForDecoratorResolution(node) {
+            switch (node.parent.kind) {
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                    return ts.Diagnostics.Unable_to_resolve_signature_of_class_decorator_when_called_as_an_expression;
+                case 160 /* Parameter */:
+                    return ts.Diagnostics.Unable_to_resolve_signature_of_parameter_decorator_when_called_as_an_expression;
+                case 163 /* PropertyDeclaration */:
+                    return ts.Diagnostics.Unable_to_resolve_signature_of_property_decorator_when_called_as_an_expression;
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    return ts.Diagnostics.Unable_to_resolve_signature_of_method_decorator_when_called_as_an_expression;
+                default:
+                    return ts.Debug.fail();
+            }
+        }
+        /**
+         * Resolves a decorator as if it were a call expression.
+         */
+        function resolveDecorator(node, candidatesOutArray, checkMode) {
+            var funcType = checkExpression(node.expression);
+            var apparentType = getApparentType(funcType);
+            if (apparentType === errorType) {
+                return resolveErrorCall(node);
+            }
+            var callSignatures = getSignaturesOfType(apparentType, 0 /* Call */);
+            var numConstructSignatures = getSignaturesOfType(apparentType, 1 /* Construct */).length;
+            if (isUntypedFunctionCall(funcType, apparentType, callSignatures.length, numConstructSignatures)) {
+                return resolveUntypedCall(node);
+            }
+            if (isPotentiallyUncalledDecorator(node, callSignatures)) {
+                var nodeStr = ts.getTextOfNode(node.expression, /*includeTrivia*/ false);
+                error(node, ts.Diagnostics._0_accepts_too_few_arguments_to_be_used_as_a_decorator_here_Did_you_mean_to_call_it_first_and_write_0, nodeStr);
+                return resolveErrorCall(node);
+            }
+            var headMessage = getDiagnosticHeadMessageForDecoratorResolution(node);
+            if (!callSignatures.length) {
+                var errorDetails = invocationErrorDetails(node.expression, apparentType, 0 /* Call */);
+                var messageChain = ts.chainDiagnosticMessages(errorDetails.messageChain, headMessage);
+                var diag = ts.createDiagnosticForNodeFromMessageChain(node.expression, messageChain);
+                if (errorDetails.relatedMessage) {
+                    ts.addRelatedInfo(diag, ts.createDiagnosticForNode(node.expression, errorDetails.relatedMessage));
                 }
+                diagnostics.add(diag);
+                invocationErrorRecovery(apparentType, 0 /* Call */, diag);
+                return resolveErrorCall(node);
             }
+            return resolveCall(node, callSignatures, candidatesOutArray, checkMode, 0 /* None */, headMessage);
         }
-        function checkPropertyDeclaration(node) {
-            // Grammar checking
-            if (!checkGrammarDecoratorsAndModifiers(node) && !checkGrammarProperty(node))
-                checkGrammarComputedPropertyName(node.name);
-            checkVariableLikeDeclaration(node);
-            // Private class fields transformation relies on WeakMaps.
-            if (ts.isPrivateIdentifier(node.name) && languageVersion < 99 /* ESNext */) {
-                for (var lexicalScope = ts.getEnclosingBlockScopeContainer(node); !!lexicalScope; lexicalScope = ts.getEnclosingBlockScopeContainer(lexicalScope)) {
-                    getNodeLinks(lexicalScope).flags |= 67108864 /* ContainsClassWithPrivateIdentifiers */;
+        function createSignatureForJSXIntrinsic(node, result) {
+            var namespace = getJsxNamespaceAt(node);
+            var exports = namespace && getExportsOfSymbol(namespace);
+            // We fake up a SFC signature for each intrinsic, however a more specific per-element signature drawn from the JSX declaration
+            // file would probably be preferable.
+            var typeSymbol = exports && getSymbol(exports, JsxNames.Element, 788968 /* Type */);
+            var returnNode = typeSymbol && nodeBuilder.symbolToEntityName(typeSymbol, 788968 /* Type */, node);
+            var declaration = ts.factory.createFunctionTypeNode(/*typeParameters*/ undefined, [ts.factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotdotdot*/ undefined, "props", /*questionMark*/ undefined, nodeBuilder.typeToTypeNode(result, node))], returnNode ? ts.factory.createTypeReferenceNode(returnNode, /*typeArguments*/ undefined) : ts.factory.createKeywordTypeNode(128 /* AnyKeyword */));
+            var parameterSymbol = createSymbol(1 /* FunctionScopedVariable */, "props");
+            parameterSymbol.type = result;
+            return createSignature(declaration, 
+            /*typeParameters*/ undefined, 
+            /*thisParameter*/ undefined, [parameterSymbol], typeSymbol ? getDeclaredTypeOfSymbol(typeSymbol) : errorType, 
+            /*returnTypePredicate*/ undefined, 1, 0 /* None */);
+        }
+        function resolveJsxOpeningLikeElement(node, candidatesOutArray, checkMode) {
+            if (isJsxIntrinsicIdentifier(node.tagName)) {
+                var result = getIntrinsicAttributesTypeFromJsxOpeningLikeElement(node);
+                var fakeSignature = createSignatureForJSXIntrinsic(node, result);
+                checkTypeAssignableToAndOptionallyElaborate(checkExpressionWithContextualType(node.attributes, getEffectiveFirstArgumentForJsxSignature(fakeSignature, node), /*mapper*/ undefined, 0 /* Normal */), result, node.tagName, node.attributes);
+                if (ts.length(node.typeArguments)) {
+                    ts.forEach(node.typeArguments, checkSourceElement);
+                    diagnostics.add(ts.createDiagnosticForNodeArray(ts.getSourceFileOfNode(node), node.typeArguments, ts.Diagnostics.Expected_0_type_arguments_but_got_1, 0, ts.length(node.typeArguments)));
                 }
+                return fakeSignature;
+            }
+            var exprTypes = checkExpression(node.tagName);
+            var apparentType = getApparentType(exprTypes);
+            if (apparentType === errorType) {
+                return resolveErrorCall(node);
+            }
+            var signatures = getUninstantiatedJsxSignaturesOfType(exprTypes, node);
+            if (isUntypedFunctionCall(exprTypes, apparentType, signatures.length, /*constructSignatures*/ 0)) {
+                return resolveUntypedCall(node);
             }
+            if (signatures.length === 0) {
+                // We found no signatures at all, which is an error
+                error(node.tagName, ts.Diagnostics.JSX_element_type_0_does_not_have_any_construct_or_call_signatures, ts.getTextOfNode(node.tagName));
+                return resolveErrorCall(node);
+            }
+            return resolveCall(node, signatures, candidatesOutArray, checkMode, 0 /* None */);
         }
-        function checkPropertySignature(node) {
-            if (ts.isPrivateIdentifier(node.name)) {
-                error(node, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
+        /**
+         * Sometimes, we have a decorator that could accept zero arguments,
+         * but is receiving too many arguments as part of the decorator invocation.
+         * In those cases, a user may have meant to *call* the expression before using it as a decorator.
+         */
+        function isPotentiallyUncalledDecorator(decorator, signatures) {
+            return signatures.length && ts.every(signatures, function (signature) {
+                return signature.minArgumentCount === 0 &&
+                    !signatureHasRestParameter(signature) &&
+                    signature.parameters.length < getDecoratorArgumentCount(decorator, signature);
+            });
+        }
+        function resolveSignature(node, candidatesOutArray, checkMode) {
+            switch (node.kind) {
+                case 203 /* CallExpression */:
+                    return resolveCallExpression(node, candidatesOutArray, checkMode);
+                case 204 /* NewExpression */:
+                    return resolveNewExpression(node, candidatesOutArray, checkMode);
+                case 205 /* TaggedTemplateExpression */:
+                    return resolveTaggedTemplateExpression(node, candidatesOutArray, checkMode);
+                case 161 /* Decorator */:
+                    return resolveDecorator(node, candidatesOutArray, checkMode);
+                case 275 /* JsxOpeningElement */:
+                case 274 /* JsxSelfClosingElement */:
+                    return resolveJsxOpeningLikeElement(node, candidatesOutArray, checkMode);
             }
-            return checkPropertyDeclaration(node);
+            throw ts.Debug.assertNever(node, "Branch in 'resolveSignature' should be unreachable.");
         }
-        function checkMethodDeclaration(node) {
-            // Grammar checking
-            if (!checkGrammarMethod(node))
-                checkGrammarComputedPropertyName(node.name);
-            if (ts.isPrivateIdentifier(node.name)) {
-                error(node, ts.Diagnostics.A_method_cannot_be_named_with_a_private_identifier);
+        /**
+         * Resolve a signature of a given call-like expression.
+         * @param node a call-like expression to try resolve a signature for
+         * @param candidatesOutArray an array of signature to be filled in by the function. It is passed by signature help in the language service;
+         *                           the function will fill it up with appropriate candidate signatures
+         * @return a signature of the call-like expression or undefined if one can't be found
+         */
+        function getResolvedSignature(node, candidatesOutArray, checkMode) {
+            var links = getNodeLinks(node);
+            // If getResolvedSignature has already been called, we will have cached the resolvedSignature.
+            // However, it is possible that either candidatesOutArray was not passed in the first time,
+            // or that a different candidatesOutArray was passed in. Therefore, we need to redo the work
+            // to correctly fill the candidatesOutArray.
+            var cached = links.resolvedSignature;
+            if (cached && cached !== resolvingSignature && !candidatesOutArray) {
+                return cached;
             }
-            // Grammar checking for modifiers is done inside the function checkGrammarFunctionLikeDeclaration
-            checkFunctionOrMethodDeclaration(node);
-            // Abstract methods cannot have an implementation.
-            // Extra checks are to avoid reporting multiple errors relating to the "abstractness" of the node.
-            if (ts.hasModifier(node, 128 /* Abstract */) && node.kind === 161 /* MethodDeclaration */ && node.body) {
-                error(node, ts.Diagnostics.Method_0_cannot_have_an_implementation_because_it_is_marked_abstract, ts.declarationNameToString(node.name));
+            links.resolvedSignature = resolvingSignature;
+            var result = resolveSignature(node, candidatesOutArray, checkMode || 0 /* Normal */);
+            // When CheckMode.SkipGenericFunctions is set we use resolvingSignature to indicate that call
+            // resolution should be deferred.
+            if (result !== resolvingSignature) {
+                // If signature resolution originated in control flow type analysis (for example to compute the
+                // assigned type in a flow assignment) we don't cache the result as it may be based on temporary
+                // types from the control flow analysis.
+                links.resolvedSignature = flowLoopStart === flowLoopCount ? result : cached;
             }
+            return result;
         }
-        function checkConstructorDeclaration(node) {
-            // Grammar check on signature of constructor and modifier of the constructor is done in checkSignatureDeclaration function.
-            checkSignatureDeclaration(node);
-            // Grammar check for checking only related to constructorDeclaration
-            if (!checkGrammarConstructorTypeParameters(node))
-                checkGrammarConstructorTypeAnnotation(node);
-            checkSourceElement(node.body);
-            var symbol = getSymbolOfNode(node);
-            var firstDeclaration = ts.getDeclarationOfKind(symbol, node.kind);
-            // Only type check the symbol once
-            if (node === firstDeclaration) {
-                checkFunctionOrConstructorSymbol(symbol);
+        /**
+         * Indicates whether a declaration can be treated as a constructor in a JavaScript
+         * file.
+         */
+        function isJSConstructor(node) {
+            var _a;
+            if (!node || !ts.isInJSFile(node)) {
+                return false;
             }
-            // exit early in the case of signature - super checks are not relevant to them
-            if (ts.nodeIsMissing(node.body)) {
-                return;
+            var func = ts.isFunctionDeclaration(node) || ts.isFunctionExpression(node) ? node :
+                ts.isVariableDeclaration(node) && node.initializer && ts.isFunctionExpression(node.initializer) ? node.initializer :
+                    undefined;
+            if (func) {
+                // If the node has a @class tag, treat it like a constructor.
+                if (ts.getJSDocClassTag(node))
+                    return true;
+                // If the symbol of the node has members, treat it like a constructor.
+                var symbol = getSymbolOfNode(func);
+                return !!((_a = symbol === null || symbol === void 0 ? void 0 : symbol.members) === null || _a === void 0 ? void 0 : _a.size);
             }
-            if (!produceDiagnostics) {
-                return;
+            return false;
+        }
+        function mergeJSSymbols(target, source) {
+            var _a, _b;
+            if (source) {
+                var links = getSymbolLinks(source);
+                if (!links.inferredClassSymbol || !links.inferredClassSymbol.has(getSymbolId(target))) {
+                    var inferred = ts.isTransientSymbol(target) ? target : cloneSymbol(target);
+                    inferred.exports = inferred.exports || ts.createSymbolTable();
+                    inferred.members = inferred.members || ts.createSymbolTable();
+                    inferred.flags |= source.flags & 32 /* Class */;
+                    if ((_a = source.exports) === null || _a === void 0 ? void 0 : _a.size) {
+                        mergeSymbolTable(inferred.exports, source.exports);
+                    }
+                    if ((_b = source.members) === null || _b === void 0 ? void 0 : _b.size) {
+                        mergeSymbolTable(inferred.members, source.members);
+                    }
+                    (links.inferredClassSymbol || (links.inferredClassSymbol = new ts.Map())).set(getSymbolId(inferred), inferred);
+                    return inferred;
+                }
+                return links.inferredClassSymbol.get(getSymbolId(target));
             }
-            function isInstancePropertyWithInitializerOrPrivateIdentifierProperty(n) {
-                if (ts.isPrivateIdentifierPropertyDeclaration(n)) {
-                    return true;
+        }
+        function getAssignedClassSymbol(decl) {
+            var _a;
+            var assignmentSymbol = decl && getSymbolOfExpando(decl, /*allowDeclaration*/ true);
+            var prototype = (_a = assignmentSymbol === null || assignmentSymbol === void 0 ? void 0 : assignmentSymbol.exports) === null || _a === void 0 ? void 0 : _a.get("prototype");
+            var init = (prototype === null || prototype === void 0 ? void 0 : prototype.valueDeclaration) && getAssignedJSPrototype(prototype.valueDeclaration);
+            return init ? getSymbolOfNode(init) : undefined;
+        }
+        function getSymbolOfExpando(node, allowDeclaration) {
+            if (!node.parent) {
+                return undefined;
+            }
+            var name;
+            var decl;
+            if (ts.isVariableDeclaration(node.parent) && node.parent.initializer === node) {
+                if (!ts.isInJSFile(node) && !(ts.isVarConst(node.parent) && ts.isFunctionLikeDeclaration(node))) {
+                    return undefined;
                 }
-                return n.kind === 159 /* PropertyDeclaration */ &&
-                    !ts.hasModifier(n, 32 /* Static */) &&
-                    !!n.initializer;
+                name = node.parent.name;
+                decl = node.parent;
             }
-            // TS 1.0 spec (April 2014): 8.3.2
-            // Constructors of classes with no extends clause may not contain super calls, whereas
-            // constructors of derived classes must contain at least one super call somewhere in their function body.
-            var containingClassDecl = node.parent;
-            if (ts.getClassExtendsHeritageElement(containingClassDecl)) {
-                captureLexicalThis(node.parent, containingClassDecl);
-                var classExtendsNull = classDeclarationExtendsNull(containingClassDecl);
-                var superCall = getSuperCallInConstructor(node);
-                if (superCall) {
-                    if (classExtendsNull) {
-                        error(superCall, ts.Diagnostics.A_constructor_cannot_contain_a_super_call_when_its_class_extends_null);
+            else if (ts.isBinaryExpression(node.parent)) {
+                var parentNode = node.parent;
+                var parentNodeOperator = node.parent.operatorToken.kind;
+                if (parentNodeOperator === 62 /* EqualsToken */ && (allowDeclaration || parentNode.right === node)) {
+                    name = parentNode.left;
+                    decl = name;
+                }
+                else if (parentNodeOperator === 56 /* BarBarToken */ || parentNodeOperator === 60 /* QuestionQuestionToken */) {
+                    if (ts.isVariableDeclaration(parentNode.parent) && parentNode.parent.initializer === parentNode) {
+                        name = parentNode.parent.name;
+                        decl = parentNode.parent;
                     }
-                    // The first statement in the body of a constructor (excluding prologue directives) must be a super call
-                    // if both of the following are true:
-                    // - The containing class is a derived class.
-                    // - The constructor declares parameter properties
-                    //   or the containing class declares instance member variables with initializers.
-                    var superCallShouldBeFirst = (compilerOptions.target !== 99 /* ESNext */ || !compilerOptions.useDefineForClassFields) &&
-                        (ts.some(node.parent.members, isInstancePropertyWithInitializerOrPrivateIdentifierProperty) ||
-                            ts.some(node.parameters, function (p) { return ts.hasModifier(p, 92 /* ParameterPropertyModifier */); }));
-                    // Skip past any prologue directives to find the first statement
-                    // to ensure that it was a super call.
-                    if (superCallShouldBeFirst) {
-                        var statements = node.body.statements;
-                        var superCallStatement = void 0;
-                        for (var _i = 0, statements_3 = statements; _i < statements_3.length; _i++) {
-                            var statement = statements_3[_i];
-                            if (statement.kind === 226 /* ExpressionStatement */ && ts.isSuperCall(statement.expression)) {
-                                superCallStatement = statement;
-                                break;
-                            }
-                            if (!ts.isPrologueDirective(statement)) {
-                                break;
-                            }
-                        }
-                        if (!superCallStatement) {
-                            error(node, ts.Diagnostics.A_super_call_must_be_the_first_statement_in_the_constructor_when_a_class_contains_initialized_properties_parameter_properties_or_private_identifiers);
-                        }
+                    else if (ts.isBinaryExpression(parentNode.parent) && parentNode.parent.operatorToken.kind === 62 /* EqualsToken */ && (allowDeclaration || parentNode.parent.right === parentNode)) {
+                        name = parentNode.parent.left;
+                        decl = name;
+                    }
+                    if (!name || !ts.isBindableStaticNameExpression(name) || !ts.isSameEntityName(name, parentNode.left)) {
+                        return undefined;
                     }
                 }
-                else if (!classExtendsNull) {
-                    error(node, ts.Diagnostics.Constructors_for_derived_classes_must_contain_a_super_call);
-                }
             }
+            else if (allowDeclaration && ts.isFunctionDeclaration(node)) {
+                name = node.name;
+                decl = node;
+            }
+            if (!decl || !name || (!allowDeclaration && !ts.getExpandoInitializer(node, ts.isPrototypeAccess(name)))) {
+                return undefined;
+            }
+            return getSymbolOfNode(decl);
         }
-        function checkAccessorDeclaration(node) {
-            if (produceDiagnostics) {
-                // Grammar checking accessors
-                if (!checkGrammarFunctionLikeDeclaration(node) && !checkGrammarAccessor(node))
-                    checkGrammarComputedPropertyName(node.name);
-                checkDecorators(node);
-                checkSignatureDeclaration(node);
-                if (node.kind === 163 /* GetAccessor */) {
-                    if (!(node.flags & 8388608 /* Ambient */) && ts.nodeIsPresent(node.body) && (node.flags & 256 /* HasImplicitReturn */)) {
-                        if (!(node.flags & 512 /* HasExplicitReturn */)) {
-                            error(node.name, ts.Diagnostics.A_get_accessor_must_return_a_value);
-                        }
+        function getAssignedJSPrototype(node) {
+            if (!node.parent) {
+                return false;
+            }
+            var parent = node.parent;
+            while (parent && parent.kind === 201 /* PropertyAccessExpression */) {
+                parent = parent.parent;
+            }
+            if (parent && ts.isBinaryExpression(parent) && ts.isPrototypeAccess(parent.left) && parent.operatorToken.kind === 62 /* EqualsToken */) {
+                var right = ts.getInitializerOfBinaryExpression(parent);
+                return ts.isObjectLiteralExpression(right) && right;
+            }
+        }
+        /**
+         * Syntactically and semantically checks a call or new expression.
+         * @param node The call/new expression to be checked.
+         * @returns On success, the expression's signature's return type. On failure, anyType.
+         */
+        function checkCallExpression(node, checkMode) {
+            var _a;
+            if (!checkGrammarTypeArguments(node, node.typeArguments))
+                checkGrammarArguments(node.arguments);
+            var signature = getResolvedSignature(node, /*candidatesOutArray*/ undefined, checkMode);
+            if (signature === resolvingSignature) {
+                // CheckMode.SkipGenericFunctions is enabled and this is a call to a generic function that
+                // returns a function type. We defer checking and return nonInferrableType.
+                return nonInferrableType;
+            }
+            checkDeprecatedSignature(signature, node);
+            if (node.expression.kind === 105 /* SuperKeyword */) {
+                return voidType;
+            }
+            if (node.kind === 204 /* NewExpression */) {
+                var declaration = signature.declaration;
+                if (declaration &&
+                    declaration.kind !== 166 /* Constructor */ &&
+                    declaration.kind !== 170 /* ConstructSignature */ &&
+                    declaration.kind !== 175 /* ConstructorType */ &&
+                    !ts.isJSDocConstructSignature(declaration) &&
+                    !isJSConstructor(declaration)) {
+                    // When resolved signature is a call signature (and not a construct signature) the result type is any
+                    if (noImplicitAny) {
+                        error(node, ts.Diagnostics.new_expression_whose_target_lacks_a_construct_signature_implicitly_has_an_any_type);
                     }
+                    return anyType;
                 }
-                // Do not use hasDynamicName here, because that returns false for well known symbols.
-                // We want to perform checkComputedPropertyName for all computed properties, including
-                // well known symbols.
-                if (node.name.kind === 154 /* ComputedPropertyName */) {
-                    checkComputedPropertyName(node.name);
-                }
-                if (ts.isPrivateIdentifier(node.name)) {
-                    error(node.name, ts.Diagnostics.An_accessor_cannot_be_named_with_a_private_identifier);
+            }
+            // In JavaScript files, calls to any identifier 'require' are treated as external module imports
+            if (ts.isInJSFile(node) && isCommonJsRequire(node)) {
+                return resolveExternalModuleTypeByLiteral(node.arguments[0]);
+            }
+            var returnType = getReturnTypeOfSignature(signature);
+            // Treat any call to the global 'Symbol' function that is part of a const variable or readonly property
+            // as a fresh unique symbol literal type.
+            if (returnType.flags & 12288 /* ESSymbolLike */ && isSymbolOrSymbolForCall(node)) {
+                return getESSymbolLikeTypeForNode(ts.walkUpParenthesizedExpressions(node.parent));
+            }
+            if (node.kind === 203 /* CallExpression */ && node.parent.kind === 233 /* ExpressionStatement */ &&
+                returnType.flags & 16384 /* Void */ && getTypePredicateOfSignature(signature)) {
+                if (!ts.isDottedName(node.expression)) {
+                    error(node.expression, ts.Diagnostics.Assertions_require_the_call_target_to_be_an_identifier_or_qualified_name);
                 }
-                if (!hasNonBindableDynamicName(node)) {
-                    // TypeScript 1.0 spec (April 2014): 8.4.3
-                    // Accessors for the same member name must specify the same accessibility.
-                    var otherKind = node.kind === 163 /* GetAccessor */ ? 164 /* SetAccessor */ : 163 /* GetAccessor */;
-                    var otherAccessor = ts.getDeclarationOfKind(getSymbolOfNode(node), otherKind);
-                    if (otherAccessor) {
-                        var nodeFlags = ts.getModifierFlags(node);
-                        var otherFlags = ts.getModifierFlags(otherAccessor);
-                        if ((nodeFlags & 28 /* AccessibilityModifier */) !== (otherFlags & 28 /* AccessibilityModifier */)) {
-                            error(node.name, ts.Diagnostics.Getter_and_setter_accessors_do_not_agree_in_visibility);
-                        }
-                        if ((nodeFlags & 128 /* Abstract */) !== (otherFlags & 128 /* Abstract */)) {
-                            error(node.name, ts.Diagnostics.Accessors_must_both_be_abstract_or_non_abstract);
-                        }
-                        // TypeScript 1.0 spec (April 2014): 4.5
-                        // If both accessors include type annotations, the specified types must be identical.
-                        checkAccessorDeclarationTypesIdentical(node, otherAccessor, getAnnotatedAccessorType, ts.Diagnostics.get_and_set_accessor_must_have_the_same_type);
-                        checkAccessorDeclarationTypesIdentical(node, otherAccessor, getThisTypeOfDeclaration, ts.Diagnostics.get_and_set_accessor_must_have_the_same_this_type);
-                    }
+                else if (!getEffectsSignature(node)) {
+                    var diagnostic = error(node.expression, ts.Diagnostics.Assertions_require_every_name_in_the_call_target_to_be_declared_with_an_explicit_type_annotation);
+                    getTypeOfDottedName(node.expression, diagnostic);
                 }
-                var returnType = getTypeOfAccessors(getSymbolOfNode(node));
-                if (node.kind === 163 /* GetAccessor */) {
-                    checkAllCodePathsInNonVoidFunctionReturnOrThrow(node, returnType);
+            }
+            if (ts.isInJSFile(node)) {
+                var jsSymbol = getSymbolOfExpando(node, /*allowDeclaration*/ false);
+                if ((_a = jsSymbol === null || jsSymbol === void 0 ? void 0 : jsSymbol.exports) === null || _a === void 0 ? void 0 : _a.size) {
+                    var jsAssignmentType = createAnonymousType(jsSymbol, jsSymbol.exports, ts.emptyArray, ts.emptyArray, undefined, undefined);
+                    jsAssignmentType.objectFlags |= 16384 /* JSLiteral */;
+                    return getIntersectionType([returnType, jsAssignmentType]);
                 }
             }
-            checkSourceElement(node.body);
+            return returnType;
         }
-        function checkAccessorDeclarationTypesIdentical(first, second, getAnnotatedType, message) {
-            var firstType = getAnnotatedType(first);
-            var secondType = getAnnotatedType(second);
-            if (firstType && secondType && !isTypeIdenticalTo(firstType, secondType)) {
-                error(first, message);
+        function checkDeprecatedSignature(signature, node) {
+            if (signature.declaration && signature.declaration.flags & 134217728 /* Deprecated */) {
+                var suggestionNode = getDeprecatedSuggestionNode(node);
+                errorOrSuggestion(/*isError*/ false, suggestionNode, ts.Diagnostics._0_is_deprecated, signatureToString(signature));
             }
         }
-        function checkMissingDeclaration(node) {
-            checkDecorators(node);
-        }
-        function getEffectiveTypeArguments(node, typeParameters) {
-            return fillMissingTypeArguments(ts.map(node.typeArguments, getTypeFromTypeNode), typeParameters, getMinTypeArgumentCount(typeParameters), ts.isInJSFile(node));
+        function getDeprecatedSuggestionNode(node) {
+            node = ts.skipParentheses(node);
+            switch (node.kind) {
+                case 203 /* CallExpression */:
+                case 161 /* Decorator */:
+                case 204 /* NewExpression */:
+                    return getDeprecatedSuggestionNode(node.expression);
+                case 205 /* TaggedTemplateExpression */:
+                    return getDeprecatedSuggestionNode(node.tag);
+                case 275 /* JsxOpeningElement */:
+                case 274 /* JsxSelfClosingElement */:
+                    return getDeprecatedSuggestionNode(node.tagName);
+                case 202 /* ElementAccessExpression */:
+                    return node.argumentExpression;
+                case 201 /* PropertyAccessExpression */:
+                    return node.name;
+                case 173 /* TypeReference */:
+                    var typeReference = node;
+                    return ts.isQualifiedName(typeReference.typeName) ? typeReference.typeName.right : typeReference;
+                default:
+                    return node;
+            }
         }
-        function checkTypeArgumentConstraints(node, typeParameters) {
-            var typeArguments;
-            var mapper;
-            var result = true;
-            for (var i = 0; i < typeParameters.length; i++) {
-                var constraint = getConstraintOfTypeParameter(typeParameters[i]);
-                if (constraint) {
-                    if (!typeArguments) {
-                        typeArguments = getEffectiveTypeArguments(node, typeParameters);
-                        mapper = createTypeMapper(typeParameters, typeArguments);
-                    }
-                    result = result && checkTypeAssignableTo(typeArguments[i], instantiateType(constraint, mapper), node.typeArguments[i], ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1);
-                }
+        function isSymbolOrSymbolForCall(node) {
+            if (!ts.isCallExpression(node))
+                return false;
+            var left = node.expression;
+            if (ts.isPropertyAccessExpression(left) && left.name.escapedText === "for") {
+                left = left.expression;
             }
-            return result;
+            if (!ts.isIdentifier(left) || left.escapedText !== "Symbol") {
+                return false;
+            }
+            // make sure `Symbol` is the global symbol
+            var globalESSymbol = getGlobalESSymbolConstructorSymbol(/*reportErrors*/ false);
+            if (!globalESSymbol) {
+                return false;
+            }
+            return globalESSymbol === resolveName(left, "Symbol", 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
         }
-        function getTypeParametersForTypeReference(node) {
-            var type = getTypeFromTypeReference(node);
-            if (type !== errorType) {
-                var symbol = getNodeLinks(node).resolvedSymbol;
-                if (symbol) {
-                    return symbol.flags & 524288 /* TypeAlias */ && getSymbolLinks(symbol).typeParameters ||
-                        (ts.getObjectFlags(type) & 4 /* Reference */ ? type.target.localTypeParameters : undefined);
+        function checkImportCallExpression(node) {
+            // Check grammar of dynamic import
+            if (!checkGrammarArguments(node.arguments))
+                checkGrammarImportCallExpression(node);
+            if (node.arguments.length === 0) {
+                return createPromiseReturnType(node, anyType);
+            }
+            var specifier = node.arguments[0];
+            var specifierType = checkExpressionCached(specifier);
+            // Even though multiple arguments is grammatically incorrect, type-check extra arguments for completion
+            for (var i = 1; i < node.arguments.length; ++i) {
+                checkExpressionCached(node.arguments[i]);
+            }
+            if (specifierType.flags & 32768 /* Undefined */ || specifierType.flags & 65536 /* Null */ || !isTypeAssignableTo(specifierType, stringType)) {
+                error(specifier, ts.Diagnostics.Dynamic_import_s_specifier_must_be_of_type_string_but_here_has_type_0, typeToString(specifierType));
+            }
+            // resolveExternalModuleName will return undefined if the moduleReferenceExpression is not a string literal
+            var moduleSymbol = resolveExternalModuleName(node, specifier);
+            if (moduleSymbol) {
+                var esModuleSymbol = resolveESModuleSymbol(moduleSymbol, specifier, /*dontRecursivelyResolve*/ true, /*suppressUsageError*/ false);
+                if (esModuleSymbol) {
+                    return createPromiseReturnType(node, getTypeWithSyntheticDefaultImportType(getTypeOfSymbol(esModuleSymbol), esModuleSymbol, moduleSymbol));
                 }
             }
-            return undefined;
+            return createPromiseReturnType(node, anyType);
         }
-        function checkTypeReferenceNode(node) {
-            checkGrammarTypeArguments(node, node.typeArguments);
-            if (node.kind === 169 /* TypeReference */ && node.typeName.jsdocDotPos !== undefined && !ts.isInJSFile(node) && !ts.isInJSDoc(node)) {
-                grammarErrorAtPos(node, node.typeName.jsdocDotPos, 1, ts.Diagnostics.JSDoc_types_can_only_be_used_inside_documentation_comments);
-            }
-            ts.forEach(node.typeArguments, checkSourceElement);
-            var type = getTypeFromTypeReference(node);
-            if (type !== errorType) {
-                if (node.typeArguments && produceDiagnostics) {
-                    var typeParameters = getTypeParametersForTypeReference(node);
-                    if (typeParameters) {
-                        checkTypeArgumentConstraints(node, typeParameters);
+        function getTypeWithSyntheticDefaultImportType(type, symbol, originalSymbol) {
+            if (allowSyntheticDefaultImports && type && type !== errorType) {
+                var synthType = type;
+                if (!synthType.syntheticType) {
+                    var file = ts.find(originalSymbol.declarations, ts.isSourceFile);
+                    var hasSyntheticDefault = canHaveSyntheticDefault(file, originalSymbol, /*dontResolveAlias*/ false);
+                    if (hasSyntheticDefault) {
+                        var memberTable = ts.createSymbolTable();
+                        var newSymbol = createSymbol(2097152 /* Alias */, "default" /* Default */);
+                        newSymbol.parent = originalSymbol;
+                        newSymbol.nameType = getLiteralType("default");
+                        newSymbol.target = resolveSymbol(symbol);
+                        memberTable.set("default" /* Default */, newSymbol);
+                        var anonymousSymbol = createSymbol(2048 /* TypeLiteral */, "__type" /* Type */);
+                        var defaultContainingObject = createAnonymousType(anonymousSymbol, memberTable, ts.emptyArray, ts.emptyArray, /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined);
+                        anonymousSymbol.type = defaultContainingObject;
+                        synthType.syntheticType = isValidSpreadType(type) ? getSpreadType(type, defaultContainingObject, anonymousSymbol, /*objectFlags*/ 0, /*readonly*/ false) : defaultContainingObject;
+                    }
+                    else {
+                        synthType.syntheticType = type;
                     }
                 }
-                if (type.flags & 32 /* Enum */ && getNodeLinks(node).resolvedSymbol.flags & 8 /* EnumMember */) {
-                    error(node, ts.Diagnostics.Enum_type_0_has_members_with_initializers_that_are_not_literals, typeToString(type));
-                }
+                return synthType.syntheticType;
             }
+            return type;
         }
-        function getTypeArgumentConstraint(node) {
-            var typeReferenceNode = ts.tryCast(node.parent, ts.isTypeReferenceType);
-            if (!typeReferenceNode)
-                return undefined;
-            var typeParameters = getTypeParametersForTypeReference(typeReferenceNode); // TODO: GH#18217
-            var constraint = getConstraintOfTypeParameter(typeParameters[typeReferenceNode.typeArguments.indexOf(node)]);
-            return constraint && instantiateType(constraint, createTypeMapper(typeParameters, getEffectiveTypeArguments(typeReferenceNode, typeParameters)));
-        }
-        function checkTypeQuery(node) {
-            getTypeFromTypeQueryNode(node);
+        function isCommonJsRequire(node) {
+            if (!ts.isRequireCall(node, /*checkArgumentIsStringLiteralLike*/ true)) {
+                return false;
+            }
+            // Make sure require is not a local function
+            if (!ts.isIdentifier(node.expression))
+                return ts.Debug.fail();
+            var resolvedRequire = resolveName(node.expression, node.expression.escapedText, 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true); // TODO: GH#18217
+            if (resolvedRequire === requireSymbol) {
+                return true;
+            }
+            // project includes symbol named 'require' - make sure that it is ambient and local non-alias
+            if (resolvedRequire.flags & 2097152 /* Alias */) {
+                return false;
+            }
+            var targetDeclarationKind = resolvedRequire.flags & 16 /* Function */
+                ? 251 /* FunctionDeclaration */
+                : resolvedRequire.flags & 3 /* Variable */
+                    ? 249 /* VariableDeclaration */
+                    : 0 /* Unknown */;
+            if (targetDeclarationKind !== 0 /* Unknown */) {
+                var decl = ts.getDeclarationOfKind(resolvedRequire, targetDeclarationKind);
+                // function/variable declaration should be ambient
+                return !!decl && !!(decl.flags & 8388608 /* Ambient */);
+            }
+            return false;
         }
-        function checkTypeLiteral(node) {
-            ts.forEach(node.members, checkSourceElement);
-            if (produceDiagnostics) {
-                var type = getTypeFromTypeLiteralOrFunctionOrConstructorTypeNode(node);
-                checkIndexConstraints(type);
-                checkTypeForDuplicateIndexSignatures(node);
-                checkObjectTypeForDuplicateDeclarations(node);
+        function checkTaggedTemplateExpression(node) {
+            if (!checkGrammarTaggedTemplateChain(node))
+                checkGrammarTypeArguments(node, node.typeArguments);
+            if (languageVersion < 2 /* ES2015 */) {
+                checkExternalEmitHelpers(node, 524288 /* MakeTemplateObject */);
             }
+            var signature = getResolvedSignature(node);
+            checkDeprecatedSignature(signature, node);
+            return getReturnTypeOfSignature(signature);
         }
-        function checkArrayType(node) {
-            checkSourceElement(node.elementType);
+        function checkAssertion(node) {
+            return checkAssertionWorker(node, node.type, node.expression);
         }
-        function checkTupleType(node) {
-            var elementTypes = node.elementTypes;
-            var seenOptionalElement = false;
-            for (var i = 0; i < elementTypes.length; i++) {
-                var e = elementTypes[i];
-                if (e.kind === 177 /* RestType */) {
-                    if (i !== elementTypes.length - 1) {
-                        grammarErrorOnNode(e, ts.Diagnostics.A_rest_element_must_be_last_in_a_tuple_type);
-                        break;
-                    }
-                    if (!isArrayType(getTypeFromTypeNode(e.type))) {
-                        error(e, ts.Diagnostics.A_rest_element_type_must_be_an_array_type);
+        function isValidConstAssertionArgument(node) {
+            switch (node.kind) {
+                case 10 /* StringLiteral */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 8 /* NumericLiteral */:
+                case 9 /* BigIntLiteral */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                case 199 /* ArrayLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
+                case 218 /* TemplateExpression */:
+                    return true;
+                case 207 /* ParenthesizedExpression */:
+                    return isValidConstAssertionArgument(node.expression);
+                case 214 /* PrefixUnaryExpression */:
+                    var op = node.operator;
+                    var arg = node.operand;
+                    return op === 40 /* MinusToken */ && (arg.kind === 8 /* NumericLiteral */ || arg.kind === 9 /* BigIntLiteral */) ||
+                        op === 39 /* PlusToken */ && arg.kind === 8 /* NumericLiteral */;
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    var expr = node.expression;
+                    if (ts.isIdentifier(expr)) {
+                        var symbol = getSymbolAtLocation(expr);
+                        if (symbol && symbol.flags & 2097152 /* Alias */) {
+                            symbol = resolveAlias(symbol);
+                        }
+                        return !!(symbol && (symbol.flags & 384 /* Enum */) && getEnumKind(symbol) === 1 /* Literal */);
                     }
+            }
+            return false;
+        }
+        function checkAssertionWorker(errNode, type, expression, checkMode) {
+            var exprType = checkExpression(expression, checkMode);
+            if (ts.isConstTypeReference(type)) {
+                if (!isValidConstAssertionArgument(expression)) {
+                    error(expression, ts.Diagnostics.A_const_assertions_can_only_be_applied_to_references_to_enum_members_or_string_number_boolean_array_or_object_literals);
                 }
-                else if (e.kind === 176 /* OptionalType */) {
-                    seenOptionalElement = true;
-                }
-                else if (seenOptionalElement) {
-                    grammarErrorOnNode(e, ts.Diagnostics.A_required_element_cannot_follow_an_optional_element);
-                    break;
+                return getRegularTypeOfLiteralType(exprType);
+            }
+            checkSourceElement(type);
+            exprType = getRegularTypeOfObjectLiteral(getBaseTypeOfLiteralType(exprType));
+            var targetType = getTypeFromTypeNode(type);
+            if (produceDiagnostics && targetType !== errorType) {
+                var widenedType = getWidenedType(exprType);
+                if (!isTypeComparableTo(targetType, widenedType)) {
+                    checkTypeComparableTo(exprType, targetType, errNode, ts.Diagnostics.Conversion_of_type_0_to_type_1_may_be_a_mistake_because_neither_type_sufficiently_overlaps_with_the_other_If_this_was_intentional_convert_the_expression_to_unknown_first);
                 }
             }
-            ts.forEach(node.elementTypes, checkSourceElement);
+            return targetType;
         }
-        function checkUnionOrIntersectionType(node) {
-            ts.forEach(node.types, checkSourceElement);
+        function checkNonNullChain(node) {
+            var leftType = checkExpression(node.expression);
+            var nonOptionalType = getOptionalExpressionType(leftType, node.expression);
+            return propagateOptionalTypeMarker(getNonNullableType(nonOptionalType), node, nonOptionalType !== leftType);
         }
-        function checkIndexedAccessIndexType(type, accessNode) {
-            if (!(type.flags & 8388608 /* IndexedAccess */)) {
-                return type;
+        function checkNonNullAssertion(node) {
+            return node.flags & 32 /* OptionalChain */ ? checkNonNullChain(node) :
+                getNonNullableType(checkExpression(node.expression));
+        }
+        function checkMetaProperty(node) {
+            checkGrammarMetaProperty(node);
+            if (node.keywordToken === 102 /* NewKeyword */) {
+                return checkNewTargetMetaProperty(node);
             }
-            // Check if the index type is assignable to 'keyof T' for the object type.
-            var objectType = type.objectType;
-            var indexType = type.indexType;
-            if (isTypeAssignableTo(indexType, getIndexType(objectType, /*stringsOnly*/ false))) {
-                if (accessNode.kind === 195 /* ElementAccessExpression */ && ts.isAssignmentTarget(accessNode) &&
-                    ts.getObjectFlags(objectType) & 32 /* Mapped */ && getMappedTypeModifiers(objectType) & 1 /* IncludeReadonly */) {
-                    error(accessNode, ts.Diagnostics.Index_signature_in_type_0_only_permits_reading, typeToString(objectType));
-                }
-                return type;
+            if (node.keywordToken === 99 /* ImportKeyword */) {
+                return checkImportMetaProperty(node);
             }
-            // Check if we're indexing with a numeric type and if either object or index types
-            // is a generic type with a constraint that has a numeric index signature.
-            var apparentObjectType = getApparentType(objectType);
-            if (getIndexInfoOfType(apparentObjectType, 1 /* Number */) && isTypeAssignableToKind(indexType, 296 /* NumberLike */)) {
-                return type;
+            return ts.Debug.assertNever(node.keywordToken);
+        }
+        function checkNewTargetMetaProperty(node) {
+            var container = ts.getNewTargetContainer(node);
+            if (!container) {
+                error(node, ts.Diagnostics.Meta_property_0_is_only_allowed_in_the_body_of_a_function_declaration_function_expression_or_constructor, "new.target");
+                return errorType;
             }
-            if (isGenericObjectType(objectType)) {
-                var propertyName_1 = getPropertyNameFromIndex(indexType, accessNode);
-                if (propertyName_1) {
-                    var propertySymbol = forEachType(apparentObjectType, function (t) { return getPropertyOfType(t, propertyName_1); });
-                    if (propertySymbol && ts.getDeclarationModifierFlagsFromSymbol(propertySymbol) & 24 /* NonPublicAccessibilityModifier */) {
-                        error(accessNode, ts.Diagnostics.Private_or_protected_member_0_cannot_be_accessed_on_a_type_parameter, ts.unescapeLeadingUnderscores(propertyName_1));
-                        return errorType;
-                    }
-                }
+            else if (container.kind === 166 /* Constructor */) {
+                var symbol = getSymbolOfNode(container.parent);
+                return getTypeOfSymbol(symbol);
+            }
+            else {
+                var symbol = getSymbolOfNode(container);
+                return getTypeOfSymbol(symbol);
             }
-            error(accessNode, ts.Diagnostics.Type_0_cannot_be_used_to_index_type_1, typeToString(indexType), typeToString(objectType));
-            return errorType;
         }
-        function checkIndexedAccessType(node) {
-            checkSourceElement(node.objectType);
-            checkSourceElement(node.indexType);
-            checkIndexedAccessIndexType(getTypeFromIndexedAccessTypeNode(node), node);
+        function checkImportMetaProperty(node) {
+            if (moduleKind !== ts.ModuleKind.ES2020 && moduleKind !== ts.ModuleKind.ESNext && moduleKind !== ts.ModuleKind.System) {
+                error(node, ts.Diagnostics.The_import_meta_meta_property_is_only_allowed_when_the_module_option_is_es2020_esnext_or_system);
+            }
+            var file = ts.getSourceFileOfNode(node);
+            ts.Debug.assert(!!(file.flags & 2097152 /* PossiblyContainsImportMeta */), "Containing file is missing import meta node flag.");
+            ts.Debug.assert(!!file.externalModuleIndicator, "Containing file should be a module.");
+            return node.name.escapedText === "meta" ? getGlobalImportMetaType() : errorType;
         }
-        function checkMappedType(node) {
-            checkSourceElement(node.typeParameter);
-            checkSourceElement(node.type);
-            if (!node.type) {
-                reportImplicitAny(node, anyType);
+        function getTypeOfParameter(symbol) {
+            var type = getTypeOfSymbol(symbol);
+            if (strictNullChecks) {
+                var declaration = symbol.valueDeclaration;
+                if (declaration && ts.hasInitializer(declaration)) {
+                    return getOptionalType(type);
+                }
             }
-            var type = getTypeFromMappedTypeNode(node);
-            var constraintType = getConstraintTypeFromMappedType(type);
-            checkTypeAssignableTo(constraintType, keyofConstraintType, ts.getEffectiveConstraintOfTypeParameter(node.typeParameter));
+            return type;
         }
-        function checkThisType(node) {
-            getTypeFromThisTypeNode(node);
+        function getTupleElementLabel(d) {
+            ts.Debug.assert(ts.isIdentifier(d.name)); // Parameter declarations could be binding patterns, but we only allow identifier names
+            return d.name.escapedText;
         }
-        function checkTypeOperator(node) {
-            checkGrammarTypeOperatorNode(node);
-            checkSourceElement(node.type);
+        function getParameterNameAtPosition(signature, pos, overrideRestType) {
+            var paramCount = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
+            if (pos < paramCount) {
+                return signature.parameters[pos].escapedName;
+            }
+            var restParameter = signature.parameters[paramCount] || unknownSymbol;
+            var restType = overrideRestType || getTypeOfSymbol(restParameter);
+            if (isTupleType(restType)) {
+                var associatedNames = restType.target.labeledElementDeclarations;
+                var index = pos - paramCount;
+                return associatedNames && getTupleElementLabel(associatedNames[index]) || restParameter.escapedName + "_" + index;
+            }
+            return restParameter.escapedName;
         }
-        function checkConditionalType(node) {
-            ts.forEachChild(node, checkSourceElement);
+        function isValidDeclarationForTupleLabel(d) {
+            return d.kind === 192 /* NamedTupleMember */ || (ts.isParameter(d) && d.name && ts.isIdentifier(d.name));
         }
-        function checkInferType(node) {
-            if (!ts.findAncestor(node, function (n) { return n.parent && n.parent.kind === 180 /* ConditionalType */ && n.parent.extendsType === n; })) {
-                grammarErrorOnNode(node, ts.Diagnostics.infer_declarations_are_only_permitted_in_the_extends_clause_of_a_conditional_type);
+        function getNameableDeclarationAtPosition(signature, pos) {
+            var paramCount = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
+            if (pos < paramCount) {
+                var decl = signature.parameters[pos].valueDeclaration;
+                return decl && isValidDeclarationForTupleLabel(decl) ? decl : undefined;
             }
-            checkSourceElement(node.typeParameter);
-            registerForUnusedIdentifiersCheck(node);
-        }
-        function checkImportType(node) {
-            checkSourceElement(node.argument);
-            getTypeFromTypeNode(node);
+            var restParameter = signature.parameters[paramCount] || unknownSymbol;
+            var restType = getTypeOfSymbol(restParameter);
+            if (isTupleType(restType)) {
+                var associatedNames = restType.target.labeledElementDeclarations;
+                var index = pos - paramCount;
+                return associatedNames && associatedNames[index];
+            }
+            return restParameter.valueDeclaration && isValidDeclarationForTupleLabel(restParameter.valueDeclaration) ? restParameter.valueDeclaration : undefined;
         }
-        function isPrivateWithinAmbient(node) {
-            return (ts.hasModifier(node, 8 /* Private */) || ts.isPrivateIdentifierPropertyDeclaration(node)) && !!(node.flags & 8388608 /* Ambient */);
+        function getTypeAtPosition(signature, pos) {
+            return tryGetTypeAtPosition(signature, pos) || anyType;
         }
-        function getEffectiveDeclarationFlags(n, flagsToCheck) {
-            var flags = ts.getCombinedModifierFlags(n);
-            // children of classes (even ambient classes) should not be marked as ambient or export
-            // because those flags have no useful semantics there.
-            if (n.parent.kind !== 246 /* InterfaceDeclaration */ &&
-                n.parent.kind !== 245 /* ClassDeclaration */ &&
-                n.parent.kind !== 214 /* ClassExpression */ &&
-                n.flags & 8388608 /* Ambient */) {
-                if (!(flags & 2 /* Ambient */) && !(ts.isModuleBlock(n.parent) && ts.isModuleDeclaration(n.parent.parent) && ts.isGlobalScopeAugmentation(n.parent.parent))) {
-                    // It is nested in an ambient context, which means it is automatically exported
-                    flags |= 1 /* Export */;
+        function tryGetTypeAtPosition(signature, pos) {
+            var paramCount = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
+            if (pos < paramCount) {
+                return getTypeOfParameter(signature.parameters[pos]);
+            }
+            if (signatureHasRestParameter(signature)) {
+                // We want to return the value undefined for an out of bounds parameter position,
+                // so we need to check bounds here before calling getIndexedAccessType (which
+                // otherwise would return the type 'undefined').
+                var restType = getTypeOfSymbol(signature.parameters[paramCount]);
+                var index = pos - paramCount;
+                if (!isTupleType(restType) || restType.target.hasRestElement || index < restType.target.fixedLength) {
+                    return getIndexedAccessType(restType, getLiteralType(index));
                 }
-                flags |= 2 /* Ambient */;
             }
-            return flags & flagsToCheck;
+            return undefined;
         }
-        function checkFunctionOrConstructorSymbol(symbol) {
-            if (!produceDiagnostics) {
-                return;
-            }
-            function getCanonicalOverload(overloads, implementation) {
-                // Consider the canonical set of flags to be the flags of the bodyDeclaration or the first declaration
-                // Error on all deviations from this canonical set of flags
-                // The caveat is that if some overloads are defined in lib.d.ts, we don't want to
-                // report the errors on those. To achieve this, we will say that the implementation is
-                // the canonical signature only if it is in the same container as the first overload
-                var implementationSharesContainerWithFirstOverload = implementation !== undefined && implementation.parent === overloads[0].parent;
-                return implementationSharesContainerWithFirstOverload ? implementation : overloads[0];
+        function getRestTypeAtPosition(source, pos) {
+            var parameterCount = getParameterCount(source);
+            var minArgumentCount = getMinArgumentCount(source);
+            var restType = getEffectiveRestType(source);
+            if (restType && pos >= parameterCount - 1) {
+                return pos === parameterCount - 1 ? restType : createArrayType(getIndexedAccessType(restType, numberType));
             }
-            function checkFlagAgreementBetweenOverloads(overloads, implementation, flagsToCheck, someOverloadFlags, allOverloadFlags) {
-                // Error if some overloads have a flag that is not shared by all overloads. To find the
-                // deviations, we XOR someOverloadFlags with allOverloadFlags
-                var someButNotAllOverloadFlags = someOverloadFlags ^ allOverloadFlags;
-                if (someButNotAllOverloadFlags !== 0) {
-                    var canonicalFlags_1 = getEffectiveDeclarationFlags(getCanonicalOverload(overloads, implementation), flagsToCheck);
-                    ts.forEach(overloads, function (o) {
-                        var deviation = getEffectiveDeclarationFlags(o, flagsToCheck) ^ canonicalFlags_1;
-                        if (deviation & 1 /* Export */) {
-                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_exported_or_non_exported);
-                        }
-                        else if (deviation & 2 /* Ambient */) {
-                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_ambient_or_non_ambient);
-                        }
-                        else if (deviation & (8 /* Private */ | 16 /* Protected */)) {
-                            error(ts.getNameOfDeclaration(o) || o, ts.Diagnostics.Overload_signatures_must_all_be_public_private_or_protected);
-                        }
-                        else if (deviation & 128 /* Abstract */) {
-                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_abstract_or_non_abstract);
-                        }
-                    });
+            var types = [];
+            var flags = [];
+            var names = [];
+            for (var i = pos; i < parameterCount; i++) {
+                if (!restType || i < parameterCount - 1) {
+                    types.push(getTypeAtPosition(source, i));
+                    flags.push(i < minArgumentCount ? 1 /* Required */ : 2 /* Optional */);
                 }
-            }
-            function checkQuestionTokenAgreementBetweenOverloads(overloads, implementation, someHaveQuestionToken, allHaveQuestionToken) {
-                if (someHaveQuestionToken !== allHaveQuestionToken) {
-                    var canonicalHasQuestionToken_1 = ts.hasQuestionToken(getCanonicalOverload(overloads, implementation));
-                    ts.forEach(overloads, function (o) {
-                        var deviation = ts.hasQuestionToken(o) !== canonicalHasQuestionToken_1;
-                        if (deviation) {
-                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_optional_or_required);
-                        }
-                    });
+                else {
+                    types.push(restType);
+                    flags.push(8 /* Variadic */);
+                }
+                var name = getNameableDeclarationAtPosition(source, i);
+                if (name) {
+                    names.push(name);
                 }
             }
-            var flagsToCheck = 1 /* Export */ | 2 /* Ambient */ | 8 /* Private */ | 16 /* Protected */ | 128 /* Abstract */;
-            var someNodeFlags = 0 /* None */;
-            var allNodeFlags = flagsToCheck;
-            var someHaveQuestionToken = false;
-            var allHaveQuestionToken = true;
-            var hasOverloads = false;
-            var bodyDeclaration;
-            var lastSeenNonAmbientDeclaration;
-            var previousDeclaration;
-            var declarations = symbol.declarations;
-            var isConstructor = (symbol.flags & 16384 /* Constructor */) !== 0;
-            function reportImplementationExpectedError(node) {
-                if (node.name && ts.nodeIsMissing(node.name)) {
-                    return;
+            return createTupleType(types, flags, /*readonly*/ false, ts.length(names) === ts.length(types) ? names : undefined);
+        }
+        function getParameterCount(signature) {
+            var length = signature.parameters.length;
+            if (signatureHasRestParameter(signature)) {
+                var restType = getTypeOfSymbol(signature.parameters[length - 1]);
+                if (isTupleType(restType)) {
+                    return length + restType.target.fixedLength - (restType.target.hasRestElement ? 0 : 1);
                 }
-                var seen = false;
-                var subsequentNode = ts.forEachChild(node.parent, function (c) {
-                    if (seen) {
-                        return c;
-                    }
-                    else {
-                        seen = c === node;
-                    }
-                });
-                // We may be here because of some extra nodes between overloads that could not be parsed into a valid node.
-                // In this case the subsequent node is not really consecutive (.pos !== node.end), and we must ignore it here.
-                if (subsequentNode && subsequentNode.pos === node.end) {
-                    if (subsequentNode.kind === node.kind) {
-                        var errorNode_1 = subsequentNode.name || subsequentNode;
-                        var subsequentName = subsequentNode.name;
-                        if (node.name && subsequentName && (
-                        // both are private identifiers
-                        ts.isPrivateIdentifier(node.name) && ts.isPrivateIdentifier(subsequentName) && node.name.escapedText === subsequentName.escapedText ||
-                            // Both are computed property names
-                            // TODO: GH#17345: These are methods, so handle computed name case. (`Always allowing computed property names is *not* the correct behavior!)
-                            ts.isComputedPropertyName(node.name) && ts.isComputedPropertyName(subsequentName) ||
-                            // Both are literal property names that are the same.
-                            ts.isPropertyNameLiteral(node.name) && ts.isPropertyNameLiteral(subsequentName) &&
-                                ts.getEscapedTextOfIdentifierOrLiteral(node.name) === ts.getEscapedTextOfIdentifierOrLiteral(subsequentName))) {
-                            var reportError = (node.kind === 161 /* MethodDeclaration */ || node.kind === 160 /* MethodSignature */) &&
-                                ts.hasModifier(node, 32 /* Static */) !== ts.hasModifier(subsequentNode, 32 /* Static */);
-                            // we can get here in two cases
-                            // 1. mixed static and instance class members
-                            // 2. something with the same name was defined before the set of overloads that prevents them from merging
-                            // here we'll report error only for the first case since for second we should already report error in binder
-                            if (reportError) {
-                                var diagnostic = ts.hasModifier(node, 32 /* Static */) ? ts.Diagnostics.Function_overload_must_be_static : ts.Diagnostics.Function_overload_must_not_be_static;
-                                error(errorNode_1, diagnostic);
-                            }
-                            return;
-                        }
-                        if (ts.nodeIsPresent(subsequentNode.body)) {
-                            error(errorNode_1, ts.Diagnostics.Function_implementation_name_must_be_0, ts.declarationNameToString(node.name));
-                            return;
+            }
+            return length;
+        }
+        function getMinArgumentCount(signature, flags) {
+            var strongArityForUntypedJS = flags & 1 /* StrongArityForUntypedJS */;
+            var voidIsNonOptional = flags & 2 /* VoidIsNonOptional */;
+            if (voidIsNonOptional || signature.resolvedMinArgumentCount === undefined) {
+                var minArgumentCount = void 0;
+                if (signatureHasRestParameter(signature)) {
+                    var restType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
+                    if (isTupleType(restType)) {
+                        var firstOptionalIndex = ts.findIndex(restType.target.elementFlags, function (f) { return !(f & 1 /* Required */); });
+                        var requiredCount = firstOptionalIndex < 0 ? restType.target.fixedLength : firstOptionalIndex;
+                        if (requiredCount > 0) {
+                            minArgumentCount = signature.parameters.length - 1 + requiredCount;
                         }
                     }
                 }
-                var errorNode = node.name || node;
-                if (isConstructor) {
-                    error(errorNode, ts.Diagnostics.Constructor_implementation_is_missing);
-                }
-                else {
-                    // Report different errors regarding non-consecutive blocks of declarations depending on whether
-                    // the node in question is abstract.
-                    if (ts.hasModifier(node, 128 /* Abstract */)) {
-                        error(errorNode, ts.Diagnostics.All_declarations_of_an_abstract_method_must_be_consecutive);
-                    }
-                    else {
-                        error(errorNode, ts.Diagnostics.Function_implementation_is_missing_or_not_immediately_following_the_declaration);
+                if (minArgumentCount === undefined) {
+                    if (!strongArityForUntypedJS && signature.flags & 16 /* IsUntypedSignatureInJSFile */) {
+                        return 0;
                     }
+                    minArgumentCount = signature.minArgumentCount;
                 }
-            }
-            var duplicateFunctionDeclaration = false;
-            var multipleConstructorImplementation = false;
-            var hasNonAmbientClass = false;
-            for (var _i = 0, declarations_4 = declarations; _i < declarations_4.length; _i++) {
-                var current = declarations_4[_i];
-                var node = current;
-                var inAmbientContext = node.flags & 8388608 /* Ambient */;
-                var inAmbientContextOrInterface = node.parent.kind === 246 /* InterfaceDeclaration */ || node.parent.kind === 173 /* TypeLiteral */ || inAmbientContext;
-                if (inAmbientContextOrInterface) {
-                    // check if declarations are consecutive only if they are non-ambient
-                    // 1. ambient declarations can be interleaved
-                    // i.e. this is legal
-                    //     declare function foo();
-                    //     declare function bar();
-                    //     declare function foo();
-                    // 2. mixing ambient and non-ambient declarations is a separate error that will be reported - do not want to report an extra one
-                    previousDeclaration = undefined;
-                }
-                if ((node.kind === 245 /* ClassDeclaration */ || node.kind === 214 /* ClassExpression */) && !inAmbientContext) {
-                    hasNonAmbientClass = true;
+                if (voidIsNonOptional) {
+                    return minArgumentCount;
                 }
-                if (node.kind === 244 /* FunctionDeclaration */ || node.kind === 161 /* MethodDeclaration */ || node.kind === 160 /* MethodSignature */ || node.kind === 162 /* Constructor */) {
-                    var currentNodeFlags = getEffectiveDeclarationFlags(node, flagsToCheck);
-                    someNodeFlags |= currentNodeFlags;
-                    allNodeFlags &= currentNodeFlags;
-                    someHaveQuestionToken = someHaveQuestionToken || ts.hasQuestionToken(node);
-                    allHaveQuestionToken = allHaveQuestionToken && ts.hasQuestionToken(node);
-                    if (ts.nodeIsPresent(node.body) && bodyDeclaration) {
-                        if (isConstructor) {
-                            multipleConstructorImplementation = true;
-                        }
-                        else {
-                            duplicateFunctionDeclaration = true;
-                        }
-                    }
-                    else if (previousDeclaration && previousDeclaration.parent === node.parent && previousDeclaration.end !== node.pos) {
-                        reportImplementationExpectedError(previousDeclaration);
-                    }
-                    if (ts.nodeIsPresent(node.body)) {
-                        if (!bodyDeclaration) {
-                            bodyDeclaration = node;
-                        }
-                    }
-                    else {
-                        hasOverloads = true;
-                    }
-                    previousDeclaration = node;
-                    if (!inAmbientContextOrInterface) {
-                        lastSeenNonAmbientDeclaration = node;
+                for (var i = minArgumentCount - 1; i >= 0; i--) {
+                    var type = getTypeAtPosition(signature, i);
+                    if (filterType(type, acceptsVoid).flags & 131072 /* Never */) {
+                        break;
                     }
+                    minArgumentCount = i;
                 }
+                signature.resolvedMinArgumentCount = minArgumentCount;
             }
-            if (multipleConstructorImplementation) {
-                ts.forEach(declarations, function (declaration) {
-                    error(declaration, ts.Diagnostics.Multiple_constructor_implementations_are_not_allowed);
-                });
+            return signature.resolvedMinArgumentCount;
+        }
+        function hasEffectiveRestParameter(signature) {
+            if (signatureHasRestParameter(signature)) {
+                var restType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
+                return !isTupleType(restType) || restType.target.hasRestElement;
             }
-            if (duplicateFunctionDeclaration) {
-                ts.forEach(declarations, function (declaration) {
-                    error(ts.getNameOfDeclaration(declaration), ts.Diagnostics.Duplicate_function_implementation);
-                });
+            return false;
+        }
+        function getEffectiveRestType(signature) {
+            if (signatureHasRestParameter(signature)) {
+                var restType = getTypeOfSymbol(signature.parameters[signature.parameters.length - 1]);
+                if (!isTupleType(restType)) {
+                    return restType;
+                }
+                if (restType.target.hasRestElement) {
+                    return sliceTupleType(restType, restType.target.fixedLength);
+                }
             }
-            if (hasNonAmbientClass && !isConstructor && symbol.flags & 16 /* Function */) {
-                // A non-ambient class cannot be an implementation for a non-constructor function/class merge
-                // TODO: The below just replicates our older error from when classes and functions were
-                // entirely unable to merge - a more helpful message like "Class declaration cannot implement overload list"
-                // might be warranted. :shrug:
-                ts.forEach(declarations, function (declaration) {
-                    addDuplicateDeclarationError(declaration, ts.Diagnostics.Duplicate_identifier_0, ts.symbolName(symbol), declarations);
-                });
+            return undefined;
+        }
+        function getNonArrayRestType(signature) {
+            var restType = getEffectiveRestType(signature);
+            return restType && !isArrayType(restType) && !isTypeAny(restType) && (getReducedType(restType).flags & 131072 /* Never */) === 0 ? restType : undefined;
+        }
+        function getTypeOfFirstParameterOfSignature(signature) {
+            return getTypeOfFirstParameterOfSignatureWithFallback(signature, neverType);
+        }
+        function getTypeOfFirstParameterOfSignatureWithFallback(signature, fallbackType) {
+            return signature.parameters.length > 0 ? getTypeAtPosition(signature, 0) : fallbackType;
+        }
+        function inferFromAnnotatedParameters(signature, context, inferenceContext) {
+            var len = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
+            for (var i = 0; i < len; i++) {
+                var declaration = signature.parameters[i].valueDeclaration;
+                if (declaration.type) {
+                    var typeNode = ts.getEffectiveTypeAnnotationNode(declaration);
+                    if (typeNode) {
+                        inferTypes(inferenceContext.inferences, getTypeFromTypeNode(typeNode), getTypeAtPosition(context, i));
+                    }
+                }
             }
-            // Abstract methods can't have an implementation -- in particular, they don't need one.
-            if (lastSeenNonAmbientDeclaration && !lastSeenNonAmbientDeclaration.body &&
-                !ts.hasModifier(lastSeenNonAmbientDeclaration, 128 /* Abstract */) && !lastSeenNonAmbientDeclaration.questionToken) {
-                reportImplementationExpectedError(lastSeenNonAmbientDeclaration);
+            var restType = getEffectiveRestType(context);
+            if (restType && restType.flags & 262144 /* TypeParameter */) {
+                // The contextual signature has a generic rest parameter. We first instantiate the contextual
+                // signature (without fixing type parameters) and assign types to contextually typed parameters.
+                var instantiatedContext = instantiateSignature(context, inferenceContext.nonFixingMapper);
+                assignContextualParameterTypes(signature, instantiatedContext);
+                // We then infer from a tuple type representing the parameters that correspond to the contextual
+                // rest parameter.
+                var restPos = getParameterCount(context) - 1;
+                inferTypes(inferenceContext.inferences, getRestTypeAtPosition(signature, restPos), restType);
             }
-            if (hasOverloads) {
-                checkFlagAgreementBetweenOverloads(declarations, bodyDeclaration, flagsToCheck, someNodeFlags, allNodeFlags);
-                checkQuestionTokenAgreementBetweenOverloads(declarations, bodyDeclaration, someHaveQuestionToken, allHaveQuestionToken);
-                if (bodyDeclaration) {
-                    var signatures = getSignaturesOfSymbol(symbol);
-                    var bodySignature = getSignatureFromDeclaration(bodyDeclaration);
-                    for (var _a = 0, signatures_10 = signatures; _a < signatures_10.length; _a++) {
-                        var signature = signatures_10[_a];
-                        if (!isImplementationCompatibleWithOverload(bodySignature, signature)) {
-                            ts.addRelatedInfo(error(signature.declaration, ts.Diagnostics.This_overload_signature_is_not_compatible_with_its_implementation_signature), ts.createDiagnosticForNode(bodyDeclaration, ts.Diagnostics.The_implementation_signature_is_declared_here));
-                            break;
-                        }
+        }
+        function assignContextualParameterTypes(signature, context) {
+            signature.typeParameters = context.typeParameters;
+            if (context.thisParameter) {
+                var parameter = signature.thisParameter;
+                if (!parameter || parameter.valueDeclaration && !parameter.valueDeclaration.type) {
+                    if (!parameter) {
+                        signature.thisParameter = createSymbolWithType(context.thisParameter, /*type*/ undefined);
                     }
+                    assignParameterType(signature.thisParameter, getTypeOfSymbol(context.thisParameter));
                 }
             }
-        }
-        function checkExportsOnMergedDeclarations(node) {
-            if (!produceDiagnostics) {
-                return;
+            var len = signature.parameters.length - (signatureHasRestParameter(signature) ? 1 : 0);
+            for (var i = 0; i < len; i++) {
+                var parameter = signature.parameters[i];
+                if (!ts.getEffectiveTypeAnnotationNode(parameter.valueDeclaration)) {
+                    var contextualParameterType = tryGetTypeAtPosition(context, i);
+                    assignParameterType(parameter, contextualParameterType);
+                }
             }
-            // if localSymbol is defined on node then node itself is exported - check is required
-            var symbol = node.localSymbol;
-            if (!symbol) {
-                // local symbol is undefined => this declaration is non-exported.
-                // however symbol might contain other declarations that are exported
-                symbol = getSymbolOfNode(node);
-                if (!symbol.exportSymbol) {
-                    // this is a pure local symbol (all declarations are non-exported) - no need to check anything
-                    return;
+            if (signatureHasRestParameter(signature)) {
+                // parameter might be a transient symbol generated by use of `arguments` in the function body.
+                var parameter = ts.last(signature.parameters);
+                if (ts.isTransientSymbol(parameter) || !ts.getEffectiveTypeAnnotationNode(parameter.valueDeclaration)) {
+                    var contextualParameterType = getRestTypeAtPosition(context, len);
+                    assignParameterType(parameter, contextualParameterType);
                 }
             }
-            // run the check only for the first declaration in the list
-            if (ts.getDeclarationOfKind(symbol, node.kind) !== node) {
-                return;
+        }
+        function assignNonContextualParameterTypes(signature) {
+            if (signature.thisParameter) {
+                assignParameterType(signature.thisParameter);
             }
-            var exportedDeclarationSpaces = 0 /* None */;
-            var nonExportedDeclarationSpaces = 0 /* None */;
-            var defaultExportedDeclarationSpaces = 0 /* None */;
-            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                var d = _a[_i];
-                var declarationSpaces = getDeclarationSpaces(d);
-                var effectiveDeclarationFlags = getEffectiveDeclarationFlags(d, 1 /* Export */ | 512 /* Default */);
-                if (effectiveDeclarationFlags & 1 /* Export */) {
-                    if (effectiveDeclarationFlags & 512 /* Default */) {
-                        defaultExportedDeclarationSpaces |= declarationSpaces;
-                    }
-                    else {
-                        exportedDeclarationSpaces |= declarationSpaces;
+            for (var _i = 0, _a = signature.parameters; _i < _a.length; _i++) {
+                var parameter = _a[_i];
+                assignParameterType(parameter);
+            }
+        }
+        function assignParameterType(parameter, type) {
+            var links = getSymbolLinks(parameter);
+            if (!links.type) {
+                var declaration = parameter.valueDeclaration;
+                links.type = type || getWidenedTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ true);
+                if (declaration.name.kind !== 78 /* Identifier */) {
+                    // if inference didn't come up with anything but unknown, fall back to the binding pattern if present.
+                    if (links.type === unknownType) {
+                        links.type = getTypeFromBindingPattern(declaration.name);
                     }
-                }
-                else {
-                    nonExportedDeclarationSpaces |= declarationSpaces;
+                    assignBindingElementTypes(declaration.name);
                 }
             }
-            // Spaces for anything not declared a 'default export'.
-            var nonDefaultExportedDeclarationSpaces = exportedDeclarationSpaces | nonExportedDeclarationSpaces;
-            var commonDeclarationSpacesForExportsAndLocals = exportedDeclarationSpaces & nonExportedDeclarationSpaces;
-            var commonDeclarationSpacesForDefaultAndNonDefault = defaultExportedDeclarationSpaces & nonDefaultExportedDeclarationSpaces;
-            if (commonDeclarationSpacesForExportsAndLocals || commonDeclarationSpacesForDefaultAndNonDefault) {
-                // declaration spaces for exported and non-exported declarations intersect
-                for (var _b = 0, _c = symbol.declarations; _b < _c.length; _b++) {
-                    var d = _c[_b];
-                    var declarationSpaces = getDeclarationSpaces(d);
-                    var name = ts.getNameOfDeclaration(d);
-                    // Only error on the declarations that contributed to the intersecting spaces.
-                    if (declarationSpaces & commonDeclarationSpacesForDefaultAndNonDefault) {
-                        error(name, ts.Diagnostics.Merged_declaration_0_cannot_include_a_default_export_declaration_Consider_adding_a_separate_export_default_0_declaration_instead, ts.declarationNameToString(name));
+        }
+        // When contextual typing assigns a type to a parameter that contains a binding pattern, we also need to push
+        // the destructured type into the contained binding elements.
+        function assignBindingElementTypes(pattern) {
+            for (var _i = 0, _a = pattern.elements; _i < _a.length; _i++) {
+                var element = _a[_i];
+                if (!ts.isOmittedExpression(element)) {
+                    if (element.name.kind === 78 /* Identifier */) {
+                        getSymbolLinks(getSymbolOfNode(element)).type = getTypeForBindingElement(element);
                     }
-                    else if (declarationSpaces & commonDeclarationSpacesForExportsAndLocals) {
-                        error(name, ts.Diagnostics.Individual_declarations_in_merged_declaration_0_must_be_all_exported_or_all_local, ts.declarationNameToString(name));
+                    else {
+                        assignBindingElementTypes(element.name);
                     }
                 }
             }
-            function getDeclarationSpaces(decl) {
-                var d = decl;
-                switch (d.kind) {
-                    case 246 /* InterfaceDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                    // A jsdoc typedef and callback are, by definition, type aliases.
-                    // falls through
-                    case 322 /* JSDocTypedefTag */:
-                    case 315 /* JSDocCallbackTag */:
-                    case 316 /* JSDocEnumTag */:
-                        return 2 /* ExportType */;
-                    case 249 /* ModuleDeclaration */:
-                        return ts.isAmbientModule(d) || ts.getModuleInstanceState(d) !== 0 /* NonInstantiated */
-                            ? 4 /* ExportNamespace */ | 1 /* ExportValue */
-                            : 4 /* ExportNamespace */;
-                    case 245 /* ClassDeclaration */:
-                    case 248 /* EnumDeclaration */:
-                    case 284 /* EnumMember */:
-                        return 2 /* ExportType */ | 1 /* ExportValue */;
-                    case 290 /* SourceFile */:
-                        return 2 /* ExportType */ | 1 /* ExportValue */ | 4 /* ExportNamespace */;
-                    case 259 /* ExportAssignment */:
-                        // Export assigned entity name expressions act as aliases and should fall through, otherwise they export values
-                        if (!ts.isEntityNameExpression(d.expression)) {
-                            return 1 /* ExportValue */;
-                        }
-                        d = d.expression;
-                    // The below options all declare an Alias, which is allowed to merge with other values within the importing module.
-                    // falls through
-                    case 253 /* ImportEqualsDeclaration */:
-                    case 256 /* NamespaceImport */:
-                    case 255 /* ImportClause */:
-                        var result_10 = 0 /* None */;
-                        var target = resolveAlias(getSymbolOfNode(d));
-                        ts.forEach(target.declarations, function (d) { result_10 |= getDeclarationSpaces(d); });
-                        return result_10;
-                    case 242 /* VariableDeclaration */:
-                    case 191 /* BindingElement */:
-                    case 244 /* FunctionDeclaration */:
-                    case 258 /* ImportSpecifier */: // https://github.com/Microsoft/TypeScript/pull/7591
-                    case 75 /* Identifier */: // https://github.com/microsoft/TypeScript/issues/36098
-                        // Identifiers are used as declarations of assignment declarations whose parents may be
-                        // SyntaxKind.CallExpression - `Object.defineProperty(thing, "aField", {value: 42});`
-                        // SyntaxKind.ElementAccessExpression - `thing["aField"] = 42;` or `thing["aField"];` (with a doc comment on it)
-                        // or SyntaxKind.PropertyAccessExpression - `thing.aField = 42;`
-                        // all of which are pretty much always values, or at least imply a value meaning.
-                        // It may be apprpriate to treat these as aliases in the future.
-                        return 1 /* ExportValue */;
-                    default:
-                        return ts.Debug.failBadSyntaxKind(d);
-                }
+        }
+        function createPromiseType(promisedType) {
+            // creates a `Promise<T>` type where `T` is the promisedType argument
+            var globalPromiseType = getGlobalPromiseType(/*reportErrors*/ true);
+            if (globalPromiseType !== emptyGenericType) {
+                // if the promised type is itself a promise, get the underlying type; otherwise, fallback to the promised type
+                promisedType = getAwaitedType(promisedType) || unknownType;
+                return createTypeReference(globalPromiseType, [promisedType]);
             }
+            return unknownType;
         }
-        function getAwaitedTypeOfPromise(type, errorNode, diagnosticMessage, arg0) {
-            var promisedType = getPromisedTypeOfPromise(type, errorNode);
-            return promisedType && getAwaitedType(promisedType, errorNode, diagnosticMessage, arg0);
+        function createPromiseLikeType(promisedType) {
+            // creates a `PromiseLike<T>` type where `T` is the promisedType argument
+            var globalPromiseLikeType = getGlobalPromiseLikeType(/*reportErrors*/ true);
+            if (globalPromiseLikeType !== emptyGenericType) {
+                // if the promised type is itself a promise, get the underlying type; otherwise, fallback to the promised type
+                promisedType = getAwaitedType(promisedType) || unknownType;
+                return createTypeReference(globalPromiseLikeType, [promisedType]);
+            }
+            return unknownType;
         }
-        /**
-         * Gets the "promised type" of a promise.
-         * @param type The type of the promise.
-         * @remarks The "promised type" of a type is the type of the "value" parameter of the "onfulfilled" callback.
-         */
-        function getPromisedTypeOfPromise(type, errorNode) {
-            //
-            //  { // type
-            //      then( // thenFunction
-            //          onfulfilled: ( // onfulfilledParameterType
-            //              value: T // valueParameterType
-            //          ) => any
-            //      ): any;
-            //  }
-            //
-            if (isTypeAny(type)) {
-                return undefined;
+        function createPromiseReturnType(func, promisedType) {
+            var promiseType = createPromiseType(promisedType);
+            if (promiseType === unknownType) {
+                error(func, ts.isImportCall(func) ?
+                    ts.Diagnostics.A_dynamic_import_call_returns_a_Promise_Make_sure_you_have_a_declaration_for_Promise_or_include_ES2015_in_your_lib_option :
+                    ts.Diagnostics.An_async_function_or_method_must_return_a_Promise_Make_sure_you_have_a_declaration_for_Promise_or_include_ES2015_in_your_lib_option);
+                return errorType;
             }
-            var typeAsPromise = type;
-            if (typeAsPromise.promisedTypeOfPromise) {
-                return typeAsPromise.promisedTypeOfPromise;
+            else if (!getGlobalPromiseConstructorSymbol(/*reportErrors*/ true)) {
+                error(func, ts.isImportCall(func) ?
+                    ts.Diagnostics.A_dynamic_import_call_in_ES5_SlashES3_requires_the_Promise_constructor_Make_sure_you_have_a_declaration_for_the_Promise_constructor_or_include_ES2015_in_your_lib_option :
+                    ts.Diagnostics.An_async_function_or_method_in_ES5_SlashES3_requires_the_Promise_constructor_Make_sure_you_have_a_declaration_for_the_Promise_constructor_or_include_ES2015_in_your_lib_option);
             }
-            if (isReferenceToType(type, getGlobalPromiseType(/*reportErrors*/ false))) {
-                return typeAsPromise.promisedTypeOfPromise = getTypeArguments(type)[0];
+            return promiseType;
+        }
+        function getReturnTypeFromBody(func, checkMode) {
+            if (!func.body) {
+                return errorType;
             }
-            var thenFunction = getTypeOfPropertyOfType(type, "then"); // TODO: GH#18217
-            if (isTypeAny(thenFunction)) {
-                return undefined;
+            var functionFlags = ts.getFunctionFlags(func);
+            var isAsync = (functionFlags & 2 /* Async */) !== 0;
+            var isGenerator = (functionFlags & 1 /* Generator */) !== 0;
+            var returnType;
+            var yieldType;
+            var nextType;
+            var fallbackReturnType = voidType;
+            if (func.body.kind !== 230 /* Block */) { // Async or normal arrow function
+                returnType = checkExpressionCached(func.body, checkMode && checkMode & ~8 /* SkipGenericFunctions */);
+                if (isAsync) {
+                    // From within an async function you can return either a non-promise value or a promise. Any
+                    // Promise/A+ compatible implementation will always assimilate any foreign promise, so the
+                    // return type of the body should be unwrapped to its awaited type, which we will wrap in
+                    // the native Promise<T> type later in this function.
+                    returnType = checkAwaitedType(returnType, /*errorNode*/ func, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+                }
             }
-            var thenSignatures = thenFunction ? getSignaturesOfType(thenFunction, 0 /* Call */) : ts.emptyArray;
-            if (thenSignatures.length === 0) {
-                if (errorNode) {
-                    error(errorNode, ts.Diagnostics.A_promise_must_have_a_then_method);
+            else if (isGenerator) { // Generator or AsyncGenerator function
+                var returnTypes = checkAndAggregateReturnExpressionTypes(func, checkMode);
+                if (!returnTypes) {
+                    fallbackReturnType = neverType;
                 }
-                return undefined;
+                else if (returnTypes.length > 0) {
+                    returnType = getUnionType(returnTypes, 2 /* Subtype */);
+                }
+                var _a = checkAndAggregateYieldOperandTypes(func, checkMode), yieldTypes = _a.yieldTypes, nextTypes = _a.nextTypes;
+                yieldType = ts.some(yieldTypes) ? getUnionType(yieldTypes, 2 /* Subtype */) : undefined;
+                nextType = ts.some(nextTypes) ? getIntersectionType(nextTypes) : undefined;
             }
-            var onfulfilledParameterType = getTypeWithFacts(getUnionType(ts.map(thenSignatures, getTypeOfFirstParameterOfSignature)), 2097152 /* NEUndefinedOrNull */);
-            if (isTypeAny(onfulfilledParameterType)) {
-                return undefined;
+            else { // Async or normal function
+                var types = checkAndAggregateReturnExpressionTypes(func, checkMode);
+                if (!types) {
+                    // For an async function, the return type will not be never, but rather a Promise for never.
+                    return functionFlags & 2 /* Async */
+                        ? createPromiseReturnType(func, neverType) // Async function
+                        : neverType; // Normal function
+                }
+                if (types.length === 0) {
+                    // For an async function, the return type will not be void, but rather a Promise for void.
+                    return functionFlags & 2 /* Async */
+                        ? createPromiseReturnType(func, voidType) // Async function
+                        : voidType; // Normal function
+                }
+                // Return a union of the return expression types.
+                returnType = getUnionType(types, 2 /* Subtype */);
             }
-            var onfulfilledParameterSignatures = getSignaturesOfType(onfulfilledParameterType, 0 /* Call */);
-            if (onfulfilledParameterSignatures.length === 0) {
-                if (errorNode) {
-                    error(errorNode, ts.Diagnostics.The_first_parameter_of_the_then_method_of_a_promise_must_be_a_callback);
+            if (returnType || yieldType || nextType) {
+                if (yieldType)
+                    reportErrorsFromWidening(func, yieldType, 3 /* GeneratorYield */);
+                if (returnType)
+                    reportErrorsFromWidening(func, returnType, 1 /* FunctionReturn */);
+                if (nextType)
+                    reportErrorsFromWidening(func, nextType, 2 /* GeneratorNext */);
+                if (returnType && isUnitType(returnType) ||
+                    yieldType && isUnitType(yieldType) ||
+                    nextType && isUnitType(nextType)) {
+                    var contextualSignature = getContextualSignatureForFunctionLikeDeclaration(func);
+                    var contextualType = !contextualSignature ? undefined :
+                        contextualSignature === getSignatureFromDeclaration(func) ? isGenerator ? undefined : returnType :
+                            instantiateContextualType(getReturnTypeOfSignature(contextualSignature), func);
+                    if (isGenerator) {
+                        yieldType = getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(yieldType, contextualType, 0 /* Yield */, isAsync);
+                        returnType = getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(returnType, contextualType, 1 /* Return */, isAsync);
+                        nextType = getWidenedLiteralLikeTypeForContextualIterationTypeIfNeeded(nextType, contextualType, 2 /* Next */, isAsync);
+                    }
+                    else {
+                        returnType = getWidenedLiteralLikeTypeForContextualReturnTypeIfNeeded(returnType, contextualType, isAsync);
+                    }
                 }
-                return undefined;
+                if (yieldType)
+                    yieldType = getWidenedType(yieldType);
+                if (returnType)
+                    returnType = getWidenedType(returnType);
+                if (nextType)
+                    nextType = getWidenedType(nextType);
             }
-            return typeAsPromise.promisedTypeOfPromise = getUnionType(ts.map(onfulfilledParameterSignatures, getTypeOfFirstParameterOfSignature), 2 /* Subtype */);
-        }
-        /**
-         * Gets the "awaited type" of a type.
-         * @param type The type to await.
-         * @remarks The "awaited type" of an expression is its "promised type" if the expression is a
-         * Promise-like type; otherwise, it is the type of the expression. This is used to reflect
-         * The runtime behavior of the `await` keyword.
-         */
-        function checkAwaitedType(type, errorNode, diagnosticMessage, arg0) {
-            var awaitedType = getAwaitedType(type, errorNode, diagnosticMessage, arg0);
-            return awaitedType || errorType;
-        }
-        /**
-         * Determines whether a type has a callable `then` member.
-         */
-        function isThenableType(type) {
-            var thenFunction = getTypeOfPropertyOfType(type, "then");
-            return !!thenFunction && getSignaturesOfType(getTypeWithFacts(thenFunction, 2097152 /* NEUndefinedOrNull */), 0 /* Call */).length > 0;
-        }
-        /**
-         * Gets the "awaited type" of a type.
-         *
-         * The "awaited type" of an expression is its "promised type" if the expression is a
-         * Promise-like type; otherwise, it is the type of the expression. If the "promised
-         * type" is itself a Promise-like, the "promised type" is recursively unwrapped until a
-         * non-promise type is found.
-         *
-         * This is used to reflect the runtime behavior of the `await` keyword.
-         */
-        function getAwaitedType(type, errorNode, diagnosticMessage, arg0) {
-            if (isTypeAny(type)) {
-                return type;
+            if (isGenerator) {
+                return createGeneratorReturnType(yieldType || neverType, returnType || fallbackReturnType, nextType || getContextualIterationType(2 /* Next */, func) || unknownType, isAsync);
             }
-            var typeAsAwaitable = type;
-            if (typeAsAwaitable.awaitedTypeOfType) {
-                return typeAsAwaitable.awaitedTypeOfType;
+            else {
+                // From within an async function you can return either a non-promise value or a promise. Any
+                // Promise/A+ compatible implementation will always assimilate any foreign promise, so the
+                // return type of the body is awaited type of the body, wrapped in a native Promise<T> type.
+                return isAsync
+                    ? createPromiseType(returnType || fallbackReturnType)
+                    : returnType || fallbackReturnType;
             }
-            // For a union, get a union of the awaited types of each constituent.
-            //
-            return typeAsAwaitable.awaitedTypeOfType =
-                mapType(type, errorNode ? function (constituentType) { return getAwaitedTypeWorker(constituentType, errorNode, diagnosticMessage, arg0); } : getAwaitedTypeWorker);
         }
-        function getAwaitedTypeWorker(type, errorNode, diagnosticMessage, arg0) {
-            var typeAsAwaitable = type;
-            if (typeAsAwaitable.awaitedTypeOfType) {
-                return typeAsAwaitable.awaitedTypeOfType;
-            }
-            var promisedType = getPromisedTypeOfPromise(type);
-            if (promisedType) {
-                if (type.id === promisedType.id || awaitedTypeStack.lastIndexOf(promisedType.id) >= 0) {
-                    // Verify that we don't have a bad actor in the form of a promise whose
-                    // promised type is the same as the promise type, or a mutually recursive
-                    // promise. If so, we return undefined as we cannot guess the shape. If this
-                    // were the actual case in the JavaScript, this Promise would never resolve.
-                    //
-                    // An example of a bad actor with a singly-recursive promise type might
-                    // be:
-                    //
-                    //  interface BadPromise {
-                    //      then(
-                    //          onfulfilled: (value: BadPromise) => any,
-                    //          onrejected: (error: any) => any): BadPromise;
-                    //  }
-                    //
-                    // The above interface will pass the PromiseLike check, and return a
-                    // promised type of `BadPromise`. Since this is a self reference, we
-                    // don't want to keep recursing ad infinitum.
-                    //
-                    // An example of a bad actor in the form of a mutually-recursive
-                    // promise type might be:
-                    //
-                    //  interface BadPromiseA {
-                    //      then(
-                    //          onfulfilled: (value: BadPromiseB) => any,
-                    //          onrejected: (error: any) => any): BadPromiseB;
-                    //  }
-                    //
-                    //  interface BadPromiseB {
-                    //      then(
-                    //          onfulfilled: (value: BadPromiseA) => any,
-                    //          onrejected: (error: any) => any): BadPromiseA;
-                    //  }
-                    //
-                    if (errorNode) {
-                        error(errorNode, ts.Diagnostics.Type_is_referenced_directly_or_indirectly_in_the_fulfillment_callback_of_its_own_then_method);
+        function createGeneratorReturnType(yieldType, returnType, nextType, isAsyncGenerator) {
+            var resolver = isAsyncGenerator ? asyncIterationTypesResolver : syncIterationTypesResolver;
+            var globalGeneratorType = resolver.getGlobalGeneratorType(/*reportErrors*/ false);
+            yieldType = resolver.resolveIterationType(yieldType, /*errorNode*/ undefined) || unknownType;
+            returnType = resolver.resolveIterationType(returnType, /*errorNode*/ undefined) || unknownType;
+            nextType = resolver.resolveIterationType(nextType, /*errorNode*/ undefined) || unknownType;
+            if (globalGeneratorType === emptyGenericType) {
+                // Fall back to the global IterableIterator if returnType is assignable to the expected return iteration
+                // type of IterableIterator, and the expected next iteration type of IterableIterator is assignable to
+                // nextType.
+                var globalType = resolver.getGlobalIterableIteratorType(/*reportErrors*/ false);
+                var iterationTypes = globalType !== emptyGenericType ? getIterationTypesOfGlobalIterableType(globalType, resolver) : undefined;
+                var iterableIteratorReturnType = iterationTypes ? iterationTypes.returnType : anyType;
+                var iterableIteratorNextType = iterationTypes ? iterationTypes.nextType : undefinedType;
+                if (isTypeAssignableTo(returnType, iterableIteratorReturnType) &&
+                    isTypeAssignableTo(iterableIteratorNextType, nextType)) {
+                    if (globalType !== emptyGenericType) {
+                        return createTypeFromGenericGlobalType(globalType, [yieldType]);
                     }
-                    return undefined;
-                }
-                // Keep track of the type we're about to unwrap to avoid bad recursive promise types.
-                // See the comments above for more information.
-                awaitedTypeStack.push(type.id);
-                var awaitedType = getAwaitedType(promisedType, errorNode, diagnosticMessage, arg0);
-                awaitedTypeStack.pop();
-                if (!awaitedType) {
-                    return undefined;
+                    // The global IterableIterator type doesn't exist, so report an error
+                    resolver.getGlobalIterableIteratorType(/*reportErrors*/ true);
+                    return emptyObjectType;
                 }
-                return typeAsAwaitable.awaitedTypeOfType = awaitedType;
+                // The global Generator type doesn't exist, so report an error
+                resolver.getGlobalGeneratorType(/*reportErrors*/ true);
+                return emptyObjectType;
             }
-            // The type was not a promise, so it could not be unwrapped any further.
-            // As long as the type does not have a callable "then" property, it is
-            // safe to return the type; otherwise, an error is reported and we return
-            // undefined.
-            //
-            // An example of a non-promise "thenable" might be:
-            //
-            //  await { then(): void {} }
-            //
-            // The "thenable" does not match the minimal definition for a promise. When
-            // a Promise/A+-compatible or ES6 promise tries to adopt this value, the promise
-            // will never settle. We treat this as an error to help flag an early indicator
-            // of a runtime problem. If the user wants to return this value from an async
-            // function, they would need to wrap it in some other value. If they want it to
-            // be treated as a promise, they can cast to <any>.
-            if (isThenableType(type)) {
-                if (errorNode) {
-                    if (!diagnosticMessage)
-                        return ts.Debug.fail();
-                    error(errorNode, diagnosticMessage, arg0);
+            return createTypeFromGenericGlobalType(globalGeneratorType, [yieldType, returnType, nextType]);
+        }
+        function checkAndAggregateYieldOperandTypes(func, checkMode) {
+            var yieldTypes = [];
+            var nextTypes = [];
+            var isAsync = (ts.getFunctionFlags(func) & 2 /* Async */) !== 0;
+            ts.forEachYieldExpression(func.body, function (yieldExpression) {
+                var yieldExpressionType = yieldExpression.expression ? checkExpression(yieldExpression.expression, checkMode) : undefinedWideningType;
+                ts.pushIfUnique(yieldTypes, getYieldedTypeOfYieldExpression(yieldExpression, yieldExpressionType, anyType, isAsync));
+                var nextType;
+                if (yieldExpression.asteriskToken) {
+                    var iterationTypes = getIterationTypesOfIterable(yieldExpressionType, isAsync ? 19 /* AsyncYieldStar */ : 17 /* YieldStar */, yieldExpression.expression);
+                    nextType = iterationTypes && iterationTypes.nextType;
                 }
-                return undefined;
-            }
-            return typeAsAwaitable.awaitedTypeOfType = type;
+                else {
+                    nextType = getContextualType(yieldExpression);
+                }
+                if (nextType)
+                    ts.pushIfUnique(nextTypes, nextType);
+            });
+            return { yieldTypes: yieldTypes, nextTypes: nextTypes };
+        }
+        function getYieldedTypeOfYieldExpression(node, expressionType, sentType, isAsync) {
+            var errorNode = node.expression || node;
+            // A `yield*` expression effectively yields everything that its operand yields
+            var yieldedType = node.asteriskToken ? checkIteratedTypeOrElementType(isAsync ? 19 /* AsyncYieldStar */ : 17 /* YieldStar */, expressionType, sentType, errorNode) : expressionType;
+            return !isAsync ? yieldedType : getAwaitedType(yieldedType, errorNode, node.asteriskToken
+                ? ts.Diagnostics.Type_of_iterated_elements_of_a_yield_Asterisk_operand_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member
+                : ts.Diagnostics.Type_of_yield_operand_in_an_async_generator_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
         }
         /**
-         * Checks the return type of an async function to ensure it is a compatible
-         * Promise implementation.
-         *
-         * This checks that an async function has a valid Promise-compatible return type.
-         * An async function has a valid Promise-compatible return type if the resolved value
-         * of the return type has a construct signature that takes in an `initializer` function
-         * that in turn supplies a `resolve` function as one of its arguments and results in an
-         * object with a callable `then` signature.
-         *
-         * @param node The signature to check
+         * Collect the TypeFacts learned from a typeof switch with
+         * total clauses `witnesses`, and the active clause ranging
+         * from `start` to `end`. Parameter `hasDefault` denotes
+         * whether the active clause contains a default clause.
          */
-        function checkAsyncFunctionReturnType(node, returnTypeNode) {
-            // As part of our emit for an async function, we will need to emit the entity name of
-            // the return type annotation as an expression. To meet the necessary runtime semantics
-            // for __awaiter, we must also check that the type of the declaration (e.g. the static
-            // side or "constructor" of the promise type) is compatible `PromiseConstructorLike`.
-            //
-            // An example might be (from lib.es6.d.ts):
-            //
-            //  interface Promise<T> { ... }
-            //  interface PromiseConstructor {
-            //      new <T>(...): Promise<T>;
-            //  }
-            //  declare var Promise: PromiseConstructor;
-            //
-            // When an async function declares a return type annotation of `Promise<T>`, we
-            // need to get the type of the `Promise` variable declaration above, which would
-            // be `PromiseConstructor`.
-            //
-            // The same case applies to a class:
-            //
-            //  declare class Promise<T> {
-            //      constructor(...);
-            //      then<U>(...): Promise<U>;
-            //  }
-            //
-            var returnType = getTypeFromTypeNode(returnTypeNode);
-            if (languageVersion >= 2 /* ES2015 */) {
-                if (returnType === errorType) {
-                    return;
+        function getFactsFromTypeofSwitch(start, end, witnesses, hasDefault) {
+            var facts = 0 /* None */;
+            // When in the default we only collect inequality facts
+            // because default is 'in theory' a set of infinite
+            // equalities.
+            if (hasDefault) {
+                // Value is not equal to any types after the active clause.
+                for (var i = end; i < witnesses.length; i++) {
+                    facts |= typeofNEFacts.get(witnesses[i]) || 32768 /* TypeofNEHostObject */;
                 }
-                var globalPromiseType = getGlobalPromiseType(/*reportErrors*/ true);
-                if (globalPromiseType !== emptyGenericType && !isReferenceToType(returnType, globalPromiseType)) {
-                    // The promise type was not a valid type reference to the global promise type, so we
-                    // report an error and return the unknown type.
-                    error(returnTypeNode, ts.Diagnostics.The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type);
-                    return;
+                // Remove inequalities for types that appear in the
+                // active clause because they appear before other
+                // types collected so far.
+                for (var i = start; i < end; i++) {
+                    facts &= ~(typeofNEFacts.get(witnesses[i]) || 0);
+                }
+                // Add inequalities for types before the active clause unconditionally.
+                for (var i = 0; i < start; i++) {
+                    facts |= typeofNEFacts.get(witnesses[i]) || 32768 /* TypeofNEHostObject */;
                 }
             }
+            // When in an active clause without default the set of
+            // equalities is finite.
             else {
-                // Always mark the type node as referenced if it points to a value
-                markTypeNodeAsReferenced(returnTypeNode);
-                if (returnType === errorType) {
-                    return;
-                }
-                var promiseConstructorName = ts.getEntityNameFromTypeNode(returnTypeNode);
-                if (promiseConstructorName === undefined) {
-                    error(returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value, typeToString(returnType));
-                    return;
-                }
-                var promiseConstructorSymbol = resolveEntityName(promiseConstructorName, 111551 /* Value */, /*ignoreErrors*/ true);
-                var promiseConstructorType = promiseConstructorSymbol ? getTypeOfSymbol(promiseConstructorSymbol) : errorType;
-                if (promiseConstructorType === errorType) {
-                    if (promiseConstructorName.kind === 75 /* Identifier */ && promiseConstructorName.escapedText === "Promise" && getTargetType(returnType) === getGlobalPromiseType(/*reportErrors*/ false)) {
-                        error(returnTypeNode, ts.Diagnostics.An_async_function_or_method_in_ES5_SlashES3_requires_the_Promise_constructor_Make_sure_you_have_a_declaration_for_the_Promise_constructor_or_include_ES2015_in_your_lib_option);
-                    }
-                    else {
-                        error(returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value, ts.entityNameToString(promiseConstructorName));
-                    }
-                    return;
-                }
-                var globalPromiseConstructorLikeType = getGlobalPromiseConstructorLikeType(/*reportErrors*/ true);
-                if (globalPromiseConstructorLikeType === emptyObjectType) {
-                    // If we couldn't resolve the global PromiseConstructorLike type we cannot verify
-                    // compatibility with __awaiter.
-                    error(returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value, ts.entityNameToString(promiseConstructorName));
-                    return;
-                }
-                if (!checkTypeAssignableTo(promiseConstructorType, globalPromiseConstructorLikeType, returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value)) {
-                    return;
+                // Add equalities for all types in the active clause.
+                for (var i = start; i < end; i++) {
+                    facts |= typeofEQFacts.get(witnesses[i]) || 128 /* TypeofEQHostObject */;
                 }
-                // Verify there is no local declaration that could collide with the promise constructor.
-                var rootName = promiseConstructorName && ts.getFirstIdentifier(promiseConstructorName);
-                var collidingSymbol = getSymbol(node.locals, rootName.escapedText, 111551 /* Value */);
-                if (collidingSymbol) {
-                    error(collidingSymbol.valueDeclaration, ts.Diagnostics.Duplicate_identifier_0_Compiler_uses_declaration_1_to_support_async_functions, ts.idText(rootName), ts.entityNameToString(promiseConstructorName));
-                    return;
+                // Remove equalities for types that appear before the
+                // active clause.
+                for (var i = 0; i < start; i++) {
+                    facts &= ~(typeofEQFacts.get(witnesses[i]) || 0);
                 }
             }
-            checkAwaitedType(returnType, node, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
-        }
-        /** Check a decorator */
-        function checkDecorator(node) {
-            var signature = getResolvedSignature(node);
-            var returnType = getReturnTypeOfSignature(signature);
-            if (returnType.flags & 1 /* Any */) {
-                return;
-            }
-            var expectedReturnType;
-            var headMessage = getDiagnosticHeadMessageForDecoratorResolution(node);
-            var errorInfo;
-            switch (node.parent.kind) {
-                case 245 /* ClassDeclaration */:
-                    var classSymbol = getSymbolOfNode(node.parent);
-                    var classConstructorType = getTypeOfSymbol(classSymbol);
-                    expectedReturnType = getUnionType([classConstructorType, voidType]);
-                    break;
-                case 156 /* Parameter */:
-                    expectedReturnType = voidType;
-                    errorInfo = ts.chainDiagnosticMessages(
-                    /*details*/ undefined, ts.Diagnostics.The_return_type_of_a_parameter_decorator_function_must_be_either_void_or_any);
-                    break;
-                case 159 /* PropertyDeclaration */:
-                    expectedReturnType = voidType;
-                    errorInfo = ts.chainDiagnosticMessages(
-                    /*details*/ undefined, ts.Diagnostics.The_return_type_of_a_property_decorator_function_must_be_either_void_or_any);
-                    break;
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    var methodType = getTypeOfNode(node.parent);
-                    var descriptorType = createTypedPropertyDescriptorType(methodType);
-                    expectedReturnType = getUnionType([descriptorType, voidType]);
-                    break;
-                default:
-                    return ts.Debug.fail();
-            }
-            checkTypeAssignableTo(returnType, expectedReturnType, node, headMessage, function () { return errorInfo; });
+            return facts;
         }
-        /**
-         * If a TypeNode can be resolved to a value symbol imported from an external module, it is
-         * marked as referenced to prevent import elision.
-         */
-        function markTypeNodeAsReferenced(node) {
-            markEntityNameOrEntityExpressionAsReference(node && ts.getEntityNameFromTypeNode(node));
+        function isExhaustiveSwitchStatement(node) {
+            var links = getNodeLinks(node);
+            return links.isExhaustive !== undefined ? links.isExhaustive : (links.isExhaustive = computeExhaustiveSwitchStatement(node));
         }
-        function markEntityNameOrEntityExpressionAsReference(typeName) {
-            if (!typeName)
-                return;
-            var rootName = ts.getFirstIdentifier(typeName);
-            var meaning = (typeName.kind === 75 /* Identifier */ ? 788968 /* Type */ : 1920 /* Namespace */) | 2097152 /* Alias */;
-            var rootSymbol = resolveName(rootName, rootName.escapedText, meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isRefernce*/ true);
-            if (rootSymbol
-                && rootSymbol.flags & 2097152 /* Alias */
-                && symbolIsValue(rootSymbol)
-                && !isConstEnumOrConstEnumOnlyModule(resolveAlias(rootSymbol))
-                && !getTypeOnlyAliasDeclaration(rootSymbol)) {
-                markAliasSymbolAsReferenced(rootSymbol);
+        function computeExhaustiveSwitchStatement(node) {
+            if (node.expression.kind === 211 /* TypeOfExpression */) {
+                var operandType = getTypeOfExpression(node.expression.expression);
+                var witnesses = getSwitchClauseTypeOfWitnesses(node, /*retainDefault*/ false);
+                // notEqualFacts states that the type of the switched value is not equal to every type in the switch.
+                var notEqualFacts_1 = getFactsFromTypeofSwitch(0, 0, witnesses, /*hasDefault*/ true);
+                var type_4 = getBaseConstraintOfType(operandType) || operandType;
+                // Take any/unknown as a special condition. Or maybe we could change `type` to a union containing all primitive types.
+                if (type_4.flags & 3 /* AnyOrUnknown */) {
+                    return (556800 /* AllTypeofNE */ & notEqualFacts_1) === 556800 /* AllTypeofNE */;
+                }
+                return !!(filterType(type_4, function (t) { return (getTypeFacts(t) & notEqualFacts_1) === notEqualFacts_1; }).flags & 131072 /* Never */);
             }
-        }
-        /**
-         * This function marks the type used for metadata decorator as referenced if it is import
-         * from external module.
-         * This is different from markTypeNodeAsReferenced because it tries to simplify type nodes in
-         * union and intersection type
-         * @param node
-         */
-        function markDecoratorMedataDataTypeNodeAsReferenced(node) {
-            var entityName = getEntityNameForDecoratorMetadata(node);
-            if (entityName && ts.isEntityName(entityName)) {
-                markEntityNameOrEntityExpressionAsReference(entityName);
+            var type = getTypeOfExpression(node.expression);
+            if (!isLiteralType(type)) {
+                return false;
             }
-        }
-        function getEntityNameForDecoratorMetadata(node) {
-            if (node) {
-                switch (node.kind) {
-                    case 179 /* IntersectionType */:
-                    case 178 /* UnionType */:
-                        return getEntityNameForDecoratorMetadataFromTypeList(node.types);
-                    case 180 /* ConditionalType */:
-                        return getEntityNameForDecoratorMetadataFromTypeList([node.trueType, node.falseType]);
-                    case 182 /* ParenthesizedType */:
-                        return getEntityNameForDecoratorMetadata(node.type);
-                    case 169 /* TypeReference */:
-                        return node.typeName;
-                }
+            var switchTypes = getSwitchClauseTypes(node);
+            if (!switchTypes.length || ts.some(switchTypes, isNeitherUnitTypeNorNever)) {
+                return false;
             }
+            return eachTypeContainedIn(mapType(type, getRegularTypeOfLiteralType), switchTypes);
         }
-        function getEntityNameForDecoratorMetadataFromTypeList(types) {
-            var commonEntityName;
-            for (var _i = 0, types_20 = types; _i < types_20.length; _i++) {
-                var typeNode = types_20[_i];
-                while (typeNode.kind === 182 /* ParenthesizedType */) {
-                    typeNode = typeNode.type; // Skip parens if need be
-                }
-                if (typeNode.kind === 137 /* NeverKeyword */) {
-                    continue; // Always elide `never` from the union/intersection if possible
-                }
-                if (!strictNullChecks && (typeNode.kind === 100 /* NullKeyword */ || typeNode.kind === 146 /* UndefinedKeyword */)) {
-                    continue; // Elide null and undefined from unions for metadata, just like what we did prior to the implementation of strict null checks
-                }
-                var individualEntityName = getEntityNameForDecoratorMetadata(typeNode);
-                if (!individualEntityName) {
-                    // Individual is something like string number
-                    // So it would be serialized to either that type or object
-                    // Safe to return here
-                    return undefined;
-                }
-                if (commonEntityName) {
-                    // Note this is in sync with the transformation that happens for type node.
-                    // Keep this in sync with serializeUnionOrIntersectionType
-                    // Verify if they refer to same entity and is identifier
-                    // return undefined if they dont match because we would emit object
-                    if (!ts.isIdentifier(commonEntityName) ||
-                        !ts.isIdentifier(individualEntityName) ||
-                        commonEntityName.escapedText !== individualEntityName.escapedText) {
-                        return undefined;
+        function functionHasImplicitReturn(func) {
+            return func.endFlowNode && isReachableFlowNode(func.endFlowNode);
+        }
+        /** NOTE: Return value of `[]` means a different thing than `undefined`. `[]` means func returns `void`, `undefined` means it returns `never`. */
+        function checkAndAggregateReturnExpressionTypes(func, checkMode) {
+            var functionFlags = ts.getFunctionFlags(func);
+            var aggregatedTypes = [];
+            var hasReturnWithNoExpression = functionHasImplicitReturn(func);
+            var hasReturnOfTypeNever = false;
+            ts.forEachReturnStatement(func.body, function (returnStatement) {
+                var expr = returnStatement.expression;
+                if (expr) {
+                    var type = checkExpressionCached(expr, checkMode && checkMode & ~8 /* SkipGenericFunctions */);
+                    if (functionFlags & 2 /* Async */) {
+                        // From within an async function you can return either a non-promise value or a promise. Any
+                        // Promise/A+ compatible implementation will always assimilate any foreign promise, so the
+                        // return type of the body should be unwrapped to its awaited type, which should be wrapped in
+                        // the native Promise<T> type by the caller.
+                        type = checkAwaitedType(type, func, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+                    }
+                    if (type.flags & 131072 /* Never */) {
+                        hasReturnOfTypeNever = true;
                     }
+                    ts.pushIfUnique(aggregatedTypes, type);
                 }
                 else {
-                    commonEntityName = individualEntityName;
+                    hasReturnWithNoExpression = true;
                 }
+            });
+            if (aggregatedTypes.length === 0 && !hasReturnWithNoExpression && (hasReturnOfTypeNever || mayReturnNever(func))) {
+                return undefined;
             }
-            return commonEntityName;
+            if (strictNullChecks && aggregatedTypes.length && hasReturnWithNoExpression &&
+                !(isJSConstructor(func) && aggregatedTypes.some(function (t) { return t.symbol === func.symbol; }))) {
+                // Javascript "callable constructors", containing eg `if (!(this instanceof A)) return new A()` should not add undefined
+                ts.pushIfUnique(aggregatedTypes, undefinedType);
+            }
+            return aggregatedTypes;
         }
-        function getParameterTypeNodeForDecoratorCheck(node) {
-            var typeNode = ts.getEffectiveTypeAnnotationNode(node);
-            return ts.isRestParameter(node) ? ts.getRestParameterElementType(typeNode) : typeNode;
+        function mayReturnNever(func) {
+            switch (func.kind) {
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                    return true;
+                case 165 /* MethodDeclaration */:
+                    return func.parent.kind === 200 /* ObjectLiteralExpression */;
+                default:
+                    return false;
+            }
         }
-        /** Check the decorators of a node */
-        function checkDecorators(node) {
-            if (!node.decorators) {
+        /**
+         * TypeScript Specification 1.0 (6.3) - July 2014
+         *   An explicitly typed function whose return type isn't the Void type,
+         *   the Any type, or a union type containing the Void or Any type as a constituent
+         *   must have at least one return statement somewhere in its body.
+         *   An exception to this rule is if the function implementation consists of a single 'throw' statement.
+         *
+         * @param returnType - return type of the function, can be undefined if return type is not explicitly specified
+         */
+        function checkAllCodePathsInNonVoidFunctionReturnOrThrow(func, returnType) {
+            if (!produceDiagnostics) {
                 return;
             }
-            // skip this check for nodes that cannot have decorators. These should have already had an error reported by
-            // checkGrammarDecorators.
-            if (!ts.nodeCanBeDecorated(node, node.parent, node.parent.parent)) {
+            var functionFlags = ts.getFunctionFlags(func);
+            var type = returnType && unwrapReturnType(returnType, functionFlags);
+            // Functions with with an explicitly specified 'void' or 'any' return type don't need any return expressions.
+            if (type && maybeTypeOfKind(type, 1 /* Any */ | 16384 /* Void */)) {
                 return;
             }
-            if (!compilerOptions.experimentalDecorators) {
-                error(node, ts.Diagnostics.Experimental_support_for_decorators_is_a_feature_that_is_subject_to_change_in_a_future_release_Set_the_experimentalDecorators_option_in_your_tsconfig_or_jsconfig_to_remove_this_warning);
-            }
-            var firstDecorator = node.decorators[0];
-            checkExternalEmitHelpers(firstDecorator, 8 /* Decorate */);
-            if (node.kind === 156 /* Parameter */) {
-                checkExternalEmitHelpers(firstDecorator, 32 /* Param */);
+            // If all we have is a function signature, or an arrow function with an expression body, then there is nothing to check.
+            // also if HasImplicitReturn flag is not set this means that all codepaths in function body end with return or throw
+            if (func.kind === 164 /* MethodSignature */ || ts.nodeIsMissing(func.body) || func.body.kind !== 230 /* Block */ || !functionHasImplicitReturn(func)) {
+                return;
             }
-            if (compilerOptions.emitDecoratorMetadata) {
-                checkExternalEmitHelpers(firstDecorator, 16 /* Metadata */);
-                // we only need to perform these checks if we are emitting serialized type metadata for the target of a decorator.
-                switch (node.kind) {
-                    case 245 /* ClassDeclaration */:
-                        var constructor = ts.getFirstConstructorWithBody(node);
-                        if (constructor) {
-                            for (var _i = 0, _a = constructor.parameters; _i < _a.length; _i++) {
-                                var parameter = _a[_i];
-                                markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(parameter));
-                            }
-                        }
-                        break;
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        var otherKind = node.kind === 163 /* GetAccessor */ ? 164 /* SetAccessor */ : 163 /* GetAccessor */;
-                        var otherAccessor = ts.getDeclarationOfKind(getSymbolOfNode(node), otherKind);
-                        markDecoratorMedataDataTypeNodeAsReferenced(getAnnotatedAccessorTypeNode(node) || otherAccessor && getAnnotatedAccessorTypeNode(otherAccessor));
-                        break;
-                    case 161 /* MethodDeclaration */:
-                        for (var _b = 0, _c = node.parameters; _b < _c.length; _b++) {
-                            var parameter = _c[_b];
-                            markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(parameter));
-                        }
-                        markDecoratorMedataDataTypeNodeAsReferenced(ts.getEffectiveReturnTypeNode(node));
-                        break;
-                    case 159 /* PropertyDeclaration */:
-                        markDecoratorMedataDataTypeNodeAsReferenced(ts.getEffectiveTypeAnnotationNode(node));
-                        break;
-                    case 156 /* Parameter */:
-                        markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(node));
-                        var containingSignature = node.parent;
-                        for (var _d = 0, _e = containingSignature.parameters; _d < _e.length; _d++) {
-                            var parameter = _e[_d];
-                            markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(parameter));
-                        }
-                        break;
-                }
+            var hasExplicitReturn = func.flags & 512 /* HasExplicitReturn */;
+            if (type && type.flags & 131072 /* Never */) {
+                error(ts.getEffectiveReturnTypeNode(func), ts.Diagnostics.A_function_returning_never_cannot_have_a_reachable_end_point);
             }
-            ts.forEach(node.decorators, checkDecorator);
-        }
-        function checkFunctionDeclaration(node) {
-            if (produceDiagnostics) {
-                checkFunctionOrMethodDeclaration(node);
-                checkGrammarForGenerator(node);
-                checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
-                checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+            else if (type && !hasExplicitReturn) {
+                // minimal check: function has syntactic return type annotation and no explicit return statements in the body
+                // this function does not conform to the specification.
+                // NOTE: having returnType !== undefined is a precondition for entering this branch so func.type will always be present
+                error(ts.getEffectiveReturnTypeNode(func), ts.Diagnostics.A_function_whose_declared_type_is_neither_void_nor_any_must_return_a_value);
             }
-        }
-        function checkJSDocTypeAliasTag(node) {
-            if (!node.typeExpression) {
-                // If the node had `@property` tags, `typeExpression` would have been set to the first property tag.
-                error(node.name, ts.Diagnostics.JSDoc_typedef_tag_should_either_have_a_type_annotation_or_be_followed_by_property_or_member_tags);
+            else if (type && strictNullChecks && !isTypeAssignableTo(undefinedType, type)) {
+                error(ts.getEffectiveReturnTypeNode(func) || func, ts.Diagnostics.Function_lacks_ending_return_statement_and_return_type_does_not_include_undefined);
             }
-            if (node.name) {
-                checkTypeNameIsReserved(node.name, ts.Diagnostics.Type_alias_name_cannot_be_0);
+            else if (compilerOptions.noImplicitReturns) {
+                if (!type) {
+                    // If return type annotation is omitted check if function has any explicit return statements.
+                    // If it does not have any - its inferred return type is void - don't do any checks.
+                    // Otherwise get inferred return type from function body and report error only if it is not void / anytype
+                    if (!hasExplicitReturn) {
+                        return;
+                    }
+                    var inferredReturnType = getReturnTypeOfSignature(getSignatureFromDeclaration(func));
+                    if (isUnwrappedReturnTypeVoidOrAny(func, inferredReturnType)) {
+                        return;
+                    }
+                }
+                error(ts.getEffectiveReturnTypeNode(func) || func, ts.Diagnostics.Not_all_code_paths_return_a_value);
             }
-            checkSourceElement(node.typeExpression);
         }
-        function checkJSDocTemplateTag(node) {
-            checkSourceElement(node.constraint);
-            for (var _i = 0, _a = node.typeParameters; _i < _a.length; _i++) {
-                var tp = _a[_i];
-                checkSourceElement(tp);
+        function checkFunctionExpressionOrObjectLiteralMethod(node, checkMode) {
+            ts.Debug.assert(node.kind !== 165 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
+            checkNodeDeferred(node);
+            // The identityMapper object is used to indicate that function expressions are wildcards
+            if (checkMode && checkMode & 4 /* SkipContextSensitive */ && isContextSensitive(node)) {
+                // Skip parameters, return signature with return type that retains noncontextual parts so inferences can still be drawn in an early stage
+                if (!ts.getEffectiveReturnTypeNode(node) && !hasContextSensitiveParameters(node)) {
+                    // Return plain anyFunctionType if there is no possibility we'll make inferences from the return type
+                    var contextualSignature = getContextualSignature(node);
+                    if (contextualSignature && couldContainTypeVariables(getReturnTypeOfSignature(contextualSignature))) {
+                        var links = getNodeLinks(node);
+                        if (links.contextFreeType) {
+                            return links.contextFreeType;
+                        }
+                        var returnType = getReturnTypeFromBody(node, checkMode);
+                        var returnOnlySignature = createSignature(undefined, undefined, undefined, ts.emptyArray, returnType, /*resolvedTypePredicate*/ undefined, 0, 0 /* None */);
+                        var returnOnlyType = createAnonymousType(node.symbol, emptySymbols, [returnOnlySignature], ts.emptyArray, undefined, undefined);
+                        returnOnlyType.objectFlags |= 2097152 /* NonInferrableType */;
+                        return links.contextFreeType = returnOnlyType;
+                    }
+                }
+                return anyFunctionType;
             }
+            // Grammar checking
+            var hasGrammarError = checkGrammarFunctionLikeDeclaration(node);
+            if (!hasGrammarError && node.kind === 208 /* FunctionExpression */) {
+                checkGrammarForGenerator(node);
+            }
+            contextuallyCheckFunctionExpressionOrObjectLiteralMethod(node, checkMode);
+            return getTypeOfSymbol(getSymbolOfNode(node));
         }
-        function checkJSDocTypeTag(node) {
-            checkSourceElement(node.typeExpression);
-        }
-        function checkJSDocParameterTag(node) {
-            checkSourceElement(node.typeExpression);
-            if (!ts.getParameterSymbolFromJSDoc(node)) {
-                var decl = ts.getHostSignatureFromJSDoc(node);
-                // don't issue an error for invalid hosts -- just functions --
-                // and give a better error message when the host function mentions `arguments`
-                // but the tag doesn't have an array type
-                if (decl) {
-                    var i = ts.getJSDocTags(decl).filter(ts.isJSDocParameterTag).indexOf(node);
-                    if (i > -1 && i < decl.parameters.length && ts.isBindingPattern(decl.parameters[i].name)) {
+        function contextuallyCheckFunctionExpressionOrObjectLiteralMethod(node, checkMode) {
+            var links = getNodeLinks(node);
+            // Check if function expression is contextually typed and assign parameter types if so.
+            if (!(links.flags & 1024 /* ContextChecked */)) {
+                var contextualSignature = getContextualSignature(node);
+                // If a type check is started at a function expression that is an argument of a function call, obtaining the
+                // contextual type may recursively get back to here during overload resolution of the call. If so, we will have
+                // already assigned contextual types.
+                if (!(links.flags & 1024 /* ContextChecked */)) {
+                    links.flags |= 1024 /* ContextChecked */;
+                    var signature = ts.firstOrUndefined(getSignaturesOfType(getTypeOfSymbol(getSymbolOfNode(node)), 0 /* Call */));
+                    if (!signature) {
                         return;
                     }
-                    if (!containsArgumentsReference(decl)) {
-                        if (ts.isQualifiedName(node.name)) {
-                            error(node.name, ts.Diagnostics.Qualified_name_0_is_not_allowed_without_a_leading_param_object_1, ts.entityNameToString(node.name), ts.entityNameToString(node.name.left));
+                    if (isContextSensitive(node)) {
+                        if (contextualSignature) {
+                            var inferenceContext = getInferenceContext(node);
+                            if (checkMode && checkMode & 2 /* Inferential */) {
+                                inferFromAnnotatedParameters(signature, contextualSignature, inferenceContext);
+                            }
+                            var instantiatedContextualSignature = inferenceContext ?
+                                instantiateSignature(contextualSignature, inferenceContext.mapper) : contextualSignature;
+                            assignContextualParameterTypes(signature, instantiatedContextualSignature);
                         }
                         else {
-                            error(node.name, ts.Diagnostics.JSDoc_param_tag_has_name_0_but_there_is_no_parameter_with_that_name, ts.idText(node.name));
+                            // Force resolution of all parameter types such that the absence of a contextual type is consistently reflected.
+                            assignNonContextualParameterTypes(signature);
                         }
                     }
-                    else if (ts.findLast(ts.getJSDocTags(decl), ts.isJSDocParameterTag) === node &&
-                        node.typeExpression && node.typeExpression.type &&
-                        !isArrayType(getTypeFromTypeNode(node.typeExpression.type))) {
-                        error(node.name, ts.Diagnostics.JSDoc_param_tag_has_name_0_but_there_is_no_parameter_with_that_name_It_would_match_arguments_if_it_had_an_array_type, ts.idText(node.name.kind === 153 /* QualifiedName */ ? node.name.right : node.name));
+                    if (contextualSignature && !getReturnTypeFromAnnotation(node) && !signature.resolvedReturnType) {
+                        var returnType = getReturnTypeFromBody(node, checkMode);
+                        if (!signature.resolvedReturnType) {
+                            signature.resolvedReturnType = returnType;
+                        }
                     }
+                    checkSignatureDeclaration(node);
                 }
             }
         }
-        function checkJSDocPropertyTag(node) {
-            checkSourceElement(node.typeExpression);
-        }
-        function checkJSDocFunctionType(node) {
-            if (produceDiagnostics && !node.type && !ts.isJSDocConstructSignature(node)) {
-                reportImplicitAny(node, anyType);
+        function checkFunctionExpressionOrObjectLiteralMethodDeferred(node) {
+            ts.Debug.assert(node.kind !== 165 /* MethodDeclaration */ || ts.isObjectLiteralMethod(node));
+            var functionFlags = ts.getFunctionFlags(node);
+            var returnType = getReturnTypeFromAnnotation(node);
+            checkAllCodePathsInNonVoidFunctionReturnOrThrow(node, returnType);
+            if (node.body) {
+                if (!ts.getEffectiveReturnTypeNode(node)) {
+                    // There are some checks that are only performed in getReturnTypeFromBody, that may produce errors
+                    // we need. An example is the noImplicitAny errors resulting from widening the return expression
+                    // of a function. Because checking of function expression bodies is deferred, there was never an
+                    // appropriate time to do this during the main walk of the file (see the comment at the top of
+                    // checkFunctionExpressionBodies). So it must be done now.
+                    getReturnTypeOfSignature(getSignatureFromDeclaration(node));
+                }
+                if (node.body.kind === 230 /* Block */) {
+                    checkSourceElement(node.body);
+                }
+                else {
+                    // From within an async function you can return either a non-promise value or a promise. Any
+                    // Promise/A+ compatible implementation will always assimilate any foreign promise, so we
+                    // should not be checking assignability of a promise to the return type. Instead, we need to
+                    // check assignability of the awaited type of the expression body against the promised type of
+                    // its return type annotation.
+                    var exprType = checkExpression(node.body);
+                    var returnOrPromisedType = returnType && unwrapReturnType(returnType, functionFlags);
+                    if (returnOrPromisedType) {
+                        if ((functionFlags & 3 /* AsyncGenerator */) === 2 /* Async */) { // Async function
+                            var awaitedType = checkAwaitedType(exprType, node.body, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+                            checkTypeAssignableToAndOptionallyElaborate(awaitedType, returnOrPromisedType, node.body, node.body);
+                        }
+                        else { // Normal function
+                            checkTypeAssignableToAndOptionallyElaborate(exprType, returnOrPromisedType, node.body, node.body);
+                        }
+                    }
+                }
             }
-            checkSignatureDeclaration(node);
         }
-        function checkJSDocImplementsTag(node) {
-            var classLike = ts.getEffectiveJSDocHost(node);
-            if (!classLike || !ts.isClassDeclaration(classLike) && !ts.isClassExpression(classLike)) {
-                error(classLike, ts.Diagnostics.JSDoc_0_is_not_attached_to_a_class, ts.idText(node.tagName));
+        function checkArithmeticOperandType(operand, type, diagnostic, isAwaitValid) {
+            if (isAwaitValid === void 0) { isAwaitValid = false; }
+            if (!isTypeAssignableTo(type, numberOrBigIntType)) {
+                var awaitedType = isAwaitValid && getAwaitedTypeOfPromise(type);
+                errorAndMaybeSuggestAwait(operand, !!awaitedType && isTypeAssignableTo(awaitedType, numberOrBigIntType), diagnostic);
+                return false;
             }
+            return true;
         }
-        function checkJSDocAugmentsTag(node) {
-            var classLike = ts.getEffectiveJSDocHost(node);
-            if (!classLike || !ts.isClassDeclaration(classLike) && !ts.isClassExpression(classLike)) {
-                error(classLike, ts.Diagnostics.JSDoc_0_is_not_attached_to_a_class, ts.idText(node.tagName));
-                return;
+        function isReadonlyAssignmentDeclaration(d) {
+            if (!ts.isCallExpression(d)) {
+                return false;
             }
-            var augmentsTags = ts.getJSDocTags(classLike).filter(ts.isJSDocAugmentsTag);
-            ts.Debug.assert(augmentsTags.length > 0);
-            if (augmentsTags.length > 1) {
-                error(augmentsTags[1], ts.Diagnostics.Class_declarations_cannot_have_more_than_one_augments_or_extends_tag);
+            if (!ts.isBindableObjectDefinePropertyCall(d)) {
+                return false;
             }
-            var name = getIdentifierFromEntityNameExpression(node.class.expression);
-            var extend = ts.getClassExtendsHeritageElement(classLike);
-            if (extend) {
-                var className = getIdentifierFromEntityNameExpression(extend.expression);
-                if (className && name.escapedText !== className.escapedText) {
-                    error(name, ts.Diagnostics.JSDoc_0_1_does_not_match_the_extends_2_clause, ts.idText(node.tagName), ts.idText(name), ts.idText(className));
+            var objectLitType = checkExpressionCached(d.arguments[2]);
+            var valueType = getTypeOfPropertyOfType(objectLitType, "value");
+            if (valueType) {
+                var writableProp = getPropertyOfType(objectLitType, "writable");
+                var writableType = writableProp && getTypeOfSymbol(writableProp);
+                if (!writableType || writableType === falseType || writableType === regularFalseType) {
+                    return true;
+                }
+                // We include this definition whereupon we walk back and check the type at the declaration because
+                // The usual definition of `Object.defineProperty` will _not_ cause literal types to be preserved in the
+                // argument types, should the type be contextualized by the call itself.
+                if (writableProp && writableProp.valueDeclaration && ts.isPropertyAssignment(writableProp.valueDeclaration)) {
+                    var initializer = writableProp.valueDeclaration.initializer;
+                    var rawOriginalType = checkExpression(initializer);
+                    if (rawOriginalType === falseType || rawOriginalType === regularFalseType) {
+                        return true;
+                    }
                 }
+                return false;
             }
+            var setProp = getPropertyOfType(objectLitType, "set");
+            return !setProp;
         }
-        function getIdentifierFromEntityNameExpression(node) {
-            switch (node.kind) {
-                case 75 /* Identifier */:
-                    return node;
-                case 194 /* PropertyAccessExpression */:
-                    return node.name;
-                default:
-                    return undefined;
-            }
+        function isReadonlySymbol(symbol) {
+            // The following symbols are considered read-only:
+            // Properties with a 'readonly' modifier
+            // Variables declared with 'const'
+            // Get accessors without matching set accessors
+            // Enum members
+            // Object.defineProperty assignments with writable false or no setter
+            // Unions and intersections of the above (unions and intersections eagerly set isReadonly on creation)
+            return !!(ts.getCheckFlags(symbol) & 8 /* Readonly */ ||
+                symbol.flags & 4 /* Property */ && ts.getDeclarationModifierFlagsFromSymbol(symbol) & 64 /* Readonly */ ||
+                symbol.flags & 3 /* Variable */ && getDeclarationNodeFlagsFromSymbol(symbol) & 2 /* Const */ ||
+                symbol.flags & 98304 /* Accessor */ && !(symbol.flags & 65536 /* SetAccessor */) ||
+                symbol.flags & 8 /* EnumMember */ ||
+                ts.some(symbol.declarations, isReadonlyAssignmentDeclaration));
         }
-        function checkFunctionOrMethodDeclaration(node) {
-            checkDecorators(node);
-            checkSignatureDeclaration(node);
-            var functionFlags = ts.getFunctionFlags(node);
-            // Do not use hasDynamicName here, because that returns false for well known symbols.
-            // We want to perform checkComputedPropertyName for all computed properties, including
-            // well known symbols.
-            if (node.name && node.name.kind === 154 /* ComputedPropertyName */) {
-                // This check will account for methods in class/interface declarations,
-                // as well as accessors in classes/object literals
-                checkComputedPropertyName(node.name);
+        function isAssignmentToReadonlyEntity(expr, symbol, assignmentKind) {
+            var _a, _b;
+            if (assignmentKind === 0 /* None */) {
+                // no assigment means it doesn't matter whether the entity is readonly
+                return false;
             }
-            if (!hasNonBindableDynamicName(node)) {
-                // first we want to check the local symbol that contain this declaration
-                // - if node.localSymbol !== undefined - this is current declaration is exported and localSymbol points to the local symbol
-                // - if node.localSymbol === undefined - this node is non-exported so we can just pick the result of getSymbolOfNode
-                var symbol = getSymbolOfNode(node);
-                var localSymbol = node.localSymbol || symbol;
-                // Since the javascript won't do semantic analysis like typescript,
-                // if the javascript file comes before the typescript file and both contain same name functions,
-                // checkFunctionOrConstructorSymbol wouldn't be called if we didnt ignore javascript function.
-                var firstDeclaration = ts.find(localSymbol.declarations, 
-                // Get first non javascript function declaration
-                function (declaration) { return declaration.kind === node.kind && !(declaration.flags & 131072 /* JavaScriptFile */); });
-                // Only type check the symbol once
-                if (node === firstDeclaration) {
-                    checkFunctionOrConstructorSymbol(localSymbol);
-                }
-                if (symbol.parent) {
-                    // run check once for the first declaration
-                    if (ts.getDeclarationOfKind(symbol, node.kind) === node) {
-                        // run check on export symbol to check that modifiers agree across all exported declarations
-                        checkFunctionOrConstructorSymbol(symbol);
+            if (isReadonlySymbol(symbol)) {
+                // Allow assignments to readonly properties within constructors of the same class declaration.
+                if (symbol.flags & 4 /* Property */ &&
+                    ts.isAccessExpression(expr) &&
+                    expr.expression.kind === 107 /* ThisKeyword */) {
+                    // Look for if this is the constructor for the class that `symbol` is a property of.
+                    var ctor = ts.getContainingFunction(expr);
+                    if (!(ctor && (ctor.kind === 166 /* Constructor */ || isJSConstructor(ctor)))) {
+                        return true;
+                    }
+                    if (symbol.valueDeclaration) {
+                        var isAssignmentDeclaration_1 = ts.isBinaryExpression(symbol.valueDeclaration);
+                        var isLocalPropertyDeclaration = ctor.parent === symbol.valueDeclaration.parent;
+                        var isLocalParameterProperty = ctor === symbol.valueDeclaration.parent;
+                        var isLocalThisPropertyAssignment = isAssignmentDeclaration_1 && ((_a = symbol.parent) === null || _a === void 0 ? void 0 : _a.valueDeclaration) === ctor.parent;
+                        var isLocalThisPropertyAssignmentConstructorFunction = isAssignmentDeclaration_1 && ((_b = symbol.parent) === null || _b === void 0 ? void 0 : _b.valueDeclaration) === ctor;
+                        var isWriteableSymbol = isLocalPropertyDeclaration
+                            || isLocalParameterProperty
+                            || isLocalThisPropertyAssignment
+                            || isLocalThisPropertyAssignmentConstructorFunction;
+                        return !isWriteableSymbol;
                     }
                 }
+                return true;
             }
-            var body = node.kind === 160 /* MethodSignature */ ? undefined : node.body;
-            checkSourceElement(body);
-            checkAllCodePathsInNonVoidFunctionReturnOrThrow(node, getReturnTypeFromAnnotation(node));
-            if (produceDiagnostics && !ts.getEffectiveReturnTypeNode(node)) {
-                // Report an implicit any error if there is no body, no explicit return type, and node is not a private method
-                // in an ambient context
-                if (ts.nodeIsMissing(body) && !isPrivateWithinAmbient(node)) {
-                    reportImplicitAny(node, anyType);
-                }
-                if (functionFlags & 1 /* Generator */ && ts.nodeIsPresent(body)) {
-                    // A generator with a body and no type annotation can still cause errors. It can error if the
-                    // yielded values have no common supertype, or it can give an implicit any error if it has no
-                    // yielded values. The only way to trigger these errors is to try checking its return type.
-                    getReturnTypeOfSignature(getSignatureFromDeclaration(node));
-                }
-            }
-            // A js function declaration can have a @type tag instead of a return type node, but that type must have a call signature
-            if (ts.isInJSFile(node)) {
-                var typeTag = ts.getJSDocTypeTag(node);
-                if (typeTag && typeTag.typeExpression && !getContextualCallSignature(getTypeFromTypeNode(typeTag.typeExpression), node)) {
-                    error(typeTag, ts.Diagnostics.The_type_of_a_function_declaration_must_match_the_function_s_signature);
+            if (ts.isAccessExpression(expr)) {
+                // references through namespace import should be readonly
+                var node = ts.skipParentheses(expr.expression);
+                if (node.kind === 78 /* Identifier */) {
+                    var symbol_2 = getNodeLinks(node).resolvedSymbol;
+                    if (symbol_2.flags & 2097152 /* Alias */) {
+                        var declaration = getDeclarationOfAliasSymbol(symbol_2);
+                        return !!declaration && declaration.kind === 263 /* NamespaceImport */;
+                    }
                 }
             }
+            return false;
         }
-        function registerForUnusedIdentifiersCheck(node) {
-            // May be in a call such as getTypeOfNode that happened to call this. But potentiallyUnusedIdentifiers is only defined in the scope of `checkSourceFile`.
-            if (produceDiagnostics) {
-                var sourceFile = ts.getSourceFileOfNode(node);
-                var potentiallyUnusedIdentifiers = allPotentiallyUnusedIdentifiers.get(sourceFile.path);
-                if (!potentiallyUnusedIdentifiers) {
-                    potentiallyUnusedIdentifiers = [];
-                    allPotentiallyUnusedIdentifiers.set(sourceFile.path, potentiallyUnusedIdentifiers);
-                }
-                // TODO: GH#22580
-                // Debug.assert(addToSeen(seenPotentiallyUnusedIdentifiers, getNodeId(node)), "Adding potentially-unused identifier twice");
-                potentiallyUnusedIdentifiers.push(node);
+        function checkReferenceExpression(expr, invalidReferenceMessage, invalidOptionalChainMessage) {
+            // References are combinations of identifiers, parentheses, and property accesses.
+            var node = ts.skipOuterExpressions(expr, 6 /* Assertions */ | 1 /* Parentheses */);
+            if (node.kind !== 78 /* Identifier */ && !ts.isAccessExpression(node)) {
+                error(expr, invalidReferenceMessage);
+                return false;
             }
-        }
-        function checkUnusedIdentifiers(potentiallyUnusedIdentifiers, addDiagnostic) {
-            for (var _i = 0, potentiallyUnusedIdentifiers_1 = potentiallyUnusedIdentifiers; _i < potentiallyUnusedIdentifiers_1.length; _i++) {
-                var node = potentiallyUnusedIdentifiers_1[_i];
-                switch (node.kind) {
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                        checkUnusedClassMembers(node, addDiagnostic);
-                        checkUnusedTypeParameters(node, addDiagnostic);
-                        break;
-                    case 290 /* SourceFile */:
-                    case 249 /* ModuleDeclaration */:
-                    case 223 /* Block */:
-                    case 251 /* CaseBlock */:
-                    case 230 /* ForStatement */:
-                    case 231 /* ForInStatement */:
-                    case 232 /* ForOfStatement */:
-                        checkUnusedLocalsAndParameters(node, addDiagnostic);
-                        break;
-                    case 162 /* Constructor */:
-                    case 201 /* FunctionExpression */:
-                    case 244 /* FunctionDeclaration */:
-                    case 202 /* ArrowFunction */:
-                    case 161 /* MethodDeclaration */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        if (node.body) { // Don't report unused parameters in overloads
-                            checkUnusedLocalsAndParameters(node, addDiagnostic);
-                        }
-                        checkUnusedTypeParameters(node, addDiagnostic);
-                        break;
-                    case 160 /* MethodSignature */:
-                    case 165 /* CallSignature */:
-                    case 166 /* ConstructSignature */:
-                    case 170 /* FunctionType */:
-                    case 171 /* ConstructorType */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                        checkUnusedTypeParameters(node, addDiagnostic);
-                        break;
-                    case 181 /* InferType */:
-                        checkUnusedInferTypeParameter(node, addDiagnostic);
-                        break;
-                    default:
-                        ts.Debug.assertNever(node, "Node should not have been registered for unused identifiers check");
-                }
+            if (node.flags & 32 /* OptionalChain */) {
+                error(expr, invalidOptionalChainMessage);
+                return false;
             }
+            return true;
         }
-        function errorUnusedLocal(declaration, name, addDiagnostic) {
-            var node = ts.getNameOfDeclaration(declaration) || declaration;
-            var message = isTypeDeclaration(declaration) ? ts.Diagnostics._0_is_declared_but_never_used : ts.Diagnostics._0_is_declared_but_its_value_is_never_read;
-            addDiagnostic(declaration, 0 /* Local */, ts.createDiagnosticForNode(node, message, name));
-        }
-        function isIdentifierThatStartsWithUnderscore(node) {
-            return ts.isIdentifier(node) && ts.idText(node).charCodeAt(0) === 95 /* _ */;
-        }
-        function checkUnusedClassMembers(node, addDiagnostic) {
-            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                var member = _a[_i];
-                switch (member.kind) {
-                    case 161 /* MethodDeclaration */:
-                    case 159 /* PropertyDeclaration */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        if (member.kind === 164 /* SetAccessor */ && member.symbol.flags & 32768 /* GetAccessor */) {
-                            // Already would have reported an error on the getter.
-                            break;
-                        }
-                        var symbol = getSymbolOfNode(member);
-                        if (!symbol.isReferenced
-                            && (ts.hasModifier(member, 8 /* Private */) || ts.isNamedDeclaration(member) && ts.isPrivateIdentifier(member.name))
-                            && !(member.flags & 8388608 /* Ambient */)) {
-                            addDiagnostic(member, 0 /* Local */, ts.createDiagnosticForNode(member.name, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, symbolToString(symbol)));
-                        }
-                        break;
-                    case 162 /* Constructor */:
-                        for (var _b = 0, _c = member.parameters; _b < _c.length; _b++) {
-                            var parameter = _c[_b];
-                            if (!parameter.symbol.isReferenced && ts.hasModifier(parameter, 8 /* Private */)) {
-                                addDiagnostic(parameter, 0 /* Local */, ts.createDiagnosticForNode(parameter.name, ts.Diagnostics.Property_0_is_declared_but_its_value_is_never_read, ts.symbolName(parameter.symbol)));
-                            }
-                        }
-                        break;
-                    case 167 /* IndexSignature */:
-                    case 222 /* SemicolonClassElement */:
-                        // Can't be private
-                        break;
-                    default:
-                        ts.Debug.fail();
-                }
+        function checkDeleteExpression(node) {
+            checkExpression(node.expression);
+            var expr = ts.skipParentheses(node.expression);
+            if (!ts.isAccessExpression(expr)) {
+                error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_must_be_a_property_reference);
+                return booleanType;
             }
-        }
-        function checkUnusedInferTypeParameter(node, addDiagnostic) {
-            var typeParameter = node.typeParameter;
-            if (isTypeParameterUnused(typeParameter)) {
-                addDiagnostic(node, 1 /* Parameter */, ts.createDiagnosticForNode(node, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, ts.idText(typeParameter.name)));
+            if (ts.isPropertyAccessExpression(expr) && ts.isPrivateIdentifier(expr.name)) {
+                error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_cannot_be_a_private_identifier);
             }
-        }
-        function checkUnusedTypeParameters(node, addDiagnostic) {
-            // Only report errors on the last declaration for the type parameter container;
-            // this ensures that all uses have been accounted for.
-            if (ts.last(getSymbolOfNode(node).declarations) !== node)
-                return;
-            var typeParameters = ts.getEffectiveTypeParameterDeclarations(node);
-            var seenParentsWithEveryUnused = new ts.NodeSet();
-            for (var _i = 0, typeParameters_3 = typeParameters; _i < typeParameters_3.length; _i++) {
-                var typeParameter = typeParameters_3[_i];
-                if (!isTypeParameterUnused(typeParameter))
-                    continue;
-                var name = ts.idText(typeParameter.name);
-                var parent = typeParameter.parent;
-                if (parent.kind !== 181 /* InferType */ && parent.typeParameters.every(isTypeParameterUnused)) {
-                    if (seenParentsWithEveryUnused.tryAdd(parent)) {
-                        var range = ts.isJSDocTemplateTag(parent)
-                            // Whole @template tag
-                            ? ts.rangeOfNode(parent)
-                            // Include the `<>` in the error message
-                            : ts.rangeOfTypeParameters(parent.typeParameters);
-                        var only = parent.typeParameters.length === 1;
-                        var message = only ? ts.Diagnostics._0_is_declared_but_its_value_is_never_read : ts.Diagnostics.All_type_parameters_are_unused;
-                        var arg0 = only ? name : undefined;
-                        addDiagnostic(typeParameter, 1 /* Parameter */, ts.createFileDiagnostic(ts.getSourceFileOfNode(parent), range.pos, range.end - range.pos, message, arg0));
-                    }
-                }
-                else {
-                    addDiagnostic(typeParameter, 1 /* Parameter */, ts.createDiagnosticForNode(typeParameter, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, name));
+            var links = getNodeLinks(expr);
+            var symbol = getExportSymbolOfValueSymbolIfExported(links.resolvedSymbol);
+            if (symbol) {
+                if (isReadonlySymbol(symbol)) {
+                    error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_cannot_be_a_read_only_property);
                 }
+                checkDeleteExpressionMustBeOptional(expr, getTypeOfSymbol(symbol));
             }
+            return booleanType;
         }
-        function isTypeParameterUnused(typeParameter) {
-            return !(getMergedSymbol(typeParameter.symbol).isReferenced & 262144 /* TypeParameter */) && !isIdentifierThatStartsWithUnderscore(typeParameter.name);
-        }
-        function addToGroup(map, key, value, getKey) {
-            var keyString = String(getKey(key));
-            var group = map.get(keyString);
-            if (group) {
-                group[1].push(value);
-            }
-            else {
-                map.set(keyString, [key, [value]]);
+        function checkDeleteExpressionMustBeOptional(expr, type) {
+            var AnyOrUnknownOrNeverFlags = 3 /* AnyOrUnknown */ | 131072 /* Never */;
+            if (strictNullChecks && !(type.flags & AnyOrUnknownOrNeverFlags) && !(getFalsyFlags(type) & 32768 /* Undefined */)) {
+                error(expr, ts.Diagnostics.The_operand_of_a_delete_operator_must_be_optional);
             }
         }
-        function tryGetRootParameterDeclaration(node) {
-            return ts.tryCast(ts.getRootDeclaration(node), ts.isParameter);
+        function checkTypeOfExpression(node) {
+            checkExpression(node.expression);
+            return typeofType;
         }
-        function isValidUnusedLocalDeclaration(declaration) {
-            if (ts.isBindingElement(declaration) && isIdentifierThatStartsWithUnderscore(declaration.name)) {
-                return !!ts.findAncestor(declaration.parent, function (ancestor) {
-                    return ts.isArrayBindingPattern(ancestor) || ts.isVariableDeclaration(ancestor) || ts.isVariableDeclarationList(ancestor) ? false :
-                        ts.isForOfStatement(ancestor) ? true : "quit";
-                });
-            }
-            return ts.isAmbientModule(declaration) ||
-                (ts.isVariableDeclaration(declaration) && ts.isForInOrOfStatement(declaration.parent.parent) || isImportedDeclaration(declaration)) && isIdentifierThatStartsWithUnderscore(declaration.name);
+        function checkVoidExpression(node) {
+            checkExpression(node.expression);
+            return undefinedWideningType;
         }
-        function checkUnusedLocalsAndParameters(nodeWithLocals, addDiagnostic) {
-            // Ideally we could use the ImportClause directly as a key, but must wait until we have full ES6 maps. So must store key along with value.
-            var unusedImports = ts.createMap();
-            var unusedDestructures = ts.createMap();
-            var unusedVariables = ts.createMap();
-            nodeWithLocals.locals.forEach(function (local) {
-                // If it's purely a type parameter, ignore, will be checked in `checkUnusedTypeParameters`.
-                // If it's a type parameter merged with a parameter, check if the parameter-side is used.
-                if (local.flags & 262144 /* TypeParameter */ ? !(local.flags & 3 /* Variable */ && !(local.isReferenced & 3 /* Variable */)) : local.isReferenced || local.exportSymbol) {
-                    return;
-                }
-                for (var _i = 0, _a = local.declarations; _i < _a.length; _i++) {
-                    var declaration = _a[_i];
-                    if (isValidUnusedLocalDeclaration(declaration)) {
-                        continue;
-                    }
-                    if (isImportedDeclaration(declaration)) {
-                        addToGroup(unusedImports, importClauseFromImported(declaration), declaration, getNodeId);
-                    }
-                    else if (ts.isBindingElement(declaration) && ts.isObjectBindingPattern(declaration.parent)) {
-                        // In `{ a, ...b }, `a` is considered used since it removes a property from `b`. `b` may still be unused though.
-                        var lastElement = ts.last(declaration.parent.elements);
-                        if (declaration === lastElement || !ts.last(declaration.parent.elements).dotDotDotToken) {
-                            addToGroup(unusedDestructures, declaration.parent, declaration, getNodeId);
+        function checkAwaitExpression(node) {
+            // Grammar checking
+            if (produceDiagnostics) {
+                if (!(node.flags & 32768 /* AwaitContext */)) {
+                    if (ts.isInTopLevelContext(node)) {
+                        var sourceFile = ts.getSourceFileOfNode(node);
+                        if (!hasParseDiagnostics(sourceFile)) {
+                            var span = void 0;
+                            if (!ts.isEffectiveExternalModule(sourceFile, compilerOptions)) {
+                                if (!span)
+                                    span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
+                                var diagnostic = ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.await_expressions_are_only_allowed_at_the_top_level_of_a_file_when_that_file_is_a_module_but_this_file_has_no_imports_or_exports_Consider_adding_an_empty_export_to_make_this_file_a_module);
+                                diagnostics.add(diagnostic);
+                            }
+                            if ((moduleKind !== ts.ModuleKind.ESNext && moduleKind !== ts.ModuleKind.System) || languageVersion < 4 /* ES2017 */) {
+                                span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
+                                var diagnostic = ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.Top_level_await_expressions_are_only_allowed_when_the_module_option_is_set_to_esnext_or_system_and_the_target_option_is_set_to_es2017_or_higher);
+                                diagnostics.add(diagnostic);
+                            }
                         }
                     }
-                    else if (ts.isVariableDeclaration(declaration)) {
-                        addToGroup(unusedVariables, declaration.parent, declaration, getNodeId);
-                    }
                     else {
-                        var parameter = local.valueDeclaration && tryGetRootParameterDeclaration(local.valueDeclaration);
-                        var name = local.valueDeclaration && ts.getNameOfDeclaration(local.valueDeclaration);
-                        if (parameter && name) {
-                            if (!ts.isParameterPropertyDeclaration(parameter, parameter.parent) && !ts.parameterIsThisKeyword(parameter) && !isIdentifierThatStartsWithUnderscore(name)) {
-                                addDiagnostic(parameter, 1 /* Parameter */, ts.createDiagnosticForNode(name, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, ts.symbolName(local)));
+                        // use of 'await' in non-async function
+                        var sourceFile = ts.getSourceFileOfNode(node);
+                        if (!hasParseDiagnostics(sourceFile)) {
+                            var span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
+                            var diagnostic = ts.createFileDiagnostic(sourceFile, span.start, span.length, ts.Diagnostics.await_expressions_are_only_allowed_within_async_functions_and_at_the_top_levels_of_modules);
+                            var func = ts.getContainingFunction(node);
+                            if (func && func.kind !== 166 /* Constructor */ && (ts.getFunctionFlags(func) & 2 /* Async */) === 0) {
+                                var relatedInfo = ts.createDiagnosticForNode(func, ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async);
+                                ts.addRelatedInfo(diagnostic, relatedInfo);
                             }
-                        }
-                        else {
-                            errorUnusedLocal(declaration, ts.symbolName(local), addDiagnostic);
+                            diagnostics.add(diagnostic);
                         }
                     }
                 }
-            });
-            unusedImports.forEach(function (_a) {
-                var importClause = _a[0], unuseds = _a[1];
-                var importDecl = importClause.parent;
-                var nDeclarations = (importClause.name ? 1 : 0) +
-                    (importClause.namedBindings ?
-                        (importClause.namedBindings.kind === 256 /* NamespaceImport */ ? 1 : importClause.namedBindings.elements.length)
-                        : 0);
-                if (nDeclarations === unuseds.length) {
-                    addDiagnostic(importDecl, 0 /* Local */, unuseds.length === 1
-                        ? ts.createDiagnosticForNode(importDecl, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, ts.idText(ts.first(unuseds).name))
-                        : ts.createDiagnosticForNode(importDecl, ts.Diagnostics.All_imports_in_import_declaration_are_unused));
+                if (isInParameterInitializerBeforeContainingFunction(node)) {
+                    error(node, ts.Diagnostics.await_expressions_cannot_be_used_in_a_parameter_initializer);
                 }
-                else {
-                    for (var _i = 0, unuseds_1 = unuseds; _i < unuseds_1.length; _i++) {
-                        var unused = unuseds_1[_i];
-                        errorUnusedLocal(unused, ts.idText(unused.name), addDiagnostic);
+            }
+            var operandType = checkExpression(node.expression);
+            var awaitedType = checkAwaitedType(operandType, node, ts.Diagnostics.Type_of_await_operand_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
+            if (awaitedType === operandType && awaitedType !== errorType && !(operandType.flags & 3 /* AnyOrUnknown */)) {
+                addErrorOrSuggestion(/*isError*/ false, ts.createDiagnosticForNode(node, ts.Diagnostics.await_has_no_effect_on_the_type_of_this_expression));
+            }
+            return awaitedType;
+        }
+        function checkPrefixUnaryExpression(node) {
+            var operandType = checkExpression(node.operand);
+            if (operandType === silentNeverType) {
+                return silentNeverType;
+            }
+            switch (node.operand.kind) {
+                case 8 /* NumericLiteral */:
+                    switch (node.operator) {
+                        case 40 /* MinusToken */:
+                            return getFreshTypeOfLiteralType(getLiteralType(-node.operand.text));
+                        case 39 /* PlusToken */:
+                            return getFreshTypeOfLiteralType(getLiteralType(+node.operand.text));
                     }
-                }
-            });
-            unusedDestructures.forEach(function (_a) {
-                var bindingPattern = _a[0], bindingElements = _a[1];
-                var kind = tryGetRootParameterDeclaration(bindingPattern.parent) ? 1 /* Parameter */ : 0 /* Local */;
-                if (bindingPattern.elements.length === bindingElements.length) {
-                    if (bindingElements.length === 1 && bindingPattern.parent.kind === 242 /* VariableDeclaration */ && bindingPattern.parent.parent.kind === 243 /* VariableDeclarationList */) {
-                        addToGroup(unusedVariables, bindingPattern.parent.parent, bindingPattern.parent, getNodeId);
+                    break;
+                case 9 /* BigIntLiteral */:
+                    if (node.operator === 40 /* MinusToken */) {
+                        return getFreshTypeOfLiteralType(getLiteralType({
+                            negative: true,
+                            base10Value: ts.parsePseudoBigInt(node.operand.text)
+                        }));
                     }
-                    else {
-                        addDiagnostic(bindingPattern, kind, bindingElements.length === 1
-                            ? ts.createDiagnosticForNode(bindingPattern, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(ts.first(bindingElements).name))
-                            : ts.createDiagnosticForNode(bindingPattern, ts.Diagnostics.All_destructured_elements_are_unused));
+            }
+            switch (node.operator) {
+                case 39 /* PlusToken */:
+                case 40 /* MinusToken */:
+                case 54 /* TildeToken */:
+                    checkNonNullType(operandType, node.operand);
+                    if (maybeTypeOfKind(operandType, 12288 /* ESSymbolLike */)) {
+                        error(node.operand, ts.Diagnostics.The_0_operator_cannot_be_applied_to_type_symbol, ts.tokenToString(node.operator));
                     }
-                }
-                else {
-                    for (var _i = 0, bindingElements_1 = bindingElements; _i < bindingElements_1.length; _i++) {
-                        var e = bindingElements_1[_i];
-                        addDiagnostic(e, kind, ts.createDiagnosticForNode(e, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(e.name)));
+                    if (node.operator === 39 /* PlusToken */) {
+                        if (maybeTypeOfKind(operandType, 2112 /* BigIntLike */)) {
+                            error(node.operand, ts.Diagnostics.Operator_0_cannot_be_applied_to_type_1, ts.tokenToString(node.operator), typeToString(getBaseTypeOfLiteralType(operandType)));
+                        }
+                        return numberType;
                     }
-                }
-            });
-            unusedVariables.forEach(function (_a) {
-                var declarationList = _a[0], declarations = _a[1];
-                if (declarationList.declarations.length === declarations.length) {
-                    addDiagnostic(declarationList, 0 /* Local */, declarations.length === 1
-                        ? ts.createDiagnosticForNode(ts.first(declarations).name, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(ts.first(declarations).name))
-                        : ts.createDiagnosticForNode(declarationList.parent.kind === 225 /* VariableStatement */ ? declarationList.parent : declarationList, ts.Diagnostics.All_variables_are_unused));
-                }
-                else {
-                    for (var _i = 0, declarations_5 = declarations; _i < declarations_5.length; _i++) {
-                        var decl = declarations_5[_i];
-                        addDiagnostic(decl, 0 /* Local */, ts.createDiagnosticForNode(decl, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(decl.name)));
+                    return getUnaryResultType(operandType);
+                case 53 /* ExclamationToken */:
+                    checkTruthinessExpression(node.operand);
+                    var facts = getTypeFacts(operandType) & (4194304 /* Truthy */ | 8388608 /* Falsy */);
+                    return facts === 4194304 /* Truthy */ ? falseType :
+                        facts === 8388608 /* Falsy */ ? trueType :
+                            booleanType;
+                case 45 /* PlusPlusToken */:
+                case 46 /* MinusMinusToken */:
+                    var ok = checkArithmeticOperandType(node.operand, checkNonNullType(operandType, node.operand), ts.Diagnostics.An_arithmetic_operand_must_be_of_type_any_number_bigint_or_an_enum_type);
+                    if (ok) {
+                        // run check only if former checks succeeded to avoid reporting cascading errors
+                        checkReferenceExpression(node.operand, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_may_not_be_an_optional_property_access);
                     }
-                }
-            });
-        }
-        function bindingNameText(name) {
-            switch (name.kind) {
-                case 75 /* Identifier */:
-                    return ts.idText(name);
-                case 190 /* ArrayBindingPattern */:
-                case 189 /* ObjectBindingPattern */:
-                    return bindingNameText(ts.cast(ts.first(name.elements), ts.isBindingElement).name);
-                default:
-                    return ts.Debug.assertNever(name);
+                    return getUnaryResultType(operandType);
             }
+            return errorType;
         }
-        function isImportedDeclaration(node) {
-            return node.kind === 255 /* ImportClause */ || node.kind === 258 /* ImportSpecifier */ || node.kind === 256 /* NamespaceImport */;
-        }
-        function importClauseFromImported(decl) {
-            return decl.kind === 255 /* ImportClause */ ? decl : decl.kind === 256 /* NamespaceImport */ ? decl.parent : decl.parent.parent;
-        }
-        function checkBlock(node) {
-            // Grammar checking for SyntaxKind.Block
-            if (node.kind === 223 /* Block */) {
-                checkGrammarStatementInAmbientContext(node);
-            }
-            if (ts.isFunctionOrModuleBlock(node)) {
-                var saveFlowAnalysisDisabled = flowAnalysisDisabled;
-                ts.forEach(node.statements, checkSourceElement);
-                flowAnalysisDisabled = saveFlowAnalysisDisabled;
-            }
-            else {
-                ts.forEach(node.statements, checkSourceElement);
+        function checkPostfixUnaryExpression(node) {
+            var operandType = checkExpression(node.operand);
+            if (operandType === silentNeverType) {
+                return silentNeverType;
             }
-            if (node.locals) {
-                registerForUnusedIdentifiersCheck(node);
+            var ok = checkArithmeticOperandType(node.operand, checkNonNullType(operandType, node.operand), ts.Diagnostics.An_arithmetic_operand_must_be_of_type_any_number_bigint_or_an_enum_type);
+            if (ok) {
+                // run check only if former checks succeeded to avoid reporting cascading errors
+                checkReferenceExpression(node.operand, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_operand_of_an_increment_or_decrement_operator_may_not_be_an_optional_property_access);
             }
+            return getUnaryResultType(operandType);
         }
-        function checkCollisionWithArgumentsInGeneratedCode(node) {
-            // no rest parameters \ declaration context \ overload - no codegen impact
-            if (languageVersion >= 2 /* ES2015 */ || compilerOptions.noEmit || !ts.hasRestParameter(node) || node.flags & 8388608 /* Ambient */ || ts.nodeIsMissing(node.body)) {
-                return;
+        function getUnaryResultType(operandType) {
+            if (maybeTypeOfKind(operandType, 2112 /* BigIntLike */)) {
+                return isTypeAssignableToKind(operandType, 3 /* AnyOrUnknown */) || maybeTypeOfKind(operandType, 296 /* NumberLike */)
+                    ? numberOrBigIntType
+                    : bigintType;
             }
-            ts.forEach(node.parameters, function (p) {
-                if (p.name && !ts.isBindingPattern(p.name) && p.name.escapedText === argumentsSymbol.escapedName) {
-                    error(p, ts.Diagnostics.Duplicate_identifier_arguments_Compiler_uses_arguments_to_initialize_rest_parameters);
-                }
-            });
+            // If it's not a bigint type, implicit coercion will result in a number
+            return numberType;
         }
-        function needCollisionCheckForIdentifier(node, identifier, name) {
-            if (!(identifier && identifier.escapedText === name)) {
-                return false;
+        // Return true if type might be of the given kind. A union or intersection type might be of a given
+        // kind if at least one constituent type is of the given kind.
+        function maybeTypeOfKind(type, kind) {
+            if (type.flags & kind) {
+                return true;
             }
-            if (node.kind === 159 /* PropertyDeclaration */ ||
-                node.kind === 158 /* PropertySignature */ ||
-                node.kind === 161 /* MethodDeclaration */ ||
-                node.kind === 160 /* MethodSignature */ ||
-                node.kind === 163 /* GetAccessor */ ||
-                node.kind === 164 /* SetAccessor */) {
-                // it is ok to have member named '_super' or '_this' - member access is always qualified
-                return false;
+            if (type.flags & 3145728 /* UnionOrIntersection */) {
+                var types = type.types;
+                for (var _i = 0, types_20 = types; _i < types_20.length; _i++) {
+                    var t = types_20[_i];
+                    if (maybeTypeOfKind(t, kind)) {
+                        return true;
+                    }
+                }
             }
-            if (node.flags & 8388608 /* Ambient */) {
-                // ambient context - no codegen impact
-                return false;
+            return false;
+        }
+        function isTypeAssignableToKind(source, kind, strict) {
+            if (source.flags & kind) {
+                return true;
             }
-            var root = ts.getRootDeclaration(node);
-            if (root.kind === 156 /* Parameter */ && ts.nodeIsMissing(root.parent.body)) {
-                // just an overload - no codegen impact
+            if (strict && source.flags & (3 /* AnyOrUnknown */ | 16384 /* Void */ | 32768 /* Undefined */ | 65536 /* Null */)) {
                 return false;
             }
-            return true;
+            return !!(kind & 296 /* NumberLike */) && isTypeAssignableTo(source, numberType) ||
+                !!(kind & 2112 /* BigIntLike */) && isTypeAssignableTo(source, bigintType) ||
+                !!(kind & 402653316 /* StringLike */) && isTypeAssignableTo(source, stringType) ||
+                !!(kind & 528 /* BooleanLike */) && isTypeAssignableTo(source, booleanType) ||
+                !!(kind & 16384 /* Void */) && isTypeAssignableTo(source, voidType) ||
+                !!(kind & 131072 /* Never */) && isTypeAssignableTo(source, neverType) ||
+                !!(kind & 65536 /* Null */) && isTypeAssignableTo(source, nullType) ||
+                !!(kind & 32768 /* Undefined */) && isTypeAssignableTo(source, undefinedType) ||
+                !!(kind & 4096 /* ESSymbol */) && isTypeAssignableTo(source, esSymbolType) ||
+                !!(kind & 67108864 /* NonPrimitive */) && isTypeAssignableTo(source, nonPrimitiveType);
         }
-        // this function will run after checking the source file so 'CaptureThis' is correct for all nodes
-        function checkIfThisIsCapturedInEnclosingScope(node) {
-            ts.findAncestor(node, function (current) {
-                if (getNodeCheckFlags(current) & 4 /* CaptureThis */) {
-                    var isDeclaration_1 = node.kind !== 75 /* Identifier */;
-                    if (isDeclaration_1) {
-                        error(ts.getNameOfDeclaration(node), ts.Diagnostics.Duplicate_identifier_this_Compiler_uses_variable_declaration_this_to_capture_this_reference);
-                    }
-                    else {
-                        error(node, ts.Diagnostics.Expression_resolves_to_variable_declaration_this_that_compiler_uses_to_capture_this_reference);
-                    }
-                    return true;
-                }
-                return false;
-            });
+        function allTypesAssignableToKind(source, kind, strict) {
+            return source.flags & 1048576 /* Union */ ?
+                ts.every(source.types, function (subType) { return allTypesAssignableToKind(subType, kind, strict); }) :
+                isTypeAssignableToKind(source, kind, strict);
         }
-        function checkIfNewTargetIsCapturedInEnclosingScope(node) {
-            ts.findAncestor(node, function (current) {
-                if (getNodeCheckFlags(current) & 8 /* CaptureNewTarget */) {
-                    var isDeclaration_2 = node.kind !== 75 /* Identifier */;
-                    if (isDeclaration_2) {
-                        error(ts.getNameOfDeclaration(node), ts.Diagnostics.Duplicate_identifier_newTarget_Compiler_uses_variable_declaration_newTarget_to_capture_new_target_meta_property_reference);
-                    }
-                    else {
-                        error(node, ts.Diagnostics.Expression_resolves_to_variable_declaration_newTarget_that_compiler_uses_to_capture_new_target_meta_property_reference);
-                    }
-                    return true;
-                }
-                return false;
-            });
+        function isConstEnumObjectType(type) {
+            return !!(ts.getObjectFlags(type) & 16 /* Anonymous */) && !!type.symbol && isConstEnumSymbol(type.symbol);
         }
-        function checkWeakMapCollision(node) {
-            var enclosingBlockScope = ts.getEnclosingBlockScopeContainer(node);
-            if (getNodeCheckFlags(enclosingBlockScope) & 67108864 /* ContainsClassWithPrivateIdentifiers */) {
-                error(node, ts.Diagnostics.Compiler_reserves_name_0_when_emitting_private_identifier_downlevel, "WeakMap");
-            }
+        function isConstEnumSymbol(symbol) {
+            return (symbol.flags & 128 /* ConstEnum */) !== 0;
         }
-        function checkCollisionWithRequireExportsInGeneratedCode(node, name) {
-            // No need to check for require or exports for ES6 modules and later
-            if (moduleKind >= ts.ModuleKind.ES2015 || compilerOptions.noEmit) {
-                return;
-            }
-            if (!needCollisionCheckForIdentifier(node, name, "require") && !needCollisionCheckForIdentifier(node, name, "exports")) {
-                return;
+        function checkInstanceOfExpression(left, right, leftType, rightType) {
+            if (leftType === silentNeverType || rightType === silentNeverType) {
+                return silentNeverType;
             }
-            // Uninstantiated modules shouldnt do this check
-            if (ts.isModuleDeclaration(node) && ts.getModuleInstanceState(node) !== 1 /* Instantiated */) {
-                return;
+            // TypeScript 1.0 spec (April 2014): 4.15.4
+            // The instanceof operator requires the left operand to be of type Any, an object type, or a type parameter type,
+            // and the right operand to be of type Any, a subtype of the 'Function' interface type, or have a call or construct signature.
+            // The result is always of the Boolean primitive type.
+            // NOTE: do not raise error if leftType is unknown as related error was already reported
+            if (!isTypeAny(leftType) &&
+                allTypesAssignableToKind(leftType, 131068 /* Primitive */)) {
+                error(left, ts.Diagnostics.The_left_hand_side_of_an_instanceof_expression_must_be_of_type_any_an_object_type_or_a_type_parameter);
             }
-            // In case of variable declaration, node.parent is variable statement so look at the variable statement's parent
-            var parent = getDeclarationContainer(node);
-            if (parent.kind === 290 /* SourceFile */ && ts.isExternalOrCommonJsModule(parent)) {
-                // If the declaration happens to be in external module, report error that require and exports are reserved keywords
-                error(name, ts.Diagnostics.Duplicate_identifier_0_Compiler_reserves_name_1_in_top_level_scope_of_a_module, ts.declarationNameToString(name), ts.declarationNameToString(name));
+            // NOTE: do not raise error if right is unknown as related error was already reported
+            if (!(isTypeAny(rightType) || typeHasCallOrConstructSignatures(rightType) || isTypeSubtypeOf(rightType, globalFunctionType))) {
+                error(right, ts.Diagnostics.The_right_hand_side_of_an_instanceof_expression_must_be_of_type_any_or_of_a_type_assignable_to_the_Function_interface_type);
             }
+            return booleanType;
         }
-        function checkCollisionWithGlobalPromiseInGeneratedCode(node, name) {
-            if (languageVersion >= 4 /* ES2017 */ || compilerOptions.noEmit || !needCollisionCheckForIdentifier(node, name, "Promise")) {
-                return;
+        function checkInExpression(left, right, leftType, rightType) {
+            if (leftType === silentNeverType || rightType === silentNeverType) {
+                return silentNeverType;
             }
-            // Uninstantiated modules shouldnt do this check
-            if (ts.isModuleDeclaration(node) && ts.getModuleInstanceState(node) !== 1 /* Instantiated */) {
-                return;
+            leftType = checkNonNullType(leftType, left);
+            rightType = checkNonNullType(rightType, right);
+            // TypeScript 1.0 spec (April 2014): 4.15.5
+            // The in operator requires the left operand to be of type Any, the String primitive type, or the Number primitive type,
+            // and the right operand to be of type Any, an object type, or a type parameter type.
+            // The result is always of the Boolean primitive type.
+            if (!(allTypesAssignableToKind(leftType, 402653316 /* StringLike */ | 296 /* NumberLike */ | 12288 /* ESSymbolLike */) ||
+                isTypeAssignableToKind(leftType, 4194304 /* Index */ | 134217728 /* TemplateLiteral */ | 268435456 /* StringMapping */ | 262144 /* TypeParameter */))) {
+                error(left, ts.Diagnostics.The_left_hand_side_of_an_in_expression_must_be_of_type_any_string_number_or_symbol);
             }
-            // In case of variable declaration, node.parent is variable statement so look at the variable statement's parent
-            var parent = getDeclarationContainer(node);
-            if (parent.kind === 290 /* SourceFile */ && ts.isExternalOrCommonJsModule(parent) && parent.flags & 2048 /* HasAsyncFunctions */) {
-                // If the declaration happens to be in external module, report error that Promise is a reserved identifier.
-                error(name, ts.Diagnostics.Duplicate_identifier_0_Compiler_reserves_name_1_in_top_level_scope_of_a_module_containing_async_functions, ts.declarationNameToString(name), ts.declarationNameToString(name));
+            if (!allTypesAssignableToKind(rightType, 67108864 /* NonPrimitive */ | 58982400 /* InstantiableNonPrimitive */)) {
+                error(right, ts.Diagnostics.The_right_hand_side_of_an_in_expression_must_be_of_type_any_an_object_type_or_a_type_parameter);
             }
+            return booleanType;
         }
-        function checkVarDeclaredNamesNotShadowed(node) {
-            // - ScriptBody : StatementList
-            // It is a Syntax Error if any element of the LexicallyDeclaredNames of StatementList
-            // also occurs in the VarDeclaredNames of StatementList.
-            // - Block : { StatementList }
-            // It is a Syntax Error if any element of the LexicallyDeclaredNames of StatementList
-            // also occurs in the VarDeclaredNames of StatementList.
-            // Variable declarations are hoisted to the top of their function scope. They can shadow
-            // block scoped declarations, which bind tighter. this will not be flagged as duplicate definition
-            // by the binder as the declaration scope is different.
-            // A non-initialized declaration is a no-op as the block declaration will resolve before the var
-            // declaration. the problem is if the declaration has an initializer. this will act as a write to the
-            // block declared value. this is fine for let, but not const.
-            // Only consider declarations with initializers, uninitialized const declarations will not
-            // step on a let/const variable.
-            // Do not consider const and const declarations, as duplicate block-scoped declarations
-            // are handled by the binder.
-            // We are only looking for const declarations that step on let\const declarations from a
-            // different scope. e.g.:
-            //      {
-            //          const x = 0; // localDeclarationSymbol obtained after name resolution will correspond to this declaration
-            //          const x = 0; // symbol for this declaration will be 'symbol'
-            //      }
-            // skip block-scoped variables and parameters
-            if ((ts.getCombinedNodeFlags(node) & 3 /* BlockScoped */) !== 0 || ts.isParameterDeclaration(node)) {
-                return;
+        function checkObjectLiteralAssignment(node, sourceType, rightIsThis) {
+            var properties = node.properties;
+            if (strictNullChecks && properties.length === 0) {
+                return checkNonNullType(sourceType, node);
             }
-            // skip variable declarations that don't have initializers
-            // NOTE: in ES6 spec initializer is required in variable declarations where name is binding pattern
-            // so we'll always treat binding elements as initialized
-            if (node.kind === 242 /* VariableDeclaration */ && !node.initializer) {
-                return;
+            for (var i = 0; i < properties.length; i++) {
+                checkObjectLiteralDestructuringPropertyAssignment(node, sourceType, i, properties, rightIsThis);
             }
-            var symbol = getSymbolOfNode(node);
-            if (symbol.flags & 1 /* FunctionScopedVariable */) {
-                if (!ts.isIdentifier(node.name))
-                    return ts.Debug.fail();
-                var localDeclarationSymbol = resolveName(node, node.name.escapedText, 3 /* Variable */, /*nodeNotFoundErrorMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
-                if (localDeclarationSymbol &&
-                    localDeclarationSymbol !== symbol &&
-                    localDeclarationSymbol.flags & 2 /* BlockScopedVariable */) {
-                    if (getDeclarationNodeFlagsFromSymbol(localDeclarationSymbol) & 3 /* BlockScoped */) {
-                        var varDeclList = ts.getAncestor(localDeclarationSymbol.valueDeclaration, 243 /* VariableDeclarationList */);
-                        var container = varDeclList.parent.kind === 225 /* VariableStatement */ && varDeclList.parent.parent
-                            ? varDeclList.parent.parent
-                            : undefined;
-                        // names of block-scoped and function scoped variables can collide only
-                        // if block scoped variable is defined in the function\module\source file scope (because of variable hoisting)
-                        var namesShareScope = container &&
-                            (container.kind === 223 /* Block */ && ts.isFunctionLike(container.parent) ||
-                                container.kind === 250 /* ModuleBlock */ ||
-                                container.kind === 249 /* ModuleDeclaration */ ||
-                                container.kind === 290 /* SourceFile */);
-                        // here we know that function scoped variable is shadowed by block scoped one
-                        // if they are defined in the same scope - binder has already reported redeclaration error
-                        // otherwise if variable has an initializer - show error that initialization will fail
-                        // since LHS will be block scoped name instead of function scoped
-                        if (!namesShareScope) {
-                            var name = symbolToString(localDeclarationSymbol);
-                            error(node, ts.Diagnostics.Cannot_initialize_outer_scoped_variable_0_in_the_same_scope_as_block_scoped_declaration_1, name, name);
+            return sourceType;
+        }
+        /** Note: If property cannot be a SpreadAssignment, then allProperties does not need to be provided */
+        function checkObjectLiteralDestructuringPropertyAssignment(node, objectLiteralType, propertyIndex, allProperties, rightIsThis) {
+            if (rightIsThis === void 0) { rightIsThis = false; }
+            var properties = node.properties;
+            var property = properties[propertyIndex];
+            if (property.kind === 288 /* PropertyAssignment */ || property.kind === 289 /* ShorthandPropertyAssignment */) {
+                var name = property.name;
+                var exprType = getLiteralTypeFromPropertyName(name);
+                if (isTypeUsableAsPropertyName(exprType)) {
+                    var text = getPropertyNameFromType(exprType);
+                    var prop = getPropertyOfType(objectLiteralType, text);
+                    if (prop) {
+                        markPropertyAsReferenced(prop, property, rightIsThis);
+                        checkPropertyAccessibility(property, /*isSuper*/ false, objectLiteralType, prop);
+                    }
+                }
+                var elementType = getIndexedAccessType(objectLiteralType, exprType, /*noUncheckedIndexedAccessCandidate*/ undefined, name, /*aliasSymbol*/ undefined, /*aliasTypeArguments*/ undefined, 16 /* ExpressionPosition */);
+                var type = getFlowTypeOfDestructuring(property, elementType);
+                return checkDestructuringAssignment(property.kind === 289 /* ShorthandPropertyAssignment */ ? property : property.initializer, type);
+            }
+            else if (property.kind === 290 /* SpreadAssignment */) {
+                if (propertyIndex < properties.length - 1) {
+                    error(property, ts.Diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern);
+                }
+                else {
+                    if (languageVersion < 99 /* ESNext */) {
+                        checkExternalEmitHelpers(property, 4 /* Rest */);
+                    }
+                    var nonRestNames = [];
+                    if (allProperties) {
+                        for (var _i = 0, allProperties_1 = allProperties; _i < allProperties_1.length; _i++) {
+                            var otherProperty = allProperties_1[_i];
+                            if (!ts.isSpreadAssignment(otherProperty)) {
+                                nonRestNames.push(otherProperty.name);
+                            }
                         }
                     }
+                    var type = getRestType(objectLiteralType, nonRestNames, objectLiteralType.symbol);
+                    checkGrammarForDisallowedTrailingComma(allProperties, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
+                    return checkDestructuringAssignment(property.expression, type);
                 }
             }
-        }
-        function convertAutoToAny(type) {
-            return type === autoType ? anyType : type === autoArrayType ? anyArrayType : type;
-        }
-        // Check variable, parameter, or property declaration
-        function checkVariableLikeDeclaration(node) {
-            checkDecorators(node);
-            if (!ts.isBindingElement(node)) {
-                checkSourceElement(node.type);
+            else {
+                error(property, ts.Diagnostics.Property_assignment_expected);
             }
-            // JSDoc `function(string, string): string` syntax results in parameters with no name
-            if (!node.name) {
-                return;
+        }
+        function checkArrayLiteralAssignment(node, sourceType, checkMode) {
+            var elements = node.elements;
+            if (languageVersion < 2 /* ES2015 */ && compilerOptions.downlevelIteration) {
+                checkExternalEmitHelpers(node, 512 /* Read */);
             }
-            // For a computed property, just check the initializer and exit
-            // Do not use hasDynamicName here, because that returns false for well known symbols.
-            // We want to perform checkComputedPropertyName for all computed properties, including
-            // well known symbols.
-            if (node.name.kind === 154 /* ComputedPropertyName */) {
-                checkComputedPropertyName(node.name);
-                if (node.initializer) {
-                    checkExpressionCached(node.initializer);
+            // This elementType will be used if the specific property corresponding to this index is not
+            // present (aka the tuple element property). This call also checks that the parentType is in
+            // fact an iterable or array (depending on target language).
+            var possiblyOutOfBoundsType = checkIteratedTypeOrElementType(65 /* Destructuring */ | 128 /* PossiblyOutOfBounds */, sourceType, undefinedType, node) || errorType;
+            var inBoundsType = compilerOptions.noUncheckedIndexedAccess ? undefined : possiblyOutOfBoundsType;
+            for (var i = 0; i < elements.length; i++) {
+                var type = possiblyOutOfBoundsType;
+                if (node.elements[i].kind === 220 /* SpreadElement */) {
+                    type = inBoundsType = inBoundsType !== null && inBoundsType !== void 0 ? inBoundsType : (checkIteratedTypeOrElementType(65 /* Destructuring */, sourceType, undefinedType, node) || errorType);
                 }
+                checkArrayLiteralDestructuringElementAssignment(node, sourceType, i, type, checkMode);
             }
-            if (node.kind === 191 /* BindingElement */) {
-                if (node.parent.kind === 189 /* ObjectBindingPattern */ && languageVersion < 99 /* ESNext */) {
-                    checkExternalEmitHelpers(node, 4 /* Rest */);
-                }
-                // check computed properties inside property names of binding elements
-                if (node.propertyName && node.propertyName.kind === 154 /* ComputedPropertyName */) {
-                    checkComputedPropertyName(node.propertyName);
-                }
-                // check private/protected variable access
-                var parent = node.parent.parent;
-                var parentType = getTypeForBindingElementParent(parent);
-                var name = node.propertyName || node.name;
-                if (parentType && !ts.isBindingPattern(name)) {
-                    var exprType = getLiteralTypeFromPropertyName(name);
-                    if (isTypeUsableAsPropertyName(exprType)) {
-                        var nameText = getPropertyNameFromType(exprType);
-                        var property = getPropertyOfType(parentType, nameText);
-                        if (property) {
-                            markPropertyAsReferenced(property, /*nodeForCheckWriteOnly*/ undefined, /*isThisAccess*/ false); // A destructuring is never a write-only reference.
-                            checkPropertyAccessibility(parent, !!parent.initializer && parent.initializer.kind === 102 /* SuperKeyword */, parentType, property);
-                        }
+            return sourceType;
+        }
+        function checkArrayLiteralDestructuringElementAssignment(node, sourceType, elementIndex, elementType, checkMode) {
+            var elements = node.elements;
+            var element = elements[elementIndex];
+            if (element.kind !== 222 /* OmittedExpression */) {
+                if (element.kind !== 220 /* SpreadElement */) {
+                    var indexType = getLiteralType(elementIndex);
+                    if (isArrayLikeType(sourceType)) {
+                        // We create a synthetic expression so that getIndexedAccessType doesn't get confused
+                        // when the element is a SyntaxKind.ElementAccessExpression.
+                        var accessFlags = 16 /* ExpressionPosition */ | (hasDefaultValue(element) ? 8 /* NoTupleBoundsCheck */ : 0);
+                        var elementType_2 = getIndexedAccessTypeOrUndefined(sourceType, indexType, /*noUncheckedIndexedAccessCandidate*/ undefined, createSyntheticExpression(element, indexType), accessFlags) || errorType;
+                        var assignedType = hasDefaultValue(element) ? getTypeWithFacts(elementType_2, 524288 /* NEUndefined */) : elementType_2;
+                        var type = getFlowTypeOfDestructuring(element, assignedType);
+                        return checkDestructuringAssignment(element, type, checkMode);
                     }
+                    return checkDestructuringAssignment(element, elementType, checkMode);
                 }
-            }
-            // For a binding pattern, check contained binding elements
-            if (ts.isBindingPattern(node.name)) {
-                if (node.name.kind === 190 /* ArrayBindingPattern */ && languageVersion < 2 /* ES2015 */ && compilerOptions.downlevelIteration) {
-                    checkExternalEmitHelpers(node, 512 /* Read */);
+                if (elementIndex < elements.length - 1) {
+                    error(element, ts.Diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern);
                 }
-                ts.forEach(node.name.elements, checkSourceElement);
-            }
-            // For a parameter declaration with an initializer, error and exit if the containing function doesn't have a body
-            if (node.initializer && ts.getRootDeclaration(node).kind === 156 /* Parameter */ && ts.nodeIsMissing(ts.getContainingFunction(node).body)) {
-                error(node, ts.Diagnostics.A_parameter_initializer_is_only_allowed_in_a_function_or_constructor_implementation);
-                return;
-            }
-            // For a binding pattern, validate the initializer and exit
-            if (ts.isBindingPattern(node.name)) {
-                var needCheckInitializer = node.initializer && node.parent.parent.kind !== 231 /* ForInStatement */;
-                var needCheckWidenedType = node.name.elements.length === 0;
-                if (needCheckInitializer || needCheckWidenedType) {
-                    // Don't validate for-in initializer as it is already an error
-                    var widenedType = getWidenedTypeForVariableLikeDeclaration(node);
-                    if (needCheckInitializer) {
-                        var initializerType = checkExpressionCached(node.initializer);
-                        if (strictNullChecks && needCheckWidenedType) {
-                            checkNonNullNonVoidType(initializerType, node);
-                        }
-                        else {
-                            checkTypeAssignableToAndOptionallyElaborate(initializerType, getWidenedTypeForVariableLikeDeclaration(node), node, node.initializer);
-                        }
+                else {
+                    var restExpression = element.expression;
+                    if (restExpression.kind === 216 /* BinaryExpression */ && restExpression.operatorToken.kind === 62 /* EqualsToken */) {
+                        error(restExpression.operatorToken, ts.Diagnostics.A_rest_element_cannot_have_an_initializer);
                     }
-                    // check the binding pattern with empty elements
-                    if (needCheckWidenedType) {
-                        if (ts.isArrayBindingPattern(node.name)) {
-                            checkIteratedTypeOrElementType(65 /* Destructuring */, widenedType, undefinedType, node);
-                        }
-                        else if (strictNullChecks) {
-                            checkNonNullNonVoidType(widenedType, node);
-                        }
+                    else {
+                        checkGrammarForDisallowedTrailingComma(node.elements, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
+                        var type = everyType(sourceType, isTupleType) ?
+                            mapType(sourceType, function (t) { return sliceTupleType(t, elementIndex); }) :
+                            createArrayType(elementType);
+                        return checkDestructuringAssignment(restExpression, type, checkMode);
                     }
                 }
-                return;
             }
-            var symbol = getSymbolOfNode(node);
-            var type = convertAutoToAny(getTypeOfSymbol(symbol));
-            if (node === symbol.valueDeclaration) {
-                // Node is the primary declaration of the symbol, just validate the initializer
-                // Don't validate for-in initializer as it is already an error
-                var initializer = ts.getEffectiveInitializer(node);
-                if (initializer) {
-                    var isJSObjectLiteralInitializer = ts.isInJSFile(node) &&
-                        ts.isObjectLiteralExpression(initializer) &&
-                        (initializer.properties.length === 0 || ts.isPrototypeAccess(node.name)) &&
-                        ts.hasEntries(symbol.exports);
-                    if (!isJSObjectLiteralInitializer && node.parent.parent.kind !== 231 /* ForInStatement */) {
-                        checkTypeAssignableToAndOptionallyElaborate(checkExpressionCached(initializer), type, node, initializer, /*headMessage*/ undefined);
-                    }
-                }
-                if (symbol.declarations.length > 1) {
-                    if (ts.some(symbol.declarations, function (d) { return d !== node && ts.isVariableLike(d) && !areDeclarationFlagsIdentical(d, node); })) {
-                        error(node.name, ts.Diagnostics.All_declarations_of_0_must_have_identical_modifiers, ts.declarationNameToString(node.name));
+            return undefined;
+        }
+        function checkDestructuringAssignment(exprOrAssignment, sourceType, checkMode, rightIsThis) {
+            var target;
+            if (exprOrAssignment.kind === 289 /* ShorthandPropertyAssignment */) {
+                var prop = exprOrAssignment;
+                if (prop.objectAssignmentInitializer) {
+                    // In strict null checking mode, if a default value of a non-undefined type is specified, remove
+                    // undefined from the final type.
+                    if (strictNullChecks &&
+                        !(getFalsyFlags(checkExpression(prop.objectAssignmentInitializer)) & 32768 /* Undefined */)) {
+                        sourceType = getTypeWithFacts(sourceType, 524288 /* NEUndefined */);
                     }
+                    checkBinaryLikeExpression(prop.name, prop.equalsToken, prop.objectAssignmentInitializer, checkMode);
                 }
+                target = exprOrAssignment.name;
             }
             else {
-                // Node is a secondary declaration, check that type is identical to primary declaration and check that
-                // initializer is consistent with type associated with the node
-                var declarationType = convertAutoToAny(getWidenedTypeForVariableLikeDeclaration(node));
-                if (type !== errorType && declarationType !== errorType &&
-                    !isTypeIdenticalTo(type, declarationType) &&
-                    !(symbol.flags & 67108864 /* Assignment */)) {
-                    errorNextVariableOrPropertyDeclarationMustHaveSameType(symbol.valueDeclaration, type, node, declarationType);
-                }
-                if (node.initializer) {
-                    checkTypeAssignableToAndOptionallyElaborate(checkExpressionCached(node.initializer), declarationType, node, node.initializer, /*headMessage*/ undefined);
-                }
-                if (!areDeclarationFlagsIdentical(node, symbol.valueDeclaration)) {
-                    error(node.name, ts.Diagnostics.All_declarations_of_0_must_have_identical_modifiers, ts.declarationNameToString(node.name));
-                }
+                target = exprOrAssignment;
             }
-            if (node.kind !== 159 /* PropertyDeclaration */ && node.kind !== 158 /* PropertySignature */) {
-                // We know we don't have a binding pattern or computed name here
-                checkExportsOnMergedDeclarations(node);
-                if (node.kind === 242 /* VariableDeclaration */ || node.kind === 191 /* BindingElement */) {
-                    checkVarDeclaredNamesNotShadowed(node);
-                }
-                checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
-                checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
-                if (!compilerOptions.noEmit && languageVersion < 99 /* ESNext */ && needCollisionCheckForIdentifier(node, node.name, "WeakMap")) {
-                    potentialWeakMapCollisions.push(node);
-                }
+            if (target.kind === 216 /* BinaryExpression */ && target.operatorToken.kind === 62 /* EqualsToken */) {
+                checkBinaryExpression(target, checkMode);
+                target = target.left;
             }
-        }
-        function errorNextVariableOrPropertyDeclarationMustHaveSameType(firstDeclaration, firstType, nextDeclaration, nextType) {
-            var nextDeclarationName = ts.getNameOfDeclaration(nextDeclaration);
-            var message = nextDeclaration.kind === 159 /* PropertyDeclaration */ || nextDeclaration.kind === 158 /* PropertySignature */
-                ? ts.Diagnostics.Subsequent_property_declarations_must_have_the_same_type_Property_0_must_be_of_type_1_but_here_has_type_2
-                : ts.Diagnostics.Subsequent_variable_declarations_must_have_the_same_type_Variable_0_must_be_of_type_1_but_here_has_type_2;
-            var declName = ts.declarationNameToString(nextDeclarationName);
-            var err = error(nextDeclarationName, message, declName, typeToString(firstType), typeToString(nextType));
-            if (firstDeclaration) {
-                ts.addRelatedInfo(err, ts.createDiagnosticForNode(firstDeclaration, ts.Diagnostics._0_was_also_declared_here, declName));
+            if (target.kind === 200 /* ObjectLiteralExpression */) {
+                return checkObjectLiteralAssignment(target, sourceType, rightIsThis);
             }
+            if (target.kind === 199 /* ArrayLiteralExpression */) {
+                return checkArrayLiteralAssignment(target, sourceType, checkMode);
+            }
+            return checkReferenceAssignment(target, sourceType, checkMode);
         }
-        function areDeclarationFlagsIdentical(left, right) {
-            if ((left.kind === 156 /* Parameter */ && right.kind === 242 /* VariableDeclaration */) ||
-                (left.kind === 242 /* VariableDeclaration */ && right.kind === 156 /* Parameter */)) {
-                // Differences in optionality between parameters and variables are allowed.
-                return true;
+        function checkReferenceAssignment(target, sourceType, checkMode) {
+            var targetType = checkExpression(target, checkMode);
+            var error = target.parent.kind === 290 /* SpreadAssignment */ ?
+                ts.Diagnostics.The_target_of_an_object_rest_assignment_must_be_a_variable_or_a_property_access :
+                ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_must_be_a_variable_or_a_property_access;
+            var optionalError = target.parent.kind === 290 /* SpreadAssignment */ ?
+                ts.Diagnostics.The_target_of_an_object_rest_assignment_may_not_be_an_optional_property_access :
+                ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_may_not_be_an_optional_property_access;
+            if (checkReferenceExpression(target, error, optionalError)) {
+                checkTypeAssignableToAndOptionallyElaborate(sourceType, targetType, target, target);
             }
-            if (ts.hasQuestionToken(left) !== ts.hasQuestionToken(right)) {
-                return false;
+            if (ts.isPrivateIdentifierPropertyAccessExpression(target)) {
+                checkExternalEmitHelpers(target.parent, 2097152 /* ClassPrivateFieldSet */);
             }
-            var interestingFlags = 8 /* Private */ |
-                16 /* Protected */ |
-                256 /* Async */ |
-                128 /* Abstract */ |
-                64 /* Readonly */ |
-                32 /* Static */;
-            return ts.getSelectedModifierFlags(left, interestingFlags) === ts.getSelectedModifierFlags(right, interestingFlags);
-        }
-        function checkVariableDeclaration(node) {
-            checkGrammarVariableDeclaration(node);
-            return checkVariableLikeDeclaration(node);
-        }
-        function checkBindingElement(node) {
-            checkGrammarBindingElement(node);
-            return checkVariableLikeDeclaration(node);
-        }
-        function checkVariableStatement(node) {
-            // Grammar checking
-            if (!checkGrammarDecoratorsAndModifiers(node) && !checkGrammarVariableDeclarationList(node.declarationList))
-                checkGrammarForDisallowedLetOrConstStatement(node);
-            ts.forEach(node.declarationList.declarations, checkSourceElement);
-        }
-        function checkExpressionStatement(node) {
-            // Grammar checking
-            checkGrammarStatementInAmbientContext(node);
-            checkExpression(node.expression);
+            return sourceType;
         }
-        function checkIfStatement(node) {
-            // Grammar checking
-            checkGrammarStatementInAmbientContext(node);
-            var type = checkTruthinessExpression(node.expression);
-            checkTestingKnownTruthyCallableType(node.expression, node.thenStatement, type);
-            checkSourceElement(node.thenStatement);
-            if (node.thenStatement.kind === 224 /* EmptyStatement */) {
-                error(node.thenStatement, ts.Diagnostics.The_body_of_an_if_statement_cannot_be_the_empty_statement);
+        /**
+         * This is a *shallow* check: An expression is side-effect-free if the
+         * evaluation of the expression *itself* cannot produce side effects.
+         * For example, x++ / 3 is side-effect free because the / operator
+         * does not have side effects.
+         * The intent is to "smell test" an expression for correctness in positions where
+         * its value is discarded (e.g. the left side of the comma operator).
+         */
+        function isSideEffectFree(node) {
+            node = ts.skipParentheses(node);
+            switch (node.kind) {
+                case 78 /* Identifier */:
+                case 10 /* StringLiteral */:
+                case 13 /* RegularExpressionLiteral */:
+                case 205 /* TaggedTemplateExpression */:
+                case 218 /* TemplateExpression */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 8 /* NumericLiteral */:
+                case 9 /* BigIntLiteral */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
+                case 103 /* NullKeyword */:
+                case 150 /* UndefinedKeyword */:
+                case 208 /* FunctionExpression */:
+                case 221 /* ClassExpression */:
+                case 209 /* ArrowFunction */:
+                case 199 /* ArrayLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
+                case 211 /* TypeOfExpression */:
+                case 225 /* NonNullExpression */:
+                case 274 /* JsxSelfClosingElement */:
+                case 273 /* JsxElement */:
+                    return true;
+                case 217 /* ConditionalExpression */:
+                    return isSideEffectFree(node.whenTrue) &&
+                        isSideEffectFree(node.whenFalse);
+                case 216 /* BinaryExpression */:
+                    if (ts.isAssignmentOperator(node.operatorToken.kind)) {
+                        return false;
+                    }
+                    return isSideEffectFree(node.left) &&
+                        isSideEffectFree(node.right);
+                case 214 /* PrefixUnaryExpression */:
+                case 215 /* PostfixUnaryExpression */:
+                    // Unary operators ~, !, +, and - have no side effects.
+                    // The rest do.
+                    switch (node.operator) {
+                        case 53 /* ExclamationToken */:
+                        case 39 /* PlusToken */:
+                        case 40 /* MinusToken */:
+                        case 54 /* TildeToken */:
+                            return true;
+                    }
+                    return false;
+                // Some forms listed here for clarity
+                case 212 /* VoidExpression */: // Explicit opt-out
+                case 206 /* TypeAssertionExpression */: // Not SEF, but can produce useful type warnings
+                case 224 /* AsExpression */: // Not SEF, but can produce useful type warnings
+                default:
+                    return false;
             }
-            checkSourceElement(node.elseStatement);
         }
-        function checkTestingKnownTruthyCallableType(condExpr, body, type) {
-            if (!strictNullChecks) {
-                return;
-            }
-            var testedNode = ts.isIdentifier(condExpr)
-                ? condExpr
-                : ts.isPropertyAccessExpression(condExpr)
-                    ? condExpr.name
-                    : undefined;
-            if (!testedNode) {
-                return;
-            }
-            var possiblyFalsy = getFalsyFlags(type);
-            if (possiblyFalsy) {
-                return;
-            }
-            // While it technically should be invalid for any known-truthy value
-            // to be tested, we de-scope to functions unrefenced in the block as a
-            // heuristic to identify the most common bugs. There are too many
-            // false positives for values sourced from type definitions without
-            // strictNullChecks otherwise.
-            var callSignatures = getSignaturesOfType(type, 0 /* Call */);
-            if (callSignatures.length === 0) {
-                return;
-            }
-            var testedFunctionSymbol = getSymbolAtLocation(testedNode);
-            if (!testedFunctionSymbol) {
-                return;
-            }
-            var functionIsUsedInBody = ts.forEachChild(body, function check(childNode) {
-                if (ts.isIdentifier(childNode)) {
-                    var childSymbol = getSymbolAtLocation(childNode);
-                    if (childSymbol && childSymbol === testedFunctionSymbol) {
-                        // If the test was a simple identifier, the above check is sufficient
-                        if (ts.isIdentifier(condExpr)) {
-                            return true;
+        function isTypeEqualityComparableTo(source, target) {
+            return (target.flags & 98304 /* Nullable */) !== 0 || isTypeComparableTo(source, target);
+        }
+        var CheckBinaryExpressionState;
+        (function (CheckBinaryExpressionState) {
+            CheckBinaryExpressionState[CheckBinaryExpressionState["MaybeCheckLeft"] = 0] = "MaybeCheckLeft";
+            CheckBinaryExpressionState[CheckBinaryExpressionState["CheckRight"] = 1] = "CheckRight";
+            CheckBinaryExpressionState[CheckBinaryExpressionState["FinishCheck"] = 2] = "FinishCheck";
+        })(CheckBinaryExpressionState || (CheckBinaryExpressionState = {}));
+        function checkBinaryExpression(node, checkMode) {
+            var workStacks = {
+                expr: [node],
+                state: [0 /* MaybeCheckLeft */],
+                leftType: [undefined]
+            };
+            var stackIndex = 0;
+            var lastResult;
+            while (stackIndex >= 0) {
+                node = workStacks.expr[stackIndex];
+                switch (workStacks.state[stackIndex]) {
+                    case 0 /* MaybeCheckLeft */: {
+                        if (ts.isInJSFile(node) && ts.getAssignedExpandoInitializer(node)) {
+                            finishInvocation(checkExpression(node.right, checkMode));
+                            break;
                         }
-                        // Otherwise we need to ensure the symbol is called on the same target
-                        var testedExpression = testedNode.parent;
-                        var childExpression = childNode.parent;
-                        while (testedExpression && childExpression) {
-                            if (ts.isIdentifier(testedExpression) && ts.isIdentifier(childExpression) ||
-                                testedExpression.kind === 104 /* ThisKeyword */ && childExpression.kind === 104 /* ThisKeyword */) {
-                                return getSymbolAtLocation(testedExpression) === getSymbolAtLocation(childExpression);
-                            }
-                            if (ts.isPropertyAccessExpression(testedExpression) && ts.isPropertyAccessExpression(childExpression)) {
-                                if (getSymbolAtLocation(testedExpression.name) !== getSymbolAtLocation(childExpression.name)) {
-                                    return false;
-                                }
-                                childExpression = childExpression.expression;
-                                testedExpression = testedExpression.expression;
-                            }
-                            else {
-                                return false;
-                            }
+                        checkGrammarNullishCoalesceWithLogicalExpression(node);
+                        var operator = node.operatorToken.kind;
+                        if (operator === 62 /* EqualsToken */ && (node.left.kind === 200 /* ObjectLiteralExpression */ || node.left.kind === 199 /* ArrayLiteralExpression */)) {
+                            finishInvocation(checkDestructuringAssignment(node.left, checkExpression(node.right, checkMode), checkMode, node.right.kind === 107 /* ThisKeyword */));
+                            break;
+                        }
+                        advanceState(1 /* CheckRight */);
+                        maybeCheckExpression(node.left);
+                        break;
+                    }
+                    case 1 /* CheckRight */: {
+                        var leftType = lastResult;
+                        workStacks.leftType[stackIndex] = leftType;
+                        var operator = node.operatorToken.kind;
+                        if (operator === 55 /* AmpersandAmpersandToken */ || operator === 56 /* BarBarToken */ || operator === 60 /* QuestionQuestionToken */) {
+                            checkTruthinessOfType(leftType, node.left);
                         }
+                        advanceState(2 /* FinishCheck */);
+                        maybeCheckExpression(node.right);
+                        break;
+                    }
+                    case 2 /* FinishCheck */: {
+                        var leftType = workStacks.leftType[stackIndex];
+                        var rightType = lastResult;
+                        finishInvocation(checkBinaryLikeExpressionWorker(node.left, node.operatorToken, node.right, leftType, rightType, node));
+                        break;
                     }
+                    default: return ts.Debug.fail("Invalid state " + workStacks.state[stackIndex] + " for checkBinaryExpression");
                 }
-                return ts.forEachChild(childNode, check);
-            });
-            if (!functionIsUsedInBody) {
-                error(condExpr, ts.Diagnostics.This_condition_will_always_return_true_since_the_function_is_always_defined_Did_you_mean_to_call_it_instead);
             }
-        }
-        function checkDoStatement(node) {
-            // Grammar checking
-            checkGrammarStatementInAmbientContext(node);
-            checkSourceElement(node.statement);
-            checkTruthinessExpression(node.expression);
-        }
-        function checkWhileStatement(node) {
-            // Grammar checking
-            checkGrammarStatementInAmbientContext(node);
-            checkTruthinessExpression(node.expression);
-            checkSourceElement(node.statement);
-        }
-        function checkTruthinessOfType(type, node) {
-            if (type.flags & 16384 /* Void */) {
-                error(node, ts.Diagnostics.An_expression_of_type_void_cannot_be_tested_for_truthiness);
+            return lastResult;
+            function finishInvocation(result) {
+                lastResult = result;
+                stackIndex--;
             }
-            return type;
-        }
-        function checkTruthinessExpression(node, checkMode) {
-            return checkTruthinessOfType(checkExpression(node, checkMode), node);
-        }
-        function checkForStatement(node) {
-            // Grammar checking
-            if (!checkGrammarStatementInAmbientContext(node)) {
-                if (node.initializer && node.initializer.kind === 243 /* VariableDeclarationList */) {
-                    checkGrammarVariableDeclarationList(node.initializer);
-                }
+            /**
+             * Note that `advanceState` sets the _current_ head state, and that `maybeCheckExpression` potentially pushes on a new
+             * head state; so `advanceState` must be called before any `maybeCheckExpression` during a state's execution.
+             */
+            function advanceState(nextState) {
+                workStacks.state[stackIndex] = nextState;
             }
-            if (node.initializer) {
-                if (node.initializer.kind === 243 /* VariableDeclarationList */) {
-                    ts.forEach(node.initializer.declarations, checkVariableDeclaration);
+            function maybeCheckExpression(node) {
+                if (ts.isBinaryExpression(node)) {
+                    stackIndex++;
+                    workStacks.expr[stackIndex] = node;
+                    workStacks.state[stackIndex] = 0 /* MaybeCheckLeft */;
+                    workStacks.leftType[stackIndex] = undefined;
                 }
                 else {
-                    checkExpression(node.initializer);
+                    lastResult = checkExpression(node, checkMode);
                 }
             }
-            if (node.condition)
-                checkTruthinessExpression(node.condition);
-            if (node.incrementor)
-                checkExpression(node.incrementor);
-            checkSourceElement(node.statement);
-            if (node.locals) {
-                registerForUnusedIdentifiersCheck(node);
-            }
         }
-        function checkForOfStatement(node) {
-            checkGrammarForInOrForOfStatement(node);
-            if (node.awaitModifier) {
-                var functionFlags = ts.getFunctionFlags(ts.getContainingFunction(node));
-                if ((functionFlags & (4 /* Invalid */ | 2 /* Async */)) === 2 /* Async */ && languageVersion < 99 /* ESNext */) {
-                    // for..await..of in an async function or async generator function prior to ESNext requires the __asyncValues helper
-                    checkExternalEmitHelpers(node, 32768 /* ForAwaitOfIncludes */);
+        function checkGrammarNullishCoalesceWithLogicalExpression(node) {
+            var left = node.left, operatorToken = node.operatorToken, right = node.right;
+            if (operatorToken.kind === 60 /* QuestionQuestionToken */) {
+                if (ts.isBinaryExpression(left) && (left.operatorToken.kind === 56 /* BarBarToken */ || left.operatorToken.kind === 55 /* AmpersandAmpersandToken */)) {
+                    grammarErrorOnNode(left, ts.Diagnostics._0_and_1_operations_cannot_be_mixed_without_parentheses, ts.tokenToString(left.operatorToken.kind), ts.tokenToString(operatorToken.kind));
+                }
+                if (ts.isBinaryExpression(right) && (right.operatorToken.kind === 56 /* BarBarToken */ || right.operatorToken.kind === 55 /* AmpersandAmpersandToken */)) {
+                    grammarErrorOnNode(right, ts.Diagnostics._0_and_1_operations_cannot_be_mixed_without_parentheses, ts.tokenToString(right.operatorToken.kind), ts.tokenToString(operatorToken.kind));
                 }
             }
-            else if (compilerOptions.downlevelIteration && languageVersion < 2 /* ES2015 */) {
-                // for..of prior to ES2015 requires the __values helper when downlevelIteration is enabled
-                checkExternalEmitHelpers(node, 256 /* ForOfIncludes */);
+        }
+        // Note that this and `checkBinaryExpression` above should behave mostly the same, except this elides some
+        // expression-wide checks and does not use a work stack to fold nested binary expressions into the same callstack frame
+        function checkBinaryLikeExpression(left, operatorToken, right, checkMode, errorNode) {
+            var operator = operatorToken.kind;
+            if (operator === 62 /* EqualsToken */ && (left.kind === 200 /* ObjectLiteralExpression */ || left.kind === 199 /* ArrayLiteralExpression */)) {
+                return checkDestructuringAssignment(left, checkExpression(right, checkMode), checkMode, right.kind === 107 /* ThisKeyword */);
             }
-            // Check the LHS and RHS
-            // If the LHS is a declaration, just check it as a variable declaration, which will in turn check the RHS
-            // via checkRightHandSideOfForOf.
-            // If the LHS is an expression, check the LHS, as a destructuring assignment or as a reference.
-            // Then check that the RHS is assignable to it.
-            if (node.initializer.kind === 243 /* VariableDeclarationList */) {
-                checkForInOrForOfVariableDeclaration(node);
+            var leftType;
+            if (operator === 55 /* AmpersandAmpersandToken */ || operator === 56 /* BarBarToken */ || operator === 60 /* QuestionQuestionToken */) {
+                leftType = checkTruthinessExpression(left, checkMode);
             }
             else {
-                var varExpr = node.initializer;
-                var iteratedType = checkRightHandSideOfForOf(node);
-                // There may be a destructuring assignment on the left side
-                if (varExpr.kind === 192 /* ArrayLiteralExpression */ || varExpr.kind === 193 /* ObjectLiteralExpression */) {
-                    // iteratedType may be undefined. In this case, we still want to check the structure of
-                    // varExpr, in particular making sure it's a valid LeftHandSideExpression. But we'd like
-                    // to short circuit the type relation checking as much as possible, so we pass the unknownType.
-                    checkDestructuringAssignment(varExpr, iteratedType || errorType);
-                }
-                else {
-                    var leftType = checkExpression(varExpr);
-                    checkReferenceExpression(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_of_statement_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_left_hand_side_of_a_for_of_statement_may_not_be_an_optional_property_access);
-                    // iteratedType will be undefined if the rightType was missing properties/signatures
-                    // required to get its iteratedType (like [Symbol.iterator] or next). This may be
-                    // because we accessed properties from anyType, or it may have led to an error inside
-                    // getElementTypeOfIterable.
-                    if (iteratedType) {
-                        checkTypeAssignableToAndOptionallyElaborate(iteratedType, leftType, varExpr, node.expression);
-                    }
-                }
-            }
-            checkSourceElement(node.statement);
-            if (node.locals) {
-                registerForUnusedIdentifiersCheck(node);
+                leftType = checkExpression(left, checkMode);
             }
+            var rightType = checkExpression(right, checkMode);
+            return checkBinaryLikeExpressionWorker(left, operatorToken, right, leftType, rightType, errorNode);
         }
-        function checkForInStatement(node) {
-            // Grammar checking
-            checkGrammarForInOrForOfStatement(node);
-            var rightType = getNonNullableTypeIfNeeded(checkExpression(node.expression));
-            // TypeScript 1.0 spec (April 2014): 5.4
-            // In a 'for-in' statement of the form
-            // for (let VarDecl in Expr) Statement
-            //   VarDecl must be a variable declaration without a type annotation that declares a variable of type Any,
-            //   and Expr must be an expression of type Any, an object type, or a type parameter type.
-            if (node.initializer.kind === 243 /* VariableDeclarationList */) {
-                var variable = node.initializer.declarations[0];
-                if (variable && ts.isBindingPattern(variable.name)) {
-                    error(variable.name, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_cannot_be_a_destructuring_pattern);
-                }
-                checkForInOrForOfVariableDeclaration(node);
-            }
-            else {
-                // In a 'for-in' statement of the form
-                // for (Var in Expr) Statement
-                //   Var must be an expression classified as a reference of type Any or the String primitive type,
-                //   and Expr must be an expression of type Any, an object type, or a type parameter type.
-                var varExpr = node.initializer;
-                var leftType = checkExpression(varExpr);
-                if (varExpr.kind === 192 /* ArrayLiteralExpression */ || varExpr.kind === 193 /* ObjectLiteralExpression */) {
-                    error(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_cannot_be_a_destructuring_pattern);
+        function checkBinaryLikeExpressionWorker(left, operatorToken, right, leftType, rightType, errorNode) {
+            var operator = operatorToken.kind;
+            switch (operator) {
+                case 41 /* AsteriskToken */:
+                case 42 /* AsteriskAsteriskToken */:
+                case 65 /* AsteriskEqualsToken */:
+                case 66 /* AsteriskAsteriskEqualsToken */:
+                case 43 /* SlashToken */:
+                case 67 /* SlashEqualsToken */:
+                case 44 /* PercentToken */:
+                case 68 /* PercentEqualsToken */:
+                case 40 /* MinusToken */:
+                case 64 /* MinusEqualsToken */:
+                case 47 /* LessThanLessThanToken */:
+                case 69 /* LessThanLessThanEqualsToken */:
+                case 48 /* GreaterThanGreaterThanToken */:
+                case 70 /* GreaterThanGreaterThanEqualsToken */:
+                case 49 /* GreaterThanGreaterThanGreaterThanToken */:
+                case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
+                case 51 /* BarToken */:
+                case 73 /* BarEqualsToken */:
+                case 52 /* CaretToken */:
+                case 77 /* CaretEqualsToken */:
+                case 50 /* AmpersandToken */:
+                case 72 /* AmpersandEqualsToken */:
+                    if (leftType === silentNeverType || rightType === silentNeverType) {
+                        return silentNeverType;
+                    }
+                    leftType = checkNonNullType(leftType, left);
+                    rightType = checkNonNullType(rightType, right);
+                    var suggestedOperator = void 0;
+                    // if a user tries to apply a bitwise operator to 2 boolean operands
+                    // try and return them a helpful suggestion
+                    if ((leftType.flags & 528 /* BooleanLike */) &&
+                        (rightType.flags & 528 /* BooleanLike */) &&
+                        (suggestedOperator = getSuggestedBooleanOperator(operatorToken.kind)) !== undefined) {
+                        error(errorNode || operatorToken, ts.Diagnostics.The_0_operator_is_not_allowed_for_boolean_types_Consider_using_1_instead, ts.tokenToString(operatorToken.kind), ts.tokenToString(suggestedOperator));
+                        return numberType;
+                    }
+                    else {
+                        // otherwise just check each operand separately and report errors as normal
+                        var leftOk = checkArithmeticOperandType(left, leftType, ts.Diagnostics.The_left_hand_side_of_an_arithmetic_operation_must_be_of_type_any_number_bigint_or_an_enum_type, /*isAwaitValid*/ true);
+                        var rightOk = checkArithmeticOperandType(right, rightType, ts.Diagnostics.The_right_hand_side_of_an_arithmetic_operation_must_be_of_type_any_number_bigint_or_an_enum_type, /*isAwaitValid*/ true);
+                        var resultType_1;
+                        // If both are any or unknown, allow operation; assume it will resolve to number
+                        if ((isTypeAssignableToKind(leftType, 3 /* AnyOrUnknown */) && isTypeAssignableToKind(rightType, 3 /* AnyOrUnknown */)) ||
+                            // Or, if neither could be bigint, implicit coercion results in a number result
+                            !(maybeTypeOfKind(leftType, 2112 /* BigIntLike */) || maybeTypeOfKind(rightType, 2112 /* BigIntLike */))) {
+                            resultType_1 = numberType;
+                        }
+                        // At least one is assignable to bigint, so check that both are
+                        else if (bothAreBigIntLike(leftType, rightType)) {
+                            switch (operator) {
+                                case 49 /* GreaterThanGreaterThanGreaterThanToken */:
+                                case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
+                                    reportOperatorError();
+                                    break;
+                                case 42 /* AsteriskAsteriskToken */:
+                                case 66 /* AsteriskAsteriskEqualsToken */:
+                                    if (languageVersion < 3 /* ES2016 */) {
+                                        error(errorNode, ts.Diagnostics.Exponentiation_cannot_be_performed_on_bigint_values_unless_the_target_option_is_set_to_es2016_or_later);
+                                    }
+                            }
+                            resultType_1 = bigintType;
+                        }
+                        // Exactly one of leftType/rightType is assignable to bigint
+                        else {
+                            reportOperatorError(bothAreBigIntLike);
+                            resultType_1 = errorType;
+                        }
+                        if (leftOk && rightOk) {
+                            checkAssignmentOperator(resultType_1);
+                        }
+                        return resultType_1;
+                    }
+                case 39 /* PlusToken */:
+                case 63 /* PlusEqualsToken */:
+                    if (leftType === silentNeverType || rightType === silentNeverType) {
+                        return silentNeverType;
+                    }
+                    if (!isTypeAssignableToKind(leftType, 402653316 /* StringLike */) && !isTypeAssignableToKind(rightType, 402653316 /* StringLike */)) {
+                        leftType = checkNonNullType(leftType, left);
+                        rightType = checkNonNullType(rightType, right);
+                    }
+                    var resultType = void 0;
+                    if (isTypeAssignableToKind(leftType, 296 /* NumberLike */, /*strict*/ true) && isTypeAssignableToKind(rightType, 296 /* NumberLike */, /*strict*/ true)) {
+                        // Operands of an enum type are treated as having the primitive type Number.
+                        // If both operands are of the Number primitive type, the result is of the Number primitive type.
+                        resultType = numberType;
+                    }
+                    else if (isTypeAssignableToKind(leftType, 2112 /* BigIntLike */, /*strict*/ true) && isTypeAssignableToKind(rightType, 2112 /* BigIntLike */, /*strict*/ true)) {
+                        // If both operands are of the BigInt primitive type, the result is of the BigInt primitive type.
+                        resultType = bigintType;
+                    }
+                    else if (isTypeAssignableToKind(leftType, 402653316 /* StringLike */, /*strict*/ true) || isTypeAssignableToKind(rightType, 402653316 /* StringLike */, /*strict*/ true)) {
+                        // If one or both operands are of the String primitive type, the result is of the String primitive type.
+                        resultType = stringType;
+                    }
+                    else if (isTypeAny(leftType) || isTypeAny(rightType)) {
+                        // Otherwise, the result is of type Any.
+                        // NOTE: unknown type here denotes error type. Old compiler treated this case as any type so do we.
+                        resultType = leftType === errorType || rightType === errorType ? errorType : anyType;
+                    }
+                    // Symbols are not allowed at all in arithmetic expressions
+                    if (resultType && !checkForDisallowedESSymbolOperand(operator)) {
+                        return resultType;
+                    }
+                    if (!resultType) {
+                        // Types that have a reasonably good chance of being a valid operand type.
+                        // If both types have an awaited type of one of these, we'll assume the user
+                        // might be missing an await without doing an exhaustive check that inserting
+                        // await(s) will actually be a completely valid binary expression.
+                        var closeEnoughKind_1 = 296 /* NumberLike */ | 2112 /* BigIntLike */ | 402653316 /* StringLike */ | 3 /* AnyOrUnknown */;
+                        reportOperatorError(function (left, right) {
+                            return isTypeAssignableToKind(left, closeEnoughKind_1) &&
+                                isTypeAssignableToKind(right, closeEnoughKind_1);
+                        });
+                        return anyType;
+                    }
+                    if (operator === 63 /* PlusEqualsToken */) {
+                        checkAssignmentOperator(resultType);
+                    }
+                    return resultType;
+                case 29 /* LessThanToken */:
+                case 31 /* GreaterThanToken */:
+                case 32 /* LessThanEqualsToken */:
+                case 33 /* GreaterThanEqualsToken */:
+                    if (checkForDisallowedESSymbolOperand(operator)) {
+                        leftType = getBaseTypeOfLiteralType(checkNonNullType(leftType, left));
+                        rightType = getBaseTypeOfLiteralType(checkNonNullType(rightType, right));
+                        reportOperatorErrorUnless(function (left, right) {
+                            return isTypeComparableTo(left, right) || isTypeComparableTo(right, left) || (isTypeAssignableTo(left, numberOrBigIntType) && isTypeAssignableTo(right, numberOrBigIntType));
+                        });
+                    }
+                    return booleanType;
+                case 34 /* EqualsEqualsToken */:
+                case 35 /* ExclamationEqualsToken */:
+                case 36 /* EqualsEqualsEqualsToken */:
+                case 37 /* ExclamationEqualsEqualsToken */:
+                    reportOperatorErrorUnless(function (left, right) { return isTypeEqualityComparableTo(left, right) || isTypeEqualityComparableTo(right, left); });
+                    return booleanType;
+                case 101 /* InstanceOfKeyword */:
+                    return checkInstanceOfExpression(left, right, leftType, rightType);
+                case 100 /* InKeyword */:
+                    return checkInExpression(left, right, leftType, rightType);
+                case 55 /* AmpersandAmpersandToken */:
+                case 75 /* AmpersandAmpersandEqualsToken */: {
+                    var resultType_2 = getTypeFacts(leftType) & 4194304 /* Truthy */ ?
+                        getUnionType([extractDefinitelyFalsyTypes(strictNullChecks ? leftType : getBaseTypeOfLiteralType(rightType)), rightType]) :
+                        leftType;
+                    if (operator === 75 /* AmpersandAmpersandEqualsToken */) {
+                        checkAssignmentOperator(rightType);
+                    }
+                    return resultType_2;
                 }
-                else if (!isTypeAssignableTo(getIndexTypeOrString(rightType), leftType)) {
-                    error(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_must_be_of_type_string_or_any);
+                case 56 /* BarBarToken */:
+                case 74 /* BarBarEqualsToken */: {
+                    var resultType_3 = getTypeFacts(leftType) & 8388608 /* Falsy */ ?
+                        getUnionType([removeDefinitelyFalsyTypes(leftType), rightType], 2 /* Subtype */) :
+                        leftType;
+                    if (operator === 74 /* BarBarEqualsToken */) {
+                        checkAssignmentOperator(rightType);
+                    }
+                    return resultType_3;
                 }
-                else {
-                    // run check only former check succeeded to avoid cascading errors
-                    checkReferenceExpression(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_may_not_be_an_optional_property_access);
+                case 60 /* QuestionQuestionToken */:
+                case 76 /* QuestionQuestionEqualsToken */: {
+                    var resultType_4 = getTypeFacts(leftType) & 262144 /* EQUndefinedOrNull */ ?
+                        getUnionType([getNonNullableType(leftType), rightType], 2 /* Subtype */) :
+                        leftType;
+                    if (operator === 76 /* QuestionQuestionEqualsToken */) {
+                        checkAssignmentOperator(rightType);
+                    }
+                    return resultType_4;
                 }
+                case 62 /* EqualsToken */:
+                    var declKind = ts.isBinaryExpression(left.parent) ? ts.getAssignmentDeclarationKind(left.parent) : 0 /* None */;
+                    checkAssignmentDeclaration(declKind, rightType);
+                    if (isAssignmentDeclaration(declKind)) {
+                        if (!(rightType.flags & 524288 /* Object */) ||
+                            declKind !== 2 /* ModuleExports */ &&
+                                declKind !== 6 /* Prototype */ &&
+                                !isEmptyObjectType(rightType) &&
+                                !isFunctionObjectType(rightType) &&
+                                !(ts.getObjectFlags(rightType) & 1 /* Class */)) {
+                            // don't check assignability of module.exports=, C.prototype=, or expando types because they will necessarily be incomplete
+                            checkAssignmentOperator(rightType);
+                        }
+                        return leftType;
+                    }
+                    else {
+                        checkAssignmentOperator(rightType);
+                        return getRegularTypeOfObjectLiteral(rightType);
+                    }
+                case 27 /* CommaToken */:
+                    if (!compilerOptions.allowUnreachableCode && isSideEffectFree(left) && !isEvalNode(right)) {
+                        var sf = ts.getSourceFileOfNode(left);
+                        var sourceText = sf.text;
+                        var start_3 = ts.skipTrivia(sourceText, left.pos);
+                        var isInDiag2657 = sf.parseDiagnostics.some(function (diag) {
+                            if (diag.code !== ts.Diagnostics.JSX_expressions_must_have_one_parent_element.code)
+                                return false;
+                            return ts.textSpanContainsPosition(diag, start_3);
+                        });
+                        if (!isInDiag2657)
+                            error(left, ts.Diagnostics.Left_side_of_comma_operator_is_unused_and_has_no_side_effects);
+                    }
+                    return rightType;
+                default:
+                    return ts.Debug.fail();
             }
-            // unknownType is returned i.e. if node.expression is identifier whose name cannot be resolved
-            // in this case error about missing name is already reported - do not report extra one
-            if (rightType === neverType || !isTypeAssignableToKind(rightType, 67108864 /* NonPrimitive */ | 58982400 /* InstantiableNonPrimitive */)) {
-                error(node.expression, ts.Diagnostics.The_right_hand_side_of_a_for_in_statement_must_be_of_type_any_an_object_type_or_a_type_parameter_but_here_has_type_0, typeToString(rightType));
+            function bothAreBigIntLike(left, right) {
+                return isTypeAssignableToKind(left, 2112 /* BigIntLike */) && isTypeAssignableToKind(right, 2112 /* BigIntLike */);
             }
-            checkSourceElement(node.statement);
-            if (node.locals) {
-                registerForUnusedIdentifiersCheck(node);
+            function checkAssignmentDeclaration(kind, rightType) {
+                if (kind === 2 /* ModuleExports */) {
+                    for (var _i = 0, _a = getPropertiesOfObjectType(rightType); _i < _a.length; _i++) {
+                        var prop = _a[_i];
+                        var propType = getTypeOfSymbol(prop);
+                        if (propType.symbol && propType.symbol.flags & 32 /* Class */) {
+                            var name = prop.escapedName;
+                            var symbol = resolveName(prop.valueDeclaration, name, 788968 /* Type */, undefined, name, /*isUse*/ false);
+                            if (symbol && symbol.declarations.some(ts.isJSDocTypedefTag)) {
+                                addDuplicateDeclarationErrorsForSymbols(symbol, ts.Diagnostics.Duplicate_identifier_0, ts.unescapeLeadingUnderscores(name), prop);
+                                addDuplicateDeclarationErrorsForSymbols(prop, ts.Diagnostics.Duplicate_identifier_0, ts.unescapeLeadingUnderscores(name), symbol);
+                            }
+                        }
+                    }
+                }
             }
-        }
-        function checkForInOrForOfVariableDeclaration(iterationStatement) {
-            var variableDeclarationList = iterationStatement.initializer;
-            // checkGrammarForInOrForOfStatement will check that there is exactly one declaration.
-            if (variableDeclarationList.declarations.length >= 1) {
-                var decl = variableDeclarationList.declarations[0];
-                checkVariableDeclaration(decl);
+            function isEvalNode(node) {
+                return node.kind === 78 /* Identifier */ && node.escapedText === "eval";
             }
-        }
-        function checkRightHandSideOfForOf(statement) {
-            var use = statement.awaitModifier ? 15 /* ForAwaitOf */ : 13 /* ForOf */;
-            return checkIteratedTypeOrElementType(use, checkNonNullExpression(statement.expression), undefinedType, statement.expression);
-        }
-        function checkIteratedTypeOrElementType(use, inputType, sentType, errorNode) {
-            if (isTypeAny(inputType)) {
-                return inputType;
+            // Return true if there was no error, false if there was an error.
+            function checkForDisallowedESSymbolOperand(operator) {
+                var offendingSymbolOperand = maybeTypeOfKind(leftType, 12288 /* ESSymbolLike */) ? left :
+                    maybeTypeOfKind(rightType, 12288 /* ESSymbolLike */) ? right :
+                        undefined;
+                if (offendingSymbolOperand) {
+                    error(offendingSymbolOperand, ts.Diagnostics.The_0_operator_cannot_be_applied_to_type_symbol, ts.tokenToString(operator));
+                    return false;
+                }
+                return true;
             }
-            return getIteratedTypeOrElementType(use, inputType, sentType, errorNode, /*checkAssignability*/ true) || anyType;
-        }
-        /**
-         * When consuming an iterable type in a for..of, spread, or iterator destructuring assignment
-         * we want to get the iterated type of an iterable for ES2015 or later, or the iterated type
-         * of a iterable (if defined globally) or element type of an array like for ES2015 or earlier.
-         */
-        function getIteratedTypeOrElementType(use, inputType, sentType, errorNode, checkAssignability) {
-            var allowAsyncIterables = (use & 2 /* AllowsAsyncIterablesFlag */) !== 0;
-            if (inputType === neverType) {
-                reportTypeNotIterableError(errorNode, inputType, allowAsyncIterables); // TODO: GH#18217
-                return undefined;
+            function getSuggestedBooleanOperator(operator) {
+                switch (operator) {
+                    case 51 /* BarToken */:
+                    case 73 /* BarEqualsToken */:
+                        return 56 /* BarBarToken */;
+                    case 52 /* CaretToken */:
+                    case 77 /* CaretEqualsToken */:
+                        return 37 /* ExclamationEqualsEqualsToken */;
+                    case 50 /* AmpersandToken */:
+                    case 72 /* AmpersandEqualsToken */:
+                        return 55 /* AmpersandAmpersandToken */;
+                    default:
+                        return undefined;
+                }
             }
-            var uplevelIteration = languageVersion >= 2 /* ES2015 */;
-            var downlevelIteration = !uplevelIteration && compilerOptions.downlevelIteration;
-            // Get the iterated type of an `Iterable<T>` or `IterableIterator<T>` only in ES2015
-            // or higher, when inside of an async generator or for-await-if, or when
-            // downlevelIteration is requested.
-            if (uplevelIteration || downlevelIteration || allowAsyncIterables) {
-                // We only report errors for an invalid iterable type in ES2015 or higher.
-                var iterationTypes = getIterationTypesOfIterable(inputType, use, uplevelIteration ? errorNode : undefined);
-                if (checkAssignability) {
-                    if (iterationTypes) {
-                        var diagnostic = use & 8 /* ForOfFlag */ ? ts.Diagnostics.Cannot_iterate_value_because_the_next_method_of_its_iterator_expects_type_1_but_for_of_will_always_send_0 :
-                            use & 32 /* SpreadFlag */ ? ts.Diagnostics.Cannot_iterate_value_because_the_next_method_of_its_iterator_expects_type_1_but_array_spread_will_always_send_0 :
-                                use & 64 /* DestructuringFlag */ ? ts.Diagnostics.Cannot_iterate_value_because_the_next_method_of_its_iterator_expects_type_1_but_array_destructuring_will_always_send_0 :
-                                    use & 16 /* YieldStarFlag */ ? ts.Diagnostics.Cannot_delegate_iteration_to_value_because_the_next_method_of_its_iterator_expects_type_1_but_the_containing_generator_will_always_send_0 :
-                                        undefined;
-                        if (diagnostic) {
-                            checkTypeAssignableTo(sentType, iterationTypes.nextType, errorNode, diagnostic);
-                        }
+            function checkAssignmentOperator(valueType) {
+                if (produceDiagnostics && ts.isAssignmentOperator(operator)) {
+                    // TypeScript 1.0 spec (April 2014): 4.17
+                    // An assignment of the form
+                    //    VarExpr = ValueExpr
+                    // requires VarExpr to be classified as a reference
+                    // A compound assignment furthermore requires VarExpr to be classified as a reference (section 4.1)
+                    // and the type of the non-compound operation to be assignable to the type of VarExpr.
+                    if (checkReferenceExpression(left, ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_left_hand_side_of_an_assignment_expression_may_not_be_an_optional_property_access)
+                        && (!ts.isIdentifier(left) || ts.unescapeLeadingUnderscores(left.escapedText) !== "exports")) {
+                        // to avoid cascading errors check assignability only if 'isReference' check succeeded and no errors were reported
+                        checkTypeAssignableToAndOptionallyElaborate(valueType, leftType, left, right);
                     }
                 }
-                if (iterationTypes || uplevelIteration) {
-                    return iterationTypes && iterationTypes.yieldType;
+            }
+            function isAssignmentDeclaration(kind) {
+                var _a;
+                switch (kind) {
+                    case 2 /* ModuleExports */:
+                        return true;
+                    case 1 /* ExportsProperty */:
+                    case 5 /* Property */:
+                    case 6 /* Prototype */:
+                    case 3 /* PrototypeProperty */:
+                    case 4 /* ThisProperty */:
+                        var symbol = getSymbolOfNode(left);
+                        var init = ts.getAssignedExpandoInitializer(right);
+                        return !!init && ts.isObjectLiteralExpression(init) &&
+                            !!((_a = symbol === null || symbol === void 0 ? void 0 : symbol.exports) === null || _a === void 0 ? void 0 : _a.size);
+                    default:
+                        return false;
                 }
             }
-            var arrayType = inputType;
-            var reportedError = false;
-            var hasStringConstituent = false;
-            // If strings are permitted, remove any string-like constituents from the array type.
-            // This allows us to find other non-string element types from an array unioned with
-            // a string.
-            if (use & 4 /* AllowsStringInputFlag */) {
-                if (arrayType.flags & 1048576 /* Union */) {
-                    // After we remove all types that are StringLike, we will know if there was a string constituent
-                    // based on whether the result of filter is a new array.
-                    var arrayTypes = inputType.types;
-                    var filteredTypes = ts.filter(arrayTypes, function (t) { return !(t.flags & 132 /* StringLike */); });
-                    if (filteredTypes !== arrayTypes) {
-                        arrayType = getUnionType(filteredTypes, 2 /* Subtype */);
-                    }
+            /**
+             * Returns true if an error is reported
+             */
+            function reportOperatorErrorUnless(typesAreCompatible) {
+                if (!typesAreCompatible(leftType, rightType)) {
+                    reportOperatorError(typesAreCompatible);
+                    return true;
+                }
+                return false;
+            }
+            function reportOperatorError(isRelated) {
+                var _a;
+                var wouldWorkWithAwait = false;
+                var errNode = errorNode || operatorToken;
+                if (isRelated) {
+                    var awaitedLeftType = getAwaitedType(leftType);
+                    var awaitedRightType = getAwaitedType(rightType);
+                    wouldWorkWithAwait = !(awaitedLeftType === leftType && awaitedRightType === rightType)
+                        && !!(awaitedLeftType && awaitedRightType)
+                        && isRelated(awaitedLeftType, awaitedRightType);
                 }
-                else if (arrayType.flags & 132 /* StringLike */) {
-                    arrayType = neverType;
+                var effectiveLeft = leftType;
+                var effectiveRight = rightType;
+                if (!wouldWorkWithAwait && isRelated) {
+                    _a = getBaseTypesIfUnrelated(leftType, rightType, isRelated), effectiveLeft = _a[0], effectiveRight = _a[1];
                 }
-                hasStringConstituent = arrayType !== inputType;
-                if (hasStringConstituent) {
-                    if (languageVersion < 1 /* ES5 */) {
-                        if (errorNode) {
-                            error(errorNode, ts.Diagnostics.Using_a_string_in_a_for_of_statement_is_only_supported_in_ECMAScript_5_and_higher);
-                            reportedError = true;
-                        }
-                    }
-                    // Now that we've removed all the StringLike types, if no constituents remain, then the entire
-                    // arrayOrStringType was a string.
-                    if (arrayType.flags & 131072 /* Never */) {
-                        return stringType;
-                    }
+                var _b = getTypeNamesForErrorDisplay(effectiveLeft, effectiveRight), leftStr = _b[0], rightStr = _b[1];
+                if (!tryGiveBetterPrimaryError(errNode, wouldWorkWithAwait, leftStr, rightStr)) {
+                    errorAndMaybeSuggestAwait(errNode, wouldWorkWithAwait, ts.Diagnostics.Operator_0_cannot_be_applied_to_types_1_and_2, ts.tokenToString(operatorToken.kind), leftStr, rightStr);
                 }
             }
-            if (!isArrayLikeType(arrayType)) {
-                if (errorNode && !reportedError) {
-                    // Which error we report depends on whether we allow strings or if there was a
-                    // string constituent. For example, if the input type is number | string, we
-                    // want to say that number is not an array type. But if the input was just
-                    // number and string input is allowed, we want to say that number is not an
-                    // array type or a string type.
-                    var yieldType = getIterationTypeOfIterable(use, 0 /* Yield */, inputType, /*errorNode*/ undefined);
-                    var _a = !(use & 4 /* AllowsStringInputFlag */) || hasStringConstituent
-                        ? downlevelIteration
-                            ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator, true]
-                            : yieldType
-                                ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type_Use_compiler_option_downlevelIteration_to_allow_iterating_of_iterators, false]
-                                : [ts.Diagnostics.Type_0_is_not_an_array_type, true]
-                        : downlevelIteration
-                            ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator, true]
-                            : yieldType
-                                ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type_Use_compiler_option_downlevelIteration_to_allow_iterating_of_iterators, false]
-                                : [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type, true], defaultDiagnostic = _a[0], maybeMissingAwait = _a[1];
-                    errorAndMaybeSuggestAwait(errorNode, maybeMissingAwait && !!getAwaitedTypeOfPromise(arrayType), defaultDiagnostic, typeToString(arrayType));
+            function tryGiveBetterPrimaryError(errNode, maybeMissingAwait, leftStr, rightStr) {
+                var typeName;
+                switch (operatorToken.kind) {
+                    case 36 /* EqualsEqualsEqualsToken */:
+                    case 34 /* EqualsEqualsToken */:
+                        typeName = "false";
+                        break;
+                    case 37 /* ExclamationEqualsEqualsToken */:
+                    case 35 /* ExclamationEqualsToken */:
+                        typeName = "true";
                 }
-                return hasStringConstituent ? stringType : undefined;
-            }
-            var arrayElementType = getIndexTypeOfType(arrayType, 1 /* Number */);
-            if (hasStringConstituent && arrayElementType) {
-                // This is just an optimization for the case where arrayOrStringType is string | string[]
-                if (arrayElementType.flags & 132 /* StringLike */) {
-                    return stringType;
+                if (typeName) {
+                    return errorAndMaybeSuggestAwait(errNode, maybeMissingAwait, ts.Diagnostics.This_condition_will_always_return_0_since_the_types_1_and_2_have_no_overlap, typeName, leftStr, rightStr);
                 }
-                return getUnionType([arrayElementType, stringType], 2 /* Subtype */);
+                return undefined;
             }
-            return arrayElementType;
         }
-        /**
-         * Gets the requested "iteration type" from an `Iterable`-like or `AsyncIterable`-like type.
-         */
-        function getIterationTypeOfIterable(use, typeKind, inputType, errorNode) {
-            if (isTypeAny(inputType)) {
-                return undefined;
+        function getBaseTypesIfUnrelated(leftType, rightType, isRelated) {
+            var effectiveLeft = leftType;
+            var effectiveRight = rightType;
+            var leftBase = getBaseTypeOfLiteralType(leftType);
+            var rightBase = getBaseTypeOfLiteralType(rightType);
+            if (!isRelated(leftBase, rightBase)) {
+                effectiveLeft = leftBase;
+                effectiveRight = rightBase;
             }
-            var iterationTypes = getIterationTypesOfIterable(inputType, use, errorNode);
-            return iterationTypes && iterationTypes[getIterationTypesKeyFromIterationTypeKind(typeKind)];
+            return [effectiveLeft, effectiveRight];
         }
-        function createIterationTypes(yieldType, returnType, nextType) {
-            // `yieldType` and `returnType` are defaulted to `neverType` they each will be combined
-            // via `getUnionType` when merging iteration types. `nextType` is defined as `unknownType`
-            // as it is combined via `getIntersectionType` when merging iteration types.
-            if (yieldType === void 0) { yieldType = neverType; }
-            if (returnType === void 0) { returnType = neverType; }
-            if (nextType === void 0) { nextType = unknownType; }
-            // Use the cache only for intrinsic types to keep it small as they are likely to be
-            // more frequently created (i.e. `Iterator<number, void, unknown>`). Iteration types
-            // are also cached on the type they are requested for, so we shouldn't need to maintain
-            // the cache for less-frequently used types.
-            if (yieldType.flags & 67359327 /* Intrinsic */ &&
-                returnType.flags & (1 /* Any */ | 131072 /* Never */ | 2 /* Unknown */ | 16384 /* Void */ | 32768 /* Undefined */) &&
-                nextType.flags & (1 /* Any */ | 131072 /* Never */ | 2 /* Unknown */ | 16384 /* Void */ | 32768 /* Undefined */)) {
-                var id = getTypeListId([yieldType, returnType, nextType]);
-                var iterationTypes = iterationTypesCache.get(id);
-                if (!iterationTypes) {
-                    iterationTypes = { yieldType: yieldType, returnType: returnType, nextType: nextType };
-                    iterationTypesCache.set(id, iterationTypes);
+        function checkYieldExpression(node) {
+            // Grammar checking
+            if (produceDiagnostics) {
+                if (!(node.flags & 8192 /* YieldContext */)) {
+                    grammarErrorOnFirstToken(node, ts.Diagnostics.A_yield_expression_is_only_allowed_in_a_generator_body);
+                }
+                if (isInParameterInitializerBeforeContainingFunction(node)) {
+                    error(node, ts.Diagnostics.yield_expressions_cannot_be_used_in_a_parameter_initializer);
                 }
-                return iterationTypes;
             }
-            return { yieldType: yieldType, returnType: returnType, nextType: nextType };
-        }
-        /**
-         * Combines multiple `IterationTypes` records.
-         *
-         * If `array` is empty or all elements are missing or are references to `noIterationTypes`,
-         * then `noIterationTypes` is returned. Otherwise, an `IterationTypes` record is returned
-         * for the combined iteration types.
-         */
-        function combineIterationTypes(array) {
-            var yieldTypes;
-            var returnTypes;
-            var nextTypes;
-            for (var _i = 0, array_10 = array; _i < array_10.length; _i++) {
-                var iterationTypes = array_10[_i];
-                if (iterationTypes === undefined || iterationTypes === noIterationTypes) {
-                    continue;
+            var func = ts.getContainingFunction(node);
+            if (!func)
+                return anyType;
+            var functionFlags = ts.getFunctionFlags(func);
+            if (!(functionFlags & 1 /* Generator */)) {
+                // If the user's code is syntactically correct, the func should always have a star. After all, we are in a yield context.
+                return anyType;
+            }
+            var isAsync = (functionFlags & 2 /* Async */) !== 0;
+            if (node.asteriskToken) {
+                // Async generator functions prior to ESNext require the __await, __asyncDelegator,
+                // and __asyncValues helpers
+                if (isAsync && languageVersion < 99 /* ESNext */) {
+                    checkExternalEmitHelpers(node, 53248 /* AsyncDelegatorIncludes */);
                 }
-                if (iterationTypes === anyIterationTypes) {
-                    return anyIterationTypes;
+                // Generator functions prior to ES2015 require the __values helper
+                if (!isAsync && languageVersion < 2 /* ES2015 */ && compilerOptions.downlevelIteration) {
+                    checkExternalEmitHelpers(node, 256 /* Values */);
                 }
-                yieldTypes = ts.append(yieldTypes, iterationTypes.yieldType);
-                returnTypes = ts.append(returnTypes, iterationTypes.returnType);
-                nextTypes = ts.append(nextTypes, iterationTypes.nextType);
             }
-            if (yieldTypes || returnTypes || nextTypes) {
-                return createIterationTypes(yieldTypes && getUnionType(yieldTypes), returnTypes && getUnionType(returnTypes), nextTypes && getIntersectionType(nextTypes));
+            // There is no point in doing an assignability check if the function
+            // has no explicit return type because the return type is directly computed
+            // from the yield expressions.
+            var returnType = getReturnTypeFromAnnotation(func);
+            var iterationTypes = returnType && getIterationTypesOfGeneratorFunctionReturnType(returnType, isAsync);
+            var signatureYieldType = iterationTypes && iterationTypes.yieldType || anyType;
+            var signatureNextType = iterationTypes && iterationTypes.nextType || anyType;
+            var resolvedSignatureNextType = isAsync ? getAwaitedType(signatureNextType) || anyType : signatureNextType;
+            var yieldExpressionType = node.expression ? checkExpression(node.expression) : undefinedWideningType;
+            var yieldedType = getYieldedTypeOfYieldExpression(node, yieldExpressionType, resolvedSignatureNextType, isAsync);
+            if (returnType && yieldedType) {
+                checkTypeAssignableToAndOptionallyElaborate(yieldedType, signatureYieldType, node.expression || node, node.expression);
             }
-            return noIterationTypes;
-        }
-        function getCachedIterationTypes(type, cacheKey) {
-            return type[cacheKey];
-        }
-        function setCachedIterationTypes(type, cacheKey, cachedTypes) {
-            return type[cacheKey] = cachedTypes;
-        }
-        /**
-         * Gets the *yield*, *return*, and *next* types from an `Iterable`-like or `AsyncIterable`-like type.
-         *
-         * At every level that involves analyzing return types of signatures, we union the return types of all the signatures.
-         *
-         * Another thing to note is that at any step of this process, we could run into a dead end,
-         * meaning either the property is missing, or we run into the anyType. If either of these things
-         * happens, we return `undefined` to signal that we could not find the iteration type. If a property
-         * is missing, and the previous step did not result in `any`, then we also give an error if the
-         * caller requested it. Then the caller can decide what to do in the case where there is no iterated
-         * type.
-         *
-         * For a **for-of** statement, `yield*` (in a normal generator), spread, array
-         * destructuring, or normal generator we will only ever look for a `[Symbol.iterator]()`
-         * method.
-         *
-         * For an async generator we will only ever look at the `[Symbol.asyncIterator]()` method.
-         *
-         * For a **for-await-of** statement or a `yield*` in an async generator we will look for
-         * the `[Symbol.asyncIterator]()` method first, and then the `[Symbol.iterator]()` method.
-         */
-        function getIterationTypesOfIterable(type, use, errorNode) {
-            if (isTypeAny(type)) {
-                return anyIterationTypes;
+            if (node.asteriskToken) {
+                var use = isAsync ? 19 /* AsyncYieldStar */ : 17 /* YieldStar */;
+                return getIterationTypeOfIterable(use, 1 /* Return */, yieldExpressionType, node.expression)
+                    || anyType;
             }
-            if (!(type.flags & 1048576 /* Union */)) {
-                var iterationTypes_1 = getIterationTypesOfIterableWorker(type, use, errorNode);
-                if (iterationTypes_1 === noIterationTypes) {
-                    if (errorNode) {
-                        reportTypeNotIterableError(errorNode, type, !!(use & 2 /* AllowsAsyncIterablesFlag */));
-                    }
-                    return undefined;
-                }
-                return iterationTypes_1;
+            else if (returnType) {
+                return getIterationTypeOfGeneratorFunctionReturnType(2 /* Next */, returnType, isAsync)
+                    || anyType;
             }
-            var cacheKey = use & 2 /* AllowsAsyncIterablesFlag */ ? "iterationTypesOfAsyncIterable" : "iterationTypesOfIterable";
-            var cachedTypes = getCachedIterationTypes(type, cacheKey);
-            if (cachedTypes)
-                return cachedTypes === noIterationTypes ? undefined : cachedTypes;
-            var allIterationTypes;
-            for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
-                var constituent = _a[_i];
-                var iterationTypes_2 = getIterationTypesOfIterableWorker(constituent, use, errorNode);
-                if (iterationTypes_2 === noIterationTypes) {
-                    if (errorNode) {
-                        reportTypeNotIterableError(errorNode, type, !!(use & 2 /* AllowsAsyncIterablesFlag */));
-                        errorNode = undefined;
-                    }
-                }
-                else {
-                    allIterationTypes = ts.append(allIterationTypes, iterationTypes_2);
+            return getContextualIterationType(2 /* Next */, func) || anyType;
+        }
+        function checkConditionalExpression(node, checkMode) {
+            var type = checkTruthinessExpression(node.condition);
+            checkTestingKnownTruthyCallableType(node.condition, node.whenTrue, type);
+            var type1 = checkExpression(node.whenTrue, checkMode);
+            var type2 = checkExpression(node.whenFalse, checkMode);
+            return getUnionType([type1, type2], 2 /* Subtype */);
+        }
+        function checkTemplateExpression(node) {
+            var texts = [node.head.text];
+            var types = [];
+            for (var _i = 0, _a = node.templateSpans; _i < _a.length; _i++) {
+                var span = _a[_i];
+                var type = checkExpression(span.expression);
+                if (maybeTypeOfKind(type, 12288 /* ESSymbolLike */)) {
+                    error(span.expression, ts.Diagnostics.Implicit_conversion_of_a_symbol_to_a_string_will_fail_at_runtime_Consider_wrapping_this_expression_in_String);
                 }
+                texts.push(span.literal.text);
+                types.push(isTypeAssignableTo(type, templateConstraintType) ? type : stringType);
             }
-            var iterationTypes = allIterationTypes ? combineIterationTypes(allIterationTypes) : noIterationTypes;
-            setCachedIterationTypes(type, cacheKey, iterationTypes);
-            return iterationTypes === noIterationTypes ? undefined : iterationTypes;
+            return isConstContext(node) ? getTemplateLiteralType(texts, types) : stringType;
         }
-        function getAsyncFromSyncIterationTypes(iterationTypes, errorNode) {
-            if (iterationTypes === noIterationTypes)
-                return noIterationTypes;
-            if (iterationTypes === anyIterationTypes)
-                return anyIterationTypes;
-            var yieldType = iterationTypes.yieldType, returnType = iterationTypes.returnType, nextType = iterationTypes.nextType;
-            return createIterationTypes(getAwaitedType(yieldType, errorNode) || anyType, getAwaitedType(returnType, errorNode) || anyType, nextType);
+        function getContextNode(node) {
+            if (node.kind === 281 /* JsxAttributes */ && !ts.isJsxSelfClosingElement(node.parent)) {
+                return node.parent.parent; // Needs to be the root JsxElement, so it encompasses the attributes _and_ the children (which are essentially part of the attributes)
+            }
+            return node;
         }
-        /**
-         * Gets the *yield*, *return*, and *next* types from a non-union type.
-         *
-         * If we are unable to find the *yield*, *return*, and *next* types, `noIterationTypes` is
-         * returned to indicate to the caller that it should report an error. Otherwise, an
-         * `IterationTypes` record is returned.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterable` instead.
-         */
-        function getIterationTypesOfIterableWorker(type, use, errorNode) {
-            if (isTypeAny(type)) {
-                return anyIterationTypes;
+        function checkExpressionWithContextualType(node, contextualType, inferenceContext, checkMode) {
+            var context = getContextNode(node);
+            var saveContextualType = context.contextualType;
+            var saveInferenceContext = context.inferenceContext;
+            try {
+                context.contextualType = contextualType;
+                context.inferenceContext = inferenceContext;
+                var type = checkExpression(node, checkMode | 1 /* Contextual */ | (inferenceContext ? 2 /* Inferential */ : 0));
+                // We strip literal freshness when an appropriate contextual type is present such that contextually typed
+                // literals always preserve their literal types (otherwise they might widen during type inference). An alternative
+                // here would be to not mark contextually typed literals as fresh in the first place.
+                var result = maybeTypeOfKind(type, 2944 /* Literal */) && isLiteralOfContextualType(type, instantiateContextualType(contextualType, node)) ?
+                    getRegularTypeOfLiteralType(type) : type;
+                return result;
             }
-            if (use & 2 /* AllowsAsyncIterablesFlag */) {
-                var iterationTypes = getIterationTypesOfIterableCached(type, asyncIterationTypesResolver) ||
-                    getIterationTypesOfIterableFast(type, asyncIterationTypesResolver);
-                if (iterationTypes) {
-                    return iterationTypes;
+            finally {
+                // In the event our operation is canceled or some other exception occurs, reset the contextual type
+                // so that we do not accidentally hold onto an instance of the checker, as a Type created in the services layer
+                // may hold onto the checker that created it.
+                context.contextualType = saveContextualType;
+                context.inferenceContext = saveInferenceContext;
+            }
+        }
+        function checkExpressionCached(node, checkMode) {
+            var links = getNodeLinks(node);
+            if (!links.resolvedType) {
+                if (checkMode && checkMode !== 0 /* Normal */) {
+                    return checkExpression(node, checkMode);
                 }
+                // When computing a type that we're going to cache, we need to ignore any ongoing control flow
+                // analysis because variables may have transient types in indeterminable states. Moving flowLoopStart
+                // to the top of the stack ensures all transient types are computed from a known point.
+                var saveFlowLoopStart = flowLoopStart;
+                var saveFlowTypeCache = flowTypeCache;
+                flowLoopStart = flowLoopCount;
+                flowTypeCache = undefined;
+                links.resolvedType = checkExpression(node, checkMode);
+                flowTypeCache = saveFlowTypeCache;
+                flowLoopStart = saveFlowLoopStart;
             }
-            if (use & 1 /* AllowsSyncIterablesFlag */) {
-                var iterationTypes = getIterationTypesOfIterableCached(type, syncIterationTypesResolver) ||
-                    getIterationTypesOfIterableFast(type, syncIterationTypesResolver);
-                if (iterationTypes) {
-                    if (use & 2 /* AllowsAsyncIterablesFlag */) {
-                        // for a sync iterable in an async context, only use the cached types if they are valid.
-                        if (iterationTypes !== noIterationTypes) {
-                            return setCachedIterationTypes(type, "iterationTypesOfAsyncIterable", getAsyncFromSyncIterationTypes(iterationTypes, errorNode));
-                        }
-                    }
-                    else {
-                        return iterationTypes;
+            return links.resolvedType;
+        }
+        function isTypeAssertion(node) {
+            node = ts.skipParentheses(node);
+            return node.kind === 206 /* TypeAssertionExpression */ || node.kind === 224 /* AsExpression */;
+        }
+        function checkDeclarationInitializer(declaration, contextualType) {
+            var initializer = ts.getEffectiveInitializer(declaration);
+            var type = getQuickTypeOfExpression(initializer) ||
+                (contextualType ? checkExpressionWithContextualType(initializer, contextualType, /*inferenceContext*/ undefined, 0 /* Normal */) : checkExpressionCached(initializer));
+            return ts.isParameter(declaration) && declaration.name.kind === 197 /* ArrayBindingPattern */ &&
+                isTupleType(type) && !type.target.hasRestElement && getTypeReferenceArity(type) < declaration.name.elements.length ?
+                padTupleType(type, declaration.name) : type;
+        }
+        function padTupleType(type, pattern) {
+            var patternElements = pattern.elements;
+            var elementTypes = getTypeArguments(type).slice();
+            var elementFlags = type.target.elementFlags.slice();
+            for (var i = getTypeReferenceArity(type); i < patternElements.length; i++) {
+                var e = patternElements[i];
+                if (i < patternElements.length - 1 || !(e.kind === 198 /* BindingElement */ && e.dotDotDotToken)) {
+                    elementTypes.push(!ts.isOmittedExpression(e) && hasDefaultValue(e) ? getTypeFromBindingElement(e, /*includePatternInType*/ false, /*reportErrors*/ false) : anyType);
+                    elementFlags.push(2 /* Optional */);
+                    if (!ts.isOmittedExpression(e) && !hasDefaultValue(e)) {
+                        reportImplicitAny(e, anyType);
                     }
                 }
             }
-            if (use & 2 /* AllowsAsyncIterablesFlag */) {
-                var iterationTypes = getIterationTypesOfIterableSlow(type, asyncIterationTypesResolver, errorNode);
-                if (iterationTypes !== noIterationTypes) {
-                    return iterationTypes;
+            return createTupleType(elementTypes, elementFlags, type.target.readonly);
+        }
+        function widenTypeInferredFromInitializer(declaration, type) {
+            var widened = ts.getCombinedNodeFlags(declaration) & 2 /* Const */ || ts.isDeclarationReadonly(declaration) ? type : getWidenedLiteralType(type);
+            if (ts.isInJSFile(declaration)) {
+                if (widened.flags & 98304 /* Nullable */) {
+                    reportImplicitAny(declaration, anyType);
+                    return anyType;
+                }
+                else if (isEmptyArrayLiteralType(widened)) {
+                    reportImplicitAny(declaration, anyArrayType);
+                    return anyArrayType;
                 }
             }
-            if (use & 1 /* AllowsSyncIterablesFlag */) {
-                var iterationTypes = getIterationTypesOfIterableSlow(type, syncIterationTypesResolver, errorNode);
-                if (iterationTypes !== noIterationTypes) {
-                    if (use & 2 /* AllowsAsyncIterablesFlag */) {
-                        return setCachedIterationTypes(type, "iterationTypesOfAsyncIterable", iterationTypes
-                            ? getAsyncFromSyncIterationTypes(iterationTypes, errorNode)
-                            : noIterationTypes);
-                    }
-                    else {
-                        return iterationTypes;
-                    }
+            return widened;
+        }
+        function isLiteralOfContextualType(candidateType, contextualType) {
+            if (contextualType) {
+                if (contextualType.flags & 3145728 /* UnionOrIntersection */) {
+                    var types = contextualType.types;
+                    return ts.some(types, function (t) { return isLiteralOfContextualType(candidateType, t); });
+                }
+                if (contextualType.flags & 58982400 /* InstantiableNonPrimitive */) {
+                    // If the contextual type is a type variable constrained to a primitive type, consider
+                    // this a literal context for literals of that primitive type. For example, given a
+                    // type parameter 'T extends string', infer string literal types for T.
+                    var constraint = getBaseConstraintOfType(contextualType) || unknownType;
+                    return maybeTypeOfKind(constraint, 4 /* String */) && maybeTypeOfKind(candidateType, 128 /* StringLiteral */) ||
+                        maybeTypeOfKind(constraint, 8 /* Number */) && maybeTypeOfKind(candidateType, 256 /* NumberLiteral */) ||
+                        maybeTypeOfKind(constraint, 64 /* BigInt */) && maybeTypeOfKind(candidateType, 2048 /* BigIntLiteral */) ||
+                        maybeTypeOfKind(constraint, 4096 /* ESSymbol */) && maybeTypeOfKind(candidateType, 8192 /* UniqueESSymbol */) ||
+                        isLiteralOfContextualType(candidateType, constraint);
                 }
+                // If the contextual type is a literal of a particular primitive type, we consider this a
+                // literal context for all literals of that primitive type.
+                return !!(contextualType.flags & (128 /* StringLiteral */ | 4194304 /* Index */ | 134217728 /* TemplateLiteral */ | 268435456 /* StringMapping */) && maybeTypeOfKind(candidateType, 128 /* StringLiteral */) ||
+                    contextualType.flags & 256 /* NumberLiteral */ && maybeTypeOfKind(candidateType, 256 /* NumberLiteral */) ||
+                    contextualType.flags & 2048 /* BigIntLiteral */ && maybeTypeOfKind(candidateType, 2048 /* BigIntLiteral */) ||
+                    contextualType.flags & 512 /* BooleanLiteral */ && maybeTypeOfKind(candidateType, 512 /* BooleanLiteral */) ||
+                    contextualType.flags & 8192 /* UniqueESSymbol */ && maybeTypeOfKind(candidateType, 8192 /* UniqueESSymbol */));
             }
-            return noIterationTypes;
+            return false;
         }
-        /**
-         * Gets the *yield*, *return*, and *next* types of an `Iterable`-like or
-         * `AsyncIterable`-like type from the cache.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterable` instead.
-         */
-        function getIterationTypesOfIterableCached(type, resolver) {
-            return getCachedIterationTypes(type, resolver.iterableCacheKey);
+        function isConstContext(node) {
+            var parent = node.parent;
+            return ts.isAssertionExpression(parent) && ts.isConstTypeReference(parent.type) ||
+                (ts.isParenthesizedExpression(parent) || ts.isArrayLiteralExpression(parent) || ts.isSpreadElement(parent)) && isConstContext(parent) ||
+                (ts.isPropertyAssignment(parent) || ts.isShorthandPropertyAssignment(parent) || ts.isTemplateSpan(parent)) && isConstContext(parent.parent);
         }
-        function getIterationTypesOfGlobalIterableType(globalType, resolver) {
-            var globalIterationTypes = getIterationTypesOfIterableCached(globalType, resolver) ||
-                getIterationTypesOfIterableSlow(globalType, resolver, /*errorNode*/ undefined);
-            return globalIterationTypes === noIterationTypes ? defaultIterationTypes : globalIterationTypes;
+        function checkExpressionForMutableLocation(node, checkMode, contextualType, forceTuple) {
+            var type = checkExpression(node, checkMode, forceTuple);
+            return isConstContext(node) ? getRegularTypeOfLiteralType(type) :
+                isTypeAssertion(node) ? type :
+                    getWidenedLiteralLikeTypeForContextualType(type, instantiateContextualType(arguments.length === 2 ? getContextualType(node) : contextualType, node));
         }
-        /**
-         * Gets the *yield*, *return*, and *next* types of an `Iterable`-like or `AsyncIterable`-like
-         * type from from common heuristics.
-         *
-         * If we previously analyzed this type and found no iteration types, `noIterationTypes` is
-         * returned. If we found iteration types, an `IterationTypes` record is returned.
-         * Otherwise, we return `undefined` to indicate to the caller it should perform a more
-         * exhaustive analysis.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterable` instead.
-         */
-        function getIterationTypesOfIterableFast(type, resolver) {
-            // As an optimization, if the type is an instantiation of one of the following global types, then
-            // just grab its related type argument:
-            // - `Iterable<T>` or `AsyncIterable<T>`
-            // - `IterableIterator<T>` or `AsyncIterableIterator<T>`
-            var globalType;
-            if (isReferenceToType(type, globalType = resolver.getGlobalIterableType(/*reportErrors*/ false)) ||
-                isReferenceToType(type, globalType = resolver.getGlobalIterableIteratorType(/*reportErrors*/ false))) {
-                var yieldType = getTypeArguments(type)[0];
-                // The "return" and "next" types of `Iterable` and `IterableIterator` are defined by the
-                // iteration types of their `[Symbol.iterator]()` method. The same is true for their async cousins.
-                // While we define these as `any` and `undefined` in our libs by default, a custom lib *could* use
-                // different definitions.
-                var _a = getIterationTypesOfGlobalIterableType(globalType, resolver), returnType = _a.returnType, nextType = _a.nextType;
-                return setCachedIterationTypes(type, resolver.iterableCacheKey, createIterationTypes(yieldType, returnType, nextType));
+        function checkPropertyAssignment(node, checkMode) {
+            // Do not use hasDynamicName here, because that returns false for well known symbols.
+            // We want to perform checkComputedPropertyName for all computed properties, including
+            // well known symbols.
+            if (node.name.kind === 158 /* ComputedPropertyName */) {
+                checkComputedPropertyName(node.name);
             }
-            // As an optimization, if the type is an instantiation of the following global type, then
-            // just grab its related type arguments:
-            // - `Generator<T, TReturn, TNext>` or `AsyncGenerator<T, TReturn, TNext>`
-            if (isReferenceToType(type, resolver.getGlobalGeneratorType(/*reportErrors*/ false))) {
-                var _b = getTypeArguments(type), yieldType = _b[0], returnType = _b[1], nextType = _b[2];
-                return setCachedIterationTypes(type, resolver.iterableCacheKey, createIterationTypes(yieldType, returnType, nextType));
+            return checkExpressionForMutableLocation(node.initializer, checkMode);
+        }
+        function checkObjectLiteralMethod(node, checkMode) {
+            // Grammar checking
+            checkGrammarMethod(node);
+            // Do not use hasDynamicName here, because that returns false for well known symbols.
+            // We want to perform checkComputedPropertyName for all computed properties, including
+            // well known symbols.
+            if (node.name.kind === 158 /* ComputedPropertyName */) {
+                checkComputedPropertyName(node.name);
             }
+            var uninstantiatedType = checkFunctionExpressionOrObjectLiteralMethod(node, checkMode);
+            return instantiateTypeWithSingleGenericCallSignature(node, uninstantiatedType, checkMode);
         }
-        /**
-         * Gets the *yield*, *return*, and *next* types of an `Iterable`-like or `AsyncIterable`-like
-         * type from its members.
-         *
-         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
-         * record is returned. Otherwise, `noIterationTypes` is returned.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterable` instead.
-         */
-        function getIterationTypesOfIterableSlow(type, resolver, errorNode) {
-            var _a;
-            var method = getPropertyOfType(type, ts.getPropertyNameForKnownSymbolName(resolver.iteratorSymbolName));
-            var methodType = method && !(method.flags & 16777216 /* Optional */) ? getTypeOfSymbol(method) : undefined;
-            if (isTypeAny(methodType)) {
-                return setCachedIterationTypes(type, resolver.iterableCacheKey, anyIterationTypes);
+        function instantiateTypeWithSingleGenericCallSignature(node, type, checkMode) {
+            if (checkMode && checkMode & (2 /* Inferential */ | 8 /* SkipGenericFunctions */)) {
+                var callSignature = getSingleSignature(type, 0 /* Call */, /*allowMembers*/ true);
+                var constructSignature = getSingleSignature(type, 1 /* Construct */, /*allowMembers*/ true);
+                var signature = callSignature || constructSignature;
+                if (signature && signature.typeParameters) {
+                    var contextualType = getApparentTypeOfContextualType(node, 2 /* NoConstraints */);
+                    if (contextualType) {
+                        var contextualSignature = getSingleSignature(getNonNullableType(contextualType), callSignature ? 0 /* Call */ : 1 /* Construct */, /*allowMembers*/ false);
+                        if (contextualSignature && !contextualSignature.typeParameters) {
+                            if (checkMode & 8 /* SkipGenericFunctions */) {
+                                skippedGenericFunction(node, checkMode);
+                                return anyFunctionType;
+                            }
+                            var context = getInferenceContext(node);
+                            // We have an expression that is an argument of a generic function for which we are performing
+                            // type argument inference. The expression is of a function type with a single generic call
+                            // signature and a contextual function type with a single non-generic call signature. Now check
+                            // if the outer function returns a function type with a single non-generic call signature and
+                            // if some of the outer function type parameters have no inferences so far. If so, we can
+                            // potentially add inferred type parameters to the outer function return type.
+                            var returnType = context.signature && getReturnTypeOfSignature(context.signature);
+                            var returnSignature = returnType && getSingleCallOrConstructSignature(returnType);
+                            if (returnSignature && !returnSignature.typeParameters && !ts.every(context.inferences, hasInferenceCandidates)) {
+                                // Instantiate the signature with its own type parameters as type arguments, possibly
+                                // renaming the type parameters to ensure they have unique names.
+                                var uniqueTypeParameters = getUniqueTypeParameters(context, signature.typeParameters);
+                                var instantiatedSignature = getSignatureInstantiationWithoutFillingInTypeArguments(signature, uniqueTypeParameters);
+                                // Infer from the parameters of the instantiated signature to the parameters of the
+                                // contextual signature starting with an empty set of inference candidates.
+                                var inferences_3 = ts.map(context.inferences, function (info) { return createInferenceInfo(info.typeParameter); });
+                                applyToParameterTypes(instantiatedSignature, contextualSignature, function (source, target) {
+                                    inferTypes(inferences_3, source, target, /*priority*/ 0, /*contravariant*/ true);
+                                });
+                                if (ts.some(inferences_3, hasInferenceCandidates)) {
+                                    // We have inference candidates, indicating that one or more type parameters are referenced
+                                    // in the parameter types of the contextual signature. Now also infer from the return type.
+                                    applyToReturnTypes(instantiatedSignature, contextualSignature, function (source, target) {
+                                        inferTypes(inferences_3, source, target);
+                                    });
+                                    // If the type parameters for which we produced candidates do not have any inferences yet,
+                                    // we adopt the new inference candidates and add the type parameters of the expression type
+                                    // to the set of inferred type parameters for the outer function return type.
+                                    if (!hasOverlappingInferences(context.inferences, inferences_3)) {
+                                        mergeInferences(context.inferences, inferences_3);
+                                        context.inferredTypeParameters = ts.concatenate(context.inferredTypeParameters, uniqueTypeParameters);
+                                        return getOrCreateTypeFromSignature(instantiatedSignature);
+                                    }
+                                }
+                            }
+                            return getOrCreateTypeFromSignature(instantiateSignatureInContextOf(signature, contextualSignature, context));
+                        }
+                    }
+                }
             }
-            var signatures = methodType ? getSignaturesOfType(methodType, 0 /* Call */) : undefined;
-            if (!ts.some(signatures)) {
-                return setCachedIterationTypes(type, resolver.iterableCacheKey, noIterationTypes);
+            return type;
+        }
+        function skippedGenericFunction(node, checkMode) {
+            if (checkMode & 2 /* Inferential */) {
+                // We have skipped a generic function during inferential typing. Obtain the inference context and
+                // indicate this has occurred such that we know a second pass of inference is be needed.
+                var context = getInferenceContext(node);
+                context.flags |= 4 /* SkippedGenericFunction */;
             }
-            var iteratorType = getUnionType(ts.map(signatures, getReturnTypeOfSignature), 2 /* Subtype */);
-            var iterationTypes = (_a = getIterationTypesOfIterator(iteratorType, resolver, errorNode)) !== null && _a !== void 0 ? _a : noIterationTypes;
-            return setCachedIterationTypes(type, resolver.iterableCacheKey, iterationTypes);
         }
-        function reportTypeNotIterableError(errorNode, type, allowAsyncIterables) {
-            var message = allowAsyncIterables
-                ? ts.Diagnostics.Type_0_must_have_a_Symbol_asyncIterator_method_that_returns_an_async_iterator
-                : ts.Diagnostics.Type_0_must_have_a_Symbol_iterator_method_that_returns_an_iterator;
-            errorAndMaybeSuggestAwait(errorNode, !!getAwaitedTypeOfPromise(type), message, typeToString(type));
+        function hasInferenceCandidates(info) {
+            return !!(info.candidates || info.contraCandidates);
         }
-        /**
-         * Gets the *yield*, *return*, and *next* types from an `Iterator`-like or `AsyncIterator`-like type.
-         *
-         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
-         * record is returned. Otherwise, `undefined` is returned.
-         */
-        function getIterationTypesOfIterator(type, resolver, errorNode) {
-            if (isTypeAny(type)) {
-                return anyIterationTypes;
+        function hasOverlappingInferences(a, b) {
+            for (var i = 0; i < a.length; i++) {
+                if (hasInferenceCandidates(a[i]) && hasInferenceCandidates(b[i])) {
+                    return true;
+                }
             }
-            var iterationTypes = getIterationTypesOfIteratorCached(type, resolver) ||
-                getIterationTypesOfIteratorFast(type, resolver) ||
-                getIterationTypesOfIteratorSlow(type, resolver, errorNode);
-            return iterationTypes === noIterationTypes ? undefined : iterationTypes;
+            return false;
         }
-        /**
-         * Gets the iteration types of an `Iterator`-like or `AsyncIterator`-like type from the
-         * cache.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterator` instead.
-         */
-        function getIterationTypesOfIteratorCached(type, resolver) {
-            return getCachedIterationTypes(type, resolver.iteratorCacheKey);
+        function mergeInferences(target, source) {
+            for (var i = 0; i < target.length; i++) {
+                if (!hasInferenceCandidates(target[i]) && hasInferenceCandidates(source[i])) {
+                    target[i] = source[i];
+                }
+            }
         }
-        /**
-         * Gets the iteration types of an `Iterator`-like or `AsyncIterator`-like type from the
-         * cache or from common heuristics.
-         *
-         * If we previously analyzed this type and found no iteration types, `noIterationTypes` is
-         * returned. If we found iteration types, an `IterationTypes` record is returned.
-         * Otherwise, we return `undefined` to indicate to the caller it should perform a more
-         * exhaustive analysis.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterator` instead.
-         */
-        function getIterationTypesOfIteratorFast(type, resolver) {
-            // As an optimization, if the type is an instantiation of one of the following global types,
-            // then just grab its related type argument:
-            // - `IterableIterator<T>` or `AsyncIterableIterator<T>`
-            // - `Iterator<T, TReturn, TNext>` or `AsyncIterator<T, TReturn, TNext>`
-            // - `Generator<T, TReturn, TNext>` or `AsyncGenerator<T, TReturn, TNext>`
-            var globalType = resolver.getGlobalIterableIteratorType(/*reportErrors*/ false);
-            if (isReferenceToType(type, globalType)) {
-                var yieldType = getTypeArguments(type)[0];
-                // The "return" and "next" types of `IterableIterator` and `AsyncIterableIterator` are defined by the
-                // iteration types of their `next`, `return`, and `throw` methods. While we define these as `any`
-                // and `undefined` in our libs by default, a custom lib *could* use different definitions.
-                var globalIterationTypes = getIterationTypesOfIteratorCached(globalType, resolver) ||
-                    getIterationTypesOfIteratorSlow(globalType, resolver, /*errorNode*/ undefined);
-                var _a = globalIterationTypes === noIterationTypes ? defaultIterationTypes : globalIterationTypes, returnType = _a.returnType, nextType = _a.nextType;
-                return setCachedIterationTypes(type, resolver.iteratorCacheKey, createIterationTypes(yieldType, returnType, nextType));
+        function getUniqueTypeParameters(context, typeParameters) {
+            var result = [];
+            var oldTypeParameters;
+            var newTypeParameters;
+            for (var _i = 0, typeParameters_2 = typeParameters; _i < typeParameters_2.length; _i++) {
+                var tp = typeParameters_2[_i];
+                var name = tp.symbol.escapedName;
+                if (hasTypeParameterByName(context.inferredTypeParameters, name) || hasTypeParameterByName(result, name)) {
+                    var newName = getUniqueTypeParameterName(ts.concatenate(context.inferredTypeParameters, result), name);
+                    var symbol = createSymbol(262144 /* TypeParameter */, newName);
+                    var newTypeParameter = createTypeParameter(symbol);
+                    newTypeParameter.target = tp;
+                    oldTypeParameters = ts.append(oldTypeParameters, tp);
+                    newTypeParameters = ts.append(newTypeParameters, newTypeParameter);
+                    result.push(newTypeParameter);
+                }
+                else {
+                    result.push(tp);
+                }
             }
-            if (isReferenceToType(type, resolver.getGlobalIteratorType(/*reportErrors*/ false)) ||
-                isReferenceToType(type, resolver.getGlobalGeneratorType(/*reportErrors*/ false))) {
-                var _b = getTypeArguments(type), yieldType = _b[0], returnType = _b[1], nextType = _b[2];
-                return setCachedIterationTypes(type, resolver.iteratorCacheKey, createIterationTypes(yieldType, returnType, nextType));
+            if (newTypeParameters) {
+                var mapper = createTypeMapper(oldTypeParameters, newTypeParameters);
+                for (var _a = 0, newTypeParameters_1 = newTypeParameters; _a < newTypeParameters_1.length; _a++) {
+                    var tp = newTypeParameters_1[_a];
+                    tp.mapper = mapper;
+                }
             }
+            return result;
         }
-        function isIteratorResult(type, kind) {
-            // From https://tc39.github.io/ecma262/#sec-iteratorresult-interface:
-            // > [done] is the result status of an iterator `next` method call. If the end of the iterator was reached `done` is `true`.
-            // > If the end was not reached `done` is `false` and a value is available.
-            // > If a `done` property (either own or inherited) does not exist, it is consider to have the value `false`.
-            var doneType = getTypeOfPropertyOfType(type, "done") || falseType;
-            return isTypeAssignableTo(kind === 0 /* Yield */ ? falseType : trueType, doneType);
+        function hasTypeParameterByName(typeParameters, name) {
+            return ts.some(typeParameters, function (tp) { return tp.symbol.escapedName === name; });
         }
-        function isYieldIteratorResult(type) {
-            return isIteratorResult(type, 0 /* Yield */);
+        function getUniqueTypeParameterName(typeParameters, baseName) {
+            var len = baseName.length;
+            while (len > 1 && baseName.charCodeAt(len - 1) >= 48 /* _0 */ && baseName.charCodeAt(len - 1) <= 57 /* _9 */)
+                len--;
+            var s = baseName.slice(0, len);
+            for (var index = 1; true; index++) {
+                var augmentedName = (s + index);
+                if (!hasTypeParameterByName(typeParameters, augmentedName)) {
+                    return augmentedName;
+                }
+            }
         }
-        function isReturnIteratorResult(type) {
-            return isIteratorResult(type, 1 /* Return */);
+        function getReturnTypeOfSingleNonGenericCallSignature(funcType) {
+            var signature = getSingleCallSignature(funcType);
+            if (signature && !signature.typeParameters) {
+                return getReturnTypeOfSignature(signature);
+            }
+        }
+        function getReturnTypeOfSingleNonGenericSignatureOfCallChain(expr) {
+            var funcType = checkExpression(expr.expression);
+            var nonOptionalType = getOptionalExpressionType(funcType, expr.expression);
+            var returnType = getReturnTypeOfSingleNonGenericCallSignature(funcType);
+            return returnType && propagateOptionalTypeMarker(returnType, expr, nonOptionalType !== funcType);
         }
         /**
-         * Gets the *yield* and *return* types of an `IteratorResult`-like type.
-         *
-         * If we are unable to determine a *yield* or a *return* type, `noIterationTypes` is
-         * returned to indicate to the caller that it should handle the error. Otherwise, an
-         * `IterationTypes` record is returned.
+         * Returns the type of an expression. Unlike checkExpression, this function is simply concerned
+         * with computing the type and may not fully check all contained sub-expressions for errors.
          */
-        function getIterationTypesOfIteratorResult(type) {
-            if (isTypeAny(type)) {
-                return anyIterationTypes;
+        function getTypeOfExpression(node) {
+            // Don't bother caching types that require no flow analysis and are quick to compute.
+            var quickType = getQuickTypeOfExpression(node);
+            if (quickType) {
+                return quickType;
             }
-            var cachedTypes = getCachedIterationTypes(type, "iterationTypesOfIteratorResult");
-            if (cachedTypes) {
-                return cachedTypes;
+            // If a type has been cached for the node, return it.
+            if (node.flags & 67108864 /* TypeCached */ && flowTypeCache) {
+                var cachedType = flowTypeCache[getNodeId(node)];
+                if (cachedType) {
+                    return cachedType;
+                }
             }
-            // As an optimization, if the type is an instantiation of one of the global `IteratorYieldResult<T>`
-            // or `IteratorReturnResult<TReturn>` types, then just grab its type argument.
-            if (isReferenceToType(type, getGlobalIteratorYieldResultType(/*reportErrors*/ false))) {
-                var yieldType_1 = getTypeArguments(type)[0];
-                return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", createIterationTypes(yieldType_1, /*returnType*/ undefined, /*nextType*/ undefined));
+            var startInvocationCount = flowInvocationCount;
+            var type = checkExpression(node);
+            // If control flow analysis was required to determine the type, it is worth caching.
+            if (flowInvocationCount !== startInvocationCount) {
+                var cache = flowTypeCache || (flowTypeCache = []);
+                cache[getNodeId(node)] = type;
+                ts.setNodeFlags(node, node.flags | 67108864 /* TypeCached */);
             }
-            if (isReferenceToType(type, getGlobalIteratorReturnResultType(/*reportErrors*/ false))) {
-                var returnType_1 = getTypeArguments(type)[0];
-                return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", createIterationTypes(/*yieldType*/ undefined, returnType_1, /*nextType*/ undefined));
+            return type;
+        }
+        function getQuickTypeOfExpression(node) {
+            var expr = ts.skipParentheses(node);
+            // Optimize for the common case of a call to a function with a single non-generic call
+            // signature where we can just fetch the return type without checking the arguments.
+            if (ts.isCallExpression(expr) && expr.expression.kind !== 105 /* SuperKeyword */ && !ts.isRequireCall(expr, /*checkArgumentIsStringLiteralLike*/ true) && !isSymbolOrSymbolForCall(expr)) {
+                var type = ts.isCallChain(expr) ? getReturnTypeOfSingleNonGenericSignatureOfCallChain(expr) :
+                    getReturnTypeOfSingleNonGenericCallSignature(checkNonNullExpression(expr.expression));
+                if (type) {
+                    return type;
+                }
             }
-            // Choose any constituents that can produce the requested iteration type.
-            var yieldIteratorResult = filterType(type, isYieldIteratorResult);
-            var yieldType = yieldIteratorResult !== neverType ? getTypeOfPropertyOfType(yieldIteratorResult, "value") : undefined;
-            var returnIteratorResult = filterType(type, isReturnIteratorResult);
-            var returnType = returnIteratorResult !== neverType ? getTypeOfPropertyOfType(returnIteratorResult, "value") : undefined;
-            if (!yieldType && !returnType) {
-                return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", noIterationTypes);
+            else if (ts.isAssertionExpression(expr) && !ts.isConstTypeReference(expr.type)) {
+                return getTypeFromTypeNode(expr.type);
             }
-            // From https://tc39.github.io/ecma262/#sec-iteratorresult-interface
-            // > ... If the iterator does not have a return value, `value` is `undefined`. In that case, the
-            // > `value` property may be absent from the conforming object if it does not inherit an explicit
-            // > `value` property.
-            return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", createIterationTypes(yieldType, returnType || voidType, /*nextType*/ undefined));
+            else if (node.kind === 8 /* NumericLiteral */ || node.kind === 10 /* StringLiteral */ ||
+                node.kind === 109 /* TrueKeyword */ || node.kind === 94 /* FalseKeyword */) {
+                return checkExpression(node);
+            }
+            return undefined;
         }
         /**
-         * Gets the *yield*, *return*, and *next* types of a the `next()`, `return()`, or
-         * `throw()` method of an `Iterator`-like or `AsyncIterator`-like type.
-         *
-         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
-         * record is returned. Otherwise, we return `undefined`.
+         * Returns the type of an expression. Unlike checkExpression, this function is simply concerned
+         * with computing the type and may not fully check all contained sub-expressions for errors.
+         * It is intended for uses where you know there is no contextual type,
+         * and requesting the contextual type might cause a circularity or other bad behaviour.
+         * It sets the contextual type of the node to any before calling getTypeOfExpression.
          */
-        function getIterationTypesOfMethod(type, resolver, methodName, errorNode) {
-            var method = getPropertyOfType(type, methodName);
-            // Ignore 'return' or 'throw' if they are missing.
-            if (!method && methodName !== "next") {
-                return undefined;
+        function getContextFreeTypeOfExpression(node) {
+            var links = getNodeLinks(node);
+            if (links.contextFreeType) {
+                return links.contextFreeType;
             }
-            var methodType = method && !(methodName === "next" && (method.flags & 16777216 /* Optional */))
-                ? methodName === "next" ? getTypeOfSymbol(method) : getTypeWithFacts(getTypeOfSymbol(method), 2097152 /* NEUndefinedOrNull */)
-                : undefined;
-            if (isTypeAny(methodType)) {
-                // `return()` and `throw()` don't provide a *next* type.
-                return methodName === "next" ? anyIterationTypes : anyIterationTypesExceptNext;
+            var saveContextualType = node.contextualType;
+            node.contextualType = anyType;
+            try {
+                var type = links.contextFreeType = checkExpression(node, 4 /* SkipContextSensitive */);
+                return type;
             }
-            // Both async and non-async iterators *must* have a `next` method.
-            var methodSignatures = methodType ? getSignaturesOfType(methodType, 0 /* Call */) : ts.emptyArray;
-            if (methodSignatures.length === 0) {
-                if (errorNode) {
-                    var diagnostic = methodName === "next"
-                        ? resolver.mustHaveANextMethodDiagnostic
-                        : resolver.mustBeAMethodDiagnostic;
-                    error(errorNode, diagnostic, methodName);
-                }
-                return methodName === "next" ? anyIterationTypes : undefined;
+            finally {
+                // In the event our operation is canceled or some other exception occurs, reset the contextual type
+                // so that we do not accidentally hold onto an instance of the checker, as a Type created in the services layer
+                // may hold onto the checker that created it.
+                node.contextualType = saveContextualType;
             }
-            // Extract the first parameter and return type of each signature.
-            var methodParameterTypes;
-            var methodReturnTypes;
-            for (var _i = 0, methodSignatures_1 = methodSignatures; _i < methodSignatures_1.length; _i++) {
-                var signature = methodSignatures_1[_i];
-                if (methodName !== "throw" && ts.some(signature.parameters)) {
-                    methodParameterTypes = ts.append(methodParameterTypes, getTypeAtPosition(signature, 0));
-                }
-                methodReturnTypes = ts.append(methodReturnTypes, getReturnTypeOfSignature(signature));
+        }
+        function checkExpression(node, checkMode, forceTuple) {
+            ts.tracing.push("check" /* Check */, "checkExpression", { kind: node.kind, pos: node.pos, end: node.end });
+            var saveCurrentNode = currentNode;
+            currentNode = node;
+            instantiationCount = 0;
+            var uninstantiatedType = checkExpressionWorker(node, checkMode, forceTuple);
+            var type = instantiateTypeWithSingleGenericCallSignature(node, uninstantiatedType, checkMode);
+            if (isConstEnumObjectType(type)) {
+                checkConstEnumAccess(node, type);
             }
-            // Resolve the *next* or *return* type from the first parameter of a `next()` or
-            // `return()` method, respectively.
-            var returnTypes;
-            var nextType;
-            if (methodName !== "throw") {
-                var methodParameterType = methodParameterTypes ? getUnionType(methodParameterTypes) : unknownType;
-                if (methodName === "next") {
-                    // The value of `next(value)` is *not* awaited by async generators
-                    nextType = methodParameterType;
-                }
-                else if (methodName === "return") {
-                    // The value of `return(value)` *is* awaited by async generators
-                    var resolvedMethodParameterType = resolver.resolveIterationType(methodParameterType, errorNode) || anyType;
-                    returnTypes = ts.append(returnTypes, resolvedMethodParameterType);
-                }
+            currentNode = saveCurrentNode;
+            ts.tracing.pop();
+            return type;
+        }
+        function checkConstEnumAccess(node, type) {
+            // enum object type for const enums are only permitted in:
+            // - 'left' in property access
+            // - 'object' in indexed access
+            // - target in rhs of import statement
+            var ok = (node.parent.kind === 201 /* PropertyAccessExpression */ && node.parent.expression === node) ||
+                (node.parent.kind === 202 /* ElementAccessExpression */ && node.parent.expression === node) ||
+                ((node.kind === 78 /* Identifier */ || node.kind === 157 /* QualifiedName */) && isInRightSideOfImportOrExportAssignment(node) ||
+                    (node.parent.kind === 176 /* TypeQuery */ && node.parent.exprName === node)) ||
+                (node.parent.kind === 270 /* ExportSpecifier */); // We allow reexporting const enums
+            if (!ok) {
+                error(node, ts.Diagnostics.const_enums_can_only_be_used_in_property_or_index_access_expressions_or_the_right_hand_side_of_an_import_declaration_or_export_assignment_or_type_query);
             }
-            // Resolve the *yield* and *return* types from the return type of the method (i.e. `IteratorResult`)
-            var yieldType;
-            var methodReturnType = methodReturnTypes ? getUnionType(methodReturnTypes, 2 /* Subtype */) : neverType;
-            var resolvedMethodReturnType = resolver.resolveIterationType(methodReturnType, errorNode) || anyType;
-            var iterationTypes = getIterationTypesOfIteratorResult(resolvedMethodReturnType);
-            if (iterationTypes === noIterationTypes) {
-                if (errorNode) {
-                    error(errorNode, resolver.mustHaveAValueDiagnostic, methodName);
+            if (compilerOptions.isolatedModules) {
+                ts.Debug.assert(!!(type.symbol.flags & 128 /* ConstEnum */));
+                var constEnumDeclaration = type.symbol.valueDeclaration;
+                if (constEnumDeclaration.flags & 8388608 /* Ambient */) {
+                    error(node, ts.Diagnostics.Cannot_access_ambient_const_enums_when_the_isolatedModules_flag_is_provided);
                 }
-                yieldType = anyType;
-                returnTypes = ts.append(returnTypes, anyType);
-            }
-            else {
-                yieldType = iterationTypes.yieldType;
-                returnTypes = ts.append(returnTypes, iterationTypes.returnType);
             }
-            return createIterationTypes(yieldType, getUnionType(returnTypes), nextType);
-        }
-        /**
-         * Gets the *yield*, *return*, and *next* types of an `Iterator`-like or `AsyncIterator`-like
-         * type from its members.
-         *
-         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
-         * record is returned. Otherwise, `noIterationTypes` is returned.
-         *
-         * NOTE: You probably don't want to call this directly and should be calling
-         * `getIterationTypesOfIterator` instead.
-         */
-        function getIterationTypesOfIteratorSlow(type, resolver, errorNode) {
-            var iterationTypes = combineIterationTypes([
-                getIterationTypesOfMethod(type, resolver, "next", errorNode),
-                getIterationTypesOfMethod(type, resolver, "return", errorNode),
-                getIterationTypesOfMethod(type, resolver, "throw", errorNode),
-            ]);
-            return setCachedIterationTypes(type, resolver.iteratorCacheKey, iterationTypes);
         }
-        /**
-         * Gets the requested "iteration type" from a type that is either `Iterable`-like, `Iterator`-like,
-         * `IterableIterator`-like, or `Generator`-like (for a non-async generator); or `AsyncIterable`-like,
-         * `AsyncIterator`-like, `AsyncIterableIterator`-like, or `AsyncGenerator`-like (for an async generator).
-         */
-        function getIterationTypeOfGeneratorFunctionReturnType(kind, returnType, isAsyncGenerator) {
-            if (isTypeAny(returnType)) {
-                return undefined;
+        function checkParenthesizedExpression(node, checkMode) {
+            var tag = ts.isInJSFile(node) ? ts.getJSDocTypeTag(node) : undefined;
+            if (tag) {
+                return checkAssertionWorker(tag, tag.typeExpression.type, node.expression, checkMode);
             }
-            var iterationTypes = getIterationTypesOfGeneratorFunctionReturnType(returnType, isAsyncGenerator);
-            return iterationTypes && iterationTypes[getIterationTypesKeyFromIterationTypeKind(kind)];
+            return checkExpression(node.expression, checkMode);
         }
-        function getIterationTypesOfGeneratorFunctionReturnType(type, isAsyncGenerator) {
-            if (isTypeAny(type)) {
-                return anyIterationTypes;
+        function checkExpressionWorker(node, checkMode, forceTuple) {
+            var kind = node.kind;
+            if (cancellationToken) {
+                // Only bother checking on a few construct kinds.  We don't want to be excessively
+                // hitting the cancellation token on every node we check.
+                switch (kind) {
+                    case 221 /* ClassExpression */:
+                    case 208 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
+                        cancellationToken.throwIfCancellationRequested();
+                }
             }
-            var use = isAsyncGenerator ? 2 /* AsyncGeneratorReturnType */ : 1 /* GeneratorReturnType */;
-            var resolver = isAsyncGenerator ? asyncIterationTypesResolver : syncIterationTypesResolver;
-            return getIterationTypesOfIterable(type, use, /*errorNode*/ undefined) ||
-                getIterationTypesOfIterator(type, resolver, /*errorNode*/ undefined);
-        }
-        function checkBreakOrContinueStatement(node) {
-            // Grammar checking
-            if (!checkGrammarStatementInAmbientContext(node))
-                checkGrammarBreakOrContinueStatement(node);
-            // TODO: Check that target label is valid
-        }
-        function unwrapReturnType(returnType, functionFlags) {
-            var _a, _b;
-            var isGenerator = !!(functionFlags & 1 /* Generator */);
-            var isAsync = !!(functionFlags & 2 /* Async */);
-            return isGenerator ? (_a = getIterationTypeOfGeneratorFunctionReturnType(1 /* Return */, returnType, isAsync)) !== null && _a !== void 0 ? _a : errorType :
-                isAsync ? (_b = getAwaitedType(returnType)) !== null && _b !== void 0 ? _b : errorType :
-                    returnType;
+            switch (kind) {
+                case 78 /* Identifier */:
+                    return checkIdentifier(node);
+                case 107 /* ThisKeyword */:
+                    return checkThisExpression(node);
+                case 105 /* SuperKeyword */:
+                    return checkSuperExpression(node);
+                case 103 /* NullKeyword */:
+                    return nullWideningType;
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                case 10 /* StringLiteral */:
+                    return getFreshTypeOfLiteralType(getLiteralType(node.text));
+                case 8 /* NumericLiteral */:
+                    checkGrammarNumericLiteral(node);
+                    return getFreshTypeOfLiteralType(getLiteralType(+node.text));
+                case 9 /* BigIntLiteral */:
+                    checkGrammarBigIntLiteral(node);
+                    return getFreshTypeOfLiteralType(getBigIntLiteralType(node));
+                case 109 /* TrueKeyword */:
+                    return trueType;
+                case 94 /* FalseKeyword */:
+                    return falseType;
+                case 218 /* TemplateExpression */:
+                    return checkTemplateExpression(node);
+                case 13 /* RegularExpressionLiteral */:
+                    return globalRegExpType;
+                case 199 /* ArrayLiteralExpression */:
+                    return checkArrayLiteral(node, checkMode, forceTuple);
+                case 200 /* ObjectLiteralExpression */:
+                    return checkObjectLiteral(node, checkMode);
+                case 201 /* PropertyAccessExpression */:
+                    return checkPropertyAccessExpression(node);
+                case 157 /* QualifiedName */:
+                    return checkQualifiedName(node);
+                case 202 /* ElementAccessExpression */:
+                    return checkIndexedAccess(node);
+                case 203 /* CallExpression */:
+                    if (node.expression.kind === 99 /* ImportKeyword */) {
+                        return checkImportCallExpression(node);
+                    }
+                // falls through
+                case 204 /* NewExpression */:
+                    return checkCallExpression(node, checkMode);
+                case 205 /* TaggedTemplateExpression */:
+                    return checkTaggedTemplateExpression(node);
+                case 207 /* ParenthesizedExpression */:
+                    return checkParenthesizedExpression(node, checkMode);
+                case 221 /* ClassExpression */:
+                    return checkClassExpression(node);
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                    return checkFunctionExpressionOrObjectLiteralMethod(node, checkMode);
+                case 211 /* TypeOfExpression */:
+                    return checkTypeOfExpression(node);
+                case 206 /* TypeAssertionExpression */:
+                case 224 /* AsExpression */:
+                    return checkAssertion(node);
+                case 225 /* NonNullExpression */:
+                    return checkNonNullAssertion(node);
+                case 226 /* MetaProperty */:
+                    return checkMetaProperty(node);
+                case 210 /* DeleteExpression */:
+                    return checkDeleteExpression(node);
+                case 212 /* VoidExpression */:
+                    return checkVoidExpression(node);
+                case 213 /* AwaitExpression */:
+                    return checkAwaitExpression(node);
+                case 214 /* PrefixUnaryExpression */:
+                    return checkPrefixUnaryExpression(node);
+                case 215 /* PostfixUnaryExpression */:
+                    return checkPostfixUnaryExpression(node);
+                case 216 /* BinaryExpression */:
+                    return checkBinaryExpression(node, checkMode);
+                case 217 /* ConditionalExpression */:
+                    return checkConditionalExpression(node, checkMode);
+                case 220 /* SpreadElement */:
+                    return checkSpreadExpression(node, checkMode);
+                case 222 /* OmittedExpression */:
+                    return undefinedWideningType;
+                case 219 /* YieldExpression */:
+                    return checkYieldExpression(node);
+                case 227 /* SyntheticExpression */:
+                    return checkSyntheticExpression(node);
+                case 283 /* JsxExpression */:
+                    return checkJsxExpression(node, checkMode);
+                case 273 /* JsxElement */:
+                    return checkJsxElement(node, checkMode);
+                case 274 /* JsxSelfClosingElement */:
+                    return checkJsxSelfClosingElement(node, checkMode);
+                case 277 /* JsxFragment */:
+                    return checkJsxFragment(node);
+                case 281 /* JsxAttributes */:
+                    return checkJsxAttributes(node, checkMode);
+                case 275 /* JsxOpeningElement */:
+                    ts.Debug.fail("Shouldn't ever directly check a JsxOpeningElement");
+            }
+            return errorType;
         }
-        function isUnwrappedReturnTypeVoidOrAny(func, returnType) {
-            var unwrappedReturnType = unwrapReturnType(returnType, ts.getFunctionFlags(func));
-            return !!unwrappedReturnType && maybeTypeOfKind(unwrappedReturnType, 16384 /* Void */ | 3 /* AnyOrUnknown */);
+        // DECLARATION AND STATEMENT TYPE CHECKING
+        function checkTypeParameter(node) {
+            // Grammar Checking
+            if (node.expression) {
+                grammarErrorOnFirstToken(node.expression, ts.Diagnostics.Type_expected);
+            }
+            checkSourceElement(node.constraint);
+            checkSourceElement(node.default);
+            var typeParameter = getDeclaredTypeOfTypeParameter(getSymbolOfNode(node));
+            // Resolve base constraint to reveal circularity errors
+            getBaseConstraintOfType(typeParameter);
+            if (!hasNonCircularTypeParameterDefault(typeParameter)) {
+                error(node.default, ts.Diagnostics.Type_parameter_0_has_a_circular_default, typeToString(typeParameter));
+            }
+            var constraintType = getConstraintOfTypeParameter(typeParameter);
+            var defaultType = getDefaultFromTypeParameter(typeParameter);
+            if (constraintType && defaultType) {
+                checkTypeAssignableTo(defaultType, getTypeWithThisArgument(instantiateType(constraintType, makeUnaryTypeMapper(typeParameter, defaultType)), defaultType), node.default, ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1);
+            }
+            if (produceDiagnostics) {
+                checkTypeNameIsReserved(node.name, ts.Diagnostics.Type_parameter_name_cannot_be_0);
+            }
         }
-        function checkReturnStatement(node) {
-            var _a;
+        function checkParameter(node) {
             // Grammar checking
-            if (checkGrammarStatementInAmbientContext(node)) {
-                return;
-            }
+            // It is a SyntaxError if the Identifier "eval" or the Identifier "arguments" occurs as the
+            // Identifier in a PropertySetParameterList of a PropertyAssignment that is contained in strict code
+            // or if its FunctionBody is strict code(11.1.5).
+            checkGrammarDecoratorsAndModifiers(node);
+            checkVariableLikeDeclaration(node);
             var func = ts.getContainingFunction(node);
-            if (!func) {
-                grammarErrorOnFirstToken(node, ts.Diagnostics.A_return_statement_can_only_be_used_within_a_function_body);
-                return;
+            if (ts.hasSyntacticModifier(node, 92 /* ParameterPropertyModifier */)) {
+                if (!(func.kind === 166 /* Constructor */ && ts.nodeIsPresent(func.body))) {
+                    error(node, ts.Diagnostics.A_parameter_property_is_only_allowed_in_a_constructor_implementation);
+                }
+                if (func.kind === 166 /* Constructor */ && ts.isIdentifier(node.name) && node.name.escapedText === "constructor") {
+                    error(node.name, ts.Diagnostics.constructor_cannot_be_used_as_a_parameter_property_name);
+                }
             }
-            var signature = getSignatureFromDeclaration(func);
-            var returnType = getReturnTypeOfSignature(signature);
-            var functionFlags = ts.getFunctionFlags(func);
-            if (strictNullChecks || node.expression || returnType.flags & 131072 /* Never */) {
-                var exprType = node.expression ? checkExpressionCached(node.expression) : undefinedType;
-                if (func.kind === 164 /* SetAccessor */) {
-                    if (node.expression) {
-                        error(node, ts.Diagnostics.Setters_cannot_return_a_value);
-                    }
+            if (node.questionToken && ts.isBindingPattern(node.name) && func.body) {
+                error(node, ts.Diagnostics.A_binding_pattern_parameter_cannot_be_optional_in_an_implementation_signature);
+            }
+            if (node.name && ts.isIdentifier(node.name) && (node.name.escapedText === "this" || node.name.escapedText === "new")) {
+                if (func.parameters.indexOf(node) !== 0) {
+                    error(node, ts.Diagnostics.A_0_parameter_must_be_the_first_parameter, node.name.escapedText);
                 }
-                else if (func.kind === 162 /* Constructor */) {
-                    if (node.expression && !checkTypeAssignableToAndOptionallyElaborate(exprType, returnType, node, node.expression)) {
-                        error(node, ts.Diagnostics.Return_type_of_constructor_signature_must_be_assignable_to_the_instance_type_of_the_class);
-                    }
+                if (func.kind === 166 /* Constructor */ || func.kind === 170 /* ConstructSignature */ || func.kind === 175 /* ConstructorType */) {
+                    error(node, ts.Diagnostics.A_constructor_cannot_have_a_this_parameter);
                 }
-                else if (getReturnTypeFromAnnotation(func)) {
-                    var unwrappedReturnType = (_a = unwrapReturnType(returnType, functionFlags)) !== null && _a !== void 0 ? _a : returnType;
-                    var unwrappedExprType = functionFlags & 2 /* Async */
-                        ? checkAwaitedType(exprType, node, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member)
-                        : exprType;
-                    if (unwrappedReturnType) {
-                        // If the function has a return type, but promisedType is
-                        // undefined, an error will be reported in checkAsyncFunctionReturnType
-                        // so we don't need to report one here.
-                        checkTypeAssignableToAndOptionallyElaborate(unwrappedExprType, unwrappedReturnType, node, node.expression);
-                    }
+                if (func.kind === 209 /* ArrowFunction */) {
+                    error(node, ts.Diagnostics.An_arrow_function_cannot_have_a_this_parameter);
+                }
+                if (func.kind === 167 /* GetAccessor */ || func.kind === 168 /* SetAccessor */) {
+                    error(node, ts.Diagnostics.get_and_set_accessors_cannot_declare_this_parameters);
                 }
             }
-            else if (func.kind !== 162 /* Constructor */ && compilerOptions.noImplicitReturns && !isUnwrappedReturnTypeVoidOrAny(func, returnType)) {
-                // The function has a return type, but the return statement doesn't have an expression.
-                error(node, ts.Diagnostics.Not_all_code_paths_return_a_value);
+            // Only check rest parameter type if it's not a binding pattern. Since binding patterns are
+            // not allowed in a rest parameter, we already have an error from checkGrammarParameterList.
+            if (node.dotDotDotToken && !ts.isBindingPattern(node.name) && !isTypeAssignableTo(getReducedType(getTypeOfSymbol(node.symbol)), anyReadonlyArrayType)) {
+                error(node, ts.Diagnostics.A_rest_parameter_must_be_of_an_array_type);
             }
         }
-        function checkWithStatement(node) {
-            // Grammar checking for withStatement
-            if (!checkGrammarStatementInAmbientContext(node)) {
-                if (node.flags & 32768 /* AwaitContext */) {
-                    grammarErrorOnFirstToken(node, ts.Diagnostics.with_statements_are_not_allowed_in_an_async_function_block);
-                }
+        function checkTypePredicate(node) {
+            var parent = getTypePredicateParent(node);
+            if (!parent) {
+                // The parent must not be valid.
+                error(node, ts.Diagnostics.A_type_predicate_is_only_allowed_in_return_type_position_for_functions_and_methods);
+                return;
             }
-            checkExpression(node.expression);
-            var sourceFile = ts.getSourceFileOfNode(node);
-            if (!hasParseDiagnostics(sourceFile)) {
-                var start = ts.getSpanOfTokenAtPosition(sourceFile, node.pos).start;
-                var end = node.statement.pos;
-                grammarErrorAtPos(sourceFile, start, end - start, ts.Diagnostics.The_with_statement_is_not_supported_All_symbols_in_a_with_block_will_have_type_any);
+            var signature = getSignatureFromDeclaration(parent);
+            var typePredicate = getTypePredicateOfSignature(signature);
+            if (!typePredicate) {
+                return;
             }
-        }
-        function checkSwitchStatement(node) {
-            // Grammar checking
-            checkGrammarStatementInAmbientContext(node);
-            var firstDefaultClause;
-            var hasDuplicateDefaultClause = false;
-            var expressionType = checkExpression(node.expression);
-            var expressionIsLiteral = isLiteralType(expressionType);
-            ts.forEach(node.caseBlock.clauses, function (clause) {
-                // Grammar check for duplicate default clauses, skip if we already report duplicate default clause
-                if (clause.kind === 278 /* DefaultClause */ && !hasDuplicateDefaultClause) {
-                    if (firstDefaultClause === undefined) {
-                        firstDefaultClause = clause;
+            checkSourceElement(node.type);
+            var parameterName = node.parameterName;
+            if (typePredicate.kind === 0 /* This */ || typePredicate.kind === 2 /* AssertsThis */) {
+                getTypeFromThisTypeNode(parameterName);
+            }
+            else {
+                if (typePredicate.parameterIndex >= 0) {
+                    if (signatureHasRestParameter(signature) && typePredicate.parameterIndex === signature.parameters.length - 1) {
+                        error(parameterName, ts.Diagnostics.A_type_predicate_cannot_reference_a_rest_parameter);
                     }
                     else {
-                        grammarErrorOnNode(clause, ts.Diagnostics.A_default_clause_cannot_appear_more_than_once_in_a_switch_statement);
-                        hasDuplicateDefaultClause = true;
+                        if (typePredicate.type) {
+                            var leadingError = function () { return ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.A_type_predicate_s_type_must_be_assignable_to_its_parameter_s_type); };
+                            checkTypeAssignableTo(typePredicate.type, getTypeOfSymbol(signature.parameters[typePredicate.parameterIndex]), node.type, 
+                            /*headMessage*/ undefined, leadingError);
+                        }
                     }
                 }
-                if (produceDiagnostics && clause.kind === 277 /* CaseClause */) {
-                    // TypeScript 1.0 spec (April 2014): 5.9
-                    // In a 'switch' statement, each 'case' expression must be of a type that is comparable
-                    // to or from the type of the 'switch' expression.
-                    var caseType = checkExpression(clause.expression);
-                    var caseIsLiteral = isLiteralType(caseType);
-                    var comparedExpressionType = expressionType;
-                    if (!caseIsLiteral || !expressionIsLiteral) {
-                        caseType = caseIsLiteral ? getBaseTypeOfLiteralType(caseType) : caseType;
-                        comparedExpressionType = getBaseTypeOfLiteralType(expressionType);
+                else if (parameterName) {
+                    var hasReportedError = false;
+                    for (var _i = 0, _a = parent.parameters; _i < _a.length; _i++) {
+                        var name = _a[_i].name;
+                        if (ts.isBindingPattern(name) &&
+                            checkIfTypePredicateVariableIsDeclaredInBindingPattern(name, parameterName, typePredicate.parameterName)) {
+                            hasReportedError = true;
+                            break;
+                        }
                     }
-                    if (!isTypeEqualityComparableTo(comparedExpressionType, caseType)) {
-                        // expressionType is not comparable to caseType, try the reversed check and report errors if it fails
-                        checkTypeComparableTo(caseType, comparedExpressionType, clause.expression, /*headMessage*/ undefined);
+                    if (!hasReportedError) {
+                        error(node.parameterName, ts.Diagnostics.Cannot_find_parameter_0, typePredicate.parameterName);
                     }
                 }
-                ts.forEach(clause.statements, checkSourceElement);
-                if (compilerOptions.noFallthroughCasesInSwitch && clause.fallthroughFlowNode && isReachableFlowNode(clause.fallthroughFlowNode)) {
-                    error(clause, ts.Diagnostics.Fallthrough_case_in_switch);
-                }
-            });
-            if (node.caseBlock.locals) {
-                registerForUnusedIdentifiersCheck(node.caseBlock);
             }
         }
-        function checkLabeledStatement(node) {
-            // Grammar checking
-            if (!checkGrammarStatementInAmbientContext(node)) {
-                ts.findAncestor(node.parent, function (current) {
-                    if (ts.isFunctionLike(current)) {
-                        return "quit";
+        function getTypePredicateParent(node) {
+            switch (node.parent.kind) {
+                case 209 /* ArrowFunction */:
+                case 169 /* CallSignature */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 174 /* FunctionType */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    var parent = node.parent;
+                    if (node === parent.type) {
+                        return parent;
                     }
-                    if (current.kind === 238 /* LabeledStatement */ && current.label.escapedText === node.label.escapedText) {
-                        grammarErrorOnNode(node.label, ts.Diagnostics.Duplicate_label_0, ts.getTextOfNode(node.label));
+            }
+        }
+        function checkIfTypePredicateVariableIsDeclaredInBindingPattern(pattern, predicateVariableNode, predicateVariableName) {
+            for (var _i = 0, _a = pattern.elements; _i < _a.length; _i++) {
+                var element = _a[_i];
+                if (ts.isOmittedExpression(element)) {
+                    continue;
+                }
+                var name = element.name;
+                if (name.kind === 78 /* Identifier */ && name.escapedText === predicateVariableName) {
+                    error(predicateVariableNode, ts.Diagnostics.A_type_predicate_cannot_reference_element_0_in_a_binding_pattern, predicateVariableName);
+                    return true;
+                }
+                else if (name.kind === 197 /* ArrayBindingPattern */ || name.kind === 196 /* ObjectBindingPattern */) {
+                    if (checkIfTypePredicateVariableIsDeclaredInBindingPattern(name, predicateVariableNode, predicateVariableName)) {
                         return true;
                     }
-                    return false;
-                });
+                }
             }
-            // ensure that label is unique
-            checkSourceElement(node.statement);
         }
-        function checkThrowStatement(node) {
+        function checkSignatureDeclaration(node) {
             // Grammar checking
-            if (!checkGrammarStatementInAmbientContext(node)) {
-                if (node.expression === undefined) {
-                    grammarErrorAfterFirstToken(node, ts.Diagnostics.Line_break_not_permitted_here);
+            if (node.kind === 171 /* IndexSignature */) {
+                checkGrammarIndexSignature(node);
+            }
+            // TODO (yuisu): Remove this check in else-if when SyntaxKind.Construct is moved and ambient context is handled
+            else if (node.kind === 174 /* FunctionType */ || node.kind === 251 /* FunctionDeclaration */ || node.kind === 175 /* ConstructorType */ ||
+                node.kind === 169 /* CallSignature */ || node.kind === 166 /* Constructor */ ||
+                node.kind === 170 /* ConstructSignature */) {
+                checkGrammarFunctionLikeDeclaration(node);
+            }
+            var functionFlags = ts.getFunctionFlags(node);
+            if (!(functionFlags & 4 /* Invalid */)) {
+                // Async generators prior to ESNext require the __await and __asyncGenerator helpers
+                if ((functionFlags & 3 /* AsyncGenerator */) === 3 /* AsyncGenerator */ && languageVersion < 99 /* ESNext */) {
+                    checkExternalEmitHelpers(node, 12288 /* AsyncGeneratorIncludes */);
+                }
+                // Async functions prior to ES2017 require the __awaiter helper
+                if ((functionFlags & 3 /* AsyncGenerator */) === 2 /* Async */ && languageVersion < 4 /* ES2017 */) {
+                    checkExternalEmitHelpers(node, 64 /* Awaiter */);
+                }
+                // Generator functions, Async functions, and Async Generator functions prior to
+                // ES2015 require the __generator helper
+                if ((functionFlags & 3 /* AsyncGenerator */) !== 0 /* Normal */ && languageVersion < 2 /* ES2015 */) {
+                    checkExternalEmitHelpers(node, 128 /* Generator */);
                 }
             }
-            if (node.expression) {
-                checkExpression(node.expression);
+            checkTypeParameters(node.typeParameters);
+            ts.forEach(node.parameters, checkParameter);
+            // TODO(rbuckton): Should we start checking JSDoc types?
+            if (node.type) {
+                checkSourceElement(node.type);
             }
-        }
-        function checkTryStatement(node) {
-            // Grammar checking
-            checkGrammarStatementInAmbientContext(node);
-            checkBlock(node.tryBlock);
-            var catchClause = node.catchClause;
-            if (catchClause) {
-                // Grammar checking
-                if (catchClause.variableDeclaration) {
-                    if (catchClause.variableDeclaration.type) {
-                        grammarErrorOnFirstToken(catchClause.variableDeclaration.type, ts.Diagnostics.Catch_clause_variable_cannot_have_a_type_annotation);
-                    }
-                    else if (catchClause.variableDeclaration.initializer) {
-                        grammarErrorOnFirstToken(catchClause.variableDeclaration.initializer, ts.Diagnostics.Catch_clause_variable_cannot_have_an_initializer);
+            if (produceDiagnostics) {
+                checkCollisionWithArgumentsInGeneratedCode(node);
+                var returnTypeNode = ts.getEffectiveReturnTypeNode(node);
+                if (noImplicitAny && !returnTypeNode) {
+                    switch (node.kind) {
+                        case 170 /* ConstructSignature */:
+                            error(node, ts.Diagnostics.Construct_signature_which_lacks_return_type_annotation_implicitly_has_an_any_return_type);
+                            break;
+                        case 169 /* CallSignature */:
+                            error(node, ts.Diagnostics.Call_signature_which_lacks_return_type_annotation_implicitly_has_an_any_return_type);
+                            break;
                     }
-                    else {
-                        var blockLocals_1 = catchClause.block.locals;
-                        if (blockLocals_1) {
-                            ts.forEachKey(catchClause.locals, function (caughtName) {
-                                var blockLocal = blockLocals_1.get(caughtName);
-                                if (blockLocal && (blockLocal.flags & 2 /* BlockScopedVariable */) !== 0) {
-                                    grammarErrorOnNode(blockLocal.valueDeclaration, ts.Diagnostics.Cannot_redeclare_identifier_0_in_catch_clause, caughtName);
-                                }
-                            });
+                }
+                if (returnTypeNode) {
+                    var functionFlags_1 = ts.getFunctionFlags(node);
+                    if ((functionFlags_1 & (4 /* Invalid */ | 1 /* Generator */)) === 1 /* Generator */) {
+                        var returnType = getTypeFromTypeNode(returnTypeNode);
+                        if (returnType === voidType) {
+                            error(returnTypeNode, ts.Diagnostics.A_generator_cannot_have_a_void_type_annotation);
+                        }
+                        else {
+                            // Naively, one could check that Generator<any, any, any> is assignable to the return type annotation.
+                            // However, that would not catch the error in the following case.
+                            //
+                            //    interface BadGenerator extends Iterable<number>, Iterator<string> { }
+                            //    function* g(): BadGenerator { } // Iterable and Iterator have different types!
+                            //
+                            var generatorYieldType = getIterationTypeOfGeneratorFunctionReturnType(0 /* Yield */, returnType, (functionFlags_1 & 2 /* Async */) !== 0) || anyType;
+                            var generatorReturnType = getIterationTypeOfGeneratorFunctionReturnType(1 /* Return */, returnType, (functionFlags_1 & 2 /* Async */) !== 0) || generatorYieldType;
+                            var generatorNextType = getIterationTypeOfGeneratorFunctionReturnType(2 /* Next */, returnType, (functionFlags_1 & 2 /* Async */) !== 0) || unknownType;
+                            var generatorInstantiation = createGeneratorReturnType(generatorYieldType, generatorReturnType, generatorNextType, !!(functionFlags_1 & 2 /* Async */));
+                            checkTypeAssignableTo(generatorInstantiation, returnType, returnTypeNode);
                         }
                     }
+                    else if ((functionFlags_1 & 3 /* AsyncGenerator */) === 2 /* Async */) {
+                        checkAsyncFunctionReturnType(node, returnTypeNode);
+                    }
+                }
+                if (node.kind !== 171 /* IndexSignature */ && node.kind !== 308 /* JSDocFunctionType */) {
+                    registerForUnusedIdentifiersCheck(node);
                 }
-                checkBlock(catchClause.block);
-            }
-            if (node.finallyBlock) {
-                checkBlock(node.finallyBlock);
             }
         }
-        function checkIndexConstraints(type) {
-            var declaredNumberIndexer = getIndexDeclarationOfSymbol(type.symbol, 1 /* Number */);
-            var declaredStringIndexer = getIndexDeclarationOfSymbol(type.symbol, 0 /* String */);
-            var stringIndexType = getIndexTypeOfType(type, 0 /* String */);
-            var numberIndexType = getIndexTypeOfType(type, 1 /* Number */);
-            if (stringIndexType || numberIndexType) {
-                ts.forEach(getPropertiesOfObjectType(type), function (prop) {
-                    var propType = getTypeOfSymbol(prop);
-                    checkIndexConstraintForProperty(prop, propType, type, declaredStringIndexer, stringIndexType, 0 /* String */);
-                    checkIndexConstraintForProperty(prop, propType, type, declaredNumberIndexer, numberIndexType, 1 /* Number */);
-                });
-                var classDeclaration = type.symbol.valueDeclaration;
-                if (ts.getObjectFlags(type) & 1 /* Class */ && ts.isClassLike(classDeclaration)) {
-                    for (var _i = 0, _a = classDeclaration.members; _i < _a.length; _i++) {
-                        var member = _a[_i];
-                        // Only process instance properties with computed names here.
-                        // Static properties cannot be in conflict with indexers,
-                        // and properties with literal names were already checked.
-                        if (!ts.hasModifier(member, 32 /* Static */) && hasNonBindableDynamicName(member)) {
-                            var symbol = getSymbolOfNode(member);
-                            var propType = getTypeOfSymbol(symbol);
-                            checkIndexConstraintForProperty(symbol, propType, type, declaredStringIndexer, stringIndexType, 0 /* String */);
-                            checkIndexConstraintForProperty(symbol, propType, type, declaredNumberIndexer, numberIndexType, 1 /* Number */);
+        function checkClassForDuplicateDeclarations(node) {
+            var instanceNames = new ts.Map();
+            var staticNames = new ts.Map();
+            // instance and static private identifiers share the same scope
+            var privateIdentifiers = new ts.Map();
+            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                var member = _a[_i];
+                if (member.kind === 166 /* Constructor */) {
+                    for (var _b = 0, _c = member.parameters; _b < _c.length; _b++) {
+                        var param = _c[_b];
+                        if (ts.isParameterPropertyDeclaration(param, member) && !ts.isBindingPattern(param.name)) {
+                            addName(instanceNames, param.name, param.name.escapedText, 3 /* GetOrSetAccessor */);
                         }
                     }
                 }
-            }
-            var errorNode;
-            if (stringIndexType && numberIndexType) {
-                errorNode = declaredNumberIndexer || declaredStringIndexer;
-                // condition 'errorNode === undefined' may appear if types does not declare nor string neither number indexer
-                if (!errorNode && (ts.getObjectFlags(type) & 2 /* Interface */)) {
-                    var someBaseTypeHasBothIndexers = ts.forEach(getBaseTypes(type), function (base) { return getIndexTypeOfType(base, 0 /* String */) && getIndexTypeOfType(base, 1 /* Number */); });
-                    errorNode = someBaseTypeHasBothIndexers ? undefined : type.symbol.declarations[0];
+                else {
+                    var isStatic = ts.hasSyntacticModifier(member, 32 /* Static */);
+                    var name = member.name;
+                    if (!name) {
+                        return;
+                    }
+                    var names = ts.isPrivateIdentifier(name) ? privateIdentifiers :
+                        isStatic ? staticNames :
+                            instanceNames;
+                    var memberName = name && ts.getPropertyNameForPropertyNameNode(name);
+                    if (memberName) {
+                        switch (member.kind) {
+                            case 167 /* GetAccessor */:
+                                addName(names, name, memberName, 1 /* GetAccessor */);
+                                break;
+                            case 168 /* SetAccessor */:
+                                addName(names, name, memberName, 2 /* SetAccessor */);
+                                break;
+                            case 163 /* PropertyDeclaration */:
+                                addName(names, name, memberName, 3 /* GetOrSetAccessor */);
+                                break;
+                            case 165 /* MethodDeclaration */:
+                                addName(names, name, memberName, 8 /* Method */);
+                                break;
+                        }
+                    }
                 }
             }
-            if (errorNode && !isTypeAssignableTo(numberIndexType, stringIndexType)) { // TODO: GH#18217
-                error(errorNode, ts.Diagnostics.Numeric_index_type_0_is_not_assignable_to_string_index_type_1, typeToString(numberIndexType), typeToString(stringIndexType));
-            }
-            function checkIndexConstraintForProperty(prop, propertyType, containingType, indexDeclaration, indexType, indexKind) {
-                // ESSymbol properties apply to neither string nor numeric indexers.
-                if (!indexType || ts.isKnownSymbol(prop)) {
-                    return;
-                }
-                var propDeclaration = prop.valueDeclaration;
-                var name = propDeclaration && ts.getNameOfDeclaration(propDeclaration);
-                if (name && ts.isPrivateIdentifier(name)) {
-                    return;
-                }
-                // index is numeric and property name is not valid numeric literal
-                if (indexKind === 1 /* Number */ && !(name ? isNumericName(name) : isNumericLiteralName(prop.escapedName))) {
-                    return;
-                }
-                // perform property check if property or indexer is declared in 'type'
-                // this allows us to rule out cases when both property and indexer are inherited from the base class
-                var errorNode;
-                if (propDeclaration && name &&
-                    (propDeclaration.kind === 209 /* BinaryExpression */ ||
-                        name.kind === 154 /* ComputedPropertyName */ ||
-                        prop.parent === containingType.symbol)) {
-                    errorNode = propDeclaration;
-                }
-                else if (indexDeclaration) {
-                    errorNode = indexDeclaration;
-                }
-                else if (ts.getObjectFlags(containingType) & 2 /* Interface */) {
-                    // for interfaces property and indexer might be inherited from different bases
-                    // check if any base class already has both property and indexer.
-                    // check should be performed only if 'type' is the first type that brings property\indexer together
-                    var someBaseClassHasBothPropertyAndIndexer = ts.forEach(getBaseTypes(containingType), function (base) { return getPropertyOfObjectType(base, prop.escapedName) && getIndexTypeOfType(base, indexKind); });
-                    errorNode = someBaseClassHasBothPropertyAndIndexer ? undefined : containingType.symbol.declarations[0];
+            function addName(names, location, name, meaning) {
+                var prev = names.get(name);
+                if (prev) {
+                    if (prev & 8 /* Method */) {
+                        if (meaning !== 8 /* Method */) {
+                            error(location, ts.Diagnostics.Duplicate_identifier_0, ts.getTextOfNode(location));
+                        }
+                    }
+                    else if (prev & meaning) {
+                        error(location, ts.Diagnostics.Duplicate_identifier_0, ts.getTextOfNode(location));
+                    }
+                    else {
+                        names.set(name, prev | meaning);
+                    }
                 }
-                if (errorNode && !isTypeAssignableTo(propertyType, indexType)) {
-                    var errorMessage = indexKind === 0 /* String */
-                        ? ts.Diagnostics.Property_0_of_type_1_is_not_assignable_to_string_index_type_2
-                        : ts.Diagnostics.Property_0_of_type_1_is_not_assignable_to_numeric_index_type_2;
-                    error(errorNode, errorMessage, symbolToString(prop), typeToString(propertyType), typeToString(indexType));
+                else {
+                    names.set(name, meaning);
                 }
             }
         }
-        function checkTypeNameIsReserved(name, message) {
-            // TS 1.0 spec (April 2014): 3.6.1
-            // The predefined type keywords are reserved and cannot be used as names of user defined types.
-            switch (name.escapedText) {
-                case "any":
-                case "unknown":
-                case "number":
-                case "bigint":
-                case "boolean":
-                case "string":
-                case "symbol":
-                case "void":
-                case "object":
-                    error(name, message, name.escapedText);
-            }
-        }
-        /**
-         * The name cannot be used as 'Object' of user defined types with special target.
-         */
-        function checkClassNameCollisionWithObject(name) {
-            if (languageVersion === 1 /* ES5 */ && name.escapedText === "Object"
-                && moduleKind < ts.ModuleKind.ES2015) {
-                error(name, ts.Diagnostics.Class_name_cannot_be_Object_when_targeting_ES5_with_module_0, ts.ModuleKind[moduleKind]); // https://github.com/Microsoft/TypeScript/issues/17494
-            }
-        }
         /**
-         * Check each type parameter and check that type parameters have no duplicate type parameter declarations
+         * Static members being set on a constructor function may conflict with built-in properties
+         * of Function. Esp. in ECMAScript 5 there are non-configurable and non-writable
+         * built-in properties. This check issues a transpile error when a class has a static
+         * member with the same name as a non-writable built-in property.
+         *
+         * @see http://www.ecma-international.org/ecma-262/5.1/#sec-15.3.3
+         * @see http://www.ecma-international.org/ecma-262/5.1/#sec-15.3.5
+         * @see http://www.ecma-international.org/ecma-262/6.0/#sec-properties-of-the-function-constructor
+         * @see http://www.ecma-international.org/ecma-262/6.0/#sec-function-instances
          */
-        function checkTypeParameters(typeParameterDeclarations) {
-            if (typeParameterDeclarations) {
-                var seenDefault = false;
-                for (var i = 0; i < typeParameterDeclarations.length; i++) {
-                    var node = typeParameterDeclarations[i];
-                    checkTypeParameter(node);
-                    if (produceDiagnostics) {
-                        if (node.default) {
-                            seenDefault = true;
-                            checkTypeParametersNotReferenced(node.default, typeParameterDeclarations, i);
-                        }
-                        else if (seenDefault) {
-                            error(node, ts.Diagnostics.Required_type_parameters_may_not_follow_optional_type_parameters);
-                        }
-                        for (var j = 0; j < i; j++) {
-                            if (typeParameterDeclarations[j].symbol === node.symbol) {
-                                error(node.name, ts.Diagnostics.Duplicate_identifier_0, ts.declarationNameToString(node.name));
-                            }
-                        }
+        function checkClassForStaticPropertyNameConflicts(node) {
+            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                var member = _a[_i];
+                var memberNameNode = member.name;
+                var isStatic = ts.hasSyntacticModifier(member, 32 /* Static */);
+                if (isStatic && memberNameNode) {
+                    var memberName = ts.getPropertyNameForPropertyNameNode(memberNameNode);
+                    switch (memberName) {
+                        case "name":
+                        case "length":
+                        case "caller":
+                        case "arguments":
+                        case "prototype":
+                            var message = ts.Diagnostics.Static_property_0_conflicts_with_built_in_property_Function_0_of_constructor_function_1;
+                            var className = getNameOfSymbolAsWritten(getSymbolOfNode(node));
+                            error(memberNameNode, message, memberName, className);
+                            break;
                     }
                 }
             }
         }
-        /** Check that type parameter defaults only reference previously declared type parameters */
-        function checkTypeParametersNotReferenced(root, typeParameters, index) {
-            visit(root);
-            function visit(node) {
-                if (node.kind === 169 /* TypeReference */) {
-                    var type = getTypeFromTypeReference(node);
-                    if (type.flags & 262144 /* TypeParameter */) {
-                        for (var i = index; i < typeParameters.length; i++) {
-                            if (type.symbol === getSymbolOfNode(typeParameters[i])) {
-                                error(node, ts.Diagnostics.Type_parameter_defaults_can_only_reference_previously_declared_type_parameters);
-                            }
-                        }
+        function checkObjectTypeForDuplicateDeclarations(node) {
+            var names = new ts.Map();
+            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                var member = _a[_i];
+                if (member.kind === 162 /* PropertySignature */) {
+                    var memberName = void 0;
+                    var name = member.name;
+                    switch (name.kind) {
+                        case 10 /* StringLiteral */:
+                        case 8 /* NumericLiteral */:
+                            memberName = name.text;
+                            break;
+                        case 78 /* Identifier */:
+                            memberName = ts.idText(name);
+                            break;
+                        default:
+                            continue;
+                    }
+                    if (names.get(memberName)) {
+                        error(ts.getNameOfDeclaration(member.symbol.valueDeclaration), ts.Diagnostics.Duplicate_identifier_0, memberName);
+                        error(member.name, ts.Diagnostics.Duplicate_identifier_0, memberName);
+                    }
+                    else {
+                        names.set(memberName, true);
                     }
                 }
-                ts.forEachChild(node, visit);
             }
         }
-        /** Check that type parameter lists are identical across multiple declarations */
-        function checkTypeParameterListsIdentical(symbol) {
-            if (symbol.declarations.length === 1) {
-                return;
-            }
-            var links = getSymbolLinks(symbol);
-            if (!links.typeParametersChecked) {
-                links.typeParametersChecked = true;
-                var declarations = getClassOrInterfaceDeclarationsOfSymbol(symbol);
-                if (declarations.length <= 1) {
+        function checkTypeForDuplicateIndexSignatures(node) {
+            if (node.kind === 253 /* InterfaceDeclaration */) {
+                var nodeSymbol = getSymbolOfNode(node);
+                // in case of merging interface declaration it is possible that we'll enter this check procedure several times for every declaration
+                // to prevent this run check only for the first declaration of a given kind
+                if (nodeSymbol.declarations.length > 0 && nodeSymbol.declarations[0] !== node) {
                     return;
                 }
-                var type = getDeclaredTypeOfSymbol(symbol);
-                if (!areTypeParametersIdentical(declarations, type.localTypeParameters)) {
-                    // Report an error on every conflicting declaration.
-                    var name = symbolToString(symbol);
-                    for (var _i = 0, declarations_6 = declarations; _i < declarations_6.length; _i++) {
-                        var declaration = declarations_6[_i];
-                        error(declaration.name, ts.Diagnostics.All_declarations_of_0_must_have_identical_type_parameters, name);
+            }
+            // TypeScript 1.0 spec (April 2014)
+            // 3.7.4: An object type can contain at most one string index signature and one numeric index signature.
+            // 8.5: A class declaration can have at most one string index member declaration and one numeric index member declaration
+            var indexSymbol = getIndexSymbol(getSymbolOfNode(node));
+            if (indexSymbol) {
+                var seenNumericIndexer = false;
+                var seenStringIndexer = false;
+                for (var _i = 0, _a = indexSymbol.declarations; _i < _a.length; _i++) {
+                    var decl = _a[_i];
+                    var declaration = decl;
+                    if (declaration.parameters.length === 1 && declaration.parameters[0].type) {
+                        switch (declaration.parameters[0].type.kind) {
+                            case 147 /* StringKeyword */:
+                                if (!seenStringIndexer) {
+                                    seenStringIndexer = true;
+                                }
+                                else {
+                                    error(declaration, ts.Diagnostics.Duplicate_string_index_signature);
+                                }
+                                break;
+                            case 144 /* NumberKeyword */:
+                                if (!seenNumericIndexer) {
+                                    seenNumericIndexer = true;
+                                }
+                                else {
+                                    error(declaration, ts.Diagnostics.Duplicate_number_index_signature);
+                                }
+                                break;
+                        }
                     }
                 }
             }
         }
-        function areTypeParametersIdentical(declarations, targetParameters) {
-            var maxTypeArgumentCount = ts.length(targetParameters);
-            var minTypeArgumentCount = getMinTypeArgumentCount(targetParameters);
-            for (var _i = 0, declarations_7 = declarations; _i < declarations_7.length; _i++) {
-                var declaration = declarations_7[_i];
-                // If this declaration has too few or too many type parameters, we report an error
-                var sourceParameters = ts.getEffectiveTypeParameterDeclarations(declaration);
-                var numTypeParameters = sourceParameters.length;
-                if (numTypeParameters < minTypeArgumentCount || numTypeParameters > maxTypeArgumentCount) {
-                    return false;
-                }
-                for (var i = 0; i < numTypeParameters; i++) {
-                    var source = sourceParameters[i];
-                    var target = targetParameters[i];
-                    // If the type parameter node does not have the same as the resolved type
-                    // parameter at this position, we report an error.
-                    if (source.name.escapedText !== target.symbol.escapedName) {
-                        return false;
-                    }
-                    // If the type parameter node does not have an identical constraint as the resolved
-                    // type parameter at this position, we report an error.
-                    var constraint = ts.getEffectiveConstraintOfTypeParameter(source);
-                    var sourceConstraint = constraint && getTypeFromTypeNode(constraint);
-                    var targetConstraint = getConstraintOfTypeParameter(target);
-                    // relax check if later interface augmentation has no constraint, it's more broad and is OK to merge with
-                    // a more constrained interface (this could be generalized to a full hierarchy check, but that's maybe overkill)
-                    if (sourceConstraint && targetConstraint && !isTypeIdenticalTo(sourceConstraint, targetConstraint)) {
-                        return false;
-                    }
-                    // If the type parameter node has a default and it is not identical to the default
-                    // for the type parameter at this position, we report an error.
-                    var sourceDefault = source.default && getTypeFromTypeNode(source.default);
-                    var targetDefault = getDefaultFromTypeParameter(target);
-                    if (sourceDefault && targetDefault && !isTypeIdenticalTo(sourceDefault, targetDefault)) {
-                        return false;
-                    }
+        function checkPropertyDeclaration(node) {
+            // Grammar checking
+            if (!checkGrammarDecoratorsAndModifiers(node) && !checkGrammarProperty(node))
+                checkGrammarComputedPropertyName(node.name);
+            checkVariableLikeDeclaration(node);
+            // Private class fields transformation relies on WeakMaps.
+            if (ts.isPrivateIdentifier(node.name) && languageVersion < 99 /* ESNext */) {
+                for (var lexicalScope = ts.getEnclosingBlockScopeContainer(node); !!lexicalScope; lexicalScope = ts.getEnclosingBlockScopeContainer(lexicalScope)) {
+                    getNodeLinks(lexicalScope).flags |= 67108864 /* ContainsClassWithPrivateIdentifiers */;
                 }
             }
-            return true;
-        }
-        function checkClassExpression(node) {
-            checkClassLikeDeclaration(node);
-            checkNodeDeferred(node);
-            return getTypeOfSymbol(getSymbolOfNode(node));
-        }
-        function checkClassExpressionDeferred(node) {
-            ts.forEach(node.members, checkSourceElement);
-            registerForUnusedIdentifiersCheck(node);
         }
-        function checkClassDeclaration(node) {
-            if (!node.name && !ts.hasModifier(node, 512 /* Default */)) {
-                grammarErrorOnFirstToken(node, ts.Diagnostics.A_class_declaration_without_the_default_modifier_must_have_a_name);
+        function checkPropertySignature(node) {
+            if (ts.isPrivateIdentifier(node.name)) {
+                error(node, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
             }
-            checkClassLikeDeclaration(node);
-            ts.forEach(node.members, checkSourceElement);
-            registerForUnusedIdentifiersCheck(node);
+            return checkPropertyDeclaration(node);
         }
-        function checkClassLikeDeclaration(node) {
-            checkGrammarClassLikeDeclaration(node);
-            checkDecorators(node);
-            if (node.name) {
-                checkTypeNameIsReserved(node.name, ts.Diagnostics.Class_name_cannot_be_0);
-                checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
-                checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
-                if (!(node.flags & 8388608 /* Ambient */)) {
-                    checkClassNameCollisionWithObject(node.name);
-                }
+        function checkMethodDeclaration(node) {
+            // Grammar checking
+            if (!checkGrammarMethod(node))
+                checkGrammarComputedPropertyName(node.name);
+            if (ts.isPrivateIdentifier(node.name)) {
+                error(node, ts.Diagnostics.A_method_cannot_be_named_with_a_private_identifier);
             }
-            checkTypeParameters(ts.getEffectiveTypeParameterDeclarations(node));
-            checkExportsOnMergedDeclarations(node);
+            // Grammar checking for modifiers is done inside the function checkGrammarFunctionLikeDeclaration
+            checkFunctionOrMethodDeclaration(node);
+            // Abstract methods cannot have an implementation.
+            // Extra checks are to avoid reporting multiple errors relating to the "abstractness" of the node.
+            if (ts.hasSyntacticModifier(node, 128 /* Abstract */) && node.kind === 165 /* MethodDeclaration */ && node.body) {
+                error(node, ts.Diagnostics.Method_0_cannot_have_an_implementation_because_it_is_marked_abstract, ts.declarationNameToString(node.name));
+            }
+        }
+        function checkConstructorDeclaration(node) {
+            // Grammar check on signature of constructor and modifier of the constructor is done in checkSignatureDeclaration function.
+            checkSignatureDeclaration(node);
+            // Grammar check for checking only related to constructorDeclaration
+            if (!checkGrammarConstructorTypeParameters(node))
+                checkGrammarConstructorTypeAnnotation(node);
+            checkSourceElement(node.body);
             var symbol = getSymbolOfNode(node);
-            var type = getDeclaredTypeOfSymbol(symbol);
-            var typeWithThis = getTypeWithThisArgument(type);
-            var staticType = getTypeOfSymbol(symbol);
-            checkTypeParameterListsIdentical(symbol);
-            checkClassForDuplicateDeclarations(node);
-            // Only check for reserved static identifiers on non-ambient context.
-            if (!(node.flags & 8388608 /* Ambient */)) {
-                checkClassForStaticPropertyNameConflicts(node);
+            var firstDeclaration = ts.getDeclarationOfKind(symbol, node.kind);
+            // Only type check the symbol once
+            if (node === firstDeclaration) {
+                checkFunctionOrConstructorSymbol(symbol);
             }
-            var baseTypeNode = ts.getEffectiveBaseTypeNode(node);
-            if (baseTypeNode) {
-                ts.forEach(baseTypeNode.typeArguments, checkSourceElement);
-                if (languageVersion < 2 /* ES2015 */) {
-                    checkExternalEmitHelpers(baseTypeNode.parent, 1 /* Extends */);
-                }
-                // check both @extends and extends if both are specified.
-                var extendsNode = ts.getClassExtendsHeritageElement(node);
-                if (extendsNode && extendsNode !== baseTypeNode) {
-                    checkExpression(extendsNode.expression);
+            // exit early in the case of signature - super checks are not relevant to them
+            if (ts.nodeIsMissing(node.body)) {
+                return;
+            }
+            if (!produceDiagnostics) {
+                return;
+            }
+            function isInstancePropertyWithInitializerOrPrivateIdentifierProperty(n) {
+                if (ts.isPrivateIdentifierPropertyDeclaration(n)) {
+                    return true;
                 }
-                var baseTypes = getBaseTypes(type);
-                if (baseTypes.length && produceDiagnostics) {
-                    var baseType_1 = baseTypes[0];
-                    var baseConstructorType = getBaseConstructorTypeOfClass(type);
-                    var staticBaseType = getApparentType(baseConstructorType);
-                    checkBaseTypeAccessibility(staticBaseType, baseTypeNode);
-                    checkSourceElement(baseTypeNode.expression);
-                    if (ts.some(baseTypeNode.typeArguments)) {
-                        ts.forEach(baseTypeNode.typeArguments, checkSourceElement);
-                        for (var _i = 0, _a = getConstructorsForTypeArguments(staticBaseType, baseTypeNode.typeArguments, baseTypeNode); _i < _a.length; _i++) {
-                            var constructor = _a[_i];
-                            if (!checkTypeArgumentConstraints(baseTypeNode, constructor.typeParameters)) {
+                return n.kind === 163 /* PropertyDeclaration */ &&
+                    !ts.hasSyntacticModifier(n, 32 /* Static */) &&
+                    !!n.initializer;
+            }
+            // TS 1.0 spec (April 2014): 8.3.2
+            // Constructors of classes with no extends clause may not contain super calls, whereas
+            // constructors of derived classes must contain at least one super call somewhere in their function body.
+            var containingClassDecl = node.parent;
+            if (ts.getClassExtendsHeritageElement(containingClassDecl)) {
+                captureLexicalThis(node.parent, containingClassDecl);
+                var classExtendsNull = classDeclarationExtendsNull(containingClassDecl);
+                var superCall = findFirstSuperCall(node.body);
+                if (superCall) {
+                    if (classExtendsNull) {
+                        error(superCall, ts.Diagnostics.A_constructor_cannot_contain_a_super_call_when_its_class_extends_null);
+                    }
+                    // The first statement in the body of a constructor (excluding prologue directives) must be a super call
+                    // if both of the following are true:
+                    // - The containing class is a derived class.
+                    // - The constructor declares parameter properties
+                    //   or the containing class declares instance member variables with initializers.
+                    var superCallShouldBeFirst = (compilerOptions.target !== 99 /* ESNext */ || !compilerOptions.useDefineForClassFields) &&
+                        (ts.some(node.parent.members, isInstancePropertyWithInitializerOrPrivateIdentifierProperty) ||
+                            ts.some(node.parameters, function (p) { return ts.hasSyntacticModifier(p, 92 /* ParameterPropertyModifier */); }));
+                    // Skip past any prologue directives to find the first statement
+                    // to ensure that it was a super call.
+                    if (superCallShouldBeFirst) {
+                        var statements = node.body.statements;
+                        var superCallStatement = void 0;
+                        for (var _i = 0, statements_4 = statements; _i < statements_4.length; _i++) {
+                            var statement = statements_4[_i];
+                            if (statement.kind === 233 /* ExpressionStatement */ && ts.isSuperCall(statement.expression)) {
+                                superCallStatement = statement;
+                                break;
+                            }
+                            if (!ts.isPrologueDirective(statement)) {
                                 break;
                             }
                         }
+                        if (!superCallStatement) {
+                            error(node, ts.Diagnostics.A_super_call_must_be_the_first_statement_in_the_constructor_when_a_class_contains_initialized_properties_parameter_properties_or_private_identifiers);
+                        }
                     }
-                    var baseWithThis = getTypeWithThisArgument(baseType_1, type.thisType);
-                    if (!checkTypeAssignableTo(typeWithThis, baseWithThis, /*errorNode*/ undefined)) {
-                        issueMemberSpecificError(node, typeWithThis, baseWithThis, ts.Diagnostics.Class_0_incorrectly_extends_base_class_1);
-                    }
-                    else {
-                        // Report static side error only when instance type is assignable
-                        checkTypeAssignableTo(staticType, getTypeWithoutSignatures(staticBaseType), node.name || node, ts.Diagnostics.Class_static_side_0_incorrectly_extends_base_class_static_side_1);
-                    }
-                    if (baseConstructorType.flags & 8650752 /* TypeVariable */ && !isMixinConstructorType(staticType)) {
-                        error(node.name || node, ts.Diagnostics.A_mixin_class_must_have_a_constructor_with_a_single_rest_parameter_of_type_any);
+                }
+                else if (!classExtendsNull) {
+                    error(node, ts.Diagnostics.Constructors_for_derived_classes_must_contain_a_super_call);
+                }
+            }
+        }
+        function checkAccessorDeclaration(node) {
+            if (produceDiagnostics) {
+                // Grammar checking accessors
+                if (!checkGrammarFunctionLikeDeclaration(node) && !checkGrammarAccessor(node))
+                    checkGrammarComputedPropertyName(node.name);
+                checkDecorators(node);
+                checkSignatureDeclaration(node);
+                if (node.kind === 167 /* GetAccessor */) {
+                    if (!(node.flags & 8388608 /* Ambient */) && ts.nodeIsPresent(node.body) && (node.flags & 256 /* HasImplicitReturn */)) {
+                        if (!(node.flags & 512 /* HasExplicitReturn */)) {
+                            error(node.name, ts.Diagnostics.A_get_accessor_must_return_a_value);
+                        }
                     }
-                    if (!(staticBaseType.symbol && staticBaseType.symbol.flags & 32 /* Class */) && !(baseConstructorType.flags & 8650752 /* TypeVariable */)) {
-                        // When the static base type is a "class-like" constructor function (but not actually a class), we verify
-                        // that all instantiated base constructor signatures return the same type.
-                        var constructors = getInstantiatedConstructorsForTypeArguments(staticBaseType, baseTypeNode.typeArguments, baseTypeNode);
-                        if (ts.forEach(constructors, function (sig) { return !isJSConstructor(sig.declaration) && !isTypeIdenticalTo(getReturnTypeOfSignature(sig), baseType_1); })) {
-                            error(baseTypeNode.expression, ts.Diagnostics.Base_constructors_must_all_have_the_same_return_type);
+                }
+                // Do not use hasDynamicName here, because that returns false for well known symbols.
+                // We want to perform checkComputedPropertyName for all computed properties, including
+                // well known symbols.
+                if (node.name.kind === 158 /* ComputedPropertyName */) {
+                    checkComputedPropertyName(node.name);
+                }
+                if (ts.isPrivateIdentifier(node.name)) {
+                    error(node.name, ts.Diagnostics.An_accessor_cannot_be_named_with_a_private_identifier);
+                }
+                if (!hasNonBindableDynamicName(node)) {
+                    // TypeScript 1.0 spec (April 2014): 8.4.3
+                    // Accessors for the same member name must specify the same accessibility.
+                    var otherKind = node.kind === 167 /* GetAccessor */ ? 168 /* SetAccessor */ : 167 /* GetAccessor */;
+                    var otherAccessor = ts.getDeclarationOfKind(getSymbolOfNode(node), otherKind);
+                    if (otherAccessor) {
+                        var nodeFlags = ts.getEffectiveModifierFlags(node);
+                        var otherFlags = ts.getEffectiveModifierFlags(otherAccessor);
+                        if ((nodeFlags & 28 /* AccessibilityModifier */) !== (otherFlags & 28 /* AccessibilityModifier */)) {
+                            error(node.name, ts.Diagnostics.Getter_and_setter_accessors_do_not_agree_in_visibility);
+                        }
+                        if ((nodeFlags & 128 /* Abstract */) !== (otherFlags & 128 /* Abstract */)) {
+                            error(node.name, ts.Diagnostics.Accessors_must_both_be_abstract_or_non_abstract);
                         }
+                        // TypeScript 1.0 spec (April 2014): 4.5
+                        // If both accessors include type annotations, the specified types must be identical.
+                        checkAccessorDeclarationTypesIdentical(node, otherAccessor, getAnnotatedAccessorType, ts.Diagnostics.get_and_set_accessor_must_have_the_same_type);
+                        checkAccessorDeclarationTypesIdentical(node, otherAccessor, getThisTypeOfDeclaration, ts.Diagnostics.get_and_set_accessor_must_have_the_same_this_type);
                     }
-                    checkKindsOfPropertyMemberOverrides(type, baseType_1);
+                }
+                var returnType = getTypeOfAccessors(getSymbolOfNode(node));
+                if (node.kind === 167 /* GetAccessor */) {
+                    checkAllCodePathsInNonVoidFunctionReturnOrThrow(node, returnType);
                 }
             }
-            var implementedTypeNodes = ts.getEffectiveImplementsTypeNodes(node);
-            if (implementedTypeNodes) {
-                for (var _b = 0, implementedTypeNodes_1 = implementedTypeNodes; _b < implementedTypeNodes_1.length; _b++) {
-                    var typeRefNode = implementedTypeNodes_1[_b];
-                    if (!ts.isEntityNameExpression(typeRefNode.expression)) {
-                        error(typeRefNode.expression, ts.Diagnostics.A_class_can_only_implement_an_identifier_Slashqualified_name_with_optional_type_arguments);
+            checkSourceElement(node.body);
+        }
+        function checkAccessorDeclarationTypesIdentical(first, second, getAnnotatedType, message) {
+            var firstType = getAnnotatedType(first);
+            var secondType = getAnnotatedType(second);
+            if (firstType && secondType && !isTypeIdenticalTo(firstType, secondType)) {
+                error(first, message);
+            }
+        }
+        function checkMissingDeclaration(node) {
+            checkDecorators(node);
+        }
+        function getEffectiveTypeArguments(node, typeParameters) {
+            return fillMissingTypeArguments(ts.map(node.typeArguments, getTypeFromTypeNode), typeParameters, getMinTypeArgumentCount(typeParameters), ts.isInJSFile(node));
+        }
+        function checkTypeArgumentConstraints(node, typeParameters) {
+            var typeArguments;
+            var mapper;
+            var result = true;
+            for (var i = 0; i < typeParameters.length; i++) {
+                var constraint = getConstraintOfTypeParameter(typeParameters[i]);
+                if (constraint) {
+                    if (!typeArguments) {
+                        typeArguments = getEffectiveTypeArguments(node, typeParameters);
+                        mapper = createTypeMapper(typeParameters, typeArguments);
                     }
-                    checkTypeReferenceNode(typeRefNode);
-                    if (produceDiagnostics) {
-                        var t = getReducedType(getTypeFromTypeNode(typeRefNode));
-                        if (t !== errorType) {
-                            if (isValidBaseType(t)) {
-                                var genericDiag = t.symbol && t.symbol.flags & 32 /* Class */ ?
-                                    ts.Diagnostics.Class_0_incorrectly_implements_class_1_Did_you_mean_to_extend_1_and_inherit_its_members_as_a_subclass :
-                                    ts.Diagnostics.Class_0_incorrectly_implements_interface_1;
-                                var baseWithThis = getTypeWithThisArgument(t, type.thisType);
-                                if (!checkTypeAssignableTo(typeWithThis, baseWithThis, /*errorNode*/ undefined)) {
-                                    issueMemberSpecificError(node, typeWithThis, baseWithThis, genericDiag);
-                                }
-                            }
-                            else {
-                                error(typeRefNode, ts.Diagnostics.A_class_can_only_implement_an_object_type_or_intersection_of_object_types_with_statically_known_members);
-                            }
-                        }
+                    result = result && checkTypeAssignableTo(typeArguments[i], instantiateType(constraint, mapper), node.typeArguments[i], ts.Diagnostics.Type_0_does_not_satisfy_the_constraint_1);
+                }
+            }
+            return result;
+        }
+        function getTypeParametersForTypeReference(node) {
+            var type = getTypeFromTypeReference(node);
+            if (type !== errorType) {
+                var symbol = getNodeLinks(node).resolvedSymbol;
+                if (symbol) {
+                    return symbol.flags & 524288 /* TypeAlias */ && getSymbolLinks(symbol).typeParameters ||
+                        (ts.getObjectFlags(type) & 4 /* Reference */ ? type.target.localTypeParameters : undefined);
+                }
+            }
+            return undefined;
+        }
+        function checkTypeReferenceNode(node) {
+            checkGrammarTypeArguments(node, node.typeArguments);
+            if (node.kind === 173 /* TypeReference */ && node.typeName.jsdocDotPos !== undefined && !ts.isInJSFile(node) && !ts.isInJSDoc(node)) {
+                grammarErrorAtPos(node, node.typeName.jsdocDotPos, 1, ts.Diagnostics.JSDoc_types_can_only_be_used_inside_documentation_comments);
+            }
+            ts.forEach(node.typeArguments, checkSourceElement);
+            var type = getTypeFromTypeReference(node);
+            if (type !== errorType) {
+                if (node.typeArguments && produceDiagnostics) {
+                    var typeParameters = getTypeParametersForTypeReference(node);
+                    if (typeParameters) {
+                        checkTypeArgumentConstraints(node, typeParameters);
+                    }
+                }
+                var symbol = getNodeLinks(node).resolvedSymbol;
+                if (symbol) {
+                    if (ts.some(symbol.declarations, function (d) { return isTypeDeclaration(d) && !!(d.flags & 134217728 /* Deprecated */); })) {
+                        errorOrSuggestion(/* isError */ false, getDeprecatedSuggestionNode(node), ts.Diagnostics._0_is_deprecated, symbol.escapedName);
+                    }
+                    if (type.flags & 32 /* Enum */ && symbol.flags & 8 /* EnumMember */) {
+                        error(node, ts.Diagnostics.Enum_type_0_has_members_with_initializers_that_are_not_literals, typeToString(type));
                     }
                 }
             }
+        }
+        function getTypeArgumentConstraint(node) {
+            var typeReferenceNode = ts.tryCast(node.parent, ts.isTypeReferenceType);
+            if (!typeReferenceNode)
+                return undefined;
+            var typeParameters = getTypeParametersForTypeReference(typeReferenceNode); // TODO: GH#18217
+            var constraint = getConstraintOfTypeParameter(typeParameters[typeReferenceNode.typeArguments.indexOf(node)]);
+            return constraint && instantiateType(constraint, createTypeMapper(typeParameters, getEffectiveTypeArguments(typeReferenceNode, typeParameters)));
+        }
+        function checkTypeQuery(node) {
+            getTypeFromTypeQueryNode(node);
+        }
+        function checkTypeLiteral(node) {
+            ts.forEach(node.members, checkSourceElement);
             if (produceDiagnostics) {
+                var type = getTypeFromTypeLiteralOrFunctionOrConstructorTypeNode(node);
                 checkIndexConstraints(type);
                 checkTypeForDuplicateIndexSignatures(node);
-                checkPropertyInitialization(node);
+                checkObjectTypeForDuplicateDeclarations(node);
             }
         }
-        function issueMemberSpecificError(node, typeWithThis, baseWithThis, broadDiag) {
-            // iterate over all implemented properties and issue errors on each one which isn't compatible, rather than the class as a whole, if possible
-            var issuedMemberError = false;
-            var _loop_19 = function (member) {
-                if (ts.hasStaticModifier(member)) {
-                    return "continue";
+        function checkArrayType(node) {
+            checkSourceElement(node.elementType);
+        }
+        function checkTupleType(node) {
+            var elementTypes = node.elements;
+            var seenOptionalElement = false;
+            var seenRestElement = false;
+            var hasNamedElement = ts.some(elementTypes, ts.isNamedTupleMember);
+            for (var i = 0; i < elementTypes.length; i++) {
+                var e = elementTypes[i];
+                if (e.kind !== 192 /* NamedTupleMember */ && hasNamedElement) {
+                    grammarErrorOnNode(e, ts.Diagnostics.Tuple_members_must_all_have_names_or_all_not_have_names);
+                    break;
                 }
-                var declaredProp = member.name && getSymbolAtLocation(member.name) || getSymbolAtLocation(member);
-                if (declaredProp) {
-                    var prop = getPropertyOfType(typeWithThis, declaredProp.escapedName);
-                    var baseProp = getPropertyOfType(baseWithThis, declaredProp.escapedName);
-                    if (prop && baseProp) {
-                        var rootChain = function () { return ts.chainDiagnosticMessages(
-                        /*details*/ undefined, ts.Diagnostics.Property_0_in_type_1_is_not_assignable_to_the_same_property_in_base_type_2, symbolToString(declaredProp), typeToString(typeWithThis), typeToString(baseWithThis)); };
-                        if (!checkTypeAssignableTo(getTypeOfSymbol(prop), getTypeOfSymbol(baseProp), member.name || member, /*message*/ undefined, rootChain)) {
-                            issuedMemberError = true;
-                        }
+                var flags = getTupleElementFlags(e);
+                if (flags & 8 /* Variadic */) {
+                    var type = getTypeFromTypeNode(e.type);
+                    if (!isArrayLikeType(type)) {
+                        error(e, ts.Diagnostics.A_rest_element_type_must_be_an_array_type);
+                        break;
+                    }
+                    if (isArrayType(type) || isTupleType(type) && type.target.combinedFlags & 4 /* Rest */) {
+                        seenRestElement = true;
                     }
                 }
-            };
-            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                var member = _a[_i];
-                _loop_19(member);
-            }
-            if (!issuedMemberError) {
-                // check again with diagnostics to generate a less-specific error
-                checkTypeAssignableTo(typeWithThis, baseWithThis, node.name || node, broadDiag);
+                else if (flags & 4 /* Rest */) {
+                    seenRestElement = true;
+                }
+                else if (flags & 2 /* Optional */) {
+                    seenOptionalElement = true;
+                }
+                else if (seenOptionalElement) {
+                    grammarErrorOnNode(e, ts.Diagnostics.A_required_element_cannot_follow_an_optional_element);
+                    break;
+                }
+                if (seenRestElement && i !== elementTypes.length - 1) {
+                    grammarErrorOnNode(e, ts.Diagnostics.A_rest_element_must_be_last_in_a_tuple_type);
+                    break;
+                }
             }
+            ts.forEach(node.elements, checkSourceElement);
+            getTypeFromTypeNode(node);
         }
-        function checkBaseTypeAccessibility(type, node) {
-            var signatures = getSignaturesOfType(type, 1 /* Construct */);
-            if (signatures.length) {
-                var declaration = signatures[0].declaration;
-                if (declaration && ts.hasModifier(declaration, 8 /* Private */)) {
-                    var typeClassDeclaration = ts.getClassLikeDeclarationOfSymbol(type.symbol);
-                    if (!isNodeWithinClass(node, typeClassDeclaration)) {
-                        error(node, ts.Diagnostics.Cannot_extend_a_class_0_Class_constructor_is_marked_as_private, getFullyQualifiedName(type.symbol));
+        function checkUnionOrIntersectionType(node) {
+            ts.forEach(node.types, checkSourceElement);
+            getTypeFromTypeNode(node);
+        }
+        function checkIndexedAccessIndexType(type, accessNode) {
+            if (!(type.flags & 8388608 /* IndexedAccess */)) {
+                return type;
+            }
+            // Check if the index type is assignable to 'keyof T' for the object type.
+            var objectType = type.objectType;
+            var indexType = type.indexType;
+            if (isTypeAssignableTo(indexType, getIndexType(objectType, /*stringsOnly*/ false))) {
+                if (accessNode.kind === 202 /* ElementAccessExpression */ && ts.isAssignmentTarget(accessNode) &&
+                    ts.getObjectFlags(objectType) & 32 /* Mapped */ && getMappedTypeModifiers(objectType) & 1 /* IncludeReadonly */) {
+                    error(accessNode, ts.Diagnostics.Index_signature_in_type_0_only_permits_reading, typeToString(objectType));
+                }
+                return type;
+            }
+            // Check if we're indexing with a numeric type and if either object or index types
+            // is a generic type with a constraint that has a numeric index signature.
+            var apparentObjectType = getApparentType(objectType);
+            if (getIndexInfoOfType(apparentObjectType, 1 /* Number */) && isTypeAssignableToKind(indexType, 296 /* NumberLike */)) {
+                return type;
+            }
+            if (isGenericObjectType(objectType)) {
+                var propertyName_1 = getPropertyNameFromIndex(indexType, accessNode);
+                if (propertyName_1) {
+                    var propertySymbol = forEachType(apparentObjectType, function (t) { return getPropertyOfType(t, propertyName_1); });
+                    if (propertySymbol && ts.getDeclarationModifierFlagsFromSymbol(propertySymbol) & 24 /* NonPublicAccessibilityModifier */) {
+                        error(accessNode, ts.Diagnostics.Private_or_protected_member_0_cannot_be_accessed_on_a_type_parameter, ts.unescapeLeadingUnderscores(propertyName_1));
+                        return errorType;
                     }
                 }
             }
+            error(accessNode, ts.Diagnostics.Type_0_cannot_be_used_to_index_type_1, typeToString(indexType), typeToString(objectType));
+            return errorType;
         }
-        function getTargetSymbol(s) {
-            // if symbol is instantiated its flags are not copied from the 'target'
-            // so we'll need to get back original 'target' symbol to work with correct set of flags
-            return ts.getCheckFlags(s) & 1 /* Instantiated */ ? s.target : s;
+        function checkIndexedAccessType(node) {
+            checkSourceElement(node.objectType);
+            checkSourceElement(node.indexType);
+            checkIndexedAccessIndexType(getTypeFromIndexedAccessTypeNode(node), node);
         }
-        function getClassOrInterfaceDeclarationsOfSymbol(symbol) {
-            return ts.filter(symbol.declarations, function (d) {
-                return d.kind === 245 /* ClassDeclaration */ || d.kind === 246 /* InterfaceDeclaration */;
-            });
+        function checkMappedType(node) {
+            checkSourceElement(node.typeParameter);
+            checkSourceElement(node.nameType);
+            checkSourceElement(node.type);
+            if (!node.type) {
+                reportImplicitAny(node, anyType);
+            }
+            var type = getTypeFromMappedTypeNode(node);
+            var nameType = getNameTypeFromMappedType(type);
+            if (nameType) {
+                checkTypeAssignableTo(nameType, keyofConstraintType, node.nameType);
+            }
+            else {
+                var constraintType = getConstraintTypeFromMappedType(type);
+                checkTypeAssignableTo(constraintType, keyofConstraintType, ts.getEffectiveConstraintOfTypeParameter(node.typeParameter));
+            }
         }
-        function checkKindsOfPropertyMemberOverrides(type, baseType) {
-            // TypeScript 1.0 spec (April 2014): 8.2.3
-            // A derived class inherits all members from its base class it doesn't override.
-            // Inheritance means that a derived class implicitly contains all non - overridden members of the base class.
-            // Both public and private property members are inherited, but only public property members can be overridden.
-            // A property member in a derived class is said to override a property member in a base class
-            // when the derived class property member has the same name and kind(instance or static)
-            // as the base class property member.
-            // The type of an overriding property member must be assignable(section 3.8.4)
-            // to the type of the overridden property member, or otherwise a compile - time error occurs.
-            // Base class instance member functions can be overridden by derived class instance member functions,
-            // but not by other kinds of members.
-            // Base class instance member variables and accessors can be overridden by
-            // derived class instance member variables and accessors, but not by other kinds of members.
-            // NOTE: assignability is checked in checkClassDeclaration
-            var baseProperties = getPropertiesOfType(baseType);
-            basePropertyCheck: for (var _i = 0, baseProperties_1 = baseProperties; _i < baseProperties_1.length; _i++) {
-                var baseProperty = baseProperties_1[_i];
-                var base = getTargetSymbol(baseProperty);
-                if (base.flags & 4194304 /* Prototype */) {
-                    continue;
+        function checkThisType(node) {
+            getTypeFromThisTypeNode(node);
+        }
+        function checkTypeOperator(node) {
+            checkGrammarTypeOperatorNode(node);
+            checkSourceElement(node.type);
+        }
+        function checkConditionalType(node) {
+            ts.forEachChild(node, checkSourceElement);
+        }
+        function checkInferType(node) {
+            if (!ts.findAncestor(node, function (n) { return n.parent && n.parent.kind === 184 /* ConditionalType */ && n.parent.extendsType === n; })) {
+                grammarErrorOnNode(node, ts.Diagnostics.infer_declarations_are_only_permitted_in_the_extends_clause_of_a_conditional_type);
+            }
+            checkSourceElement(node.typeParameter);
+            registerForUnusedIdentifiersCheck(node);
+        }
+        function checkTemplateLiteralType(node) {
+            for (var _i = 0, _a = node.templateSpans; _i < _a.length; _i++) {
+                var span = _a[_i];
+                checkSourceElement(span.type);
+                var type = getTypeFromTypeNode(span.type);
+                checkTypeAssignableTo(type, templateConstraintType, span.type);
+            }
+            getTypeFromTypeNode(node);
+        }
+        function checkImportType(node) {
+            checkSourceElement(node.argument);
+            getTypeFromTypeNode(node);
+        }
+        function checkNamedTupleMember(node) {
+            if (node.dotDotDotToken && node.questionToken) {
+                grammarErrorOnNode(node, ts.Diagnostics.A_tuple_member_cannot_be_both_optional_and_rest);
+            }
+            if (node.type.kind === 180 /* OptionalType */) {
+                grammarErrorOnNode(node.type, ts.Diagnostics.A_labeled_tuple_element_is_declared_as_optional_with_a_question_mark_after_the_name_and_before_the_colon_rather_than_after_the_type);
+            }
+            if (node.type.kind === 181 /* RestType */) {
+                grammarErrorOnNode(node.type, ts.Diagnostics.A_labeled_tuple_element_is_declared_as_rest_with_a_before_the_name_rather_than_before_the_type);
+            }
+            checkSourceElement(node.type);
+            getTypeFromTypeNode(node);
+        }
+        function isPrivateWithinAmbient(node) {
+            return (ts.hasEffectiveModifier(node, 8 /* Private */) || ts.isPrivateIdentifierPropertyDeclaration(node)) && !!(node.flags & 8388608 /* Ambient */);
+        }
+        function getEffectiveDeclarationFlags(n, flagsToCheck) {
+            var flags = ts.getCombinedModifierFlags(n);
+            // children of classes (even ambient classes) should not be marked as ambient or export
+            // because those flags have no useful semantics there.
+            if (n.parent.kind !== 253 /* InterfaceDeclaration */ &&
+                n.parent.kind !== 252 /* ClassDeclaration */ &&
+                n.parent.kind !== 221 /* ClassExpression */ &&
+                n.flags & 8388608 /* Ambient */) {
+                if (!(flags & 2 /* Ambient */) && !(ts.isModuleBlock(n.parent) && ts.isModuleDeclaration(n.parent.parent) && ts.isGlobalScopeAugmentation(n.parent.parent))) {
+                    // It is nested in an ambient context, which means it is automatically exported
+                    flags |= 1 /* Export */;
+                }
+                flags |= 2 /* Ambient */;
+            }
+            return flags & flagsToCheck;
+        }
+        function checkFunctionOrConstructorSymbol(symbol) {
+            if (!produceDiagnostics) {
+                return;
+            }
+            function getCanonicalOverload(overloads, implementation) {
+                // Consider the canonical set of flags to be the flags of the bodyDeclaration or the first declaration
+                // Error on all deviations from this canonical set of flags
+                // The caveat is that if some overloads are defined in lib.d.ts, we don't want to
+                // report the errors on those. To achieve this, we will say that the implementation is
+                // the canonical signature only if it is in the same container as the first overload
+                var implementationSharesContainerWithFirstOverload = implementation !== undefined && implementation.parent === overloads[0].parent;
+                return implementationSharesContainerWithFirstOverload ? implementation : overloads[0];
+            }
+            function checkFlagAgreementBetweenOverloads(overloads, implementation, flagsToCheck, someOverloadFlags, allOverloadFlags) {
+                // Error if some overloads have a flag that is not shared by all overloads. To find the
+                // deviations, we XOR someOverloadFlags with allOverloadFlags
+                var someButNotAllOverloadFlags = someOverloadFlags ^ allOverloadFlags;
+                if (someButNotAllOverloadFlags !== 0) {
+                    var canonicalFlags_1 = getEffectiveDeclarationFlags(getCanonicalOverload(overloads, implementation), flagsToCheck);
+                    ts.forEach(overloads, function (o) {
+                        var deviation = getEffectiveDeclarationFlags(o, flagsToCheck) ^ canonicalFlags_1;
+                        if (deviation & 1 /* Export */) {
+                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_exported_or_non_exported);
+                        }
+                        else if (deviation & 2 /* Ambient */) {
+                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_ambient_or_non_ambient);
+                        }
+                        else if (deviation & (8 /* Private */ | 16 /* Protected */)) {
+                            error(ts.getNameOfDeclaration(o) || o, ts.Diagnostics.Overload_signatures_must_all_be_public_private_or_protected);
+                        }
+                        else if (deviation & 128 /* Abstract */) {
+                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_abstract_or_non_abstract);
+                        }
+                    });
+                }
+            }
+            function checkQuestionTokenAgreementBetweenOverloads(overloads, implementation, someHaveQuestionToken, allHaveQuestionToken) {
+                if (someHaveQuestionToken !== allHaveQuestionToken) {
+                    var canonicalHasQuestionToken_1 = ts.hasQuestionToken(getCanonicalOverload(overloads, implementation));
+                    ts.forEach(overloads, function (o) {
+                        var deviation = ts.hasQuestionToken(o) !== canonicalHasQuestionToken_1;
+                        if (deviation) {
+                            error(ts.getNameOfDeclaration(o), ts.Diagnostics.Overload_signatures_must_all_be_optional_or_required);
+                        }
+                    });
                 }
-                var baseSymbol = getPropertyOfObjectType(type, base.escapedName);
-                if (!baseSymbol) {
-                    continue;
+            }
+            var flagsToCheck = 1 /* Export */ | 2 /* Ambient */ | 8 /* Private */ | 16 /* Protected */ | 128 /* Abstract */;
+            var someNodeFlags = 0 /* None */;
+            var allNodeFlags = flagsToCheck;
+            var someHaveQuestionToken = false;
+            var allHaveQuestionToken = true;
+            var hasOverloads = false;
+            var bodyDeclaration;
+            var lastSeenNonAmbientDeclaration;
+            var previousDeclaration;
+            var declarations = symbol.declarations;
+            var isConstructor = (symbol.flags & 16384 /* Constructor */) !== 0;
+            function reportImplementationExpectedError(node) {
+                if (node.name && ts.nodeIsMissing(node.name)) {
+                    return;
                 }
-                var derived = getTargetSymbol(baseSymbol);
-                var baseDeclarationFlags = ts.getDeclarationModifierFlagsFromSymbol(base);
-                ts.Debug.assert(!!derived, "derived should point to something, even if it is the base class' declaration.");
-                // In order to resolve whether the inherited method was overridden in the base class or not,
-                // we compare the Symbols obtained. Since getTargetSymbol returns the symbol on the *uninstantiated*
-                // type declaration, derived and base resolve to the same symbol even in the case of generic classes.
-                if (derived === base) {
-                    // derived class inherits base without override/redeclaration
-                    var derivedClassDecl = ts.getClassLikeDeclarationOfSymbol(type.symbol);
-                    // It is an error to inherit an abstract member without implementing it or being declared abstract.
-                    // If there is no declaration for the derived class (as in the case of class expressions),
-                    // then the class cannot be declared abstract.
-                    if (baseDeclarationFlags & 128 /* Abstract */ && (!derivedClassDecl || !ts.hasModifier(derivedClassDecl, 128 /* Abstract */))) {
-                        // Searches other base types for a declaration that would satisfy the inherited abstract member.
-                        // (The class may have more than one base type via declaration merging with an interface with the
-                        // same name.)
-                        for (var _a = 0, _b = getBaseTypes(type); _a < _b.length; _a++) {
-                            var otherBaseType = _b[_a];
-                            if (otherBaseType === baseType)
-                                continue;
-                            var baseSymbol_1 = getPropertyOfObjectType(otherBaseType, base.escapedName);
-                            var derivedElsewhere = baseSymbol_1 && getTargetSymbol(baseSymbol_1);
-                            if (derivedElsewhere && derivedElsewhere !== base) {
-                                continue basePropertyCheck;
+                var seen = false;
+                var subsequentNode = ts.forEachChild(node.parent, function (c) {
+                    if (seen) {
+                        return c;
+                    }
+                    else {
+                        seen = c === node;
+                    }
+                });
+                // We may be here because of some extra nodes between overloads that could not be parsed into a valid node.
+                // In this case the subsequent node is not really consecutive (.pos !== node.end), and we must ignore it here.
+                if (subsequentNode && subsequentNode.pos === node.end) {
+                    if (subsequentNode.kind === node.kind) {
+                        var errorNode_1 = subsequentNode.name || subsequentNode;
+                        var subsequentName = subsequentNode.name;
+                        if (node.name && subsequentName && (
+                        // both are private identifiers
+                        ts.isPrivateIdentifier(node.name) && ts.isPrivateIdentifier(subsequentName) && node.name.escapedText === subsequentName.escapedText ||
+                            // Both are computed property names
+                            // TODO: GH#17345: These are methods, so handle computed name case. (`Always allowing computed property names is *not* the correct behavior!)
+                            ts.isComputedPropertyName(node.name) && ts.isComputedPropertyName(subsequentName) ||
+                            // Both are literal property names that are the same.
+                            ts.isPropertyNameLiteral(node.name) && ts.isPropertyNameLiteral(subsequentName) &&
+                                ts.getEscapedTextOfIdentifierOrLiteral(node.name) === ts.getEscapedTextOfIdentifierOrLiteral(subsequentName))) {
+                            var reportError = (node.kind === 165 /* MethodDeclaration */ || node.kind === 164 /* MethodSignature */) &&
+                                ts.hasSyntacticModifier(node, 32 /* Static */) !== ts.hasSyntacticModifier(subsequentNode, 32 /* Static */);
+                            // we can get here in two cases
+                            // 1. mixed static and instance class members
+                            // 2. something with the same name was defined before the set of overloads that prevents them from merging
+                            // here we'll report error only for the first case since for second we should already report error in binder
+                            if (reportError) {
+                                var diagnostic = ts.hasSyntacticModifier(node, 32 /* Static */) ? ts.Diagnostics.Function_overload_must_be_static : ts.Diagnostics.Function_overload_must_not_be_static;
+                                error(errorNode_1, diagnostic);
                             }
+                            return;
                         }
-                        if (derivedClassDecl.kind === 214 /* ClassExpression */) {
-                            error(derivedClassDecl, ts.Diagnostics.Non_abstract_class_expression_does_not_implement_inherited_abstract_member_0_from_class_1, symbolToString(baseProperty), typeToString(baseType));
-                        }
-                        else {
-                            error(derivedClassDecl, ts.Diagnostics.Non_abstract_class_0_does_not_implement_inherited_abstract_member_1_from_class_2, typeToString(type), symbolToString(baseProperty), typeToString(baseType));
+                        if (ts.nodeIsPresent(subsequentNode.body)) {
+                            error(errorNode_1, ts.Diagnostics.Function_implementation_name_must_be_0, ts.declarationNameToString(node.name));
+                            return;
                         }
                     }
                 }
+                var errorNode = node.name || node;
+                if (isConstructor) {
+                    error(errorNode, ts.Diagnostics.Constructor_implementation_is_missing);
+                }
                 else {
-                    // derived overrides base.
-                    var derivedDeclarationFlags = ts.getDeclarationModifierFlagsFromSymbol(derived);
-                    if (baseDeclarationFlags & 8 /* Private */ || derivedDeclarationFlags & 8 /* Private */) {
-                        // either base or derived property is private - not override, skip it
-                        continue;
+                    // Report different errors regarding non-consecutive blocks of declarations depending on whether
+                    // the node in question is abstract.
+                    if (ts.hasSyntacticModifier(node, 128 /* Abstract */)) {
+                        error(errorNode, ts.Diagnostics.All_declarations_of_an_abstract_method_must_be_consecutive);
                     }
-                    var errorMessage = void 0;
-                    var basePropertyFlags = base.flags & 98308 /* PropertyOrAccessor */;
-                    var derivedPropertyFlags = derived.flags & 98308 /* PropertyOrAccessor */;
-                    if (basePropertyFlags && derivedPropertyFlags) {
-                        // property/accessor is overridden with property/accessor
-                        if (!compilerOptions.useDefineForClassFields
-                            || baseDeclarationFlags & 128 /* Abstract */ && !(base.valueDeclaration && ts.isPropertyDeclaration(base.valueDeclaration) && base.valueDeclaration.initializer)
-                            || base.valueDeclaration && base.valueDeclaration.parent.kind === 246 /* InterfaceDeclaration */
-                            || derived.valueDeclaration && ts.isBinaryExpression(derived.valueDeclaration)) {
-                            // when the base property is abstract or from an interface, base/derived flags don't need to match
-                            // same when the derived property is from an assignment
-                            continue;
-                        }
-                        var overriddenInstanceProperty = basePropertyFlags !== 4 /* Property */ && derivedPropertyFlags === 4 /* Property */;
-                        var overriddenInstanceAccessor = basePropertyFlags === 4 /* Property */ && derivedPropertyFlags !== 4 /* Property */;
-                        if (overriddenInstanceProperty || overriddenInstanceAccessor) {
-                            var errorMessage_1 = overriddenInstanceProperty ?
-                                ts.Diagnostics._0_is_defined_as_an_accessor_in_class_1_but_is_overridden_here_in_2_as_an_instance_property :
-                                ts.Diagnostics._0_is_defined_as_a_property_in_class_1_but_is_overridden_here_in_2_as_an_accessor;
-                            error(ts.getNameOfDeclaration(derived.valueDeclaration) || derived.valueDeclaration, errorMessage_1, symbolToString(base), typeToString(baseType), typeToString(type));
-                        }
-                        else {
-                            var uninitialized = ts.find(derived.declarations, function (d) { return d.kind === 159 /* PropertyDeclaration */ && !d.initializer; });
-                            if (uninitialized
-                                && !(derived.flags & 33554432 /* Transient */)
-                                && !(baseDeclarationFlags & 128 /* Abstract */)
-                                && !(derivedDeclarationFlags & 128 /* Abstract */)
-                                && !derived.declarations.some(function (d) { return !!(d.flags & 8388608 /* Ambient */); })) {
-                                var constructor = findConstructorDeclaration(ts.getClassLikeDeclarationOfSymbol(type.symbol));
-                                var propName = uninitialized.name;
-                                if (uninitialized.exclamationToken
-                                    || !constructor
-                                    || !ts.isIdentifier(propName)
-                                    || !strictNullChecks
-                                    || !isPropertyInitializedInConstructor(propName, type, constructor)) {
-                                    var errorMessage_2 = ts.Diagnostics.Property_0_will_overwrite_the_base_property_in_1_If_this_is_intentional_add_an_initializer_Otherwise_add_a_declare_modifier_or_remove_the_redundant_declaration;
-                                    error(ts.getNameOfDeclaration(derived.valueDeclaration) || derived.valueDeclaration, errorMessage_2, symbolToString(base), typeToString(baseType));
-                                }
-                            }
-                        }
-                        // correct case
-                        continue;
+                    else {
+                        error(errorNode, ts.Diagnostics.Function_implementation_is_missing_or_not_immediately_following_the_declaration);
                     }
-                    else if (isPrototypeProperty(base)) {
-                        if (isPrototypeProperty(derived) || derived.flags & 4 /* Property */) {
-                            // method is overridden with method or property -- correct case
-                            continue;
+                }
+            }
+            var duplicateFunctionDeclaration = false;
+            var multipleConstructorImplementation = false;
+            var hasNonAmbientClass = false;
+            var functionDeclarations = [];
+            for (var _i = 0, declarations_4 = declarations; _i < declarations_4.length; _i++) {
+                var current = declarations_4[_i];
+                var node = current;
+                var inAmbientContext = node.flags & 8388608 /* Ambient */;
+                var inAmbientContextOrInterface = node.parent && (node.parent.kind === 253 /* InterfaceDeclaration */ || node.parent.kind === 177 /* TypeLiteral */) || inAmbientContext;
+                if (inAmbientContextOrInterface) {
+                    // check if declarations are consecutive only if they are non-ambient
+                    // 1. ambient declarations can be interleaved
+                    // i.e. this is legal
+                    //     declare function foo();
+                    //     declare function bar();
+                    //     declare function foo();
+                    // 2. mixing ambient and non-ambient declarations is a separate error that will be reported - do not want to report an extra one
+                    previousDeclaration = undefined;
+                }
+                if ((node.kind === 252 /* ClassDeclaration */ || node.kind === 221 /* ClassExpression */) && !inAmbientContext) {
+                    hasNonAmbientClass = true;
+                }
+                if (node.kind === 251 /* FunctionDeclaration */ || node.kind === 165 /* MethodDeclaration */ || node.kind === 164 /* MethodSignature */ || node.kind === 166 /* Constructor */) {
+                    functionDeclarations.push(node);
+                    var currentNodeFlags = getEffectiveDeclarationFlags(node, flagsToCheck);
+                    someNodeFlags |= currentNodeFlags;
+                    allNodeFlags &= currentNodeFlags;
+                    someHaveQuestionToken = someHaveQuestionToken || ts.hasQuestionToken(node);
+                    allHaveQuestionToken = allHaveQuestionToken && ts.hasQuestionToken(node);
+                    var bodyIsPresent = ts.nodeIsPresent(node.body);
+                    if (bodyIsPresent && bodyDeclaration) {
+                        if (isConstructor) {
+                            multipleConstructorImplementation = true;
                         }
                         else {
-                            ts.Debug.assert(!!(derived.flags & 98304 /* Accessor */));
-                            errorMessage = ts.Diagnostics.Class_0_defines_instance_member_function_1_but_extended_class_2_defines_it_as_instance_member_accessor;
+                            duplicateFunctionDeclaration = true;
                         }
                     }
-                    else if (base.flags & 98304 /* Accessor */) {
-                        errorMessage = ts.Diagnostics.Class_0_defines_instance_member_accessor_1_but_extended_class_2_defines_it_as_instance_member_function;
+                    else if ((previousDeclaration === null || previousDeclaration === void 0 ? void 0 : previousDeclaration.parent) === node.parent && previousDeclaration.end !== node.pos) {
+                        reportImplementationExpectedError(previousDeclaration);
+                    }
+                    if (bodyIsPresent) {
+                        if (!bodyDeclaration) {
+                            bodyDeclaration = node;
+                        }
                     }
                     else {
-                        errorMessage = ts.Diagnostics.Class_0_defines_instance_member_property_1_but_extended_class_2_defines_it_as_instance_member_function;
+                        hasOverloads = true;
+                    }
+                    previousDeclaration = node;
+                    if (!inAmbientContextOrInterface) {
+                        lastSeenNonAmbientDeclaration = node;
                     }
-                    error(ts.getNameOfDeclaration(derived.valueDeclaration) || derived.valueDeclaration, errorMessage, typeToString(baseType), symbolToString(base), typeToString(type));
                 }
             }
-        }
-        function getNonInterhitedProperties(type, baseTypes, properties) {
-            if (!ts.length(baseTypes)) {
-                return properties;
+            if (multipleConstructorImplementation) {
+                ts.forEach(functionDeclarations, function (declaration) {
+                    error(declaration, ts.Diagnostics.Multiple_constructor_implementations_are_not_allowed);
+                });
             }
-            var seen = ts.createUnderscoreEscapedMap();
-            ts.forEach(properties, function (p) { seen.set(p.escapedName, p); });
-            for (var _i = 0, baseTypes_2 = baseTypes; _i < baseTypes_2.length; _i++) {
-                var base = baseTypes_2[_i];
-                var properties_5 = getPropertiesOfType(getTypeWithThisArgument(base, type.thisType));
-                for (var _a = 0, properties_4 = properties_5; _a < properties_4.length; _a++) {
-                    var prop = properties_4[_a];
-                    var existing = seen.get(prop.escapedName);
-                    if (existing && !isPropertyIdenticalTo(existing, prop)) {
-                        seen.delete(prop.escapedName);
-                    }
-                }
+            if (duplicateFunctionDeclaration) {
+                ts.forEach(functionDeclarations, function (declaration) {
+                    error(ts.getNameOfDeclaration(declaration) || declaration, ts.Diagnostics.Duplicate_function_implementation);
+                });
             }
-            return ts.arrayFrom(seen.values());
-        }
-        function checkInheritedPropertiesAreIdentical(type, typeNode) {
-            var baseTypes = getBaseTypes(type);
-            if (baseTypes.length < 2) {
-                return true;
+            if (hasNonAmbientClass && !isConstructor && symbol.flags & 16 /* Function */) {
+                // A non-ambient class cannot be an implementation for a non-constructor function/class merge
+                // TODO: The below just replicates our older error from when classes and functions were
+                // entirely unable to merge - a more helpful message like "Class declaration cannot implement overload list"
+                // might be warranted. :shrug:
+                ts.forEach(declarations, function (declaration) {
+                    addDuplicateDeclarationError(declaration, ts.Diagnostics.Duplicate_identifier_0, ts.symbolName(symbol), declarations);
+                });
             }
-            var seen = ts.createUnderscoreEscapedMap();
-            ts.forEach(resolveDeclaredMembers(type).declaredProperties, function (p) { seen.set(p.escapedName, { prop: p, containingType: type }); });
-            var ok = true;
-            for (var _i = 0, baseTypes_3 = baseTypes; _i < baseTypes_3.length; _i++) {
-                var base = baseTypes_3[_i];
-                var properties = getPropertiesOfType(getTypeWithThisArgument(base, type.thisType));
-                for (var _a = 0, properties_6 = properties; _a < properties_6.length; _a++) {
-                    var prop = properties_6[_a];
-                    var existing = seen.get(prop.escapedName);
-                    if (!existing) {
-                        seen.set(prop.escapedName, { prop: prop, containingType: base });
-                    }
-                    else {
-                        var isInheritedProperty = existing.containingType !== type;
-                        if (isInheritedProperty && !isPropertyIdenticalTo(existing.prop, prop)) {
-                            ok = false;
-                            var typeName1 = typeToString(existing.containingType);
-                            var typeName2 = typeToString(base);
-                            var errorInfo = ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.Named_property_0_of_types_1_and_2_are_not_identical, symbolToString(prop), typeName1, typeName2);
-                            errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Interface_0_cannot_simultaneously_extend_types_1_and_2, typeToString(type), typeName1, typeName2);
-                            diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(typeNode, errorInfo));
+            // Abstract methods can't have an implementation -- in particular, they don't need one.
+            if (lastSeenNonAmbientDeclaration && !lastSeenNonAmbientDeclaration.body &&
+                !ts.hasSyntacticModifier(lastSeenNonAmbientDeclaration, 128 /* Abstract */) && !lastSeenNonAmbientDeclaration.questionToken) {
+                reportImplementationExpectedError(lastSeenNonAmbientDeclaration);
+            }
+            if (hasOverloads) {
+                checkFlagAgreementBetweenOverloads(declarations, bodyDeclaration, flagsToCheck, someNodeFlags, allNodeFlags);
+                checkQuestionTokenAgreementBetweenOverloads(declarations, bodyDeclaration, someHaveQuestionToken, allHaveQuestionToken);
+                if (bodyDeclaration) {
+                    var signatures = getSignaturesOfSymbol(symbol);
+                    var bodySignature = getSignatureFromDeclaration(bodyDeclaration);
+                    for (var _a = 0, signatures_10 = signatures; _a < signatures_10.length; _a++) {
+                        var signature = signatures_10[_a];
+                        if (!isImplementationCompatibleWithOverload(bodySignature, signature)) {
+                            ts.addRelatedInfo(error(signature.declaration, ts.Diagnostics.This_overload_signature_is_not_compatible_with_its_implementation_signature), ts.createDiagnosticForNode(bodyDeclaration, ts.Diagnostics.The_implementation_signature_is_declared_here));
+                            break;
                         }
                     }
                 }
             }
-            return ok;
         }
-        function checkPropertyInitialization(node) {
-            if (!strictNullChecks || !strictPropertyInitialization || node.flags & 8388608 /* Ambient */) {
+        function checkExportsOnMergedDeclarations(node) {
+            if (!produceDiagnostics) {
                 return;
             }
-            var constructor = findConstructorDeclaration(node);
-            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                var member = _a[_i];
-                if (ts.getModifierFlags(member) & 2 /* Ambient */) {
-                    continue;
+            // if localSymbol is defined on node then node itself is exported - check is required
+            var symbol = node.localSymbol;
+            if (!symbol) {
+                // local symbol is undefined => this declaration is non-exported.
+                // however symbol might contain other declarations that are exported
+                symbol = getSymbolOfNode(node);
+                if (!symbol.exportSymbol) {
+                    // this is a pure local symbol (all declarations are non-exported) - no need to check anything
+                    return;
                 }
-                if (isInstancePropertyWithoutInitializer(member)) {
-                    var propName = member.name;
-                    if (ts.isIdentifier(propName) || ts.isPrivateIdentifier(propName)) {
-                        var type = getTypeOfSymbol(getSymbolOfNode(member));
-                        if (!(type.flags & 3 /* AnyOrUnknown */ || getFalsyFlags(type) & 32768 /* Undefined */)) {
-                            if (!constructor || !isPropertyInitializedInConstructor(propName, type, constructor)) {
-                                error(member.name, ts.Diagnostics.Property_0_has_no_initializer_and_is_not_definitely_assigned_in_the_constructor, ts.declarationNameToString(propName));
-                            }
-                        }
+            }
+            // run the check only for the first declaration in the list
+            if (ts.getDeclarationOfKind(symbol, node.kind) !== node) {
+                return;
+            }
+            var exportedDeclarationSpaces = 0 /* None */;
+            var nonExportedDeclarationSpaces = 0 /* None */;
+            var defaultExportedDeclarationSpaces = 0 /* None */;
+            for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                var d = _a[_i];
+                var declarationSpaces = getDeclarationSpaces(d);
+                var effectiveDeclarationFlags = getEffectiveDeclarationFlags(d, 1 /* Export */ | 512 /* Default */);
+                if (effectiveDeclarationFlags & 1 /* Export */) {
+                    if (effectiveDeclarationFlags & 512 /* Default */) {
+                        defaultExportedDeclarationSpaces |= declarationSpaces;
                     }
+                    else {
+                        exportedDeclarationSpaces |= declarationSpaces;
+                    }
+                }
+                else {
+                    nonExportedDeclarationSpaces |= declarationSpaces;
                 }
             }
-        }
-        function isInstancePropertyWithoutInitializer(node) {
-            return node.kind === 159 /* PropertyDeclaration */ &&
-                !ts.hasModifier(node, 32 /* Static */ | 128 /* Abstract */) &&
-                !node.exclamationToken &&
-                !node.initializer;
-        }
-        function isPropertyInitializedInConstructor(propName, propType, constructor) {
-            var reference = ts.createPropertyAccess(ts.createThis(), propName);
-            reference.expression.parent = reference;
-            reference.parent = constructor;
-            reference.flowNode = constructor.returnFlowNode;
-            var flowType = getFlowTypeOfReference(reference, propType, getOptionalType(propType));
-            return !(getFalsyFlags(flowType) & 32768 /* Undefined */);
-        }
-        function checkInterfaceDeclaration(node) {
-            // Grammar checking
-            if (!checkGrammarDecoratorsAndModifiers(node))
-                checkGrammarInterfaceDeclaration(node);
-            checkTypeParameters(node.typeParameters);
-            if (produceDiagnostics) {
-                checkTypeNameIsReserved(node.name, ts.Diagnostics.Interface_name_cannot_be_0);
-                checkExportsOnMergedDeclarations(node);
-                var symbol = getSymbolOfNode(node);
-                checkTypeParameterListsIdentical(symbol);
-                // Only check this symbol once
-                var firstInterfaceDecl = ts.getDeclarationOfKind(symbol, 246 /* InterfaceDeclaration */);
-                if (node === firstInterfaceDecl) {
-                    var type = getDeclaredTypeOfSymbol(symbol);
-                    var typeWithThis = getTypeWithThisArgument(type);
-                    // run subsequent checks only if first set succeeded
-                    if (checkInheritedPropertiesAreIdentical(type, node.name)) {
-                        for (var _i = 0, _a = getBaseTypes(type); _i < _a.length; _i++) {
-                            var baseType = _a[_i];
-                            checkTypeAssignableTo(typeWithThis, getTypeWithThisArgument(baseType, type.thisType), node.name, ts.Diagnostics.Interface_0_incorrectly_extends_interface_1);
-                        }
-                        checkIndexConstraints(type);
+            // Spaces for anything not declared a 'default export'.
+            var nonDefaultExportedDeclarationSpaces = exportedDeclarationSpaces | nonExportedDeclarationSpaces;
+            var commonDeclarationSpacesForExportsAndLocals = exportedDeclarationSpaces & nonExportedDeclarationSpaces;
+            var commonDeclarationSpacesForDefaultAndNonDefault = defaultExportedDeclarationSpaces & nonDefaultExportedDeclarationSpaces;
+            if (commonDeclarationSpacesForExportsAndLocals || commonDeclarationSpacesForDefaultAndNonDefault) {
+                // declaration spaces for exported and non-exported declarations intersect
+                for (var _b = 0, _c = symbol.declarations; _b < _c.length; _b++) {
+                    var d = _c[_b];
+                    var declarationSpaces = getDeclarationSpaces(d);
+                    var name = ts.getNameOfDeclaration(d);
+                    // Only error on the declarations that contributed to the intersecting spaces.
+                    if (declarationSpaces & commonDeclarationSpacesForDefaultAndNonDefault) {
+                        error(name, ts.Diagnostics.Merged_declaration_0_cannot_include_a_default_export_declaration_Consider_adding_a_separate_export_default_0_declaration_instead, ts.declarationNameToString(name));
+                    }
+                    else if (declarationSpaces & commonDeclarationSpacesForExportsAndLocals) {
+                        error(name, ts.Diagnostics.Individual_declarations_in_merged_declaration_0_must_be_all_exported_or_all_local, ts.declarationNameToString(name));
                     }
                 }
-                checkObjectTypeForDuplicateDeclarations(node);
             }
-            ts.forEach(ts.getInterfaceBaseTypeNodes(node), function (heritageElement) {
-                if (!ts.isEntityNameExpression(heritageElement.expression)) {
-                    error(heritageElement.expression, ts.Diagnostics.An_interface_can_only_extend_an_identifier_Slashqualified_name_with_optional_type_arguments);
+            function getDeclarationSpaces(decl) {
+                var d = decl;
+                switch (d.kind) {
+                    case 253 /* InterfaceDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                    // A jsdoc typedef and callback are, by definition, type aliases.
+                    // falls through
+                    case 331 /* JSDocTypedefTag */:
+                    case 324 /* JSDocCallbackTag */:
+                    case 325 /* JSDocEnumTag */:
+                        return 2 /* ExportType */;
+                    case 256 /* ModuleDeclaration */:
+                        return ts.isAmbientModule(d) || ts.getModuleInstanceState(d) !== 0 /* NonInstantiated */
+                            ? 4 /* ExportNamespace */ | 1 /* ExportValue */
+                            : 4 /* ExportNamespace */;
+                    case 252 /* ClassDeclaration */:
+                    case 255 /* EnumDeclaration */:
+                    case 291 /* EnumMember */:
+                        return 2 /* ExportType */ | 1 /* ExportValue */;
+                    case 297 /* SourceFile */:
+                        return 2 /* ExportType */ | 1 /* ExportValue */ | 4 /* ExportNamespace */;
+                    case 266 /* ExportAssignment */:
+                        // Export assigned entity name expressions act as aliases and should fall through, otherwise they export values
+                        if (!ts.isEntityNameExpression(d.expression)) {
+                            return 1 /* ExportValue */;
+                        }
+                        d = d.expression;
+                    // The below options all declare an Alias, which is allowed to merge with other values within the importing module.
+                    // falls through
+                    case 260 /* ImportEqualsDeclaration */:
+                    case 263 /* NamespaceImport */:
+                    case 262 /* ImportClause */:
+                        var result_12 = 0 /* None */;
+                        var target = resolveAlias(getSymbolOfNode(d));
+                        ts.forEach(target.declarations, function (d) { result_12 |= getDeclarationSpaces(d); });
+                        return result_12;
+                    case 249 /* VariableDeclaration */:
+                    case 198 /* BindingElement */:
+                    case 251 /* FunctionDeclaration */:
+                    case 265 /* ImportSpecifier */: // https://github.com/Microsoft/TypeScript/pull/7591
+                    case 78 /* Identifier */: // https://github.com/microsoft/TypeScript/issues/36098
+                        // Identifiers are used as declarations of assignment declarations whose parents may be
+                        // SyntaxKind.CallExpression - `Object.defineProperty(thing, "aField", {value: 42});`
+                        // SyntaxKind.ElementAccessExpression - `thing["aField"] = 42;` or `thing["aField"];` (with a doc comment on it)
+                        // or SyntaxKind.PropertyAccessExpression - `thing.aField = 42;`
+                        // all of which are pretty much always values, or at least imply a value meaning.
+                        // It may be apprpriate to treat these as aliases in the future.
+                        return 1 /* ExportValue */;
+                    default:
+                        return ts.Debug.failBadSyntaxKind(d);
                 }
-                checkTypeReferenceNode(heritageElement);
-            });
-            ts.forEach(node.members, checkSourceElement);
-            if (produceDiagnostics) {
-                checkTypeForDuplicateIndexSignatures(node);
-                registerForUnusedIdentifiersCheck(node);
             }
         }
-        function checkTypeAliasDeclaration(node) {
-            // Grammar checking
-            checkGrammarDecoratorsAndModifiers(node);
-            checkTypeNameIsReserved(node.name, ts.Diagnostics.Type_alias_name_cannot_be_0);
-            checkExportsOnMergedDeclarations(node);
-            checkTypeParameters(node.typeParameters);
-            checkSourceElement(node.type);
-            registerForUnusedIdentifiersCheck(node);
-        }
-        function computeEnumMemberValues(node) {
-            var nodeLinks = getNodeLinks(node);
-            if (!(nodeLinks.flags & 16384 /* EnumValuesComputed */)) {
-                nodeLinks.flags |= 16384 /* EnumValuesComputed */;
-                var autoValue = 0;
-                for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
-                    var member = _a[_i];
-                    var value = computeMemberValue(member, autoValue);
-                    getNodeLinks(member).enumMemberValue = value;
-                    autoValue = typeof value === "number" ? value + 1 : undefined;
-                }
-            }
+        function getAwaitedTypeOfPromise(type, errorNode, diagnosticMessage, arg0) {
+            var promisedType = getPromisedTypeOfPromise(type, errorNode);
+            return promisedType && getAwaitedType(promisedType, errorNode, diagnosticMessage, arg0);
         }
-        function computeMemberValue(member, autoValue) {
-            if (ts.isComputedNonLiteralName(member.name)) {
-                error(member.name, ts.Diagnostics.Computed_property_names_are_not_allowed_in_enums);
-            }
-            else {
-                var text = ts.getTextOfPropertyName(member.name);
-                if (isNumericLiteralName(text) && !isInfinityOrNaNString(text)) {
-                    error(member.name, ts.Diagnostics.An_enum_member_cannot_have_a_numeric_name);
-                }
-            }
-            if (member.initializer) {
-                return computeConstantValue(member);
-            }
-            // In ambient non-const numeric enum declarations, enum members without initializers are
-            // considered computed members (as opposed to having auto-incremented values).
-            if (member.parent.flags & 8388608 /* Ambient */ && !ts.isEnumConst(member.parent) && getEnumKind(getSymbolOfNode(member.parent)) === 0 /* Numeric */) {
+        /**
+         * Gets the "promised type" of a promise.
+         * @param type The type of the promise.
+         * @remarks The "promised type" of a type is the type of the "value" parameter of the "onfulfilled" callback.
+         */
+        function getPromisedTypeOfPromise(type, errorNode) {
+            //
+            //  { // type
+            //      then( // thenFunction
+            //          onfulfilled: ( // onfulfilledParameterType
+            //              value: T // valueParameterType
+            //          ) => any
+            //      ): any;
+            //  }
+            //
+            if (isTypeAny(type)) {
                 return undefined;
             }
-            // If the member declaration specifies no value, the member is considered a constant enum member.
-            // If the member is the first member in the enum declaration, it is assigned the value zero.
-            // Otherwise, it is assigned the value of the immediately preceding member plus one, and an error
-            // occurs if the immediately preceding member is not a constant enum member.
-            if (autoValue !== undefined) {
-                return autoValue;
-            }
-            error(member.name, ts.Diagnostics.Enum_member_must_have_initializer);
-            return undefined;
-        }
-        function computeConstantValue(member) {
-            var enumKind = getEnumKind(getSymbolOfNode(member.parent));
-            var isConstEnum = ts.isEnumConst(member.parent);
-            var initializer = member.initializer;
-            var value = enumKind === 1 /* Literal */ && !isLiteralEnumMember(member) ? undefined : evaluate(initializer);
-            if (value !== undefined) {
-                if (isConstEnum && typeof value === "number" && !isFinite(value)) {
-                    error(initializer, isNaN(value) ?
-                        ts.Diagnostics.const_enum_member_initializer_was_evaluated_to_disallowed_value_NaN :
-                        ts.Diagnostics.const_enum_member_initializer_was_evaluated_to_a_non_finite_value);
-                }
-            }
-            else if (enumKind === 1 /* Literal */) {
-                error(initializer, ts.Diagnostics.Computed_values_are_not_permitted_in_an_enum_with_string_valued_members);
-                return 0;
+            var typeAsPromise = type;
+            if (typeAsPromise.promisedTypeOfPromise) {
+                return typeAsPromise.promisedTypeOfPromise;
             }
-            else if (isConstEnum) {
-                error(initializer, ts.Diagnostics.const_enum_member_initializers_can_only_contain_literal_values_and_other_computed_enum_values);
+            if (isReferenceToType(type, getGlobalPromiseType(/*reportErrors*/ false))) {
+                return typeAsPromise.promisedTypeOfPromise = getTypeArguments(type)[0];
             }
-            else if (member.parent.flags & 8388608 /* Ambient */) {
-                error(initializer, ts.Diagnostics.In_ambient_enum_declarations_member_initializer_must_be_constant_expression);
+            var thenFunction = getTypeOfPropertyOfType(type, "then"); // TODO: GH#18217
+            if (isTypeAny(thenFunction)) {
+                return undefined;
             }
-            else {
-                // Only here do we need to check that the initializer is assignable to the enum type.
-                var source = checkExpression(initializer);
-                if (!isTypeAssignableToKind(source, 296 /* NumberLike */)) {
-                    error(initializer, ts.Diagnostics.Only_numeric_enums_can_have_computed_members_but_this_expression_has_type_0_If_you_do_not_need_exhaustiveness_checks_consider_using_an_object_literal_instead, typeToString(source));
-                }
-                else {
-                    checkTypeAssignableTo(source, getDeclaredTypeOfSymbol(getSymbolOfNode(member.parent)), initializer, /*headMessage*/ undefined);
+            var thenSignatures = thenFunction ? getSignaturesOfType(thenFunction, 0 /* Call */) : ts.emptyArray;
+            if (thenSignatures.length === 0) {
+                if (errorNode) {
+                    error(errorNode, ts.Diagnostics.A_promise_must_have_a_then_method);
                 }
+                return undefined;
             }
-            return value;
-            function evaluate(expr) {
-                switch (expr.kind) {
-                    case 207 /* PrefixUnaryExpression */:
-                        var value_2 = evaluate(expr.operand);
-                        if (typeof value_2 === "number") {
-                            switch (expr.operator) {
-                                case 39 /* PlusToken */: return value_2;
-                                case 40 /* MinusToken */: return -value_2;
-                                case 54 /* TildeToken */: return ~value_2;
-                            }
-                        }
-                        break;
-                    case 209 /* BinaryExpression */:
-                        var left = evaluate(expr.left);
-                        var right = evaluate(expr.right);
-                        if (typeof left === "number" && typeof right === "number") {
-                            switch (expr.operatorToken.kind) {
-                                case 51 /* BarToken */: return left | right;
-                                case 50 /* AmpersandToken */: return left & right;
-                                case 48 /* GreaterThanGreaterThanToken */: return left >> right;
-                                case 49 /* GreaterThanGreaterThanGreaterThanToken */: return left >>> right;
-                                case 47 /* LessThanLessThanToken */: return left << right;
-                                case 52 /* CaretToken */: return left ^ right;
-                                case 41 /* AsteriskToken */: return left * right;
-                                case 43 /* SlashToken */: return left / right;
-                                case 39 /* PlusToken */: return left + right;
-                                case 40 /* MinusToken */: return left - right;
-                                case 44 /* PercentToken */: return left % right;
-                                case 42 /* AsteriskAsteriskToken */: return Math.pow(left, right);
-                            }
-                        }
-                        else if (typeof left === "string" && typeof right === "string" && expr.operatorToken.kind === 39 /* PlusToken */) {
-                            return left + right;
-                        }
-                        break;
-                    case 10 /* StringLiteral */:
-                    case 14 /* NoSubstitutionTemplateLiteral */:
-                        return expr.text;
-                    case 8 /* NumericLiteral */:
-                        checkGrammarNumericLiteral(expr);
-                        return +expr.text;
-                    case 200 /* ParenthesizedExpression */:
-                        return evaluate(expr.expression);
-                    case 75 /* Identifier */:
-                        var identifier = expr;
-                        if (isInfinityOrNaNString(identifier.escapedText)) {
-                            return +(identifier.escapedText);
-                        }
-                        return ts.nodeIsMissing(expr) ? 0 : evaluateEnumMember(expr, getSymbolOfNode(member.parent), identifier.escapedText);
-                    case 195 /* ElementAccessExpression */:
-                    case 194 /* PropertyAccessExpression */:
-                        var ex = expr;
-                        if (isConstantMemberAccess(ex)) {
-                            var type = getTypeOfExpression(ex.expression);
-                            if (type.symbol && type.symbol.flags & 384 /* Enum */) {
-                                var name = void 0;
-                                if (ex.kind === 194 /* PropertyAccessExpression */) {
-                                    name = ex.name.escapedText;
-                                }
-                                else {
-                                    name = ts.escapeLeadingUnderscores(ts.cast(ex.argumentExpression, ts.isLiteralExpression).text);
-                                }
-                                return evaluateEnumMember(expr, type.symbol, name);
-                            }
-                        }
-                        break;
-                }
+            var onfulfilledParameterType = getTypeWithFacts(getUnionType(ts.map(thenSignatures, getTypeOfFirstParameterOfSignature)), 2097152 /* NEUndefinedOrNull */);
+            if (isTypeAny(onfulfilledParameterType)) {
                 return undefined;
             }
-            function evaluateEnumMember(expr, enumSymbol, name) {
-                var memberSymbol = enumSymbol.exports.get(name);
-                if (memberSymbol) {
-                    var declaration = memberSymbol.valueDeclaration;
-                    if (declaration !== member) {
-                        if (isBlockScopedNameDeclaredBeforeUse(declaration, member)) {
-                            return getEnumMemberValue(declaration);
-                        }
-                        error(expr, ts.Diagnostics.A_member_initializer_in_a_enum_declaration_cannot_reference_members_declared_after_it_including_members_defined_in_other_enums);
-                        return 0;
-                    }
-                    else {
-                        error(expr, ts.Diagnostics.Property_0_is_used_before_being_assigned, symbolToString(memberSymbol));
-                    }
+            var onfulfilledParameterSignatures = getSignaturesOfType(onfulfilledParameterType, 0 /* Call */);
+            if (onfulfilledParameterSignatures.length === 0) {
+                if (errorNode) {
+                    error(errorNode, ts.Diagnostics.The_first_parameter_of_the_then_method_of_a_promise_must_be_a_callback);
                 }
                 return undefined;
             }
+            return typeAsPromise.promisedTypeOfPromise = getUnionType(ts.map(onfulfilledParameterSignatures, getTypeOfFirstParameterOfSignature), 2 /* Subtype */);
         }
-        function isConstantMemberAccess(node) {
-            return node.kind === 75 /* Identifier */ ||
-                node.kind === 194 /* PropertyAccessExpression */ && isConstantMemberAccess(node.expression) ||
-                node.kind === 195 /* ElementAccessExpression */ && isConstantMemberAccess(node.expression) &&
-                    ts.isStringLiteralLike(node.argumentExpression);
+        /**
+         * Gets the "awaited type" of a type.
+         * @param type The type to await.
+         * @remarks The "awaited type" of an expression is its "promised type" if the expression is a
+         * Promise-like type; otherwise, it is the type of the expression. This is used to reflect
+         * The runtime behavior of the `await` keyword.
+         */
+        function checkAwaitedType(type, errorNode, diagnosticMessage, arg0) {
+            var awaitedType = getAwaitedType(type, errorNode, diagnosticMessage, arg0);
+            return awaitedType || errorType;
         }
-        function checkEnumDeclaration(node) {
-            if (!produceDiagnostics) {
-                return;
-            }
-            // Grammar checking
-            checkGrammarDecoratorsAndModifiers(node);
-            checkTypeNameIsReserved(node.name, ts.Diagnostics.Enum_name_cannot_be_0);
-            checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
-            checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
-            checkExportsOnMergedDeclarations(node);
-            node.members.forEach(checkEnumMember);
-            computeEnumMemberValues(node);
-            // Spec 2014 - Section 9.3:
-            // It isn't possible for one enum declaration to continue the automatic numbering sequence of another,
-            // and when an enum type has multiple declarations, only one declaration is permitted to omit a value
-            // for the first member.
-            //
-            // Only perform this check once per symbol
-            var enumSymbol = getSymbolOfNode(node);
-            var firstDeclaration = ts.getDeclarationOfKind(enumSymbol, node.kind);
-            if (node === firstDeclaration) {
-                if (enumSymbol.declarations.length > 1) {
-                    var enumIsConst_1 = ts.isEnumConst(node);
-                    // check that const is placed\omitted on all enum declarations
-                    ts.forEach(enumSymbol.declarations, function (decl) {
-                        if (ts.isEnumDeclaration(decl) && ts.isEnumConst(decl) !== enumIsConst_1) {
-                            error(ts.getNameOfDeclaration(decl), ts.Diagnostics.Enum_declarations_must_all_be_const_or_non_const);
-                        }
-                    });
-                }
-                var seenEnumMissingInitialInitializer_1 = false;
-                ts.forEach(enumSymbol.declarations, function (declaration) {
-                    // return true if we hit a violation of the rule, false otherwise
-                    if (declaration.kind !== 248 /* EnumDeclaration */) {
-                        return false;
-                    }
-                    var enumDeclaration = declaration;
-                    if (!enumDeclaration.members.length) {
-                        return false;
-                    }
-                    var firstEnumMember = enumDeclaration.members[0];
-                    if (!firstEnumMember.initializer) {
-                        if (seenEnumMissingInitialInitializer_1) {
-                            error(firstEnumMember.name, ts.Diagnostics.In_an_enum_with_multiple_declarations_only_one_declaration_can_omit_an_initializer_for_its_first_enum_element);
-                        }
-                        else {
-                            seenEnumMissingInitialInitializer_1 = true;
-                        }
-                    }
-                });
-            }
+        /**
+         * Determines whether a type has a callable `then` member.
+         */
+        function isThenableType(type) {
+            var thenFunction = getTypeOfPropertyOfType(type, "then");
+            return !!thenFunction && getSignaturesOfType(getTypeWithFacts(thenFunction, 2097152 /* NEUndefinedOrNull */), 0 /* Call */).length > 0;
         }
-        function checkEnumMember(node) {
-            if (ts.isPrivateIdentifier(node.name)) {
-                error(node, ts.Diagnostics.An_enum_member_cannot_be_named_with_a_private_identifier);
+        /**
+         * Gets the "awaited type" of a type.
+         *
+         * The "awaited type" of an expression is its "promised type" if the expression is a
+         * Promise-like type; otherwise, it is the type of the expression. If the "promised
+         * type" is itself a Promise-like, the "promised type" is recursively unwrapped until a
+         * non-promise type is found.
+         *
+         * This is used to reflect the runtime behavior of the `await` keyword.
+         */
+        function getAwaitedType(type, errorNode, diagnosticMessage, arg0) {
+            if (isTypeAny(type)) {
+                return type;
             }
-        }
-        function getFirstNonAmbientClassOrFunctionDeclaration(symbol) {
-            var declarations = symbol.declarations;
-            for (var _i = 0, declarations_8 = declarations; _i < declarations_8.length; _i++) {
-                var declaration = declarations_8[_i];
-                if ((declaration.kind === 245 /* ClassDeclaration */ ||
-                    (declaration.kind === 244 /* FunctionDeclaration */ && ts.nodeIsPresent(declaration.body))) &&
-                    !(declaration.flags & 8388608 /* Ambient */)) {
-                    return declaration;
-                }
+            var typeAsAwaitable = type;
+            if (typeAsAwaitable.awaitedTypeOfType) {
+                return typeAsAwaitable.awaitedTypeOfType;
             }
-            return undefined;
+            // For a union, get a union of the awaited types of each constituent.
+            //
+            return typeAsAwaitable.awaitedTypeOfType =
+                mapType(type, errorNode ? function (constituentType) { return getAwaitedTypeWorker(constituentType, errorNode, diagnosticMessage, arg0); } : getAwaitedTypeWorker);
         }
-        function inSameLexicalScope(node1, node2) {
-            var container1 = ts.getEnclosingBlockScopeContainer(node1);
-            var container2 = ts.getEnclosingBlockScopeContainer(node2);
-            if (isGlobalSourceFile(container1)) {
-                return isGlobalSourceFile(container2);
+        function getAwaitedTypeWorker(type, errorNode, diagnosticMessage, arg0) {
+            var typeAsAwaitable = type;
+            if (typeAsAwaitable.awaitedTypeOfType) {
+                return typeAsAwaitable.awaitedTypeOfType;
             }
-            else if (isGlobalSourceFile(container2)) {
-                return false;
+            var promisedType = getPromisedTypeOfPromise(type);
+            if (promisedType) {
+                if (type.id === promisedType.id || awaitedTypeStack.lastIndexOf(promisedType.id) >= 0) {
+                    // Verify that we don't have a bad actor in the form of a promise whose
+                    // promised type is the same as the promise type, or a mutually recursive
+                    // promise. If so, we return undefined as we cannot guess the shape. If this
+                    // were the actual case in the JavaScript, this Promise would never resolve.
+                    //
+                    // An example of a bad actor with a singly-recursive promise type might
+                    // be:
+                    //
+                    //  interface BadPromise {
+                    //      then(
+                    //          onfulfilled: (value: BadPromise) => any,
+                    //          onrejected: (error: any) => any): BadPromise;
+                    //  }
+                    //
+                    // The above interface will pass the PromiseLike check, and return a
+                    // promised type of `BadPromise`. Since this is a self reference, we
+                    // don't want to keep recursing ad infinitum.
+                    //
+                    // An example of a bad actor in the form of a mutually-recursive
+                    // promise type might be:
+                    //
+                    //  interface BadPromiseA {
+                    //      then(
+                    //          onfulfilled: (value: BadPromiseB) => any,
+                    //          onrejected: (error: any) => any): BadPromiseB;
+                    //  }
+                    //
+                    //  interface BadPromiseB {
+                    //      then(
+                    //          onfulfilled: (value: BadPromiseA) => any,
+                    //          onrejected: (error: any) => any): BadPromiseA;
+                    //  }
+                    //
+                    if (errorNode) {
+                        error(errorNode, ts.Diagnostics.Type_is_referenced_directly_or_indirectly_in_the_fulfillment_callback_of_its_own_then_method);
+                    }
+                    return undefined;
+                }
+                // Keep track of the type we're about to unwrap to avoid bad recursive promise types.
+                // See the comments above for more information.
+                awaitedTypeStack.push(type.id);
+                var awaitedType = getAwaitedType(promisedType, errorNode, diagnosticMessage, arg0);
+                awaitedTypeStack.pop();
+                if (!awaitedType) {
+                    return undefined;
+                }
+                return typeAsAwaitable.awaitedTypeOfType = awaitedType;
             }
-            else {
-                return container1 === container2;
+            // The type was not a promise, so it could not be unwrapped any further.
+            // As long as the type does not have a callable "then" property, it is
+            // safe to return the type; otherwise, an error is reported and we return
+            // undefined.
+            //
+            // An example of a non-promise "thenable" might be:
+            //
+            //  await { then(): void {} }
+            //
+            // The "thenable" does not match the minimal definition for a promise. When
+            // a Promise/A+-compatible or ES6 promise tries to adopt this value, the promise
+            // will never settle. We treat this as an error to help flag an early indicator
+            // of a runtime problem. If the user wants to return this value from an async
+            // function, they would need to wrap it in some other value. If they want it to
+            // be treated as a promise, they can cast to <any>.
+            if (isThenableType(type)) {
+                if (errorNode) {
+                    if (!diagnosticMessage)
+                        return ts.Debug.fail();
+                    error(errorNode, diagnosticMessage, arg0);
+                }
+                return undefined;
             }
+            return typeAsAwaitable.awaitedTypeOfType = type;
         }
-        function checkModuleDeclaration(node) {
-            if (produceDiagnostics) {
-                // Grammar checking
-                var isGlobalAugmentation = ts.isGlobalScopeAugmentation(node);
-                var inAmbientContext = node.flags & 8388608 /* Ambient */;
-                if (isGlobalAugmentation && !inAmbientContext) {
-                    error(node.name, ts.Diagnostics.Augmentations_for_the_global_scope_should_have_declare_modifier_unless_they_appear_in_already_ambient_context);
-                }
-                var isAmbientExternalModule = ts.isAmbientModule(node);
-                var contextErrorMessage = isAmbientExternalModule
-                    ? ts.Diagnostics.An_ambient_module_declaration_is_only_allowed_at_the_top_level_in_a_file
-                    : ts.Diagnostics.A_namespace_declaration_is_only_allowed_in_a_namespace_or_module;
-                if (checkGrammarModuleElementContext(node, contextErrorMessage)) {
-                    // If we hit a module declaration in an illegal context, just bail out to avoid cascading errors.
+        /**
+         * Checks the return type of an async function to ensure it is a compatible
+         * Promise implementation.
+         *
+         * This checks that an async function has a valid Promise-compatible return type.
+         * An async function has a valid Promise-compatible return type if the resolved value
+         * of the return type has a construct signature that takes in an `initializer` function
+         * that in turn supplies a `resolve` function as one of its arguments and results in an
+         * object with a callable `then` signature.
+         *
+         * @param node The signature to check
+         */
+        function checkAsyncFunctionReturnType(node, returnTypeNode) {
+            // As part of our emit for an async function, we will need to emit the entity name of
+            // the return type annotation as an expression. To meet the necessary runtime semantics
+            // for __awaiter, we must also check that the type of the declaration (e.g. the static
+            // side or "constructor" of the promise type) is compatible `PromiseConstructorLike`.
+            //
+            // An example might be (from lib.es6.d.ts):
+            //
+            //  interface Promise<T> { ... }
+            //  interface PromiseConstructor {
+            //      new <T>(...): Promise<T>;
+            //  }
+            //  declare var Promise: PromiseConstructor;
+            //
+            // When an async function declares a return type annotation of `Promise<T>`, we
+            // need to get the type of the `Promise` variable declaration above, which would
+            // be `PromiseConstructor`.
+            //
+            // The same case applies to a class:
+            //
+            //  declare class Promise<T> {
+            //      constructor(...);
+            //      then<U>(...): Promise<U>;
+            //  }
+            //
+            var returnType = getTypeFromTypeNode(returnTypeNode);
+            if (languageVersion >= 2 /* ES2015 */) {
+                if (returnType === errorType) {
                     return;
                 }
-                if (!checkGrammarDecoratorsAndModifiers(node)) {
-                    if (!inAmbientContext && node.name.kind === 10 /* StringLiteral */) {
-                        grammarErrorOnNode(node.name, ts.Diagnostics.Only_ambient_modules_can_use_quoted_names);
-                    }
+                var globalPromiseType = getGlobalPromiseType(/*reportErrors*/ true);
+                if (globalPromiseType !== emptyGenericType && !isReferenceToType(returnType, globalPromiseType)) {
+                    // The promise type was not a valid type reference to the global promise type, so we
+                    // report an error and return the unknown type.
+                    error(returnTypeNode, ts.Diagnostics.The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type_Did_you_mean_to_write_Promise_0, typeToString(getAwaitedType(returnType) || voidType));
+                    return;
                 }
-                if (ts.isIdentifier(node.name)) {
-                    checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
-                    checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+            }
+            else {
+                // Always mark the type node as referenced if it points to a value
+                markTypeNodeAsReferenced(returnTypeNode);
+                if (returnType === errorType) {
+                    return;
                 }
-                checkExportsOnMergedDeclarations(node);
-                var symbol = getSymbolOfNode(node);
-                // The following checks only apply on a non-ambient instantiated module declaration.
-                if (symbol.flags & 512 /* ValueModule */
-                    && !inAmbientContext
-                    && symbol.declarations.length > 1
-                    && isInstantiatedModule(node, !!compilerOptions.preserveConstEnums || !!compilerOptions.isolatedModules)) {
-                    var firstNonAmbientClassOrFunc = getFirstNonAmbientClassOrFunctionDeclaration(symbol);
-                    if (firstNonAmbientClassOrFunc) {
-                        if (ts.getSourceFileOfNode(node) !== ts.getSourceFileOfNode(firstNonAmbientClassOrFunc)) {
-                            error(node.name, ts.Diagnostics.A_namespace_declaration_cannot_be_in_a_different_file_from_a_class_or_function_with_which_it_is_merged);
-                        }
-                        else if (node.pos < firstNonAmbientClassOrFunc.pos) {
-                            error(node.name, ts.Diagnostics.A_namespace_declaration_cannot_be_located_prior_to_a_class_or_function_with_which_it_is_merged);
-                        }
-                    }
-                    // if the module merges with a class declaration in the same lexical scope,
-                    // we need to track this to ensure the correct emit.
-                    var mergedClass = ts.getDeclarationOfKind(symbol, 245 /* ClassDeclaration */);
-                    if (mergedClass &&
-                        inSameLexicalScope(node, mergedClass)) {
-                        getNodeLinks(node).flags |= 32768 /* LexicalModuleMergesWithClass */;
-                    }
+                var promiseConstructorName = ts.getEntityNameFromTypeNode(returnTypeNode);
+                if (promiseConstructorName === undefined) {
+                    error(returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value, typeToString(returnType));
+                    return;
                 }
-                if (isAmbientExternalModule) {
-                    if (ts.isExternalModuleAugmentation(node)) {
-                        // body of the augmentation should be checked for consistency only if augmentation was applied to its target (either global scope or module)
-                        // otherwise we'll be swamped in cascading errors.
-                        // We can detect if augmentation was applied using following rules:
-                        // - augmentation for a global scope is always applied
-                        // - augmentation for some external module is applied if symbol for augmentation is merged (it was combined with target module).
-                        var checkBody = isGlobalAugmentation || (getSymbolOfNode(node).flags & 33554432 /* Transient */);
-                        if (checkBody && node.body) {
-                            for (var _i = 0, _a = node.body.statements; _i < _a.length; _i++) {
-                                var statement = _a[_i];
-                                checkModuleAugmentationElement(statement, isGlobalAugmentation);
-                            }
-                        }
-                    }
-                    else if (isGlobalSourceFile(node.parent)) {
-                        if (isGlobalAugmentation) {
-                            error(node.name, ts.Diagnostics.Augmentations_for_the_global_scope_can_only_be_directly_nested_in_external_modules_or_ambient_module_declarations);
-                        }
-                        else if (ts.isExternalModuleNameRelative(ts.getTextOfIdentifierOrLiteral(node.name))) {
-                            error(node.name, ts.Diagnostics.Ambient_module_declaration_cannot_specify_relative_module_name);
-                        }
+                var promiseConstructorSymbol = resolveEntityName(promiseConstructorName, 111551 /* Value */, /*ignoreErrors*/ true);
+                var promiseConstructorType = promiseConstructorSymbol ? getTypeOfSymbol(promiseConstructorSymbol) : errorType;
+                if (promiseConstructorType === errorType) {
+                    if (promiseConstructorName.kind === 78 /* Identifier */ && promiseConstructorName.escapedText === "Promise" && getTargetType(returnType) === getGlobalPromiseType(/*reportErrors*/ false)) {
+                        error(returnTypeNode, ts.Diagnostics.An_async_function_or_method_in_ES5_SlashES3_requires_the_Promise_constructor_Make_sure_you_have_a_declaration_for_the_Promise_constructor_or_include_ES2015_in_your_lib_option);
                     }
                     else {
-                        if (isGlobalAugmentation) {
-                            error(node.name, ts.Diagnostics.Augmentations_for_the_global_scope_can_only_be_directly_nested_in_external_modules_or_ambient_module_declarations);
-                        }
-                        else {
-                            // Node is not an augmentation and is not located on the script level.
-                            // This means that this is declaration of ambient module that is located in other module or namespace which is prohibited.
-                            error(node.name, ts.Diagnostics.Ambient_modules_cannot_be_nested_in_other_modules_or_namespaces);
-                        }
+                        error(returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value, ts.entityNameToString(promiseConstructorName));
                     }
+                    return;
                 }
-            }
-            if (node.body) {
-                checkSourceElement(node.body);
-                if (!ts.isGlobalScopeAugmentation(node)) {
-                    registerForUnusedIdentifiersCheck(node);
+                var globalPromiseConstructorLikeType = getGlobalPromiseConstructorLikeType(/*reportErrors*/ true);
+                if (globalPromiseConstructorLikeType === emptyObjectType) {
+                    // If we couldn't resolve the global PromiseConstructorLike type we cannot verify
+                    // compatibility with __awaiter.
+                    error(returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value, ts.entityNameToString(promiseConstructorName));
+                    return;
+                }
+                if (!checkTypeAssignableTo(promiseConstructorType, globalPromiseConstructorLikeType, returnTypeNode, ts.Diagnostics.Type_0_is_not_a_valid_async_function_return_type_in_ES5_SlashES3_because_it_does_not_refer_to_a_Promise_compatible_constructor_value)) {
+                    return;
+                }
+                // Verify there is no local declaration that could collide with the promise constructor.
+                var rootName = promiseConstructorName && ts.getFirstIdentifier(promiseConstructorName);
+                var collidingSymbol = getSymbol(node.locals, rootName.escapedText, 111551 /* Value */);
+                if (collidingSymbol) {
+                    error(collidingSymbol.valueDeclaration, ts.Diagnostics.Duplicate_identifier_0_Compiler_uses_declaration_1_to_support_async_functions, ts.idText(rootName), ts.entityNameToString(promiseConstructorName));
+                    return;
                 }
             }
+            checkAwaitedType(returnType, node, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member);
         }
-        function checkModuleAugmentationElement(node, isGlobalAugmentation) {
-            switch (node.kind) {
-                case 225 /* VariableStatement */:
-                    // error each individual name in variable statement instead of marking the entire variable statement
-                    for (var _i = 0, _a = node.declarationList.declarations; _i < _a.length; _i++) {
-                        var decl = _a[_i];
-                        checkModuleAugmentationElement(decl, isGlobalAugmentation);
-                    }
+        /** Check a decorator */
+        function checkDecorator(node) {
+            var signature = getResolvedSignature(node);
+            checkDeprecatedSignature(signature, node);
+            var returnType = getReturnTypeOfSignature(signature);
+            if (returnType.flags & 1 /* Any */) {
+                return;
+            }
+            var expectedReturnType;
+            var headMessage = getDiagnosticHeadMessageForDecoratorResolution(node);
+            var errorInfo;
+            switch (node.parent.kind) {
+                case 252 /* ClassDeclaration */:
+                    var classSymbol = getSymbolOfNode(node.parent);
+                    var classConstructorType = getTypeOfSymbol(classSymbol);
+                    expectedReturnType = getUnionType([classConstructorType, voidType]);
                     break;
-                case 259 /* ExportAssignment */:
-                case 260 /* ExportDeclaration */:
-                    grammarErrorOnFirstToken(node, ts.Diagnostics.Exports_and_export_assignments_are_not_permitted_in_module_augmentations);
+                case 160 /* Parameter */:
+                    expectedReturnType = voidType;
+                    errorInfo = ts.chainDiagnosticMessages(
+                    /*details*/ undefined, ts.Diagnostics.The_return_type_of_a_parameter_decorator_function_must_be_either_void_or_any);
                     break;
-                case 253 /* ImportEqualsDeclaration */:
-                case 254 /* ImportDeclaration */:
-                    grammarErrorOnFirstToken(node, ts.Diagnostics.Imports_are_not_permitted_in_module_augmentations_Consider_moving_them_to_the_enclosing_external_module);
+                case 163 /* PropertyDeclaration */:
+                    expectedReturnType = voidType;
+                    errorInfo = ts.chainDiagnosticMessages(
+                    /*details*/ undefined, ts.Diagnostics.The_return_type_of_a_property_decorator_function_must_be_either_void_or_any);
                     break;
-                case 191 /* BindingElement */:
-                case 242 /* VariableDeclaration */:
-                    var name = node.name;
-                    if (ts.isBindingPattern(name)) {
-                        for (var _b = 0, _c = name.elements; _b < _c.length; _b++) {
-                            var el = _c[_b];
-                            // mark individual names in binding pattern
-                            checkModuleAugmentationElement(el, isGlobalAugmentation);
-                        }
-                        break;
-                    }
-                // falls through
-                case 245 /* ClassDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 244 /* FunctionDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 249 /* ModuleDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                    if (isGlobalAugmentation) {
-                        return;
-                    }
-                    var symbol = getSymbolOfNode(node);
-                    if (symbol) {
-                        // module augmentations cannot introduce new names on the top level scope of the module
-                        // this is done it two steps
-                        // 1. quick check - if symbol for node is not merged - this is local symbol to this augmentation - report error
-                        // 2. main check - report error if value declaration of the parent symbol is module augmentation)
-                        var reportError = !(symbol.flags & 33554432 /* Transient */);
-                        if (!reportError) {
-                            // symbol should not originate in augmentation
-                            reportError = !!symbol.parent && ts.isExternalModuleAugmentation(symbol.parent.declarations[0]);
-                        }
-                    }
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    var methodType = getTypeOfNode(node.parent);
+                    var descriptorType = createTypedPropertyDescriptorType(methodType);
+                    expectedReturnType = getUnionType([descriptorType, voidType]);
                     break;
+                default:
+                    return ts.Debug.fail();
             }
+            checkTypeAssignableTo(returnType, expectedReturnType, node, headMessage, function () { return errorInfo; });
         }
-        function getFirstNonModuleExportsIdentifier(node) {
-            switch (node.kind) {
-                case 75 /* Identifier */:
-                    return node;
-                case 153 /* QualifiedName */:
-                    do {
-                        node = node.left;
-                    } while (node.kind !== 75 /* Identifier */);
-                    return node;
-                case 194 /* PropertyAccessExpression */:
-                    do {
-                        if (ts.isModuleExportsAccessExpression(node.expression) && !ts.isPrivateIdentifier(node.name)) {
-                            return node.name;
-                        }
-                        node = node.expression;
-                    } while (node.kind !== 75 /* Identifier */);
-                    return node;
+        /**
+         * If a TypeNode can be resolved to a value symbol imported from an external module, it is
+         * marked as referenced to prevent import elision.
+         */
+        function markTypeNodeAsReferenced(node) {
+            markEntityNameOrEntityExpressionAsReference(node && ts.getEntityNameFromTypeNode(node));
+        }
+        function markEntityNameOrEntityExpressionAsReference(typeName) {
+            if (!typeName)
+                return;
+            var rootName = ts.getFirstIdentifier(typeName);
+            var meaning = (typeName.kind === 78 /* Identifier */ ? 788968 /* Type */ : 1920 /* Namespace */) | 2097152 /* Alias */;
+            var rootSymbol = resolveName(rootName, rootName.escapedText, meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isRefernce*/ true);
+            if (rootSymbol
+                && rootSymbol.flags & 2097152 /* Alias */
+                && symbolIsValue(rootSymbol)
+                && !isConstEnumOrConstEnumOnlyModule(resolveAlias(rootSymbol))
+                && !getTypeOnlyAliasDeclaration(rootSymbol)) {
+                markAliasSymbolAsReferenced(rootSymbol);
+            }
+        }
+        /**
+         * This function marks the type used for metadata decorator as referenced if it is import
+         * from external module.
+         * This is different from markTypeNodeAsReferenced because it tries to simplify type nodes in
+         * union and intersection type
+         * @param node
+         */
+        function markDecoratorMedataDataTypeNodeAsReferenced(node) {
+            var entityName = getEntityNameForDecoratorMetadata(node);
+            if (entityName && ts.isEntityName(entityName)) {
+                markEntityNameOrEntityExpressionAsReference(entityName);
             }
         }
-        function checkExternalImportOrExportDeclaration(node) {
-            var moduleName = ts.getExternalModuleName(node);
-            if (!moduleName || ts.nodeIsMissing(moduleName)) {
-                // Should be a parse error.
-                return false;
-            }
-            if (!ts.isStringLiteral(moduleName)) {
-                error(moduleName, ts.Diagnostics.String_literal_expected);
-                return false;
-            }
-            var inAmbientExternalModule = node.parent.kind === 250 /* ModuleBlock */ && ts.isAmbientModule(node.parent.parent);
-            if (node.parent.kind !== 290 /* SourceFile */ && !inAmbientExternalModule) {
-                error(moduleName, node.kind === 260 /* ExportDeclaration */ ?
-                    ts.Diagnostics.Export_declarations_are_not_permitted_in_a_namespace :
-                    ts.Diagnostics.Import_declarations_in_a_namespace_cannot_reference_a_module);
-                return false;
-            }
-            if (inAmbientExternalModule && ts.isExternalModuleNameRelative(moduleName.text)) {
-                // we have already reported errors on top level imports/exports in external module augmentations in checkModuleDeclaration
-                // no need to do this again.
-                if (!isTopLevelInExternalModuleAugmentation(node)) {
-                    // TypeScript 1.0 spec (April 2013): 12.1.6
-                    // An ExternalImportDeclaration in an AmbientExternalModuleDeclaration may reference
-                    // other external modules only through top - level external module names.
-                    // Relative external module names are not permitted.
-                    error(node, ts.Diagnostics.Import_or_export_declaration_in_an_ambient_module_declaration_cannot_reference_module_through_relative_module_name);
-                    return false;
+        function getEntityNameForDecoratorMetadata(node) {
+            if (node) {
+                switch (node.kind) {
+                    case 183 /* IntersectionType */:
+                    case 182 /* UnionType */:
+                        return getEntityNameForDecoratorMetadataFromTypeList(node.types);
+                    case 184 /* ConditionalType */:
+                        return getEntityNameForDecoratorMetadataFromTypeList([node.trueType, node.falseType]);
+                    case 186 /* ParenthesizedType */:
+                    case 192 /* NamedTupleMember */:
+                        return getEntityNameForDecoratorMetadata(node.type);
+                    case 173 /* TypeReference */:
+                        return node.typeName;
                 }
             }
-            return true;
         }
-        function checkAliasSymbol(node) {
-            var symbol = getSymbolOfNode(node);
-            var target = resolveAlias(symbol);
-            var shouldSkipWithJSExpandoTargets = symbol.flags & 67108864 /* Assignment */;
-            if (!shouldSkipWithJSExpandoTargets && target !== unknownSymbol) {
-                // For external modules symbol represents local symbol for an alias.
-                // This local symbol will merge any other local declarations (excluding other aliases)
-                // and symbol.flags will contains combined representation for all merged declaration.
-                // Based on symbol.flags we can compute a set of excluded meanings (meaning that resolved alias should not have,
-                // otherwise it will conflict with some local declaration). Note that in addition to normal flags we include matching SymbolFlags.Export*
-                // in order to prevent collisions with declarations that were exported from the current module (they still contribute to local names).
-                symbol = getMergedSymbol(symbol.exportSymbol || symbol);
-                var excludedMeanings = (symbol.flags & (111551 /* Value */ | 1048576 /* ExportValue */) ? 111551 /* Value */ : 0) |
-                    (symbol.flags & 788968 /* Type */ ? 788968 /* Type */ : 0) |
-                    (symbol.flags & 1920 /* Namespace */ ? 1920 /* Namespace */ : 0);
-                if (target.flags & excludedMeanings) {
-                    var message = node.kind === 263 /* ExportSpecifier */ ?
-                        ts.Diagnostics.Export_declaration_conflicts_with_exported_declaration_of_0 :
-                        ts.Diagnostics.Import_declaration_conflicts_with_local_declaration_of_0;
-                    error(node, message, symbolToString(symbol));
+        function getEntityNameForDecoratorMetadataFromTypeList(types) {
+            var commonEntityName;
+            for (var _i = 0, types_21 = types; _i < types_21.length; _i++) {
+                var typeNode = types_21[_i];
+                while (typeNode.kind === 186 /* ParenthesizedType */ || typeNode.kind === 192 /* NamedTupleMember */) {
+                    typeNode = typeNode.type; // Skip parens if need be
                 }
-                // Don't allow to re-export something with no value side when `--isolatedModules` is set.
-                if (compilerOptions.isolatedModules
-                    && node.kind === 263 /* ExportSpecifier */
-                    && !node.parent.parent.isTypeOnly
-                    && !(target.flags & 111551 /* Value */)
-                    && !(node.flags & 8388608 /* Ambient */)) {
-                    error(node, ts.Diagnostics.Re_exporting_a_type_when_the_isolatedModules_flag_is_provided_requires_using_export_type);
+                if (typeNode.kind === 141 /* NeverKeyword */) {
+                    continue; // Always elide `never` from the union/intersection if possible
+                }
+                if (!strictNullChecks && (typeNode.kind === 191 /* LiteralType */ && typeNode.literal.kind === 103 /* NullKeyword */ || typeNode.kind === 150 /* UndefinedKeyword */)) {
+                    continue; // Elide null and undefined from unions for metadata, just like what we did prior to the implementation of strict null checks
+                }
+                var individualEntityName = getEntityNameForDecoratorMetadata(typeNode);
+                if (!individualEntityName) {
+                    // Individual is something like string number
+                    // So it would be serialized to either that type or object
+                    // Safe to return here
+                    return undefined;
+                }
+                if (commonEntityName) {
+                    // Note this is in sync with the transformation that happens for type node.
+                    // Keep this in sync with serializeUnionOrIntersectionType
+                    // Verify if they refer to same entity and is identifier
+                    // return undefined if they dont match because we would emit object
+                    if (!ts.isIdentifier(commonEntityName) ||
+                        !ts.isIdentifier(individualEntityName) ||
+                        commonEntityName.escapedText !== individualEntityName.escapedText) {
+                        return undefined;
+                    }
+                }
+                else {
+                    commonEntityName = individualEntityName;
                 }
             }
+            return commonEntityName;
         }
-        function checkImportBinding(node) {
-            checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
-            checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
-            checkAliasSymbol(node);
+        function getParameterTypeNodeForDecoratorCheck(node) {
+            var typeNode = ts.getEffectiveTypeAnnotationNode(node);
+            return ts.isRestParameter(node) ? ts.getRestParameterElementType(typeNode) : typeNode;
         }
-        function checkImportDeclaration(node) {
-            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_import_declaration_can_only_be_used_in_a_namespace_or_module)) {
-                // If we hit an import declaration in an illegal context, just bail out to avoid cascading errors.
+        /** Check the decorators of a node */
+        function checkDecorators(node) {
+            if (!node.decorators) {
                 return;
             }
-            if (!checkGrammarDecoratorsAndModifiers(node) && ts.hasModifiers(node)) {
-                grammarErrorOnFirstToken(node, ts.Diagnostics.An_import_declaration_cannot_have_modifiers);
+            // skip this check for nodes that cannot have decorators. These should have already had an error reported by
+            // checkGrammarDecorators.
+            if (!ts.nodeCanBeDecorated(node, node.parent, node.parent.parent)) {
+                return;
             }
-            if (checkExternalImportOrExportDeclaration(node)) {
-                var importClause = node.importClause;
-                if (importClause && !checkGrammarImportClause(importClause)) {
-                    if (importClause.name) {
-                        checkImportBinding(importClause);
-                    }
-                    if (importClause.namedBindings) {
-                        if (importClause.namedBindings.kind === 256 /* NamespaceImport */) {
-                            checkImportBinding(importClause.namedBindings);
-                        }
-                        else {
-                            var moduleExisted = resolveExternalModuleName(node, node.moduleSpecifier);
-                            if (moduleExisted) {
-                                ts.forEach(importClause.namedBindings.elements, checkImportBinding);
-                            }
-                        }
-                    }
-                }
+            if (!compilerOptions.experimentalDecorators) {
+                error(node, ts.Diagnostics.Experimental_support_for_decorators_is_a_feature_that_is_subject_to_change_in_a_future_release_Set_the_experimentalDecorators_option_in_your_tsconfig_or_jsconfig_to_remove_this_warning);
             }
-        }
-        function checkImportEqualsDeclaration(node) {
-            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_import_declaration_can_only_be_used_in_a_namespace_or_module)) {
-                // If we hit an import declaration in an illegal context, just bail out to avoid cascading errors.
-                return;
+            var firstDecorator = node.decorators[0];
+            checkExternalEmitHelpers(firstDecorator, 8 /* Decorate */);
+            if (node.kind === 160 /* Parameter */) {
+                checkExternalEmitHelpers(firstDecorator, 32 /* Param */);
             }
-            checkGrammarDecoratorsAndModifiers(node);
-            if (ts.isInternalModuleImportEqualsDeclaration(node) || checkExternalImportOrExportDeclaration(node)) {
-                checkImportBinding(node);
-                if (ts.hasModifier(node, 1 /* Export */)) {
-                    markExportAsReferenced(node);
-                }
-                if (node.moduleReference.kind !== 265 /* ExternalModuleReference */) {
-                    var target = resolveAlias(getSymbolOfNode(node));
-                    if (target !== unknownSymbol) {
-                        if (target.flags & 111551 /* Value */) {
-                            // Target is a value symbol, check that it is not hidden by a local declaration with the same name
-                            var moduleName = ts.getFirstIdentifier(node.moduleReference);
-                            if (!(resolveEntityName(moduleName, 111551 /* Value */ | 1920 /* Namespace */).flags & 1920 /* Namespace */)) {
-                                error(moduleName, ts.Diagnostics.Module_0_is_hidden_by_a_local_declaration_with_the_same_name, ts.declarationNameToString(moduleName));
+            if (compilerOptions.emitDecoratorMetadata) {
+                checkExternalEmitHelpers(firstDecorator, 16 /* Metadata */);
+                // we only need to perform these checks if we are emitting serialized type metadata for the target of a decorator.
+                switch (node.kind) {
+                    case 252 /* ClassDeclaration */:
+                        var constructor = ts.getFirstConstructorWithBody(node);
+                        if (constructor) {
+                            for (var _i = 0, _a = constructor.parameters; _i < _a.length; _i++) {
+                                var parameter = _a[_i];
+                                markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(parameter));
                             }
                         }
-                        if (target.flags & 788968 /* Type */) {
-                            checkTypeNameIsReserved(node.name, ts.Diagnostics.Import_name_cannot_be_0);
+                        break;
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        var otherKind = node.kind === 167 /* GetAccessor */ ? 168 /* SetAccessor */ : 167 /* GetAccessor */;
+                        var otherAccessor = ts.getDeclarationOfKind(getSymbolOfNode(node), otherKind);
+                        markDecoratorMedataDataTypeNodeAsReferenced(getAnnotatedAccessorTypeNode(node) || otherAccessor && getAnnotatedAccessorTypeNode(otherAccessor));
+                        break;
+                    case 165 /* MethodDeclaration */:
+                        for (var _b = 0, _c = node.parameters; _b < _c.length; _b++) {
+                            var parameter = _c[_b];
+                            markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(parameter));
                         }
-                    }
-                }
-                else {
-                    if (moduleKind >= ts.ModuleKind.ES2015 && !(node.flags & 8388608 /* Ambient */)) {
-                        // Import equals declaration is deprecated in es6 or above
-                        grammarErrorOnNode(node, ts.Diagnostics.Import_assignment_cannot_be_used_when_targeting_ECMAScript_modules_Consider_using_import_Asterisk_as_ns_from_mod_import_a_from_mod_import_d_from_mod_or_another_module_format_instead);
-                    }
+                        markDecoratorMedataDataTypeNodeAsReferenced(ts.getEffectiveReturnTypeNode(node));
+                        break;
+                    case 163 /* PropertyDeclaration */:
+                        markDecoratorMedataDataTypeNodeAsReferenced(ts.getEffectiveTypeAnnotationNode(node));
+                        break;
+                    case 160 /* Parameter */:
+                        markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(node));
+                        var containingSignature = node.parent;
+                        for (var _d = 0, _e = containingSignature.parameters; _d < _e.length; _d++) {
+                            var parameter = _e[_d];
+                            markDecoratorMedataDataTypeNodeAsReferenced(getParameterTypeNodeForDecoratorCheck(parameter));
+                        }
+                        break;
                 }
             }
+            ts.forEach(node.decorators, checkDecorator);
         }
-        function checkExportDeclaration(node) {
-            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_export_declaration_can_only_be_used_in_a_module)) {
-                // If we hit an export in an illegal context, just bail out to avoid cascading errors.
-                return;
+        function checkFunctionDeclaration(node) {
+            if (produceDiagnostics) {
+                checkFunctionOrMethodDeclaration(node);
+                checkGrammarForGenerator(node);
+                checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
+                checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
             }
-            if (!checkGrammarDecoratorsAndModifiers(node) && ts.hasModifiers(node)) {
-                grammarErrorOnFirstToken(node, ts.Diagnostics.An_export_declaration_cannot_have_modifiers);
+        }
+        function checkJSDocTypeAliasTag(node) {
+            if (!node.typeExpression) {
+                // If the node had `@property` tags, `typeExpression` would have been set to the first property tag.
+                error(node.name, ts.Diagnostics.JSDoc_typedef_tag_should_either_have_a_type_annotation_or_be_followed_by_property_or_member_tags);
             }
-            if (node.moduleSpecifier && node.exportClause && ts.isNamedExports(node.exportClause) && ts.length(node.exportClause.elements) && languageVersion === 0 /* ES3 */) {
-                checkExternalEmitHelpers(node, 1048576 /* CreateBinding */);
+            if (node.name) {
+                checkTypeNameIsReserved(node.name, ts.Diagnostics.Type_alias_name_cannot_be_0);
             }
-            checkGrammarExportDeclaration(node);
-            if (!node.moduleSpecifier || checkExternalImportOrExportDeclaration(node)) {
-                if (node.exportClause && !ts.isNamespaceExport(node.exportClause)) {
-                    // export { x, y }
-                    // export { x, y } from "foo"
-                    ts.forEach(node.exportClause.elements, checkExportSpecifier);
-                    var inAmbientExternalModule = node.parent.kind === 250 /* ModuleBlock */ && ts.isAmbientModule(node.parent.parent);
-                    var inAmbientNamespaceDeclaration = !inAmbientExternalModule && node.parent.kind === 250 /* ModuleBlock */ &&
-                        !node.moduleSpecifier && node.flags & 8388608 /* Ambient */;
-                    if (node.parent.kind !== 290 /* SourceFile */ && !inAmbientExternalModule && !inAmbientNamespaceDeclaration) {
-                        error(node, ts.Diagnostics.Export_declarations_are_not_permitted_in_a_namespace);
-                    }
-                }
-                else {
-                    // export * from "foo"
-                    var moduleSymbol = resolveExternalModuleName(node, node.moduleSpecifier);
-                    if (moduleSymbol && hasExportAssignmentSymbol(moduleSymbol)) {
-                        error(node.moduleSpecifier, ts.Diagnostics.Module_0_uses_export_and_cannot_be_used_with_export_Asterisk, symbolToString(moduleSymbol));
+            checkSourceElement(node.typeExpression);
+        }
+        function checkJSDocTemplateTag(node) {
+            checkSourceElement(node.constraint);
+            for (var _i = 0, _a = node.typeParameters; _i < _a.length; _i++) {
+                var tp = _a[_i];
+                checkSourceElement(tp);
+            }
+        }
+        function checkJSDocTypeTag(node) {
+            checkSourceElement(node.typeExpression);
+        }
+        function checkJSDocParameterTag(node) {
+            checkSourceElement(node.typeExpression);
+            if (!ts.getParameterSymbolFromJSDoc(node)) {
+                var decl = ts.getHostSignatureFromJSDoc(node);
+                // don't issue an error for invalid hosts -- just functions --
+                // and give a better error message when the host function mentions `arguments`
+                // but the tag doesn't have an array type
+                if (decl) {
+                    var i = ts.getJSDocTags(decl).filter(ts.isJSDocParameterTag).indexOf(node);
+                    if (i > -1 && i < decl.parameters.length && ts.isBindingPattern(decl.parameters[i].name)) {
+                        return;
                     }
-                    else if (node.exportClause) {
-                        checkAliasSymbol(node.exportClause);
+                    if (!containsArgumentsReference(decl)) {
+                        if (ts.isQualifiedName(node.name)) {
+                            error(node.name, ts.Diagnostics.Qualified_name_0_is_not_allowed_without_a_leading_param_object_1, ts.entityNameToString(node.name), ts.entityNameToString(node.name.left));
+                        }
+                        else {
+                            error(node.name, ts.Diagnostics.JSDoc_param_tag_has_name_0_but_there_is_no_parameter_with_that_name, ts.idText(node.name));
+                        }
                     }
-                    if (moduleKind !== ts.ModuleKind.System && moduleKind < ts.ModuleKind.ES2015) {
-                        checkExternalEmitHelpers(node, 65536 /* ExportStar */);
+                    else if (ts.findLast(ts.getJSDocTags(decl), ts.isJSDocParameterTag) === node &&
+                        node.typeExpression && node.typeExpression.type &&
+                        !isArrayType(getTypeFromTypeNode(node.typeExpression.type))) {
+                        error(node.name, ts.Diagnostics.JSDoc_param_tag_has_name_0_but_there_is_no_parameter_with_that_name_It_would_match_arguments_if_it_had_an_array_type, ts.idText(node.name.kind === 157 /* QualifiedName */ ? node.name.right : node.name));
                     }
                 }
             }
         }
-        function checkGrammarExportDeclaration(node) {
-            var _a;
-            var isTypeOnlyExportStar = node.isTypeOnly && ((_a = node.exportClause) === null || _a === void 0 ? void 0 : _a.kind) !== 261 /* NamedExports */;
-            if (isTypeOnlyExportStar) {
-                grammarErrorOnNode(node, ts.Diagnostics.Only_named_exports_may_use_export_type);
-            }
-            return !isTypeOnlyExportStar;
+        function checkJSDocPropertyTag(node) {
+            checkSourceElement(node.typeExpression);
         }
-        function checkGrammarModuleElementContext(node, errorMessage) {
-            var isInAppropriateContext = node.parent.kind === 290 /* SourceFile */ || node.parent.kind === 250 /* ModuleBlock */ || node.parent.kind === 249 /* ModuleDeclaration */;
-            if (!isInAppropriateContext) {
-                grammarErrorOnFirstToken(node, errorMessage);
+        function checkJSDocFunctionType(node) {
+            if (produceDiagnostics && !node.type && !ts.isJSDocConstructSignature(node)) {
+                reportImplicitAny(node, anyType);
             }
-            return !isInAppropriateContext;
-        }
-        function importClauseContainsReferencedImport(importClause) {
-            return ts.forEachImportClauseDeclaration(importClause, function (declaration) {
-                return !!getSymbolOfNode(declaration).isReferenced;
-            });
-        }
-        function importClauseContainsConstEnumUsedAsValue(importClause) {
-            return ts.forEachImportClauseDeclaration(importClause, function (declaration) {
-                return !!getSymbolLinks(getSymbolOfNode(declaration)).constEnumReferenced;
-            });
+            checkSignatureDeclaration(node);
         }
-        function checkImportsForTypeOnlyConversion(sourceFile) {
-            for (var _i = 0, _a = sourceFile.statements; _i < _a.length; _i++) {
-                var statement = _a[_i];
-                if (ts.isImportDeclaration(statement) &&
-                    statement.importClause &&
-                    !statement.importClause.isTypeOnly &&
-                    importClauseContainsReferencedImport(statement.importClause) &&
-                    !isReferencedAliasDeclaration(statement.importClause, /*checkChildren*/ true) &&
-                    !importClauseContainsConstEnumUsedAsValue(statement.importClause)) {
-                    error(statement, ts.Diagnostics.This_import_is_never_used_as_a_value_and_must_use_import_type_because_the_importsNotUsedAsValues_is_set_to_error);
-                }
+        function checkJSDocImplementsTag(node) {
+            var classLike = ts.getEffectiveJSDocHost(node);
+            if (!classLike || !ts.isClassDeclaration(classLike) && !ts.isClassExpression(classLike)) {
+                error(classLike, ts.Diagnostics.JSDoc_0_is_not_attached_to_a_class, ts.idText(node.tagName));
             }
         }
-        function checkExportSpecifier(node) {
-            checkAliasSymbol(node);
-            if (ts.getEmitDeclarations(compilerOptions)) {
-                collectLinkedAliases(node.propertyName || node.name, /*setVisibility*/ true);
+        function checkJSDocAugmentsTag(node) {
+            var classLike = ts.getEffectiveJSDocHost(node);
+            if (!classLike || !ts.isClassDeclaration(classLike) && !ts.isClassExpression(classLike)) {
+                error(classLike, ts.Diagnostics.JSDoc_0_is_not_attached_to_a_class, ts.idText(node.tagName));
+                return;
             }
-            if (!node.parent.parent.moduleSpecifier) {
-                var exportedName = node.propertyName || node.name;
-                // find immediate value referenced by exported name (SymbolFlags.Alias is set so we don't chase down aliases)
-                var symbol = resolveName(exportedName, exportedName.escapedText, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */, 
-                /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true);
-                if (symbol && (symbol === undefinedSymbol || symbol === globalThisSymbol || isGlobalSourceFile(getDeclarationContainer(symbol.declarations[0])))) {
-                    error(exportedName, ts.Diagnostics.Cannot_export_0_Only_local_declarations_can_be_exported_from_a_module, ts.idText(exportedName));
-                }
-                else {
-                    markExportAsReferenced(node);
-                    var target = symbol && (symbol.flags & 2097152 /* Alias */ ? resolveAlias(symbol) : symbol);
-                    if (!target || target === unknownSymbol || target.flags & 111551 /* Value */) {
-                        checkExpressionCached(node.propertyName || node.name);
-                    }
+            var augmentsTags = ts.getJSDocTags(classLike).filter(ts.isJSDocAugmentsTag);
+            ts.Debug.assert(augmentsTags.length > 0);
+            if (augmentsTags.length > 1) {
+                error(augmentsTags[1], ts.Diagnostics.Class_declarations_cannot_have_more_than_one_augments_or_extends_tag);
+            }
+            var name = getIdentifierFromEntityNameExpression(node.class.expression);
+            var extend = ts.getClassExtendsHeritageElement(classLike);
+            if (extend) {
+                var className = getIdentifierFromEntityNameExpression(extend.expression);
+                if (className && name.escapedText !== className.escapedText) {
+                    error(name, ts.Diagnostics.JSDoc_0_1_does_not_match_the_extends_2_clause, ts.idText(node.tagName), ts.idText(name), ts.idText(className));
                 }
             }
         }
-        function checkExportAssignment(node) {
-            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_export_assignment_can_only_be_used_in_a_module)) {
-                // If we hit an export assignment in an illegal context, just bail out to avoid cascading errors.
-                return;
+        function getIdentifierFromEntityNameExpression(node) {
+            switch (node.kind) {
+                case 78 /* Identifier */:
+                    return node;
+                case 201 /* PropertyAccessExpression */:
+                    return node.name;
+                default:
+                    return undefined;
             }
-            var container = node.parent.kind === 290 /* SourceFile */ ? node.parent : node.parent.parent;
-            if (container.kind === 249 /* ModuleDeclaration */ && !ts.isAmbientModule(container)) {
-                if (node.isExportEquals) {
-                    error(node, ts.Diagnostics.An_export_assignment_cannot_be_used_in_a_namespace);
+        }
+        function checkFunctionOrMethodDeclaration(node) {
+            checkDecorators(node);
+            checkSignatureDeclaration(node);
+            var functionFlags = ts.getFunctionFlags(node);
+            // Do not use hasDynamicName here, because that returns false for well known symbols.
+            // We want to perform checkComputedPropertyName for all computed properties, including
+            // well known symbols.
+            if (node.name && node.name.kind === 158 /* ComputedPropertyName */) {
+                // This check will account for methods in class/interface declarations,
+                // as well as accessors in classes/object literals
+                checkComputedPropertyName(node.name);
+            }
+            if (!hasNonBindableDynamicName(node)) {
+                // first we want to check the local symbol that contain this declaration
+                // - if node.localSymbol !== undefined - this is current declaration is exported and localSymbol points to the local symbol
+                // - if node.localSymbol === undefined - this node is non-exported so we can just pick the result of getSymbolOfNode
+                var symbol = getSymbolOfNode(node);
+                var localSymbol = node.localSymbol || symbol;
+                // Since the javascript won't do semantic analysis like typescript,
+                // if the javascript file comes before the typescript file and both contain same name functions,
+                // checkFunctionOrConstructorSymbol wouldn't be called if we didnt ignore javascript function.
+                var firstDeclaration = ts.find(localSymbol.declarations, 
+                // Get first non javascript function declaration
+                function (declaration) { return declaration.kind === node.kind && !(declaration.flags & 131072 /* JavaScriptFile */); });
+                // Only type check the symbol once
+                if (node === firstDeclaration) {
+                    checkFunctionOrConstructorSymbol(localSymbol);
                 }
-                else {
-                    error(node, ts.Diagnostics.A_default_export_can_only_be_used_in_an_ECMAScript_style_module);
+                if (symbol.parent) {
+                    // run check on export symbol to check that modifiers agree across all exported declarations
+                    checkFunctionOrConstructorSymbol(symbol);
                 }
-                return;
-            }
-            // Grammar checking
-            if (!checkGrammarDecoratorsAndModifiers(node) && ts.hasModifiers(node)) {
-                grammarErrorOnFirstToken(node, ts.Diagnostics.An_export_assignment_cannot_have_modifiers);
             }
-            if (node.expression.kind === 75 /* Identifier */) {
-                var id = node.expression;
-                var sym = resolveEntityName(id, 67108863 /* All */, /*ignoreErrors*/ true, /*dontResolveAlias*/ true, node);
-                if (sym) {
-                    markAliasReferenced(sym, id);
-                    // If not a value, we're interpreting the identifier as a type export, along the lines of (`export { Id as default }`)
-                    var target = sym.flags & 2097152 /* Alias */ ? resolveAlias(sym) : sym;
-                    if (target === unknownSymbol || target.flags & 111551 /* Value */) {
-                        // However if it is a value, we need to check it's being used correctly
-                        checkExpressionCached(node.expression);
-                    }
+            var body = node.kind === 164 /* MethodSignature */ ? undefined : node.body;
+            checkSourceElement(body);
+            checkAllCodePathsInNonVoidFunctionReturnOrThrow(node, getReturnTypeFromAnnotation(node));
+            if (produceDiagnostics && !ts.getEffectiveReturnTypeNode(node)) {
+                // Report an implicit any error if there is no body, no explicit return type, and node is not a private method
+                // in an ambient context
+                if (ts.nodeIsMissing(body) && !isPrivateWithinAmbient(node)) {
+                    reportImplicitAny(node, anyType);
                 }
-                if (ts.getEmitDeclarations(compilerOptions)) {
-                    collectLinkedAliases(node.expression, /*setVisibility*/ true);
+                if (functionFlags & 1 /* Generator */ && ts.nodeIsPresent(body)) {
+                    // A generator with a body and no type annotation can still cause errors. It can error if the
+                    // yielded values have no common supertype, or it can give an implicit any error if it has no
+                    // yielded values. The only way to trigger these errors is to try checking its return type.
+                    getReturnTypeOfSignature(getSignatureFromDeclaration(node));
                 }
             }
-            else {
-                checkExpressionCached(node.expression);
-            }
-            checkExternalModuleExports(container);
-            if ((node.flags & 8388608 /* Ambient */) && !ts.isEntityNameExpression(node.expression)) {
-                grammarErrorOnNode(node.expression, ts.Diagnostics.The_expression_of_an_export_assignment_must_be_an_identifier_or_qualified_name_in_an_ambient_context);
+            // A js function declaration can have a @type tag instead of a return type node, but that type must have a call signature
+            if (ts.isInJSFile(node)) {
+                var typeTag = ts.getJSDocTypeTag(node);
+                if (typeTag && typeTag.typeExpression && !getContextualCallSignature(getTypeFromTypeNode(typeTag.typeExpression), node)) {
+                    error(typeTag, ts.Diagnostics.The_type_of_a_function_declaration_must_match_the_function_s_signature);
+                }
             }
-            if (node.isExportEquals && !(node.flags & 8388608 /* Ambient */)) {
-                if (moduleKind >= ts.ModuleKind.ES2015) {
-                    // export assignment is not supported in es6 modules
-                    grammarErrorOnNode(node, ts.Diagnostics.Export_assignment_cannot_be_used_when_targeting_ECMAScript_modules_Consider_using_export_default_or_another_module_format_instead);
+        }
+        function registerForUnusedIdentifiersCheck(node) {
+            // May be in a call such as getTypeOfNode that happened to call this. But potentiallyUnusedIdentifiers is only defined in the scope of `checkSourceFile`.
+            if (produceDiagnostics) {
+                var sourceFile = ts.getSourceFileOfNode(node);
+                var potentiallyUnusedIdentifiers = allPotentiallyUnusedIdentifiers.get(sourceFile.path);
+                if (!potentiallyUnusedIdentifiers) {
+                    potentiallyUnusedIdentifiers = [];
+                    allPotentiallyUnusedIdentifiers.set(sourceFile.path, potentiallyUnusedIdentifiers);
                 }
-                else if (moduleKind === ts.ModuleKind.System) {
-                    // system modules does not support export assignment
-                    grammarErrorOnNode(node, ts.Diagnostics.Export_assignment_is_not_supported_when_module_flag_is_system);
+                // TODO: GH#22580
+                // Debug.assert(addToSeen(seenPotentiallyUnusedIdentifiers, getNodeId(node)), "Adding potentially-unused identifier twice");
+                potentiallyUnusedIdentifiers.push(node);
+            }
+        }
+        function checkUnusedIdentifiers(potentiallyUnusedIdentifiers, addDiagnostic) {
+            for (var _i = 0, potentiallyUnusedIdentifiers_1 = potentiallyUnusedIdentifiers; _i < potentiallyUnusedIdentifiers_1.length; _i++) {
+                var node = potentiallyUnusedIdentifiers_1[_i];
+                switch (node.kind) {
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                        checkUnusedClassMembers(node, addDiagnostic);
+                        checkUnusedTypeParameters(node, addDiagnostic);
+                        break;
+                    case 297 /* SourceFile */:
+                    case 256 /* ModuleDeclaration */:
+                    case 230 /* Block */:
+                    case 258 /* CaseBlock */:
+                    case 237 /* ForStatement */:
+                    case 238 /* ForInStatement */:
+                    case 239 /* ForOfStatement */:
+                        checkUnusedLocalsAndParameters(node, addDiagnostic);
+                        break;
+                    case 166 /* Constructor */:
+                    case 208 /* FunctionExpression */:
+                    case 251 /* FunctionDeclaration */:
+                    case 209 /* ArrowFunction */:
+                    case 165 /* MethodDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        if (node.body) { // Don't report unused parameters in overloads
+                            checkUnusedLocalsAndParameters(node, addDiagnostic);
+                        }
+                        checkUnusedTypeParameters(node, addDiagnostic);
+                        break;
+                    case 164 /* MethodSignature */:
+                    case 169 /* CallSignature */:
+                    case 170 /* ConstructSignature */:
+                    case 174 /* FunctionType */:
+                    case 175 /* ConstructorType */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                        checkUnusedTypeParameters(node, addDiagnostic);
+                        break;
+                    case 185 /* InferType */:
+                        checkUnusedInferTypeParameter(node, addDiagnostic);
+                        break;
+                    default:
+                        ts.Debug.assertNever(node, "Node should not have been registered for unused identifiers check");
                 }
             }
         }
-        function hasExportedMembers(moduleSymbol) {
-            return ts.forEachEntry(moduleSymbol.exports, function (_, id) { return id !== "export="; });
+        function errorUnusedLocal(declaration, name, addDiagnostic) {
+            var node = ts.getNameOfDeclaration(declaration) || declaration;
+            var message = isTypeDeclaration(declaration) ? ts.Diagnostics._0_is_declared_but_never_used : ts.Diagnostics._0_is_declared_but_its_value_is_never_read;
+            addDiagnostic(declaration, 0 /* Local */, ts.createDiagnosticForNode(node, message, name));
         }
-        function checkExternalModuleExports(node) {
-            var moduleSymbol = getSymbolOfNode(node);
-            var links = getSymbolLinks(moduleSymbol);
-            if (!links.exportsChecked) {
-                var exportEqualsSymbol = moduleSymbol.exports.get("export=");
-                if (exportEqualsSymbol && hasExportedMembers(moduleSymbol)) {
-                    var declaration = getDeclarationOfAliasSymbol(exportEqualsSymbol) || exportEqualsSymbol.valueDeclaration;
-                    if (!isTopLevelInExternalModuleAugmentation(declaration) && !ts.isInJSFile(declaration)) {
-                        error(declaration, ts.Diagnostics.An_export_assignment_cannot_be_used_in_a_module_with_other_exported_elements);
-                    }
-                }
-                // Checks for export * conflicts
-                var exports_2 = getExportsOfModule(moduleSymbol);
-                if (exports_2) {
-                    exports_2.forEach(function (_a, id) {
-                        var declarations = _a.declarations, flags = _a.flags;
-                        if (id === "__export") {
-                            return;
-                        }
-                        // ECMA262: 15.2.1.1 It is a Syntax Error if the ExportedNames of ModuleItemList contains any duplicate entries.
-                        // (TS Exceptions: namespaces, function overloads, enums, and interfaces)
-                        if (flags & (1920 /* Namespace */ | 64 /* Interface */ | 384 /* Enum */)) {
-                            return;
+        function isIdentifierThatStartsWithUnderscore(node) {
+            return ts.isIdentifier(node) && ts.idText(node).charCodeAt(0) === 95 /* _ */;
+        }
+        function checkUnusedClassMembers(node, addDiagnostic) {
+            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                var member = _a[_i];
+                switch (member.kind) {
+                    case 165 /* MethodDeclaration */:
+                    case 163 /* PropertyDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        if (member.kind === 168 /* SetAccessor */ && member.symbol.flags & 32768 /* GetAccessor */) {
+                            // Already would have reported an error on the getter.
+                            break;
                         }
-                        var exportedDeclarationsCount = ts.countWhere(declarations, isNotOverloadAndNotAccessor);
-                        if (flags & 524288 /* TypeAlias */ && exportedDeclarationsCount <= 2) {
-                            // it is legal to merge type alias with other values
-                            // so count should be either 1 (just type alias) or 2 (type alias + merged value)
-                            return;
+                        var symbol = getSymbolOfNode(member);
+                        if (!symbol.isReferenced
+                            && (ts.hasEffectiveModifier(member, 8 /* Private */) || ts.isNamedDeclaration(member) && ts.isPrivateIdentifier(member.name))
+                            && !(member.flags & 8388608 /* Ambient */)) {
+                            addDiagnostic(member, 0 /* Local */, ts.createDiagnosticForNode(member.name, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, symbolToString(symbol)));
                         }
-                        if (exportedDeclarationsCount > 1) {
-                            for (var _i = 0, declarations_9 = declarations; _i < declarations_9.length; _i++) {
-                                var declaration = declarations_9[_i];
-                                if (isNotOverload(declaration)) {
-                                    diagnostics.add(ts.createDiagnosticForNode(declaration, ts.Diagnostics.Cannot_redeclare_exported_variable_0, ts.unescapeLeadingUnderscores(id)));
-                                }
+                        break;
+                    case 166 /* Constructor */:
+                        for (var _b = 0, _c = member.parameters; _b < _c.length; _b++) {
+                            var parameter = _c[_b];
+                            if (!parameter.symbol.isReferenced && ts.hasSyntacticModifier(parameter, 8 /* Private */)) {
+                                addDiagnostic(parameter, 0 /* Local */, ts.createDiagnosticForNode(parameter.name, ts.Diagnostics.Property_0_is_declared_but_its_value_is_never_read, ts.symbolName(parameter.symbol)));
                             }
                         }
-                    });
-                }
-                links.exportsChecked = true;
-            }
-        }
-        function checkSourceElement(node) {
-            if (node) {
-                var saveCurrentNode = currentNode;
-                currentNode = node;
-                instantiationCount = 0;
-                checkSourceElementWorker(node);
-                currentNode = saveCurrentNode;
-            }
-        }
-        function checkSourceElementWorker(node) {
-            if (ts.isInJSFile(node)) {
-                ts.forEach(node.jsDoc, function (_a) {
-                    var tags = _a.tags;
-                    return ts.forEach(tags, checkSourceElement);
-                });
-            }
-            var kind = node.kind;
-            if (cancellationToken) {
-                // Only bother checking on a few construct kinds.  We don't want to be excessively
-                // hitting the cancellation token on every node we check.
-                switch (kind) {
-                    case 249 /* ModuleDeclaration */:
-                    case 245 /* ClassDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 244 /* FunctionDeclaration */:
-                        cancellationToken.throwIfCancellationRequested();
+                        break;
+                    case 171 /* IndexSignature */:
+                    case 229 /* SemicolonClassElement */:
+                        // Can't be private
+                        break;
+                    default:
+                        ts.Debug.fail();
                 }
             }
-            if (kind >= 225 /* FirstStatement */ && kind <= 241 /* LastStatement */ && node.flowNode && !isReachableFlowNode(node.flowNode)) {
-                errorOrSuggestion(compilerOptions.allowUnreachableCode === false, node, ts.Diagnostics.Unreachable_code_detected);
-            }
-            switch (kind) {
-                case 155 /* TypeParameter */:
-                    return checkTypeParameter(node);
-                case 156 /* Parameter */:
-                    return checkParameter(node);
-                case 159 /* PropertyDeclaration */:
-                    return checkPropertyDeclaration(node);
-                case 158 /* PropertySignature */:
-                    return checkPropertySignature(node);
-                case 170 /* FunctionType */:
-                case 171 /* ConstructorType */:
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 167 /* IndexSignature */:
-                    return checkSignatureDeclaration(node);
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    return checkMethodDeclaration(node);
-                case 162 /* Constructor */:
-                    return checkConstructorDeclaration(node);
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    return checkAccessorDeclaration(node);
-                case 169 /* TypeReference */:
-                    return checkTypeReferenceNode(node);
-                case 168 /* TypePredicate */:
-                    return checkTypePredicate(node);
-                case 172 /* TypeQuery */:
-                    return checkTypeQuery(node);
-                case 173 /* TypeLiteral */:
-                    return checkTypeLiteral(node);
-                case 174 /* ArrayType */:
-                    return checkArrayType(node);
-                case 175 /* TupleType */:
-                    return checkTupleType(node);
-                case 178 /* UnionType */:
-                case 179 /* IntersectionType */:
-                    return checkUnionOrIntersectionType(node);
-                case 182 /* ParenthesizedType */:
-                case 176 /* OptionalType */:
-                case 177 /* RestType */:
-                    return checkSourceElement(node.type);
-                case 183 /* ThisType */:
-                    return checkThisType(node);
-                case 184 /* TypeOperator */:
-                    return checkTypeOperator(node);
-                case 180 /* ConditionalType */:
-                    return checkConditionalType(node);
-                case 181 /* InferType */:
-                    return checkInferType(node);
-                case 188 /* ImportType */:
-                    return checkImportType(node);
-                case 307 /* JSDocAugmentsTag */:
-                    return checkJSDocAugmentsTag(node);
-                case 308 /* JSDocImplementsTag */:
-                    return checkJSDocImplementsTag(node);
-                case 322 /* JSDocTypedefTag */:
-                case 315 /* JSDocCallbackTag */:
-                case 316 /* JSDocEnumTag */:
-                    return checkJSDocTypeAliasTag(node);
-                case 321 /* JSDocTemplateTag */:
-                    return checkJSDocTemplateTag(node);
-                case 320 /* JSDocTypeTag */:
-                    return checkJSDocTypeTag(node);
-                case 317 /* JSDocParameterTag */:
-                    return checkJSDocParameterTag(node);
-                case 323 /* JSDocPropertyTag */:
-                    return checkJSDocPropertyTag(node);
-                case 300 /* JSDocFunctionType */:
-                    checkJSDocFunctionType(node);
-                // falls through
-                case 298 /* JSDocNonNullableType */:
-                case 297 /* JSDocNullableType */:
-                case 295 /* JSDocAllType */:
-                case 296 /* JSDocUnknownType */:
-                case 304 /* JSDocTypeLiteral */:
-                    checkJSDocTypeIsInJsFile(node);
-                    ts.forEachChild(node, checkSourceElement);
-                    return;
-                case 301 /* JSDocVariadicType */:
-                    checkJSDocVariadicType(node);
-                    return;
-                case 294 /* JSDocTypeExpression */:
-                    return checkSourceElement(node.type);
-                case 185 /* IndexedAccessType */:
-                    return checkIndexedAccessType(node);
-                case 186 /* MappedType */:
-                    return checkMappedType(node);
-                case 244 /* FunctionDeclaration */:
-                    return checkFunctionDeclaration(node);
-                case 223 /* Block */:
-                case 250 /* ModuleBlock */:
-                    return checkBlock(node);
-                case 225 /* VariableStatement */:
-                    return checkVariableStatement(node);
-                case 226 /* ExpressionStatement */:
-                    return checkExpressionStatement(node);
-                case 227 /* IfStatement */:
-                    return checkIfStatement(node);
-                case 228 /* DoStatement */:
-                    return checkDoStatement(node);
-                case 229 /* WhileStatement */:
-                    return checkWhileStatement(node);
-                case 230 /* ForStatement */:
-                    return checkForStatement(node);
-                case 231 /* ForInStatement */:
-                    return checkForInStatement(node);
-                case 232 /* ForOfStatement */:
-                    return checkForOfStatement(node);
-                case 233 /* ContinueStatement */:
-                case 234 /* BreakStatement */:
-                    return checkBreakOrContinueStatement(node);
-                case 235 /* ReturnStatement */:
-                    return checkReturnStatement(node);
-                case 236 /* WithStatement */:
-                    return checkWithStatement(node);
-                case 237 /* SwitchStatement */:
-                    return checkSwitchStatement(node);
-                case 238 /* LabeledStatement */:
-                    return checkLabeledStatement(node);
-                case 239 /* ThrowStatement */:
-                    return checkThrowStatement(node);
-                case 240 /* TryStatement */:
-                    return checkTryStatement(node);
-                case 242 /* VariableDeclaration */:
-                    return checkVariableDeclaration(node);
-                case 191 /* BindingElement */:
-                    return checkBindingElement(node);
-                case 245 /* ClassDeclaration */:
-                    return checkClassDeclaration(node);
-                case 246 /* InterfaceDeclaration */:
-                    return checkInterfaceDeclaration(node);
-                case 247 /* TypeAliasDeclaration */:
-                    return checkTypeAliasDeclaration(node);
-                case 248 /* EnumDeclaration */:
-                    return checkEnumDeclaration(node);
-                case 249 /* ModuleDeclaration */:
-                    return checkModuleDeclaration(node);
-                case 254 /* ImportDeclaration */:
-                    return checkImportDeclaration(node);
-                case 253 /* ImportEqualsDeclaration */:
-                    return checkImportEqualsDeclaration(node);
-                case 260 /* ExportDeclaration */:
-                    return checkExportDeclaration(node);
-                case 259 /* ExportAssignment */:
-                    return checkExportAssignment(node);
-                case 224 /* EmptyStatement */:
-                case 241 /* DebuggerStatement */:
-                    checkGrammarStatementInAmbientContext(node);
-                    return;
-                case 264 /* MissingDeclaration */:
-                    return checkMissingDeclaration(node);
-            }
         }
-        function checkJSDocTypeIsInJsFile(node) {
-            if (!ts.isInJSFile(node)) {
-                grammarErrorOnNode(node, ts.Diagnostics.JSDoc_types_can_only_be_used_inside_documentation_comments);
+        function checkUnusedInferTypeParameter(node, addDiagnostic) {
+            var typeParameter = node.typeParameter;
+            if (isTypeParameterUnused(typeParameter)) {
+                addDiagnostic(node, 1 /* Parameter */, ts.createDiagnosticForNode(node, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, ts.idText(typeParameter.name)));
             }
         }
-        function checkJSDocVariadicType(node) {
-            checkJSDocTypeIsInJsFile(node);
-            checkSourceElement(node.type);
-            // Only legal location is in the *last* parameter tag or last parameter of a JSDoc function.
-            var parent = node.parent;
-            if (ts.isParameter(parent) && ts.isJSDocFunctionType(parent.parent)) {
-                if (ts.last(parent.parent.parameters) !== parent) {
-                    error(node, ts.Diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list);
-                }
-                return;
-            }
-            if (!ts.isJSDocTypeExpression(parent)) {
-                error(node, ts.Diagnostics.JSDoc_may_only_appear_in_the_last_parameter_of_a_signature);
-            }
-            var paramTag = node.parent.parent;
-            if (!ts.isJSDocParameterTag(paramTag)) {
-                error(node, ts.Diagnostics.JSDoc_may_only_appear_in_the_last_parameter_of_a_signature);
-                return;
-            }
-            var param = ts.getParameterSymbolFromJSDoc(paramTag);
-            if (!param) {
-                // We will error in `checkJSDocParameterTag`.
+        function checkUnusedTypeParameters(node, addDiagnostic) {
+            // Only report errors on the last declaration for the type parameter container;
+            // this ensures that all uses have been accounted for.
+            if (ts.last(getSymbolOfNode(node).declarations) !== node)
                 return;
-            }
-            var host = ts.getHostSignatureFromJSDoc(paramTag);
-            if (!host || ts.last(host.parameters).symbol !== param) {
-                error(node, ts.Diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list);
-            }
-        }
-        function getTypeFromJSDocVariadicType(node) {
-            var type = getTypeFromTypeNode(node.type);
-            var parent = node.parent;
-            var paramTag = node.parent.parent;
-            if (ts.isJSDocTypeExpression(node.parent) && ts.isJSDocParameterTag(paramTag)) {
-                // Else we will add a diagnostic, see `checkJSDocVariadicType`.
-                var host_1 = ts.getHostSignatureFromJSDoc(paramTag);
-                if (host_1) {
-                    /*
-                    Only return an array type if the corresponding parameter is marked as a rest parameter, or if there are no parameters.
-                    So in the following situation we will not create an array type:
-                        /** @param {...number} a * /
-                        function f(a) {}
-                    Because `a` will just be of type `number | undefined`. A synthetic `...args` will also be added, which *will* get an array type.
-                    */
-                    var lastParamDeclaration = ts.lastOrUndefined(host_1.parameters);
-                    var symbol = ts.getParameterSymbolFromJSDoc(paramTag);
-                    if (!lastParamDeclaration ||
-                        symbol && lastParamDeclaration.symbol === symbol && ts.isRestParameter(lastParamDeclaration)) {
-                        return createArrayType(type);
+            var typeParameters = ts.getEffectiveTypeParameterDeclarations(node);
+            var seenParentsWithEveryUnused = new ts.Set();
+            for (var _i = 0, typeParameters_3 = typeParameters; _i < typeParameters_3.length; _i++) {
+                var typeParameter = typeParameters_3[_i];
+                if (!isTypeParameterUnused(typeParameter))
+                    continue;
+                var name = ts.idText(typeParameter.name);
+                var parent = typeParameter.parent;
+                if (parent.kind !== 185 /* InferType */ && parent.typeParameters.every(isTypeParameterUnused)) {
+                    if (ts.tryAddToSet(seenParentsWithEveryUnused, parent)) {
+                        var range = ts.isJSDocTemplateTag(parent)
+                            // Whole @template tag
+                            ? ts.rangeOfNode(parent)
+                            // Include the `<>` in the error message
+                            : ts.rangeOfTypeParameters(parent.typeParameters);
+                        var only = parent.typeParameters.length === 1;
+                        var message = only ? ts.Diagnostics._0_is_declared_but_its_value_is_never_read : ts.Diagnostics.All_type_parameters_are_unused;
+                        var arg0 = only ? name : undefined;
+                        addDiagnostic(typeParameter, 1 /* Parameter */, ts.createFileDiagnostic(ts.getSourceFileOfNode(parent), range.pos, range.end - range.pos, message, arg0));
                     }
                 }
+                else {
+                    addDiagnostic(typeParameter, 1 /* Parameter */, ts.createDiagnosticForNode(typeParameter, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, name));
+                }
             }
-            if (ts.isParameter(parent) && ts.isJSDocFunctionType(parent.parent)) {
-                return createArrayType(type);
-            }
-            return addOptionality(type);
         }
-        // Function and class expression bodies are checked after all statements in the enclosing body. This is
-        // to ensure constructs like the following are permitted:
-        //     const foo = function () {
-        //        const s = foo();
-        //        return "hello";
-        //     }
-        // Here, performing a full type check of the body of the function expression whilst in the process of
-        // determining the type of foo would cause foo to be given type any because of the recursive reference.
-        // Delaying the type check of the body ensures foo has been assigned a type.
-        function checkNodeDeferred(node) {
-            var enclosingFile = ts.getSourceFileOfNode(node);
-            var links = getNodeLinks(enclosingFile);
-            if (!(links.flags & 1 /* TypeChecked */)) {
-                links.deferredNodes = links.deferredNodes || ts.createMap();
-                var id = "" + getNodeId(node);
-                links.deferredNodes.set(id, node);
-            }
+        function isTypeParameterUnused(typeParameter) {
+            return !(getMergedSymbol(typeParameter.symbol).isReferenced & 262144 /* TypeParameter */) && !isIdentifierThatStartsWithUnderscore(typeParameter.name);
         }
-        function checkDeferredNodes(context) {
-            var links = getNodeLinks(context);
-            if (links.deferredNodes) {
-                links.deferredNodes.forEach(checkDeferredNode);
+        function addToGroup(map, key, value, getKey) {
+            var keyString = String(getKey(key));
+            var group = map.get(keyString);
+            if (group) {
+                group[1].push(value);
             }
-        }
-        function checkDeferredNode(node) {
-            var saveCurrentNode = currentNode;
-            currentNode = node;
-            instantiationCount = 0;
-            switch (node.kind) {
-                case 196 /* CallExpression */:
-                case 197 /* NewExpression */:
-                case 198 /* TaggedTemplateExpression */:
-                case 157 /* Decorator */:
-                case 268 /* JsxOpeningElement */:
-                    // These node kinds are deferred checked when overload resolution fails
-                    // To save on work, we ensure the arguments are checked just once, in
-                    // a deferred way
-                    resolveUntypedCall(node);
-                    break;
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    checkFunctionExpressionOrObjectLiteralMethodDeferred(node);
-                    break;
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    checkAccessorDeclaration(node);
-                    break;
-                case 214 /* ClassExpression */:
-                    checkClassExpressionDeferred(node);
-                    break;
-                case 267 /* JsxSelfClosingElement */:
-                    checkJsxSelfClosingElementDeferred(node);
-                    break;
-                case 266 /* JsxElement */:
-                    checkJsxElementDeferred(node);
-                    break;
+            else {
+                map.set(keyString, [key, [value]]);
             }
-            currentNode = saveCurrentNode;
         }
-        function checkSourceFile(node) {
-            ts.performance.mark("beforeCheck");
-            checkSourceFileWorker(node);
-            ts.performance.mark("afterCheck");
-            ts.performance.measure("Check", "beforeCheck", "afterCheck");
+        function tryGetRootParameterDeclaration(node) {
+            return ts.tryCast(ts.getRootDeclaration(node), ts.isParameter);
         }
-        function unusedIsError(kind, isAmbient) {
-            if (isAmbient) {
-                return false;
-            }
-            switch (kind) {
-                case 0 /* Local */:
-                    return !!compilerOptions.noUnusedLocals;
-                case 1 /* Parameter */:
-                    return !!compilerOptions.noUnusedParameters;
-                default:
-                    return ts.Debug.assertNever(kind);
+        function isValidUnusedLocalDeclaration(declaration) {
+            if (ts.isBindingElement(declaration) && isIdentifierThatStartsWithUnderscore(declaration.name)) {
+                return !!ts.findAncestor(declaration.parent, function (ancestor) {
+                    return ts.isArrayBindingPattern(ancestor) || ts.isVariableDeclaration(ancestor) || ts.isVariableDeclarationList(ancestor) ? false :
+                        ts.isForOfStatement(ancestor) ? true : "quit";
+                });
             }
+            return ts.isAmbientModule(declaration) ||
+                (ts.isVariableDeclaration(declaration) && ts.isForInOrOfStatement(declaration.parent.parent) || isImportedDeclaration(declaration)) && isIdentifierThatStartsWithUnderscore(declaration.name);
         }
-        function getPotentiallyUnusedIdentifiers(sourceFile) {
-            return allPotentiallyUnusedIdentifiers.get(sourceFile.path) || ts.emptyArray;
-        }
-        // Fully type check a source file and collect the relevant diagnostics.
-        function checkSourceFileWorker(node) {
-            var links = getNodeLinks(node);
-            if (!(links.flags & 1 /* TypeChecked */)) {
-                if (ts.skipTypeChecking(node, compilerOptions, host)) {
+        function checkUnusedLocalsAndParameters(nodeWithLocals, addDiagnostic) {
+            // Ideally we could use the ImportClause directly as a key, but must wait until we have full ES6 maps. So must store key along with value.
+            var unusedImports = new ts.Map();
+            var unusedDestructures = new ts.Map();
+            var unusedVariables = new ts.Map();
+            nodeWithLocals.locals.forEach(function (local) {
+                // If it's purely a type parameter, ignore, will be checked in `checkUnusedTypeParameters`.
+                // If it's a type parameter merged with a parameter, check if the parameter-side is used.
+                if (local.flags & 262144 /* TypeParameter */ ? !(local.flags & 3 /* Variable */ && !(local.isReferenced & 3 /* Variable */)) : local.isReferenced || local.exportSymbol) {
                     return;
                 }
-                // Grammar checking
-                checkGrammarSourceFile(node);
-                ts.clear(potentialThisCollisions);
-                ts.clear(potentialNewTargetCollisions);
-                ts.clear(potentialWeakMapCollisions);
-                ts.forEach(node.statements, checkSourceElement);
-                checkSourceElement(node.endOfFileToken);
-                checkDeferredNodes(node);
-                if (ts.isExternalOrCommonJsModule(node)) {
-                    registerForUnusedIdentifiersCheck(node);
-                }
-                if (!node.isDeclarationFile && (compilerOptions.noUnusedLocals || compilerOptions.noUnusedParameters)) {
-                    checkUnusedIdentifiers(getPotentiallyUnusedIdentifiers(node), function (containingNode, kind, diag) {
-                        if (!ts.containsParseError(containingNode) && unusedIsError(kind, !!(containingNode.flags & 8388608 /* Ambient */))) {
-                            diagnostics.add(diag);
+                for (var _i = 0, _a = local.declarations; _i < _a.length; _i++) {
+                    var declaration = _a[_i];
+                    if (isValidUnusedLocalDeclaration(declaration)) {
+                        continue;
+                    }
+                    if (isImportedDeclaration(declaration)) {
+                        addToGroup(unusedImports, importClauseFromImported(declaration), declaration, getNodeId);
+                    }
+                    else if (ts.isBindingElement(declaration) && ts.isObjectBindingPattern(declaration.parent)) {
+                        // In `{ a, ...b }, `a` is considered used since it removes a property from `b`. `b` may still be unused though.
+                        var lastElement = ts.last(declaration.parent.elements);
+                        if (declaration === lastElement || !ts.last(declaration.parent.elements).dotDotDotToken) {
+                            addToGroup(unusedDestructures, declaration.parent, declaration, getNodeId);
                         }
-                    });
-                }
-                if (compilerOptions.importsNotUsedAsValues === 2 /* Error */ &&
-                    !node.isDeclarationFile &&
-                    ts.isExternalModule(node)) {
-                    checkImportsForTypeOnlyConversion(node);
+                    }
+                    else if (ts.isVariableDeclaration(declaration)) {
+                        addToGroup(unusedVariables, declaration.parent, declaration, getNodeId);
+                    }
+                    else {
+                        var parameter = local.valueDeclaration && tryGetRootParameterDeclaration(local.valueDeclaration);
+                        var name = local.valueDeclaration && ts.getNameOfDeclaration(local.valueDeclaration);
+                        if (parameter && name) {
+                            if (!ts.isParameterPropertyDeclaration(parameter, parameter.parent) && !ts.parameterIsThisKeyword(parameter) && !isIdentifierThatStartsWithUnderscore(name)) {
+                                addDiagnostic(parameter, 1 /* Parameter */, ts.createDiagnosticForNode(name, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, ts.symbolName(local)));
+                            }
+                        }
+                        else {
+                            errorUnusedLocal(declaration, ts.symbolName(local), addDiagnostic);
+                        }
+                    }
                 }
-                if (ts.isExternalOrCommonJsModule(node)) {
-                    checkExternalModuleExports(node);
+            });
+            unusedImports.forEach(function (_a) {
+                var importClause = _a[0], unuseds = _a[1];
+                var importDecl = importClause.parent;
+                var nDeclarations = (importClause.name ? 1 : 0) +
+                    (importClause.namedBindings ?
+                        (importClause.namedBindings.kind === 263 /* NamespaceImport */ ? 1 : importClause.namedBindings.elements.length)
+                        : 0);
+                if (nDeclarations === unuseds.length) {
+                    addDiagnostic(importDecl, 0 /* Local */, unuseds.length === 1
+                        ? ts.createDiagnosticForNode(importDecl, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, ts.idText(ts.first(unuseds).name))
+                        : ts.createDiagnosticForNode(importDecl, ts.Diagnostics.All_imports_in_import_declaration_are_unused));
                 }
-                if (potentialThisCollisions.length) {
-                    ts.forEach(potentialThisCollisions, checkIfThisIsCapturedInEnclosingScope);
-                    ts.clear(potentialThisCollisions);
+                else {
+                    for (var _i = 0, unuseds_1 = unuseds; _i < unuseds_1.length; _i++) {
+                        var unused = unuseds_1[_i];
+                        errorUnusedLocal(unused, ts.idText(unused.name), addDiagnostic);
+                    }
                 }
-                if (potentialNewTargetCollisions.length) {
-                    ts.forEach(potentialNewTargetCollisions, checkIfNewTargetIsCapturedInEnclosingScope);
-                    ts.clear(potentialNewTargetCollisions);
+            });
+            unusedDestructures.forEach(function (_a) {
+                var bindingPattern = _a[0], bindingElements = _a[1];
+                var kind = tryGetRootParameterDeclaration(bindingPattern.parent) ? 1 /* Parameter */ : 0 /* Local */;
+                if (bindingPattern.elements.length === bindingElements.length) {
+                    if (bindingElements.length === 1 && bindingPattern.parent.kind === 249 /* VariableDeclaration */ && bindingPattern.parent.parent.kind === 250 /* VariableDeclarationList */) {
+                        addToGroup(unusedVariables, bindingPattern.parent.parent, bindingPattern.parent, getNodeId);
+                    }
+                    else {
+                        addDiagnostic(bindingPattern, kind, bindingElements.length === 1
+                            ? ts.createDiagnosticForNode(bindingPattern, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(ts.first(bindingElements).name))
+                            : ts.createDiagnosticForNode(bindingPattern, ts.Diagnostics.All_destructured_elements_are_unused));
+                    }
                 }
-                if (potentialWeakMapCollisions.length) {
-                    ts.forEach(potentialWeakMapCollisions, checkWeakMapCollision);
-                    ts.clear(potentialWeakMapCollisions);
+                else {
+                    for (var _i = 0, bindingElements_1 = bindingElements; _i < bindingElements_1.length; _i++) {
+                        var e = bindingElements_1[_i];
+                        addDiagnostic(e, kind, ts.createDiagnosticForNode(e, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(e.name)));
+                    }
                 }
-                links.flags |= 1 /* TypeChecked */;
-            }
-        }
-        function getDiagnostics(sourceFile, ct) {
-            try {
-                // Record the cancellation token so it can be checked later on during checkSourceElement.
-                // Do this in a finally block so we can ensure that it gets reset back to nothing after
-                // this call is done.
-                cancellationToken = ct;
-                return getDiagnosticsWorker(sourceFile);
-            }
-            finally {
-                cancellationToken = undefined;
-            }
-        }
-        function getDiagnosticsWorker(sourceFile) {
-            throwIfNonDiagnosticsProducing();
-            if (sourceFile) {
-                // Some global diagnostics are deferred until they are needed and
-                // may not be reported in the first call to getGlobalDiagnostics.
-                // We should catch these changes and report them.
-                var previousGlobalDiagnostics = diagnostics.getGlobalDiagnostics();
-                var previousGlobalDiagnosticsSize = previousGlobalDiagnostics.length;
-                checkSourceFile(sourceFile);
-                var semanticDiagnostics = diagnostics.getDiagnostics(sourceFile.fileName);
-                var currentGlobalDiagnostics = diagnostics.getGlobalDiagnostics();
-                if (currentGlobalDiagnostics !== previousGlobalDiagnostics) {
-                    // If the arrays are not the same reference, new diagnostics were added.
-                    var deferredGlobalDiagnostics = ts.relativeComplement(previousGlobalDiagnostics, currentGlobalDiagnostics, ts.compareDiagnostics);
-                    return ts.concatenate(deferredGlobalDiagnostics, semanticDiagnostics);
+            });
+            unusedVariables.forEach(function (_a) {
+                var declarationList = _a[0], declarations = _a[1];
+                if (declarationList.declarations.length === declarations.length) {
+                    addDiagnostic(declarationList, 0 /* Local */, declarations.length === 1
+                        ? ts.createDiagnosticForNode(ts.first(declarations).name, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(ts.first(declarations).name))
+                        : ts.createDiagnosticForNode(declarationList.parent.kind === 232 /* VariableStatement */ ? declarationList.parent : declarationList, ts.Diagnostics.All_variables_are_unused));
                 }
-                else if (previousGlobalDiagnosticsSize === 0 && currentGlobalDiagnostics.length > 0) {
-                    // If the arrays are the same reference, but the length has changed, a single
-                    // new diagnostic was added as DiagnosticCollection attempts to reuse the
-                    // same array.
-                    return ts.concatenate(currentGlobalDiagnostics, semanticDiagnostics);
+                else {
+                    for (var _i = 0, declarations_5 = declarations; _i < declarations_5.length; _i++) {
+                        var decl = declarations_5[_i];
+                        addDiagnostic(decl, 0 /* Local */, ts.createDiagnosticForNode(decl, ts.Diagnostics._0_is_declared_but_its_value_is_never_read, bindingNameText(decl.name)));
+                    }
                 }
-                return semanticDiagnostics;
+            });
+        }
+        function bindingNameText(name) {
+            switch (name.kind) {
+                case 78 /* Identifier */:
+                    return ts.idText(name);
+                case 197 /* ArrayBindingPattern */:
+                case 196 /* ObjectBindingPattern */:
+                    return bindingNameText(ts.cast(ts.first(name.elements), ts.isBindingElement).name);
+                default:
+                    return ts.Debug.assertNever(name);
             }
-            // Global diagnostics are always added when a file is not provided to
-            // getDiagnostics
-            ts.forEach(host.getSourceFiles(), checkSourceFile);
-            return diagnostics.getDiagnostics();
         }
-        function getGlobalDiagnostics() {
-            throwIfNonDiagnosticsProducing();
-            return diagnostics.getGlobalDiagnostics();
+        function isImportedDeclaration(node) {
+            return node.kind === 262 /* ImportClause */ || node.kind === 265 /* ImportSpecifier */ || node.kind === 263 /* NamespaceImport */;
         }
-        function throwIfNonDiagnosticsProducing() {
-            if (!produceDiagnostics) {
-                throw new Error("Trying to get diagnostics from a type checker that does not produce them.");
-            }
+        function importClauseFromImported(decl) {
+            return decl.kind === 262 /* ImportClause */ ? decl : decl.kind === 263 /* NamespaceImport */ ? decl.parent : decl.parent.parent;
         }
-        // Language service support
-        function getSymbolsInScope(location, meaning) {
-            if (location.flags & 16777216 /* InWithStatement */) {
-                // We cannot answer semantic questions within a with block, do not proceed any further
-                return [];
-            }
-            var symbols = ts.createSymbolTable();
-            var isStatic = false;
-            populateSymbols();
-            symbols.delete("this" /* This */); // Not a symbol, a keyword
-            return symbolsToArray(symbols);
-            function populateSymbols() {
-                while (location) {
-                    if (location.locals && !isGlobalSourceFile(location)) {
-                        copySymbols(location.locals, meaning);
-                    }
-                    switch (location.kind) {
-                        case 290 /* SourceFile */:
-                            if (!ts.isExternalOrCommonJsModule(location))
-                                break;
-                        // falls through
-                        case 249 /* ModuleDeclaration */:
-                            copySymbols(getSymbolOfNode(location).exports, meaning & 2623475 /* ModuleMember */);
-                            break;
-                        case 248 /* EnumDeclaration */:
-                            copySymbols(getSymbolOfNode(location).exports, meaning & 8 /* EnumMember */);
-                            break;
-                        case 214 /* ClassExpression */:
-                            var className = location.name;
-                            if (className) {
-                                copySymbol(location.symbol, meaning);
-                            }
-                        // this fall-through is necessary because we would like to handle
-                        // type parameter inside class expression similar to how we handle it in classDeclaration and interface Declaration.
-                        // falls through
-                        case 245 /* ClassDeclaration */:
-                        case 246 /* InterfaceDeclaration */:
-                            // If we didn't come from static member of class or interface,
-                            // add the type parameters into the symbol table
-                            // (type parameters of classDeclaration/classExpression and interface are in member property of the symbol.
-                            // Note: that the memberFlags come from previous iteration.
-                            if (!isStatic) {
-                                copySymbols(getMembersOfSymbol(getSymbolOfNode(location)), meaning & 788968 /* Type */);
-                            }
-                            break;
-                        case 201 /* FunctionExpression */:
-                            var funcName = location.name;
-                            if (funcName) {
-                                copySymbol(location.symbol, meaning);
-                            }
-                            break;
-                    }
-                    if (ts.introducesArgumentsExoticObject(location)) {
-                        copySymbol(argumentsSymbol, meaning);
-                    }
-                    isStatic = ts.hasModifier(location, 32 /* Static */);
-                    location = location.parent;
-                }
-                copySymbols(globals, meaning);
-            }
-            /**
-             * Copy the given symbol into symbol tables if the symbol has the given meaning
-             * and it doesn't already existed in the symbol table
-             * @param key a key for storing in symbol table; if undefined, use symbol.name
-             * @param symbol the symbol to be added into symbol table
-             * @param meaning meaning of symbol to filter by before adding to symbol table
-             */
-            function copySymbol(symbol, meaning) {
-                if (ts.getCombinedLocalAndExportSymbolFlags(symbol) & meaning) {
-                    var id = symbol.escapedName;
-                    // We will copy all symbol regardless of its reserved name because
-                    // symbolsToArray will check whether the key is a reserved name and
-                    // it will not copy symbol with reserved name to the array
-                    if (!symbols.has(id)) {
-                        symbols.set(id, symbol);
-                    }
-                }
+        function checkBlock(node) {
+            // Grammar checking for SyntaxKind.Block
+            if (node.kind === 230 /* Block */) {
+                checkGrammarStatementInAmbientContext(node);
             }
-            function copySymbols(source, meaning) {
-                if (meaning) {
-                    source.forEach(function (symbol) {
-                        copySymbol(symbol, meaning);
-                    });
-                }
+            if (ts.isFunctionOrModuleBlock(node)) {
+                var saveFlowAnalysisDisabled = flowAnalysisDisabled;
+                ts.forEach(node.statements, checkSourceElement);
+                flowAnalysisDisabled = saveFlowAnalysisDisabled;
             }
-        }
-        function isTypeDeclarationName(name) {
-            return name.kind === 75 /* Identifier */ &&
-                isTypeDeclaration(name.parent) &&
-                name.parent.name === name;
-        }
-        function isTypeDeclaration(node) {
-            switch (node.kind) {
-                case 155 /* TypeParameter */:
-                case 245 /* ClassDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 248 /* EnumDeclaration */:
-                    return true;
-                case 255 /* ImportClause */:
-                    return node.isTypeOnly;
-                case 258 /* ImportSpecifier */:
-                case 263 /* ExportSpecifier */:
-                    return node.parent.parent.isTypeOnly;
-                default:
-                    return false;
+            else {
+                ts.forEach(node.statements, checkSourceElement);
             }
-        }
-        // True if the given identifier is part of a type reference
-        function isTypeReferenceIdentifier(node) {
-            while (node.parent.kind === 153 /* QualifiedName */) {
-                node = node.parent;
+            if (node.locals) {
+                registerForUnusedIdentifiersCheck(node);
             }
-            return node.parent.kind === 169 /* TypeReference */;
         }
-        function isHeritageClauseElementIdentifier(node) {
-            while (node.parent.kind === 194 /* PropertyAccessExpression */) {
-                node = node.parent;
+        function checkCollisionWithArgumentsInGeneratedCode(node) {
+            // no rest parameters \ declaration context \ overload - no codegen impact
+            if (languageVersion >= 2 /* ES2015 */ || !ts.hasRestParameter(node) || node.flags & 8388608 /* Ambient */ || ts.nodeIsMissing(node.body)) {
+                return;
             }
-            return node.parent.kind === 216 /* ExpressionWithTypeArguments */;
+            ts.forEach(node.parameters, function (p) {
+                if (p.name && !ts.isBindingPattern(p.name) && p.name.escapedText === argumentsSymbol.escapedName) {
+                    errorSkippedOn("noEmit", p, ts.Diagnostics.Duplicate_identifier_arguments_Compiler_uses_arguments_to_initialize_rest_parameters);
+                }
+            });
         }
-        function forEachEnclosingClass(node, callback) {
-            var result;
-            while (true) {
-                node = ts.getContainingClass(node);
-                if (!node)
-                    break;
-                if (result = callback(node))
-                    break;
+        function needCollisionCheckForIdentifier(node, identifier, name) {
+            if (!(identifier && identifier.escapedText === name)) {
+                return false;
             }
-            return result;
+            if (node.kind === 163 /* PropertyDeclaration */ ||
+                node.kind === 162 /* PropertySignature */ ||
+                node.kind === 165 /* MethodDeclaration */ ||
+                node.kind === 164 /* MethodSignature */ ||
+                node.kind === 167 /* GetAccessor */ ||
+                node.kind === 168 /* SetAccessor */) {
+                // it is ok to have member named '_super' or '_this' - member access is always qualified
+                return false;
+            }
+            if (node.flags & 8388608 /* Ambient */) {
+                // ambient context - no codegen impact
+                return false;
+            }
+            var root = ts.getRootDeclaration(node);
+            if (root.kind === 160 /* Parameter */ && ts.nodeIsMissing(root.parent.body)) {
+                // just an overload - no codegen impact
+                return false;
+            }
+            return true;
         }
-        function isNodeUsedDuringClassInitialization(node) {
-            return !!ts.findAncestor(node, function (element) {
-                if (ts.isConstructorDeclaration(element) && ts.nodeIsPresent(element.body) || ts.isPropertyDeclaration(element)) {
+        // this function will run after checking the source file so 'CaptureThis' is correct for all nodes
+        function checkIfThisIsCapturedInEnclosingScope(node) {
+            ts.findAncestor(node, function (current) {
+                if (getNodeCheckFlags(current) & 4 /* CaptureThis */) {
+                    var isDeclaration_1 = node.kind !== 78 /* Identifier */;
+                    if (isDeclaration_1) {
+                        error(ts.getNameOfDeclaration(node), ts.Diagnostics.Duplicate_identifier_this_Compiler_uses_variable_declaration_this_to_capture_this_reference);
+                    }
+                    else {
+                        error(node, ts.Diagnostics.Expression_resolves_to_variable_declaration_this_that_compiler_uses_to_capture_this_reference);
+                    }
                     return true;
                 }
-                else if (ts.isClassLike(element) || ts.isFunctionLikeDeclaration(element)) {
-                    return "quit";
+                return false;
+            });
+        }
+        function checkIfNewTargetIsCapturedInEnclosingScope(node) {
+            ts.findAncestor(node, function (current) {
+                if (getNodeCheckFlags(current) & 8 /* CaptureNewTarget */) {
+                    var isDeclaration_2 = node.kind !== 78 /* Identifier */;
+                    if (isDeclaration_2) {
+                        error(ts.getNameOfDeclaration(node), ts.Diagnostics.Duplicate_identifier_newTarget_Compiler_uses_variable_declaration_newTarget_to_capture_new_target_meta_property_reference);
+                    }
+                    else {
+                        error(node, ts.Diagnostics.Expression_resolves_to_variable_declaration_newTarget_that_compiler_uses_to_capture_new_target_meta_property_reference);
+                    }
+                    return true;
                 }
                 return false;
             });
         }
-        function isNodeWithinClass(node, classDeclaration) {
-            return !!forEachEnclosingClass(node, function (n) { return n === classDeclaration; });
+        function checkWeakMapCollision(node) {
+            var enclosingBlockScope = ts.getEnclosingBlockScopeContainer(node);
+            if (getNodeCheckFlags(enclosingBlockScope) & 67108864 /* ContainsClassWithPrivateIdentifiers */) {
+                errorSkippedOn("noEmit", node, ts.Diagnostics.Compiler_reserves_name_0_when_emitting_private_identifier_downlevel, "WeakMap");
+            }
         }
-        function getLeftSideOfImportEqualsOrExportAssignment(nodeOnRightSide) {
-            while (nodeOnRightSide.parent.kind === 153 /* QualifiedName */) {
-                nodeOnRightSide = nodeOnRightSide.parent;
+        function checkCollisionWithRequireExportsInGeneratedCode(node, name) {
+            // No need to check for require or exports for ES6 modules and later
+            if (moduleKind >= ts.ModuleKind.ES2015) {
+                return;
             }
-            if (nodeOnRightSide.parent.kind === 253 /* ImportEqualsDeclaration */) {
-                return nodeOnRightSide.parent.moduleReference === nodeOnRightSide ? nodeOnRightSide.parent : undefined;
+            if (!needCollisionCheckForIdentifier(node, name, "require") && !needCollisionCheckForIdentifier(node, name, "exports")) {
+                return;
             }
-            if (nodeOnRightSide.parent.kind === 259 /* ExportAssignment */) {
-                return nodeOnRightSide.parent.expression === nodeOnRightSide ? nodeOnRightSide.parent : undefined;
+            // Uninstantiated modules shouldnt do this check
+            if (ts.isModuleDeclaration(node) && ts.getModuleInstanceState(node) !== 1 /* Instantiated */) {
+                return;
             }
-            return undefined;
-        }
-        function isInRightSideOfImportOrExportAssignment(node) {
-            return getLeftSideOfImportEqualsOrExportAssignment(node) !== undefined;
-        }
-        function getSpecialPropertyAssignmentSymbolFromEntityName(entityName) {
-            var specialPropertyAssignmentKind = ts.getAssignmentDeclarationKind(entityName.parent.parent);
-            switch (specialPropertyAssignmentKind) {
-                case 1 /* ExportsProperty */:
-                case 3 /* PrototypeProperty */:
-                    return getSymbolOfNode(entityName.parent);
-                case 4 /* ThisProperty */:
-                case 2 /* ModuleExports */:
-                case 5 /* Property */:
-                    return getSymbolOfNode(entityName.parent.parent);
+            // In case of variable declaration, node.parent is variable statement so look at the variable statement's parent
+            var parent = getDeclarationContainer(node);
+            if (parent.kind === 297 /* SourceFile */ && ts.isExternalOrCommonJsModule(parent)) {
+                // If the declaration happens to be in external module, report error that require and exports are reserved keywords
+                errorSkippedOn("noEmit", name, ts.Diagnostics.Duplicate_identifier_0_Compiler_reserves_name_1_in_top_level_scope_of_a_module, ts.declarationNameToString(name), ts.declarationNameToString(name));
             }
         }
-        function isImportTypeQualifierPart(node) {
-            var parent = node.parent;
-            while (ts.isQualifiedName(parent)) {
-                node = parent;
-                parent = parent.parent;
+        function checkCollisionWithGlobalPromiseInGeneratedCode(node, name) {
+            if (languageVersion >= 4 /* ES2017 */ || !needCollisionCheckForIdentifier(node, name, "Promise")) {
+                return;
             }
-            if (parent && parent.kind === 188 /* ImportType */ && parent.qualifier === node) {
-                return parent;
+            // Uninstantiated modules shouldnt do this check
+            if (ts.isModuleDeclaration(node) && ts.getModuleInstanceState(node) !== 1 /* Instantiated */) {
+                return;
+            }
+            // In case of variable declaration, node.parent is variable statement so look at the variable statement's parent
+            var parent = getDeclarationContainer(node);
+            if (parent.kind === 297 /* SourceFile */ && ts.isExternalOrCommonJsModule(parent) && parent.flags & 2048 /* HasAsyncFunctions */) {
+                // If the declaration happens to be in external module, report error that Promise is a reserved identifier.
+                errorSkippedOn("noEmit", name, ts.Diagnostics.Duplicate_identifier_0_Compiler_reserves_name_1_in_top_level_scope_of_a_module_containing_async_functions, ts.declarationNameToString(name), ts.declarationNameToString(name));
             }
-            return undefined;
         }
-        function getSymbolOfNameOrPropertyAccessExpression(name) {
-            if (ts.isDeclarationName(name)) {
-                return getSymbolOfNode(name.parent);
+        function checkVarDeclaredNamesNotShadowed(node) {
+            // - ScriptBody : StatementList
+            // It is a Syntax Error if any element of the LexicallyDeclaredNames of StatementList
+            // also occurs in the VarDeclaredNames of StatementList.
+            // - Block : { StatementList }
+            // It is a Syntax Error if any element of the LexicallyDeclaredNames of StatementList
+            // also occurs in the VarDeclaredNames of StatementList.
+            // Variable declarations are hoisted to the top of their function scope. They can shadow
+            // block scoped declarations, which bind tighter. this will not be flagged as duplicate definition
+            // by the binder as the declaration scope is different.
+            // A non-initialized declaration is a no-op as the block declaration will resolve before the var
+            // declaration. the problem is if the declaration has an initializer. this will act as a write to the
+            // block declared value. this is fine for let, but not const.
+            // Only consider declarations with initializers, uninitialized const declarations will not
+            // step on a let/const variable.
+            // Do not consider const and const declarations, as duplicate block-scoped declarations
+            // are handled by the binder.
+            // We are only looking for const declarations that step on let\const declarations from a
+            // different scope. e.g.:
+            //      {
+            //          const x = 0; // localDeclarationSymbol obtained after name resolution will correspond to this declaration
+            //          const x = 0; // symbol for this declaration will be 'symbol'
+            //      }
+            // skip block-scoped variables and parameters
+            if ((ts.getCombinedNodeFlags(node) & 3 /* BlockScoped */) !== 0 || ts.isParameterDeclaration(node)) {
+                return;
             }
-            if (ts.isInJSFile(name) &&
-                name.parent.kind === 194 /* PropertyAccessExpression */ &&
-                name.parent === name.parent.parent.left) {
-                // Check if this is a special property assignment
-                if (!ts.isPrivateIdentifier(name)) {
-                    var specialPropertyAssignmentSymbol = getSpecialPropertyAssignmentSymbolFromEntityName(name);
-                    if (specialPropertyAssignmentSymbol) {
-                        return specialPropertyAssignmentSymbol;
+            // skip variable declarations that don't have initializers
+            // NOTE: in ES6 spec initializer is required in variable declarations where name is binding pattern
+            // so we'll always treat binding elements as initialized
+            if (node.kind === 249 /* VariableDeclaration */ && !node.initializer) {
+                return;
+            }
+            var symbol = getSymbolOfNode(node);
+            if (symbol.flags & 1 /* FunctionScopedVariable */) {
+                if (!ts.isIdentifier(node.name))
+                    return ts.Debug.fail();
+                var localDeclarationSymbol = resolveName(node, node.name.escapedText, 3 /* Variable */, /*nodeNotFoundErrorMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false);
+                if (localDeclarationSymbol &&
+                    localDeclarationSymbol !== symbol &&
+                    localDeclarationSymbol.flags & 2 /* BlockScopedVariable */) {
+                    if (getDeclarationNodeFlagsFromSymbol(localDeclarationSymbol) & 3 /* BlockScoped */) {
+                        var varDeclList = ts.getAncestor(localDeclarationSymbol.valueDeclaration, 250 /* VariableDeclarationList */);
+                        var container = varDeclList.parent.kind === 232 /* VariableStatement */ && varDeclList.parent.parent
+                            ? varDeclList.parent.parent
+                            : undefined;
+                        // names of block-scoped and function scoped variables can collide only
+                        // if block scoped variable is defined in the function\module\source file scope (because of variable hoisting)
+                        var namesShareScope = container &&
+                            (container.kind === 230 /* Block */ && ts.isFunctionLike(container.parent) ||
+                                container.kind === 257 /* ModuleBlock */ ||
+                                container.kind === 256 /* ModuleDeclaration */ ||
+                                container.kind === 297 /* SourceFile */);
+                        // here we know that function scoped variable is shadowed by block scoped one
+                        // if they are defined in the same scope - binder has already reported redeclaration error
+                        // otherwise if variable has an initializer - show error that initialization will fail
+                        // since LHS will be block scoped name instead of function scoped
+                        if (!namesShareScope) {
+                            var name = symbolToString(localDeclarationSymbol);
+                            error(node, ts.Diagnostics.Cannot_initialize_outer_scoped_variable_0_in_the_same_scope_as_block_scoped_declaration_1, name, name);
+                        }
                     }
                 }
             }
-            if (name.parent.kind === 259 /* ExportAssignment */ && ts.isEntityNameExpression(name)) {
-                // Even an entity name expression that doesn't resolve as an entityname may still typecheck as a property access expression
-                var success = resolveEntityName(name, 
-                /*all meanings*/ 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */, /*ignoreErrors*/ true);
-                if (success && success !== unknownSymbol) {
-                    return success;
-                }
+        }
+        function convertAutoToAny(type) {
+            return type === autoType ? anyType : type === autoArrayType ? anyArrayType : type;
+        }
+        // Check variable, parameter, or property declaration
+        function checkVariableLikeDeclaration(node) {
+            var _a;
+            checkDecorators(node);
+            if (!ts.isBindingElement(node)) {
+                checkSourceElement(node.type);
             }
-            else if (!ts.isPropertyAccessExpression(name) && !ts.isPrivateIdentifier(name) && isInRightSideOfImportOrExportAssignment(name)) {
-                // Since we already checked for ExportAssignment, this really could only be an Import
-                var importEqualsDeclaration = ts.getAncestor(name, 253 /* ImportEqualsDeclaration */);
-                ts.Debug.assert(importEqualsDeclaration !== undefined);
-                return getSymbolOfPartOfRightHandSideOfImportEquals(name, /*dontResolveAlias*/ true);
+            // JSDoc `function(string, string): string` syntax results in parameters with no name
+            if (!node.name) {
+                return;
             }
-            if (!ts.isPropertyAccessExpression(name) && !ts.isPrivateIdentifier(name)) {
-                var possibleImportNode = isImportTypeQualifierPart(name);
-                if (possibleImportNode) {
-                    getTypeFromTypeNode(possibleImportNode);
-                    var sym = getNodeLinks(name).resolvedSymbol;
-                    return sym === unknownSymbol ? undefined : sym;
+            // For a computed property, just check the initializer and exit
+            // Do not use hasDynamicName here, because that returns false for well known symbols.
+            // We want to perform checkComputedPropertyName for all computed properties, including
+            // well known symbols.
+            if (node.name.kind === 158 /* ComputedPropertyName */) {
+                checkComputedPropertyName(node.name);
+                if (node.initializer) {
+                    checkExpressionCached(node.initializer);
                 }
             }
-            while (ts.isRightSideOfQualifiedNameOrPropertyAccess(name)) {
-                name = name.parent;
-            }
-            if (isHeritageClauseElementIdentifier(name)) {
-                var meaning = 0 /* None */;
-                // In an interface or class, we're definitely interested in a type.
-                if (name.parent.kind === 216 /* ExpressionWithTypeArguments */) {
-                    meaning = 788968 /* Type */;
-                    // In a class 'extends' clause we are also looking for a value.
-                    if (ts.isExpressionWithTypeArgumentsInClassExtendsClause(name.parent)) {
-                        meaning |= 111551 /* Value */;
-                    }
+            if (node.kind === 198 /* BindingElement */) {
+                if (node.parent.kind === 196 /* ObjectBindingPattern */ && languageVersion < 99 /* ESNext */) {
+                    checkExternalEmitHelpers(node, 4 /* Rest */);
                 }
-                else {
-                    meaning = 1920 /* Namespace */;
+                // check computed properties inside property names of binding elements
+                if (node.propertyName && node.propertyName.kind === 158 /* ComputedPropertyName */) {
+                    checkComputedPropertyName(node.propertyName);
                 }
-                meaning |= 2097152 /* Alias */;
-                var entityNameSymbol = ts.isEntityNameExpression(name) ? resolveEntityName(name, meaning) : undefined;
-                if (entityNameSymbol) {
-                    return entityNameSymbol;
+                // check private/protected variable access
+                var parent = node.parent.parent;
+                var parentType = getTypeForBindingElementParent(parent);
+                var name = node.propertyName || node.name;
+                if (parentType && !ts.isBindingPattern(name)) {
+                    var exprType = getLiteralTypeFromPropertyName(name);
+                    if (isTypeUsableAsPropertyName(exprType)) {
+                        var nameText = getPropertyNameFromType(exprType);
+                        var property = getPropertyOfType(parentType, nameText);
+                        if (property) {
+                            markPropertyAsReferenced(property, /*nodeForCheckWriteOnly*/ undefined, /*isThisAccess*/ false); // A destructuring is never a write-only reference.
+                            checkPropertyAccessibility(parent, !!parent.initializer && parent.initializer.kind === 105 /* SuperKeyword */, parentType, property);
+                        }
+                    }
                 }
             }
-            if (name.parent.kind === 317 /* JSDocParameterTag */) {
-                return ts.getParameterSymbolFromJSDoc(name.parent);
+            // For a binding pattern, check contained binding elements
+            if (ts.isBindingPattern(node.name)) {
+                if (node.name.kind === 197 /* ArrayBindingPattern */ && languageVersion < 2 /* ES2015 */ && compilerOptions.downlevelIteration) {
+                    checkExternalEmitHelpers(node, 512 /* Read */);
+                }
+                ts.forEach(node.name.elements, checkSourceElement);
             }
-            if (name.parent.kind === 155 /* TypeParameter */ && name.parent.parent.kind === 321 /* JSDocTemplateTag */) {
-                ts.Debug.assert(!ts.isInJSFile(name)); // Otherwise `isDeclarationName` would have been true.
-                var typeParameter = ts.getTypeParameterFromJsDoc(name.parent);
-                return typeParameter && typeParameter.symbol;
+            // For a parameter declaration with an initializer, error and exit if the containing function doesn't have a body
+            if (node.initializer && ts.isParameterDeclaration(node) && ts.nodeIsMissing(ts.getContainingFunction(node).body)) {
+                error(node, ts.Diagnostics.A_parameter_initializer_is_only_allowed_in_a_function_or_constructor_implementation);
+                return;
             }
-            if (ts.isExpressionNode(name)) {
-                if (ts.nodeIsMissing(name)) {
-                    // Missing entity name.
-                    return undefined;
-                }
-                if (name.kind === 75 /* Identifier */) {
-                    if (ts.isJSXTagName(name) && isJsxIntrinsicIdentifier(name)) {
-                        var symbol = getIntrinsicTagSymbol(name.parent);
-                        return symbol === unknownSymbol ? undefined : symbol;
+            // For a binding pattern, validate the initializer and exit
+            if (ts.isBindingPattern(node.name)) {
+                var needCheckInitializer = node.initializer && node.parent.parent.kind !== 238 /* ForInStatement */;
+                var needCheckWidenedType = node.name.elements.length === 0;
+                if (needCheckInitializer || needCheckWidenedType) {
+                    // Don't validate for-in initializer as it is already an error
+                    var widenedType = getWidenedTypeForVariableLikeDeclaration(node);
+                    if (needCheckInitializer) {
+                        var initializerType = checkExpressionCached(node.initializer);
+                        if (strictNullChecks && needCheckWidenedType) {
+                            checkNonNullNonVoidType(initializerType, node);
+                        }
+                        else {
+                            checkTypeAssignableToAndOptionallyElaborate(initializerType, getWidenedTypeForVariableLikeDeclaration(node), node, node.initializer);
+                        }
                     }
-                    return resolveEntityName(name, 111551 /* Value */, /*ignoreErrors*/ false, /*dontResolveAlias*/ true);
-                }
-                else if (name.kind === 194 /* PropertyAccessExpression */ || name.kind === 153 /* QualifiedName */) {
-                    var links = getNodeLinks(name);
-                    if (links.resolvedSymbol) {
-                        return links.resolvedSymbol;
+                    // check the binding pattern with empty elements
+                    if (needCheckWidenedType) {
+                        if (ts.isArrayBindingPattern(node.name)) {
+                            checkIteratedTypeOrElementType(65 /* Destructuring */, widenedType, undefinedType, node);
+                        }
+                        else if (strictNullChecks) {
+                            checkNonNullNonVoidType(widenedType, node);
+                        }
                     }
-                    if (name.kind === 194 /* PropertyAccessExpression */) {
-                        checkPropertyAccessExpression(name);
+                }
+                return;
+            }
+            // For a commonjs `const x = require`, validate the alias and exit
+            var symbol = getSymbolOfNode(node);
+            if (symbol.flags & 2097152 /* Alias */ && ts.isRequireVariableDeclaration(node, /*requireStringLiteralLikeArgument*/ true)) {
+                checkAliasSymbol(node);
+                return;
+            }
+            var type = convertAutoToAny(getTypeOfSymbol(symbol));
+            if (node === symbol.valueDeclaration) {
+                // Node is the primary declaration of the symbol, just validate the initializer
+                // Don't validate for-in initializer as it is already an error
+                var initializer = ts.getEffectiveInitializer(node);
+                if (initializer) {
+                    var isJSObjectLiteralInitializer = ts.isInJSFile(node) &&
+                        ts.isObjectLiteralExpression(initializer) &&
+                        (initializer.properties.length === 0 || ts.isPrototypeAccess(node.name)) &&
+                        !!((_a = symbol.exports) === null || _a === void 0 ? void 0 : _a.size);
+                    if (!isJSObjectLiteralInitializer && node.parent.parent.kind !== 238 /* ForInStatement */) {
+                        checkTypeAssignableToAndOptionallyElaborate(checkExpressionCached(initializer), type, node, initializer, /*headMessage*/ undefined);
                     }
-                    else {
-                        checkQualifiedName(name);
+                }
+                if (symbol.declarations.length > 1) {
+                    if (ts.some(symbol.declarations, function (d) { return d !== node && ts.isVariableLike(d) && !areDeclarationFlagsIdentical(d, node); })) {
+                        error(node.name, ts.Diagnostics.All_declarations_of_0_must_have_identical_modifiers, ts.declarationNameToString(node.name));
                     }
-                    return links.resolvedSymbol;
                 }
             }
-            else if (isTypeReferenceIdentifier(name)) {
-                var meaning = name.parent.kind === 169 /* TypeReference */ ? 788968 /* Type */ : 1920 /* Namespace */;
-                return resolveEntityName(name, meaning, /*ignoreErrors*/ false, /*dontResolveAlias*/ true);
+            else {
+                // Node is a secondary declaration, check that type is identical to primary declaration and check that
+                // initializer is consistent with type associated with the node
+                var declarationType = convertAutoToAny(getWidenedTypeForVariableLikeDeclaration(node));
+                if (type !== errorType && declarationType !== errorType &&
+                    !isTypeIdenticalTo(type, declarationType) &&
+                    !(symbol.flags & 67108864 /* Assignment */)) {
+                    errorNextVariableOrPropertyDeclarationMustHaveSameType(symbol.valueDeclaration, type, node, declarationType);
+                }
+                if (node.initializer) {
+                    checkTypeAssignableToAndOptionallyElaborate(checkExpressionCached(node.initializer), declarationType, node, node.initializer, /*headMessage*/ undefined);
+                }
+                if (!areDeclarationFlagsIdentical(node, symbol.valueDeclaration)) {
+                    error(node.name, ts.Diagnostics.All_declarations_of_0_must_have_identical_modifiers, ts.declarationNameToString(node.name));
+                }
             }
-            if (name.parent.kind === 168 /* TypePredicate */) {
-                return resolveEntityName(name, /*meaning*/ 1 /* FunctionScopedVariable */);
+            if (node.kind !== 163 /* PropertyDeclaration */ && node.kind !== 162 /* PropertySignature */) {
+                // We know we don't have a binding pattern or computed name here
+                checkExportsOnMergedDeclarations(node);
+                if (node.kind === 249 /* VariableDeclaration */ || node.kind === 198 /* BindingElement */) {
+                    checkVarDeclaredNamesNotShadowed(node);
+                }
+                checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
+                checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+                if (languageVersion < 99 /* ESNext */ && needCollisionCheckForIdentifier(node, node.name, "WeakMap")) {
+                    potentialWeakMapCollisions.push(node);
+                }
             }
-            // Do we want to return undefined here?
-            return undefined;
         }
-        function getSymbolAtLocation(node, ignoreErrors) {
-            if (node.kind === 290 /* SourceFile */) {
-                return ts.isExternalModule(node) ? getMergedSymbol(node.symbol) : undefined;
+        function errorNextVariableOrPropertyDeclarationMustHaveSameType(firstDeclaration, firstType, nextDeclaration, nextType) {
+            var nextDeclarationName = ts.getNameOfDeclaration(nextDeclaration);
+            var message = nextDeclaration.kind === 163 /* PropertyDeclaration */ || nextDeclaration.kind === 162 /* PropertySignature */
+                ? ts.Diagnostics.Subsequent_property_declarations_must_have_the_same_type_Property_0_must_be_of_type_1_but_here_has_type_2
+                : ts.Diagnostics.Subsequent_variable_declarations_must_have_the_same_type_Variable_0_must_be_of_type_1_but_here_has_type_2;
+            var declName = ts.declarationNameToString(nextDeclarationName);
+            var err = error(nextDeclarationName, message, declName, typeToString(firstType), typeToString(nextType));
+            if (firstDeclaration) {
+                ts.addRelatedInfo(err, ts.createDiagnosticForNode(firstDeclaration, ts.Diagnostics._0_was_also_declared_here, declName));
             }
-            var parent = node.parent;
-            var grandParent = parent.parent;
-            if (node.flags & 16777216 /* InWithStatement */) {
-                // We cannot answer semantic questions within a with block, do not proceed any further
-                return undefined;
+        }
+        function areDeclarationFlagsIdentical(left, right) {
+            if ((left.kind === 160 /* Parameter */ && right.kind === 249 /* VariableDeclaration */) ||
+                (left.kind === 249 /* VariableDeclaration */ && right.kind === 160 /* Parameter */)) {
+                // Differences in optionality between parameters and variables are allowed.
+                return true;
             }
-            if (isDeclarationNameOrImportPropertyName(node)) {
-                // This is a declaration, call getSymbolOfNode
-                var parentSymbol = getSymbolOfNode(parent);
-                return ts.isImportOrExportSpecifier(node.parent) && node.parent.propertyName === node
-                    ? getImmediateAliasedSymbol(parentSymbol)
-                    : parentSymbol;
+            if (ts.hasQuestionToken(left) !== ts.hasQuestionToken(right)) {
+                return false;
             }
-            else if (ts.isLiteralComputedPropertyDeclarationName(node)) {
-                return getSymbolOfNode(parent.parent);
+            var interestingFlags = 8 /* Private */ |
+                16 /* Protected */ |
+                256 /* Async */ |
+                128 /* Abstract */ |
+                64 /* Readonly */ |
+                32 /* Static */;
+            return ts.getSelectedEffectiveModifierFlags(left, interestingFlags) === ts.getSelectedEffectiveModifierFlags(right, interestingFlags);
+        }
+        function checkVariableDeclaration(node) {
+            ts.tracing.push("check" /* Check */, "checkVariableDeclaration", { kind: node.kind, pos: node.pos, end: node.end });
+            checkGrammarVariableDeclaration(node);
+            checkVariableLikeDeclaration(node);
+            ts.tracing.pop();
+        }
+        function checkBindingElement(node) {
+            checkGrammarBindingElement(node);
+            return checkVariableLikeDeclaration(node);
+        }
+        function checkVariableStatement(node) {
+            // Grammar checking
+            if (!checkGrammarDecoratorsAndModifiers(node) && !checkGrammarVariableDeclarationList(node.declarationList))
+                checkGrammarForDisallowedLetOrConstStatement(node);
+            ts.forEach(node.declarationList.declarations, checkSourceElement);
+        }
+        function checkExpressionStatement(node) {
+            // Grammar checking
+            checkGrammarStatementInAmbientContext(node);
+            checkExpression(node.expression);
+        }
+        function checkIfStatement(node) {
+            // Grammar checking
+            checkGrammarStatementInAmbientContext(node);
+            var type = checkTruthinessExpression(node.expression);
+            checkTestingKnownTruthyCallableType(node.expression, node.thenStatement, type);
+            checkSourceElement(node.thenStatement);
+            if (node.thenStatement.kind === 231 /* EmptyStatement */) {
+                error(node.thenStatement, ts.Diagnostics.The_body_of_an_if_statement_cannot_be_the_empty_statement);
             }
-            if (node.kind === 75 /* Identifier */) {
-                if (isInRightSideOfImportOrExportAssignment(node)) {
-                    return getSymbolOfNameOrPropertyAccessExpression(node);
-                }
-                else if (parent.kind === 191 /* BindingElement */ &&
-                    grandParent.kind === 189 /* ObjectBindingPattern */ &&
-                    node === parent.propertyName) {
-                    var typeOfPattern = getTypeOfNode(grandParent);
-                    var propertyDeclaration = getPropertyOfType(typeOfPattern, node.escapedText);
-                    if (propertyDeclaration) {
-                        return propertyDeclaration;
-                    }
-                }
+            checkSourceElement(node.elseStatement);
+        }
+        function checkTestingKnownTruthyCallableType(condExpr, body, type) {
+            if (!strictNullChecks) {
+                return;
             }
-            switch (node.kind) {
-                case 75 /* Identifier */:
-                case 76 /* PrivateIdentifier */:
-                case 194 /* PropertyAccessExpression */:
-                case 153 /* QualifiedName */:
-                    return getSymbolOfNameOrPropertyAccessExpression(node);
-                case 104 /* ThisKeyword */:
-                    var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
-                    if (ts.isFunctionLike(container)) {
-                        var sig = getSignatureFromDeclaration(container);
-                        if (sig.thisParameter) {
-                            return sig.thisParameter;
+            var testedNode = ts.isIdentifier(condExpr)
+                ? condExpr
+                : ts.isPropertyAccessExpression(condExpr)
+                    ? condExpr.name
+                    : undefined;
+            if (!testedNode) {
+                return;
+            }
+            var possiblyFalsy = getFalsyFlags(type);
+            if (possiblyFalsy) {
+                return;
+            }
+            // While it technically should be invalid for any known-truthy value
+            // to be tested, we de-scope to functions unrefenced in the block as a
+            // heuristic to identify the most common bugs. There are too many
+            // false positives for values sourced from type definitions without
+            // strictNullChecks otherwise.
+            var callSignatures = getSignaturesOfType(type, 0 /* Call */);
+            if (callSignatures.length === 0) {
+                return;
+            }
+            var testedFunctionSymbol = getSymbolAtLocation(testedNode);
+            if (!testedFunctionSymbol) {
+                return;
+            }
+            var functionIsUsedInBody = ts.forEachChild(body, function check(childNode) {
+                if (ts.isIdentifier(childNode)) {
+                    var childSymbol = getSymbolAtLocation(childNode);
+                    if (childSymbol && childSymbol === testedFunctionSymbol) {
+                        // If the test was a simple identifier, the above check is sufficient
+                        if (ts.isIdentifier(condExpr)) {
+                            return true;
+                        }
+                        // Otherwise we need to ensure the symbol is called on the same target
+                        var testedExpression = testedNode.parent;
+                        var childExpression = childNode.parent;
+                        while (testedExpression && childExpression) {
+                            if (ts.isIdentifier(testedExpression) && ts.isIdentifier(childExpression) ||
+                                testedExpression.kind === 107 /* ThisKeyword */ && childExpression.kind === 107 /* ThisKeyword */) {
+                                return getSymbolAtLocation(testedExpression) === getSymbolAtLocation(childExpression);
+                            }
+                            if (ts.isPropertyAccessExpression(testedExpression) && ts.isPropertyAccessExpression(childExpression)) {
+                                if (getSymbolAtLocation(testedExpression.name) !== getSymbolAtLocation(childExpression.name)) {
+                                    return false;
+                                }
+                                childExpression = childExpression.expression;
+                                testedExpression = testedExpression.expression;
+                            }
+                            else {
+                                return false;
+                            }
                         }
                     }
-                    if (ts.isInExpressionContext(node)) {
-                        return checkExpression(node).symbol;
-                    }
-                // falls through
-                case 183 /* ThisType */:
-                    return getTypeFromThisTypeNode(node).symbol;
-                case 102 /* SuperKeyword */:
-                    return checkExpression(node).symbol;
-                case 129 /* ConstructorKeyword */:
-                    // constructor keyword for an overload, should take us to the definition if it exist
-                    var constructorDeclaration = node.parent;
-                    if (constructorDeclaration && constructorDeclaration.kind === 162 /* Constructor */) {
-                        return constructorDeclaration.parent.symbol;
-                    }
-                    return undefined;
-                case 10 /* StringLiteral */:
-                case 14 /* NoSubstitutionTemplateLiteral */:
-                    // 1). import x = require("./mo/*gotToDefinitionHere*/d")
-                    // 2). External module name in an import declaration
-                    // 3). Dynamic import call or require in javascript
-                    // 4). type A = import("./f/*gotToDefinitionHere*/oo")
-                    if ((ts.isExternalModuleImportEqualsDeclaration(node.parent.parent) && ts.getExternalModuleImportEqualsDeclarationExpression(node.parent.parent) === node) ||
-                        ((node.parent.kind === 254 /* ImportDeclaration */ || node.parent.kind === 260 /* ExportDeclaration */) && node.parent.moduleSpecifier === node) ||
-                        ((ts.isInJSFile(node) && ts.isRequireCall(node.parent, /*checkArgumentIsStringLiteralLike*/ false)) || ts.isImportCall(node.parent)) ||
-                        (ts.isLiteralTypeNode(node.parent) && ts.isLiteralImportTypeNode(node.parent.parent) && node.parent.parent.argument === node.parent)) {
-                        return resolveExternalModuleName(node, node, ignoreErrors);
-                    }
-                    if (ts.isCallExpression(parent) && ts.isBindableObjectDefinePropertyCall(parent) && parent.arguments[1] === node) {
-                        return getSymbolOfNode(parent);
-                    }
-                // falls through
-                case 8 /* NumericLiteral */:
-                    // index access
-                    var objectType = ts.isElementAccessExpression(parent)
-                        ? parent.argumentExpression === node ? getTypeOfExpression(parent.expression) : undefined
-                        : ts.isLiteralTypeNode(parent) && ts.isIndexedAccessTypeNode(grandParent)
-                            ? getTypeFromTypeNode(grandParent.objectType)
-                            : undefined;
-                    return objectType && getPropertyOfType(objectType, ts.escapeLeadingUnderscores(node.text));
-                case 84 /* DefaultKeyword */:
-                case 94 /* FunctionKeyword */:
-                case 38 /* EqualsGreaterThanToken */:
-                case 80 /* ClassKeyword */:
-                    return getSymbolOfNode(node.parent);
-                case 188 /* ImportType */:
-                    return ts.isLiteralImportTypeNode(node) ? getSymbolAtLocation(node.argument.literal, ignoreErrors) : undefined;
-                case 89 /* ExportKeyword */:
-                    return ts.isExportAssignment(node.parent) ? ts.Debug.checkDefined(node.parent.symbol) : undefined;
-                default:
-                    return undefined;
+                }
+                return ts.forEachChild(childNode, check);
+            });
+            if (!functionIsUsedInBody) {
+                error(condExpr, ts.Diagnostics.This_condition_will_always_return_true_since_the_function_is_always_defined_Did_you_mean_to_call_it_instead);
             }
         }
-        function getShorthandAssignmentValueSymbol(location) {
-            if (location && location.kind === 282 /* ShorthandPropertyAssignment */) {
-                return resolveEntityName(location.name, 111551 /* Value */ | 2097152 /* Alias */);
-            }
-            return undefined;
+        function checkDoStatement(node) {
+            // Grammar checking
+            checkGrammarStatementInAmbientContext(node);
+            checkSourceElement(node.statement);
+            checkTruthinessExpression(node.expression);
         }
-        /** Returns the target of an export specifier without following aliases */
-        function getExportSpecifierLocalTargetSymbol(node) {
-            return node.parent.parent.moduleSpecifier ?
-                getExternalModuleMember(node.parent.parent, node) :
-                resolveEntityName(node.propertyName || node.name, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */);
+        function checkWhileStatement(node) {
+            // Grammar checking
+            checkGrammarStatementInAmbientContext(node);
+            checkTruthinessExpression(node.expression);
+            checkSourceElement(node.statement);
         }
-        function getTypeOfNode(node) {
-            if (node.flags & 16777216 /* InWithStatement */) {
-                // We cannot answer semantic questions within a with block, do not proceed any further
-                return errorType;
+        function checkTruthinessOfType(type, node) {
+            if (type.flags & 16384 /* Void */) {
+                error(node, ts.Diagnostics.An_expression_of_type_void_cannot_be_tested_for_truthiness);
             }
-            var classDecl = ts.tryGetClassImplementingOrExtendingExpressionWithTypeArguments(node);
-            var classType = classDecl && getDeclaredTypeOfClassOrInterface(getSymbolOfNode(classDecl.class));
-            if (ts.isPartOfTypeNode(node)) {
-                var typeFromTypeNode = getTypeFromTypeNode(node);
-                return classType ? getTypeWithThisArgument(typeFromTypeNode, classType.thisType) : typeFromTypeNode;
+            return type;
+        }
+        function checkTruthinessExpression(node, checkMode) {
+            return checkTruthinessOfType(checkExpression(node, checkMode), node);
+        }
+        function checkForStatement(node) {
+            // Grammar checking
+            if (!checkGrammarStatementInAmbientContext(node)) {
+                if (node.initializer && node.initializer.kind === 250 /* VariableDeclarationList */) {
+                    checkGrammarVariableDeclarationList(node.initializer);
+                }
             }
-            if (ts.isExpressionNode(node)) {
-                return getRegularTypeOfExpression(node);
+            if (node.initializer) {
+                if (node.initializer.kind === 250 /* VariableDeclarationList */) {
+                    ts.forEach(node.initializer.declarations, checkVariableDeclaration);
+                }
+                else {
+                    checkExpression(node.initializer);
+                }
             }
-            if (classType && !classDecl.isImplements) {
-                // A SyntaxKind.ExpressionWithTypeArguments is considered a type node, except when it occurs in the
-                // extends clause of a class. We handle that case here.
-                var baseType = ts.firstOrUndefined(getBaseTypes(classType));
-                return baseType ? getTypeWithThisArgument(baseType, classType.thisType) : errorType;
+            if (node.condition)
+                checkTruthinessExpression(node.condition);
+            if (node.incrementor)
+                checkExpression(node.incrementor);
+            checkSourceElement(node.statement);
+            if (node.locals) {
+                registerForUnusedIdentifiersCheck(node);
             }
-            if (isTypeDeclaration(node)) {
-                // In this case, we call getSymbolOfNode instead of getSymbolAtLocation because it is a declaration
-                var symbol = getSymbolOfNode(node);
-                return getDeclaredTypeOfSymbol(symbol);
+        }
+        function checkForOfStatement(node) {
+            checkGrammarForInOrForOfStatement(node);
+            if (node.awaitModifier) {
+                var functionFlags = ts.getFunctionFlags(ts.getContainingFunction(node));
+                if ((functionFlags & (4 /* Invalid */ | 2 /* Async */)) === 2 /* Async */ && languageVersion < 99 /* ESNext */) {
+                    // for..await..of in an async function or async generator function prior to ESNext requires the __asyncValues helper
+                    checkExternalEmitHelpers(node, 32768 /* ForAwaitOfIncludes */);
+                }
             }
-            if (isTypeDeclarationName(node)) {
-                var symbol = getSymbolAtLocation(node);
-                return symbol ? getDeclaredTypeOfSymbol(symbol) : errorType;
+            else if (compilerOptions.downlevelIteration && languageVersion < 2 /* ES2015 */) {
+                // for..of prior to ES2015 requires the __values helper when downlevelIteration is enabled
+                checkExternalEmitHelpers(node, 256 /* ForOfIncludes */);
             }
-            if (ts.isDeclaration(node)) {
-                // In this case, we call getSymbolOfNode instead of getSymbolAtLocation because it is a declaration
-                var symbol = getSymbolOfNode(node);
-                return getTypeOfSymbol(symbol);
+            // Check the LHS and RHS
+            // If the LHS is a declaration, just check it as a variable declaration, which will in turn check the RHS
+            // via checkRightHandSideOfForOf.
+            // If the LHS is an expression, check the LHS, as a destructuring assignment or as a reference.
+            // Then check that the RHS is assignable to it.
+            if (node.initializer.kind === 250 /* VariableDeclarationList */) {
+                checkForInOrForOfVariableDeclaration(node);
             }
-            if (isDeclarationNameOrImportPropertyName(node)) {
-                var symbol = getSymbolAtLocation(node);
-                if (symbol) {
-                    return getTypeOfSymbol(symbol);
+            else {
+                var varExpr = node.initializer;
+                var iteratedType = checkRightHandSideOfForOf(node);
+                // There may be a destructuring assignment on the left side
+                if (varExpr.kind === 199 /* ArrayLiteralExpression */ || varExpr.kind === 200 /* ObjectLiteralExpression */) {
+                    // iteratedType may be undefined. In this case, we still want to check the structure of
+                    // varExpr, in particular making sure it's a valid LeftHandSideExpression. But we'd like
+                    // to short circuit the type relation checking as much as possible, so we pass the unknownType.
+                    checkDestructuringAssignment(varExpr, iteratedType || errorType);
+                }
+                else {
+                    var leftType = checkExpression(varExpr);
+                    checkReferenceExpression(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_of_statement_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_left_hand_side_of_a_for_of_statement_may_not_be_an_optional_property_access);
+                    // iteratedType will be undefined if the rightType was missing properties/signatures
+                    // required to get its iteratedType (like [Symbol.iterator] or next). This may be
+                    // because we accessed properties from anyType, or it may have led to an error inside
+                    // getElementTypeOfIterable.
+                    if (iteratedType) {
+                        checkTypeAssignableToAndOptionallyElaborate(iteratedType, leftType, varExpr, node.expression);
+                    }
                 }
-                return errorType;
             }
-            if (ts.isBindingPattern(node)) {
-                return getTypeForVariableLikeDeclaration(node.parent, /*includeOptionality*/ true) || errorType;
+            checkSourceElement(node.statement);
+            if (node.locals) {
+                registerForUnusedIdentifiersCheck(node);
             }
-            if (isInRightSideOfImportOrExportAssignment(node)) {
-                var symbol = getSymbolAtLocation(node);
-                if (symbol) {
-                    var declaredType = getDeclaredTypeOfSymbol(symbol);
-                    return declaredType !== errorType ? declaredType : getTypeOfSymbol(symbol);
+        }
+        function checkForInStatement(node) {
+            // Grammar checking
+            checkGrammarForInOrForOfStatement(node);
+            var rightType = getNonNullableTypeIfNeeded(checkExpression(node.expression));
+            // TypeScript 1.0 spec (April 2014): 5.4
+            // In a 'for-in' statement of the form
+            // for (let VarDecl in Expr) Statement
+            //   VarDecl must be a variable declaration without a type annotation that declares a variable of type Any,
+            //   and Expr must be an expression of type Any, an object type, or a type parameter type.
+            if (node.initializer.kind === 250 /* VariableDeclarationList */) {
+                var variable = node.initializer.declarations[0];
+                if (variable && ts.isBindingPattern(variable.name)) {
+                    error(variable.name, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_cannot_be_a_destructuring_pattern);
                 }
+                checkForInOrForOfVariableDeclaration(node);
             }
-            return errorType;
-        }
-        // Gets the type of object literal or array literal of destructuring assignment.
-        // { a } from
-        //     for ( { a } of elems) {
-        //     }
-        // [ a ] from
-        //     [a] = [ some array ...]
-        function getTypeOfAssignmentPattern(expr) {
-            ts.Debug.assert(expr.kind === 193 /* ObjectLiteralExpression */ || expr.kind === 192 /* ArrayLiteralExpression */);
-            // If this is from "for of"
-            //     for ( { a } of elems) {
-            //     }
-            if (expr.parent.kind === 232 /* ForOfStatement */) {
-                var iteratedType = checkRightHandSideOfForOf(expr.parent);
-                return checkDestructuringAssignment(expr, iteratedType || errorType);
+            else {
+                // In a 'for-in' statement of the form
+                // for (Var in Expr) Statement
+                //   Var must be an expression classified as a reference of type Any or the String primitive type,
+                //   and Expr must be an expression of type Any, an object type, or a type parameter type.
+                var varExpr = node.initializer;
+                var leftType = checkExpression(varExpr);
+                if (varExpr.kind === 199 /* ArrayLiteralExpression */ || varExpr.kind === 200 /* ObjectLiteralExpression */) {
+                    error(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_cannot_be_a_destructuring_pattern);
+                }
+                else if (!isTypeAssignableTo(getIndexTypeOrString(rightType), leftType)) {
+                    error(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_must_be_of_type_string_or_any);
+                }
+                else {
+                    // run check only former check succeeded to avoid cascading errors
+                    checkReferenceExpression(varExpr, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_must_be_a_variable_or_a_property_access, ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_may_not_be_an_optional_property_access);
+                }
             }
-            // If this is from "for" initializer
-            //     for ({a } = elems[0];.....) { }
-            if (expr.parent.kind === 209 /* BinaryExpression */) {
-                var iteratedType = getTypeOfExpression(expr.parent.right);
-                return checkDestructuringAssignment(expr, iteratedType || errorType);
+            // unknownType is returned i.e. if node.expression is identifier whose name cannot be resolved
+            // in this case error about missing name is already reported - do not report extra one
+            if (rightType === neverType || !isTypeAssignableToKind(rightType, 67108864 /* NonPrimitive */ | 58982400 /* InstantiableNonPrimitive */)) {
+                error(node.expression, ts.Diagnostics.The_right_hand_side_of_a_for_in_statement_must_be_of_type_any_an_object_type_or_a_type_parameter_but_here_has_type_0, typeToString(rightType));
             }
-            // If this is from nested object binding pattern
-            //     for ({ skills: { primary, secondary } } = multiRobot, i = 0; i < 1; i++) {
-            if (expr.parent.kind === 281 /* PropertyAssignment */) {
-                var node_4 = ts.cast(expr.parent.parent, ts.isObjectLiteralExpression);
-                var typeOfParentObjectLiteral = getTypeOfAssignmentPattern(node_4) || errorType;
-                var propertyIndex = ts.indexOfNode(node_4.properties, expr.parent);
-                return checkObjectLiteralDestructuringPropertyAssignment(node_4, typeOfParentObjectLiteral, propertyIndex);
+            checkSourceElement(node.statement);
+            if (node.locals) {
+                registerForUnusedIdentifiersCheck(node);
             }
-            // Array literal assignment - array destructuring pattern
-            var node = ts.cast(expr.parent, ts.isArrayLiteralExpression);
-            //    [{ property1: p1, property2 }] = elems;
-            var typeOfArrayLiteral = getTypeOfAssignmentPattern(node) || errorType;
-            var elementType = checkIteratedTypeOrElementType(65 /* Destructuring */, typeOfArrayLiteral, undefinedType, expr.parent) || errorType;
-            return checkArrayLiteralDestructuringElementAssignment(node, typeOfArrayLiteral, node.elements.indexOf(expr), elementType);
         }
-        // Gets the property symbol corresponding to the property in destructuring assignment
-        // 'property1' from
-        //     for ( { property1: a } of elems) {
-        //     }
-        // 'property1' at location 'a' from:
-        //     [a] = [ property1, property2 ]
-        function getPropertySymbolOfDestructuringAssignment(location) {
-            // Get the type of the object or array literal and then look for property of given name in the type
-            var typeOfObjectLiteral = getTypeOfAssignmentPattern(ts.cast(location.parent.parent, ts.isAssignmentPattern));
-            return typeOfObjectLiteral && getPropertyOfType(typeOfObjectLiteral, location.escapedText);
+        function checkForInOrForOfVariableDeclaration(iterationStatement) {
+            var variableDeclarationList = iterationStatement.initializer;
+            // checkGrammarForInOrForOfStatement will check that there is exactly one declaration.
+            if (variableDeclarationList.declarations.length >= 1) {
+                var decl = variableDeclarationList.declarations[0];
+                checkVariableDeclaration(decl);
+            }
         }
-        function getRegularTypeOfExpression(expr) {
-            if (ts.isRightSideOfQualifiedNameOrPropertyAccess(expr)) {
-                expr = expr.parent;
+        function checkRightHandSideOfForOf(statement) {
+            var use = statement.awaitModifier ? 15 /* ForAwaitOf */ : 13 /* ForOf */;
+            return checkIteratedTypeOrElementType(use, checkNonNullExpression(statement.expression), undefinedType, statement.expression);
+        }
+        function checkIteratedTypeOrElementType(use, inputType, sentType, errorNode) {
+            if (isTypeAny(inputType)) {
+                return inputType;
             }
-            return getRegularTypeOfLiteralType(getTypeOfExpression(expr));
+            return getIteratedTypeOrElementType(use, inputType, sentType, errorNode, /*checkAssignability*/ true) || anyType;
         }
         /**
-         * Gets either the static or instance type of a class element, based on
-         * whether the element is declared as "static".
+         * When consuming an iterable type in a for..of, spread, or iterator destructuring assignment
+         * we want to get the iterated type of an iterable for ES2015 or later, or the iterated type
+         * of a iterable (if defined globally) or element type of an array like for ES2015 or earlier.
          */
-        function getParentTypeOfClassElement(node) {
-            var classSymbol = getSymbolOfNode(node.parent);
-            return ts.hasModifier(node, 32 /* Static */)
-                ? getTypeOfSymbol(classSymbol)
-                : getDeclaredTypeOfSymbol(classSymbol);
-        }
-        function getClassElementPropertyKeyType(element) {
-            var name = element.name;
-            switch (name.kind) {
-                case 75 /* Identifier */:
-                    return getLiteralType(ts.idText(name));
-                case 8 /* NumericLiteral */:
-                case 10 /* StringLiteral */:
-                    return getLiteralType(name.text);
-                case 154 /* ComputedPropertyName */:
-                    var nameType = checkComputedPropertyName(name);
-                    return isTypeAssignableToKind(nameType, 12288 /* ESSymbolLike */) ? nameType : stringType;
-                default:
-                    return ts.Debug.fail("Unsupported property name.");
+        function getIteratedTypeOrElementType(use, inputType, sentType, errorNode, checkAssignability) {
+            var allowAsyncIterables = (use & 2 /* AllowsAsyncIterablesFlag */) !== 0;
+            if (inputType === neverType) {
+                reportTypeNotIterableError(errorNode, inputType, allowAsyncIterables); // TODO: GH#18217
+                return undefined;
             }
-        }
-        // Return the list of properties of the given type, augmented with properties from Function
-        // if the type has call or construct signatures
-        function getAugmentedPropertiesOfType(type) {
-            type = getApparentType(type);
-            var propsByName = ts.createSymbolTable(getPropertiesOfType(type));
-            var functionType = getSignaturesOfType(type, 0 /* Call */).length ? globalCallableFunctionType :
-                getSignaturesOfType(type, 1 /* Construct */).length ? globalNewableFunctionType :
-                    undefined;
-            if (functionType) {
-                ts.forEach(getPropertiesOfType(functionType), function (p) {
-                    if (!propsByName.has(p.escapedName)) {
-                        propsByName.set(p.escapedName, p);
+            var uplevelIteration = languageVersion >= 2 /* ES2015 */;
+            var downlevelIteration = !uplevelIteration && compilerOptions.downlevelIteration;
+            var possibleOutOfBounds = compilerOptions.noUncheckedIndexedAccess && !!(use & 128 /* PossiblyOutOfBounds */);
+            // Get the iterated type of an `Iterable<T>` or `IterableIterator<T>` only in ES2015
+            // or higher, when inside of an async generator or for-await-if, or when
+            // downlevelIteration is requested.
+            if (uplevelIteration || downlevelIteration || allowAsyncIterables) {
+                // We only report errors for an invalid iterable type in ES2015 or higher.
+                var iterationTypes = getIterationTypesOfIterable(inputType, use, uplevelIteration ? errorNode : undefined);
+                if (checkAssignability) {
+                    if (iterationTypes) {
+                        var diagnostic = use & 8 /* ForOfFlag */ ? ts.Diagnostics.Cannot_iterate_value_because_the_next_method_of_its_iterator_expects_type_1_but_for_of_will_always_send_0 :
+                            use & 32 /* SpreadFlag */ ? ts.Diagnostics.Cannot_iterate_value_because_the_next_method_of_its_iterator_expects_type_1_but_array_spread_will_always_send_0 :
+                                use & 64 /* DestructuringFlag */ ? ts.Diagnostics.Cannot_iterate_value_because_the_next_method_of_its_iterator_expects_type_1_but_array_destructuring_will_always_send_0 :
+                                    use & 16 /* YieldStarFlag */ ? ts.Diagnostics.Cannot_delegate_iteration_to_value_because_the_next_method_of_its_iterator_expects_type_1_but_the_containing_generator_will_always_send_0 :
+                                        undefined;
+                        if (diagnostic) {
+                            checkTypeAssignableTo(sentType, iterationTypes.nextType, errorNode, diagnostic);
+                        }
                     }
-                });
+                }
+                if (iterationTypes || uplevelIteration) {
+                    return possibleOutOfBounds ? includeUndefinedInIndexSignature(iterationTypes && iterationTypes.yieldType) : (iterationTypes && iterationTypes.yieldType);
+                }
             }
-            return getNamedMembers(propsByName);
-        }
-        function typeHasCallOrConstructSignatures(type) {
-            return ts.typeHasCallOrConstructSignatures(type, checker);
-        }
-        function getRootSymbols(symbol) {
-            var roots = getImmediateRootSymbols(symbol);
-            return roots ? ts.flatMap(roots, getRootSymbols) : [symbol];
-        }
-        function getImmediateRootSymbols(symbol) {
-            if (ts.getCheckFlags(symbol) & 6 /* Synthetic */) {
-                return ts.mapDefined(getSymbolLinks(symbol).containingType.types, function (type) { return getPropertyOfType(type, symbol.escapedName); });
+            var arrayType = inputType;
+            var reportedError = false;
+            var hasStringConstituent = false;
+            // If strings are permitted, remove any string-like constituents from the array type.
+            // This allows us to find other non-string element types from an array unioned with
+            // a string.
+            if (use & 4 /* AllowsStringInputFlag */) {
+                if (arrayType.flags & 1048576 /* Union */) {
+                    // After we remove all types that are StringLike, we will know if there was a string constituent
+                    // based on whether the result of filter is a new array.
+                    var arrayTypes = inputType.types;
+                    var filteredTypes = ts.filter(arrayTypes, function (t) { return !(t.flags & 402653316 /* StringLike */); });
+                    if (filteredTypes !== arrayTypes) {
+                        arrayType = getUnionType(filteredTypes, 2 /* Subtype */);
+                    }
+                }
+                else if (arrayType.flags & 402653316 /* StringLike */) {
+                    arrayType = neverType;
+                }
+                hasStringConstituent = arrayType !== inputType;
+                if (hasStringConstituent) {
+                    if (languageVersion < 1 /* ES5 */) {
+                        if (errorNode) {
+                            error(errorNode, ts.Diagnostics.Using_a_string_in_a_for_of_statement_is_only_supported_in_ECMAScript_5_and_higher);
+                            reportedError = true;
+                        }
+                    }
+                    // Now that we've removed all the StringLike types, if no constituents remain, then the entire
+                    // arrayOrStringType was a string.
+                    if (arrayType.flags & 131072 /* Never */) {
+                        return possibleOutOfBounds ? includeUndefinedInIndexSignature(stringType) : stringType;
+                    }
+                }
             }
-            else if (symbol.flags & 33554432 /* Transient */) {
-                var _a = symbol, leftSpread = _a.leftSpread, rightSpread = _a.rightSpread, syntheticOrigin = _a.syntheticOrigin;
-                return leftSpread ? [leftSpread, rightSpread]
-                    : syntheticOrigin ? [syntheticOrigin]
-                        : ts.singleElementArray(tryGetAliasTarget(symbol));
+            if (!isArrayLikeType(arrayType)) {
+                if (errorNode && !reportedError) {
+                    // Which error we report depends on whether we allow strings or if there was a
+                    // string constituent. For example, if the input type is number | string, we
+                    // want to say that number is not an array type. But if the input was just
+                    // number and string input is allowed, we want to say that number is not an
+                    // array type or a string type.
+                    var yieldType = getIterationTypeOfIterable(use, 0 /* Yield */, inputType, /*errorNode*/ undefined);
+                    var _a = !(use & 4 /* AllowsStringInputFlag */) || hasStringConstituent
+                        ? downlevelIteration
+                            ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator, true]
+                            : yieldType
+                                ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type_Use_compiler_option_downlevelIteration_to_allow_iterating_of_iterators, false]
+                                : [ts.Diagnostics.Type_0_is_not_an_array_type, true]
+                        : downlevelIteration
+                            ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type_or_does_not_have_a_Symbol_iterator_method_that_returns_an_iterator, true]
+                            : yieldType
+                                ? [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type_Use_compiler_option_downlevelIteration_to_allow_iterating_of_iterators, false]
+                                : [ts.Diagnostics.Type_0_is_not_an_array_type_or_a_string_type, true], defaultDiagnostic = _a[0], maybeMissingAwait = _a[1];
+                    errorAndMaybeSuggestAwait(errorNode, maybeMissingAwait && !!getAwaitedTypeOfPromise(arrayType), defaultDiagnostic, typeToString(arrayType));
+                }
+                return hasStringConstituent ? possibleOutOfBounds ? includeUndefinedInIndexSignature(stringType) : stringType : undefined;
             }
-            return undefined;
+            var arrayElementType = getIndexTypeOfType(arrayType, 1 /* Number */);
+            if (hasStringConstituent && arrayElementType) {
+                // This is just an optimization for the case where arrayOrStringType is string | string[]
+                if (arrayElementType.flags & 402653316 /* StringLike */ && !compilerOptions.noUncheckedIndexedAccess) {
+                    return stringType;
+                }
+                return getUnionType(possibleOutOfBounds ? [arrayElementType, stringType, undefinedType] : [arrayElementType, stringType], 2 /* Subtype */);
+            }
+            return (use & 128 /* PossiblyOutOfBounds */) ? includeUndefinedInIndexSignature(arrayElementType) : arrayElementType;
         }
-        function tryGetAliasTarget(symbol) {
-            var target;
-            var next = symbol;
-            while (next = getSymbolLinks(next).target) {
-                target = next;
+        /**
+         * Gets the requested "iteration type" from an `Iterable`-like or `AsyncIterable`-like type.
+         */
+        function getIterationTypeOfIterable(use, typeKind, inputType, errorNode) {
+            if (isTypeAny(inputType)) {
+                return undefined;
             }
-            return target;
+            var iterationTypes = getIterationTypesOfIterable(inputType, use, errorNode);
+            return iterationTypes && iterationTypes[getIterationTypesKeyFromIterationTypeKind(typeKind)];
         }
-        // Emitter support
-        function isArgumentsLocalBinding(nodeIn) {
-            if (!ts.isGeneratedIdentifier(nodeIn)) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
-                if (node) {
-                    var isPropertyName_1 = node.parent.kind === 194 /* PropertyAccessExpression */ && node.parent.name === node;
-                    return !isPropertyName_1 && getReferencedValueSymbol(node) === argumentsSymbol;
+        function createIterationTypes(yieldType, returnType, nextType) {
+            // `yieldType` and `returnType` are defaulted to `neverType` they each will be combined
+            // via `getUnionType` when merging iteration types. `nextType` is defined as `unknownType`
+            // as it is combined via `getIntersectionType` when merging iteration types.
+            if (yieldType === void 0) { yieldType = neverType; }
+            if (returnType === void 0) { returnType = neverType; }
+            if (nextType === void 0) { nextType = unknownType; }
+            // Use the cache only for intrinsic types to keep it small as they are likely to be
+            // more frequently created (i.e. `Iterator<number, void, unknown>`). Iteration types
+            // are also cached on the type they are requested for, so we shouldn't need to maintain
+            // the cache for less-frequently used types.
+            if (yieldType.flags & 67359327 /* Intrinsic */ &&
+                returnType.flags & (1 /* Any */ | 131072 /* Never */ | 2 /* Unknown */ | 16384 /* Void */ | 32768 /* Undefined */) &&
+                nextType.flags & (1 /* Any */ | 131072 /* Never */ | 2 /* Unknown */ | 16384 /* Void */ | 32768 /* Undefined */)) {
+                var id = getTypeListId([yieldType, returnType, nextType]);
+                var iterationTypes = iterationTypesCache.get(id);
+                if (!iterationTypes) {
+                    iterationTypes = { yieldType: yieldType, returnType: returnType, nextType: nextType };
+                    iterationTypesCache.set(id, iterationTypes);
                 }
+                return iterationTypes;
             }
-            return false;
+            return { yieldType: yieldType, returnType: returnType, nextType: nextType };
         }
-        function moduleExportsSomeValue(moduleReferenceExpression) {
-            var moduleSymbol = resolveExternalModuleName(moduleReferenceExpression.parent, moduleReferenceExpression);
-            if (!moduleSymbol || ts.isShorthandAmbientModuleSymbol(moduleSymbol)) {
-                // If the module is not found or is shorthand, assume that it may export a value.
-                return true;
-            }
-            var hasExportAssignment = hasExportAssignmentSymbol(moduleSymbol);
-            // if module has export assignment then 'resolveExternalModuleSymbol' will return resolved symbol for export assignment
-            // otherwise it will return moduleSymbol itself
-            moduleSymbol = resolveExternalModuleSymbol(moduleSymbol);
-            var symbolLinks = getSymbolLinks(moduleSymbol);
-            if (symbolLinks.exportsSomeValue === undefined) {
-                // for export assignments - check if resolved symbol for RHS is itself a value
-                // otherwise - check if at least one export is value
-                symbolLinks.exportsSomeValue = hasExportAssignment
-                    ? !!(moduleSymbol.flags & 111551 /* Value */)
-                    : ts.forEachEntry(getExportsOfModule(moduleSymbol), isValue);
+        /**
+         * Combines multiple `IterationTypes` records.
+         *
+         * If `array` is empty or all elements are missing or are references to `noIterationTypes`,
+         * then `noIterationTypes` is returned. Otherwise, an `IterationTypes` record is returned
+         * for the combined iteration types.
+         */
+        function combineIterationTypes(array) {
+            var yieldTypes;
+            var returnTypes;
+            var nextTypes;
+            for (var _i = 0, array_10 = array; _i < array_10.length; _i++) {
+                var iterationTypes = array_10[_i];
+                if (iterationTypes === undefined || iterationTypes === noIterationTypes) {
+                    continue;
+                }
+                if (iterationTypes === anyIterationTypes) {
+                    return anyIterationTypes;
+                }
+                yieldTypes = ts.append(yieldTypes, iterationTypes.yieldType);
+                returnTypes = ts.append(returnTypes, iterationTypes.returnType);
+                nextTypes = ts.append(nextTypes, iterationTypes.nextType);
             }
-            return symbolLinks.exportsSomeValue;
-            function isValue(s) {
-                s = resolveSymbol(s);
-                return s && !!(s.flags & 111551 /* Value */);
+            if (yieldTypes || returnTypes || nextTypes) {
+                return createIterationTypes(yieldTypes && getUnionType(yieldTypes), returnTypes && getUnionType(returnTypes), nextTypes && getIntersectionType(nextTypes));
             }
+            return noIterationTypes;
         }
-        function isNameOfModuleOrEnumDeclaration(node) {
-            return ts.isModuleOrEnumDeclaration(node.parent) && node === node.parent.name;
+        function getCachedIterationTypes(type, cacheKey) {
+            return type[cacheKey];
         }
-        // When resolved as an expression identifier, if the given node references an exported entity, return the declaration
-        // node of the exported entity's container. Otherwise, return undefined.
-        function getReferencedExportContainer(nodeIn, prefixLocals) {
-            var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
-            if (node) {
-                // When resolving the export container for the name of a module or enum
-                // declaration, we need to start resolution at the declaration's container.
-                // Otherwise, we could incorrectly resolve the export container as the
-                // declaration if it contains an exported member with the same name.
-                var symbol = getReferencedValueSymbol(node, /*startInDeclarationContainer*/ isNameOfModuleOrEnumDeclaration(node));
-                if (symbol) {
-                    if (symbol.flags & 1048576 /* ExportValue */) {
-                        // If we reference an exported entity within the same module declaration, then whether
-                        // we prefix depends on the kind of entity. SymbolFlags.ExportHasLocal encompasses all the
-                        // kinds that we do NOT prefix.
-                        var exportSymbol = getMergedSymbol(symbol.exportSymbol);
-                        if (!prefixLocals && exportSymbol.flags & 944 /* ExportHasLocal */ && !(exportSymbol.flags & 3 /* Variable */)) {
-                            return undefined;
-                        }
-                        symbol = exportSymbol;
-                    }
-                    var parentSymbol_1 = getParentOfSymbol(symbol);
-                    if (parentSymbol_1) {
-                        if (parentSymbol_1.flags & 512 /* ValueModule */ && parentSymbol_1.valueDeclaration.kind === 290 /* SourceFile */) {
-                            var symbolFile = parentSymbol_1.valueDeclaration;
-                            var referenceFile = ts.getSourceFileOfNode(node);
-                            // If `node` accesses an export and that export isn't in the same file, then symbol is a namespace export, so return undefined.
-                            var symbolIsUmdExport = symbolFile !== referenceFile;
-                            return symbolIsUmdExport ? undefined : symbolFile;
-                        }
-                        return ts.findAncestor(node.parent, function (n) { return ts.isModuleOrEnumDeclaration(n) && getSymbolOfNode(n) === parentSymbol_1; });
+        function setCachedIterationTypes(type, cacheKey, cachedTypes) {
+            return type[cacheKey] = cachedTypes;
+        }
+        /**
+         * Gets the *yield*, *return*, and *next* types from an `Iterable`-like or `AsyncIterable`-like type.
+         *
+         * At every level that involves analyzing return types of signatures, we union the return types of all the signatures.
+         *
+         * Another thing to note is that at any step of this process, we could run into a dead end,
+         * meaning either the property is missing, or we run into the anyType. If either of these things
+         * happens, we return `undefined` to signal that we could not find the iteration type. If a property
+         * is missing, and the previous step did not result in `any`, then we also give an error if the
+         * caller requested it. Then the caller can decide what to do in the case where there is no iterated
+         * type.
+         *
+         * For a **for-of** statement, `yield*` (in a normal generator), spread, array
+         * destructuring, or normal generator we will only ever look for a `[Symbol.iterator]()`
+         * method.
+         *
+         * For an async generator we will only ever look at the `[Symbol.asyncIterator]()` method.
+         *
+         * For a **for-await-of** statement or a `yield*` in an async generator we will look for
+         * the `[Symbol.asyncIterator]()` method first, and then the `[Symbol.iterator]()` method.
+         */
+        function getIterationTypesOfIterable(type, use, errorNode) {
+            if (isTypeAny(type)) {
+                return anyIterationTypes;
+            }
+            if (!(type.flags & 1048576 /* Union */)) {
+                var iterationTypes_1 = getIterationTypesOfIterableWorker(type, use, errorNode);
+                if (iterationTypes_1 === noIterationTypes) {
+                    if (errorNode) {
+                        reportTypeNotIterableError(errorNode, type, !!(use & 2 /* AllowsAsyncIterablesFlag */));
                     }
+                    return undefined;
                 }
+                return iterationTypes_1;
             }
-        }
-        // When resolved as an expression identifier, if the given node references an import, return the declaration of
-        // that import. Otherwise, return undefined.
-        function getReferencedImportDeclaration(nodeIn) {
-            var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
-            if (node) {
-                var symbol = getReferencedValueSymbol(node);
-                // We should only get the declaration of an alias if there isn't a local value
-                // declaration for the symbol
-                if (isNonLocalAlias(symbol, /*excludes*/ 111551 /* Value */) && !getTypeOnlyAliasDeclaration(symbol)) {
-                    return getDeclarationOfAliasSymbol(symbol);
+            var cacheKey = use & 2 /* AllowsAsyncIterablesFlag */ ? "iterationTypesOfAsyncIterable" : "iterationTypesOfIterable";
+            var cachedTypes = getCachedIterationTypes(type, cacheKey);
+            if (cachedTypes)
+                return cachedTypes === noIterationTypes ? undefined : cachedTypes;
+            var allIterationTypes;
+            for (var _i = 0, _a = type.types; _i < _a.length; _i++) {
+                var constituent = _a[_i];
+                var iterationTypes_2 = getIterationTypesOfIterableWorker(constituent, use, errorNode);
+                if (iterationTypes_2 === noIterationTypes) {
+                    if (errorNode) {
+                        reportTypeNotIterableError(errorNode, type, !!(use & 2 /* AllowsAsyncIterablesFlag */));
+                    }
+                    setCachedIterationTypes(type, cacheKey, noIterationTypes);
+                    return undefined;
+                }
+                else {
+                    allIterationTypes = ts.append(allIterationTypes, iterationTypes_2);
                 }
             }
-            return undefined;
+            var iterationTypes = allIterationTypes ? combineIterationTypes(allIterationTypes) : noIterationTypes;
+            setCachedIterationTypes(type, cacheKey, iterationTypes);
+            return iterationTypes === noIterationTypes ? undefined : iterationTypes;
         }
-        function isSymbolOfDestructuredElementOfCatchBinding(symbol) {
-            return ts.isBindingElement(symbol.valueDeclaration)
-                && ts.walkUpBindingElementsAndPatterns(symbol.valueDeclaration).parent.kind === 280 /* CatchClause */;
+        function getAsyncFromSyncIterationTypes(iterationTypes, errorNode) {
+            if (iterationTypes === noIterationTypes)
+                return noIterationTypes;
+            if (iterationTypes === anyIterationTypes)
+                return anyIterationTypes;
+            var yieldType = iterationTypes.yieldType, returnType = iterationTypes.returnType, nextType = iterationTypes.nextType;
+            return createIterationTypes(getAwaitedType(yieldType, errorNode) || anyType, getAwaitedType(returnType, errorNode) || anyType, nextType);
         }
-        function isSymbolOfDeclarationWithCollidingName(symbol) {
-            if (symbol.flags & 418 /* BlockScoped */ && !ts.isSourceFile(symbol.valueDeclaration)) {
-                var links = getSymbolLinks(symbol);
-                if (links.isDeclarationWithCollidingName === undefined) {
-                    var container = ts.getEnclosingBlockScopeContainer(symbol.valueDeclaration);
-                    if (ts.isStatementWithLocals(container) || isSymbolOfDestructuredElementOfCatchBinding(symbol)) {
-                        var nodeLinks_1 = getNodeLinks(symbol.valueDeclaration);
-                        if (resolveName(container.parent, symbol.escapedName, 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false)) {
-                            // redeclaration - always should be renamed
-                            links.isDeclarationWithCollidingName = true;
-                        }
-                        else if (nodeLinks_1.flags & 262144 /* CapturedBlockScopedBinding */) {
-                            // binding is captured in the function
-                            // should be renamed if:
-                            // - binding is not top level - top level bindings never collide with anything
-                            // AND
-                            //   - binding is not declared in loop, should be renamed to avoid name reuse across siblings
-                            //     let a, b
-                            //     { let x = 1; a = () => x; }
-                            //     { let x = 100; b = () => x; }
-                            //     console.log(a()); // should print '1'
-                            //     console.log(b()); // should print '100'
-                            //     OR
-                            //   - binding is declared inside loop but not in inside initializer of iteration statement or directly inside loop body
-                            //     * variables from initializer are passed to rewritten loop body as parameters so they are not captured directly
-                            //     * variables that are declared immediately in loop body will become top level variable after loop is rewritten and thus
-                            //       they will not collide with anything
-                            var isDeclaredInLoop = nodeLinks_1.flags & 524288 /* BlockScopedBindingInLoop */;
-                            var inLoopInitializer = ts.isIterationStatement(container, /*lookInLabeledStatements*/ false);
-                            var inLoopBodyBlock = container.kind === 223 /* Block */ && ts.isIterationStatement(container.parent, /*lookInLabeledStatements*/ false);
-                            links.isDeclarationWithCollidingName = !ts.isBlockScopedContainerTopLevel(container) && (!isDeclaredInLoop || (!inLoopInitializer && !inLoopBodyBlock));
-                        }
-                        else {
-                            links.isDeclarationWithCollidingName = false;
+        /**
+         * Gets the *yield*, *return*, and *next* types from a non-union type.
+         *
+         * If we are unable to find the *yield*, *return*, and *next* types, `noIterationTypes` is
+         * returned to indicate to the caller that it should report an error. Otherwise, an
+         * `IterationTypes` record is returned.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterable` instead.
+         */
+        function getIterationTypesOfIterableWorker(type, use, errorNode) {
+            if (isTypeAny(type)) {
+                return anyIterationTypes;
+            }
+            if (use & 2 /* AllowsAsyncIterablesFlag */) {
+                var iterationTypes = getIterationTypesOfIterableCached(type, asyncIterationTypesResolver) ||
+                    getIterationTypesOfIterableFast(type, asyncIterationTypesResolver);
+                if (iterationTypes) {
+                    return iterationTypes;
+                }
+            }
+            if (use & 1 /* AllowsSyncIterablesFlag */) {
+                var iterationTypes = getIterationTypesOfIterableCached(type, syncIterationTypesResolver) ||
+                    getIterationTypesOfIterableFast(type, syncIterationTypesResolver);
+                if (iterationTypes) {
+                    if (use & 2 /* AllowsAsyncIterablesFlag */) {
+                        // for a sync iterable in an async context, only use the cached types if they are valid.
+                        if (iterationTypes !== noIterationTypes) {
+                            return setCachedIterationTypes(type, "iterationTypesOfAsyncIterable", getAsyncFromSyncIterationTypes(iterationTypes, errorNode));
                         }
                     }
+                    else {
+                        return iterationTypes;
+                    }
                 }
-                return links.isDeclarationWithCollidingName;
             }
-            return false;
-        }
-        // When resolved as an expression identifier, if the given node references a nested block scoped entity with
-        // a name that either hides an existing name or might hide it when compiled downlevel,
-        // return the declaration of that entity. Otherwise, return undefined.
-        function getReferencedDeclarationWithCollidingName(nodeIn) {
-            if (!ts.isGeneratedIdentifier(nodeIn)) {
-                var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
-                if (node) {
-                    var symbol = getReferencedValueSymbol(node);
-                    if (symbol && isSymbolOfDeclarationWithCollidingName(symbol)) {
-                        return symbol.valueDeclaration;
-                    }
+            if (use & 2 /* AllowsAsyncIterablesFlag */) {
+                var iterationTypes = getIterationTypesOfIterableSlow(type, asyncIterationTypesResolver, errorNode);
+                if (iterationTypes !== noIterationTypes) {
+                    return iterationTypes;
                 }
             }
-            return undefined;
-        }
-        // Return true if the given node is a declaration of a nested block scoped entity with a name that either hides an
-        // existing name or might hide a name when compiled downlevel
-        function isDeclarationWithCollidingName(nodeIn) {
-            var node = ts.getParseTreeNode(nodeIn, ts.isDeclaration);
-            if (node) {
-                var symbol = getSymbolOfNode(node);
-                if (symbol) {
-                    return isSymbolOfDeclarationWithCollidingName(symbol);
+            if (use & 1 /* AllowsSyncIterablesFlag */) {
+                var iterationTypes = getIterationTypesOfIterableSlow(type, syncIterationTypesResolver, errorNode);
+                if (iterationTypes !== noIterationTypes) {
+                    if (use & 2 /* AllowsAsyncIterablesFlag */) {
+                        return setCachedIterationTypes(type, "iterationTypesOfAsyncIterable", iterationTypes
+                            ? getAsyncFromSyncIterationTypes(iterationTypes, errorNode)
+                            : noIterationTypes);
+                    }
+                    else {
+                        return iterationTypes;
+                    }
                 }
             }
-            return false;
+            return noIterationTypes;
         }
-        function isValueAliasDeclaration(node) {
-            switch (node.kind) {
-                case 253 /* ImportEqualsDeclaration */:
-                    return isAliasResolvedToValue(getSymbolOfNode(node) || unknownSymbol);
-                case 255 /* ImportClause */:
-                case 256 /* NamespaceImport */:
-                case 258 /* ImportSpecifier */:
-                case 263 /* ExportSpecifier */:
-                    var symbol = getSymbolOfNode(node) || unknownSymbol;
-                    return isAliasResolvedToValue(symbol) && !getTypeOnlyAliasDeclaration(symbol);
-                case 260 /* ExportDeclaration */:
-                    var exportClause = node.exportClause;
-                    return !!exportClause && (ts.isNamespaceExport(exportClause) ||
-                        ts.some(exportClause.elements, isValueAliasDeclaration));
-                case 259 /* ExportAssignment */:
-                    return node.expression && node.expression.kind === 75 /* Identifier */ ?
-                        isAliasResolvedToValue(getSymbolOfNode(node) || unknownSymbol) :
-                        true;
-            }
-            return false;
+        /**
+         * Gets the *yield*, *return*, and *next* types of an `Iterable`-like or
+         * `AsyncIterable`-like type from the cache.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterable` instead.
+         */
+        function getIterationTypesOfIterableCached(type, resolver) {
+            return getCachedIterationTypes(type, resolver.iterableCacheKey);
         }
-        function isTopLevelValueImportEqualsWithEntityName(nodeIn) {
-            var node = ts.getParseTreeNode(nodeIn, ts.isImportEqualsDeclaration);
-            if (node === undefined || node.parent.kind !== 290 /* SourceFile */ || !ts.isInternalModuleImportEqualsDeclaration(node)) {
-                // parent is not source file or it is not reference to internal module
-                return false;
-            }
-            var isValue = isAliasResolvedToValue(getSymbolOfNode(node));
-            return isValue && node.moduleReference && !ts.nodeIsMissing(node.moduleReference);
+        function getIterationTypesOfGlobalIterableType(globalType, resolver) {
+            var globalIterationTypes = getIterationTypesOfIterableCached(globalType, resolver) ||
+                getIterationTypesOfIterableSlow(globalType, resolver, /*errorNode*/ undefined);
+            return globalIterationTypes === noIterationTypes ? defaultIterationTypes : globalIterationTypes;
         }
-        function isAliasResolvedToValue(symbol) {
-            var target = resolveAlias(symbol);
-            if (target === unknownSymbol) {
-                return true;
+        /**
+         * Gets the *yield*, *return*, and *next* types of an `Iterable`-like or `AsyncIterable`-like
+         * type from from common heuristics.
+         *
+         * If we previously analyzed this type and found no iteration types, `noIterationTypes` is
+         * returned. If we found iteration types, an `IterationTypes` record is returned.
+         * Otherwise, we return `undefined` to indicate to the caller it should perform a more
+         * exhaustive analysis.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterable` instead.
+         */
+        function getIterationTypesOfIterableFast(type, resolver) {
+            // As an optimization, if the type is an instantiation of one of the following global types, then
+            // just grab its related type argument:
+            // - `Iterable<T>` or `AsyncIterable<T>`
+            // - `IterableIterator<T>` or `AsyncIterableIterator<T>`
+            var globalType;
+            if (isReferenceToType(type, globalType = resolver.getGlobalIterableType(/*reportErrors*/ false)) ||
+                isReferenceToType(type, globalType = resolver.getGlobalIterableIteratorType(/*reportErrors*/ false))) {
+                var yieldType = getTypeArguments(type)[0];
+                // The "return" and "next" types of `Iterable` and `IterableIterator` are defined by the
+                // iteration types of their `[Symbol.iterator]()` method. The same is true for their async cousins.
+                // While we define these as `any` and `undefined` in our libs by default, a custom lib *could* use
+                // different definitions.
+                var _a = getIterationTypesOfGlobalIterableType(globalType, resolver), returnType = _a.returnType, nextType = _a.nextType;
+                return setCachedIterationTypes(type, resolver.iterableCacheKey, createIterationTypes(yieldType, returnType, nextType));
+            }
+            // As an optimization, if the type is an instantiation of the following global type, then
+            // just grab its related type arguments:
+            // - `Generator<T, TReturn, TNext>` or `AsyncGenerator<T, TReturn, TNext>`
+            if (isReferenceToType(type, resolver.getGlobalGeneratorType(/*reportErrors*/ false))) {
+                var _b = getTypeArguments(type), yieldType = _b[0], returnType = _b[1], nextType = _b[2];
+                return setCachedIterationTypes(type, resolver.iterableCacheKey, createIterationTypes(yieldType, returnType, nextType));
             }
-            // const enums and modules that contain only const enums are not considered values from the emit perspective
-            // unless 'preserveConstEnums' option is set to true
-            return !!(target.flags & 111551 /* Value */) &&
-                (compilerOptions.preserveConstEnums || !isConstEnumOrConstEnumOnlyModule(target));
-        }
-        function isConstEnumOrConstEnumOnlyModule(s) {
-            return isConstEnumSymbol(s) || !!s.constEnumOnlyModule;
         }
-        function isReferencedAliasDeclaration(node, checkChildren) {
-            if (isAliasSymbolDeclaration(node)) {
-                var symbol = getSymbolOfNode(node);
-                if (symbol && getSymbolLinks(symbol).referenced) {
-                    return true;
-                }
-                var target = getSymbolLinks(symbol).target; // TODO: GH#18217
-                if (target && ts.getModifierFlags(node) & 1 /* Export */ &&
-                    target.flags & 111551 /* Value */ &&
-                    (compilerOptions.preserveConstEnums || !isConstEnumOrConstEnumOnlyModule(target))) {
-                    // An `export import ... =` of a value symbol is always considered referenced
-                    return true;
-                }
+        /**
+         * Gets the *yield*, *return*, and *next* types of an `Iterable`-like or `AsyncIterable`-like
+         * type from its members.
+         *
+         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
+         * record is returned. Otherwise, `noIterationTypes` is returned.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterable` instead.
+         */
+        function getIterationTypesOfIterableSlow(type, resolver, errorNode) {
+            var _a;
+            var method = getPropertyOfType(type, ts.getPropertyNameForKnownSymbolName(resolver.iteratorSymbolName));
+            var methodType = method && !(method.flags & 16777216 /* Optional */) ? getTypeOfSymbol(method) : undefined;
+            if (isTypeAny(methodType)) {
+                return setCachedIterationTypes(type, resolver.iterableCacheKey, anyIterationTypes);
             }
-            if (checkChildren) {
-                return !!ts.forEachChild(node, function (node) { return isReferencedAliasDeclaration(node, checkChildren); });
+            var signatures = methodType ? getSignaturesOfType(methodType, 0 /* Call */) : undefined;
+            if (!ts.some(signatures)) {
+                return setCachedIterationTypes(type, resolver.iterableCacheKey, noIterationTypes);
             }
-            return false;
+            var iteratorType = getIntersectionType(ts.map(signatures, getReturnTypeOfSignature));
+            var iterationTypes = (_a = getIterationTypesOfIterator(iteratorType, resolver, errorNode)) !== null && _a !== void 0 ? _a : noIterationTypes;
+            return setCachedIterationTypes(type, resolver.iterableCacheKey, iterationTypes);
         }
-        function isImplementationOfOverload(node) {
-            if (ts.nodeIsPresent(node.body)) {
-                if (ts.isGetAccessor(node) || ts.isSetAccessor(node))
-                    return false; // Get or set accessors can never be overload implementations, but can have up to 2 signatures
-                var symbol = getSymbolOfNode(node);
-                var signaturesOfSymbol = getSignaturesOfSymbol(symbol);
-                // If this function body corresponds to function with multiple signature, it is implementation of overload
-                // e.g.: function foo(a: string): string;
-                //       function foo(a: number): number;
-                //       function foo(a: any) { // This is implementation of the overloads
-                //           return a;
-                //       }
-                return signaturesOfSymbol.length > 1 ||
-                    // If there is single signature for the symbol, it is overload if that signature isn't coming from the node
-                    // e.g.: function foo(a: string): string;
-                    //       function foo(a: any) { // This is implementation of the overloads
-                    //           return a;
-                    //       }
-                    (signaturesOfSymbol.length === 1 && signaturesOfSymbol[0].declaration !== node);
-            }
-            return false;
+        function reportTypeNotIterableError(errorNode, type, allowAsyncIterables) {
+            var message = allowAsyncIterables
+                ? ts.Diagnostics.Type_0_must_have_a_Symbol_asyncIterator_method_that_returns_an_async_iterator
+                : ts.Diagnostics.Type_0_must_have_a_Symbol_iterator_method_that_returns_an_iterator;
+            errorAndMaybeSuggestAwait(errorNode, !!getAwaitedTypeOfPromise(type), message, typeToString(type));
         }
-        function isRequiredInitializedParameter(parameter) {
-            return !!strictNullChecks &&
-                !isOptionalParameter(parameter) &&
-                !ts.isJSDocParameterTag(parameter) &&
-                !!parameter.initializer &&
-                !ts.hasModifier(parameter, 92 /* ParameterPropertyModifier */);
+        /**
+         * Gets the *yield*, *return*, and *next* types from an `Iterator`-like or `AsyncIterator`-like type.
+         *
+         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
+         * record is returned. Otherwise, `undefined` is returned.
+         */
+        function getIterationTypesOfIterator(type, resolver, errorNode) {
+            if (isTypeAny(type)) {
+                return anyIterationTypes;
+            }
+            var iterationTypes = getIterationTypesOfIteratorCached(type, resolver) ||
+                getIterationTypesOfIteratorFast(type, resolver) ||
+                getIterationTypesOfIteratorSlow(type, resolver, errorNode);
+            return iterationTypes === noIterationTypes ? undefined : iterationTypes;
         }
-        function isOptionalUninitializedParameterProperty(parameter) {
-            return strictNullChecks &&
-                isOptionalParameter(parameter) &&
-                !parameter.initializer &&
-                ts.hasModifier(parameter, 92 /* ParameterPropertyModifier */);
+        /**
+         * Gets the iteration types of an `Iterator`-like or `AsyncIterator`-like type from the
+         * cache.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterator` instead.
+         */
+        function getIterationTypesOfIteratorCached(type, resolver) {
+            return getCachedIterationTypes(type, resolver.iteratorCacheKey);
         }
-        function isExpandoFunctionDeclaration(node) {
-            var declaration = ts.getParseTreeNode(node, ts.isFunctionDeclaration);
-            if (!declaration) {
-                return false;
+        /**
+         * Gets the iteration types of an `Iterator`-like or `AsyncIterator`-like type from the
+         * cache or from common heuristics.
+         *
+         * If we previously analyzed this type and found no iteration types, `noIterationTypes` is
+         * returned. If we found iteration types, an `IterationTypes` record is returned.
+         * Otherwise, we return `undefined` to indicate to the caller it should perform a more
+         * exhaustive analysis.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterator` instead.
+         */
+        function getIterationTypesOfIteratorFast(type, resolver) {
+            // As an optimization, if the type is an instantiation of one of the following global types,
+            // then just grab its related type argument:
+            // - `IterableIterator<T>` or `AsyncIterableIterator<T>`
+            // - `Iterator<T, TReturn, TNext>` or `AsyncIterator<T, TReturn, TNext>`
+            // - `Generator<T, TReturn, TNext>` or `AsyncGenerator<T, TReturn, TNext>`
+            var globalType = resolver.getGlobalIterableIteratorType(/*reportErrors*/ false);
+            if (isReferenceToType(type, globalType)) {
+                var yieldType = getTypeArguments(type)[0];
+                // The "return" and "next" types of `IterableIterator` and `AsyncIterableIterator` are defined by the
+                // iteration types of their `next`, `return`, and `throw` methods. While we define these as `any`
+                // and `undefined` in our libs by default, a custom lib *could* use different definitions.
+                var globalIterationTypes = getIterationTypesOfIteratorCached(globalType, resolver) ||
+                    getIterationTypesOfIteratorSlow(globalType, resolver, /*errorNode*/ undefined);
+                var _a = globalIterationTypes === noIterationTypes ? defaultIterationTypes : globalIterationTypes, returnType = _a.returnType, nextType = _a.nextType;
+                return setCachedIterationTypes(type, resolver.iteratorCacheKey, createIterationTypes(yieldType, returnType, nextType));
             }
-            var symbol = getSymbolOfNode(declaration);
-            if (!symbol || !(symbol.flags & 16 /* Function */)) {
-                return false;
+            if (isReferenceToType(type, resolver.getGlobalIteratorType(/*reportErrors*/ false)) ||
+                isReferenceToType(type, resolver.getGlobalGeneratorType(/*reportErrors*/ false))) {
+                var _b = getTypeArguments(type), yieldType = _b[0], returnType = _b[1], nextType = _b[2];
+                return setCachedIterationTypes(type, resolver.iteratorCacheKey, createIterationTypes(yieldType, returnType, nextType));
             }
-            return !!ts.forEachEntry(getExportsOfSymbol(symbol), function (p) { return p.flags & 111551 /* Value */ && p.valueDeclaration && ts.isPropertyAccessExpression(p.valueDeclaration); });
         }
-        function getPropertiesOfContainerFunction(node) {
-            var declaration = ts.getParseTreeNode(node, ts.isFunctionDeclaration);
-            if (!declaration) {
-                return ts.emptyArray;
-            }
-            var symbol = getSymbolOfNode(declaration);
-            return symbol && getPropertiesOfType(getTypeOfSymbol(symbol)) || ts.emptyArray;
+        function isIteratorResult(type, kind) {
+            // From https://tc39.github.io/ecma262/#sec-iteratorresult-interface:
+            // > [done] is the result status of an iterator `next` method call. If the end of the iterator was reached `done` is `true`.
+            // > If the end was not reached `done` is `false` and a value is available.
+            // > If a `done` property (either own or inherited) does not exist, it is consider to have the value `false`.
+            var doneType = getTypeOfPropertyOfType(type, "done") || falseType;
+            return isTypeAssignableTo(kind === 0 /* Yield */ ? falseType : trueType, doneType);
         }
-        function getNodeCheckFlags(node) {
-            return getNodeLinks(node).flags || 0;
+        function isYieldIteratorResult(type) {
+            return isIteratorResult(type, 0 /* Yield */);
         }
-        function getEnumMemberValue(node) {
-            computeEnumMemberValues(node.parent);
-            return getNodeLinks(node).enumMemberValue;
+        function isReturnIteratorResult(type) {
+            return isIteratorResult(type, 1 /* Return */);
         }
-        function canHaveConstantValue(node) {
-            switch (node.kind) {
-                case 284 /* EnumMember */:
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                    return true;
+        /**
+         * Gets the *yield* and *return* types of an `IteratorResult`-like type.
+         *
+         * If we are unable to determine a *yield* or a *return* type, `noIterationTypes` is
+         * returned to indicate to the caller that it should handle the error. Otherwise, an
+         * `IterationTypes` record is returned.
+         */
+        function getIterationTypesOfIteratorResult(type) {
+            if (isTypeAny(type)) {
+                return anyIterationTypes;
             }
-            return false;
-        }
-        function getConstantValue(node) {
-            if (node.kind === 284 /* EnumMember */) {
-                return getEnumMemberValue(node);
+            var cachedTypes = getCachedIterationTypes(type, "iterationTypesOfIteratorResult");
+            if (cachedTypes) {
+                return cachedTypes;
             }
-            var symbol = getNodeLinks(node).resolvedSymbol;
-            if (symbol && (symbol.flags & 8 /* EnumMember */)) {
-                // inline property\index accesses only for const enums
-                var member = symbol.valueDeclaration;
-                if (ts.isEnumConst(member.parent)) {
-                    return getEnumMemberValue(member);
-                }
+            // As an optimization, if the type is an instantiation of one of the global `IteratorYieldResult<T>`
+            // or `IteratorReturnResult<TReturn>` types, then just grab its type argument.
+            if (isReferenceToType(type, getGlobalIteratorYieldResultType(/*reportErrors*/ false))) {
+                var yieldType_1 = getTypeArguments(type)[0];
+                return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", createIterationTypes(yieldType_1, /*returnType*/ undefined, /*nextType*/ undefined));
             }
-            return undefined;
-        }
-        function isFunctionType(type) {
-            return !!(type.flags & 524288 /* Object */) && getSignaturesOfType(type, 0 /* Call */).length > 0;
+            if (isReferenceToType(type, getGlobalIteratorReturnResultType(/*reportErrors*/ false))) {
+                var returnType_1 = getTypeArguments(type)[0];
+                return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", createIterationTypes(/*yieldType*/ undefined, returnType_1, /*nextType*/ undefined));
+            }
+            // Choose any constituents that can produce the requested iteration type.
+            var yieldIteratorResult = filterType(type, isYieldIteratorResult);
+            var yieldType = yieldIteratorResult !== neverType ? getTypeOfPropertyOfType(yieldIteratorResult, "value") : undefined;
+            var returnIteratorResult = filterType(type, isReturnIteratorResult);
+            var returnType = returnIteratorResult !== neverType ? getTypeOfPropertyOfType(returnIteratorResult, "value") : undefined;
+            if (!yieldType && !returnType) {
+                return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", noIterationTypes);
+            }
+            // From https://tc39.github.io/ecma262/#sec-iteratorresult-interface
+            // > ... If the iterator does not have a return value, `value` is `undefined`. In that case, the
+            // > `value` property may be absent from the conforming object if it does not inherit an explicit
+            // > `value` property.
+            return setCachedIterationTypes(type, "iterationTypesOfIteratorResult", createIterationTypes(yieldType, returnType || voidType, /*nextType*/ undefined));
         }
-        function getTypeReferenceSerializationKind(typeNameIn, location) {
-            // ensure both `typeName` and `location` are parse tree nodes.
-            var typeName = ts.getParseTreeNode(typeNameIn, ts.isEntityName);
-            if (!typeName)
-                return ts.TypeReferenceSerializationKind.Unknown;
-            if (location) {
-                location = ts.getParseTreeNode(location);
-                if (!location)
-                    return ts.TypeReferenceSerializationKind.Unknown;
+        /**
+         * Gets the *yield*, *return*, and *next* types of a the `next()`, `return()`, or
+         * `throw()` method of an `Iterator`-like or `AsyncIterator`-like type.
+         *
+         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
+         * record is returned. Otherwise, we return `undefined`.
+         */
+        function getIterationTypesOfMethod(type, resolver, methodName, errorNode) {
+            var _a, _b, _c, _d;
+            var method = getPropertyOfType(type, methodName);
+            // Ignore 'return' or 'throw' if they are missing.
+            if (!method && methodName !== "next") {
+                return undefined;
             }
-            // Resolve the symbol as a value to ensure the type can be reached at runtime during emit.
-            var valueSymbol = resolveEntityName(typeName, 111551 /* Value */, /*ignoreErrors*/ true, /*dontResolveAlias*/ false, location);
-            // Resolve the symbol as a type so that we can provide a more useful hint for the type serializer.
-            var typeSymbol = resolveEntityName(typeName, 788968 /* Type */, /*ignoreErrors*/ true, /*dontResolveAlias*/ false, location);
-            if (valueSymbol && valueSymbol === typeSymbol) {
-                var globalPromiseSymbol = getGlobalPromiseConstructorSymbol(/*reportErrors*/ false);
-                if (globalPromiseSymbol && valueSymbol === globalPromiseSymbol) {
-                    return ts.TypeReferenceSerializationKind.Promise;
+            var methodType = method && !(methodName === "next" && (method.flags & 16777216 /* Optional */))
+                ? methodName === "next" ? getTypeOfSymbol(method) : getTypeWithFacts(getTypeOfSymbol(method), 2097152 /* NEUndefinedOrNull */)
+                : undefined;
+            if (isTypeAny(methodType)) {
+                // `return()` and `throw()` don't provide a *next* type.
+                return methodName === "next" ? anyIterationTypes : anyIterationTypesExceptNext;
+            }
+            // Both async and non-async iterators *must* have a `next` method.
+            var methodSignatures = methodType ? getSignaturesOfType(methodType, 0 /* Call */) : ts.emptyArray;
+            if (methodSignatures.length === 0) {
+                if (errorNode) {
+                    var diagnostic = methodName === "next"
+                        ? resolver.mustHaveANextMethodDiagnostic
+                        : resolver.mustBeAMethodDiagnostic;
+                    error(errorNode, diagnostic, methodName);
                 }
-                var constructorType = getTypeOfSymbol(valueSymbol);
-                if (constructorType && isConstructorType(constructorType)) {
-                    return ts.TypeReferenceSerializationKind.TypeWithConstructSignatureAndValue;
+                return methodName === "next" ? anyIterationTypes : undefined;
+            }
+            // If the method signature comes exclusively from the global iterator or generator type,
+            // create iteration types from its type arguments like `getIterationTypesOfIteratorFast`
+            // does (so as to remove `undefined` from the next and return types). We arrive here when
+            // a contextual type for a generator was not a direct reference to one of those global types,
+            // but looking up `methodType` referred to one of them (and nothing else). E.g., in
+            // `interface SpecialIterator extends Iterator<number> {}`, `SpecialIterator` is not a
+            // reference to `Iterator`, but its `next` member derives exclusively from `Iterator`.
+            if ((methodType === null || methodType === void 0 ? void 0 : methodType.symbol) && methodSignatures.length === 1) {
+                var globalGeneratorType = resolver.getGlobalGeneratorType(/*reportErrors*/ false);
+                var globalIteratorType = resolver.getGlobalIteratorType(/*reportErrors*/ false);
+                var isGeneratorMethod = ((_b = (_a = globalGeneratorType.symbol) === null || _a === void 0 ? void 0 : _a.members) === null || _b === void 0 ? void 0 : _b.get(methodName)) === methodType.symbol;
+                var isIteratorMethod = !isGeneratorMethod && ((_d = (_c = globalIteratorType.symbol) === null || _c === void 0 ? void 0 : _c.members) === null || _d === void 0 ? void 0 : _d.get(methodName)) === methodType.symbol;
+                if (isGeneratorMethod || isIteratorMethod) {
+                    var globalType = isGeneratorMethod ? globalGeneratorType : globalIteratorType;
+                    var mapper = methodType.mapper;
+                    return createIterationTypes(getMappedType(globalType.typeParameters[0], mapper), getMappedType(globalType.typeParameters[1], mapper), methodName === "next" ? getMappedType(globalType.typeParameters[2], mapper) : undefined);
                 }
             }
-            // We might not be able to resolve type symbol so use unknown type in that case (eg error case)
-            if (!typeSymbol) {
-                return ts.TypeReferenceSerializationKind.Unknown;
+            // Extract the first parameter and return type of each signature.
+            var methodParameterTypes;
+            var methodReturnTypes;
+            for (var _i = 0, methodSignatures_1 = methodSignatures; _i < methodSignatures_1.length; _i++) {
+                var signature = methodSignatures_1[_i];
+                if (methodName !== "throw" && ts.some(signature.parameters)) {
+                    methodParameterTypes = ts.append(methodParameterTypes, getTypeAtPosition(signature, 0));
+                }
+                methodReturnTypes = ts.append(methodReturnTypes, getReturnTypeOfSignature(signature));
             }
-            var type = getDeclaredTypeOfSymbol(typeSymbol);
-            if (type === errorType) {
-                return ts.TypeReferenceSerializationKind.Unknown;
+            // Resolve the *next* or *return* type from the first parameter of a `next()` or
+            // `return()` method, respectively.
+            var returnTypes;
+            var nextType;
+            if (methodName !== "throw") {
+                var methodParameterType = methodParameterTypes ? getUnionType(methodParameterTypes) : unknownType;
+                if (methodName === "next") {
+                    // The value of `next(value)` is *not* awaited by async generators
+                    nextType = methodParameterType;
+                }
+                else if (methodName === "return") {
+                    // The value of `return(value)` *is* awaited by async generators
+                    var resolvedMethodParameterType = resolver.resolveIterationType(methodParameterType, errorNode) || anyType;
+                    returnTypes = ts.append(returnTypes, resolvedMethodParameterType);
+                }
             }
-            else if (type.flags & 3 /* AnyOrUnknown */) {
-                return ts.TypeReferenceSerializationKind.ObjectType;
+            // Resolve the *yield* and *return* types from the return type of the method (i.e. `IteratorResult`)
+            var yieldType;
+            var methodReturnType = methodReturnTypes ? getIntersectionType(methodReturnTypes) : neverType;
+            var resolvedMethodReturnType = resolver.resolveIterationType(methodReturnType, errorNode) || anyType;
+            var iterationTypes = getIterationTypesOfIteratorResult(resolvedMethodReturnType);
+            if (iterationTypes === noIterationTypes) {
+                if (errorNode) {
+                    error(errorNode, resolver.mustHaveAValueDiagnostic, methodName);
+                }
+                yieldType = anyType;
+                returnTypes = ts.append(returnTypes, anyType);
             }
-            else if (isTypeAssignableToKind(type, 16384 /* Void */ | 98304 /* Nullable */ | 131072 /* Never */)) {
-                return ts.TypeReferenceSerializationKind.VoidNullableOrNeverType;
+            else {
+                yieldType = iterationTypes.yieldType;
+                returnTypes = ts.append(returnTypes, iterationTypes.returnType);
             }
-            else if (isTypeAssignableToKind(type, 528 /* BooleanLike */)) {
-                return ts.TypeReferenceSerializationKind.BooleanType;
+            return createIterationTypes(yieldType, getUnionType(returnTypes), nextType);
+        }
+        /**
+         * Gets the *yield*, *return*, and *next* types of an `Iterator`-like or `AsyncIterator`-like
+         * type from its members.
+         *
+         * If we successfully found the *yield*, *return*, and *next* types, an `IterationTypes`
+         * record is returned. Otherwise, `noIterationTypes` is returned.
+         *
+         * NOTE: You probably don't want to call this directly and should be calling
+         * `getIterationTypesOfIterator` instead.
+         */
+        function getIterationTypesOfIteratorSlow(type, resolver, errorNode) {
+            var iterationTypes = combineIterationTypes([
+                getIterationTypesOfMethod(type, resolver, "next", errorNode),
+                getIterationTypesOfMethod(type, resolver, "return", errorNode),
+                getIterationTypesOfMethod(type, resolver, "throw", errorNode),
+            ]);
+            return setCachedIterationTypes(type, resolver.iteratorCacheKey, iterationTypes);
+        }
+        /**
+         * Gets the requested "iteration type" from a type that is either `Iterable`-like, `Iterator`-like,
+         * `IterableIterator`-like, or `Generator`-like (for a non-async generator); or `AsyncIterable`-like,
+         * `AsyncIterator`-like, `AsyncIterableIterator`-like, or `AsyncGenerator`-like (for an async generator).
+         */
+        function getIterationTypeOfGeneratorFunctionReturnType(kind, returnType, isAsyncGenerator) {
+            if (isTypeAny(returnType)) {
+                return undefined;
             }
-            else if (isTypeAssignableToKind(type, 296 /* NumberLike */)) {
-                return ts.TypeReferenceSerializationKind.NumberLikeType;
+            var iterationTypes = getIterationTypesOfGeneratorFunctionReturnType(returnType, isAsyncGenerator);
+            return iterationTypes && iterationTypes[getIterationTypesKeyFromIterationTypeKind(kind)];
+        }
+        function getIterationTypesOfGeneratorFunctionReturnType(type, isAsyncGenerator) {
+            if (isTypeAny(type)) {
+                return anyIterationTypes;
             }
-            else if (isTypeAssignableToKind(type, 2112 /* BigIntLike */)) {
-                return ts.TypeReferenceSerializationKind.BigIntLikeType;
+            var use = isAsyncGenerator ? 2 /* AsyncGeneratorReturnType */ : 1 /* GeneratorReturnType */;
+            var resolver = isAsyncGenerator ? asyncIterationTypesResolver : syncIterationTypesResolver;
+            return getIterationTypesOfIterable(type, use, /*errorNode*/ undefined) ||
+                getIterationTypesOfIterator(type, resolver, /*errorNode*/ undefined);
+        }
+        function checkBreakOrContinueStatement(node) {
+            // Grammar checking
+            if (!checkGrammarStatementInAmbientContext(node))
+                checkGrammarBreakOrContinueStatement(node);
+            // TODO: Check that target label is valid
+        }
+        function unwrapReturnType(returnType, functionFlags) {
+            var _a, _b;
+            var isGenerator = !!(functionFlags & 1 /* Generator */);
+            var isAsync = !!(functionFlags & 2 /* Async */);
+            return isGenerator ? (_a = getIterationTypeOfGeneratorFunctionReturnType(1 /* Return */, returnType, isAsync)) !== null && _a !== void 0 ? _a : errorType :
+                isAsync ? (_b = getAwaitedType(returnType)) !== null && _b !== void 0 ? _b : errorType :
+                    returnType;
+        }
+        function isUnwrappedReturnTypeVoidOrAny(func, returnType) {
+            var unwrappedReturnType = unwrapReturnType(returnType, ts.getFunctionFlags(func));
+            return !!unwrappedReturnType && maybeTypeOfKind(unwrappedReturnType, 16384 /* Void */ | 3 /* AnyOrUnknown */);
+        }
+        function checkReturnStatement(node) {
+            var _a;
+            // Grammar checking
+            if (checkGrammarStatementInAmbientContext(node)) {
+                return;
             }
-            else if (isTypeAssignableToKind(type, 132 /* StringLike */)) {
-                return ts.TypeReferenceSerializationKind.StringLikeType;
+            var func = ts.getContainingFunction(node);
+            if (!func) {
+                grammarErrorOnFirstToken(node, ts.Diagnostics.A_return_statement_can_only_be_used_within_a_function_body);
+                return;
             }
-            else if (isTupleType(type)) {
-                return ts.TypeReferenceSerializationKind.ArrayLikeType;
+            var signature = getSignatureFromDeclaration(func);
+            var returnType = getReturnTypeOfSignature(signature);
+            var functionFlags = ts.getFunctionFlags(func);
+            if (strictNullChecks || node.expression || returnType.flags & 131072 /* Never */) {
+                var exprType = node.expression ? checkExpressionCached(node.expression) : undefinedType;
+                if (func.kind === 168 /* SetAccessor */) {
+                    if (node.expression) {
+                        error(node, ts.Diagnostics.Setters_cannot_return_a_value);
+                    }
+                }
+                else if (func.kind === 166 /* Constructor */) {
+                    if (node.expression && !checkTypeAssignableToAndOptionallyElaborate(exprType, returnType, node, node.expression)) {
+                        error(node, ts.Diagnostics.Return_type_of_constructor_signature_must_be_assignable_to_the_instance_type_of_the_class);
+                    }
+                }
+                else if (getReturnTypeFromAnnotation(func)) {
+                    var unwrappedReturnType = (_a = unwrapReturnType(returnType, functionFlags)) !== null && _a !== void 0 ? _a : returnType;
+                    var unwrappedExprType = functionFlags & 2 /* Async */
+                        ? checkAwaitedType(exprType, node, ts.Diagnostics.The_return_type_of_an_async_function_must_either_be_a_valid_promise_or_must_not_contain_a_callable_then_member)
+                        : exprType;
+                    if (unwrappedReturnType) {
+                        // If the function has a return type, but promisedType is
+                        // undefined, an error will be reported in checkAsyncFunctionReturnType
+                        // so we don't need to report one here.
+                        checkTypeAssignableToAndOptionallyElaborate(unwrappedExprType, unwrappedReturnType, node, node.expression);
+                    }
+                }
             }
-            else if (isTypeAssignableToKind(type, 12288 /* ESSymbolLike */)) {
-                return ts.TypeReferenceSerializationKind.ESSymbolType;
+            else if (func.kind !== 166 /* Constructor */ && compilerOptions.noImplicitReturns && !isUnwrappedReturnTypeVoidOrAny(func, returnType)) {
+                // The function has a return type, but the return statement doesn't have an expression.
+                error(node, ts.Diagnostics.Not_all_code_paths_return_a_value);
             }
-            else if (isFunctionType(type)) {
-                return ts.TypeReferenceSerializationKind.TypeWithCallSignature;
+        }
+        function checkWithStatement(node) {
+            // Grammar checking for withStatement
+            if (!checkGrammarStatementInAmbientContext(node)) {
+                if (node.flags & 32768 /* AwaitContext */) {
+                    grammarErrorOnFirstToken(node, ts.Diagnostics.with_statements_are_not_allowed_in_an_async_function_block);
+                }
             }
-            else if (isArrayType(type)) {
-                return ts.TypeReferenceSerializationKind.ArrayLikeType;
+            checkExpression(node.expression);
+            var sourceFile = ts.getSourceFileOfNode(node);
+            if (!hasParseDiagnostics(sourceFile)) {
+                var start = ts.getSpanOfTokenAtPosition(sourceFile, node.pos).start;
+                var end = node.statement.pos;
+                grammarErrorAtPos(sourceFile, start, end - start, ts.Diagnostics.The_with_statement_is_not_supported_All_symbols_in_a_with_block_will_have_type_any);
             }
-            else {
-                return ts.TypeReferenceSerializationKind.ObjectType;
+        }
+        function checkSwitchStatement(node) {
+            // Grammar checking
+            checkGrammarStatementInAmbientContext(node);
+            var firstDefaultClause;
+            var hasDuplicateDefaultClause = false;
+            var expressionType = checkExpression(node.expression);
+            var expressionIsLiteral = isLiteralType(expressionType);
+            ts.forEach(node.caseBlock.clauses, function (clause) {
+                // Grammar check for duplicate default clauses, skip if we already report duplicate default clause
+                if (clause.kind === 285 /* DefaultClause */ && !hasDuplicateDefaultClause) {
+                    if (firstDefaultClause === undefined) {
+                        firstDefaultClause = clause;
+                    }
+                    else {
+                        grammarErrorOnNode(clause, ts.Diagnostics.A_default_clause_cannot_appear_more_than_once_in_a_switch_statement);
+                        hasDuplicateDefaultClause = true;
+                    }
+                }
+                if (produceDiagnostics && clause.kind === 284 /* CaseClause */) {
+                    // TypeScript 1.0 spec (April 2014): 5.9
+                    // In a 'switch' statement, each 'case' expression must be of a type that is comparable
+                    // to or from the type of the 'switch' expression.
+                    var caseType = checkExpression(clause.expression);
+                    var caseIsLiteral = isLiteralType(caseType);
+                    var comparedExpressionType = expressionType;
+                    if (!caseIsLiteral || !expressionIsLiteral) {
+                        caseType = caseIsLiteral ? getBaseTypeOfLiteralType(caseType) : caseType;
+                        comparedExpressionType = getBaseTypeOfLiteralType(expressionType);
+                    }
+                    if (!isTypeEqualityComparableTo(comparedExpressionType, caseType)) {
+                        // expressionType is not comparable to caseType, try the reversed check and report errors if it fails
+                        checkTypeComparableTo(caseType, comparedExpressionType, clause.expression, /*headMessage*/ undefined);
+                    }
+                }
+                ts.forEach(clause.statements, checkSourceElement);
+                if (compilerOptions.noFallthroughCasesInSwitch && clause.fallthroughFlowNode && isReachableFlowNode(clause.fallthroughFlowNode)) {
+                    error(clause, ts.Diagnostics.Fallthrough_case_in_switch);
+                }
+            });
+            if (node.caseBlock.locals) {
+                registerForUnusedIdentifiersCheck(node.caseBlock);
             }
         }
-        function createTypeOfDeclaration(declarationIn, enclosingDeclaration, flags, tracker, addUndefined) {
-            var declaration = ts.getParseTreeNode(declarationIn, ts.isVariableLikeOrAccessor);
-            if (!declaration) {
-                return ts.createToken(125 /* AnyKeyword */);
+        function checkLabeledStatement(node) {
+            // Grammar checking
+            if (!checkGrammarStatementInAmbientContext(node)) {
+                ts.findAncestor(node.parent, function (current) {
+                    if (ts.isFunctionLike(current)) {
+                        return "quit";
+                    }
+                    if (current.kind === 245 /* LabeledStatement */ && current.label.escapedText === node.label.escapedText) {
+                        grammarErrorOnNode(node.label, ts.Diagnostics.Duplicate_label_0, ts.getTextOfNode(node.label));
+                        return true;
+                    }
+                    return false;
+                });
             }
-            // Get type of the symbol if this is the valid symbol otherwise get type at location
-            var symbol = getSymbolOfNode(declaration);
-            var type = symbol && !(symbol.flags & (2048 /* TypeLiteral */ | 131072 /* Signature */))
-                ? getWidenedLiteralType(getTypeOfSymbol(symbol))
-                : errorType;
-            if (type.flags & 8192 /* UniqueESSymbol */ &&
-                type.symbol === symbol) {
-                flags |= 1048576 /* AllowUniqueESSymbolType */;
+            // ensure that label is unique
+            checkSourceElement(node.statement);
+        }
+        function checkThrowStatement(node) {
+            // Grammar checking
+            if (!checkGrammarStatementInAmbientContext(node)) {
+                if (ts.isIdentifier(node.expression) && !node.expression.escapedText) {
+                    grammarErrorAfterFirstToken(node, ts.Diagnostics.Line_break_not_permitted_here);
+                }
             }
-            if (addUndefined) {
-                type = getOptionalType(type);
+            if (node.expression) {
+                checkExpression(node.expression);
             }
-            return nodeBuilder.typeToTypeNode(type, enclosingDeclaration, flags | 1024 /* MultilineObjectLiterals */, tracker);
         }
-        function createReturnTypeOfSignatureDeclaration(signatureDeclarationIn, enclosingDeclaration, flags, tracker) {
-            var signatureDeclaration = ts.getParseTreeNode(signatureDeclarationIn, ts.isFunctionLike);
-            if (!signatureDeclaration) {
-                return ts.createToken(125 /* AnyKeyword */);
+        function checkTryStatement(node) {
+            // Grammar checking
+            checkGrammarStatementInAmbientContext(node);
+            checkBlock(node.tryBlock);
+            var catchClause = node.catchClause;
+            if (catchClause) {
+                // Grammar checking
+                if (catchClause.variableDeclaration) {
+                    var declaration = catchClause.variableDeclaration;
+                    if (declaration.type) {
+                        var type = getTypeForVariableLikeDeclaration(declaration, /*includeOptionality*/ false);
+                        if (type && !(type.flags & 3 /* AnyOrUnknown */)) {
+                            grammarErrorOnFirstToken(declaration.type, ts.Diagnostics.Catch_clause_variable_type_annotation_must_be_any_or_unknown_if_specified);
+                        }
+                    }
+                    else if (declaration.initializer) {
+                        grammarErrorOnFirstToken(declaration.initializer, ts.Diagnostics.Catch_clause_variable_cannot_have_an_initializer);
+                    }
+                    else {
+                        var blockLocals_1 = catchClause.block.locals;
+                        if (blockLocals_1) {
+                            ts.forEachKey(catchClause.locals, function (caughtName) {
+                                var blockLocal = blockLocals_1.get(caughtName);
+                                if (blockLocal && (blockLocal.flags & 2 /* BlockScopedVariable */) !== 0) {
+                                    grammarErrorOnNode(blockLocal.valueDeclaration, ts.Diagnostics.Cannot_redeclare_identifier_0_in_catch_clause, caughtName);
+                                }
+                            });
+                        }
+                    }
+                }
+                checkBlock(catchClause.block);
             }
-            var signature = getSignatureFromDeclaration(signatureDeclaration);
-            return nodeBuilder.typeToTypeNode(getReturnTypeOfSignature(signature), enclosingDeclaration, flags | 1024 /* MultilineObjectLiterals */, tracker);
-        }
-        function createTypeOfExpression(exprIn, enclosingDeclaration, flags, tracker) {
-            var expr = ts.getParseTreeNode(exprIn, ts.isExpression);
-            if (!expr) {
-                return ts.createToken(125 /* AnyKeyword */);
+            if (node.finallyBlock) {
+                checkBlock(node.finallyBlock);
             }
-            var type = getWidenedType(getRegularTypeOfExpression(expr));
-            return nodeBuilder.typeToTypeNode(type, enclosingDeclaration, flags | 1024 /* MultilineObjectLiterals */, tracker);
-        }
-        function hasGlobalName(name) {
-            return globals.has(ts.escapeLeadingUnderscores(name));
         }
-        function getReferencedValueSymbol(reference, startInDeclarationContainer) {
-            var resolvedSymbol = getNodeLinks(reference).resolvedSymbol;
-            if (resolvedSymbol) {
-                return resolvedSymbol;
+        function checkIndexConstraints(type) {
+            var declaredNumberIndexer = getIndexDeclarationOfSymbol(type.symbol, 1 /* Number */);
+            var declaredStringIndexer = getIndexDeclarationOfSymbol(type.symbol, 0 /* String */);
+            var stringIndexType = getIndexTypeOfType(type, 0 /* String */);
+            var numberIndexType = getIndexTypeOfType(type, 1 /* Number */);
+            if (stringIndexType || numberIndexType) {
+                ts.forEach(getPropertiesOfObjectType(type), function (prop) {
+                    var propType = getTypeOfSymbol(prop);
+                    checkIndexConstraintForProperty(prop, propType, type, declaredStringIndexer, stringIndexType, 0 /* String */);
+                    checkIndexConstraintForProperty(prop, propType, type, declaredNumberIndexer, numberIndexType, 1 /* Number */);
+                });
+                var classDeclaration = type.symbol.valueDeclaration;
+                if (ts.getObjectFlags(type) & 1 /* Class */ && ts.isClassLike(classDeclaration)) {
+                    for (var _i = 0, _a = classDeclaration.members; _i < _a.length; _i++) {
+                        var member = _a[_i];
+                        // Only process instance properties with computed names here.
+                        // Static properties cannot be in conflict with indexers,
+                        // and properties with literal names were already checked.
+                        if (!ts.hasSyntacticModifier(member, 32 /* Static */) && hasNonBindableDynamicName(member)) {
+                            var symbol = getSymbolOfNode(member);
+                            var propType = getTypeOfSymbol(symbol);
+                            checkIndexConstraintForProperty(symbol, propType, type, declaredStringIndexer, stringIndexType, 0 /* String */);
+                            checkIndexConstraintForProperty(symbol, propType, type, declaredNumberIndexer, numberIndexType, 1 /* Number */);
+                        }
+                    }
+                }
             }
-            var location = reference;
-            if (startInDeclarationContainer) {
-                // When resolving the name of a declaration as a value, we need to start resolution
-                // at a point outside of the declaration.
-                var parent = reference.parent;
-                if (ts.isDeclaration(parent) && reference === parent.name) {
-                    location = getDeclarationContainer(parent);
+            var errorNode;
+            if (stringIndexType && numberIndexType) {
+                errorNode = declaredNumberIndexer || declaredStringIndexer;
+                // condition 'errorNode === undefined' may appear if types does not declare nor string neither number indexer
+                if (!errorNode && (ts.getObjectFlags(type) & 2 /* Interface */)) {
+                    var someBaseTypeHasBothIndexers = ts.forEach(getBaseTypes(type), function (base) { return getIndexTypeOfType(base, 0 /* String */) && getIndexTypeOfType(base, 1 /* Number */); });
+                    errorNode = someBaseTypeHasBothIndexers ? undefined : type.symbol.declarations[0];
                 }
             }
-            return resolveName(location, reference.escapedText, 111551 /* Value */ | 1048576 /* ExportValue */ | 2097152 /* Alias */, /*nodeNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true);
-        }
-        function getReferencedValueDeclaration(referenceIn) {
-            if (!ts.isGeneratedIdentifier(referenceIn)) {
-                var reference = ts.getParseTreeNode(referenceIn, ts.isIdentifier);
-                if (reference) {
-                    var symbol = getReferencedValueSymbol(reference);
-                    if (symbol) {
-                        return getExportSymbolOfValueSymbolIfExported(symbol).valueDeclaration;
-                    }
+            if (errorNode && !isTypeAssignableTo(numberIndexType, stringIndexType)) { // TODO: GH#18217
+                error(errorNode, ts.Diagnostics.Numeric_index_type_0_is_not_assignable_to_string_index_type_1, typeToString(numberIndexType), typeToString(stringIndexType));
+            }
+            function checkIndexConstraintForProperty(prop, propertyType, containingType, indexDeclaration, indexType, indexKind) {
+                // ESSymbol properties apply to neither string nor numeric indexers.
+                if (!indexType || ts.isKnownSymbol(prop)) {
+                    return;
+                }
+                var propDeclaration = prop.valueDeclaration;
+                var name = propDeclaration && ts.getNameOfDeclaration(propDeclaration);
+                if (name && ts.isPrivateIdentifier(name)) {
+                    return;
+                }
+                // index is numeric and property name is not valid numeric literal
+                if (indexKind === 1 /* Number */ && !(name ? isNumericName(name) : isNumericLiteralName(prop.escapedName))) {
+                    return;
+                }
+                // perform property check if property or indexer is declared in 'type'
+                // this allows us to rule out cases when both property and indexer are inherited from the base class
+                var errorNode;
+                if (propDeclaration && name &&
+                    (propDeclaration.kind === 216 /* BinaryExpression */ ||
+                        name.kind === 158 /* ComputedPropertyName */ ||
+                        prop.parent === containingType.symbol)) {
+                    errorNode = propDeclaration;
+                }
+                else if (indexDeclaration) {
+                    errorNode = indexDeclaration;
+                }
+                else if (ts.getObjectFlags(containingType) & 2 /* Interface */) {
+                    // for interfaces property and indexer might be inherited from different bases
+                    // check if any base class already has both property and indexer.
+                    // check should be performed only if 'type' is the first type that brings property\indexer together
+                    var someBaseClassHasBothPropertyAndIndexer = ts.forEach(getBaseTypes(containingType), function (base) { return getPropertyOfObjectType(base, prop.escapedName) && getIndexTypeOfType(base, indexKind); });
+                    errorNode = someBaseClassHasBothPropertyAndIndexer ? undefined : containingType.symbol.declarations[0];
+                }
+                if (errorNode && !isTypeAssignableTo(propertyType, indexType)) {
+                    var errorMessage = indexKind === 0 /* String */
+                        ? ts.Diagnostics.Property_0_of_type_1_is_not_assignable_to_string_index_type_2
+                        : ts.Diagnostics.Property_0_of_type_1_is_not_assignable_to_numeric_index_type_2;
+                    error(errorNode, errorMessage, symbolToString(prop), typeToString(propertyType), typeToString(indexType));
                 }
             }
-            return undefined;
         }
-        function isLiteralConstDeclaration(node) {
-            if (ts.isDeclarationReadonly(node) || ts.isVariableDeclaration(node) && ts.isVarConst(node)) {
-                return isFreshLiteralType(getTypeOfSymbol(getSymbolOfNode(node)));
+        function checkTypeNameIsReserved(name, message) {
+            // TS 1.0 spec (April 2014): 3.6.1
+            // The predefined type keywords are reserved and cannot be used as names of user defined types.
+            switch (name.escapedText) {
+                case "any":
+                case "unknown":
+                case "number":
+                case "bigint":
+                case "boolean":
+                case "string":
+                case "symbol":
+                case "void":
+                case "object":
+                    error(name, message, name.escapedText);
             }
-            return false;
-        }
-        function literalTypeToNode(type, enclosing, tracker) {
-            var enumResult = type.flags & 1024 /* EnumLiteral */ ? nodeBuilder.symbolToExpression(type.symbol, 111551 /* Value */, enclosing, /*flags*/ undefined, tracker)
-                : type === trueType ? ts.createTrue() : type === falseType && ts.createFalse();
-            return enumResult || ts.createLiteral(type.value);
-        }
-        function createLiteralConstValue(node, tracker) {
-            var type = getTypeOfSymbol(getSymbolOfNode(node));
-            return literalTypeToNode(type, node, tracker);
-        }
-        function getJsxFactoryEntity(location) {
-            return location ? (getJsxNamespace(location), (ts.getSourceFileOfNode(location).localJsxFactory || _jsxFactoryEntity)) : _jsxFactoryEntity;
         }
-        function createResolver() {
-            // this variable and functions that use it are deliberately moved here from the outer scope
-            // to avoid scope pollution
-            var resolvedTypeReferenceDirectives = host.getResolvedTypeReferenceDirectives();
-            var fileToDirective;
-            if (resolvedTypeReferenceDirectives) {
-                // populate reverse mapping: file path -> type reference directive that was resolved to this file
-                fileToDirective = ts.createMap();
-                resolvedTypeReferenceDirectives.forEach(function (resolvedDirective, key) {
-                    if (!resolvedDirective || !resolvedDirective.resolvedFileName) {
-                        return;
-                    }
-                    var file = host.getSourceFile(resolvedDirective.resolvedFileName);
-                    if (file) {
-                        // Add the transitive closure of path references loaded by this file (as long as they are not)
-                        // part of an existing type reference.
-                        addReferencedFilesToTypeDirective(file, key);
-                    }
-                });
+        /**
+         * The name cannot be used as 'Object' of user defined types with special target.
+         */
+        function checkClassNameCollisionWithObject(name) {
+            if (languageVersion === 1 /* ES5 */ && name.escapedText === "Object"
+                && moduleKind < ts.ModuleKind.ES2015) {
+                error(name, ts.Diagnostics.Class_name_cannot_be_Object_when_targeting_ES5_with_module_0, ts.ModuleKind[moduleKind]); // https://github.com/Microsoft/TypeScript/issues/17494
             }
-            return {
-                getReferencedExportContainer: getReferencedExportContainer,
-                getReferencedImportDeclaration: getReferencedImportDeclaration,
-                getReferencedDeclarationWithCollidingName: getReferencedDeclarationWithCollidingName,
-                isDeclarationWithCollidingName: isDeclarationWithCollidingName,
-                isValueAliasDeclaration: function (node) {
-                    node = ts.getParseTreeNode(node);
-                    // Synthesized nodes are always treated like values.
-                    return node ? isValueAliasDeclaration(node) : true;
-                },
-                hasGlobalName: hasGlobalName,
-                isReferencedAliasDeclaration: function (node, checkChildren) {
-                    node = ts.getParseTreeNode(node);
-                    // Synthesized nodes are always treated as referenced.
-                    return node ? isReferencedAliasDeclaration(node, checkChildren) : true;
-                },
-                getNodeCheckFlags: function (node) {
-                    node = ts.getParseTreeNode(node);
-                    return node ? getNodeCheckFlags(node) : 0;
-                },
-                isTopLevelValueImportEqualsWithEntityName: isTopLevelValueImportEqualsWithEntityName,
-                isDeclarationVisible: isDeclarationVisible,
-                isImplementationOfOverload: isImplementationOfOverload,
-                isRequiredInitializedParameter: isRequiredInitializedParameter,
-                isOptionalUninitializedParameterProperty: isOptionalUninitializedParameterProperty,
-                isExpandoFunctionDeclaration: isExpandoFunctionDeclaration,
-                getPropertiesOfContainerFunction: getPropertiesOfContainerFunction,
-                createTypeOfDeclaration: createTypeOfDeclaration,
-                createReturnTypeOfSignatureDeclaration: createReturnTypeOfSignatureDeclaration,
-                createTypeOfExpression: createTypeOfExpression,
-                createLiteralConstValue: createLiteralConstValue,
-                isSymbolAccessible: isSymbolAccessible,
-                isEntityNameVisible: isEntityNameVisible,
-                getConstantValue: function (nodeIn) {
-                    var node = ts.getParseTreeNode(nodeIn, canHaveConstantValue);
-                    return node ? getConstantValue(node) : undefined;
-                },
-                collectLinkedAliases: collectLinkedAliases,
-                getReferencedValueDeclaration: getReferencedValueDeclaration,
-                getTypeReferenceSerializationKind: getTypeReferenceSerializationKind,
-                isOptionalParameter: isOptionalParameter,
-                moduleExportsSomeValue: moduleExportsSomeValue,
-                isArgumentsLocalBinding: isArgumentsLocalBinding,
-                getExternalModuleFileFromDeclaration: getExternalModuleFileFromDeclaration,
-                getTypeReferenceDirectivesForEntityName: getTypeReferenceDirectivesForEntityName,
-                getTypeReferenceDirectivesForSymbol: getTypeReferenceDirectivesForSymbol,
-                isLiteralConstDeclaration: isLiteralConstDeclaration,
-                isLateBound: function (nodeIn) {
-                    var node = ts.getParseTreeNode(nodeIn, ts.isDeclaration);
-                    var symbol = node && getSymbolOfNode(node);
-                    return !!(symbol && ts.getCheckFlags(symbol) & 4096 /* Late */);
-                },
-                getJsxFactoryEntity: getJsxFactoryEntity,
-                getAllAccessorDeclarations: function (accessor) {
-                    accessor = ts.getParseTreeNode(accessor, ts.isGetOrSetAccessorDeclaration); // TODO: GH#18217
-                    var otherKind = accessor.kind === 164 /* SetAccessor */ ? 163 /* GetAccessor */ : 164 /* SetAccessor */;
-                    var otherAccessor = ts.getDeclarationOfKind(getSymbolOfNode(accessor), otherKind);
-                    var firstAccessor = otherAccessor && (otherAccessor.pos < accessor.pos) ? otherAccessor : accessor;
-                    var secondAccessor = otherAccessor && (otherAccessor.pos < accessor.pos) ? accessor : otherAccessor;
-                    var setAccessor = accessor.kind === 164 /* SetAccessor */ ? accessor : otherAccessor;
-                    var getAccessor = accessor.kind === 163 /* GetAccessor */ ? accessor : otherAccessor;
-                    return {
-                        firstAccessor: firstAccessor,
-                        secondAccessor: secondAccessor,
-                        setAccessor: setAccessor,
-                        getAccessor: getAccessor
-                    };
-                },
-                getSymbolOfExternalModuleSpecifier: function (moduleName) { return resolveExternalModuleNameWorker(moduleName, moduleName, /*moduleNotFoundError*/ undefined); },
-                isBindingCapturedByNode: function (node, decl) {
-                    var parseNode = ts.getParseTreeNode(node);
-                    var parseDecl = ts.getParseTreeNode(decl);
-                    return !!parseNode && !!parseDecl && (ts.isVariableDeclaration(parseDecl) || ts.isBindingElement(parseDecl)) && isBindingCapturedByNode(parseNode, parseDecl);
-                },
-                getDeclarationStatementsForSourceFile: function (node, flags, tracker, bundled) {
-                    var n = ts.getParseTreeNode(node);
-                    ts.Debug.assert(n && n.kind === 290 /* SourceFile */, "Non-sourcefile node passed into getDeclarationsForSourceFile");
-                    var sym = getSymbolOfNode(node);
-                    if (!sym) {
-                        return !node.locals ? [] : nodeBuilder.symbolTableToDeclarationStatements(node.locals, node, flags, tracker, bundled);
+        }
+        /**
+         * Check each type parameter and check that type parameters have no duplicate type parameter declarations
+         */
+        function checkTypeParameters(typeParameterDeclarations) {
+            if (typeParameterDeclarations) {
+                var seenDefault = false;
+                for (var i = 0; i < typeParameterDeclarations.length; i++) {
+                    var node = typeParameterDeclarations[i];
+                    checkTypeParameter(node);
+                    if (produceDiagnostics) {
+                        if (node.default) {
+                            seenDefault = true;
+                            checkTypeParametersNotReferenced(node.default, typeParameterDeclarations, i);
+                        }
+                        else if (seenDefault) {
+                            error(node, ts.Diagnostics.Required_type_parameters_may_not_follow_optional_type_parameters);
+                        }
+                        for (var j = 0; j < i; j++) {
+                            if (typeParameterDeclarations[j].symbol === node.symbol) {
+                                error(node.name, ts.Diagnostics.Duplicate_identifier_0, ts.declarationNameToString(node.name));
+                            }
+                        }
                     }
-                    return !sym.exports ? [] : nodeBuilder.symbolTableToDeclarationStatements(sym.exports, node, flags, tracker, bundled);
-                },
-                isImportRequiredByAugmentation: isImportRequiredByAugmentation,
-            };
-            function isImportRequiredByAugmentation(node) {
-                var file = ts.getSourceFileOfNode(node);
-                if (!file.symbol)
-                    return false;
-                var importTarget = getExternalModuleFileFromDeclaration(node);
-                if (!importTarget)
-                    return false;
-                if (importTarget === file)
-                    return false;
-                var exports = getExportsOfModule(file.symbol);
-                for (var _i = 0, _a = ts.arrayFrom(exports.values()); _i < _a.length; _i++) {
-                    var s = _a[_i];
-                    if (s.mergeId) {
-                        var merged = getMergedSymbol(s);
-                        for (var _b = 0, _c = merged.declarations; _b < _c.length; _b++) {
-                            var d = _c[_b];
-                            var declFile = ts.getSourceFileOfNode(d);
-                            if (declFile === importTarget) {
-                                return true;
+                }
+            }
+        }
+        /** Check that type parameter defaults only reference previously declared type parameters */
+        function checkTypeParametersNotReferenced(root, typeParameters, index) {
+            visit(root);
+            function visit(node) {
+                if (node.kind === 173 /* TypeReference */) {
+                    var type = getTypeFromTypeReference(node);
+                    if (type.flags & 262144 /* TypeParameter */) {
+                        for (var i = index; i < typeParameters.length; i++) {
+                            if (type.symbol === getSymbolOfNode(typeParameters[i])) {
+                                error(node, ts.Diagnostics.Type_parameter_defaults_can_only_reference_previously_declared_type_parameters);
                             }
                         }
                     }
                 }
-                return false;
+                ts.forEachChild(node, visit);
             }
-            function isInHeritageClause(node) {
-                return node.parent && node.parent.kind === 216 /* ExpressionWithTypeArguments */ && node.parent.parent && node.parent.parent.kind === 279 /* HeritageClause */;
+        }
+        /** Check that type parameter lists are identical across multiple declarations */
+        function checkTypeParameterListsIdentical(symbol) {
+            if (symbol.declarations.length === 1) {
+                return;
             }
-            // defined here to avoid outer scope pollution
-            function getTypeReferenceDirectivesForEntityName(node) {
-                // program does not have any files with type reference directives - bail out
-                if (!fileToDirective) {
-                    return undefined;
+            var links = getSymbolLinks(symbol);
+            if (!links.typeParametersChecked) {
+                links.typeParametersChecked = true;
+                var declarations = getClassOrInterfaceDeclarationsOfSymbol(symbol);
+                if (declarations.length <= 1) {
+                    return;
                 }
-                // property access can only be used as values, or types when within an expression with type arguments inside a heritage clause
-                // qualified names can only be used as types\namespaces
-                // identifiers are treated as values only if they appear in type queries
-                var meaning = 788968 /* Type */ | 1920 /* Namespace */;
-                if ((node.kind === 75 /* Identifier */ && isInTypeQuery(node)) || (node.kind === 194 /* PropertyAccessExpression */ && !isInHeritageClause(node))) {
-                    meaning = 111551 /* Value */ | 1048576 /* ExportValue */;
+                var type = getDeclaredTypeOfSymbol(symbol);
+                if (!areTypeParametersIdentical(declarations, type.localTypeParameters)) {
+                    // Report an error on every conflicting declaration.
+                    var name = symbolToString(symbol);
+                    for (var _i = 0, declarations_6 = declarations; _i < declarations_6.length; _i++) {
+                        var declaration = declarations_6[_i];
+                        error(declaration.name, ts.Diagnostics.All_declarations_of_0_must_have_identical_type_parameters, name);
+                    }
                 }
-                var symbol = resolveEntityName(node, meaning, /*ignoreErrors*/ true);
-                return symbol && symbol !== unknownSymbol ? getTypeReferenceDirectivesForSymbol(symbol, meaning) : undefined;
             }
-            // defined here to avoid outer scope pollution
-            function getTypeReferenceDirectivesForSymbol(symbol, meaning) {
-                // program does not have any files with type reference directives - bail out
-                if (!fileToDirective) {
-                    return undefined;
-                }
-                if (!isSymbolFromTypeDeclarationFile(symbol)) {
-                    return undefined;
+        }
+        function areTypeParametersIdentical(declarations, targetParameters) {
+            var maxTypeArgumentCount = ts.length(targetParameters);
+            var minTypeArgumentCount = getMinTypeArgumentCount(targetParameters);
+            for (var _i = 0, declarations_7 = declarations; _i < declarations_7.length; _i++) {
+                var declaration = declarations_7[_i];
+                // If this declaration has too few or too many type parameters, we report an error
+                var sourceParameters = ts.getEffectiveTypeParameterDeclarations(declaration);
+                var numTypeParameters = sourceParameters.length;
+                if (numTypeParameters < minTypeArgumentCount || numTypeParameters > maxTypeArgumentCount) {
+                    return false;
                 }
-                // check what declarations in the symbol can contribute to the target meaning
-                var typeReferenceDirectives;
-                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                    var decl = _a[_i];
-                    // check meaning of the local symbol to see if declaration needs to be analyzed further
-                    if (decl.symbol && decl.symbol.flags & meaning) {
-                        var file = ts.getSourceFileOfNode(decl);
-                        var typeReferenceDirective = fileToDirective.get(file.path);
-                        if (typeReferenceDirective) {
-                            (typeReferenceDirectives || (typeReferenceDirectives = [])).push(typeReferenceDirective);
-                        }
-                        else {
-                            // found at least one entry that does not originate from type reference directive
-                            return undefined;
-                        }
+                for (var i = 0; i < numTypeParameters; i++) {
+                    var source = sourceParameters[i];
+                    var target = targetParameters[i];
+                    // If the type parameter node does not have the same as the resolved type
+                    // parameter at this position, we report an error.
+                    if (source.name.escapedText !== target.symbol.escapedName) {
+                        return false;
+                    }
+                    // If the type parameter node does not have an identical constraint as the resolved
+                    // type parameter at this position, we report an error.
+                    var constraint = ts.getEffectiveConstraintOfTypeParameter(source);
+                    var sourceConstraint = constraint && getTypeFromTypeNode(constraint);
+                    var targetConstraint = getConstraintOfTypeParameter(target);
+                    // relax check if later interface augmentation has no constraint, it's more broad and is OK to merge with
+                    // a more constrained interface (this could be generalized to a full hierarchy check, but that's maybe overkill)
+                    if (sourceConstraint && targetConstraint && !isTypeIdenticalTo(sourceConstraint, targetConstraint)) {
+                        return false;
+                    }
+                    // If the type parameter node has a default and it is not identical to the default
+                    // for the type parameter at this position, we report an error.
+                    var sourceDefault = source.default && getTypeFromTypeNode(source.default);
+                    var targetDefault = getDefaultFromTypeParameter(target);
+                    if (sourceDefault && targetDefault && !isTypeIdenticalTo(sourceDefault, targetDefault)) {
+                        return false;
                     }
                 }
-                return typeReferenceDirectives;
             }
-            function isSymbolFromTypeDeclarationFile(symbol) {
-                // bail out if symbol does not have associated declarations (i.e. this is transient symbol created for property in binding pattern)
-                if (!symbol.declarations) {
-                    return false;
+            return true;
+        }
+        function checkClassExpression(node) {
+            checkClassLikeDeclaration(node);
+            checkNodeDeferred(node);
+            return getTypeOfSymbol(getSymbolOfNode(node));
+        }
+        function checkClassExpressionDeferred(node) {
+            ts.forEach(node.members, checkSourceElement);
+            registerForUnusedIdentifiersCheck(node);
+        }
+        function checkClassDeclaration(node) {
+            if (!node.name && !ts.hasSyntacticModifier(node, 512 /* Default */)) {
+                grammarErrorOnFirstToken(node, ts.Diagnostics.A_class_declaration_without_the_default_modifier_must_have_a_name);
+            }
+            checkClassLikeDeclaration(node);
+            ts.forEach(node.members, checkSourceElement);
+            registerForUnusedIdentifiersCheck(node);
+        }
+        function checkClassLikeDeclaration(node) {
+            checkGrammarClassLikeDeclaration(node);
+            checkDecorators(node);
+            if (node.name) {
+                checkTypeNameIsReserved(node.name, ts.Diagnostics.Class_name_cannot_be_0);
+                checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
+                checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+                if (!(node.flags & 8388608 /* Ambient */)) {
+                    checkClassNameCollisionWithObject(node.name);
                 }
-                // walk the parent chain for symbols to make sure that top level parent symbol is in the global scope
-                // external modules cannot define or contribute to type declaration files
-                var current = symbol;
-                while (true) {
-                    var parent = getParentOfSymbol(current);
-                    if (parent) {
-                        current = parent;
+            }
+            checkTypeParameters(ts.getEffectiveTypeParameterDeclarations(node));
+            checkExportsOnMergedDeclarations(node);
+            var symbol = getSymbolOfNode(node);
+            var type = getDeclaredTypeOfSymbol(symbol);
+            var typeWithThis = getTypeWithThisArgument(type);
+            var staticType = getTypeOfSymbol(symbol);
+            checkTypeParameterListsIdentical(symbol);
+            checkFunctionOrConstructorSymbol(symbol);
+            checkClassForDuplicateDeclarations(node);
+            // Only check for reserved static identifiers on non-ambient context.
+            if (!(node.flags & 8388608 /* Ambient */)) {
+                checkClassForStaticPropertyNameConflicts(node);
+            }
+            var baseTypeNode = ts.getEffectiveBaseTypeNode(node);
+            if (baseTypeNode) {
+                ts.forEach(baseTypeNode.typeArguments, checkSourceElement);
+                if (languageVersion < 2 /* ES2015 */) {
+                    checkExternalEmitHelpers(baseTypeNode.parent, 1 /* Extends */);
+                }
+                // check both @extends and extends if both are specified.
+                var extendsNode = ts.getClassExtendsHeritageElement(node);
+                if (extendsNode && extendsNode !== baseTypeNode) {
+                    checkExpression(extendsNode.expression);
+                }
+                var baseTypes = getBaseTypes(type);
+                if (baseTypes.length && produceDiagnostics) {
+                    var baseType_1 = baseTypes[0];
+                    var baseConstructorType = getBaseConstructorTypeOfClass(type);
+                    var staticBaseType = getApparentType(baseConstructorType);
+                    checkBaseTypeAccessibility(staticBaseType, baseTypeNode);
+                    checkSourceElement(baseTypeNode.expression);
+                    if (ts.some(baseTypeNode.typeArguments)) {
+                        ts.forEach(baseTypeNode.typeArguments, checkSourceElement);
+                        for (var _i = 0, _a = getConstructorsForTypeArguments(staticBaseType, baseTypeNode.typeArguments, baseTypeNode); _i < _a.length; _i++) {
+                            var constructor = _a[_i];
+                            if (!checkTypeArgumentConstraints(baseTypeNode, constructor.typeParameters)) {
+                                break;
+                            }
+                        }
+                    }
+                    var baseWithThis = getTypeWithThisArgument(baseType_1, type.thisType);
+                    if (!checkTypeAssignableTo(typeWithThis, baseWithThis, /*errorNode*/ undefined)) {
+                        issueMemberSpecificError(node, typeWithThis, baseWithThis, ts.Diagnostics.Class_0_incorrectly_extends_base_class_1);
                     }
                     else {
-                        break;
+                        // Report static side error only when instance type is assignable
+                        checkTypeAssignableTo(staticType, getTypeWithoutSignatures(staticBaseType), node.name || node, ts.Diagnostics.Class_static_side_0_incorrectly_extends_base_class_static_side_1);
                     }
-                }
-                if (current.valueDeclaration && current.valueDeclaration.kind === 290 /* SourceFile */ && current.flags & 512 /* ValueModule */) {
-                    return false;
-                }
-                // check that at least one declaration of top level symbol originates from type declaration file
-                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
-                    var decl = _a[_i];
-                    var file = ts.getSourceFileOfNode(decl);
-                    if (fileToDirective.has(file.path)) {
-                        return true;
+                    if (baseConstructorType.flags & 8650752 /* TypeVariable */ && !isMixinConstructorType(staticType)) {
+                        error(node.name || node, ts.Diagnostics.A_mixin_class_must_have_a_constructor_with_a_single_rest_parameter_of_type_any);
+                    }
+                    if (!(staticBaseType.symbol && staticBaseType.symbol.flags & 32 /* Class */) && !(baseConstructorType.flags & 8650752 /* TypeVariable */)) {
+                        // When the static base type is a "class-like" constructor function (but not actually a class), we verify
+                        // that all instantiated base constructor signatures return the same type.
+                        var constructors = getInstantiatedConstructorsForTypeArguments(staticBaseType, baseTypeNode.typeArguments, baseTypeNode);
+                        if (ts.forEach(constructors, function (sig) { return !isJSConstructor(sig.declaration) && !isTypeIdenticalTo(getReturnTypeOfSignature(sig), baseType_1); })) {
+                            error(baseTypeNode.expression, ts.Diagnostics.Base_constructors_must_all_have_the_same_return_type);
+                        }
                     }
+                    checkKindsOfPropertyMemberOverrides(type, baseType_1);
                 }
-                return false;
             }
-            function addReferencedFilesToTypeDirective(file, key) {
-                if (fileToDirective.has(file.path))
-                    return;
-                fileToDirective.set(file.path, key);
-                for (var _i = 0, _a = file.referencedFiles; _i < _a.length; _i++) {
-                    var fileName = _a[_i].fileName;
-                    var resolvedFile = ts.resolveTripleslashReference(fileName, file.originalFileName);
-                    var referencedFile = host.getSourceFile(resolvedFile);
-                    if (referencedFile) {
-                        addReferencedFilesToTypeDirective(referencedFile, key);
+            var implementedTypeNodes = ts.getEffectiveImplementsTypeNodes(node);
+            if (implementedTypeNodes) {
+                for (var _b = 0, implementedTypeNodes_1 = implementedTypeNodes; _b < implementedTypeNodes_1.length; _b++) {
+                    var typeRefNode = implementedTypeNodes_1[_b];
+                    if (!ts.isEntityNameExpression(typeRefNode.expression)) {
+                        error(typeRefNode.expression, ts.Diagnostics.A_class_can_only_implement_an_identifier_Slashqualified_name_with_optional_type_arguments);
+                    }
+                    checkTypeReferenceNode(typeRefNode);
+                    if (produceDiagnostics) {
+                        var t = getReducedType(getTypeFromTypeNode(typeRefNode));
+                        if (t !== errorType) {
+                            if (isValidBaseType(t)) {
+                                var genericDiag = t.symbol && t.symbol.flags & 32 /* Class */ ?
+                                    ts.Diagnostics.Class_0_incorrectly_implements_class_1_Did_you_mean_to_extend_1_and_inherit_its_members_as_a_subclass :
+                                    ts.Diagnostics.Class_0_incorrectly_implements_interface_1;
+                                var baseWithThis = getTypeWithThisArgument(t, type.thisType);
+                                if (!checkTypeAssignableTo(typeWithThis, baseWithThis, /*errorNode*/ undefined)) {
+                                    issueMemberSpecificError(node, typeWithThis, baseWithThis, genericDiag);
+                                }
+                            }
+                            else {
+                                error(typeRefNode, ts.Diagnostics.A_class_can_only_implement_an_object_type_or_intersection_of_object_types_with_statically_known_members);
+                            }
+                        }
                     }
                 }
             }
-        }
-        function getExternalModuleFileFromDeclaration(declaration) {
-            var specifier = declaration.kind === 249 /* ModuleDeclaration */ ? ts.tryCast(declaration.name, ts.isStringLiteral) : ts.getExternalModuleName(declaration);
-            var moduleSymbol = resolveExternalModuleNameWorker(specifier, specifier, /*moduleNotFoundError*/ undefined); // TODO: GH#18217
-            if (!moduleSymbol) {
-                return undefined;
+            if (produceDiagnostics) {
+                checkIndexConstraints(type);
+                checkTypeForDuplicateIndexSignatures(node);
+                checkPropertyInitialization(node);
             }
-            return ts.getDeclarationOfKind(moduleSymbol, 290 /* SourceFile */);
         }
-        function initializeTypeChecker() {
-            // Bind all source files and propagate errors
-            for (var _i = 0, _a = host.getSourceFiles(); _i < _a.length; _i++) {
-                var file = _a[_i];
-                ts.bindSourceFile(file, compilerOptions);
-            }
-            amalgamatedDuplicates = ts.createMap();
-            // Initialize global symbol table
-            var augmentations;
-            for (var _b = 0, _c = host.getSourceFiles(); _b < _c.length; _b++) {
-                var file = _c[_b];
-                if (file.redirectInfo) {
-                    continue;
+        function issueMemberSpecificError(node, typeWithThis, baseWithThis, broadDiag) {
+            // iterate over all implemented properties and issue errors on each one which isn't compatible, rather than the class as a whole, if possible
+            var issuedMemberError = false;
+            var _loop_23 = function (member) {
+                if (ts.hasStaticModifier(member)) {
+                    return "continue";
                 }
-                if (!ts.isExternalOrCommonJsModule(file)) {
-                    // It is an error for a non-external-module (i.e. script) to declare its own `globalThis`.
-                    // We can't use `builtinGlobals` for this due to synthetic expando-namespace generation in JS files.
-                    var fileGlobalThisSymbol = file.locals.get("globalThis");
-                    if (fileGlobalThisSymbol) {
-                        for (var _d = 0, _e = fileGlobalThisSymbol.declarations; _d < _e.length; _d++) {
-                            var declaration = _e[_d];
-                            diagnostics.add(ts.createDiagnosticForNode(declaration, ts.Diagnostics.Declaration_name_conflicts_with_built_in_global_identifier_0, "globalThis"));
+                var declaredProp = member.name && getSymbolAtLocation(member.name) || getSymbolAtLocation(member);
+                if (declaredProp) {
+                    var prop = getPropertyOfType(typeWithThis, declaredProp.escapedName);
+                    var baseProp = getPropertyOfType(baseWithThis, declaredProp.escapedName);
+                    if (prop && baseProp) {
+                        var rootChain = function () { return ts.chainDiagnosticMessages(
+                        /*details*/ undefined, ts.Diagnostics.Property_0_in_type_1_is_not_assignable_to_the_same_property_in_base_type_2, symbolToString(declaredProp), typeToString(typeWithThis), typeToString(baseWithThis)); };
+                        if (!checkTypeAssignableTo(getTypeOfSymbol(prop), getTypeOfSymbol(baseProp), member.name || member, /*message*/ undefined, rootChain)) {
+                            issuedMemberError = true;
                         }
                     }
-                    mergeSymbolTable(globals, file.locals);
                 }
-                if (file.jsGlobalAugmentations) {
-                    mergeSymbolTable(globals, file.jsGlobalAugmentations);
+            };
+            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                var member = _a[_i];
+                _loop_23(member);
+            }
+            if (!issuedMemberError) {
+                // check again with diagnostics to generate a less-specific error
+                checkTypeAssignableTo(typeWithThis, baseWithThis, node.name || node, broadDiag);
+            }
+        }
+        function checkBaseTypeAccessibility(type, node) {
+            var signatures = getSignaturesOfType(type, 1 /* Construct */);
+            if (signatures.length) {
+                var declaration = signatures[0].declaration;
+                if (declaration && ts.hasEffectiveModifier(declaration, 8 /* Private */)) {
+                    var typeClassDeclaration = ts.getClassLikeDeclarationOfSymbol(type.symbol);
+                    if (!isNodeWithinClass(node, typeClassDeclaration)) {
+                        error(node, ts.Diagnostics.Cannot_extend_a_class_0_Class_constructor_is_marked_as_private, getFullyQualifiedName(type.symbol));
+                    }
                 }
-                if (file.patternAmbientModules && file.patternAmbientModules.length) {
-                    patternAmbientModules = ts.concatenate(patternAmbientModules, file.patternAmbientModules);
+            }
+        }
+        function getTargetSymbol(s) {
+            // if symbol is instantiated its flags are not copied from the 'target'
+            // so we'll need to get back original 'target' symbol to work with correct set of flags
+            return ts.getCheckFlags(s) & 1 /* Instantiated */ ? s.target : s;
+        }
+        function getClassOrInterfaceDeclarationsOfSymbol(symbol) {
+            return ts.filter(symbol.declarations, function (d) {
+                return d.kind === 252 /* ClassDeclaration */ || d.kind === 253 /* InterfaceDeclaration */;
+            });
+        }
+        function checkKindsOfPropertyMemberOverrides(type, baseType) {
+            // TypeScript 1.0 spec (April 2014): 8.2.3
+            // A derived class inherits all members from its base class it doesn't override.
+            // Inheritance means that a derived class implicitly contains all non - overridden members of the base class.
+            // Both public and private property members are inherited, but only public property members can be overridden.
+            // A property member in a derived class is said to override a property member in a base class
+            // when the derived class property member has the same name and kind(instance or static)
+            // as the base class property member.
+            // The type of an overriding property member must be assignable(section 3.8.4)
+            // to the type of the overridden property member, or otherwise a compile - time error occurs.
+            // Base class instance member functions can be overridden by derived class instance member functions,
+            // but not by other kinds of members.
+            // Base class instance member variables and accessors can be overridden by
+            // derived class instance member variables and accessors, but not by other kinds of members.
+            // NOTE: assignability is checked in checkClassDeclaration
+            var baseProperties = getPropertiesOfType(baseType);
+            basePropertyCheck: for (var _i = 0, baseProperties_1 = baseProperties; _i < baseProperties_1.length; _i++) {
+                var baseProperty = baseProperties_1[_i];
+                var base = getTargetSymbol(baseProperty);
+                if (base.flags & 4194304 /* Prototype */) {
+                    continue;
                 }
-                if (file.moduleAugmentations.length) {
-                    (augmentations || (augmentations = [])).push(file.moduleAugmentations);
+                var baseSymbol = getPropertyOfObjectType(type, base.escapedName);
+                if (!baseSymbol) {
+                    continue;
                 }
-                if (file.symbol && file.symbol.globalExports) {
-                    // Merge in UMD exports with first-in-wins semantics (see #9771)
-                    var source = file.symbol.globalExports;
-                    source.forEach(function (sourceSymbol, id) {
-                        if (!globals.has(id)) {
-                            globals.set(id, sourceSymbol);
+                var derived = getTargetSymbol(baseSymbol);
+                var baseDeclarationFlags = ts.getDeclarationModifierFlagsFromSymbol(base);
+                ts.Debug.assert(!!derived, "derived should point to something, even if it is the base class' declaration.");
+                // In order to resolve whether the inherited method was overridden in the base class or not,
+                // we compare the Symbols obtained. Since getTargetSymbol returns the symbol on the *uninstantiated*
+                // type declaration, derived and base resolve to the same symbol even in the case of generic classes.
+                if (derived === base) {
+                    // derived class inherits base without override/redeclaration
+                    var derivedClassDecl = ts.getClassLikeDeclarationOfSymbol(type.symbol);
+                    // It is an error to inherit an abstract member without implementing it or being declared abstract.
+                    // If there is no declaration for the derived class (as in the case of class expressions),
+                    // then the class cannot be declared abstract.
+                    if (baseDeclarationFlags & 128 /* Abstract */ && (!derivedClassDecl || !ts.hasSyntacticModifier(derivedClassDecl, 128 /* Abstract */))) {
+                        // Searches other base types for a declaration that would satisfy the inherited abstract member.
+                        // (The class may have more than one base type via declaration merging with an interface with the
+                        // same name.)
+                        for (var _a = 0, _b = getBaseTypes(type); _a < _b.length; _a++) {
+                            var otherBaseType = _b[_a];
+                            if (otherBaseType === baseType)
+                                continue;
+                            var baseSymbol_1 = getPropertyOfObjectType(otherBaseType, base.escapedName);
+                            var derivedElsewhere = baseSymbol_1 && getTargetSymbol(baseSymbol_1);
+                            if (derivedElsewhere && derivedElsewhere !== base) {
+                                continue basePropertyCheck;
+                            }
                         }
-                    });
+                        if (derivedClassDecl.kind === 221 /* ClassExpression */) {
+                            error(derivedClassDecl, ts.Diagnostics.Non_abstract_class_expression_does_not_implement_inherited_abstract_member_0_from_class_1, symbolToString(baseProperty), typeToString(baseType));
+                        }
+                        else {
+                            error(derivedClassDecl, ts.Diagnostics.Non_abstract_class_0_does_not_implement_inherited_abstract_member_1_from_class_2, typeToString(type), symbolToString(baseProperty), typeToString(baseType));
+                        }
+                    }
                 }
-            }
-            // We do global augmentations separately from module augmentations (and before creating global types) because they
-            //  1. Affect global types. We won't have the correct global types until global augmentations are merged. Also,
-            //  2. Module augmentation instantiation requires creating the type of a module, which, in turn, can require
-            //       checking for an export or property on the module (if export=) which, in turn, can fall back to the
-            //       apparent type of the module - either globalObjectType or globalFunctionType - which wouldn't exist if we
-            //       did module augmentations prior to finalizing the global types.
-            if (augmentations) {
-                // merge _global_ module augmentations.
-                // this needs to be done after global symbol table is initialized to make sure that all ambient modules are indexed
-                for (var _f = 0, augmentations_1 = augmentations; _f < augmentations_1.length; _f++) {
-                    var list = augmentations_1[_f];
-                    for (var _g = 0, list_1 = list; _g < list_1.length; _g++) {
-                        var augmentation = list_1[_g];
-                        if (!ts.isGlobalScopeAugmentation(augmentation.parent))
+                else {
+                    // derived overrides base.
+                    var derivedDeclarationFlags = ts.getDeclarationModifierFlagsFromSymbol(derived);
+                    if (baseDeclarationFlags & 8 /* Private */ || derivedDeclarationFlags & 8 /* Private */) {
+                        // either base or derived property is private - not override, skip it
+                        continue;
+                    }
+                    var errorMessage = void 0;
+                    var basePropertyFlags = base.flags & 98308 /* PropertyOrAccessor */;
+                    var derivedPropertyFlags = derived.flags & 98308 /* PropertyOrAccessor */;
+                    if (basePropertyFlags && derivedPropertyFlags) {
+                        // property/accessor is overridden with property/accessor
+                        if (baseDeclarationFlags & 128 /* Abstract */ && !(base.valueDeclaration && ts.isPropertyDeclaration(base.valueDeclaration) && base.valueDeclaration.initializer)
+                            || base.valueDeclaration && base.valueDeclaration.parent.kind === 253 /* InterfaceDeclaration */
+                            || derived.valueDeclaration && ts.isBinaryExpression(derived.valueDeclaration)) {
+                            // when the base property is abstract or from an interface, base/derived flags don't need to match
+                            // same when the derived property is from an assignment
                             continue;
-                        mergeModuleAugmentation(augmentation);
+                        }
+                        var overriddenInstanceProperty = basePropertyFlags !== 4 /* Property */ && derivedPropertyFlags === 4 /* Property */;
+                        var overriddenInstanceAccessor = basePropertyFlags === 4 /* Property */ && derivedPropertyFlags !== 4 /* Property */;
+                        if (overriddenInstanceProperty || overriddenInstanceAccessor) {
+                            var errorMessage_1 = overriddenInstanceProperty ?
+                                ts.Diagnostics._0_is_defined_as_an_accessor_in_class_1_but_is_overridden_here_in_2_as_an_instance_property :
+                                ts.Diagnostics._0_is_defined_as_a_property_in_class_1_but_is_overridden_here_in_2_as_an_accessor;
+                            error(ts.getNameOfDeclaration(derived.valueDeclaration) || derived.valueDeclaration, errorMessage_1, symbolToString(base), typeToString(baseType), typeToString(type));
+                        }
+                        else if (compilerOptions.useDefineForClassFields) {
+                            var uninitialized = ts.find(derived.declarations, function (d) { return d.kind === 163 /* PropertyDeclaration */ && !d.initializer; });
+                            if (uninitialized
+                                && !(derived.flags & 33554432 /* Transient */)
+                                && !(baseDeclarationFlags & 128 /* Abstract */)
+                                && !(derivedDeclarationFlags & 128 /* Abstract */)
+                                && !derived.declarations.some(function (d) { return !!(d.flags & 8388608 /* Ambient */); })) {
+                                var constructor = findConstructorDeclaration(ts.getClassLikeDeclarationOfSymbol(type.symbol));
+                                var propName = uninitialized.name;
+                                if (uninitialized.exclamationToken
+                                    || !constructor
+                                    || !ts.isIdentifier(propName)
+                                    || !strictNullChecks
+                                    || !isPropertyInitializedInConstructor(propName, type, constructor)) {
+                                    var errorMessage_2 = ts.Diagnostics.Property_0_will_overwrite_the_base_property_in_1_If_this_is_intentional_add_an_initializer_Otherwise_add_a_declare_modifier_or_remove_the_redundant_declaration;
+                                    error(ts.getNameOfDeclaration(derived.valueDeclaration) || derived.valueDeclaration, errorMessage_2, symbolToString(base), typeToString(baseType));
+                                }
+                            }
+                        }
+                        // correct case
+                        continue;
+                    }
+                    else if (isPrototypeProperty(base)) {
+                        if (isPrototypeProperty(derived) || derived.flags & 4 /* Property */) {
+                            // method is overridden with method or property -- correct case
+                            continue;
+                        }
+                        else {
+                            ts.Debug.assert(!!(derived.flags & 98304 /* Accessor */));
+                            errorMessage = ts.Diagnostics.Class_0_defines_instance_member_function_1_but_extended_class_2_defines_it_as_instance_member_accessor;
+                        }
+                    }
+                    else if (base.flags & 98304 /* Accessor */) {
+                        errorMessage = ts.Diagnostics.Class_0_defines_instance_member_accessor_1_but_extended_class_2_defines_it_as_instance_member_function;
                     }
+                    else {
+                        errorMessage = ts.Diagnostics.Class_0_defines_instance_member_property_1_but_extended_class_2_defines_it_as_instance_member_function;
+                    }
+                    error(ts.getNameOfDeclaration(derived.valueDeclaration) || derived.valueDeclaration, errorMessage, typeToString(baseType), symbolToString(base), typeToString(type));
                 }
             }
-            // Setup global builtins
-            addToSymbolTable(globals, builtinGlobals, ts.Diagnostics.Declaration_name_conflicts_with_built_in_global_identifier_0);
-            getSymbolLinks(undefinedSymbol).type = undefinedWideningType;
-            getSymbolLinks(argumentsSymbol).type = getGlobalType("IArguments", /*arity*/ 0, /*reportErrors*/ true);
-            getSymbolLinks(unknownSymbol).type = errorType;
-            getSymbolLinks(globalThisSymbol).type = createObjectType(16 /* Anonymous */, globalThisSymbol);
-            // Initialize special types
-            globalArrayType = getGlobalType("Array", /*arity*/ 1, /*reportErrors*/ true);
-            globalObjectType = getGlobalType("Object", /*arity*/ 0, /*reportErrors*/ true);
-            globalFunctionType = getGlobalType("Function", /*arity*/ 0, /*reportErrors*/ true);
-            globalCallableFunctionType = strictBindCallApply && getGlobalType("CallableFunction", /*arity*/ 0, /*reportErrors*/ true) || globalFunctionType;
-            globalNewableFunctionType = strictBindCallApply && getGlobalType("NewableFunction", /*arity*/ 0, /*reportErrors*/ true) || globalFunctionType;
-            globalStringType = getGlobalType("String", /*arity*/ 0, /*reportErrors*/ true);
-            globalNumberType = getGlobalType("Number", /*arity*/ 0, /*reportErrors*/ true);
-            globalBooleanType = getGlobalType("Boolean", /*arity*/ 0, /*reportErrors*/ true);
-            globalRegExpType = getGlobalType("RegExp", /*arity*/ 0, /*reportErrors*/ true);
-            anyArrayType = createArrayType(anyType);
-            autoArrayType = createArrayType(autoType);
-            if (autoArrayType === emptyObjectType) {
-                // autoArrayType is used as a marker, so even if global Array type is not defined, it needs to be a unique type
-                autoArrayType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+        }
+        function getNonInterhitedProperties(type, baseTypes, properties) {
+            if (!ts.length(baseTypes)) {
+                return properties;
             }
-            globalReadonlyArrayType = getGlobalTypeOrUndefined("ReadonlyArray", /*arity*/ 1) || globalArrayType;
-            anyReadonlyArrayType = globalReadonlyArrayType ? createTypeFromGenericGlobalType(globalReadonlyArrayType, [anyType]) : anyArrayType;
-            globalThisType = getGlobalTypeOrUndefined("ThisType", /*arity*/ 1);
-            if (augmentations) {
-                // merge _nonglobal_ module augmentations.
-                // this needs to be done after global symbol table is initialized to make sure that all ambient modules are indexed
-                for (var _h = 0, augmentations_2 = augmentations; _h < augmentations_2.length; _h++) {
-                    var list = augmentations_2[_h];
-                    for (var _j = 0, list_2 = list; _j < list_2.length; _j++) {
-                        var augmentation = list_2[_j];
-                        if (ts.isGlobalScopeAugmentation(augmentation.parent))
-                            continue;
-                        mergeModuleAugmentation(augmentation);
+            var seen = new ts.Map();
+            ts.forEach(properties, function (p) { seen.set(p.escapedName, p); });
+            for (var _i = 0, baseTypes_2 = baseTypes; _i < baseTypes_2.length; _i++) {
+                var base = baseTypes_2[_i];
+                var properties_5 = getPropertiesOfType(getTypeWithThisArgument(base, type.thisType));
+                for (var _a = 0, properties_4 = properties_5; _a < properties_4.length; _a++) {
+                    var prop = properties_4[_a];
+                    var existing = seen.get(prop.escapedName);
+                    if (existing && !isPropertyIdenticalTo(existing, prop)) {
+                        seen.delete(prop.escapedName);
                     }
                 }
             }
-            amalgamatedDuplicates.forEach(function (_a) {
-                var firstFile = _a.firstFile, secondFile = _a.secondFile, conflictingSymbols = _a.conflictingSymbols;
-                // If not many things conflict, issue individual errors
-                if (conflictingSymbols.size < 8) {
-                    conflictingSymbols.forEach(function (_a, symbolName) {
-                        var isBlockScoped = _a.isBlockScoped, firstFileLocations = _a.firstFileLocations, secondFileLocations = _a.secondFileLocations;
-                        var message = isBlockScoped ? ts.Diagnostics.Cannot_redeclare_block_scoped_variable_0 : ts.Diagnostics.Duplicate_identifier_0;
-                        for (var _i = 0, firstFileLocations_1 = firstFileLocations; _i < firstFileLocations_1.length; _i++) {
-                            var node = firstFileLocations_1[_i];
-                            addDuplicateDeclarationError(node, message, symbolName, secondFileLocations);
-                        }
-                        for (var _b = 0, secondFileLocations_1 = secondFileLocations; _b < secondFileLocations_1.length; _b++) {
-                            var node = secondFileLocations_1[_b];
-                            addDuplicateDeclarationError(node, message, symbolName, firstFileLocations);
+            return ts.arrayFrom(seen.values());
+        }
+        function checkInheritedPropertiesAreIdentical(type, typeNode) {
+            var baseTypes = getBaseTypes(type);
+            if (baseTypes.length < 2) {
+                return true;
+            }
+            var seen = new ts.Map();
+            ts.forEach(resolveDeclaredMembers(type).declaredProperties, function (p) { seen.set(p.escapedName, { prop: p, containingType: type }); });
+            var ok = true;
+            for (var _i = 0, baseTypes_3 = baseTypes; _i < baseTypes_3.length; _i++) {
+                var base = baseTypes_3[_i];
+                var properties = getPropertiesOfType(getTypeWithThisArgument(base, type.thisType));
+                for (var _a = 0, properties_6 = properties; _a < properties_6.length; _a++) {
+                    var prop = properties_6[_a];
+                    var existing = seen.get(prop.escapedName);
+                    if (!existing) {
+                        seen.set(prop.escapedName, { prop: prop, containingType: base });
+                    }
+                    else {
+                        var isInheritedProperty = existing.containingType !== type;
+                        if (isInheritedProperty && !isPropertyIdenticalTo(existing.prop, prop)) {
+                            ok = false;
+                            var typeName1 = typeToString(existing.containingType);
+                            var typeName2 = typeToString(base);
+                            var errorInfo = ts.chainDiagnosticMessages(/*details*/ undefined, ts.Diagnostics.Named_property_0_of_types_1_and_2_are_not_identical, symbolToString(prop), typeName1, typeName2);
+                            errorInfo = ts.chainDiagnosticMessages(errorInfo, ts.Diagnostics.Interface_0_cannot_simultaneously_extend_types_1_and_2, typeToString(type), typeName1, typeName2);
+                            diagnostics.add(ts.createDiagnosticForNodeFromMessageChain(typeNode, errorInfo));
                         }
-                    });
-                }
-                else {
-                    // Otherwise issue top-level error since the files appear very identical in terms of what they contain
-                    var list = ts.arrayFrom(conflictingSymbols.keys()).join(", ");
-                    diagnostics.add(ts.addRelatedInfo(ts.createDiagnosticForNode(firstFile, ts.Diagnostics.Definitions_of_the_following_identifiers_conflict_with_those_in_another_file_Colon_0, list), ts.createDiagnosticForNode(secondFile, ts.Diagnostics.Conflicts_are_in_this_file)));
-                    diagnostics.add(ts.addRelatedInfo(ts.createDiagnosticForNode(secondFile, ts.Diagnostics.Definitions_of_the_following_identifiers_conflict_with_those_in_another_file_Colon_0, list), ts.createDiagnosticForNode(firstFile, ts.Diagnostics.Conflicts_are_in_this_file)));
+                    }
                 }
-            });
-            amalgamatedDuplicates = undefined;
+            }
+            return ok;
         }
-        function checkExternalEmitHelpers(location, helpers) {
-            if ((requestedExternalEmitHelpers & helpers) !== helpers && compilerOptions.importHelpers) {
-                var sourceFile = ts.getSourceFileOfNode(location);
-                if (ts.isEffectiveExternalModule(sourceFile, compilerOptions) && !(location.flags & 8388608 /* Ambient */)) {
-                    var helpersModule = resolveHelpersModule(sourceFile, location);
-                    if (helpersModule !== unknownSymbol) {
-                        var uncheckedHelpers = helpers & ~requestedExternalEmitHelpers;
-                        for (var helper = 1 /* FirstEmitHelper */; helper <= 1048576 /* LastEmitHelper */; helper <<= 1) {
-                            if (uncheckedHelpers & helper) {
-                                var name = getHelperName(helper);
-                                var symbol = getSymbol(helpersModule.exports, ts.escapeLeadingUnderscores(name), 111551 /* Value */);
-                                if (!symbol) {
-                                    error(location, ts.Diagnostics.This_syntax_requires_an_imported_helper_named_1_which_does_not_exist_in_0_Consider_upgrading_your_version_of_0, ts.externalHelpersModuleNameText, name);
-                                }
+        function checkPropertyInitialization(node) {
+            if (!strictNullChecks || !strictPropertyInitialization || node.flags & 8388608 /* Ambient */) {
+                return;
+            }
+            var constructor = findConstructorDeclaration(node);
+            for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                var member = _a[_i];
+                if (ts.getEffectiveModifierFlags(member) & 2 /* Ambient */) {
+                    continue;
+                }
+                if (isInstancePropertyWithoutInitializer(member)) {
+                    var propName = member.name;
+                    if (ts.isIdentifier(propName) || ts.isPrivateIdentifier(propName)) {
+                        var type = getTypeOfSymbol(getSymbolOfNode(member));
+                        if (!(type.flags & 3 /* AnyOrUnknown */ || getFalsyFlags(type) & 32768 /* Undefined */)) {
+                            if (!constructor || !isPropertyInitializedInConstructor(propName, type, constructor)) {
+                                error(member.name, ts.Diagnostics.Property_0_has_no_initializer_and_is_not_definitely_assigned_in_the_constructor, ts.declarationNameToString(propName));
                             }
                         }
                     }
-                    requestedExternalEmitHelpers |= helpers;
                 }
             }
         }
-        function getHelperName(helper) {
-            switch (helper) {
-                case 1 /* Extends */: return "__extends";
-                case 2 /* Assign */: return "__assign";
-                case 4 /* Rest */: return "__rest";
-                case 8 /* Decorate */: return "__decorate";
-                case 16 /* Metadata */: return "__metadata";
-                case 32 /* Param */: return "__param";
-                case 64 /* Awaiter */: return "__awaiter";
-                case 128 /* Generator */: return "__generator";
-                case 256 /* Values */: return "__values";
-                case 512 /* Read */: return "__read";
-                case 1024 /* Spread */: return "__spread";
-                case 2048 /* SpreadArrays */: return "__spreadArrays";
-                case 4096 /* Await */: return "__await";
-                case 8192 /* AsyncGenerator */: return "__asyncGenerator";
-                case 16384 /* AsyncDelegator */: return "__asyncDelegator";
-                case 32768 /* AsyncValues */: return "__asyncValues";
-                case 65536 /* ExportStar */: return "__exportStar";
-                case 131072 /* MakeTemplateObject */: return "__makeTemplateObject";
-                case 262144 /* ClassPrivateFieldGet */: return "__classPrivateFieldGet";
-                case 524288 /* ClassPrivateFieldSet */: return "__classPrivateFieldSet";
-                case 1048576 /* CreateBinding */: return "__createBinding";
-                default: return ts.Debug.fail("Unrecognized helper");
+        function isInstancePropertyWithoutInitializer(node) {
+            return node.kind === 163 /* PropertyDeclaration */ &&
+                !ts.hasSyntacticModifier(node, 32 /* Static */ | 128 /* Abstract */) &&
+                !node.exclamationToken &&
+                !node.initializer;
+        }
+        function isPropertyInitializedInConstructor(propName, propType, constructor) {
+            var reference = ts.factory.createPropertyAccessExpression(ts.factory.createThis(), propName);
+            ts.setParent(reference.expression, reference);
+            ts.setParent(reference, constructor);
+            reference.flowNode = constructor.returnFlowNode;
+            var flowType = getFlowTypeOfReference(reference, propType, getOptionalType(propType));
+            return !(getFalsyFlags(flowType) & 32768 /* Undefined */);
+        }
+        function checkInterfaceDeclaration(node) {
+            // Grammar checking
+            if (!checkGrammarDecoratorsAndModifiers(node))
+                checkGrammarInterfaceDeclaration(node);
+            checkTypeParameters(node.typeParameters);
+            if (produceDiagnostics) {
+                checkTypeNameIsReserved(node.name, ts.Diagnostics.Interface_name_cannot_be_0);
+                checkExportsOnMergedDeclarations(node);
+                var symbol = getSymbolOfNode(node);
+                checkTypeParameterListsIdentical(symbol);
+                // Only check this symbol once
+                var firstInterfaceDecl = ts.getDeclarationOfKind(symbol, 253 /* InterfaceDeclaration */);
+                if (node === firstInterfaceDecl) {
+                    var type = getDeclaredTypeOfSymbol(symbol);
+                    var typeWithThis = getTypeWithThisArgument(type);
+                    // run subsequent checks only if first set succeeded
+                    if (checkInheritedPropertiesAreIdentical(type, node.name)) {
+                        for (var _i = 0, _a = getBaseTypes(type); _i < _a.length; _i++) {
+                            var baseType = _a[_i];
+                            checkTypeAssignableTo(typeWithThis, getTypeWithThisArgument(baseType, type.thisType), node.name, ts.Diagnostics.Interface_0_incorrectly_extends_interface_1);
+                        }
+                        checkIndexConstraints(type);
+                    }
+                }
+                checkObjectTypeForDuplicateDeclarations(node);
+            }
+            ts.forEach(ts.getInterfaceBaseTypeNodes(node), function (heritageElement) {
+                if (!ts.isEntityNameExpression(heritageElement.expression)) {
+                    error(heritageElement.expression, ts.Diagnostics.An_interface_can_only_extend_an_identifier_Slashqualified_name_with_optional_type_arguments);
+                }
+                checkTypeReferenceNode(heritageElement);
+            });
+            ts.forEach(node.members, checkSourceElement);
+            if (produceDiagnostics) {
+                checkTypeForDuplicateIndexSignatures(node);
+                registerForUnusedIdentifiersCheck(node);
             }
         }
-        function resolveHelpersModule(node, errorNode) {
-            if (!externalHelpersModule) {
-                externalHelpersModule = resolveExternalModule(node, ts.externalHelpersModuleNameText, ts.Diagnostics.This_syntax_requires_an_imported_helper_but_module_0_cannot_be_found, errorNode) || unknownSymbol;
+        function checkTypeAliasDeclaration(node) {
+            // Grammar checking
+            checkGrammarDecoratorsAndModifiers(node);
+            checkTypeNameIsReserved(node.name, ts.Diagnostics.Type_alias_name_cannot_be_0);
+            checkExportsOnMergedDeclarations(node);
+            checkTypeParameters(node.typeParameters);
+            if (node.type.kind === 136 /* IntrinsicKeyword */) {
+                if (!intrinsicTypeKinds.has(node.name.escapedText) || ts.length(node.typeParameters) !== 1) {
+                    error(node.type, ts.Diagnostics.The_intrinsic_keyword_can_only_be_used_to_declare_compiler_provided_intrinsic_types);
+                }
+            }
+            else {
+                checkSourceElement(node.type);
+                registerForUnusedIdentifiersCheck(node);
             }
-            return externalHelpersModule;
         }
-        // GRAMMAR CHECKING
-        function checkGrammarDecoratorsAndModifiers(node) {
-            return checkGrammarDecorators(node) || checkGrammarModifiers(node);
+        function computeEnumMemberValues(node) {
+            var nodeLinks = getNodeLinks(node);
+            if (!(nodeLinks.flags & 16384 /* EnumValuesComputed */)) {
+                nodeLinks.flags |= 16384 /* EnumValuesComputed */;
+                var autoValue = 0;
+                for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
+                    var member = _a[_i];
+                    var value = computeMemberValue(member, autoValue);
+                    getNodeLinks(member).enumMemberValue = value;
+                    autoValue = typeof value === "number" ? value + 1 : undefined;
+                }
+            }
         }
-        function checkGrammarDecorators(node) {
-            if (!node.decorators) {
-                return false;
+        function computeMemberValue(member, autoValue) {
+            if (ts.isComputedNonLiteralName(member.name)) {
+                error(member.name, ts.Diagnostics.Computed_property_names_are_not_allowed_in_enums);
             }
-            if (!ts.nodeCanBeDecorated(node, node.parent, node.parent.parent)) {
-                if (node.kind === 161 /* MethodDeclaration */ && !ts.nodeIsPresent(node.body)) {
-                    return grammarErrorOnFirstToken(node, ts.Diagnostics.A_decorator_can_only_decorate_a_method_implementation_not_an_overload);
-                }
-                else {
-                    return grammarErrorOnFirstToken(node, ts.Diagnostics.Decorators_are_not_valid_here);
+            else {
+                var text = ts.getTextOfPropertyName(member.name);
+                if (isNumericLiteralName(text) && !isInfinityOrNaNString(text)) {
+                    error(member.name, ts.Diagnostics.An_enum_member_cannot_have_a_numeric_name);
                 }
             }
-            else if (node.kind === 163 /* GetAccessor */ || node.kind === 164 /* SetAccessor */) {
-                var accessors = ts.getAllAccessorDeclarations(node.parent.members, node);
-                if (accessors.firstAccessor.decorators && node === accessors.secondAccessor) {
-                    return grammarErrorOnFirstToken(node, ts.Diagnostics.Decorators_cannot_be_applied_to_multiple_get_Slashset_accessors_of_the_same_name);
-                }
+            if (member.initializer) {
+                return computeConstantValue(member);
             }
-            return false;
+            // In ambient non-const numeric enum declarations, enum members without initializers are
+            // considered computed members (as opposed to having auto-incremented values).
+            if (member.parent.flags & 8388608 /* Ambient */ && !ts.isEnumConst(member.parent) && getEnumKind(getSymbolOfNode(member.parent)) === 0 /* Numeric */) {
+                return undefined;
+            }
+            // If the member declaration specifies no value, the member is considered a constant enum member.
+            // If the member is the first member in the enum declaration, it is assigned the value zero.
+            // Otherwise, it is assigned the value of the immediately preceding member plus one, and an error
+            // occurs if the immediately preceding member is not a constant enum member.
+            if (autoValue !== undefined) {
+                return autoValue;
+            }
+            error(member.name, ts.Diagnostics.Enum_member_must_have_initializer);
+            return undefined;
         }
-        function checkGrammarModifiers(node) {
-            var quickResult = reportObviousModifierErrors(node);
-            if (quickResult !== undefined) {
-                return quickResult;
+        function computeConstantValue(member) {
+            var enumKind = getEnumKind(getSymbolOfNode(member.parent));
+            var isConstEnum = ts.isEnumConst(member.parent);
+            var initializer = member.initializer;
+            var value = enumKind === 1 /* Literal */ && !isLiteralEnumMember(member) ? undefined : evaluate(initializer);
+            if (value !== undefined) {
+                if (isConstEnum && typeof value === "number" && !isFinite(value)) {
+                    error(initializer, isNaN(value) ?
+                        ts.Diagnostics.const_enum_member_initializer_was_evaluated_to_disallowed_value_NaN :
+                        ts.Diagnostics.const_enum_member_initializer_was_evaluated_to_a_non_finite_value);
+                }
             }
-            var lastStatic, lastDeclare, lastAsync, lastReadonly;
-            var flags = 0 /* None */;
-            for (var _i = 0, _a = node.modifiers; _i < _a.length; _i++) {
-                var modifier = _a[_i];
-                if (modifier.kind !== 138 /* ReadonlyKeyword */) {
-                    if (node.kind === 158 /* PropertySignature */ || node.kind === 160 /* MethodSignature */) {
-                        return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_type_member, ts.tokenToString(modifier.kind));
-                    }
-                    if (node.kind === 167 /* IndexSignature */) {
-                        return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_an_index_signature, ts.tokenToString(modifier.kind));
-                    }
+            else if (enumKind === 1 /* Literal */) {
+                error(initializer, ts.Diagnostics.Computed_values_are_not_permitted_in_an_enum_with_string_valued_members);
+                return 0;
+            }
+            else if (isConstEnum) {
+                error(initializer, ts.Diagnostics.const_enum_member_initializers_can_only_contain_literal_values_and_other_computed_enum_values);
+            }
+            else if (member.parent.flags & 8388608 /* Ambient */) {
+                error(initializer, ts.Diagnostics.In_ambient_enum_declarations_member_initializer_must_be_constant_expression);
+            }
+            else {
+                // Only here do we need to check that the initializer is assignable to the enum type.
+                var source = checkExpression(initializer);
+                if (!isTypeAssignableToKind(source, 296 /* NumberLike */)) {
+                    error(initializer, ts.Diagnostics.Only_numeric_enums_can_have_computed_members_but_this_expression_has_type_0_If_you_do_not_need_exhaustiveness_checks_consider_using_an_object_literal_instead, typeToString(source));
                 }
-                switch (modifier.kind) {
-                    case 81 /* ConstKeyword */:
-                        if (node.kind !== 248 /* EnumDeclaration */) {
-                            return grammarErrorOnNode(node, ts.Diagnostics.A_class_member_cannot_have_the_0_keyword, ts.tokenToString(81 /* ConstKeyword */));
-                        }
-                        break;
-                    case 119 /* PublicKeyword */:
-                    case 118 /* ProtectedKeyword */:
-                    case 117 /* PrivateKeyword */:
-                        var text = visibilityToString(ts.modifierToFlag(modifier.kind));
-                        if (flags & 28 /* AccessibilityModifier */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics.Accessibility_modifier_already_seen);
-                        }
-                        else if (flags & 32 /* Static */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "static");
-                        }
-                        else if (flags & 64 /* Readonly */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "readonly");
-                        }
-                        else if (flags & 256 /* Async */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "async");
-                        }
-                        else if (node.parent.kind === 250 /* ModuleBlock */ || node.parent.kind === 290 /* SourceFile */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_module_or_namespace_element, text);
-                        }
-                        else if (flags & 128 /* Abstract */) {
-                            if (modifier.kind === 117 /* PrivateKeyword */) {
-                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, text, "abstract");
-                            }
-                            else {
-                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "abstract");
+                else {
+                    checkTypeAssignableTo(source, getDeclaredTypeOfSymbol(getSymbolOfNode(member.parent)), initializer, /*headMessage*/ undefined);
+                }
+            }
+            return value;
+            function evaluate(expr) {
+                switch (expr.kind) {
+                    case 214 /* PrefixUnaryExpression */:
+                        var value_2 = evaluate(expr.operand);
+                        if (typeof value_2 === "number") {
+                            switch (expr.operator) {
+                                case 39 /* PlusToken */: return value_2;
+                                case 40 /* MinusToken */: return -value_2;
+                                case 54 /* TildeToken */: return ~value_2;
                             }
                         }
-                        else if (ts.isPrivateIdentifierPropertyDeclaration(node)) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics.An_accessibility_modifier_cannot_be_used_with_a_private_identifier);
-                        }
-                        flags |= ts.modifierToFlag(modifier.kind);
-                        break;
-                    case 120 /* StaticKeyword */:
-                        if (flags & 32 /* Static */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "static");
-                        }
-                        else if (flags & 64 /* Readonly */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "static", "readonly");
-                        }
-                        else if (flags & 256 /* Async */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "static", "async");
-                        }
-                        else if (node.parent.kind === 250 /* ModuleBlock */ || node.parent.kind === 290 /* SourceFile */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_module_or_namespace_element, "static");
-                        }
-                        else if (node.kind === 156 /* Parameter */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "static");
-                        }
-                        else if (flags & 128 /* Abstract */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "static", "abstract");
-                        }
-                        else if (ts.isPrivateIdentifierPropertyDeclaration(node)) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_a_private_identifier, "static");
-                        }
-                        flags |= 32 /* Static */;
-                        lastStatic = modifier;
                         break;
-                    case 138 /* ReadonlyKeyword */:
-                        if (flags & 64 /* Readonly */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "readonly");
+                    case 216 /* BinaryExpression */:
+                        var left = evaluate(expr.left);
+                        var right = evaluate(expr.right);
+                        if (typeof left === "number" && typeof right === "number") {
+                            switch (expr.operatorToken.kind) {
+                                case 51 /* BarToken */: return left | right;
+                                case 50 /* AmpersandToken */: return left & right;
+                                case 48 /* GreaterThanGreaterThanToken */: return left >> right;
+                                case 49 /* GreaterThanGreaterThanGreaterThanToken */: return left >>> right;
+                                case 47 /* LessThanLessThanToken */: return left << right;
+                                case 52 /* CaretToken */: return left ^ right;
+                                case 41 /* AsteriskToken */: return left * right;
+                                case 43 /* SlashToken */: return left / right;
+                                case 39 /* PlusToken */: return left + right;
+                                case 40 /* MinusToken */: return left - right;
+                                case 44 /* PercentToken */: return left % right;
+                                case 42 /* AsteriskAsteriskToken */: return Math.pow(left, right);
+                            }
                         }
-                        else if (node.kind !== 159 /* PropertyDeclaration */ && node.kind !== 158 /* PropertySignature */ && node.kind !== 167 /* IndexSignature */ && node.kind !== 156 /* Parameter */) {
-                            // If node.kind === SyntaxKind.Parameter, checkParameter report an error if it's not a parameter property.
-                            return grammarErrorOnNode(modifier, ts.Diagnostics.readonly_modifier_can_only_appear_on_a_property_declaration_or_index_signature);
+                        else if (typeof left === "string" && typeof right === "string" && expr.operatorToken.kind === 39 /* PlusToken */) {
+                            return left + right;
                         }
-                        flags |= 64 /* Readonly */;
-                        lastReadonly = modifier;
                         break;
-                    case 89 /* ExportKeyword */:
-                        if (flags & 1 /* Export */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "export");
-                        }
-                        else if (flags & 2 /* Ambient */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "export", "declare");
-                        }
-                        else if (flags & 128 /* Abstract */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "export", "abstract");
-                        }
-                        else if (flags & 256 /* Async */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "export", "async");
-                        }
-                        else if (ts.isClassLike(node.parent)) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_class_element, "export");
-                        }
-                        else if (node.kind === 156 /* Parameter */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "export");
+                    case 10 /* StringLiteral */:
+                    case 14 /* NoSubstitutionTemplateLiteral */:
+                        return expr.text;
+                    case 8 /* NumericLiteral */:
+                        checkGrammarNumericLiteral(expr);
+                        return +expr.text;
+                    case 207 /* ParenthesizedExpression */:
+                        return evaluate(expr.expression);
+                    case 78 /* Identifier */:
+                        var identifier = expr;
+                        if (isInfinityOrNaNString(identifier.escapedText)) {
+                            return +(identifier.escapedText);
                         }
-                        flags |= 1 /* Export */;
-                        break;
-                    case 84 /* DefaultKeyword */:
-                        var container = node.parent.kind === 290 /* SourceFile */ ? node.parent : node.parent.parent;
-                        if (container.kind === 249 /* ModuleDeclaration */ && !ts.isAmbientModule(container)) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics.A_default_export_can_only_be_used_in_an_ECMAScript_style_module);
+                        return ts.nodeIsMissing(expr) ? 0 : evaluateEnumMember(expr, getSymbolOfNode(member.parent), identifier.escapedText);
+                    case 202 /* ElementAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
+                        var ex = expr;
+                        if (isConstantMemberAccess(ex)) {
+                            var type = getTypeOfExpression(ex.expression);
+                            if (type.symbol && type.symbol.flags & 384 /* Enum */) {
+                                var name = void 0;
+                                if (ex.kind === 201 /* PropertyAccessExpression */) {
+                                    name = ex.name.escapedText;
+                                }
+                                else {
+                                    name = ts.escapeLeadingUnderscores(ts.cast(ex.argumentExpression, ts.isLiteralExpression).text);
+                                }
+                                return evaluateEnumMember(expr, type.symbol, name);
+                            }
                         }
-                        flags |= 512 /* Default */;
                         break;
-                    case 130 /* DeclareKeyword */:
-                        if (flags & 2 /* Ambient */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "declare");
-                        }
-                        else if (flags & 256 /* Async */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_in_an_ambient_context, "async");
+                }
+                return undefined;
+            }
+            function evaluateEnumMember(expr, enumSymbol, name) {
+                var memberSymbol = enumSymbol.exports.get(name);
+                if (memberSymbol) {
+                    var declaration = memberSymbol.valueDeclaration;
+                    if (declaration !== member) {
+                        if (isBlockScopedNameDeclaredBeforeUse(declaration, member)) {
+                            return getEnumMemberValue(declaration);
                         }
-                        else if (ts.isClassLike(node.parent) && !ts.isPropertyDeclaration(node)) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_class_element, "declare");
+                        error(expr, ts.Diagnostics.A_member_initializer_in_a_enum_declaration_cannot_reference_members_declared_after_it_including_members_defined_in_other_enums);
+                        return 0;
+                    }
+                    else {
+                        error(expr, ts.Diagnostics.Property_0_is_used_before_being_assigned, symbolToString(memberSymbol));
+                    }
+                }
+                return undefined;
+            }
+        }
+        function isConstantMemberAccess(node) {
+            return node.kind === 78 /* Identifier */ ||
+                node.kind === 201 /* PropertyAccessExpression */ && isConstantMemberAccess(node.expression) ||
+                node.kind === 202 /* ElementAccessExpression */ && isConstantMemberAccess(node.expression) &&
+                    ts.isStringLiteralLike(node.argumentExpression);
+        }
+        function checkEnumDeclaration(node) {
+            if (!produceDiagnostics) {
+                return;
+            }
+            // Grammar checking
+            checkGrammarDecoratorsAndModifiers(node);
+            checkTypeNameIsReserved(node.name, ts.Diagnostics.Enum_name_cannot_be_0);
+            checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
+            checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+            checkExportsOnMergedDeclarations(node);
+            node.members.forEach(checkEnumMember);
+            computeEnumMemberValues(node);
+            // Spec 2014 - Section 9.3:
+            // It isn't possible for one enum declaration to continue the automatic numbering sequence of another,
+            // and when an enum type has multiple declarations, only one declaration is permitted to omit a value
+            // for the first member.
+            //
+            // Only perform this check once per symbol
+            var enumSymbol = getSymbolOfNode(node);
+            var firstDeclaration = ts.getDeclarationOfKind(enumSymbol, node.kind);
+            if (node === firstDeclaration) {
+                if (enumSymbol.declarations.length > 1) {
+                    var enumIsConst_1 = ts.isEnumConst(node);
+                    // check that const is placed\omitted on all enum declarations
+                    ts.forEach(enumSymbol.declarations, function (decl) {
+                        if (ts.isEnumDeclaration(decl) && ts.isEnumConst(decl) !== enumIsConst_1) {
+                            error(ts.getNameOfDeclaration(decl), ts.Diagnostics.Enum_declarations_must_all_be_const_or_non_const);
                         }
-                        else if (node.kind === 156 /* Parameter */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "declare");
+                    });
+                }
+                var seenEnumMissingInitialInitializer_1 = false;
+                ts.forEach(enumSymbol.declarations, function (declaration) {
+                    // return true if we hit a violation of the rule, false otherwise
+                    if (declaration.kind !== 255 /* EnumDeclaration */) {
+                        return false;
+                    }
+                    var enumDeclaration = declaration;
+                    if (!enumDeclaration.members.length) {
+                        return false;
+                    }
+                    var firstEnumMember = enumDeclaration.members[0];
+                    if (!firstEnumMember.initializer) {
+                        if (seenEnumMissingInitialInitializer_1) {
+                            error(firstEnumMember.name, ts.Diagnostics.In_an_enum_with_multiple_declarations_only_one_declaration_can_omit_an_initializer_for_its_first_enum_element);
                         }
-                        else if ((node.parent.flags & 8388608 /* Ambient */) && node.parent.kind === 250 /* ModuleBlock */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics.A_declare_modifier_cannot_be_used_in_an_already_ambient_context);
+                        else {
+                            seenEnumMissingInitialInitializer_1 = true;
                         }
-                        else if (ts.isPrivateIdentifierPropertyDeclaration(node)) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_a_private_identifier, "declare");
+                    }
+                });
+            }
+        }
+        function checkEnumMember(node) {
+            if (ts.isPrivateIdentifier(node.name)) {
+                error(node, ts.Diagnostics.An_enum_member_cannot_be_named_with_a_private_identifier);
+            }
+        }
+        function getFirstNonAmbientClassOrFunctionDeclaration(symbol) {
+            var declarations = symbol.declarations;
+            for (var _i = 0, declarations_8 = declarations; _i < declarations_8.length; _i++) {
+                var declaration = declarations_8[_i];
+                if ((declaration.kind === 252 /* ClassDeclaration */ ||
+                    (declaration.kind === 251 /* FunctionDeclaration */ && ts.nodeIsPresent(declaration.body))) &&
+                    !(declaration.flags & 8388608 /* Ambient */)) {
+                    return declaration;
+                }
+            }
+            return undefined;
+        }
+        function inSameLexicalScope(node1, node2) {
+            var container1 = ts.getEnclosingBlockScopeContainer(node1);
+            var container2 = ts.getEnclosingBlockScopeContainer(node2);
+            if (isGlobalSourceFile(container1)) {
+                return isGlobalSourceFile(container2);
+            }
+            else if (isGlobalSourceFile(container2)) {
+                return false;
+            }
+            else {
+                return container1 === container2;
+            }
+        }
+        function checkModuleDeclaration(node) {
+            if (produceDiagnostics) {
+                // Grammar checking
+                var isGlobalAugmentation = ts.isGlobalScopeAugmentation(node);
+                var inAmbientContext = node.flags & 8388608 /* Ambient */;
+                if (isGlobalAugmentation && !inAmbientContext) {
+                    error(node.name, ts.Diagnostics.Augmentations_for_the_global_scope_should_have_declare_modifier_unless_they_appear_in_already_ambient_context);
+                }
+                var isAmbientExternalModule = ts.isAmbientModule(node);
+                var contextErrorMessage = isAmbientExternalModule
+                    ? ts.Diagnostics.An_ambient_module_declaration_is_only_allowed_at_the_top_level_in_a_file
+                    : ts.Diagnostics.A_namespace_declaration_is_only_allowed_in_a_namespace_or_module;
+                if (checkGrammarModuleElementContext(node, contextErrorMessage)) {
+                    // If we hit a module declaration in an illegal context, just bail out to avoid cascading errors.
+                    return;
+                }
+                if (!checkGrammarDecoratorsAndModifiers(node)) {
+                    if (!inAmbientContext && node.name.kind === 10 /* StringLiteral */) {
+                        grammarErrorOnNode(node.name, ts.Diagnostics.Only_ambient_modules_can_use_quoted_names);
+                    }
+                }
+                if (ts.isIdentifier(node.name)) {
+                    checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
+                    checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+                }
+                checkExportsOnMergedDeclarations(node);
+                var symbol = getSymbolOfNode(node);
+                // The following checks only apply on a non-ambient instantiated module declaration.
+                if (symbol.flags & 512 /* ValueModule */
+                    && !inAmbientContext
+                    && symbol.declarations.length > 1
+                    && isInstantiatedModule(node, !!compilerOptions.preserveConstEnums || !!compilerOptions.isolatedModules)) {
+                    var firstNonAmbientClassOrFunc = getFirstNonAmbientClassOrFunctionDeclaration(symbol);
+                    if (firstNonAmbientClassOrFunc) {
+                        if (ts.getSourceFileOfNode(node) !== ts.getSourceFileOfNode(firstNonAmbientClassOrFunc)) {
+                            error(node.name, ts.Diagnostics.A_namespace_declaration_cannot_be_in_a_different_file_from_a_class_or_function_with_which_it_is_merged);
                         }
-                        flags |= 2 /* Ambient */;
-                        lastDeclare = modifier;
-                        break;
-                    case 122 /* AbstractKeyword */:
-                        if (flags & 128 /* Abstract */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "abstract");
+                        else if (node.pos < firstNonAmbientClassOrFunc.pos) {
+                            error(node.name, ts.Diagnostics.A_namespace_declaration_cannot_be_located_prior_to_a_class_or_function_with_which_it_is_merged);
                         }
-                        if (node.kind !== 245 /* ClassDeclaration */) {
-                            if (node.kind !== 161 /* MethodDeclaration */ &&
-                                node.kind !== 159 /* PropertyDeclaration */ &&
-                                node.kind !== 163 /* GetAccessor */ &&
-                                node.kind !== 164 /* SetAccessor */) {
-                                return grammarErrorOnNode(modifier, ts.Diagnostics.abstract_modifier_can_only_appear_on_a_class_method_or_property_declaration);
-                            }
-                            if (!(node.parent.kind === 245 /* ClassDeclaration */ && ts.hasModifier(node.parent, 128 /* Abstract */))) {
-                                return grammarErrorOnNode(modifier, ts.Diagnostics.Abstract_methods_can_only_appear_within_an_abstract_class);
-                            }
-                            if (flags & 32 /* Static */) {
-                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "static", "abstract");
-                            }
-                            if (flags & 8 /* Private */) {
-                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "private", "abstract");
+                    }
+                    // if the module merges with a class declaration in the same lexical scope,
+                    // we need to track this to ensure the correct emit.
+                    var mergedClass = ts.getDeclarationOfKind(symbol, 252 /* ClassDeclaration */);
+                    if (mergedClass &&
+                        inSameLexicalScope(node, mergedClass)) {
+                        getNodeLinks(node).flags |= 32768 /* LexicalModuleMergesWithClass */;
+                    }
+                }
+                if (isAmbientExternalModule) {
+                    if (ts.isExternalModuleAugmentation(node)) {
+                        // body of the augmentation should be checked for consistency only if augmentation was applied to its target (either global scope or module)
+                        // otherwise we'll be swamped in cascading errors.
+                        // We can detect if augmentation was applied using following rules:
+                        // - augmentation for a global scope is always applied
+                        // - augmentation for some external module is applied if symbol for augmentation is merged (it was combined with target module).
+                        var checkBody = isGlobalAugmentation || (getSymbolOfNode(node).flags & 33554432 /* Transient */);
+                        if (checkBody && node.body) {
+                            for (var _i = 0, _a = node.body.statements; _i < _a.length; _i++) {
+                                var statement = _a[_i];
+                                checkModuleAugmentationElement(statement, isGlobalAugmentation);
                             }
                         }
-                        if (ts.isNamedDeclaration(node) && node.name.kind === 76 /* PrivateIdentifier */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_a_private_identifier, "abstract");
+                    }
+                    else if (isGlobalSourceFile(node.parent)) {
+                        if (isGlobalAugmentation) {
+                            error(node.name, ts.Diagnostics.Augmentations_for_the_global_scope_can_only_be_directly_nested_in_external_modules_or_ambient_module_declarations);
                         }
-                        flags |= 128 /* Abstract */;
-                        break;
-                    case 126 /* AsyncKeyword */:
-                        if (flags & 256 /* Async */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "async");
+                        else if (ts.isExternalModuleNameRelative(ts.getTextOfIdentifierOrLiteral(node.name))) {
+                            error(node.name, ts.Diagnostics.Ambient_module_declaration_cannot_specify_relative_module_name);
                         }
-                        else if (flags & 2 /* Ambient */ || node.parent.flags & 8388608 /* Ambient */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_in_an_ambient_context, "async");
+                    }
+                    else {
+                        if (isGlobalAugmentation) {
+                            error(node.name, ts.Diagnostics.Augmentations_for_the_global_scope_can_only_be_directly_nested_in_external_modules_or_ambient_module_declarations);
                         }
-                        else if (node.kind === 156 /* Parameter */) {
-                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "async");
+                        else {
+                            // Node is not an augmentation and is not located on the script level.
+                            // This means that this is declaration of ambient module that is located in other module or namespace which is prohibited.
+                            error(node.name, ts.Diagnostics.Ambient_modules_cannot_be_nested_in_other_modules_or_namespaces);
                         }
-                        flags |= 256 /* Async */;
-                        lastAsync = modifier;
-                        break;
+                    }
                 }
             }
-            if (node.kind === 162 /* Constructor */) {
-                if (flags & 32 /* Static */) {
-                    return grammarErrorOnNode(lastStatic, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "static");
-                }
-                if (flags & 128 /* Abstract */) {
-                    return grammarErrorOnNode(lastStatic, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "abstract"); // TODO: GH#18217
-                }
-                else if (flags & 256 /* Async */) {
-                    return grammarErrorOnNode(lastAsync, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "async");
-                }
-                else if (flags & 64 /* Readonly */) {
-                    return grammarErrorOnNode(lastReadonly, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "readonly");
+            if (node.body) {
+                checkSourceElement(node.body);
+                if (!ts.isGlobalScopeAugmentation(node)) {
+                    registerForUnusedIdentifiersCheck(node);
                 }
-                return false;
-            }
-            else if ((node.kind === 254 /* ImportDeclaration */ || node.kind === 253 /* ImportEqualsDeclaration */) && flags & 2 /* Ambient */) {
-                return grammarErrorOnNode(lastDeclare, ts.Diagnostics.A_0_modifier_cannot_be_used_with_an_import_declaration, "declare");
             }
-            else if (node.kind === 156 /* Parameter */ && (flags & 92 /* ParameterPropertyModifier */) && ts.isBindingPattern(node.name)) {
-                return grammarErrorOnNode(node, ts.Diagnostics.A_parameter_property_may_not_be_declared_using_a_binding_pattern);
-            }
-            else if (node.kind === 156 /* Parameter */ && (flags & 92 /* ParameterPropertyModifier */) && node.dotDotDotToken) {
-                return grammarErrorOnNode(node, ts.Diagnostics.A_parameter_property_cannot_be_declared_using_a_rest_parameter);
-            }
-            if (flags & 256 /* Async */) {
-                return checkGrammarAsyncModifier(node, lastAsync);
-            }
-            return false;
-        }
-        /**
-         * true | false: Early return this value from checkGrammarModifiers.
-         * undefined: Need to do full checking on the modifiers.
-         */
-        function reportObviousModifierErrors(node) {
-            return !node.modifiers
-                ? false
-                : shouldReportBadModifier(node)
-                    ? grammarErrorOnFirstToken(node, ts.Diagnostics.Modifiers_cannot_appear_here)
-                    : undefined;
         }
-        function shouldReportBadModifier(node) {
+        function checkModuleAugmentationElement(node, isGlobalAugmentation) {
             switch (node.kind) {
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 162 /* Constructor */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 167 /* IndexSignature */:
-                case 249 /* ModuleDeclaration */:
-                case 254 /* ImportDeclaration */:
-                case 253 /* ImportEqualsDeclaration */:
-                case 260 /* ExportDeclaration */:
-                case 259 /* ExportAssignment */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 156 /* Parameter */:
-                    return false;
-                default:
-                    if (node.parent.kind === 250 /* ModuleBlock */ || node.parent.kind === 290 /* SourceFile */) {
-                        return false;
+                case 232 /* VariableStatement */:
+                    // error each individual name in variable statement instead of marking the entire variable statement
+                    for (var _i = 0, _a = node.declarationList.declarations; _i < _a.length; _i++) {
+                        var decl = _a[_i];
+                        checkModuleAugmentationElement(decl, isGlobalAugmentation);
                     }
-                    switch (node.kind) {
-                        case 244 /* FunctionDeclaration */:
-                            return nodeHasAnyModifiersExcept(node, 126 /* AsyncKeyword */);
-                        case 245 /* ClassDeclaration */:
-                            return nodeHasAnyModifiersExcept(node, 122 /* AbstractKeyword */);
-                        case 246 /* InterfaceDeclaration */:
-                        case 225 /* VariableStatement */:
-                        case 247 /* TypeAliasDeclaration */:
-                            return true;
-                        case 248 /* EnumDeclaration */:
-                            return nodeHasAnyModifiersExcept(node, 81 /* ConstKeyword */);
-                        default:
-                            ts.Debug.fail();
-                            return false;
+                    break;
+                case 266 /* ExportAssignment */:
+                case 267 /* ExportDeclaration */:
+                    grammarErrorOnFirstToken(node, ts.Diagnostics.Exports_and_export_assignments_are_not_permitted_in_module_augmentations);
+                    break;
+                case 260 /* ImportEqualsDeclaration */:
+                case 261 /* ImportDeclaration */:
+                    grammarErrorOnFirstToken(node, ts.Diagnostics.Imports_are_not_permitted_in_module_augmentations_Consider_moving_them_to_the_enclosing_external_module);
+                    break;
+                case 198 /* BindingElement */:
+                case 249 /* VariableDeclaration */:
+                    var name = node.name;
+                    if (ts.isBindingPattern(name)) {
+                        for (var _b = 0, _c = name.elements; _b < _c.length; _b++) {
+                            var el = _c[_b];
+                            // mark individual names in binding pattern
+                            checkModuleAugmentationElement(el, isGlobalAugmentation);
+                        }
+                        break;
+                    }
+                // falls through
+                case 252 /* ClassDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                    if (isGlobalAugmentation) {
+                        return;
+                    }
+                    var symbol = getSymbolOfNode(node);
+                    if (symbol) {
+                        // module augmentations cannot introduce new names on the top level scope of the module
+                        // this is done it two steps
+                        // 1. quick check - if symbol for node is not merged - this is local symbol to this augmentation - report error
+                        // 2. main check - report error if value declaration of the parent symbol is module augmentation)
+                        var reportError = !(symbol.flags & 33554432 /* Transient */);
+                        if (!reportError) {
+                            // symbol should not originate in augmentation
+                            reportError = !!symbol.parent && ts.isExternalModuleAugmentation(symbol.parent.declarations[0]);
+                        }
                     }
+                    break;
             }
         }
-        function nodeHasAnyModifiersExcept(node, allowedModifier) {
-            return node.modifiers.length > 1 || node.modifiers[0].kind !== allowedModifier;
-        }
-        function checkGrammarAsyncModifier(node, asyncModifier) {
+        function getFirstNonModuleExportsIdentifier(node) {
             switch (node.kind) {
-                case 161 /* MethodDeclaration */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                    return false;
+                case 78 /* Identifier */:
+                    return node;
+                case 157 /* QualifiedName */:
+                    do {
+                        node = node.left;
+                    } while (node.kind !== 78 /* Identifier */);
+                    return node;
+                case 201 /* PropertyAccessExpression */:
+                    do {
+                        if (ts.isModuleExportsAccessExpression(node.expression) && !ts.isPrivateIdentifier(node.name)) {
+                            return node.name;
+                        }
+                        node = node.expression;
+                    } while (node.kind !== 78 /* Identifier */);
+                    return node;
             }
-            return grammarErrorOnNode(asyncModifier, ts.Diagnostics._0_modifier_cannot_be_used_here, "async");
         }
-        function checkGrammarForDisallowedTrailingComma(list, diag) {
-            if (diag === void 0) { diag = ts.Diagnostics.Trailing_comma_not_allowed; }
-            if (list && list.hasTrailingComma) {
-                return grammarErrorAtPos(list[0], list.end - ",".length, ",".length, diag);
+        function checkExternalImportOrExportDeclaration(node) {
+            var moduleName = ts.getExternalModuleName(node);
+            if (!moduleName || ts.nodeIsMissing(moduleName)) {
+                // Should be a parse error.
+                return false;
             }
-            return false;
-        }
-        function checkGrammarTypeParameterList(typeParameters, file) {
-            if (typeParameters && typeParameters.length === 0) {
-                var start = typeParameters.pos - "<".length;
-                var end = ts.skipTrivia(file.text, typeParameters.end) + ">".length;
-                return grammarErrorAtPos(file, start, end - start, ts.Diagnostics.Type_parameter_list_cannot_be_empty);
+            if (!ts.isStringLiteral(moduleName)) {
+                error(moduleName, ts.Diagnostics.String_literal_expected);
+                return false;
             }
-            return false;
+            var inAmbientExternalModule = node.parent.kind === 257 /* ModuleBlock */ && ts.isAmbientModule(node.parent.parent);
+            if (node.parent.kind !== 297 /* SourceFile */ && !inAmbientExternalModule) {
+                error(moduleName, node.kind === 267 /* ExportDeclaration */ ?
+                    ts.Diagnostics.Export_declarations_are_not_permitted_in_a_namespace :
+                    ts.Diagnostics.Import_declarations_in_a_namespace_cannot_reference_a_module);
+                return false;
+            }
+            if (inAmbientExternalModule && ts.isExternalModuleNameRelative(moduleName.text)) {
+                // we have already reported errors on top level imports/exports in external module augmentations in checkModuleDeclaration
+                // no need to do this again.
+                if (!isTopLevelInExternalModuleAugmentation(node)) {
+                    // TypeScript 1.0 spec (April 2013): 12.1.6
+                    // An ExternalImportDeclaration in an AmbientExternalModuleDeclaration may reference
+                    // other external modules only through top - level external module names.
+                    // Relative external module names are not permitted.
+                    error(node, ts.Diagnostics.Import_or_export_declaration_in_an_ambient_module_declaration_cannot_reference_module_through_relative_module_name);
+                    return false;
+                }
+            }
+            return true;
         }
-        function checkGrammarParameterList(parameters) {
-            var seenOptionalParameter = false;
-            var parameterCount = parameters.length;
-            for (var i = 0; i < parameterCount; i++) {
-                var parameter = parameters[i];
-                if (parameter.dotDotDotToken) {
-                    if (i !== (parameterCount - 1)) {
-                        return grammarErrorOnNode(parameter.dotDotDotToken, ts.Diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list);
-                    }
-                    if (!(parameter.flags & 8388608 /* Ambient */)) { // Allow `...foo,` in ambient declarations; see GH#23070
-                        checkGrammarForDisallowedTrailingComma(parameters, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
-                    }
-                    if (parameter.questionToken) {
-                        return grammarErrorOnNode(parameter.questionToken, ts.Diagnostics.A_rest_parameter_cannot_be_optional);
-                    }
-                    if (parameter.initializer) {
-                        return grammarErrorOnNode(parameter.name, ts.Diagnostics.A_rest_parameter_cannot_have_an_initializer);
-                    }
+        function checkAliasSymbol(node) {
+            var symbol = getSymbolOfNode(node);
+            var target = resolveAlias(symbol);
+            if (target !== unknownSymbol) {
+                // For external modules, `symbol` represents the local symbol for an alias.
+                // This local symbol will merge any other local declarations (excluding other aliases)
+                // and symbol.flags will contains combined representation for all merged declaration.
+                // Based on symbol.flags we can compute a set of excluded meanings (meaning that resolved alias should not have,
+                // otherwise it will conflict with some local declaration). Note that in addition to normal flags we include matching SymbolFlags.Export*
+                // in order to prevent collisions with declarations that were exported from the current module (they still contribute to local names).
+                symbol = getMergedSymbol(symbol.exportSymbol || symbol);
+                var excludedMeanings = (symbol.flags & (111551 /* Value */ | 1048576 /* ExportValue */) ? 111551 /* Value */ : 0) |
+                    (symbol.flags & 788968 /* Type */ ? 788968 /* Type */ : 0) |
+                    (symbol.flags & 1920 /* Namespace */ ? 1920 /* Namespace */ : 0);
+                if (target.flags & excludedMeanings) {
+                    var message = node.kind === 270 /* ExportSpecifier */ ?
+                        ts.Diagnostics.Export_declaration_conflicts_with_exported_declaration_of_0 :
+                        ts.Diagnostics.Import_declaration_conflicts_with_local_declaration_of_0;
+                    error(node, message, symbolToString(symbol));
                 }
-                else if (parameter.questionToken) {
-                    seenOptionalParameter = true;
-                    if (parameter.initializer) {
-                        return grammarErrorOnNode(parameter.name, ts.Diagnostics.Parameter_cannot_have_question_mark_and_initializer);
-                    }
+                // Don't allow to re-export something with no value side when `--isolatedModules` is set.
+                if (compilerOptions.isolatedModules
+                    && node.kind === 270 /* ExportSpecifier */
+                    && !node.parent.parent.isTypeOnly
+                    && !(target.flags & 111551 /* Value */)
+                    && !(node.flags & 8388608 /* Ambient */)) {
+                    error(node, ts.Diagnostics.Re_exporting_a_type_when_the_isolatedModules_flag_is_provided_requires_using_export_type);
                 }
-                else if (seenOptionalParameter && !parameter.initializer) {
-                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.A_required_parameter_cannot_follow_an_optional_parameter);
+                if (ts.isImportSpecifier(node) && ts.every(target.declarations, function (d) { return !!(ts.getCombinedNodeFlags(d) & 134217728 /* Deprecated */); })) {
+                    errorOrSuggestion(/* isError */ false, node.name, ts.Diagnostics._0_is_deprecated, symbol.escapedName);
                 }
             }
         }
-        function getNonSimpleParameters(parameters) {
-            return ts.filter(parameters, function (parameter) { return !!parameter.initializer || ts.isBindingPattern(parameter.name) || ts.isRestParameter(parameter); });
+        function checkImportBinding(node) {
+            checkCollisionWithRequireExportsInGeneratedCode(node, node.name);
+            checkCollisionWithGlobalPromiseInGeneratedCode(node, node.name);
+            checkAliasSymbol(node);
+            if (node.kind === 265 /* ImportSpecifier */ &&
+                ts.idText(node.propertyName || node.name) === "default" &&
+                compilerOptions.esModuleInterop &&
+                moduleKind !== ts.ModuleKind.System && moduleKind < ts.ModuleKind.ES2015) {
+                checkExternalEmitHelpers(node, 262144 /* ImportDefault */);
+            }
         }
-        function checkGrammarForUseStrictSimpleParameterList(node) {
-            if (languageVersion >= 3 /* ES2016 */) {
-                var useStrictDirective_1 = node.body && ts.isBlock(node.body) && ts.findUseStrictPrologue(node.body.statements);
-                if (useStrictDirective_1) {
-                    var nonSimpleParameters = getNonSimpleParameters(node.parameters);
-                    if (ts.length(nonSimpleParameters)) {
-                        ts.forEach(nonSimpleParameters, function (parameter) {
-                            ts.addRelatedInfo(error(parameter, ts.Diagnostics.This_parameter_is_not_allowed_with_use_strict_directive), ts.createDiagnosticForNode(useStrictDirective_1, ts.Diagnostics.use_strict_directive_used_here));
-                        });
-                        var diagnostics_1 = nonSimpleParameters.map(function (parameter, index) { return (index === 0 ? ts.createDiagnosticForNode(parameter, ts.Diagnostics.Non_simple_parameter_declared_here) : ts.createDiagnosticForNode(parameter, ts.Diagnostics.and_here)); });
-                        ts.addRelatedInfo.apply(void 0, __spreadArrays([error(useStrictDirective_1, ts.Diagnostics.use_strict_directive_cannot_be_used_with_non_simple_parameter_list)], diagnostics_1));
-                        return true;
+        function checkImportDeclaration(node) {
+            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_import_declaration_can_only_be_used_in_a_namespace_or_module)) {
+                // If we hit an import declaration in an illegal context, just bail out to avoid cascading errors.
+                return;
+            }
+            if (!checkGrammarDecoratorsAndModifiers(node) && ts.hasEffectiveModifiers(node)) {
+                grammarErrorOnFirstToken(node, ts.Diagnostics.An_import_declaration_cannot_have_modifiers);
+            }
+            if (checkExternalImportOrExportDeclaration(node)) {
+                var importClause = node.importClause;
+                if (importClause && !checkGrammarImportClause(importClause)) {
+                    if (importClause.name) {
+                        checkImportBinding(importClause);
+                    }
+                    if (importClause.namedBindings) {
+                        if (importClause.namedBindings.kind === 263 /* NamespaceImport */) {
+                            checkImportBinding(importClause.namedBindings);
+                            if (moduleKind !== ts.ModuleKind.System && moduleKind < ts.ModuleKind.ES2015 && compilerOptions.esModuleInterop) {
+                                // import * as ns from "foo";
+                                checkExternalEmitHelpers(node, 131072 /* ImportStar */);
+                            }
+                        }
+                        else {
+                            var moduleExisted = resolveExternalModuleName(node, node.moduleSpecifier);
+                            if (moduleExisted) {
+                                ts.forEach(importClause.namedBindings.elements, checkImportBinding);
+                            }
+                        }
                     }
                 }
             }
-            return false;
-        }
-        function checkGrammarFunctionLikeDeclaration(node) {
-            // Prevent cascading error by short-circuit
-            var file = ts.getSourceFileOfNode(node);
-            return checkGrammarDecoratorsAndModifiers(node) || checkGrammarTypeParameterList(node.typeParameters, file) ||
-                checkGrammarParameterList(node.parameters) || checkGrammarArrowFunction(node, file) ||
-                (ts.isFunctionLikeDeclaration(node) && checkGrammarForUseStrictSimpleParameterList(node));
-        }
-        function checkGrammarClassLikeDeclaration(node) {
-            var file = ts.getSourceFileOfNode(node);
-            return checkGrammarClassDeclarationHeritageClauses(node) || checkGrammarTypeParameterList(node.typeParameters, file);
         }
-        function checkGrammarArrowFunction(node, file) {
-            if (!ts.isArrowFunction(node)) {
-                return false;
+        function checkImportEqualsDeclaration(node) {
+            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_import_declaration_can_only_be_used_in_a_namespace_or_module)) {
+                // If we hit an import declaration in an illegal context, just bail out to avoid cascading errors.
+                return;
             }
-            var equalsGreaterThanToken = node.equalsGreaterThanToken;
-            var startLine = ts.getLineAndCharacterOfPosition(file, equalsGreaterThanToken.pos).line;
-            var endLine = ts.getLineAndCharacterOfPosition(file, equalsGreaterThanToken.end).line;
-            return startLine !== endLine && grammarErrorOnNode(equalsGreaterThanToken, ts.Diagnostics.Line_terminator_not_permitted_before_arrow);
-        }
-        function checkGrammarIndexSignatureParameters(node) {
-            var parameter = node.parameters[0];
-            if (node.parameters.length !== 1) {
-                if (parameter) {
-                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_must_have_exactly_one_parameter);
+            checkGrammarDecoratorsAndModifiers(node);
+            if (ts.isInternalModuleImportEqualsDeclaration(node) || checkExternalImportOrExportDeclaration(node)) {
+                checkImportBinding(node);
+                if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                    markExportAsReferenced(node);
+                }
+                if (node.moduleReference.kind !== 272 /* ExternalModuleReference */) {
+                    var target = resolveAlias(getSymbolOfNode(node));
+                    if (target !== unknownSymbol) {
+                        if (target.flags & 111551 /* Value */) {
+                            // Target is a value symbol, check that it is not hidden by a local declaration with the same name
+                            var moduleName = ts.getFirstIdentifier(node.moduleReference);
+                            if (!(resolveEntityName(moduleName, 111551 /* Value */ | 1920 /* Namespace */).flags & 1920 /* Namespace */)) {
+                                error(moduleName, ts.Diagnostics.Module_0_is_hidden_by_a_local_declaration_with_the_same_name, ts.declarationNameToString(moduleName));
+                            }
+                        }
+                        if (target.flags & 788968 /* Type */) {
+                            checkTypeNameIsReserved(node.name, ts.Diagnostics.Import_name_cannot_be_0);
+                        }
+                    }
                 }
                 else {
-                    return grammarErrorOnNode(node, ts.Diagnostics.An_index_signature_must_have_exactly_one_parameter);
+                    if (moduleKind >= ts.ModuleKind.ES2015 && !(node.flags & 8388608 /* Ambient */)) {
+                        // Import equals declaration is deprecated in es6 or above
+                        grammarErrorOnNode(node, ts.Diagnostics.Import_assignment_cannot_be_used_when_targeting_ECMAScript_modules_Consider_using_import_Asterisk_as_ns_from_mod_import_a_from_mod_import_d_from_mod_or_another_module_format_instead);
+                    }
                 }
             }
-            checkGrammarForDisallowedTrailingComma(node.parameters, ts.Diagnostics.An_index_signature_cannot_have_a_trailing_comma);
-            if (parameter.dotDotDotToken) {
-                return grammarErrorOnNode(parameter.dotDotDotToken, ts.Diagnostics.An_index_signature_cannot_have_a_rest_parameter);
-            }
-            if (ts.hasModifiers(parameter)) {
-                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_cannot_have_an_accessibility_modifier);
-            }
-            if (parameter.questionToken) {
-                return grammarErrorOnNode(parameter.questionToken, ts.Diagnostics.An_index_signature_parameter_cannot_have_a_question_mark);
+        }
+        function checkExportDeclaration(node) {
+            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_export_declaration_can_only_be_used_in_a_module)) {
+                // If we hit an export in an illegal context, just bail out to avoid cascading errors.
+                return;
             }
-            if (parameter.initializer) {
-                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_cannot_have_an_initializer);
+            if (!checkGrammarDecoratorsAndModifiers(node) && ts.hasEffectiveModifiers(node)) {
+                grammarErrorOnFirstToken(node, ts.Diagnostics.An_export_declaration_cannot_have_modifiers);
             }
-            if (!parameter.type) {
-                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_must_have_a_type_annotation);
+            if (node.moduleSpecifier && node.exportClause && ts.isNamedExports(node.exportClause) && ts.length(node.exportClause.elements) && languageVersion === 0 /* ES3 */) {
+                checkExternalEmitHelpers(node, 4194304 /* CreateBinding */);
             }
-            if (parameter.type.kind !== 143 /* StringKeyword */ && parameter.type.kind !== 140 /* NumberKeyword */) {
-                var type = getTypeFromTypeNode(parameter.type);
-                if (type.flags & 4 /* String */ || type.flags & 8 /* Number */) {
-                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_type_cannot_be_a_type_alias_Consider_writing_0_Colon_1_Colon_2_instead, ts.getTextOfNode(parameter.name), typeToString(type), typeToString(node.type ? getTypeFromTypeNode(node.type) : anyType));
+            checkGrammarExportDeclaration(node);
+            if (!node.moduleSpecifier || checkExternalImportOrExportDeclaration(node)) {
+                if (node.exportClause && !ts.isNamespaceExport(node.exportClause)) {
+                    // export { x, y }
+                    // export { x, y } from "foo"
+                    ts.forEach(node.exportClause.elements, checkExportSpecifier);
+                    var inAmbientExternalModule = node.parent.kind === 257 /* ModuleBlock */ && ts.isAmbientModule(node.parent.parent);
+                    var inAmbientNamespaceDeclaration = !inAmbientExternalModule && node.parent.kind === 257 /* ModuleBlock */ &&
+                        !node.moduleSpecifier && node.flags & 8388608 /* Ambient */;
+                    if (node.parent.kind !== 297 /* SourceFile */ && !inAmbientExternalModule && !inAmbientNamespaceDeclaration) {
+                        error(node, ts.Diagnostics.Export_declarations_are_not_permitted_in_a_namespace);
+                    }
                 }
-                if (type.flags & 1048576 /* Union */ && allTypesAssignableToKind(type, 384 /* StringOrNumberLiteral */, /*strict*/ true)) {
-                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_type_cannot_be_a_union_type_Consider_using_a_mapped_object_type_instead);
+                else {
+                    // export * from "foo"
+                    // export * as ns from "foo";
+                    var moduleSymbol = resolveExternalModuleName(node, node.moduleSpecifier);
+                    if (moduleSymbol && hasExportAssignmentSymbol(moduleSymbol)) {
+                        error(node.moduleSpecifier, ts.Diagnostics.Module_0_uses_export_and_cannot_be_used_with_export_Asterisk, symbolToString(moduleSymbol));
+                    }
+                    else if (node.exportClause) {
+                        checkAliasSymbol(node.exportClause);
+                    }
+                    if (moduleKind !== ts.ModuleKind.System && moduleKind < ts.ModuleKind.ES2015) {
+                        if (node.exportClause) {
+                            // export * as ns from "foo";
+                            // For ES2015 modules, we emit it as a pair of `import * as a_1 ...; export { a_1 as ns }` and don't need the helper.
+                            // We only use the helper here when in esModuleInterop
+                            if (compilerOptions.esModuleInterop) {
+                                checkExternalEmitHelpers(node, 131072 /* ImportStar */);
+                            }
+                        }
+                        else {
+                            // export * from "foo"
+                            checkExternalEmitHelpers(node, 65536 /* ExportStar */);
+                        }
+                    }
                 }
-                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_type_must_be_either_string_or_number);
-            }
-            if (!node.type) {
-                return grammarErrorOnNode(node, ts.Diagnostics.An_index_signature_must_have_a_type_annotation);
             }
-            return false;
         }
-        function checkGrammarIndexSignature(node) {
-            // Prevent cascading error by short-circuit
-            return checkGrammarDecoratorsAndModifiers(node) || checkGrammarIndexSignatureParameters(node);
+        function checkGrammarExportDeclaration(node) {
+            var _a;
+            var isTypeOnlyExportStar = node.isTypeOnly && ((_a = node.exportClause) === null || _a === void 0 ? void 0 : _a.kind) !== 268 /* NamedExports */;
+            if (isTypeOnlyExportStar) {
+                grammarErrorOnNode(node, ts.Diagnostics.Only_named_exports_may_use_export_type);
+            }
+            return !isTypeOnlyExportStar;
         }
-        function checkGrammarForAtLeastOneTypeArgument(node, typeArguments) {
-            if (typeArguments && typeArguments.length === 0) {
-                var sourceFile = ts.getSourceFileOfNode(node);
-                var start = typeArguments.pos - "<".length;
-                var end = ts.skipTrivia(sourceFile.text, typeArguments.end) + ">".length;
-                return grammarErrorAtPos(sourceFile, start, end - start, ts.Diagnostics.Type_argument_list_cannot_be_empty);
+        function checkGrammarModuleElementContext(node, errorMessage) {
+            var isInAppropriateContext = node.parent.kind === 297 /* SourceFile */ || node.parent.kind === 257 /* ModuleBlock */ || node.parent.kind === 256 /* ModuleDeclaration */;
+            if (!isInAppropriateContext) {
+                grammarErrorOnFirstToken(node, errorMessage);
             }
-            return false;
+            return !isInAppropriateContext;
         }
-        function checkGrammarTypeArguments(node, typeArguments) {
-            return checkGrammarForDisallowedTrailingComma(typeArguments) ||
-                checkGrammarForAtLeastOneTypeArgument(node, typeArguments);
+        function importClauseContainsReferencedImport(importClause) {
+            return ts.forEachImportClauseDeclaration(importClause, function (declaration) {
+                return !!getSymbolOfNode(declaration).isReferenced;
+            });
         }
-        function checkGrammarTaggedTemplateChain(node) {
-            if (node.questionDotToken || node.flags & 32 /* OptionalChain */) {
-                return grammarErrorOnNode(node.template, ts.Diagnostics.Tagged_template_expressions_are_not_permitted_in_an_optional_chain);
-            }
-            return false;
+        function importClauseContainsConstEnumUsedAsValue(importClause) {
+            return ts.forEachImportClauseDeclaration(importClause, function (declaration) {
+                return !!getSymbolLinks(getSymbolOfNode(declaration)).constEnumReferenced;
+            });
         }
-        function checkGrammarForOmittedArgument(args) {
-            if (args) {
-                for (var _i = 0, args_4 = args; _i < args_4.length; _i++) {
-                    var arg = args_4[_i];
-                    if (arg.kind === 215 /* OmittedExpression */) {
-                        return grammarErrorAtPos(arg, arg.pos, 0, ts.Diagnostics.Argument_expression_expected);
-                    }
+        function checkImportsForTypeOnlyConversion(sourceFile) {
+            for (var _i = 0, _a = sourceFile.statements; _i < _a.length; _i++) {
+                var statement = _a[_i];
+                if (ts.isImportDeclaration(statement) &&
+                    statement.importClause &&
+                    !statement.importClause.isTypeOnly &&
+                    importClauseContainsReferencedImport(statement.importClause) &&
+                    !isReferencedAliasDeclaration(statement.importClause, /*checkChildren*/ true) &&
+                    !importClauseContainsConstEnumUsedAsValue(statement.importClause)) {
+                    error(statement, ts.Diagnostics.This_import_is_never_used_as_a_value_and_must_use_import_type_because_the_importsNotUsedAsValues_is_set_to_error);
                 }
             }
-            return false;
-        }
-        function checkGrammarArguments(args) {
-            return checkGrammarForOmittedArgument(args);
         }
-        function checkGrammarHeritageClause(node) {
-            var types = node.types;
-            if (checkGrammarForDisallowedTrailingComma(types)) {
-                return true;
-            }
-            if (types && types.length === 0) {
-                var listType = ts.tokenToString(node.token);
-                return grammarErrorAtPos(node, types.pos, 0, ts.Diagnostics._0_list_cannot_be_empty, listType);
+        function checkExportSpecifier(node) {
+            checkAliasSymbol(node);
+            if (ts.getEmitDeclarations(compilerOptions)) {
+                collectLinkedAliases(node.propertyName || node.name, /*setVisibility*/ true);
             }
-            return ts.some(types, checkGrammarExpressionWithTypeArguments);
-        }
-        function checkGrammarExpressionWithTypeArguments(node) {
-            return checkGrammarTypeArguments(node, node.typeArguments);
-        }
-        function checkGrammarClassDeclarationHeritageClauses(node) {
-            var seenExtendsClause = false;
-            var seenImplementsClause = false;
-            if (!checkGrammarDecoratorsAndModifiers(node) && node.heritageClauses) {
-                for (var _i = 0, _a = node.heritageClauses; _i < _a.length; _i++) {
-                    var heritageClause = _a[_i];
-                    if (heritageClause.token === 90 /* ExtendsKeyword */) {
-                        if (seenExtendsClause) {
-                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.extends_clause_already_seen);
-                        }
-                        if (seenImplementsClause) {
-                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.extends_clause_must_precede_implements_clause);
-                        }
-                        if (heritageClause.types.length > 1) {
-                            return grammarErrorOnFirstToken(heritageClause.types[1], ts.Diagnostics.Classes_can_only_extend_a_single_class);
-                        }
-                        seenExtendsClause = true;
-                    }
-                    else {
-                        ts.Debug.assert(heritageClause.token === 113 /* ImplementsKeyword */);
-                        if (seenImplementsClause) {
-                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.implements_clause_already_seen);
-                        }
-                        seenImplementsClause = true;
+            if (!node.parent.parent.moduleSpecifier) {
+                var exportedName = node.propertyName || node.name;
+                // find immediate value referenced by exported name (SymbolFlags.Alias is set so we don't chase down aliases)
+                var symbol = resolveName(exportedName, exportedName.escapedText, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */, 
+                /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true);
+                if (symbol && (symbol === undefinedSymbol || symbol === globalThisSymbol || isGlobalSourceFile(getDeclarationContainer(symbol.declarations[0])))) {
+                    error(exportedName, ts.Diagnostics.Cannot_export_0_Only_local_declarations_can_be_exported_from_a_module, ts.idText(exportedName));
+                }
+                else {
+                    markExportAsReferenced(node);
+                    var target = symbol && (symbol.flags & 2097152 /* Alias */ ? resolveAlias(symbol) : symbol);
+                    if (!target || target === unknownSymbol || target.flags & 111551 /* Value */) {
+                        checkExpressionCached(node.propertyName || node.name);
                     }
-                    // Grammar checking heritageClause inside class declaration
-                    checkGrammarHeritageClause(heritageClause);
                 }
             }
-        }
-        function checkGrammarInterfaceDeclaration(node) {
-            var seenExtendsClause = false;
-            if (node.heritageClauses) {
-                for (var _i = 0, _a = node.heritageClauses; _i < _a.length; _i++) {
-                    var heritageClause = _a[_i];
-                    if (heritageClause.token === 90 /* ExtendsKeyword */) {
-                        if (seenExtendsClause) {
-                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.extends_clause_already_seen);
-                        }
-                        seenExtendsClause = true;
-                    }
-                    else {
-                        ts.Debug.assert(heritageClause.token === 113 /* ImplementsKeyword */);
-                        return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.Interface_declaration_cannot_have_implements_clause);
-                    }
-                    // Grammar checking heritageClause inside class declaration
-                    checkGrammarHeritageClause(heritageClause);
+            else {
+                if (compilerOptions.esModuleInterop &&
+                    moduleKind !== ts.ModuleKind.System &&
+                    moduleKind < ts.ModuleKind.ES2015 &&
+                    ts.idText(node.propertyName || node.name) === "default") {
+                    checkExternalEmitHelpers(node, 262144 /* ImportDefault */);
                 }
             }
-            return false;
         }
-        function checkGrammarComputedPropertyName(node) {
-            // If node is not a computedPropertyName, just skip the grammar checking
-            if (node.kind !== 154 /* ComputedPropertyName */) {
-                return false;
-            }
-            var computedPropertyName = node;
-            if (computedPropertyName.expression.kind === 209 /* BinaryExpression */ && computedPropertyName.expression.operatorToken.kind === 27 /* CommaToken */) {
-                return grammarErrorOnNode(computedPropertyName.expression, ts.Diagnostics.A_comma_expression_is_not_allowed_in_a_computed_property_name);
+        function checkExportAssignment(node) {
+            if (checkGrammarModuleElementContext(node, ts.Diagnostics.An_export_assignment_can_only_be_used_in_a_module)) {
+                // If we hit an export assignment in an illegal context, just bail out to avoid cascading errors.
+                return;
             }
-            return false;
-        }
-        function checkGrammarForGenerator(node) {
-            if (node.asteriskToken) {
-                ts.Debug.assert(node.kind === 244 /* FunctionDeclaration */ ||
-                    node.kind === 201 /* FunctionExpression */ ||
-                    node.kind === 161 /* MethodDeclaration */);
-                if (node.flags & 8388608 /* Ambient */) {
-                    return grammarErrorOnNode(node.asteriskToken, ts.Diagnostics.Generators_are_not_allowed_in_an_ambient_context);
+            var container = node.parent.kind === 297 /* SourceFile */ ? node.parent : node.parent.parent;
+            if (container.kind === 256 /* ModuleDeclaration */ && !ts.isAmbientModule(container)) {
+                if (node.isExportEquals) {
+                    error(node, ts.Diagnostics.An_export_assignment_cannot_be_used_in_a_namespace);
                 }
-                if (!node.body) {
-                    return grammarErrorOnNode(node.asteriskToken, ts.Diagnostics.An_overload_signature_cannot_be_declared_as_a_generator);
+                else {
+                    error(node, ts.Diagnostics.A_default_export_can_only_be_used_in_an_ECMAScript_style_module);
                 }
+                return;
             }
-        }
-        function checkGrammarForInvalidQuestionMark(questionToken, message) {
-            return !!questionToken && grammarErrorOnNode(questionToken, message);
-        }
-        function checkGrammarForInvalidExclamationToken(exclamationToken, message) {
-            return !!exclamationToken && grammarErrorOnNode(exclamationToken, message);
-        }
-        function checkGrammarObjectLiteralExpression(node, inDestructuring) {
-            var seen = ts.createUnderscoreEscapedMap();
-            for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
-                var prop = _a[_i];
-                if (prop.kind === 283 /* SpreadAssignment */) {
-                    if (inDestructuring) {
-                        // a rest property cannot be destructured any further
-                        var expression = ts.skipParentheses(prop.expression);
-                        if (ts.isArrayLiteralExpression(expression) || ts.isObjectLiteralExpression(expression)) {
-                            return grammarErrorOnNode(prop.expression, ts.Diagnostics.A_rest_element_cannot_contain_a_binding_pattern);
-                        }
+            // Grammar checking
+            if (!checkGrammarDecoratorsAndModifiers(node) && ts.hasEffectiveModifiers(node)) {
+                grammarErrorOnFirstToken(node, ts.Diagnostics.An_export_assignment_cannot_have_modifiers);
+            }
+            if (node.expression.kind === 78 /* Identifier */) {
+                var id = node.expression;
+                var sym = resolveEntityName(id, 67108863 /* All */, /*ignoreErrors*/ true, /*dontResolveAlias*/ true, node);
+                if (sym) {
+                    markAliasReferenced(sym, id);
+                    // If not a value, we're interpreting the identifier as a type export, along the lines of (`export { Id as default }`)
+                    var target = sym.flags & 2097152 /* Alias */ ? resolveAlias(sym) : sym;
+                    if (target === unknownSymbol || target.flags & 111551 /* Value */) {
+                        // However if it is a value, we need to check it's being used correctly
+                        checkExpressionCached(node.expression);
                     }
-                    continue;
                 }
-                var name = prop.name;
-                if (name.kind === 154 /* ComputedPropertyName */) {
-                    // If the name is not a ComputedPropertyName, the grammar checking will skip it
-                    checkGrammarComputedPropertyName(name);
+                else {
+                    checkExpressionCached(node.expression); // doesn't resolve, check as expression to mark as error
                 }
-                if (prop.kind === 282 /* ShorthandPropertyAssignment */ && !inDestructuring && prop.objectAssignmentInitializer) {
-                    // having objectAssignmentInitializer is only valid in ObjectAssignmentPattern
-                    // outside of destructuring it is a syntax error
-                    return grammarErrorOnNode(prop.equalsToken, ts.Diagnostics.can_only_be_used_in_an_object_literal_property_inside_a_destructuring_assignment);
+                if (ts.getEmitDeclarations(compilerOptions)) {
+                    collectLinkedAliases(node.expression, /*setVisibility*/ true);
                 }
-                if (name.kind === 76 /* PrivateIdentifier */) {
-                    return grammarErrorOnNode(name, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
+            }
+            else {
+                checkExpressionCached(node.expression);
+            }
+            checkExternalModuleExports(container);
+            if ((node.flags & 8388608 /* Ambient */) && !ts.isEntityNameExpression(node.expression)) {
+                grammarErrorOnNode(node.expression, ts.Diagnostics.The_expression_of_an_export_assignment_must_be_an_identifier_or_qualified_name_in_an_ambient_context);
+            }
+            if (node.isExportEquals && !(node.flags & 8388608 /* Ambient */)) {
+                if (moduleKind >= ts.ModuleKind.ES2015) {
+                    // export assignment is not supported in es6 modules
+                    grammarErrorOnNode(node, ts.Diagnostics.Export_assignment_cannot_be_used_when_targeting_ECMAScript_modules_Consider_using_export_default_or_another_module_format_instead);
                 }
-                // Modifiers are never allowed on properties except for 'async' on a method declaration
-                if (prop.modifiers) {
-                    // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
-                    for (var _b = 0, _c = prop.modifiers; _b < _c.length; _b++) { // TODO: GH#19955
-                        var mod = _c[_b];
-                        if (mod.kind !== 126 /* AsyncKeyword */ || prop.kind !== 161 /* MethodDeclaration */) {
-                            grammarErrorOnNode(mod, ts.Diagnostics._0_modifier_cannot_be_used_here, ts.getTextOfNode(mod));
-                        }
+                else if (moduleKind === ts.ModuleKind.System) {
+                    // system modules does not support export assignment
+                    grammarErrorOnNode(node, ts.Diagnostics.Export_assignment_is_not_supported_when_module_flag_is_system);
+                }
+            }
+        }
+        function hasExportedMembers(moduleSymbol) {
+            return ts.forEachEntry(moduleSymbol.exports, function (_, id) { return id !== "export="; });
+        }
+        function checkExternalModuleExports(node) {
+            var moduleSymbol = getSymbolOfNode(node);
+            var links = getSymbolLinks(moduleSymbol);
+            if (!links.exportsChecked) {
+                var exportEqualsSymbol = moduleSymbol.exports.get("export=");
+                if (exportEqualsSymbol && hasExportedMembers(moduleSymbol)) {
+                    var declaration = getDeclarationOfAliasSymbol(exportEqualsSymbol) || exportEqualsSymbol.valueDeclaration;
+                    if (!isTopLevelInExternalModuleAugmentation(declaration) && !ts.isInJSFile(declaration)) {
+                        error(declaration, ts.Diagnostics.An_export_assignment_cannot_be_used_in_a_module_with_other_exported_elements);
                     }
                 }
-                // ECMA-262 11.1.5 Object Initializer
-                // If previous is not undefined then throw a SyntaxError exception if any of the following conditions are true
-                // a.This production is contained in strict code and IsDataDescriptor(previous) is true and
-                // IsDataDescriptor(propId.descriptor) is true.
-                //    b.IsDataDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true.
-                //    c.IsAccessorDescriptor(previous) is true and IsDataDescriptor(propId.descriptor) is true.
-                //    d.IsAccessorDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true
-                // and either both previous and propId.descriptor have[[Get]] fields or both previous and propId.descriptor have[[Set]] fields
-                var currentKind = void 0;
-                switch (prop.kind) {
-                    case 282 /* ShorthandPropertyAssignment */:
-                        checkGrammarForInvalidExclamationToken(prop.exclamationToken, ts.Diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context);
-                    // falls through
-                    case 281 /* PropertyAssignment */:
-                        // Grammar checking for computedPropertyName and shorthandPropertyAssignment
-                        checkGrammarForInvalidQuestionMark(prop.questionToken, ts.Diagnostics.An_object_member_cannot_be_declared_optional);
-                        if (name.kind === 8 /* NumericLiteral */) {
-                            checkGrammarNumericLiteral(name);
+                // Checks for export * conflicts
+                var exports_2 = getExportsOfModule(moduleSymbol);
+                if (exports_2) {
+                    exports_2.forEach(function (_a, id) {
+                        var declarations = _a.declarations, flags = _a.flags;
+                        if (id === "__export") {
+                            return;
                         }
-                        currentKind = 4 /* PropertyAssignment */;
-                        break;
-                    case 161 /* MethodDeclaration */:
-                        currentKind = 8 /* Method */;
-                        break;
-                    case 163 /* GetAccessor */:
-                        currentKind = 1 /* GetAccessor */;
-                        break;
-                    case 164 /* SetAccessor */:
-                        currentKind = 2 /* SetAccessor */;
-                        break;
-                    default:
-                        throw ts.Debug.assertNever(prop, "Unexpected syntax kind:" + prop.kind);
-                }
-                if (!inDestructuring) {
-                    var effectiveName = ts.getPropertyNameForPropertyNameNode(name);
-                    if (effectiveName === undefined) {
-                        continue;
-                    }
-                    var existingKind = seen.get(effectiveName);
-                    if (!existingKind) {
-                        seen.set(effectiveName, currentKind);
-                    }
-                    else {
-                        if ((currentKind & 12 /* PropertyAssignmentOrMethod */) && (existingKind & 12 /* PropertyAssignmentOrMethod */)) {
-                            grammarErrorOnNode(name, ts.Diagnostics.Duplicate_identifier_0, ts.getTextOfNode(name));
+                        // ECMA262: 15.2.1.1 It is a Syntax Error if the ExportedNames of ModuleItemList contains any duplicate entries.
+                        // (TS Exceptions: namespaces, function overloads, enums, and interfaces)
+                        if (flags & (1920 /* Namespace */ | 64 /* Interface */ | 384 /* Enum */)) {
+                            return;
                         }
-                        else if ((currentKind & 3 /* GetOrSetAccessor */) && (existingKind & 3 /* GetOrSetAccessor */)) {
-                            if (existingKind !== 3 /* GetOrSetAccessor */ && currentKind !== existingKind) {
-                                seen.set(effectiveName, currentKind | existingKind);
-                            }
-                            else {
-                                return grammarErrorOnNode(name, ts.Diagnostics.An_object_literal_cannot_have_multiple_get_Slashset_accessors_with_the_same_name);
-                            }
+                        var exportedDeclarationsCount = ts.countWhere(declarations, isNotOverloadAndNotAccessor);
+                        if (flags & 524288 /* TypeAlias */ && exportedDeclarationsCount <= 2) {
+                            // it is legal to merge type alias with other values
+                            // so count should be either 1 (just type alias) or 2 (type alias + merged value)
+                            return;
                         }
-                        else {
-                            return grammarErrorOnNode(name, ts.Diagnostics.An_object_literal_cannot_have_property_and_accessor_with_the_same_name);
+                        if (exportedDeclarationsCount > 1) {
+                            for (var _i = 0, declarations_9 = declarations; _i < declarations_9.length; _i++) {
+                                var declaration = declarations_9[_i];
+                                if (isNotOverload(declaration)) {
+                                    diagnostics.add(ts.createDiagnosticForNode(declaration, ts.Diagnostics.Cannot_redeclare_exported_variable_0, ts.unescapeLeadingUnderscores(id)));
+                                }
+                            }
                         }
-                    }
-                }
-            }
-        }
-        function checkGrammarJsxElement(node) {
-            checkGrammarTypeArguments(node, node.typeArguments);
-            var seen = ts.createUnderscoreEscapedMap();
-            for (var _i = 0, _a = node.attributes.properties; _i < _a.length; _i++) {
-                var attr = _a[_i];
-                if (attr.kind === 275 /* JsxSpreadAttribute */) {
-                    continue;
-                }
-                var name = attr.name, initializer = attr.initializer;
-                if (!seen.get(name.escapedText)) {
-                    seen.set(name.escapedText, true);
-                }
-                else {
-                    return grammarErrorOnNode(name, ts.Diagnostics.JSX_elements_cannot_have_multiple_attributes_with_the_same_name);
-                }
-                if (initializer && initializer.kind === 276 /* JsxExpression */ && !initializer.expression) {
-                    return grammarErrorOnNode(initializer, ts.Diagnostics.JSX_attributes_must_only_be_assigned_a_non_empty_expression);
+                    });
                 }
+                links.exportsChecked = true;
             }
         }
-        function checkGrammarJsxExpression(node) {
-            if (node.expression && ts.isCommaSequence(node.expression)) {
-                return grammarErrorOnNode(node.expression, ts.Diagnostics.JSX_expressions_may_not_use_the_comma_operator_Did_you_mean_to_write_an_array);
+        function checkSourceElement(node) {
+            if (node) {
+                var saveCurrentNode = currentNode;
+                currentNode = node;
+                instantiationCount = 0;
+                checkSourceElementWorker(node);
+                currentNode = saveCurrentNode;
             }
         }
-        function checkGrammarForInOrForOfStatement(forInOrOfStatement) {
-            if (checkGrammarStatementInAmbientContext(forInOrOfStatement)) {
-                return true;
+        function checkSourceElementWorker(node) {
+            if (ts.isInJSFile(node)) {
+                ts.forEach(node.jsDoc, function (_a) {
+                    var tags = _a.tags;
+                    return ts.forEach(tags, checkSourceElement);
+                });
             }
-            if (forInOrOfStatement.kind === 232 /* ForOfStatement */ && forInOrOfStatement.awaitModifier) {
-                if ((forInOrOfStatement.flags & 32768 /* AwaitContext */) === 0 /* None */) {
-                    // use of 'for-await-of' in non-async function
-                    var sourceFile = ts.getSourceFileOfNode(forInOrOfStatement);
-                    if (!hasParseDiagnostics(sourceFile)) {
-                        var diagnostic = ts.createDiagnosticForNode(forInOrOfStatement.awaitModifier, ts.Diagnostics.A_for_await_of_statement_is_only_allowed_within_an_async_function_or_async_generator);
-                        var func = ts.getContainingFunction(forInOrOfStatement);
-                        if (func && func.kind !== 162 /* Constructor */) {
-                            ts.Debug.assert((ts.getFunctionFlags(func) & 2 /* Async */) === 0, "Enclosing function should never be an async function.");
-                            var relatedInfo = ts.createDiagnosticForNode(func, ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async);
-                            ts.addRelatedInfo(diagnostic, relatedInfo);
-                        }
-                        diagnostics.add(diagnostic);
-                        return true;
-                    }
-                    return false;
+            var kind = node.kind;
+            if (cancellationToken) {
+                // Only bother checking on a few construct kinds.  We don't want to be excessively
+                // hitting the cancellation token on every node we check.
+                switch (kind) {
+                    case 256 /* ModuleDeclaration */:
+                    case 252 /* ClassDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 251 /* FunctionDeclaration */:
+                        cancellationToken.throwIfCancellationRequested();
                 }
             }
-            if (forInOrOfStatement.initializer.kind === 243 /* VariableDeclarationList */) {
-                var variableList = forInOrOfStatement.initializer;
-                if (!checkGrammarVariableDeclarationList(variableList)) {
-                    var declarations = variableList.declarations;
-                    // declarations.length can be zero if there is an error in variable declaration in for-of or for-in
-                    // See http://www.ecma-international.org/ecma-262/6.0/#sec-for-in-and-for-of-statements for details
-                    // For example:
-                    //      var let = 10;
-                    //      for (let of [1,2,3]) {} // this is invalid ES6 syntax
-                    //      for (let in [1,2,3]) {} // this is invalid ES6 syntax
-                    // We will then want to skip on grammar checking on variableList declaration
-                    if (!declarations.length) {
-                        return false;
-                    }
-                    if (declarations.length > 1) {
-                        var diagnostic = forInOrOfStatement.kind === 231 /* ForInStatement */
-                            ? ts.Diagnostics.Only_a_single_variable_declaration_is_allowed_in_a_for_in_statement
-                            : ts.Diagnostics.Only_a_single_variable_declaration_is_allowed_in_a_for_of_statement;
-                        return grammarErrorOnFirstToken(variableList.declarations[1], diagnostic);
-                    }
-                    var firstDeclaration = declarations[0];
-                    if (firstDeclaration.initializer) {
-                        var diagnostic = forInOrOfStatement.kind === 231 /* ForInStatement */
-                            ? ts.Diagnostics.The_variable_declaration_of_a_for_in_statement_cannot_have_an_initializer
-                            : ts.Diagnostics.The_variable_declaration_of_a_for_of_statement_cannot_have_an_initializer;
-                        return grammarErrorOnNode(firstDeclaration.name, diagnostic);
-                    }
-                    if (firstDeclaration.type) {
-                        var diagnostic = forInOrOfStatement.kind === 231 /* ForInStatement */
-                            ? ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_cannot_use_a_type_annotation
-                            : ts.Diagnostics.The_left_hand_side_of_a_for_of_statement_cannot_use_a_type_annotation;
-                        return grammarErrorOnNode(firstDeclaration, diagnostic);
-                    }
-                }
+            if (kind >= 232 /* FirstStatement */ && kind <= 248 /* LastStatement */ && node.flowNode && !isReachableFlowNode(node.flowNode)) {
+                errorOrSuggestion(compilerOptions.allowUnreachableCode === false, node, ts.Diagnostics.Unreachable_code_detected);
+            }
+            switch (kind) {
+                case 159 /* TypeParameter */:
+                    return checkTypeParameter(node);
+                case 160 /* Parameter */:
+                    return checkParameter(node);
+                case 163 /* PropertyDeclaration */:
+                    return checkPropertyDeclaration(node);
+                case 162 /* PropertySignature */:
+                    return checkPropertySignature(node);
+                case 174 /* FunctionType */:
+                case 175 /* ConstructorType */:
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 171 /* IndexSignature */:
+                    return checkSignatureDeclaration(node);
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    return checkMethodDeclaration(node);
+                case 166 /* Constructor */:
+                    return checkConstructorDeclaration(node);
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    return checkAccessorDeclaration(node);
+                case 173 /* TypeReference */:
+                    return checkTypeReferenceNode(node);
+                case 172 /* TypePredicate */:
+                    return checkTypePredicate(node);
+                case 176 /* TypeQuery */:
+                    return checkTypeQuery(node);
+                case 177 /* TypeLiteral */:
+                    return checkTypeLiteral(node);
+                case 178 /* ArrayType */:
+                    return checkArrayType(node);
+                case 179 /* TupleType */:
+                    return checkTupleType(node);
+                case 182 /* UnionType */:
+                case 183 /* IntersectionType */:
+                    return checkUnionOrIntersectionType(node);
+                case 186 /* ParenthesizedType */:
+                case 180 /* OptionalType */:
+                case 181 /* RestType */:
+                    return checkSourceElement(node.type);
+                case 187 /* ThisType */:
+                    return checkThisType(node);
+                case 188 /* TypeOperator */:
+                    return checkTypeOperator(node);
+                case 184 /* ConditionalType */:
+                    return checkConditionalType(node);
+                case 185 /* InferType */:
+                    return checkInferType(node);
+                case 193 /* TemplateLiteralType */:
+                    return checkTemplateLiteralType(node);
+                case 195 /* ImportType */:
+                    return checkImportType(node);
+                case 192 /* NamedTupleMember */:
+                    return checkNamedTupleMember(node);
+                case 315 /* JSDocAugmentsTag */:
+                    return checkJSDocAugmentsTag(node);
+                case 316 /* JSDocImplementsTag */:
+                    return checkJSDocImplementsTag(node);
+                case 331 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 325 /* JSDocEnumTag */:
+                    return checkJSDocTypeAliasTag(node);
+                case 330 /* JSDocTemplateTag */:
+                    return checkJSDocTemplateTag(node);
+                case 329 /* JSDocTypeTag */:
+                    return checkJSDocTypeTag(node);
+                case 326 /* JSDocParameterTag */:
+                    return checkJSDocParameterTag(node);
+                case 333 /* JSDocPropertyTag */:
+                    return checkJSDocPropertyTag(node);
+                case 308 /* JSDocFunctionType */:
+                    checkJSDocFunctionType(node);
+                // falls through
+                case 306 /* JSDocNonNullableType */:
+                case 305 /* JSDocNullableType */:
+                case 303 /* JSDocAllType */:
+                case 304 /* JSDocUnknownType */:
+                case 312 /* JSDocTypeLiteral */:
+                    checkJSDocTypeIsInJsFile(node);
+                    ts.forEachChild(node, checkSourceElement);
+                    return;
+                case 309 /* JSDocVariadicType */:
+                    checkJSDocVariadicType(node);
+                    return;
+                case 301 /* JSDocTypeExpression */:
+                    return checkSourceElement(node.type);
+                case 189 /* IndexedAccessType */:
+                    return checkIndexedAccessType(node);
+                case 190 /* MappedType */:
+                    return checkMappedType(node);
+                case 251 /* FunctionDeclaration */:
+                    return checkFunctionDeclaration(node);
+                case 230 /* Block */:
+                case 257 /* ModuleBlock */:
+                    return checkBlock(node);
+                case 232 /* VariableStatement */:
+                    return checkVariableStatement(node);
+                case 233 /* ExpressionStatement */:
+                    return checkExpressionStatement(node);
+                case 234 /* IfStatement */:
+                    return checkIfStatement(node);
+                case 235 /* DoStatement */:
+                    return checkDoStatement(node);
+                case 236 /* WhileStatement */:
+                    return checkWhileStatement(node);
+                case 237 /* ForStatement */:
+                    return checkForStatement(node);
+                case 238 /* ForInStatement */:
+                    return checkForInStatement(node);
+                case 239 /* ForOfStatement */:
+                    return checkForOfStatement(node);
+                case 240 /* ContinueStatement */:
+                case 241 /* BreakStatement */:
+                    return checkBreakOrContinueStatement(node);
+                case 242 /* ReturnStatement */:
+                    return checkReturnStatement(node);
+                case 243 /* WithStatement */:
+                    return checkWithStatement(node);
+                case 244 /* SwitchStatement */:
+                    return checkSwitchStatement(node);
+                case 245 /* LabeledStatement */:
+                    return checkLabeledStatement(node);
+                case 246 /* ThrowStatement */:
+                    return checkThrowStatement(node);
+                case 247 /* TryStatement */:
+                    return checkTryStatement(node);
+                case 249 /* VariableDeclaration */:
+                    return checkVariableDeclaration(node);
+                case 198 /* BindingElement */:
+                    return checkBindingElement(node);
+                case 252 /* ClassDeclaration */:
+                    return checkClassDeclaration(node);
+                case 253 /* InterfaceDeclaration */:
+                    return checkInterfaceDeclaration(node);
+                case 254 /* TypeAliasDeclaration */:
+                    return checkTypeAliasDeclaration(node);
+                case 255 /* EnumDeclaration */:
+                    return checkEnumDeclaration(node);
+                case 256 /* ModuleDeclaration */:
+                    return checkModuleDeclaration(node);
+                case 261 /* ImportDeclaration */:
+                    return checkImportDeclaration(node);
+                case 260 /* ImportEqualsDeclaration */:
+                    return checkImportEqualsDeclaration(node);
+                case 267 /* ExportDeclaration */:
+                    return checkExportDeclaration(node);
+                case 266 /* ExportAssignment */:
+                    return checkExportAssignment(node);
+                case 231 /* EmptyStatement */:
+                case 248 /* DebuggerStatement */:
+                    checkGrammarStatementInAmbientContext(node);
+                    return;
+                case 271 /* MissingDeclaration */:
+                    return checkMissingDeclaration(node);
             }
-            return false;
         }
-        function checkGrammarAccessor(accessor) {
-            if (!(accessor.flags & 8388608 /* Ambient */)) {
-                if (languageVersion < 1 /* ES5 */) {
-                    return grammarErrorOnNode(accessor.name, ts.Diagnostics.Accessors_are_only_available_when_targeting_ECMAScript_5_and_higher);
-                }
-                if (accessor.body === undefined && !ts.hasModifier(accessor, 128 /* Abstract */)) {
-                    return grammarErrorAtPos(accessor, accessor.end - 1, ";".length, ts.Diagnostics._0_expected, "{");
+        function checkJSDocTypeIsInJsFile(node) {
+            if (!ts.isInJSFile(node)) {
+                grammarErrorOnNode(node, ts.Diagnostics.JSDoc_types_can_only_be_used_inside_documentation_comments);
+            }
+        }
+        function checkJSDocVariadicType(node) {
+            checkJSDocTypeIsInJsFile(node);
+            checkSourceElement(node.type);
+            // Only legal location is in the *last* parameter tag or last parameter of a JSDoc function.
+            var parent = node.parent;
+            if (ts.isParameter(parent) && ts.isJSDocFunctionType(parent.parent)) {
+                if (ts.last(parent.parent.parameters) !== parent) {
+                    error(node, ts.Diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list);
                 }
+                return;
             }
-            if (accessor.body && ts.hasModifier(accessor, 128 /* Abstract */)) {
-                return grammarErrorOnNode(accessor, ts.Diagnostics.An_abstract_accessor_cannot_have_an_implementation);
+            if (!ts.isJSDocTypeExpression(parent)) {
+                error(node, ts.Diagnostics.JSDoc_may_only_appear_in_the_last_parameter_of_a_signature);
             }
-            if (accessor.typeParameters) {
-                return grammarErrorOnNode(accessor.name, ts.Diagnostics.An_accessor_cannot_have_type_parameters);
+            var paramTag = node.parent.parent;
+            if (!ts.isJSDocParameterTag(paramTag)) {
+                error(node, ts.Diagnostics.JSDoc_may_only_appear_in_the_last_parameter_of_a_signature);
+                return;
             }
-            if (!doesAccessorHaveCorrectParameterCount(accessor)) {
-                return grammarErrorOnNode(accessor.name, accessor.kind === 163 /* GetAccessor */ ?
-                    ts.Diagnostics.A_get_accessor_cannot_have_parameters :
-                    ts.Diagnostics.A_set_accessor_must_have_exactly_one_parameter);
+            var param = ts.getParameterSymbolFromJSDoc(paramTag);
+            if (!param) {
+                // We will error in `checkJSDocParameterTag`.
+                return;
             }
-            if (accessor.kind === 164 /* SetAccessor */) {
-                if (accessor.type) {
-                    return grammarErrorOnNode(accessor.name, ts.Diagnostics.A_set_accessor_cannot_have_a_return_type_annotation);
-                }
-                var parameter = ts.Debug.checkDefined(ts.getSetAccessorValueParameter(accessor), "Return value does not match parameter count assertion.");
-                if (parameter.dotDotDotToken) {
-                    return grammarErrorOnNode(parameter.dotDotDotToken, ts.Diagnostics.A_set_accessor_cannot_have_rest_parameter);
-                }
-                if (parameter.questionToken) {
-                    return grammarErrorOnNode(parameter.questionToken, ts.Diagnostics.A_set_accessor_cannot_have_an_optional_parameter);
-                }
-                if (parameter.initializer) {
-                    return grammarErrorOnNode(accessor.name, ts.Diagnostics.A_set_accessor_parameter_cannot_have_an_initializer);
-                }
+            var host = ts.getHostSignatureFromJSDoc(paramTag);
+            if (!host || ts.last(host.parameters).symbol !== param) {
+                error(node, ts.Diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list);
             }
-            return false;
         }
-        /** Does the accessor have the right number of parameters?
-         * A get accessor has no parameters or a single `this` parameter.
-         * A set accessor has one parameter or a `this` parameter and one more parameter.
-         */
-        function doesAccessorHaveCorrectParameterCount(accessor) {
-            return getAccessorThisParameter(accessor) || accessor.parameters.length === (accessor.kind === 163 /* GetAccessor */ ? 0 : 1);
+        function getTypeFromJSDocVariadicType(node) {
+            var type = getTypeFromTypeNode(node.type);
+            var parent = node.parent;
+            var paramTag = node.parent.parent;
+            if (ts.isJSDocTypeExpression(node.parent) && ts.isJSDocParameterTag(paramTag)) {
+                // Else we will add a diagnostic, see `checkJSDocVariadicType`.
+                var host_1 = ts.getHostSignatureFromJSDoc(paramTag);
+                if (host_1) {
+                    /*
+                    Only return an array type if the corresponding parameter is marked as a rest parameter, or if there are no parameters.
+                    So in the following situation we will not create an array type:
+                        /** @param {...number} a * /
+                        function f(a) {}
+                    Because `a` will just be of type `number | undefined`. A synthetic `...args` will also be added, which *will* get an array type.
+                    */
+                    var lastParamDeclaration = ts.lastOrUndefined(host_1.parameters);
+                    var symbol = ts.getParameterSymbolFromJSDoc(paramTag);
+                    if (!lastParamDeclaration ||
+                        symbol && lastParamDeclaration.symbol === symbol && ts.isRestParameter(lastParamDeclaration)) {
+                        return createArrayType(type);
+                    }
+                }
+            }
+            if (ts.isParameter(parent) && ts.isJSDocFunctionType(parent.parent)) {
+                return createArrayType(type);
+            }
+            return addOptionality(type);
         }
-        function getAccessorThisParameter(accessor) {
-            if (accessor.parameters.length === (accessor.kind === 163 /* GetAccessor */ ? 1 : 2)) {
-                return ts.getThisParameter(accessor);
+        // Function and class expression bodies are checked after all statements in the enclosing body. This is
+        // to ensure constructs like the following are permitted:
+        //     const foo = function () {
+        //        const s = foo();
+        //        return "hello";
+        //     }
+        // Here, performing a full type check of the body of the function expression whilst in the process of
+        // determining the type of foo would cause foo to be given type any because of the recursive reference.
+        // Delaying the type check of the body ensures foo has been assigned a type.
+        function checkNodeDeferred(node) {
+            var enclosingFile = ts.getSourceFileOfNode(node);
+            var links = getNodeLinks(enclosingFile);
+            if (!(links.flags & 1 /* TypeChecked */)) {
+                links.deferredNodes = links.deferredNodes || new ts.Map();
+                var id = getNodeId(node);
+                links.deferredNodes.set(id, node);
             }
         }
-        function checkGrammarTypeOperatorNode(node) {
-            if (node.operator === 147 /* UniqueKeyword */) {
-                if (node.type.kind !== 144 /* SymbolKeyword */) {
-                    return grammarErrorOnNode(node.type, ts.Diagnostics._0_expected, ts.tokenToString(144 /* SymbolKeyword */));
-                }
-                var parent = ts.walkUpParenthesizedTypes(node.parent);
-                switch (parent.kind) {
-                    case 242 /* VariableDeclaration */:
-                        var decl = parent;
-                        if (decl.name.kind !== 75 /* Identifier */) {
-                            return grammarErrorOnNode(node, ts.Diagnostics.unique_symbol_types_may_not_be_used_on_a_variable_declaration_with_a_binding_name);
-                        }
-                        if (!ts.isVariableDeclarationInVariableStatement(decl)) {
-                            return grammarErrorOnNode(node, ts.Diagnostics.unique_symbol_types_are_only_allowed_on_variables_in_a_variable_statement);
-                        }
-                        if (!(decl.parent.flags & 2 /* Const */)) {
-                            return grammarErrorOnNode(parent.name, ts.Diagnostics.A_variable_whose_type_is_a_unique_symbol_type_must_be_const);
-                        }
-                        break;
-                    case 159 /* PropertyDeclaration */:
-                        if (!ts.hasModifier(parent, 32 /* Static */) ||
-                            !ts.hasModifier(parent, 64 /* Readonly */)) {
-                            return grammarErrorOnNode(parent.name, ts.Diagnostics.A_property_of_a_class_whose_type_is_a_unique_symbol_type_must_be_both_static_and_readonly);
-                        }
-                        break;
-                    case 158 /* PropertySignature */:
-                        if (!ts.hasModifier(parent, 64 /* Readonly */)) {
-                            return grammarErrorOnNode(parent.name, ts.Diagnostics.A_property_of_an_interface_or_type_literal_whose_type_is_a_unique_symbol_type_must_be_readonly);
-                        }
-                        break;
-                    default:
-                        return grammarErrorOnNode(node, ts.Diagnostics.unique_symbol_types_are_not_allowed_here);
-                }
+        function checkDeferredNodes(context) {
+            var links = getNodeLinks(context);
+            if (links.deferredNodes) {
+                links.deferredNodes.forEach(checkDeferredNode);
             }
-            else if (node.operator === 138 /* ReadonlyKeyword */) {
-                if (node.type.kind !== 174 /* ArrayType */ && node.type.kind !== 175 /* TupleType */) {
-                    return grammarErrorOnFirstToken(node, ts.Diagnostics.readonly_type_modifier_is_only_permitted_on_array_and_tuple_literal_types, ts.tokenToString(144 /* SymbolKeyword */));
-                }
+        }
+        function checkDeferredNode(node) {
+            var saveCurrentNode = currentNode;
+            currentNode = node;
+            instantiationCount = 0;
+            switch (node.kind) {
+                case 203 /* CallExpression */:
+                case 204 /* NewExpression */:
+                case 205 /* TaggedTemplateExpression */:
+                case 161 /* Decorator */:
+                case 275 /* JsxOpeningElement */:
+                    // These node kinds are deferred checked when overload resolution fails
+                    // To save on work, we ensure the arguments are checked just once, in
+                    // a deferred way
+                    resolveUntypedCall(node);
+                    break;
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    checkFunctionExpressionOrObjectLiteralMethodDeferred(node);
+                    break;
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    checkAccessorDeclaration(node);
+                    break;
+                case 221 /* ClassExpression */:
+                    checkClassExpressionDeferred(node);
+                    break;
+                case 274 /* JsxSelfClosingElement */:
+                    checkJsxSelfClosingElementDeferred(node);
+                    break;
+                case 273 /* JsxElement */:
+                    checkJsxElementDeferred(node);
+                    break;
             }
+            currentNode = saveCurrentNode;
         }
-        function checkGrammarForInvalidDynamicName(node, message) {
-            if (isNonBindableDynamicName(node)) {
-                return grammarErrorOnNode(node, message);
-            }
+        function checkSourceFile(node) {
+            var tracingData = ["check" /* Check */, "checkSourceFile", { path: node.path }];
+            ts.tracing.begin.apply(ts.tracing, tracingData);
+            ts.performance.mark("beforeCheck");
+            checkSourceFileWorker(node);
+            ts.performance.mark("afterCheck");
+            ts.performance.measure("Check", "beforeCheck", "afterCheck");
+            ts.tracing.end.apply(ts.tracing, tracingData);
         }
-        function checkGrammarMethod(node) {
-            if (checkGrammarFunctionLikeDeclaration(node)) {
-                return true;
+        function unusedIsError(kind, isAmbient) {
+            if (isAmbient) {
+                return false;
             }
-            if (node.kind === 161 /* MethodDeclaration */) {
-                if (node.parent.kind === 193 /* ObjectLiteralExpression */) {
-                    // We only disallow modifier on a method declaration if it is a property of object-literal-expression
-                    if (node.modifiers && !(node.modifiers.length === 1 && ts.first(node.modifiers).kind === 126 /* AsyncKeyword */)) {
-                        return grammarErrorOnFirstToken(node, ts.Diagnostics.Modifiers_cannot_appear_here);
-                    }
-                    else if (checkGrammarForInvalidQuestionMark(node.questionToken, ts.Diagnostics.An_object_member_cannot_be_declared_optional)) {
-                        return true;
-                    }
-                    else if (checkGrammarForInvalidExclamationToken(node.exclamationToken, ts.Diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context)) {
-                        return true;
-                    }
-                    else if (node.body === undefined) {
-                        return grammarErrorAtPos(node, node.end - 1, ";".length, ts.Diagnostics._0_expected, "{");
-                    }
+            switch (kind) {
+                case 0 /* Local */:
+                    return !!compilerOptions.noUnusedLocals;
+                case 1 /* Parameter */:
+                    return !!compilerOptions.noUnusedParameters;
+                default:
+                    return ts.Debug.assertNever(kind);
+            }
+        }
+        function getPotentiallyUnusedIdentifiers(sourceFile) {
+            return allPotentiallyUnusedIdentifiers.get(sourceFile.path) || ts.emptyArray;
+        }
+        // Fully type check a source file and collect the relevant diagnostics.
+        function checkSourceFileWorker(node) {
+            var links = getNodeLinks(node);
+            if (!(links.flags & 1 /* TypeChecked */)) {
+                if (ts.skipTypeChecking(node, compilerOptions, host)) {
+                    return;
                 }
-                if (checkGrammarForGenerator(node)) {
-                    return true;
+                // Grammar checking
+                checkGrammarSourceFile(node);
+                ts.clear(potentialThisCollisions);
+                ts.clear(potentialNewTargetCollisions);
+                ts.clear(potentialWeakMapCollisions);
+                ts.forEach(node.statements, checkSourceElement);
+                checkSourceElement(node.endOfFileToken);
+                checkDeferredNodes(node);
+                if (ts.isExternalOrCommonJsModule(node)) {
+                    registerForUnusedIdentifiersCheck(node);
                 }
-            }
-            if (ts.isClassLike(node.parent)) {
-                // Technically, computed properties in ambient contexts is disallowed
-                // for property declarations and accessors too, not just methods.
-                // However, property declarations disallow computed names in general,
-                // and accessors are not allowed in ambient contexts in general,
-                // so this error only really matters for methods.
-                if (node.flags & 8388608 /* Ambient */) {
-                    return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_an_ambient_context_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
+                if (!node.isDeclarationFile && (compilerOptions.noUnusedLocals || compilerOptions.noUnusedParameters)) {
+                    checkUnusedIdentifiers(getPotentiallyUnusedIdentifiers(node), function (containingNode, kind, diag) {
+                        if (!ts.containsParseError(containingNode) && unusedIsError(kind, !!(containingNode.flags & 8388608 /* Ambient */))) {
+                            diagnostics.add(diag);
+                        }
+                    });
                 }
-                else if (node.kind === 161 /* MethodDeclaration */ && !node.body) {
-                    return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_method_overload_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
+                if (compilerOptions.importsNotUsedAsValues === 2 /* Error */ &&
+                    !node.isDeclarationFile &&
+                    ts.isExternalModule(node)) {
+                    checkImportsForTypeOnlyConversion(node);
                 }
-            }
-            else if (node.parent.kind === 246 /* InterfaceDeclaration */) {
-                return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_an_interface_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
-            }
-            else if (node.parent.kind === 173 /* TypeLiteral */) {
-                return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_type_literal_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
-            }
-        }
-        function checkGrammarBreakOrContinueStatement(node) {
-            var current = node;
-            while (current) {
-                if (ts.isFunctionLike(current)) {
-                    return grammarErrorOnNode(node, ts.Diagnostics.Jump_target_cannot_cross_function_boundary);
+                if (ts.isExternalOrCommonJsModule(node)) {
+                    checkExternalModuleExports(node);
                 }
-                switch (current.kind) {
-                    case 238 /* LabeledStatement */:
-                        if (node.label && current.label.escapedText === node.label.escapedText) {
-                            // found matching label - verify that label usage is correct
-                            // continue can only target labels that are on iteration statements
-                            var isMisplacedContinueLabel = node.kind === 233 /* ContinueStatement */
-                                && !ts.isIterationStatement(current.statement, /*lookInLabeledStatement*/ true);
-                            if (isMisplacedContinueLabel) {
-                                return grammarErrorOnNode(node, ts.Diagnostics.A_continue_statement_can_only_jump_to_a_label_of_an_enclosing_iteration_statement);
-                            }
-                            return false;
-                        }
-                        break;
-                    case 237 /* SwitchStatement */:
-                        if (node.kind === 234 /* BreakStatement */ && !node.label) {
-                            // unlabeled break within switch statement - ok
-                            return false;
-                        }
-                        break;
-                    default:
-                        if (ts.isIterationStatement(current, /*lookInLabeledStatement*/ false) && !node.label) {
-                            // unlabeled break or continue within iteration statement - ok
-                            return false;
-                        }
-                        break;
+                if (potentialThisCollisions.length) {
+                    ts.forEach(potentialThisCollisions, checkIfThisIsCapturedInEnclosingScope);
+                    ts.clear(potentialThisCollisions);
                 }
-                current = current.parent;
+                if (potentialNewTargetCollisions.length) {
+                    ts.forEach(potentialNewTargetCollisions, checkIfNewTargetIsCapturedInEnclosingScope);
+                    ts.clear(potentialNewTargetCollisions);
+                }
+                if (potentialWeakMapCollisions.length) {
+                    ts.forEach(potentialWeakMapCollisions, checkWeakMapCollision);
+                    ts.clear(potentialWeakMapCollisions);
+                }
+                links.flags |= 1 /* TypeChecked */;
             }
-            if (node.label) {
-                var message = node.kind === 234 /* BreakStatement */
-                    ? ts.Diagnostics.A_break_statement_can_only_jump_to_a_label_of_an_enclosing_statement
-                    : ts.Diagnostics.A_continue_statement_can_only_jump_to_a_label_of_an_enclosing_iteration_statement;
-                return grammarErrorOnNode(node, message);
+        }
+        function getDiagnostics(sourceFile, ct) {
+            try {
+                // Record the cancellation token so it can be checked later on during checkSourceElement.
+                // Do this in a finally block so we can ensure that it gets reset back to nothing after
+                // this call is done.
+                cancellationToken = ct;
+                return getDiagnosticsWorker(sourceFile);
             }
-            else {
-                var message = node.kind === 234 /* BreakStatement */
-                    ? ts.Diagnostics.A_break_statement_can_only_be_used_within_an_enclosing_iteration_or_switch_statement
-                    : ts.Diagnostics.A_continue_statement_can_only_be_used_within_an_enclosing_iteration_statement;
-                return grammarErrorOnNode(node, message);
+            finally {
+                cancellationToken = undefined;
             }
         }
-        function checkGrammarBindingElement(node) {
-            if (node.dotDotDotToken) {
-                var elements = node.parent.elements;
-                if (node !== ts.last(elements)) {
-                    return grammarErrorOnNode(node, ts.Diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern);
-                }
-                checkGrammarForDisallowedTrailingComma(elements, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
-                if (node.propertyName) {
-                    return grammarErrorOnNode(node.name, ts.Diagnostics.A_rest_element_cannot_have_a_property_name);
+        function getDiagnosticsWorker(sourceFile) {
+            throwIfNonDiagnosticsProducing();
+            if (sourceFile) {
+                // Some global diagnostics are deferred until they are needed and
+                // may not be reported in the first call to getGlobalDiagnostics.
+                // We should catch these changes and report them.
+                var previousGlobalDiagnostics = diagnostics.getGlobalDiagnostics();
+                var previousGlobalDiagnosticsSize = previousGlobalDiagnostics.length;
+                checkSourceFile(sourceFile);
+                var semanticDiagnostics = diagnostics.getDiagnostics(sourceFile.fileName);
+                var currentGlobalDiagnostics = diagnostics.getGlobalDiagnostics();
+                if (currentGlobalDiagnostics !== previousGlobalDiagnostics) {
+                    // If the arrays are not the same reference, new diagnostics were added.
+                    var deferredGlobalDiagnostics = ts.relativeComplement(previousGlobalDiagnostics, currentGlobalDiagnostics, ts.compareDiagnostics);
+                    return ts.concatenate(deferredGlobalDiagnostics, semanticDiagnostics);
                 }
-                if (node.initializer) {
-                    // Error on equals token which immediately precedes the initializer
-                    return grammarErrorAtPos(node, node.initializer.pos - 1, 1, ts.Diagnostics.A_rest_element_cannot_have_an_initializer);
+                else if (previousGlobalDiagnosticsSize === 0 && currentGlobalDiagnostics.length > 0) {
+                    // If the arrays are the same reference, but the length has changed, a single
+                    // new diagnostic was added as DiagnosticCollection attempts to reuse the
+                    // same array.
+                    return ts.concatenate(currentGlobalDiagnostics, semanticDiagnostics);
                 }
+                return semanticDiagnostics;
             }
+            // Global diagnostics are always added when a file is not provided to
+            // getDiagnostics
+            ts.forEach(host.getSourceFiles(), checkSourceFile);
+            return diagnostics.getDiagnostics();
         }
-        function isStringOrNumberLiteralExpression(expr) {
-            return ts.isStringOrNumericLiteralLike(expr) ||
-                expr.kind === 207 /* PrefixUnaryExpression */ && expr.operator === 40 /* MinusToken */ &&
-                    expr.operand.kind === 8 /* NumericLiteral */;
-        }
-        function isBigIntLiteralExpression(expr) {
-            return expr.kind === 9 /* BigIntLiteral */ ||
-                expr.kind === 207 /* PrefixUnaryExpression */ && expr.operator === 40 /* MinusToken */ &&
-                    expr.operand.kind === 9 /* BigIntLiteral */;
+        function getGlobalDiagnostics() {
+            throwIfNonDiagnosticsProducing();
+            return diagnostics.getGlobalDiagnostics();
         }
-        function isSimpleLiteralEnumReference(expr) {
-            if ((ts.isPropertyAccessExpression(expr) || (ts.isElementAccessExpression(expr) && isStringOrNumberLiteralExpression(expr.argumentExpression))) &&
-                ts.isEntityNameExpression(expr.expression)) {
-                return !!(checkExpressionCached(expr).flags & 1024 /* EnumLiteral */);
+        function throwIfNonDiagnosticsProducing() {
+            if (!produceDiagnostics) {
+                throw new Error("Trying to get diagnostics from a type checker that does not produce them.");
             }
         }
-        function checkAmbientInitializer(node) {
-            var initializer = node.initializer;
-            if (initializer) {
-                var isInvalidInitializer = !(isStringOrNumberLiteralExpression(initializer) ||
-                    isSimpleLiteralEnumReference(initializer) ||
-                    initializer.kind === 106 /* TrueKeyword */ || initializer.kind === 91 /* FalseKeyword */ ||
-                    isBigIntLiteralExpression(initializer));
-                var isConstOrReadonly = ts.isDeclarationReadonly(node) || ts.isVariableDeclaration(node) && ts.isVarConst(node);
-                if (isConstOrReadonly && !node.type) {
-                    if (isInvalidInitializer) {
-                        return grammarErrorOnNode(initializer, ts.Diagnostics.A_const_initializer_in_an_ambient_context_must_be_a_string_or_numeric_literal_or_literal_enum_reference);
-                    }
-                }
-                else {
-                    return grammarErrorOnNode(initializer, ts.Diagnostics.Initializers_are_not_allowed_in_ambient_contexts);
-                }
-                if (!isConstOrReadonly || isInvalidInitializer) {
-                    return grammarErrorOnNode(initializer, ts.Diagnostics.Initializers_are_not_allowed_in_ambient_contexts);
-                }
+        // Language service support
+        function getSymbolsInScope(location, meaning) {
+            if (location.flags & 16777216 /* InWithStatement */) {
+                // We cannot answer semantic questions within a with block, do not proceed any further
+                return [];
             }
-        }
-        function checkGrammarVariableDeclaration(node) {
-            if (node.parent.parent.kind !== 231 /* ForInStatement */ && node.parent.parent.kind !== 232 /* ForOfStatement */) {
-                if (node.flags & 8388608 /* Ambient */) {
-                    checkAmbientInitializer(node);
-                }
-                else if (!node.initializer) {
-                    if (ts.isBindingPattern(node.name) && !ts.isBindingPattern(node.parent)) {
-                        return grammarErrorOnNode(node, ts.Diagnostics.A_destructuring_declaration_must_have_an_initializer);
+            var symbols = ts.createSymbolTable();
+            var isStatic = false;
+            populateSymbols();
+            symbols.delete("this" /* This */); // Not a symbol, a keyword
+            return symbolsToArray(symbols);
+            function populateSymbols() {
+                while (location) {
+                    if (location.locals && !isGlobalSourceFile(location)) {
+                        copySymbols(location.locals, meaning);
                     }
-                    if (ts.isVarConst(node)) {
-                        return grammarErrorOnNode(node, ts.Diagnostics.const_declarations_must_be_initialized);
+                    switch (location.kind) {
+                        case 297 /* SourceFile */:
+                            if (!ts.isExternalOrCommonJsModule(location))
+                                break;
+                        // falls through
+                        case 256 /* ModuleDeclaration */:
+                            copySymbols(getSymbolOfNode(location).exports, meaning & 2623475 /* ModuleMember */);
+                            break;
+                        case 255 /* EnumDeclaration */:
+                            copySymbols(getSymbolOfNode(location).exports, meaning & 8 /* EnumMember */);
+                            break;
+                        case 221 /* ClassExpression */:
+                            var className = location.name;
+                            if (className) {
+                                copySymbol(location.symbol, meaning);
+                            }
+                        // this fall-through is necessary because we would like to handle
+                        // type parameter inside class expression similar to how we handle it in classDeclaration and interface Declaration.
+                        // falls through
+                        case 252 /* ClassDeclaration */:
+                        case 253 /* InterfaceDeclaration */:
+                            // If we didn't come from static member of class or interface,
+                            // add the type parameters into the symbol table
+                            // (type parameters of classDeclaration/classExpression and interface are in member property of the symbol.
+                            // Note: that the memberFlags come from previous iteration.
+                            if (!isStatic) {
+                                copySymbols(getMembersOfSymbol(getSymbolOfNode(location)), meaning & 788968 /* Type */);
+                            }
+                            break;
+                        case 208 /* FunctionExpression */:
+                            var funcName = location.name;
+                            if (funcName) {
+                                copySymbol(location.symbol, meaning);
+                            }
+                            break;
                     }
-                }
-            }
-            if (node.exclamationToken && (node.parent.parent.kind !== 225 /* VariableStatement */ || !node.type || node.initializer || node.flags & 8388608 /* Ambient */)) {
-                return grammarErrorOnNode(node.exclamationToken, ts.Diagnostics.Definite_assignment_assertions_can_only_be_used_along_with_a_type_annotation);
-            }
-            var moduleKind = ts.getEmitModuleKind(compilerOptions);
-            if (moduleKind < ts.ModuleKind.ES2015 && moduleKind !== ts.ModuleKind.System && !compilerOptions.noEmit &&
-                !(node.parent.parent.flags & 8388608 /* Ambient */) && ts.hasModifier(node.parent.parent, 1 /* Export */)) {
-                checkESModuleMarker(node.name);
-            }
-            var checkLetConstNames = (ts.isLet(node) || ts.isVarConst(node));
-            // 1. LexicalDeclaration : LetOrConst BindingList ;
-            // It is a Syntax Error if the BoundNames of BindingList contains "let".
-            // 2. ForDeclaration: ForDeclaration : LetOrConst ForBinding
-            // It is a Syntax Error if the BoundNames of ForDeclaration contains "let".
-            // It is a SyntaxError if a VariableDeclaration or VariableDeclarationNoIn occurs within strict code
-            // and its Identifier is eval or arguments
-            return checkLetConstNames && checkGrammarNameInLetOrConstDeclarations(node.name);
-        }
-        function checkESModuleMarker(name) {
-            if (name.kind === 75 /* Identifier */) {
-                if (ts.idText(name) === "__esModule") {
-                    return grammarErrorOnNode(name, ts.Diagnostics.Identifier_expected_esModule_is_reserved_as_an_exported_marker_when_transforming_ECMAScript_modules);
-                }
-            }
-            else {
-                var elements = name.elements;
-                for (var _i = 0, elements_1 = elements; _i < elements_1.length; _i++) {
-                    var element = elements_1[_i];
-                    if (!ts.isOmittedExpression(element)) {
-                        return checkESModuleMarker(element.name);
+                    if (ts.introducesArgumentsExoticObject(location)) {
+                        copySymbol(argumentsSymbol, meaning);
                     }
+                    isStatic = ts.hasSyntacticModifier(location, 32 /* Static */);
+                    location = location.parent;
                 }
+                copySymbols(globals, meaning);
             }
-            return false;
-        }
-        function checkGrammarNameInLetOrConstDeclarations(name) {
-            if (name.kind === 75 /* Identifier */) {
-                if (name.originalKeywordKind === 115 /* LetKeyword */) {
-                    return grammarErrorOnNode(name, ts.Diagnostics.let_is_not_allowed_to_be_used_as_a_name_in_let_or_const_declarations);
+            /**
+             * Copy the given symbol into symbol tables if the symbol has the given meaning
+             * and it doesn't already existed in the symbol table
+             * @param key a key for storing in symbol table; if undefined, use symbol.name
+             * @param symbol the symbol to be added into symbol table
+             * @param meaning meaning of symbol to filter by before adding to symbol table
+             */
+            function copySymbol(symbol, meaning) {
+                if (ts.getCombinedLocalAndExportSymbolFlags(symbol) & meaning) {
+                    var id = symbol.escapedName;
+                    // We will copy all symbol regardless of its reserved name because
+                    // symbolsToArray will check whether the key is a reserved name and
+                    // it will not copy symbol with reserved name to the array
+                    if (!symbols.has(id)) {
+                        symbols.set(id, symbol);
+                    }
                 }
             }
-            else {
-                var elements = name.elements;
-                for (var _i = 0, elements_2 = elements; _i < elements_2.length; _i++) {
-                    var element = elements_2[_i];
-                    if (!ts.isOmittedExpression(element)) {
-                        checkGrammarNameInLetOrConstDeclarations(element.name);
-                    }
+            function copySymbols(source, meaning) {
+                if (meaning) {
+                    source.forEach(function (symbol) {
+                        copySymbol(symbol, meaning);
+                    });
                 }
             }
-            return false;
         }
-        function checkGrammarVariableDeclarationList(declarationList) {
-            var declarations = declarationList.declarations;
-            if (checkGrammarForDisallowedTrailingComma(declarationList.declarations)) {
-                return true;
+        function isTypeDeclarationName(name) {
+            return name.kind === 78 /* Identifier */ &&
+                isTypeDeclaration(name.parent) &&
+                ts.getNameOfDeclaration(name.parent) === name;
+        }
+        function isTypeDeclaration(node) {
+            switch (node.kind) {
+                case 159 /* TypeParameter */:
+                case 252 /* ClassDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 331 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 325 /* JSDocEnumTag */:
+                    return true;
+                case 262 /* ImportClause */:
+                    return node.isTypeOnly;
+                case 265 /* ImportSpecifier */:
+                case 270 /* ExportSpecifier */:
+                    return node.parent.parent.isTypeOnly;
+                default:
+                    return false;
             }
-            if (!declarationList.declarations.length) {
-                return grammarErrorAtPos(declarationList, declarations.pos, declarations.end - declarations.pos, ts.Diagnostics.Variable_declaration_list_cannot_be_empty);
+        }
+        // True if the given identifier is part of a type reference
+        function isTypeReferenceIdentifier(node) {
+            while (node.parent.kind === 157 /* QualifiedName */) {
+                node = node.parent;
             }
-            return false;
+            return node.parent.kind === 173 /* TypeReference */;
         }
-        function allowLetAndConstDeclarations(parent) {
-            switch (parent.kind) {
-                case 227 /* IfStatement */:
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
-                case 236 /* WithStatement */:
-                case 230 /* ForStatement */:
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
-                    return false;
-                case 238 /* LabeledStatement */:
-                    return allowLetAndConstDeclarations(parent.parent);
+        function isHeritageClauseElementIdentifier(node) {
+            while (node.parent.kind === 201 /* PropertyAccessExpression */) {
+                node = node.parent;
             }
-            return true;
+            return node.parent.kind === 223 /* ExpressionWithTypeArguments */;
         }
-        function checkGrammarForDisallowedLetOrConstStatement(node) {
-            if (!allowLetAndConstDeclarations(node.parent)) {
-                if (ts.isLet(node.declarationList)) {
-                    return grammarErrorOnNode(node, ts.Diagnostics.let_declarations_can_only_be_declared_inside_a_block);
-                }
-                else if (ts.isVarConst(node.declarationList)) {
-                    return grammarErrorOnNode(node, ts.Diagnostics.const_declarations_can_only_be_declared_inside_a_block);
-                }
+        function isJSDocEntryNameReference(node) {
+            while (node.parent.kind === 157 /* QualifiedName */) {
+                node = node.parent;
+            }
+            while (node.parent.kind === 201 /* PropertyAccessExpression */) {
+                node = node.parent;
             }
+            return node.parent.kind === 302 /* JSDocNameReference */;
         }
-        function checkGrammarMetaProperty(node) {
-            var escapedText = node.name.escapedText;
-            switch (node.keywordToken) {
-                case 99 /* NewKeyword */:
-                    if (escapedText !== "target") {
-                        return grammarErrorOnNode(node.name, ts.Diagnostics._0_is_not_a_valid_meta_property_for_keyword_1_Did_you_mean_2, node.name.escapedText, ts.tokenToString(node.keywordToken), "target");
-                    }
+        function forEachEnclosingClass(node, callback) {
+            var result;
+            while (true) {
+                node = ts.getContainingClass(node);
+                if (!node)
                     break;
-                case 96 /* ImportKeyword */:
-                    if (escapedText !== "meta") {
-                        return grammarErrorOnNode(node.name, ts.Diagnostics._0_is_not_a_valid_meta_property_for_keyword_1_Did_you_mean_2, node.name.escapedText, ts.tokenToString(node.keywordToken), "meta");
-                    }
+                if (result = callback(node))
                     break;
             }
+            return result;
         }
-        function hasParseDiagnostics(sourceFile) {
-            return sourceFile.parseDiagnostics.length > 0;
+        function isNodeUsedDuringClassInitialization(node) {
+            return !!ts.findAncestor(node, function (element) {
+                if (ts.isConstructorDeclaration(element) && ts.nodeIsPresent(element.body) || ts.isPropertyDeclaration(element)) {
+                    return true;
+                }
+                else if (ts.isClassLike(element) || ts.isFunctionLikeDeclaration(element)) {
+                    return "quit";
+                }
+                return false;
+            });
         }
-        function grammarErrorOnFirstToken(node, message, arg0, arg1, arg2) {
-            var sourceFile = ts.getSourceFileOfNode(node);
-            if (!hasParseDiagnostics(sourceFile)) {
-                var span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
-                diagnostics.add(ts.createFileDiagnostic(sourceFile, span.start, span.length, message, arg0, arg1, arg2));
-                return true;
-            }
-            return false;
+        function isNodeWithinClass(node, classDeclaration) {
+            return !!forEachEnclosingClass(node, function (n) { return n === classDeclaration; });
         }
-        function grammarErrorAtPos(nodeForSourceFile, start, length, message, arg0, arg1, arg2) {
-            var sourceFile = ts.getSourceFileOfNode(nodeForSourceFile);
-            if (!hasParseDiagnostics(sourceFile)) {
-                diagnostics.add(ts.createFileDiagnostic(sourceFile, start, length, message, arg0, arg1, arg2));
-                return true;
+        function getLeftSideOfImportEqualsOrExportAssignment(nodeOnRightSide) {
+            while (nodeOnRightSide.parent.kind === 157 /* QualifiedName */) {
+                nodeOnRightSide = nodeOnRightSide.parent;
             }
-            return false;
-        }
-        function grammarErrorOnNode(node, message, arg0, arg1, arg2) {
-            var sourceFile = ts.getSourceFileOfNode(node);
-            if (!hasParseDiagnostics(sourceFile)) {
-                diagnostics.add(ts.createDiagnosticForNode(node, message, arg0, arg1, arg2));
-                return true;
+            if (nodeOnRightSide.parent.kind === 260 /* ImportEqualsDeclaration */) {
+                return nodeOnRightSide.parent.moduleReference === nodeOnRightSide ? nodeOnRightSide.parent : undefined;
             }
-            return false;
+            if (nodeOnRightSide.parent.kind === 266 /* ExportAssignment */) {
+                return nodeOnRightSide.parent.expression === nodeOnRightSide ? nodeOnRightSide.parent : undefined;
+            }
+            return undefined;
         }
-        function checkGrammarConstructorTypeParameters(node) {
-            var jsdocTypeParameters = ts.isInJSFile(node) ? ts.getJSDocTypeParameterDeclarations(node) : undefined;
-            var range = node.typeParameters || jsdocTypeParameters && ts.firstOrUndefined(jsdocTypeParameters);
-            if (range) {
-                var pos = range.pos === range.end ? range.pos : ts.skipTrivia(ts.getSourceFileOfNode(node).text, range.pos);
-                return grammarErrorAtPos(node, pos, range.end - pos, ts.Diagnostics.Type_parameters_cannot_appear_on_a_constructor_declaration);
+        function isInRightSideOfImportOrExportAssignment(node) {
+            return getLeftSideOfImportEqualsOrExportAssignment(node) !== undefined;
+        }
+        function getSpecialPropertyAssignmentSymbolFromEntityName(entityName) {
+            var specialPropertyAssignmentKind = ts.getAssignmentDeclarationKind(entityName.parent.parent);
+            switch (specialPropertyAssignmentKind) {
+                case 1 /* ExportsProperty */:
+                case 3 /* PrototypeProperty */:
+                    return getSymbolOfNode(entityName.parent);
+                case 4 /* ThisProperty */:
+                case 2 /* ModuleExports */:
+                case 5 /* Property */:
+                    return getSymbolOfNode(entityName.parent.parent);
             }
         }
-        function checkGrammarConstructorTypeAnnotation(node) {
-            var type = ts.getEffectiveReturnTypeNode(node);
-            if (type) {
-                return grammarErrorOnNode(type, ts.Diagnostics.Type_annotation_cannot_appear_on_a_constructor_declaration);
+        function isImportTypeQualifierPart(node) {
+            var parent = node.parent;
+            while (ts.isQualifiedName(parent)) {
+                node = parent;
+                parent = parent.parent;
+            }
+            if (parent && parent.kind === 195 /* ImportType */ && parent.qualifier === node) {
+                return parent;
             }
+            return undefined;
         }
-        function checkGrammarProperty(node) {
-            if (ts.isClassLike(node.parent)) {
-                if (ts.isStringLiteral(node.name) && node.name.text === "constructor") {
-                    return grammarErrorOnNode(node.name, ts.Diagnostics.Classes_may_not_have_a_field_named_constructor);
-                }
-                if (checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_class_property_declaration_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type)) {
-                    return true;
-                }
-                if (languageVersion < 2 /* ES2015 */ && ts.isPrivateIdentifier(node.name)) {
-                    return grammarErrorOnNode(node.name, ts.Diagnostics.Private_identifiers_are_only_available_when_targeting_ECMAScript_2015_and_higher);
-                }
+        function getSymbolOfNameOrPropertyAccessExpression(name) {
+            if (ts.isDeclarationName(name)) {
+                return getSymbolOfNode(name.parent);
             }
-            else if (node.parent.kind === 246 /* InterfaceDeclaration */) {
-                if (checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_an_interface_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type)) {
-                    return true;
-                }
-                if (node.initializer) {
-                    return grammarErrorOnNode(node.initializer, ts.Diagnostics.An_interface_property_cannot_have_an_initializer);
+            if (ts.isInJSFile(name) &&
+                name.parent.kind === 201 /* PropertyAccessExpression */ &&
+                name.parent === name.parent.parent.left) {
+                // Check if this is a special property assignment
+                if (!ts.isPrivateIdentifier(name)) {
+                    var specialPropertyAssignmentSymbol = getSpecialPropertyAssignmentSymbolFromEntityName(name);
+                    if (specialPropertyAssignmentSymbol) {
+                        return specialPropertyAssignmentSymbol;
+                    }
                 }
             }
-            else if (node.parent.kind === 173 /* TypeLiteral */) {
-                if (checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_type_literal_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type)) {
-                    return true;
-                }
-                if (node.initializer) {
-                    return grammarErrorOnNode(node.initializer, ts.Diagnostics.A_type_literal_property_cannot_have_an_initializer);
+            if (name.parent.kind === 266 /* ExportAssignment */ && ts.isEntityNameExpression(name)) {
+                // Even an entity name expression that doesn't resolve as an entityname may still typecheck as a property access expression
+                var success = resolveEntityName(name, 
+                /*all meanings*/ 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */, /*ignoreErrors*/ true);
+                if (success && success !== unknownSymbol) {
+                    return success;
                 }
             }
-            if (node.flags & 8388608 /* Ambient */) {
-                checkAmbientInitializer(node);
-            }
-            if (ts.isPropertyDeclaration(node) && node.exclamationToken && (!ts.isClassLike(node.parent) || !node.type || node.initializer ||
-                node.flags & 8388608 /* Ambient */ || ts.hasModifier(node, 32 /* Static */ | 128 /* Abstract */))) {
-                return grammarErrorOnNode(node.exclamationToken, ts.Diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context);
-            }
-        }
-        function checkGrammarTopLevelElementForRequiredDeclareModifier(node) {
-            // A declare modifier is required for any top level .d.ts declaration except export=, export default, export as namespace
-            // interfaces and imports categories:
-            //
-            //  DeclarationElement:
-            //     ExportAssignment
-            //     export_opt   InterfaceDeclaration
-            //     export_opt   TypeAliasDeclaration
-            //     export_opt   ImportDeclaration
-            //     export_opt   ExternalImportDeclaration
-            //     export_opt   AmbientDeclaration
-            //
-            // TODO: The spec needs to be amended to reflect this grammar.
-            if (node.kind === 246 /* InterfaceDeclaration */ ||
-                node.kind === 247 /* TypeAliasDeclaration */ ||
-                node.kind === 254 /* ImportDeclaration */ ||
-                node.kind === 253 /* ImportEqualsDeclaration */ ||
-                node.kind === 260 /* ExportDeclaration */ ||
-                node.kind === 259 /* ExportAssignment */ ||
-                node.kind === 252 /* NamespaceExportDeclaration */ ||
-                ts.hasModifier(node, 2 /* Ambient */ | 1 /* Export */ | 512 /* Default */)) {
-                return false;
+            else if (!ts.isPropertyAccessExpression(name) && !ts.isPrivateIdentifier(name) && isInRightSideOfImportOrExportAssignment(name)) {
+                // Since we already checked for ExportAssignment, this really could only be an Import
+                var importEqualsDeclaration = ts.getAncestor(name, 260 /* ImportEqualsDeclaration */);
+                ts.Debug.assert(importEqualsDeclaration !== undefined);
+                return getSymbolOfPartOfRightHandSideOfImportEquals(name, /*dontResolveAlias*/ true);
             }
-            return grammarErrorOnFirstToken(node, ts.Diagnostics.Top_level_declarations_in_d_ts_files_must_start_with_either_a_declare_or_export_modifier);
-        }
-        function checkGrammarTopLevelElementsForRequiredDeclareModifier(file) {
-            for (var _i = 0, _a = file.statements; _i < _a.length; _i++) {
-                var decl = _a[_i];
-                if (ts.isDeclaration(decl) || decl.kind === 225 /* VariableStatement */) {
-                    if (checkGrammarTopLevelElementForRequiredDeclareModifier(decl)) {
-                        return true;
-                    }
+            if (!ts.isPropertyAccessExpression(name) && !ts.isPrivateIdentifier(name)) {
+                var possibleImportNode = isImportTypeQualifierPart(name);
+                if (possibleImportNode) {
+                    getTypeFromTypeNode(possibleImportNode);
+                    var sym = getNodeLinks(name).resolvedSymbol;
+                    return sym === unknownSymbol ? undefined : sym;
                 }
             }
-            return false;
-        }
-        function checkGrammarSourceFile(node) {
-            return !!(node.flags & 8388608 /* Ambient */) && checkGrammarTopLevelElementsForRequiredDeclareModifier(node);
-        }
-        function checkGrammarStatementInAmbientContext(node) {
-            if (node.flags & 8388608 /* Ambient */) {
-                // Find containing block which is either Block, ModuleBlock, SourceFile
-                var links = getNodeLinks(node);
-                if (!links.hasReportedStatementInAmbientContext && (ts.isFunctionLike(node.parent) || ts.isAccessor(node.parent))) {
-                    return getNodeLinks(node).hasReportedStatementInAmbientContext = grammarErrorOnFirstToken(node, ts.Diagnostics.An_implementation_cannot_be_declared_in_ambient_contexts);
-                }
-                // We are either parented by another statement, or some sort of block.
-                // If we're in a block, we only want to really report an error once
-                // to prevent noisiness.  So use a bit on the block to indicate if
-                // this has already been reported, and don't report if it has.
-                //
-                if (node.parent.kind === 223 /* Block */ || node.parent.kind === 250 /* ModuleBlock */ || node.parent.kind === 290 /* SourceFile */) {
-                    var links_2 = getNodeLinks(node.parent);
-                    // Check if the containing block ever report this error
-                    if (!links_2.hasReportedStatementInAmbientContext) {
-                        return links_2.hasReportedStatementInAmbientContext = grammarErrorOnFirstToken(node, ts.Diagnostics.Statements_are_not_allowed_in_ambient_contexts);
+            while (ts.isRightSideOfQualifiedNameOrPropertyAccess(name)) {
+                name = name.parent;
+            }
+            if (isHeritageClauseElementIdentifier(name)) {
+                var meaning = 0 /* None */;
+                // In an interface or class, we're definitely interested in a type.
+                if (name.parent.kind === 223 /* ExpressionWithTypeArguments */) {
+                    meaning = 788968 /* Type */;
+                    // In a class 'extends' clause we are also looking for a value.
+                    if (ts.isExpressionWithTypeArgumentsInClassExtendsClause(name.parent)) {
+                        meaning |= 111551 /* Value */;
                     }
                 }
                 else {
-                    // We must be parented by a statement.  If so, there's no need
-                    // to report the error as our parent will have already done it.
-                    // Debug.assert(isStatement(node.parent));
+                    meaning = 1920 /* Namespace */;
                 }
-            }
-            return false;
-        }
-        function checkGrammarNumericLiteral(node) {
-            // Grammar checking
-            if (node.numericLiteralFlags & 32 /* Octal */) {
-                var diagnosticMessage = void 0;
-                if (languageVersion >= 1 /* ES5 */) {
-                    diagnosticMessage = ts.Diagnostics.Octal_literals_are_not_available_when_targeting_ECMAScript_5_and_higher_Use_the_syntax_0;
+                meaning |= 2097152 /* Alias */;
+                var entityNameSymbol = ts.isEntityNameExpression(name) ? resolveEntityName(name, meaning) : undefined;
+                if (entityNameSymbol) {
+                    return entityNameSymbol;
                 }
-                else if (ts.isChildOfNodeWithKind(node, 187 /* LiteralType */)) {
-                    diagnosticMessage = ts.Diagnostics.Octal_literal_types_must_use_ES2015_syntax_Use_the_syntax_0;
+            }
+            if (name.parent.kind === 326 /* JSDocParameterTag */) {
+                return ts.getParameterSymbolFromJSDoc(name.parent);
+            }
+            if (name.parent.kind === 159 /* TypeParameter */ && name.parent.parent.kind === 330 /* JSDocTemplateTag */) {
+                ts.Debug.assert(!ts.isInJSFile(name)); // Otherwise `isDeclarationName` would have been true.
+                var typeParameter = ts.getTypeParameterFromJsDoc(name.parent);
+                return typeParameter && typeParameter.symbol;
+            }
+            if (ts.isExpressionNode(name)) {
+                if (ts.nodeIsMissing(name)) {
+                    // Missing entity name.
+                    return undefined;
                 }
-                else if (ts.isChildOfNodeWithKind(node, 284 /* EnumMember */)) {
-                    diagnosticMessage = ts.Diagnostics.Octal_literals_are_not_allowed_in_enums_members_initializer_Use_the_syntax_0;
+                if (name.kind === 78 /* Identifier */) {
+                    if (ts.isJSXTagName(name) && isJsxIntrinsicIdentifier(name)) {
+                        var symbol = getIntrinsicTagSymbol(name.parent);
+                        return symbol === unknownSymbol ? undefined : symbol;
+                    }
+                    return resolveEntityName(name, 111551 /* Value */, /*ignoreErrors*/ false, /*dontResolveAlias*/ true);
                 }
-                if (diagnosticMessage) {
-                    var withMinus = ts.isPrefixUnaryExpression(node.parent) && node.parent.operator === 40 /* MinusToken */;
-                    var literal = (withMinus ? "-" : "") + "0o" + node.text;
-                    return grammarErrorOnNode(withMinus ? node.parent : node, diagnosticMessage, literal);
+                else if (name.kind === 201 /* PropertyAccessExpression */ || name.kind === 157 /* QualifiedName */) {
+                    var links = getNodeLinks(name);
+                    if (links.resolvedSymbol) {
+                        return links.resolvedSymbol;
+                    }
+                    if (name.kind === 201 /* PropertyAccessExpression */) {
+                        checkPropertyAccessExpression(name);
+                    }
+                    else {
+                        checkQualifiedName(name);
+                    }
+                    return links.resolvedSymbol;
                 }
             }
-            // Realism (size) checking
-            checkNumericLiteralValueSize(node);
-            return false;
-        }
-        function checkNumericLiteralValueSize(node) {
-            // Scientific notation (e.g. 2e54 and 1e00000000010) can't be converted to bigint
-            // Literals with 15 or fewer characters aren't long enough to reach past 2^53 - 1
-            // Fractional numbers (e.g. 9000000000000000.001) are inherently imprecise anyway
-            if (node.numericLiteralFlags & 16 /* Scientific */ || node.text.length <= 15 || node.text.indexOf(".") !== -1) {
-                return;
+            else if (isTypeReferenceIdentifier(name)) {
+                var meaning = name.parent.kind === 173 /* TypeReference */ ? 788968 /* Type */ : 1920 /* Namespace */;
+                return resolveEntityName(name, meaning, /*ignoreErrors*/ false, /*dontResolveAlias*/ true);
             }
-            // We can't rely on the runtime to accurately store and compare extremely large numeric values
-            // Even for internal use, we use getTextOfNode: https://github.com/microsoft/TypeScript/issues/33298
-            // Thus, if the runtime claims a too-large number is lower than Number.MAX_SAFE_INTEGER,
-            // it's likely addition operations on it will fail too
-            var apparentValue = +ts.getTextOfNode(node);
-            if (apparentValue <= Math.pow(2, 53) - 1 && apparentValue + 1 > apparentValue) {
-                return;
+            else if (isJSDocEntryNameReference(name)) {
+                var meaning = 788968 /* Type */ | 1920 /* Namespace */ | 111551 /* Value */;
+                return resolveEntityName(name, meaning, /*ignoreErrors*/ false, /*dontResolveAlias*/ true, ts.getHostSignatureFromJSDoc(name));
             }
-            addErrorOrSuggestion(/*isError*/ false, ts.createDiagnosticForNode(node, ts.Diagnostics.Numeric_literals_with_absolute_values_equal_to_2_53_or_greater_are_too_large_to_be_represented_accurately_as_integers));
+            if (name.parent.kind === 172 /* TypePredicate */) {
+                return resolveEntityName(name, /*meaning*/ 1 /* FunctionScopedVariable */);
+            }
+            // Do we want to return undefined here?
+            return undefined;
         }
-        function checkGrammarBigIntLiteral(node) {
-            var literalType = ts.isLiteralTypeNode(node.parent) ||
-                ts.isPrefixUnaryExpression(node.parent) && ts.isLiteralTypeNode(node.parent.parent);
-            if (!literalType) {
-                if (languageVersion < 7 /* ES2020 */) {
-                    if (grammarErrorOnNode(node, ts.Diagnostics.BigInt_literals_are_not_available_when_targeting_lower_than_ES2020)) {
-                        return true;
+        function getSymbolAtLocation(node, ignoreErrors) {
+            if (node.kind === 297 /* SourceFile */) {
+                return ts.isExternalModule(node) ? getMergedSymbol(node.symbol) : undefined;
+            }
+            var parent = node.parent;
+            var grandParent = parent.parent;
+            if (node.flags & 16777216 /* InWithStatement */) {
+                // We cannot answer semantic questions within a with block, do not proceed any further
+                return undefined;
+            }
+            if (isDeclarationNameOrImportPropertyName(node)) {
+                // This is a declaration, call getSymbolOfNode
+                var parentSymbol = getSymbolOfNode(parent);
+                return ts.isImportOrExportSpecifier(node.parent) && node.parent.propertyName === node
+                    ? getImmediateAliasedSymbol(parentSymbol)
+                    : parentSymbol;
+            }
+            else if (ts.isLiteralComputedPropertyDeclarationName(node)) {
+                return getSymbolOfNode(parent.parent);
+            }
+            if (node.kind === 78 /* Identifier */) {
+                if (isInRightSideOfImportOrExportAssignment(node)) {
+                    return getSymbolOfNameOrPropertyAccessExpression(node);
+                }
+                else if (parent.kind === 198 /* BindingElement */ &&
+                    grandParent.kind === 196 /* ObjectBindingPattern */ &&
+                    node === parent.propertyName) {
+                    var typeOfPattern = getTypeOfNode(grandParent);
+                    var propertyDeclaration = getPropertyOfType(typeOfPattern, node.escapedText);
+                    if (propertyDeclaration) {
+                        return propertyDeclaration;
                     }
                 }
             }
-            return false;
-        }
-        function grammarErrorAfterFirstToken(node, message, arg0, arg1, arg2) {
-            var sourceFile = ts.getSourceFileOfNode(node);
-            if (!hasParseDiagnostics(sourceFile)) {
-                var span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
-                diagnostics.add(ts.createFileDiagnostic(sourceFile, ts.textSpanEnd(span), /*length*/ 0, message, arg0, arg1, arg2));
-                return true;
+            switch (node.kind) {
+                case 78 /* Identifier */:
+                case 79 /* PrivateIdentifier */:
+                case 201 /* PropertyAccessExpression */:
+                case 157 /* QualifiedName */:
+                    return getSymbolOfNameOrPropertyAccessExpression(node);
+                case 107 /* ThisKeyword */:
+                    var container = ts.getThisContainer(node, /*includeArrowFunctions*/ false);
+                    if (ts.isFunctionLike(container)) {
+                        var sig = getSignatureFromDeclaration(container);
+                        if (sig.thisParameter) {
+                            return sig.thisParameter;
+                        }
+                    }
+                    if (ts.isInExpressionContext(node)) {
+                        return checkExpression(node).symbol;
+                    }
+                // falls through
+                case 187 /* ThisType */:
+                    return getTypeFromThisTypeNode(node).symbol;
+                case 105 /* SuperKeyword */:
+                    return checkExpression(node).symbol;
+                case 132 /* ConstructorKeyword */:
+                    // constructor keyword for an overload, should take us to the definition if it exist
+                    var constructorDeclaration = node.parent;
+                    if (constructorDeclaration && constructorDeclaration.kind === 166 /* Constructor */) {
+                        return constructorDeclaration.parent.symbol;
+                    }
+                    return undefined;
+                case 10 /* StringLiteral */:
+                case 14 /* NoSubstitutionTemplateLiteral */:
+                    // 1). import x = require("./mo/*gotToDefinitionHere*/d")
+                    // 2). External module name in an import declaration
+                    // 3). Dynamic import call or require in javascript
+                    // 4). type A = import("./f/*gotToDefinitionHere*/oo")
+                    if ((ts.isExternalModuleImportEqualsDeclaration(node.parent.parent) && ts.getExternalModuleImportEqualsDeclarationExpression(node.parent.parent) === node) ||
+                        ((node.parent.kind === 261 /* ImportDeclaration */ || node.parent.kind === 267 /* ExportDeclaration */) && node.parent.moduleSpecifier === node) ||
+                        ((ts.isInJSFile(node) && ts.isRequireCall(node.parent, /*checkArgumentIsStringLiteralLike*/ false)) || ts.isImportCall(node.parent)) ||
+                        (ts.isLiteralTypeNode(node.parent) && ts.isLiteralImportTypeNode(node.parent.parent) && node.parent.parent.argument === node.parent)) {
+                        return resolveExternalModuleName(node, node, ignoreErrors);
+                    }
+                    if (ts.isCallExpression(parent) && ts.isBindableObjectDefinePropertyCall(parent) && parent.arguments[1] === node) {
+                        return getSymbolOfNode(parent);
+                    }
+                // falls through
+                case 8 /* NumericLiteral */:
+                    // index access
+                    var objectType = ts.isElementAccessExpression(parent)
+                        ? parent.argumentExpression === node ? getTypeOfExpression(parent.expression) : undefined
+                        : ts.isLiteralTypeNode(parent) && ts.isIndexedAccessTypeNode(grandParent)
+                            ? getTypeFromTypeNode(grandParent.objectType)
+                            : undefined;
+                    return objectType && getPropertyOfType(objectType, ts.escapeLeadingUnderscores(node.text));
+                case 87 /* DefaultKeyword */:
+                case 97 /* FunctionKeyword */:
+                case 38 /* EqualsGreaterThanToken */:
+                case 83 /* ClassKeyword */:
+                    return getSymbolOfNode(node.parent);
+                case 195 /* ImportType */:
+                    return ts.isLiteralImportTypeNode(node) ? getSymbolAtLocation(node.argument.literal, ignoreErrors) : undefined;
+                case 92 /* ExportKeyword */:
+                    return ts.isExportAssignment(node.parent) ? ts.Debug.checkDefined(node.parent.symbol) : undefined;
+                default:
+                    return undefined;
             }
-            return false;
         }
-        function getAmbientModules() {
-            if (!ambientModulesCache) {
-                ambientModulesCache = [];
-                globals.forEach(function (global, sym) {
-                    // No need to `unescapeLeadingUnderscores`, an escaped symbol is never an ambient module.
-                    if (ambientModuleSymbolRegex.test(sym)) {
-                        ambientModulesCache.push(global);
-                    }
-                });
+        function getShorthandAssignmentValueSymbol(location) {
+            if (location && location.kind === 289 /* ShorthandPropertyAssignment */) {
+                return resolveEntityName(location.name, 111551 /* Value */ | 2097152 /* Alias */);
             }
-            return ambientModulesCache;
+            return undefined;
         }
-        function checkGrammarImportClause(node) {
-            if (node.isTypeOnly && node.name && node.namedBindings) {
-                return grammarErrorOnNode(node, ts.Diagnostics.A_type_only_import_can_specify_a_default_import_or_named_bindings_but_not_both);
+        /** Returns the target of an export specifier without following aliases */
+        function getExportSpecifierLocalTargetSymbol(node) {
+            if (ts.isExportSpecifier(node)) {
+                return node.parent.parent.moduleSpecifier ?
+                    getExternalModuleMember(node.parent.parent, node) :
+                    resolveEntityName(node.propertyName || node.name, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */);
+            }
+            else {
+                return resolveEntityName(node, 111551 /* Value */ | 788968 /* Type */ | 1920 /* Namespace */ | 2097152 /* Alias */);
             }
-            return false;
         }
-        function checkGrammarImportCallExpression(node) {
-            if (moduleKind === ts.ModuleKind.ES2015) {
-                return grammarErrorOnNode(node, ts.Diagnostics.Dynamic_imports_are_only_supported_when_the_module_flag_is_set_to_es2020_esnext_commonjs_amd_system_or_umd);
+        function getTypeOfNode(node) {
+            if (ts.isSourceFile(node) && !ts.isExternalModule(node)) {
+                return errorType;
             }
-            if (node.typeArguments) {
-                return grammarErrorOnNode(node, ts.Diagnostics.Dynamic_import_cannot_have_type_arguments);
+            if (node.flags & 16777216 /* InWithStatement */) {
+                // We cannot answer semantic questions within a with block, do not proceed any further
+                return errorType;
             }
-            var nodeArguments = node.arguments;
-            if (nodeArguments.length !== 1) {
-                return grammarErrorOnNode(node, ts.Diagnostics.Dynamic_import_must_have_one_specifier_as_an_argument);
+            var classDecl = ts.tryGetClassImplementingOrExtendingExpressionWithTypeArguments(node);
+            var classType = classDecl && getDeclaredTypeOfClassOrInterface(getSymbolOfNode(classDecl.class));
+            if (ts.isPartOfTypeNode(node)) {
+                var typeFromTypeNode = getTypeFromTypeNode(node);
+                return classType ? getTypeWithThisArgument(typeFromTypeNode, classType.thisType) : typeFromTypeNode;
             }
-            checkGrammarForDisallowedTrailingComma(nodeArguments);
-            // see: parseArgumentOrArrayLiteralElement...we use this function which parse arguments of callExpression to parse specifier for dynamic import.
-            // parseArgumentOrArrayLiteralElement allows spread element to be in an argument list which is not allowed as specifier in dynamic import.
-            if (ts.isSpreadElement(nodeArguments[0])) {
-                return grammarErrorOnNode(nodeArguments[0], ts.Diagnostics.Specifier_of_dynamic_import_cannot_be_spread_element);
+            if (ts.isExpressionNode(node)) {
+                return getRegularTypeOfExpression(node);
             }
-            return false;
-        }
-        function findMatchingTypeReferenceOrTypeAliasReference(source, unionTarget) {
-            var sourceObjectFlags = ts.getObjectFlags(source);
-            if (sourceObjectFlags & (4 /* Reference */ | 16 /* Anonymous */) && unionTarget.flags & 1048576 /* Union */) {
-                return ts.find(unionTarget.types, function (target) {
-                    if (target.flags & 524288 /* Object */) {
-                        var overlapObjFlags = sourceObjectFlags & ts.getObjectFlags(target);
-                        if (overlapObjFlags & 4 /* Reference */) {
-                            return source.target === target.target;
-                        }
-                        if (overlapObjFlags & 16 /* Anonymous */) {
-                            return !!source.aliasSymbol && source.aliasSymbol === target.aliasSymbol;
-                        }
-                    }
-                    return false;
-                });
+            if (classType && !classDecl.isImplements) {
+                // A SyntaxKind.ExpressionWithTypeArguments is considered a type node, except when it occurs in the
+                // extends clause of a class. We handle that case here.
+                var baseType = ts.firstOrUndefined(getBaseTypes(classType));
+                return baseType ? getTypeWithThisArgument(baseType, classType.thisType) : errorType;
             }
-        }
-        function findBestTypeForObjectLiteral(source, unionTarget) {
-            if (ts.getObjectFlags(source) & 128 /* ObjectLiteral */ && forEachType(unionTarget, isArrayLikeType)) {
-                return ts.find(unionTarget.types, function (t) { return !isArrayLikeType(t); });
+            if (isTypeDeclaration(node)) {
+                // In this case, we call getSymbolOfNode instead of getSymbolAtLocation because it is a declaration
+                var symbol = getSymbolOfNode(node);
+                return getDeclaredTypeOfSymbol(symbol);
             }
-        }
-        function findBestTypeForInvokable(source, unionTarget) {
-            var signatureKind = 0 /* Call */;
-            var hasSignatures = getSignaturesOfType(source, signatureKind).length > 0 ||
-                (signatureKind = 1 /* Construct */, getSignaturesOfType(source, signatureKind).length > 0);
-            if (hasSignatures) {
-                return ts.find(unionTarget.types, function (t) { return getSignaturesOfType(t, signatureKind).length > 0; });
+            if (isTypeDeclarationName(node)) {
+                var symbol = getSymbolAtLocation(node);
+                return symbol ? getDeclaredTypeOfSymbol(symbol) : errorType;
             }
-        }
-        function findMostOverlappyType(source, unionTarget) {
-            var bestMatch;
-            var matchingCount = 0;
-            for (var _i = 0, _a = unionTarget.types; _i < _a.length; _i++) {
-                var target = _a[_i];
-                var overlap = getIntersectionType([getIndexType(source), getIndexType(target)]);
-                if (overlap.flags & 4194304 /* Index */) {
-                    // perfect overlap of keys
-                    bestMatch = target;
-                    matchingCount = Infinity;
-                }
-                else if (overlap.flags & 1048576 /* Union */) {
-                    // We only want to account for literal types otherwise.
-                    // If we have a union of index types, it seems likely that we
-                    // needed to elaborate between two generic mapped types anyway.
-                    var len = ts.length(ts.filter(overlap.types, isUnitType));
-                    if (len >= matchingCount) {
-                        bestMatch = target;
-                        matchingCount = len;
-                    }
-                }
-                else if (isUnitType(overlap) && 1 >= matchingCount) {
-                    bestMatch = target;
-                    matchingCount = 1;
-                }
+            if (ts.isDeclaration(node)) {
+                // In this case, we call getSymbolOfNode instead of getSymbolAtLocation because it is a declaration
+                var symbol = getSymbolOfNode(node);
+                return getTypeOfSymbol(symbol);
             }
-            return bestMatch;
-        }
-        function filterPrimitivesIfContainsNonPrimitive(type) {
-            if (maybeTypeOfKind(type, 67108864 /* NonPrimitive */)) {
-                var result = filterType(type, function (t) { return !(t.flags & 131068 /* Primitive */); });
-                if (!(result.flags & 131072 /* Never */)) {
-                    return result;
+            if (isDeclarationNameOrImportPropertyName(node)) {
+                var symbol = getSymbolAtLocation(node);
+                if (symbol) {
+                    return getTypeOfSymbol(symbol);
                 }
+                return errorType;
             }
-            return type;
-        }
-        // Keep this up-to-date with the same logic within `getApparentTypeOfContextualType`, since they should behave similarly
-        function findMatchingDiscriminantType(source, target, isRelatedTo, skipPartial) {
-            if (target.flags & 1048576 /* Union */ && source.flags & (2097152 /* Intersection */ | 524288 /* Object */)) {
-                var sourceProperties = getPropertiesOfType(source);
-                if (sourceProperties) {
-                    var sourcePropertiesFiltered = findDiscriminantProperties(sourceProperties, target);
-                    if (sourcePropertiesFiltered) {
-                        return discriminateTypeByDiscriminableItems(target, ts.map(sourcePropertiesFiltered, function (p) { return [function () { return getTypeOfSymbol(p); }, p.escapedName]; }), isRelatedTo, /*defaultValue*/ undefined, skipPartial);
-                    }
+            if (ts.isBindingPattern(node)) {
+                return getTypeForVariableLikeDeclaration(node.parent, /*includeOptionality*/ true) || errorType;
+            }
+            if (isInRightSideOfImportOrExportAssignment(node)) {
+                var symbol = getSymbolAtLocation(node);
+                if (symbol) {
+                    var declaredType = getDeclaredTypeOfSymbol(symbol);
+                    return declaredType !== errorType ? declaredType : getTypeOfSymbol(symbol);
                 }
             }
-            return undefined;
-        }
-    }
-    ts.createTypeChecker = createTypeChecker;
-    function isNotAccessor(declaration) {
-        // Accessors check for their own matching duplicates, and in contexts where they are valid, there are already duplicate identifier checks
-        return !ts.isAccessor(declaration);
-    }
-    function isNotOverload(declaration) {
-        return (declaration.kind !== 244 /* FunctionDeclaration */ && declaration.kind !== 161 /* MethodDeclaration */) ||
-            !!declaration.body;
-    }
-    /** Like 'isDeclarationName', but returns true for LHS of `import { x as y }` or `export { x as y }`. */
-    function isDeclarationNameOrImportPropertyName(name) {
-        switch (name.parent.kind) {
-            case 258 /* ImportSpecifier */:
-            case 263 /* ExportSpecifier */:
-                return ts.isIdentifier(name);
-            default:
-                return ts.isDeclarationName(name);
-        }
-    }
-    function isSomeImportDeclaration(decl) {
-        switch (decl.kind) {
-            case 255 /* ImportClause */: // For default import
-            case 253 /* ImportEqualsDeclaration */:
-            case 256 /* NamespaceImport */:
-            case 258 /* ImportSpecifier */: // For rename import `x as y`
-                return true;
-            case 75 /* Identifier */:
-                // For regular import, `decl` is an Identifier under the ImportSpecifier.
-                return decl.parent.kind === 258 /* ImportSpecifier */;
-            default:
-                return false;
+            return errorType;
         }
-    }
-    var JsxNames;
-    (function (JsxNames) {
-        JsxNames.JSX = "JSX";
-        JsxNames.IntrinsicElements = "IntrinsicElements";
-        JsxNames.ElementClass = "ElementClass";
-        JsxNames.ElementAttributesPropertyNameContainer = "ElementAttributesProperty"; // TODO: Deprecate and remove support
-        JsxNames.ElementChildrenAttributeNameContainer = "ElementChildrenAttribute";
-        JsxNames.Element = "Element";
-        JsxNames.IntrinsicAttributes = "IntrinsicAttributes";
-        JsxNames.IntrinsicClassAttributes = "IntrinsicClassAttributes";
-        JsxNames.LibraryManagedAttributes = "LibraryManagedAttributes";
-    })(JsxNames || (JsxNames = {}));
-    function getIterationTypesKeyFromIterationTypeKind(typeKind) {
-        switch (typeKind) {
-            case 0 /* Yield */: return "yieldType";
-            case 1 /* Return */: return "returnType";
-            case 2 /* Next */: return "nextType";
+        // Gets the type of object literal or array literal of destructuring assignment.
+        // { a } from
+        //     for ( { a } of elems) {
+        //     }
+        // [ a ] from
+        //     [a] = [ some array ...]
+        function getTypeOfAssignmentPattern(expr) {
+            ts.Debug.assert(expr.kind === 200 /* ObjectLiteralExpression */ || expr.kind === 199 /* ArrayLiteralExpression */);
+            // If this is from "for of"
+            //     for ( { a } of elems) {
+            //     }
+            if (expr.parent.kind === 239 /* ForOfStatement */) {
+                var iteratedType = checkRightHandSideOfForOf(expr.parent);
+                return checkDestructuringAssignment(expr, iteratedType || errorType);
+            }
+            // If this is from "for" initializer
+            //     for ({a } = elems[0];.....) { }
+            if (expr.parent.kind === 216 /* BinaryExpression */) {
+                var iteratedType = getTypeOfExpression(expr.parent.right);
+                return checkDestructuringAssignment(expr, iteratedType || errorType);
+            }
+            // If this is from nested object binding pattern
+            //     for ({ skills: { primary, secondary } } = multiRobot, i = 0; i < 1; i++) {
+            if (expr.parent.kind === 288 /* PropertyAssignment */) {
+                var node_2 = ts.cast(expr.parent.parent, ts.isObjectLiteralExpression);
+                var typeOfParentObjectLiteral = getTypeOfAssignmentPattern(node_2) || errorType;
+                var propertyIndex = ts.indexOfNode(node_2.properties, expr.parent);
+                return checkObjectLiteralDestructuringPropertyAssignment(node_2, typeOfParentObjectLiteral, propertyIndex);
+            }
+            // Array literal assignment - array destructuring pattern
+            var node = ts.cast(expr.parent, ts.isArrayLiteralExpression);
+            //    [{ property1: p1, property2 }] = elems;
+            var typeOfArrayLiteral = getTypeOfAssignmentPattern(node) || errorType;
+            var elementType = checkIteratedTypeOrElementType(65 /* Destructuring */, typeOfArrayLiteral, undefinedType, expr.parent) || errorType;
+            return checkArrayLiteralDestructuringElementAssignment(node, typeOfArrayLiteral, node.elements.indexOf(expr), elementType);
         }
-    }
-    function signatureHasRestParameter(s) {
-        return !!(s.flags & 1 /* HasRestParameter */);
-    }
-    ts.signatureHasRestParameter = signatureHasRestParameter;
-    function signatureHasLiteralTypes(s) {
-        return !!(s.flags & 2 /* HasLiteralTypes */);
-    }
-    ts.signatureHasLiteralTypes = signatureHasLiteralTypes;
-})(ts || (ts = {}));
-var ts;
-(function (ts) {
-    function createSynthesizedNode(kind) {
-        var node = ts.createNode(kind, -1, -1);
-        node.flags |= 8 /* Synthesized */;
-        return node;
-    }
-    /* @internal */
-    function updateNode(updated, original) {
-        if (updated !== original) {
-            setOriginalNode(updated, original);
-            setTextRange(updated, original);
-            ts.aggregateTransformFlags(updated);
+        // Gets the property symbol corresponding to the property in destructuring assignment
+        // 'property1' from
+        //     for ( { property1: a } of elems) {
+        //     }
+        // 'property1' at location 'a' from:
+        //     [a] = [ property1, property2 ]
+        function getPropertySymbolOfDestructuringAssignment(location) {
+            // Get the type of the object or array literal and then look for property of given name in the type
+            var typeOfObjectLiteral = getTypeOfAssignmentPattern(ts.cast(location.parent.parent, ts.isAssignmentPattern));
+            return typeOfObjectLiteral && getPropertyOfType(typeOfObjectLiteral, location.escapedText);
         }
-        return updated;
-    }
-    ts.updateNode = updateNode;
-    /**
-     * Make `elements` into a `NodeArray<T>`. If `elements` is `undefined`, returns an empty `NodeArray<T>`.
-     */
-    function createNodeArray(elements, hasTrailingComma) {
-        if (!elements || elements === ts.emptyArray) {
-            elements = [];
+        function getRegularTypeOfExpression(expr) {
+            if (ts.isRightSideOfQualifiedNameOrPropertyAccess(expr)) {
+                expr = expr.parent;
+            }
+            return getRegularTypeOfLiteralType(getTypeOfExpression(expr));
         }
-        else if (ts.isNodeArray(elements)) {
-            return elements;
+        /**
+         * Gets either the static or instance type of a class element, based on
+         * whether the element is declared as "static".
+         */
+        function getParentTypeOfClassElement(node) {
+            var classSymbol = getSymbolOfNode(node.parent);
+            return ts.hasSyntacticModifier(node, 32 /* Static */)
+                ? getTypeOfSymbol(classSymbol)
+                : getDeclaredTypeOfSymbol(classSymbol);
         }
-        var array = elements;
-        array.pos = -1;
-        array.end = -1;
-        array.hasTrailingComma = hasTrailingComma;
-        return array;
-    }
-    ts.createNodeArray = createNodeArray;
-    /**
-     * Creates a shallow, memberwise clone of a node with no source map location.
-     */
-    /* @internal */
-    function getSynthesizedClone(node) {
-        // We don't use "clone" from core.ts here, as we need to preserve the prototype chain of
-        // the original node. We also need to exclude specific properties and only include own-
-        // properties (to skip members already defined on the shared prototype).
-        if (node === undefined) {
-            return node;
+        function getClassElementPropertyKeyType(element) {
+            var name = element.name;
+            switch (name.kind) {
+                case 78 /* Identifier */:
+                    return getLiteralType(ts.idText(name));
+                case 8 /* NumericLiteral */:
+                case 10 /* StringLiteral */:
+                    return getLiteralType(name.text);
+                case 158 /* ComputedPropertyName */:
+                    var nameType = checkComputedPropertyName(name);
+                    return isTypeAssignableToKind(nameType, 12288 /* ESSymbolLike */) ? nameType : stringType;
+                default:
+                    return ts.Debug.fail("Unsupported property name.");
+            }
         }
-        var clone = createSynthesizedNode(node.kind);
-        clone.flags |= node.flags;
-        setOriginalNode(clone, node);
-        for (var key in node) {
-            if (clone.hasOwnProperty(key) || !node.hasOwnProperty(key)) {
-                continue;
+        // Return the list of properties of the given type, augmented with properties from Function
+        // if the type has call or construct signatures
+        function getAugmentedPropertiesOfType(type) {
+            type = getApparentType(type);
+            var propsByName = ts.createSymbolTable(getPropertiesOfType(type));
+            var functionType = getSignaturesOfType(type, 0 /* Call */).length ? globalCallableFunctionType :
+                getSignaturesOfType(type, 1 /* Construct */).length ? globalNewableFunctionType :
+                    undefined;
+            if (functionType) {
+                ts.forEach(getPropertiesOfType(functionType), function (p) {
+                    if (!propsByName.has(p.escapedName)) {
+                        propsByName.set(p.escapedName, p);
+                    }
+                });
             }
-            clone[key] = node[key];
-        }
-        return clone;
-    }
-    ts.getSynthesizedClone = getSynthesizedClone;
-    function createLiteral(value, isSingleQuote) {
-        if (typeof value === "number") {
-            return createNumericLiteral(value + "");
-        }
-        // eslint-disable-next-line no-in-operator
-        if (typeof value === "object" && "base10Value" in value) { // PseudoBigInt
-            return createBigIntLiteral(ts.pseudoBigIntToString(value) + "n");
-        }
-        if (typeof value === "boolean") {
-            return value ? createTrue() : createFalse();
+            return getNamedMembers(propsByName);
         }
-        if (ts.isString(value)) {
-            var res = createStringLiteral(value);
-            if (isSingleQuote)
-                res.singleQuote = true;
-            return res;
+        function typeHasCallOrConstructSignatures(type) {
+            return ts.typeHasCallOrConstructSignatures(type, checker);
         }
-        return createLiteralFromNode(value);
-    }
-    ts.createLiteral = createLiteral;
-    function createNumericLiteral(value, numericLiteralFlags) {
-        if (numericLiteralFlags === void 0) { numericLiteralFlags = 0 /* None */; }
-        var node = createSynthesizedNode(8 /* NumericLiteral */);
-        node.text = value;
-        node.numericLiteralFlags = numericLiteralFlags;
-        return node;
-    }
-    ts.createNumericLiteral = createNumericLiteral;
-    function createBigIntLiteral(value) {
-        var node = createSynthesizedNode(9 /* BigIntLiteral */);
-        node.text = value;
-        return node;
-    }
-    ts.createBigIntLiteral = createBigIntLiteral;
-    function createStringLiteral(text) {
-        var node = createSynthesizedNode(10 /* StringLiteral */);
-        node.text = text;
-        return node;
-    }
-    ts.createStringLiteral = createStringLiteral;
-    function createRegularExpressionLiteral(text) {
-        var node = createSynthesizedNode(13 /* RegularExpressionLiteral */);
-        node.text = text;
-        return node;
-    }
-    ts.createRegularExpressionLiteral = createRegularExpressionLiteral;
-    function createLiteralFromNode(sourceNode) {
-        var node = createStringLiteral(ts.getTextOfIdentifierOrLiteral(sourceNode));
-        node.textSourceNode = sourceNode;
-        return node;
-    }
-    function createIdentifier(text, typeArguments) {
-        var node = createSynthesizedNode(75 /* Identifier */);
-        node.escapedText = ts.escapeLeadingUnderscores(text);
-        node.originalKeywordKind = text ? ts.stringToToken(text) : 0 /* Unknown */;
-        node.autoGenerateFlags = 0 /* None */;
-        node.autoGenerateId = 0;
-        if (typeArguments) {
-            node.typeArguments = createNodeArray(typeArguments);
+        function getRootSymbols(symbol) {
+            var roots = getImmediateRootSymbols(symbol);
+            return roots ? ts.flatMap(roots, getRootSymbols) : [symbol];
         }
-        return node;
-    }
-    ts.createIdentifier = createIdentifier;
-    function updateIdentifier(node, typeArguments) {
-        return node.typeArguments !== typeArguments
-            ? updateNode(createIdentifier(ts.idText(node), typeArguments), node)
-            : node;
-    }
-    ts.updateIdentifier = updateIdentifier;
-    var nextAutoGenerateId = 0;
-    function createTempVariable(recordTempVariable, reservedInNestedScopes) {
-        var name = createIdentifier("");
-        name.autoGenerateFlags = 1 /* Auto */;
-        name.autoGenerateId = nextAutoGenerateId;
-        nextAutoGenerateId++;
-        if (recordTempVariable) {
-            recordTempVariable(name);
+        function getImmediateRootSymbols(symbol) {
+            if (ts.getCheckFlags(symbol) & 6 /* Synthetic */) {
+                return ts.mapDefined(getSymbolLinks(symbol).containingType.types, function (type) { return getPropertyOfType(type, symbol.escapedName); });
+            }
+            else if (symbol.flags & 33554432 /* Transient */) {
+                var _a = symbol, leftSpread = _a.leftSpread, rightSpread = _a.rightSpread, syntheticOrigin = _a.syntheticOrigin;
+                return leftSpread ? [leftSpread, rightSpread]
+                    : syntheticOrigin ? [syntheticOrigin]
+                        : ts.singleElementArray(tryGetAliasTarget(symbol));
+            }
+            return undefined;
         }
-        if (reservedInNestedScopes) {
-            name.autoGenerateFlags |= 8 /* ReservedInNestedScopes */;
+        function tryGetAliasTarget(symbol) {
+            var target;
+            var next = symbol;
+            while (next = getSymbolLinks(next).target) {
+                target = next;
+            }
+            return target;
         }
-        return name;
-    }
-    ts.createTempVariable = createTempVariable;
-    /** Create a unique temporary variable for use in a loop. */
-    function createLoopVariable() {
-        var name = createIdentifier("");
-        name.autoGenerateFlags = 2 /* Loop */;
-        name.autoGenerateId = nextAutoGenerateId;
-        nextAutoGenerateId++;
-        return name;
-    }
-    ts.createLoopVariable = createLoopVariable;
-    /** Create a unique name based on the supplied text. */
-    function createUniqueName(text) {
-        var name = createIdentifier(text);
-        name.autoGenerateFlags = 3 /* Unique */;
-        name.autoGenerateId = nextAutoGenerateId;
-        nextAutoGenerateId++;
-        return name;
-    }
-    ts.createUniqueName = createUniqueName;
-    function createOptimisticUniqueName(text) {
-        var name = createIdentifier(text);
-        name.autoGenerateFlags = 3 /* Unique */ | 16 /* Optimistic */;
-        name.autoGenerateId = nextAutoGenerateId;
-        nextAutoGenerateId++;
-        return name;
-    }
-    ts.createOptimisticUniqueName = createOptimisticUniqueName;
-    /** Create a unique name based on the supplied text. This does not consider names injected by the transformer. */
-    function createFileLevelUniqueName(text) {
-        var name = createOptimisticUniqueName(text);
-        name.autoGenerateFlags |= 32 /* FileLevel */;
-        return name;
-    }
-    ts.createFileLevelUniqueName = createFileLevelUniqueName;
-    function getGeneratedNameForNode(node, flags) {
-        var name = createIdentifier(node && ts.isIdentifier(node) ? ts.idText(node) : "");
-        name.autoGenerateFlags = 4 /* Node */ | flags;
-        name.autoGenerateId = nextAutoGenerateId;
-        name.original = node;
-        nextAutoGenerateId++;
-        return name;
-    }
-    ts.getGeneratedNameForNode = getGeneratedNameForNode;
-    // Private Identifiers
-    function createPrivateIdentifier(text) {
-        if (text[0] !== "#") {
-            ts.Debug.fail("First character of private identifier must be #: " + text);
+        // Emitter support
+        function isArgumentsLocalBinding(nodeIn) {
+            // Note: does not handle isShorthandPropertyAssignment (and probably a few more)
+            if (ts.isGeneratedIdentifier(nodeIn))
+                return false;
+            var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
+            if (!node)
+                return false;
+            var parent = node.parent;
+            if (!parent)
+                return false;
+            var isPropertyName = ((ts.isPropertyAccessExpression(parent)
+                || ts.isPropertyAssignment(parent))
+                && parent.name === node);
+            return !isPropertyName && getReferencedValueSymbol(node) === argumentsSymbol;
         }
-        var node = createSynthesizedNode(76 /* PrivateIdentifier */);
-        node.escapedText = ts.escapeLeadingUnderscores(text);
-        return node;
-    }
-    ts.createPrivateIdentifier = createPrivateIdentifier;
-    // Punctuation
-    function createToken(token) {
-        return createSynthesizedNode(token);
-    }
-    ts.createToken = createToken;
-    // Reserved words
-    function createSuper() {
-        return createSynthesizedNode(102 /* SuperKeyword */);
-    }
-    ts.createSuper = createSuper;
-    function createThis() {
-        return createSynthesizedNode(104 /* ThisKeyword */);
-    }
-    ts.createThis = createThis;
-    function createNull() {
-        return createSynthesizedNode(100 /* NullKeyword */);
-    }
-    ts.createNull = createNull;
-    function createTrue() {
-        return createSynthesizedNode(106 /* TrueKeyword */);
-    }
-    ts.createTrue = createTrue;
-    function createFalse() {
-        return createSynthesizedNode(91 /* FalseKeyword */);
-    }
-    ts.createFalse = createFalse;
-    // Modifiers
-    function createModifier(kind) {
-        return createToken(kind);
-    }
-    ts.createModifier = createModifier;
-    function createModifiersFromModifierFlags(flags) {
-        var result = [];
-        if (flags & 1 /* Export */) {
-            result.push(createModifier(89 /* ExportKeyword */));
+        function moduleExportsSomeValue(moduleReferenceExpression) {
+            var moduleSymbol = resolveExternalModuleName(moduleReferenceExpression.parent, moduleReferenceExpression);
+            if (!moduleSymbol || ts.isShorthandAmbientModuleSymbol(moduleSymbol)) {
+                // If the module is not found or is shorthand, assume that it may export a value.
+                return true;
+            }
+            var hasExportAssignment = hasExportAssignmentSymbol(moduleSymbol);
+            // if module has export assignment then 'resolveExternalModuleSymbol' will return resolved symbol for export assignment
+            // otherwise it will return moduleSymbol itself
+            moduleSymbol = resolveExternalModuleSymbol(moduleSymbol);
+            var symbolLinks = getSymbolLinks(moduleSymbol);
+            if (symbolLinks.exportsSomeValue === undefined) {
+                // for export assignments - check if resolved symbol for RHS is itself a value
+                // otherwise - check if at least one export is value
+                symbolLinks.exportsSomeValue = hasExportAssignment
+                    ? !!(moduleSymbol.flags & 111551 /* Value */)
+                    : ts.forEachEntry(getExportsOfModule(moduleSymbol), isValue);
+            }
+            return symbolLinks.exportsSomeValue;
+            function isValue(s) {
+                s = resolveSymbol(s);
+                return s && !!(s.flags & 111551 /* Value */);
+            }
         }
-        if (flags & 2 /* Ambient */) {
-            result.push(createModifier(130 /* DeclareKeyword */));
+        function isNameOfModuleOrEnumDeclaration(node) {
+            return ts.isModuleOrEnumDeclaration(node.parent) && node === node.parent.name;
         }
-        if (flags & 512 /* Default */) {
-            result.push(createModifier(84 /* DefaultKeyword */));
+        // When resolved as an expression identifier, if the given node references an exported entity, return the declaration
+        // node of the exported entity's container. Otherwise, return undefined.
+        function getReferencedExportContainer(nodeIn, prefixLocals) {
+            var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
+            if (node) {
+                // When resolving the export container for the name of a module or enum
+                // declaration, we need to start resolution at the declaration's container.
+                // Otherwise, we could incorrectly resolve the export container as the
+                // declaration if it contains an exported member with the same name.
+                var symbol = getReferencedValueSymbol(node, /*startInDeclarationContainer*/ isNameOfModuleOrEnumDeclaration(node));
+                if (symbol) {
+                    if (symbol.flags & 1048576 /* ExportValue */) {
+                        // If we reference an exported entity within the same module declaration, then whether
+                        // we prefix depends on the kind of entity. SymbolFlags.ExportHasLocal encompasses all the
+                        // kinds that we do NOT prefix.
+                        var exportSymbol = getMergedSymbol(symbol.exportSymbol);
+                        if (!prefixLocals && exportSymbol.flags & 944 /* ExportHasLocal */ && !(exportSymbol.flags & 3 /* Variable */)) {
+                            return undefined;
+                        }
+                        symbol = exportSymbol;
+                    }
+                    var parentSymbol_1 = getParentOfSymbol(symbol);
+                    if (parentSymbol_1) {
+                        if (parentSymbol_1.flags & 512 /* ValueModule */ && parentSymbol_1.valueDeclaration.kind === 297 /* SourceFile */) {
+                            var symbolFile = parentSymbol_1.valueDeclaration;
+                            var referenceFile = ts.getSourceFileOfNode(node);
+                            // If `node` accesses an export and that export isn't in the same file, then symbol is a namespace export, so return undefined.
+                            var symbolIsUmdExport = symbolFile !== referenceFile;
+                            return symbolIsUmdExport ? undefined : symbolFile;
+                        }
+                        return ts.findAncestor(node.parent, function (n) { return ts.isModuleOrEnumDeclaration(n) && getSymbolOfNode(n) === parentSymbol_1; });
+                    }
+                }
+            }
         }
-        if (flags & 2048 /* Const */) {
-            result.push(createModifier(81 /* ConstKeyword */));
+        // When resolved as an expression identifier, if the given node references an import, return the declaration of
+        // that import. Otherwise, return undefined.
+        function getReferencedImportDeclaration(nodeIn) {
+            if (nodeIn.generatedImportReference) {
+                return nodeIn.generatedImportReference;
+            }
+            var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
+            if (node) {
+                var symbol = getReferencedValueSymbol(node);
+                // We should only get the declaration of an alias if there isn't a local value
+                // declaration for the symbol
+                if (isNonLocalAlias(symbol, /*excludes*/ 111551 /* Value */) && !getTypeOnlyAliasDeclaration(symbol)) {
+                    return getDeclarationOfAliasSymbol(symbol);
+                }
+            }
+            return undefined;
         }
-        if (flags & 4 /* Public */) {
-            result.push(createModifier(119 /* PublicKeyword */));
+        function isSymbolOfDestructuredElementOfCatchBinding(symbol) {
+            return ts.isBindingElement(symbol.valueDeclaration)
+                && ts.walkUpBindingElementsAndPatterns(symbol.valueDeclaration).parent.kind === 287 /* CatchClause */;
         }
-        if (flags & 8 /* Private */) {
-            result.push(createModifier(117 /* PrivateKeyword */));
+        function isSymbolOfDeclarationWithCollidingName(symbol) {
+            if (symbol.flags & 418 /* BlockScoped */ && !ts.isSourceFile(symbol.valueDeclaration)) {
+                var links = getSymbolLinks(symbol);
+                if (links.isDeclarationWithCollidingName === undefined) {
+                    var container = ts.getEnclosingBlockScopeContainer(symbol.valueDeclaration);
+                    if (ts.isStatementWithLocals(container) || isSymbolOfDestructuredElementOfCatchBinding(symbol)) {
+                        var nodeLinks_1 = getNodeLinks(symbol.valueDeclaration);
+                        if (resolveName(container.parent, symbol.escapedName, 111551 /* Value */, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ false)) {
+                            // redeclaration - always should be renamed
+                            links.isDeclarationWithCollidingName = true;
+                        }
+                        else if (nodeLinks_1.flags & 262144 /* CapturedBlockScopedBinding */) {
+                            // binding is captured in the function
+                            // should be renamed if:
+                            // - binding is not top level - top level bindings never collide with anything
+                            // AND
+                            //   - binding is not declared in loop, should be renamed to avoid name reuse across siblings
+                            //     let a, b
+                            //     { let x = 1; a = () => x; }
+                            //     { let x = 100; b = () => x; }
+                            //     console.log(a()); // should print '1'
+                            //     console.log(b()); // should print '100'
+                            //     OR
+                            //   - binding is declared inside loop but not in inside initializer of iteration statement or directly inside loop body
+                            //     * variables from initializer are passed to rewritten loop body as parameters so they are not captured directly
+                            //     * variables that are declared immediately in loop body will become top level variable after loop is rewritten and thus
+                            //       they will not collide with anything
+                            var isDeclaredInLoop = nodeLinks_1.flags & 524288 /* BlockScopedBindingInLoop */;
+                            var inLoopInitializer = ts.isIterationStatement(container, /*lookInLabeledStatements*/ false);
+                            var inLoopBodyBlock = container.kind === 230 /* Block */ && ts.isIterationStatement(container.parent, /*lookInLabeledStatements*/ false);
+                            links.isDeclarationWithCollidingName = !ts.isBlockScopedContainerTopLevel(container) && (!isDeclaredInLoop || (!inLoopInitializer && !inLoopBodyBlock));
+                        }
+                        else {
+                            links.isDeclarationWithCollidingName = false;
+                        }
+                    }
+                }
+                return links.isDeclarationWithCollidingName;
+            }
+            return false;
         }
-        if (flags & 16 /* Protected */) {
-            result.push(createModifier(118 /* ProtectedKeyword */));
+        // When resolved as an expression identifier, if the given node references a nested block scoped entity with
+        // a name that either hides an existing name or might hide it when compiled downlevel,
+        // return the declaration of that entity. Otherwise, return undefined.
+        function getReferencedDeclarationWithCollidingName(nodeIn) {
+            if (!ts.isGeneratedIdentifier(nodeIn)) {
+                var node = ts.getParseTreeNode(nodeIn, ts.isIdentifier);
+                if (node) {
+                    var symbol = getReferencedValueSymbol(node);
+                    if (symbol && isSymbolOfDeclarationWithCollidingName(symbol)) {
+                        return symbol.valueDeclaration;
+                    }
+                }
+            }
+            return undefined;
         }
-        if (flags & 128 /* Abstract */) {
-            result.push(createModifier(122 /* AbstractKeyword */));
+        // Return true if the given node is a declaration of a nested block scoped entity with a name that either hides an
+        // existing name or might hide a name when compiled downlevel
+        function isDeclarationWithCollidingName(nodeIn) {
+            var node = ts.getParseTreeNode(nodeIn, ts.isDeclaration);
+            if (node) {
+                var symbol = getSymbolOfNode(node);
+                if (symbol) {
+                    return isSymbolOfDeclarationWithCollidingName(symbol);
+                }
+            }
+            return false;
         }
-        if (flags & 32 /* Static */) {
-            result.push(createModifier(120 /* StaticKeyword */));
+        function isValueAliasDeclaration(node) {
+            switch (node.kind) {
+                case 260 /* ImportEqualsDeclaration */:
+                    return isAliasResolvedToValue(getSymbolOfNode(node) || unknownSymbol);
+                case 262 /* ImportClause */:
+                case 263 /* NamespaceImport */:
+                case 265 /* ImportSpecifier */:
+                case 270 /* ExportSpecifier */:
+                    var symbol = getSymbolOfNode(node) || unknownSymbol;
+                    return isAliasResolvedToValue(symbol) && !getTypeOnlyAliasDeclaration(symbol);
+                case 267 /* ExportDeclaration */:
+                    var exportClause = node.exportClause;
+                    return !!exportClause && (ts.isNamespaceExport(exportClause) ||
+                        ts.some(exportClause.elements, isValueAliasDeclaration));
+                case 266 /* ExportAssignment */:
+                    return node.expression && node.expression.kind === 78 /* Identifier */ ?
+                        isAliasResolvedToValue(getSymbolOfNode(node) || unknownSymbol) :
+                        true;
+            }
+            return false;
         }
-        if (flags & 64 /* Readonly */) {
-            result.push(createModifier(138 /* ReadonlyKeyword */));
+        function isTopLevelValueImportEqualsWithEntityName(nodeIn) {
+            var node = ts.getParseTreeNode(nodeIn, ts.isImportEqualsDeclaration);
+            if (node === undefined || node.parent.kind !== 297 /* SourceFile */ || !ts.isInternalModuleImportEqualsDeclaration(node)) {
+                // parent is not source file or it is not reference to internal module
+                return false;
+            }
+            var isValue = isAliasResolvedToValue(getSymbolOfNode(node));
+            return isValue && node.moduleReference && !ts.nodeIsMissing(node.moduleReference);
         }
-        if (flags & 256 /* Async */) {
-            result.push(createModifier(126 /* AsyncKeyword */));
+        function isAliasResolvedToValue(symbol) {
+            var target = resolveAlias(symbol);
+            if (target === unknownSymbol) {
+                return true;
+            }
+            // const enums and modules that contain only const enums are not considered values from the emit perspective
+            // unless 'preserveConstEnums' option is set to true
+            return !!(target.flags & 111551 /* Value */) &&
+                (compilerOptions.preserveConstEnums || !isConstEnumOrConstEnumOnlyModule(target));
         }
-        return result;
-    }
-    ts.createModifiersFromModifierFlags = createModifiersFromModifierFlags;
-    // Names
-    function createQualifiedName(left, right) {
-        var node = createSynthesizedNode(153 /* QualifiedName */);
-        node.left = left;
-        node.right = asName(right);
-        return node;
-    }
-    ts.createQualifiedName = createQualifiedName;
-    function updateQualifiedName(node, left, right) {
-        return node.left !== left
-            || node.right !== right
-            ? updateNode(createQualifiedName(left, right), node)
-            : node;
-    }
-    ts.updateQualifiedName = updateQualifiedName;
-    function parenthesizeForComputedName(expression) {
-        return ts.isCommaSequence(expression)
-            ? createParen(expression)
-            : expression;
-    }
-    function createComputedPropertyName(expression) {
-        var node = createSynthesizedNode(154 /* ComputedPropertyName */);
-        node.expression = parenthesizeForComputedName(expression);
-        return node;
-    }
-    ts.createComputedPropertyName = createComputedPropertyName;
-    function updateComputedPropertyName(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createComputedPropertyName(expression), node)
-            : node;
-    }
-    ts.updateComputedPropertyName = updateComputedPropertyName;
-    // Signature elements
-    function createTypeParameterDeclaration(name, constraint, defaultType) {
-        var node = createSynthesizedNode(155 /* TypeParameter */);
-        node.name = asName(name);
-        node.constraint = constraint;
-        node.default = defaultType;
-        return node;
-    }
-    ts.createTypeParameterDeclaration = createTypeParameterDeclaration;
-    function updateTypeParameterDeclaration(node, name, constraint, defaultType) {
-        return node.name !== name
-            || node.constraint !== constraint
-            || node.default !== defaultType
-            ? updateNode(createTypeParameterDeclaration(name, constraint, defaultType), node)
-            : node;
-    }
-    ts.updateTypeParameterDeclaration = updateTypeParameterDeclaration;
-    function createParameter(decorators, modifiers, dotDotDotToken, name, questionToken, type, initializer) {
-        var node = createSynthesizedNode(156 /* Parameter */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.dotDotDotToken = dotDotDotToken;
-        node.name = asName(name);
-        node.questionToken = questionToken;
-        node.type = type;
-        node.initializer = initializer ? ts.parenthesizeExpressionForList(initializer) : undefined;
-        return node;
-    }
-    ts.createParameter = createParameter;
-    function updateParameter(node, decorators, modifiers, dotDotDotToken, name, questionToken, type, initializer) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.dotDotDotToken !== dotDotDotToken
-            || node.name !== name
-            || node.questionToken !== questionToken
-            || node.type !== type
-            || node.initializer !== initializer
-            ? updateNode(createParameter(decorators, modifiers, dotDotDotToken, name, questionToken, type, initializer), node)
-            : node;
-    }
-    ts.updateParameter = updateParameter;
-    function createDecorator(expression) {
-        var node = createSynthesizedNode(157 /* Decorator */);
-        node.expression = ts.parenthesizeForAccess(expression);
-        return node;
-    }
-    ts.createDecorator = createDecorator;
-    function updateDecorator(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createDecorator(expression), node)
-            : node;
-    }
-    ts.updateDecorator = updateDecorator;
-    // Type Elements
-    function createPropertySignature(modifiers, name, questionToken, type, initializer) {
-        var node = createSynthesizedNode(158 /* PropertySignature */);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.questionToken = questionToken;
-        node.type = type;
-        node.initializer = initializer;
-        return node;
-    }
-    ts.createPropertySignature = createPropertySignature;
-    function updatePropertySignature(node, modifiers, name, questionToken, type, initializer) {
-        return node.modifiers !== modifiers
-            || node.name !== name
-            || node.questionToken !== questionToken
-            || node.type !== type
-            || node.initializer !== initializer
-            ? updateNode(createPropertySignature(modifiers, name, questionToken, type, initializer), node)
-            : node;
-    }
-    ts.updatePropertySignature = updatePropertySignature;
-    function createProperty(decorators, modifiers, name, questionOrExclamationToken, type, initializer) {
-        var node = createSynthesizedNode(159 /* PropertyDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.questionToken = questionOrExclamationToken !== undefined && questionOrExclamationToken.kind === 57 /* QuestionToken */ ? questionOrExclamationToken : undefined;
-        node.exclamationToken = questionOrExclamationToken !== undefined && questionOrExclamationToken.kind === 53 /* ExclamationToken */ ? questionOrExclamationToken : undefined;
-        node.type = type;
-        node.initializer = initializer;
-        return node;
-    }
-    ts.createProperty = createProperty;
-    function updateProperty(node, decorators, modifiers, name, questionOrExclamationToken, type, initializer) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.questionToken !== (questionOrExclamationToken !== undefined && questionOrExclamationToken.kind === 57 /* QuestionToken */ ? questionOrExclamationToken : undefined)
-            || node.exclamationToken !== (questionOrExclamationToken !== undefined && questionOrExclamationToken.kind === 53 /* ExclamationToken */ ? questionOrExclamationToken : undefined)
-            || node.type !== type
-            || node.initializer !== initializer
-            ? updateNode(createProperty(decorators, modifiers, name, questionOrExclamationToken, type, initializer), node)
-            : node;
-    }
-    ts.updateProperty = updateProperty;
-    function createMethodSignature(typeParameters, parameters, type, name, questionToken) {
-        var node = createSignatureDeclaration(160 /* MethodSignature */, typeParameters, parameters, type);
-        node.name = asName(name);
-        node.questionToken = questionToken;
-        return node;
-    }
-    ts.createMethodSignature = createMethodSignature;
-    function updateMethodSignature(node, typeParameters, parameters, type, name, questionToken) {
-        return node.typeParameters !== typeParameters
-            || node.parameters !== parameters
-            || node.type !== type
-            || node.name !== name
-            || node.questionToken !== questionToken
-            ? updateNode(createMethodSignature(typeParameters, parameters, type, name, questionToken), node)
-            : node;
-    }
-    ts.updateMethodSignature = updateMethodSignature;
-    function createMethod(decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body) {
-        var node = createSynthesizedNode(161 /* MethodDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.asteriskToken = asteriskToken;
-        node.name = asName(name);
-        node.questionToken = questionToken;
-        node.typeParameters = asNodeArray(typeParameters);
-        node.parameters = createNodeArray(parameters);
-        node.type = type;
-        node.body = body;
-        return node;
-    }
-    ts.createMethod = createMethod;
-    function createMethodCall(object, methodName, argumentsList) {
-        return createCall(createPropertyAccess(object, asName(methodName)), 
-        /*typeArguments*/ undefined, argumentsList);
-    }
-    function createGlobalMethodCall(globalObjectName, methodName, argumentsList) {
-        return createMethodCall(createIdentifier(globalObjectName), methodName, argumentsList);
-    }
-    /* @internal */
-    function createObjectDefinePropertyCall(target, propertyName, attributes) {
-        return createGlobalMethodCall("Object", "defineProperty", [target, asExpression(propertyName), attributes]);
-    }
-    ts.createObjectDefinePropertyCall = createObjectDefinePropertyCall;
-    function tryAddPropertyAssignment(properties, propertyName, expression) {
-        if (expression) {
-            properties.push(createPropertyAssignment(propertyName, expression));
-            return true;
+        function isConstEnumOrConstEnumOnlyModule(s) {
+            return isConstEnumSymbol(s) || !!s.constEnumOnlyModule;
         }
-        return false;
-    }
-    /* @internal */
-    function createPropertyDescriptor(attributes, singleLine) {
-        var properties = [];
-        tryAddPropertyAssignment(properties, "enumerable", asExpression(attributes.enumerable));
-        tryAddPropertyAssignment(properties, "configurable", asExpression(attributes.configurable));
-        var isData = tryAddPropertyAssignment(properties, "writable", asExpression(attributes.writable));
-        isData = tryAddPropertyAssignment(properties, "value", attributes.value) || isData;
-        var isAccessor = tryAddPropertyAssignment(properties, "get", attributes.get);
-        isAccessor = tryAddPropertyAssignment(properties, "set", attributes.set) || isAccessor;
-        ts.Debug.assert(!(isData && isAccessor), "A PropertyDescriptor may not be both an accessor descriptor and a data descriptor.");
-        return createObjectLiteral(properties, !singleLine);
-    }
-    ts.createPropertyDescriptor = createPropertyDescriptor;
-    function updateMethod(node, decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.asteriskToken !== asteriskToken
-            || node.name !== name
-            || node.questionToken !== questionToken
-            || node.typeParameters !== typeParameters
-            || node.parameters !== parameters
-            || node.type !== type
-            || node.body !== body
-            ? updateNode(createMethod(decorators, modifiers, asteriskToken, name, questionToken, typeParameters, parameters, type, body), node)
-            : node;
-    }
-    ts.updateMethod = updateMethod;
-    function createConstructor(decorators, modifiers, parameters, body) {
-        var node = createSynthesizedNode(162 /* Constructor */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.typeParameters = undefined;
-        node.parameters = createNodeArray(parameters);
-        node.type = undefined;
-        node.body = body;
-        return node;
-    }
-    ts.createConstructor = createConstructor;
-    function updateConstructor(node, decorators, modifiers, parameters, body) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.parameters !== parameters
-            || node.body !== body
-            ? updateNode(createConstructor(decorators, modifiers, parameters, body), node)
-            : node;
-    }
-    ts.updateConstructor = updateConstructor;
-    function createGetAccessor(decorators, modifiers, name, parameters, type, body) {
-        var node = createSynthesizedNode(163 /* GetAccessor */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.typeParameters = undefined;
-        node.parameters = createNodeArray(parameters);
-        node.type = type;
-        node.body = body;
-        return node;
-    }
-    ts.createGetAccessor = createGetAccessor;
-    function updateGetAccessor(node, decorators, modifiers, name, parameters, type, body) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.parameters !== parameters
-            || node.type !== type
-            || node.body !== body
-            ? updateNode(createGetAccessor(decorators, modifiers, name, parameters, type, body), node)
-            : node;
-    }
-    ts.updateGetAccessor = updateGetAccessor;
-    function createSetAccessor(decorators, modifiers, name, parameters, body) {
-        var node = createSynthesizedNode(164 /* SetAccessor */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.typeParameters = undefined;
-        node.parameters = createNodeArray(parameters);
-        node.body = body;
-        return node;
-    }
-    ts.createSetAccessor = createSetAccessor;
-    function updateSetAccessor(node, decorators, modifiers, name, parameters, body) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.parameters !== parameters
-            || node.body !== body
-            ? updateNode(createSetAccessor(decorators, modifiers, name, parameters, body), node)
-            : node;
-    }
-    ts.updateSetAccessor = updateSetAccessor;
-    function createCallSignature(typeParameters, parameters, type) {
-        return createSignatureDeclaration(165 /* CallSignature */, typeParameters, parameters, type);
-    }
-    ts.createCallSignature = createCallSignature;
-    function updateCallSignature(node, typeParameters, parameters, type) {
-        return updateSignatureDeclaration(node, typeParameters, parameters, type);
-    }
-    ts.updateCallSignature = updateCallSignature;
-    function createConstructSignature(typeParameters, parameters, type) {
-        return createSignatureDeclaration(166 /* ConstructSignature */, typeParameters, parameters, type);
-    }
-    ts.createConstructSignature = createConstructSignature;
-    function updateConstructSignature(node, typeParameters, parameters, type) {
-        return updateSignatureDeclaration(node, typeParameters, parameters, type);
-    }
-    ts.updateConstructSignature = updateConstructSignature;
-    function createIndexSignature(decorators, modifiers, parameters, type) {
-        var node = createSynthesizedNode(167 /* IndexSignature */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.parameters = createNodeArray(parameters);
-        node.type = type;
-        return node;
-    }
-    ts.createIndexSignature = createIndexSignature;
-    function updateIndexSignature(node, decorators, modifiers, parameters, type) {
-        return node.parameters !== parameters
-            || node.type !== type
-            || node.decorators !== decorators
-            || node.modifiers !== modifiers
-            ? updateNode(createIndexSignature(decorators, modifiers, parameters, type), node)
-            : node;
-    }
-    ts.updateIndexSignature = updateIndexSignature;
-    /* @internal */
-    function createSignatureDeclaration(kind, typeParameters, parameters, type, typeArguments) {
-        var node = createSynthesizedNode(kind);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.parameters = asNodeArray(parameters);
-        node.type = type;
-        node.typeArguments = asNodeArray(typeArguments);
-        return node;
-    }
-    ts.createSignatureDeclaration = createSignatureDeclaration;
-    function updateSignatureDeclaration(node, typeParameters, parameters, type) {
-        return node.typeParameters !== typeParameters
-            || node.parameters !== parameters
-            || node.type !== type
-            ? updateNode(createSignatureDeclaration(node.kind, typeParameters, parameters, type), node)
-            : node;
-    }
-    // Types
-    function createKeywordTypeNode(kind) {
-        return createSynthesizedNode(kind);
-    }
-    ts.createKeywordTypeNode = createKeywordTypeNode;
-    function createTypePredicateNode(parameterName, type) {
-        return createTypePredicateNodeWithModifier(/*assertsModifier*/ undefined, parameterName, type);
-    }
-    ts.createTypePredicateNode = createTypePredicateNode;
-    function createTypePredicateNodeWithModifier(assertsModifier, parameterName, type) {
-        var node = createSynthesizedNode(168 /* TypePredicate */);
-        node.assertsModifier = assertsModifier;
-        node.parameterName = asName(parameterName);
-        node.type = type;
-        return node;
-    }
-    ts.createTypePredicateNodeWithModifier = createTypePredicateNodeWithModifier;
-    function updateTypePredicateNode(node, parameterName, type) {
-        return updateTypePredicateNodeWithModifier(node, node.assertsModifier, parameterName, type);
-    }
-    ts.updateTypePredicateNode = updateTypePredicateNode;
-    function updateTypePredicateNodeWithModifier(node, assertsModifier, parameterName, type) {
-        return node.assertsModifier !== assertsModifier
-            || node.parameterName !== parameterName
-            || node.type !== type
-            ? updateNode(createTypePredicateNodeWithModifier(assertsModifier, parameterName, type), node)
-            : node;
-    }
-    ts.updateTypePredicateNodeWithModifier = updateTypePredicateNodeWithModifier;
-    function createTypeReferenceNode(typeName, typeArguments) {
-        var node = createSynthesizedNode(169 /* TypeReference */);
-        node.typeName = asName(typeName);
-        node.typeArguments = typeArguments && ts.parenthesizeTypeParameters(typeArguments);
-        return node;
-    }
-    ts.createTypeReferenceNode = createTypeReferenceNode;
-    function updateTypeReferenceNode(node, typeName, typeArguments) {
-        return node.typeName !== typeName
-            || node.typeArguments !== typeArguments
-            ? updateNode(createTypeReferenceNode(typeName, typeArguments), node)
-            : node;
-    }
-    ts.updateTypeReferenceNode = updateTypeReferenceNode;
-    function createFunctionTypeNode(typeParameters, parameters, type) {
-        return createSignatureDeclaration(170 /* FunctionType */, typeParameters, parameters, type);
-    }
-    ts.createFunctionTypeNode = createFunctionTypeNode;
-    function updateFunctionTypeNode(node, typeParameters, parameters, type) {
-        return updateSignatureDeclaration(node, typeParameters, parameters, type);
-    }
-    ts.updateFunctionTypeNode = updateFunctionTypeNode;
-    function createConstructorTypeNode(typeParameters, parameters, type) {
-        return createSignatureDeclaration(171 /* ConstructorType */, typeParameters, parameters, type);
-    }
-    ts.createConstructorTypeNode = createConstructorTypeNode;
-    function updateConstructorTypeNode(node, typeParameters, parameters, type) {
-        return updateSignatureDeclaration(node, typeParameters, parameters, type);
-    }
-    ts.updateConstructorTypeNode = updateConstructorTypeNode;
-    function createTypeQueryNode(exprName) {
-        var node = createSynthesizedNode(172 /* TypeQuery */);
-        node.exprName = exprName;
-        return node;
-    }
-    ts.createTypeQueryNode = createTypeQueryNode;
-    function updateTypeQueryNode(node, exprName) {
-        return node.exprName !== exprName
-            ? updateNode(createTypeQueryNode(exprName), node)
-            : node;
-    }
-    ts.updateTypeQueryNode = updateTypeQueryNode;
-    function createTypeLiteralNode(members) {
-        var node = createSynthesizedNode(173 /* TypeLiteral */);
-        node.members = createNodeArray(members);
-        return node;
-    }
-    ts.createTypeLiteralNode = createTypeLiteralNode;
-    function updateTypeLiteralNode(node, members) {
-        return node.members !== members
-            ? updateNode(createTypeLiteralNode(members), node)
-            : node;
-    }
-    ts.updateTypeLiteralNode = updateTypeLiteralNode;
-    function createArrayTypeNode(elementType) {
-        var node = createSynthesizedNode(174 /* ArrayType */);
-        node.elementType = ts.parenthesizeArrayTypeMember(elementType);
-        return node;
-    }
-    ts.createArrayTypeNode = createArrayTypeNode;
-    function updateArrayTypeNode(node, elementType) {
-        return node.elementType !== elementType
-            ? updateNode(createArrayTypeNode(elementType), node)
-            : node;
-    }
-    ts.updateArrayTypeNode = updateArrayTypeNode;
-    function createTupleTypeNode(elementTypes) {
-        var node = createSynthesizedNode(175 /* TupleType */);
-        node.elementTypes = createNodeArray(elementTypes);
-        return node;
-    }
-    ts.createTupleTypeNode = createTupleTypeNode;
-    function updateTupleTypeNode(node, elementTypes) {
-        return node.elementTypes !== elementTypes
-            ? updateNode(createTupleTypeNode(elementTypes), node)
-            : node;
-    }
-    ts.updateTupleTypeNode = updateTupleTypeNode;
-    function createOptionalTypeNode(type) {
-        var node = createSynthesizedNode(176 /* OptionalType */);
-        node.type = ts.parenthesizeArrayTypeMember(type);
-        return node;
-    }
-    ts.createOptionalTypeNode = createOptionalTypeNode;
-    function updateOptionalTypeNode(node, type) {
-        return node.type !== type
-            ? updateNode(createOptionalTypeNode(type), node)
-            : node;
-    }
-    ts.updateOptionalTypeNode = updateOptionalTypeNode;
-    function createRestTypeNode(type) {
-        var node = createSynthesizedNode(177 /* RestType */);
-        node.type = type;
-        return node;
-    }
-    ts.createRestTypeNode = createRestTypeNode;
-    function updateRestTypeNode(node, type) {
-        return node.type !== type
-            ? updateNode(createRestTypeNode(type), node)
-            : node;
-    }
-    ts.updateRestTypeNode = updateRestTypeNode;
-    function createUnionTypeNode(types) {
-        return createUnionOrIntersectionTypeNode(178 /* UnionType */, types);
-    }
-    ts.createUnionTypeNode = createUnionTypeNode;
-    function updateUnionTypeNode(node, types) {
-        return updateUnionOrIntersectionTypeNode(node, types);
-    }
-    ts.updateUnionTypeNode = updateUnionTypeNode;
-    function createIntersectionTypeNode(types) {
-        return createUnionOrIntersectionTypeNode(179 /* IntersectionType */, types);
-    }
-    ts.createIntersectionTypeNode = createIntersectionTypeNode;
-    function updateIntersectionTypeNode(node, types) {
-        return updateUnionOrIntersectionTypeNode(node, types);
-    }
-    ts.updateIntersectionTypeNode = updateIntersectionTypeNode;
-    function createUnionOrIntersectionTypeNode(kind, types) {
-        var node = createSynthesizedNode(kind);
-        node.types = ts.parenthesizeElementTypeMembers(types);
-        return node;
-    }
-    ts.createUnionOrIntersectionTypeNode = createUnionOrIntersectionTypeNode;
-    function updateUnionOrIntersectionTypeNode(node, types) {
-        return node.types !== types
-            ? updateNode(createUnionOrIntersectionTypeNode(node.kind, types), node)
-            : node;
-    }
-    function createConditionalTypeNode(checkType, extendsType, trueType, falseType) {
-        var node = createSynthesizedNode(180 /* ConditionalType */);
-        node.checkType = ts.parenthesizeConditionalTypeMember(checkType);
-        node.extendsType = ts.parenthesizeConditionalTypeMember(extendsType);
-        node.trueType = trueType;
-        node.falseType = falseType;
-        return node;
-    }
-    ts.createConditionalTypeNode = createConditionalTypeNode;
-    function updateConditionalTypeNode(node, checkType, extendsType, trueType, falseType) {
-        return node.checkType !== checkType
-            || node.extendsType !== extendsType
-            || node.trueType !== trueType
-            || node.falseType !== falseType
-            ? updateNode(createConditionalTypeNode(checkType, extendsType, trueType, falseType), node)
-            : node;
-    }
-    ts.updateConditionalTypeNode = updateConditionalTypeNode;
-    function createInferTypeNode(typeParameter) {
-        var node = createSynthesizedNode(181 /* InferType */);
-        node.typeParameter = typeParameter;
-        return node;
-    }
-    ts.createInferTypeNode = createInferTypeNode;
-    function updateInferTypeNode(node, typeParameter) {
-        return node.typeParameter !== typeParameter
-            ? updateNode(createInferTypeNode(typeParameter), node)
-            : node;
-    }
-    ts.updateInferTypeNode = updateInferTypeNode;
-    function createImportTypeNode(argument, qualifier, typeArguments, isTypeOf) {
-        var node = createSynthesizedNode(188 /* ImportType */);
-        node.argument = argument;
-        node.qualifier = qualifier;
-        node.typeArguments = ts.parenthesizeTypeParameters(typeArguments);
-        node.isTypeOf = isTypeOf;
-        return node;
-    }
-    ts.createImportTypeNode = createImportTypeNode;
-    function updateImportTypeNode(node, argument, qualifier, typeArguments, isTypeOf) {
-        return node.argument !== argument
-            || node.qualifier !== qualifier
-            || node.typeArguments !== typeArguments
-            || node.isTypeOf !== isTypeOf
-            ? updateNode(createImportTypeNode(argument, qualifier, typeArguments, isTypeOf), node)
-            : node;
-    }
-    ts.updateImportTypeNode = updateImportTypeNode;
-    function createParenthesizedType(type) {
-        var node = createSynthesizedNode(182 /* ParenthesizedType */);
-        node.type = type;
-        return node;
-    }
-    ts.createParenthesizedType = createParenthesizedType;
-    function updateParenthesizedType(node, type) {
-        return node.type !== type
-            ? updateNode(createParenthesizedType(type), node)
-            : node;
-    }
-    ts.updateParenthesizedType = updateParenthesizedType;
-    function createThisTypeNode() {
-        return createSynthesizedNode(183 /* ThisType */);
-    }
-    ts.createThisTypeNode = createThisTypeNode;
-    function createTypeOperatorNode(operatorOrType, type) {
-        var node = createSynthesizedNode(184 /* TypeOperator */);
-        node.operator = typeof operatorOrType === "number" ? operatorOrType : 134 /* KeyOfKeyword */;
-        node.type = ts.parenthesizeElementTypeMember(typeof operatorOrType === "number" ? type : operatorOrType);
-        return node;
-    }
-    ts.createTypeOperatorNode = createTypeOperatorNode;
-    function updateTypeOperatorNode(node, type) {
-        return node.type !== type ? updateNode(createTypeOperatorNode(node.operator, type), node) : node;
-    }
-    ts.updateTypeOperatorNode = updateTypeOperatorNode;
-    function createIndexedAccessTypeNode(objectType, indexType) {
-        var node = createSynthesizedNode(185 /* IndexedAccessType */);
-        node.objectType = ts.parenthesizeElementTypeMember(objectType);
-        node.indexType = indexType;
-        return node;
-    }
-    ts.createIndexedAccessTypeNode = createIndexedAccessTypeNode;
-    function updateIndexedAccessTypeNode(node, objectType, indexType) {
-        return node.objectType !== objectType
-            || node.indexType !== indexType
-            ? updateNode(createIndexedAccessTypeNode(objectType, indexType), node)
-            : node;
-    }
-    ts.updateIndexedAccessTypeNode = updateIndexedAccessTypeNode;
-    function createMappedTypeNode(readonlyToken, typeParameter, questionToken, type) {
-        var node = createSynthesizedNode(186 /* MappedType */);
-        node.readonlyToken = readonlyToken;
-        node.typeParameter = typeParameter;
-        node.questionToken = questionToken;
-        node.type = type;
-        return node;
-    }
-    ts.createMappedTypeNode = createMappedTypeNode;
-    function updateMappedTypeNode(node, readonlyToken, typeParameter, questionToken, type) {
-        return node.readonlyToken !== readonlyToken
-            || node.typeParameter !== typeParameter
-            || node.questionToken !== questionToken
-            || node.type !== type
-            ? updateNode(createMappedTypeNode(readonlyToken, typeParameter, questionToken, type), node)
-            : node;
-    }
-    ts.updateMappedTypeNode = updateMappedTypeNode;
-    function createLiteralTypeNode(literal) {
-        var node = createSynthesizedNode(187 /* LiteralType */);
-        node.literal = literal;
-        return node;
-    }
-    ts.createLiteralTypeNode = createLiteralTypeNode;
-    function updateLiteralTypeNode(node, literal) {
-        return node.literal !== literal
-            ? updateNode(createLiteralTypeNode(literal), node)
-            : node;
-    }
-    ts.updateLiteralTypeNode = updateLiteralTypeNode;
-    // Binding Patterns
-    function createObjectBindingPattern(elements) {
-        var node = createSynthesizedNode(189 /* ObjectBindingPattern */);
-        node.elements = createNodeArray(elements);
-        return node;
-    }
-    ts.createObjectBindingPattern = createObjectBindingPattern;
-    function updateObjectBindingPattern(node, elements) {
-        return node.elements !== elements
-            ? updateNode(createObjectBindingPattern(elements), node)
-            : node;
-    }
-    ts.updateObjectBindingPattern = updateObjectBindingPattern;
-    function createArrayBindingPattern(elements) {
-        var node = createSynthesizedNode(190 /* ArrayBindingPattern */);
-        node.elements = createNodeArray(elements);
-        return node;
-    }
-    ts.createArrayBindingPattern = createArrayBindingPattern;
-    function updateArrayBindingPattern(node, elements) {
-        return node.elements !== elements
-            ? updateNode(createArrayBindingPattern(elements), node)
-            : node;
-    }
-    ts.updateArrayBindingPattern = updateArrayBindingPattern;
-    function createBindingElement(dotDotDotToken, propertyName, name, initializer) {
-        var node = createSynthesizedNode(191 /* BindingElement */);
-        node.dotDotDotToken = dotDotDotToken;
-        node.propertyName = asName(propertyName);
-        node.name = asName(name);
-        node.initializer = initializer;
-        return node;
-    }
-    ts.createBindingElement = createBindingElement;
-    function updateBindingElement(node, dotDotDotToken, propertyName, name, initializer) {
-        return node.propertyName !== propertyName
-            || node.dotDotDotToken !== dotDotDotToken
-            || node.name !== name
-            || node.initializer !== initializer
-            ? updateNode(createBindingElement(dotDotDotToken, propertyName, name, initializer), node)
-            : node;
-    }
-    ts.updateBindingElement = updateBindingElement;
-    // Expression
-    function createArrayLiteral(elements, multiLine) {
-        var node = createSynthesizedNode(192 /* ArrayLiteralExpression */);
-        node.elements = ts.parenthesizeListElements(createNodeArray(elements));
-        if (multiLine)
-            node.multiLine = true;
-        return node;
-    }
-    ts.createArrayLiteral = createArrayLiteral;
-    function updateArrayLiteral(node, elements) {
-        return node.elements !== elements
-            ? updateNode(createArrayLiteral(elements, node.multiLine), node)
-            : node;
-    }
-    ts.updateArrayLiteral = updateArrayLiteral;
-    function createObjectLiteral(properties, multiLine) {
-        var node = createSynthesizedNode(193 /* ObjectLiteralExpression */);
-        node.properties = createNodeArray(properties);
-        if (multiLine)
-            node.multiLine = true;
-        return node;
-    }
-    ts.createObjectLiteral = createObjectLiteral;
-    function updateObjectLiteral(node, properties) {
-        return node.properties !== properties
-            ? updateNode(createObjectLiteral(properties, node.multiLine), node)
-            : node;
-    }
-    ts.updateObjectLiteral = updateObjectLiteral;
-    function createPropertyAccess(expression, name) {
-        var node = createSynthesizedNode(194 /* PropertyAccessExpression */);
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.name = asName(name);
-        setEmitFlags(node, 131072 /* NoIndentation */);
-        return node;
-    }
-    ts.createPropertyAccess = createPropertyAccess;
-    function updatePropertyAccess(node, expression, name) {
-        if (ts.isPropertyAccessChain(node)) {
-            return updatePropertyAccessChain(node, expression, node.questionDotToken, ts.cast(name, ts.isIdentifier));
+        function isReferencedAliasDeclaration(node, checkChildren) {
+            if (isAliasSymbolDeclaration(node)) {
+                var symbol = getSymbolOfNode(node);
+                if (symbol && getSymbolLinks(symbol).referenced) {
+                    return true;
+                }
+                var target = getSymbolLinks(symbol).target; // TODO: GH#18217
+                if (target && ts.getEffectiveModifierFlags(node) & 1 /* Export */ &&
+                    target.flags & 111551 /* Value */ &&
+                    (compilerOptions.preserveConstEnums || !isConstEnumOrConstEnumOnlyModule(target))) {
+                    // An `export import ... =` of a value symbol is always considered referenced
+                    return true;
+                }
+            }
+            if (checkChildren) {
+                return !!ts.forEachChild(node, function (node) { return isReferencedAliasDeclaration(node, checkChildren); });
+            }
+            return false;
         }
-        // Because we are updating existed propertyAccess we want to inherit its emitFlags
-        // instead of using the default from createPropertyAccess
-        return node.expression !== expression
-            || node.name !== name
-            ? updateNode(setEmitFlags(createPropertyAccess(expression, name), ts.getEmitFlags(node)), node)
-            : node;
-    }
-    ts.updatePropertyAccess = updatePropertyAccess;
-    function createPropertyAccessChain(expression, questionDotToken, name) {
-        var node = createSynthesizedNode(194 /* PropertyAccessExpression */);
-        node.flags |= 32 /* OptionalChain */;
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.questionDotToken = questionDotToken;
-        node.name = asName(name);
-        setEmitFlags(node, 131072 /* NoIndentation */);
-        return node;
-    }
-    ts.createPropertyAccessChain = createPropertyAccessChain;
-    function updatePropertyAccessChain(node, expression, questionDotToken, name) {
-        ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a PropertyAccessExpression using updatePropertyAccessChain. Use updatePropertyAccess instead.");
-        // Because we are updating an existing PropertyAccessChain we want to inherit its emitFlags
-        // instead of using the default from createPropertyAccess
-        return node.expression !== expression
-            || node.questionDotToken !== questionDotToken
-            || node.name !== name
-            ? updateNode(setEmitFlags(createPropertyAccessChain(expression, questionDotToken, name), ts.getEmitFlags(node)), node)
-            : node;
-    }
-    ts.updatePropertyAccessChain = updatePropertyAccessChain;
-    function createElementAccess(expression, index) {
-        var node = createSynthesizedNode(195 /* ElementAccessExpression */);
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.argumentExpression = asExpression(index);
-        return node;
-    }
-    ts.createElementAccess = createElementAccess;
-    function updateElementAccess(node, expression, argumentExpression) {
-        if (ts.isOptionalChain(node)) {
-            return updateElementAccessChain(node, expression, node.questionDotToken, argumentExpression);
+        function isImplementationOfOverload(node) {
+            if (ts.nodeIsPresent(node.body)) {
+                if (ts.isGetAccessor(node) || ts.isSetAccessor(node))
+                    return false; // Get or set accessors can never be overload implementations, but can have up to 2 signatures
+                var symbol = getSymbolOfNode(node);
+                var signaturesOfSymbol = getSignaturesOfSymbol(symbol);
+                // If this function body corresponds to function with multiple signature, it is implementation of overload
+                // e.g.: function foo(a: string): string;
+                //       function foo(a: number): number;
+                //       function foo(a: any) { // This is implementation of the overloads
+                //           return a;
+                //       }
+                return signaturesOfSymbol.length > 1 ||
+                    // If there is single signature for the symbol, it is overload if that signature isn't coming from the node
+                    // e.g.: function foo(a: string): string;
+                    //       function foo(a: any) { // This is implementation of the overloads
+                    //           return a;
+                    //       }
+                    (signaturesOfSymbol.length === 1 && signaturesOfSymbol[0].declaration !== node);
+            }
+            return false;
         }
-        return node.expression !== expression
-            || node.argumentExpression !== argumentExpression
-            ? updateNode(createElementAccess(expression, argumentExpression), node)
-            : node;
-    }
-    ts.updateElementAccess = updateElementAccess;
-    function createElementAccessChain(expression, questionDotToken, index) {
-        var node = createSynthesizedNode(195 /* ElementAccessExpression */);
-        node.flags |= 32 /* OptionalChain */;
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.questionDotToken = questionDotToken;
-        node.argumentExpression = asExpression(index);
-        return node;
-    }
-    ts.createElementAccessChain = createElementAccessChain;
-    function updateElementAccessChain(node, expression, questionDotToken, argumentExpression) {
-        ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update an ElementAccessExpression using updateElementAccessChain. Use updateElementAccess instead.");
-        return node.expression !== expression
-            || node.questionDotToken !== questionDotToken
-            || node.argumentExpression !== argumentExpression
-            ? updateNode(createElementAccessChain(expression, questionDotToken, argumentExpression), node)
-            : node;
-    }
-    ts.updateElementAccessChain = updateElementAccessChain;
-    function createCall(expression, typeArguments, argumentsArray) {
-        var node = createSynthesizedNode(196 /* CallExpression */);
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.typeArguments = asNodeArray(typeArguments);
-        node.arguments = ts.parenthesizeListElements(createNodeArray(argumentsArray));
-        return node;
-    }
-    ts.createCall = createCall;
-    function updateCall(node, expression, typeArguments, argumentsArray) {
-        if (ts.isOptionalChain(node)) {
-            return updateCallChain(node, expression, node.questionDotToken, typeArguments, argumentsArray);
+        function isRequiredInitializedParameter(parameter) {
+            return !!strictNullChecks &&
+                !isOptionalParameter(parameter) &&
+                !ts.isJSDocParameterTag(parameter) &&
+                !!parameter.initializer &&
+                !ts.hasSyntacticModifier(parameter, 92 /* ParameterPropertyModifier */);
         }
-        return node.expression !== expression
-            || node.typeArguments !== typeArguments
-            || node.arguments !== argumentsArray
-            ? updateNode(createCall(expression, typeArguments, argumentsArray), node)
-            : node;
-    }
-    ts.updateCall = updateCall;
-    function createCallChain(expression, questionDotToken, typeArguments, argumentsArray) {
-        var node = createSynthesizedNode(196 /* CallExpression */);
-        node.flags |= 32 /* OptionalChain */;
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.questionDotToken = questionDotToken;
-        node.typeArguments = asNodeArray(typeArguments);
-        node.arguments = ts.parenthesizeListElements(createNodeArray(argumentsArray));
-        return node;
-    }
-    ts.createCallChain = createCallChain;
-    function updateCallChain(node, expression, questionDotToken, typeArguments, argumentsArray) {
-        ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a CallExpression using updateCallChain. Use updateCall instead.");
-        return node.expression !== expression
-            || node.questionDotToken !== questionDotToken
-            || node.typeArguments !== typeArguments
-            || node.arguments !== argumentsArray
-            ? updateNode(createCallChain(expression, questionDotToken, typeArguments, argumentsArray), node)
-            : node;
-    }
-    ts.updateCallChain = updateCallChain;
-    function createNew(expression, typeArguments, argumentsArray) {
-        var node = createSynthesizedNode(197 /* NewExpression */);
-        node.expression = ts.parenthesizeForNew(expression);
-        node.typeArguments = asNodeArray(typeArguments);
-        node.arguments = argumentsArray ? ts.parenthesizeListElements(createNodeArray(argumentsArray)) : undefined;
-        return node;
-    }
-    ts.createNew = createNew;
-    function updateNew(node, expression, typeArguments, argumentsArray) {
-        return node.expression !== expression
-            || node.typeArguments !== typeArguments
-            || node.arguments !== argumentsArray
-            ? updateNode(createNew(expression, typeArguments, argumentsArray), node)
-            : node;
-    }
-    ts.updateNew = updateNew;
-    function createTaggedTemplate(tag, typeArgumentsOrTemplate, template) {
-        var node = createSynthesizedNode(198 /* TaggedTemplateExpression */);
-        node.tag = ts.parenthesizeForAccess(tag);
-        if (template) {
-            node.typeArguments = asNodeArray(typeArgumentsOrTemplate);
-            node.template = template;
+        function isOptionalUninitializedParameterProperty(parameter) {
+            return strictNullChecks &&
+                isOptionalParameter(parameter) &&
+                !parameter.initializer &&
+                ts.hasSyntacticModifier(parameter, 92 /* ParameterPropertyModifier */);
         }
-        else {
-            node.typeArguments = undefined;
-            node.template = typeArgumentsOrTemplate;
+        function isOptionalUninitializedParameter(parameter) {
+            return !!strictNullChecks &&
+                isOptionalParameter(parameter) &&
+                !parameter.initializer;
         }
-        return node;
-    }
-    ts.createTaggedTemplate = createTaggedTemplate;
-    function updateTaggedTemplate(node, tag, typeArgumentsOrTemplate, template) {
-        return node.tag !== tag
-            || (template
-                ? node.typeArguments !== typeArgumentsOrTemplate || node.template !== template
-                : node.typeArguments !== undefined || node.template !== typeArgumentsOrTemplate)
-            ? updateNode(createTaggedTemplate(tag, typeArgumentsOrTemplate, template), node)
-            : node;
-    }
-    ts.updateTaggedTemplate = updateTaggedTemplate;
-    function createTypeAssertion(type, expression) {
-        var node = createSynthesizedNode(199 /* TypeAssertionExpression */);
-        node.type = type;
-        node.expression = ts.parenthesizePrefixOperand(expression);
-        return node;
-    }
-    ts.createTypeAssertion = createTypeAssertion;
-    function updateTypeAssertion(node, type, expression) {
-        return node.type !== type
-            || node.expression !== expression
-            ? updateNode(createTypeAssertion(type, expression), node)
-            : node;
-    }
-    ts.updateTypeAssertion = updateTypeAssertion;
-    function createParen(expression) {
-        var node = createSynthesizedNode(200 /* ParenthesizedExpression */);
-        node.expression = expression;
-        return node;
-    }
-    ts.createParen = createParen;
-    function updateParen(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createParen(expression), node)
-            : node;
-    }
-    ts.updateParen = updateParen;
-    function createFunctionExpression(modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
-        var node = createSynthesizedNode(201 /* FunctionExpression */);
-        node.modifiers = asNodeArray(modifiers);
-        node.asteriskToken = asteriskToken;
-        node.name = asName(name);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.parameters = createNodeArray(parameters);
-        node.type = type;
-        node.body = body;
-        return node;
-    }
-    ts.createFunctionExpression = createFunctionExpression;
-    function updateFunctionExpression(node, modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
-        return node.name !== name
-            || node.modifiers !== modifiers
-            || node.asteriskToken !== asteriskToken
-            || node.typeParameters !== typeParameters
-            || node.parameters !== parameters
-            || node.type !== type
-            || node.body !== body
-            ? updateNode(createFunctionExpression(modifiers, asteriskToken, name, typeParameters, parameters, type, body), node)
-            : node;
-    }
-    ts.updateFunctionExpression = updateFunctionExpression;
-    function createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body) {
-        var node = createSynthesizedNode(202 /* ArrowFunction */);
-        node.modifiers = asNodeArray(modifiers);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.parameters = createNodeArray(parameters);
-        node.type = type;
-        node.equalsGreaterThanToken = equalsGreaterThanToken || createToken(38 /* EqualsGreaterThanToken */);
-        node.body = ts.parenthesizeConciseBody(body);
-        return node;
-    }
-    ts.createArrowFunction = createArrowFunction;
-    function updateArrowFunction(node, modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body) {
-        return node.modifiers !== modifiers
-            || node.typeParameters !== typeParameters
-            || node.parameters !== parameters
-            || node.type !== type
-            || node.equalsGreaterThanToken !== equalsGreaterThanToken
-            || node.body !== body
-            ? updateNode(createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanToken, body), node)
-            : node;
-    }
-    ts.updateArrowFunction = updateArrowFunction;
-    function createDelete(expression) {
-        var node = createSynthesizedNode(203 /* DeleteExpression */);
-        node.expression = ts.parenthesizePrefixOperand(expression);
-        return node;
-    }
-    ts.createDelete = createDelete;
-    function updateDelete(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createDelete(expression), node)
-            : node;
-    }
-    ts.updateDelete = updateDelete;
-    function createTypeOf(expression) {
-        var node = createSynthesizedNode(204 /* TypeOfExpression */);
-        node.expression = ts.parenthesizePrefixOperand(expression);
-        return node;
-    }
-    ts.createTypeOf = createTypeOf;
-    function updateTypeOf(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createTypeOf(expression), node)
-            : node;
-    }
-    ts.updateTypeOf = updateTypeOf;
-    function createVoid(expression) {
-        var node = createSynthesizedNode(205 /* VoidExpression */);
-        node.expression = ts.parenthesizePrefixOperand(expression);
-        return node;
-    }
-    ts.createVoid = createVoid;
-    function updateVoid(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createVoid(expression), node)
-            : node;
-    }
-    ts.updateVoid = updateVoid;
-    function createAwait(expression) {
-        var node = createSynthesizedNode(206 /* AwaitExpression */);
-        node.expression = ts.parenthesizePrefixOperand(expression);
-        return node;
-    }
-    ts.createAwait = createAwait;
-    function updateAwait(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createAwait(expression), node)
-            : node;
-    }
-    ts.updateAwait = updateAwait;
-    function createPrefix(operator, operand) {
-        var node = createSynthesizedNode(207 /* PrefixUnaryExpression */);
-        node.operator = operator;
-        node.operand = ts.parenthesizePrefixOperand(operand);
-        return node;
-    }
-    ts.createPrefix = createPrefix;
-    function updatePrefix(node, operand) {
-        return node.operand !== operand
-            ? updateNode(createPrefix(node.operator, operand), node)
-            : node;
-    }
-    ts.updatePrefix = updatePrefix;
-    function createPostfix(operand, operator) {
-        var node = createSynthesizedNode(208 /* PostfixUnaryExpression */);
-        node.operand = ts.parenthesizePostfixOperand(operand);
-        node.operator = operator;
-        return node;
-    }
-    ts.createPostfix = createPostfix;
-    function updatePostfix(node, operand) {
-        return node.operand !== operand
-            ? updateNode(createPostfix(operand, node.operator), node)
-            : node;
-    }
-    ts.updatePostfix = updatePostfix;
-    function createBinary(left, operator, right) {
-        var node = createSynthesizedNode(209 /* BinaryExpression */);
-        var operatorToken = asToken(operator);
-        var operatorKind = operatorToken.kind;
-        node.left = ts.parenthesizeBinaryOperand(operatorKind, left, /*isLeftSideOfBinary*/ true, /*leftOperand*/ undefined);
-        node.operatorToken = operatorToken;
-        node.right = ts.parenthesizeBinaryOperand(operatorKind, right, /*isLeftSideOfBinary*/ false, node.left);
-        return node;
-    }
-    ts.createBinary = createBinary;
-    function updateBinary(node, left, right, operator) {
-        return node.left !== left
-            || node.right !== right
-            ? updateNode(createBinary(left, operator || node.operatorToken, right), node)
-            : node;
-    }
-    ts.updateBinary = updateBinary;
-    function createConditional(condition, questionTokenOrWhenTrue, whenTrueOrWhenFalse, colonToken, whenFalse) {
-        var node = createSynthesizedNode(210 /* ConditionalExpression */);
-        node.condition = ts.parenthesizeForConditionalHead(condition);
-        node.questionToken = whenFalse ? questionTokenOrWhenTrue : createToken(57 /* QuestionToken */);
-        node.whenTrue = ts.parenthesizeSubexpressionOfConditionalExpression(whenFalse ? whenTrueOrWhenFalse : questionTokenOrWhenTrue);
-        node.colonToken = whenFalse ? colonToken : createToken(58 /* ColonToken */);
-        node.whenFalse = ts.parenthesizeSubexpressionOfConditionalExpression(whenFalse ? whenFalse : whenTrueOrWhenFalse);
-        return node;
-    }
-    ts.createConditional = createConditional;
-    function updateConditional(node, condition, questionToken, whenTrue, colonToken, whenFalse) {
-        return node.condition !== condition
-            || node.questionToken !== questionToken
-            || node.whenTrue !== whenTrue
-            || node.colonToken !== colonToken
-            || node.whenFalse !== whenFalse
-            ? updateNode(createConditional(condition, questionToken, whenTrue, colonToken, whenFalse), node)
-            : node;
-    }
-    ts.updateConditional = updateConditional;
-    function createTemplateExpression(head, templateSpans) {
-        var node = createSynthesizedNode(211 /* TemplateExpression */);
-        node.head = head;
-        node.templateSpans = createNodeArray(templateSpans);
-        return node;
-    }
-    ts.createTemplateExpression = createTemplateExpression;
-    function updateTemplateExpression(node, head, templateSpans) {
-        return node.head !== head
-            || node.templateSpans !== templateSpans
-            ? updateNode(createTemplateExpression(head, templateSpans), node)
-            : node;
-    }
-    ts.updateTemplateExpression = updateTemplateExpression;
-    var rawTextScanner;
-    var invalidValueSentinel = {};
-    function getCookedText(kind, rawText) {
-        if (!rawTextScanner) {
-            rawTextScanner = ts.createScanner(99 /* Latest */, /*skipTrivia*/ false, 0 /* Standard */);
+        function isExpandoFunctionDeclaration(node) {
+            var declaration = ts.getParseTreeNode(node, ts.isFunctionDeclaration);
+            if (!declaration) {
+                return false;
+            }
+            var symbol = getSymbolOfNode(declaration);
+            if (!symbol || !(symbol.flags & 16 /* Function */)) {
+                return false;
+            }
+            return !!ts.forEachEntry(getExportsOfSymbol(symbol), function (p) { return p.flags & 111551 /* Value */ && p.valueDeclaration && ts.isPropertyAccessExpression(p.valueDeclaration); });
         }
-        switch (kind) {
-            case 14 /* NoSubstitutionTemplateLiteral */:
-                rawTextScanner.setText("`" + rawText + "`");
-                break;
-            case 15 /* TemplateHead */:
-                rawTextScanner.setText("`" + rawText + "${");
-                break;
-            case 16 /* TemplateMiddle */:
-                rawTextScanner.setText("}" + rawText + "${");
-                break;
-            case 17 /* TemplateTail */:
-                rawTextScanner.setText("}" + rawText + "`");
-                break;
+        function getPropertiesOfContainerFunction(node) {
+            var declaration = ts.getParseTreeNode(node, ts.isFunctionDeclaration);
+            if (!declaration) {
+                return ts.emptyArray;
+            }
+            var symbol = getSymbolOfNode(declaration);
+            return symbol && getPropertiesOfType(getTypeOfSymbol(symbol)) || ts.emptyArray;
         }
-        var token = rawTextScanner.scan();
-        if (token === 23 /* CloseBracketToken */) {
-            token = rawTextScanner.reScanTemplateToken(/* isTaggedTemplate */ false);
+        function getNodeCheckFlags(node) {
+            return getNodeLinks(node).flags || 0;
         }
-        if (rawTextScanner.isUnterminated()) {
-            rawTextScanner.setText(undefined);
-            return invalidValueSentinel;
+        function getEnumMemberValue(node) {
+            computeEnumMemberValues(node.parent);
+            return getNodeLinks(node).enumMemberValue;
         }
-        var tokenValue;
-        switch (token) {
-            case 14 /* NoSubstitutionTemplateLiteral */:
-            case 15 /* TemplateHead */:
-            case 16 /* TemplateMiddle */:
-            case 17 /* TemplateTail */:
-                tokenValue = rawTextScanner.getTokenValue();
-                break;
+        function canHaveConstantValue(node) {
+            switch (node.kind) {
+                case 291 /* EnumMember */:
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                    return true;
+            }
+            return false;
         }
-        if (rawTextScanner.scan() !== 1 /* EndOfFileToken */) {
-            rawTextScanner.setText(undefined);
-            return invalidValueSentinel;
+        function getConstantValue(node) {
+            if (node.kind === 291 /* EnumMember */) {
+                return getEnumMemberValue(node);
+            }
+            var symbol = getNodeLinks(node).resolvedSymbol;
+            if (symbol && (symbol.flags & 8 /* EnumMember */)) {
+                // inline property\index accesses only for const enums
+                var member = symbol.valueDeclaration;
+                if (ts.isEnumConst(member.parent)) {
+                    return getEnumMemberValue(member);
+                }
+            }
+            return undefined;
         }
-        rawTextScanner.setText(undefined);
-        return tokenValue;
-    }
-    function createTemplateLiteralLikeNode(kind, text, rawText) {
-        var node = createSynthesizedNode(kind);
-        node.text = text;
-        if (rawText === undefined || text === rawText) {
-            node.rawText = rawText;
+        function isFunctionType(type) {
+            return !!(type.flags & 524288 /* Object */) && getSignaturesOfType(type, 0 /* Call */).length > 0;
         }
-        else {
-            var cooked = getCookedText(kind, rawText);
-            if (typeof cooked === "object") {
-                return ts.Debug.fail("Invalid raw text");
+        function getTypeReferenceSerializationKind(typeNameIn, location) {
+            var _a;
+            // ensure both `typeName` and `location` are parse tree nodes.
+            var typeName = ts.getParseTreeNode(typeNameIn, ts.isEntityName);
+            if (!typeName)
+                return ts.TypeReferenceSerializationKind.Unknown;
+            if (location) {
+                location = ts.getParseTreeNode(location);
+                if (!location)
+                    return ts.TypeReferenceSerializationKind.Unknown;
+            }
+            // Resolve the symbol as a value to ensure the type can be reached at runtime during emit.
+            var valueSymbol = resolveEntityName(typeName, 111551 /* Value */, /*ignoreErrors*/ true, /*dontResolveAlias*/ true, location);
+            var isTypeOnly = ((_a = valueSymbol === null || valueSymbol === void 0 ? void 0 : valueSymbol.declarations) === null || _a === void 0 ? void 0 : _a.every(ts.isTypeOnlyImportOrExportDeclaration)) || false;
+            var resolvedSymbol = valueSymbol && valueSymbol.flags & 2097152 /* Alias */ ? resolveAlias(valueSymbol) : valueSymbol;
+            // Resolve the symbol as a type so that we can provide a more useful hint for the type serializer.
+            var typeSymbol = resolveEntityName(typeName, 788968 /* Type */, /*ignoreErrors*/ true, /*dontResolveAlias*/ false, location);
+            if (resolvedSymbol && resolvedSymbol === typeSymbol) {
+                var globalPromiseSymbol = getGlobalPromiseConstructorSymbol(/*reportErrors*/ false);
+                if (globalPromiseSymbol && resolvedSymbol === globalPromiseSymbol) {
+                    return ts.TypeReferenceSerializationKind.Promise;
+                }
+                var constructorType = getTypeOfSymbol(resolvedSymbol);
+                if (constructorType && isConstructorType(constructorType)) {
+                    return isTypeOnly ? ts.TypeReferenceSerializationKind.TypeWithCallSignature : ts.TypeReferenceSerializationKind.TypeWithConstructSignatureAndValue;
+                }
+            }
+            // We might not be able to resolve type symbol so use unknown type in that case (eg error case)
+            if (!typeSymbol) {
+                return isTypeOnly ? ts.TypeReferenceSerializationKind.ObjectType : ts.TypeReferenceSerializationKind.Unknown;
+            }
+            var type = getDeclaredTypeOfSymbol(typeSymbol);
+            if (type === errorType) {
+                return isTypeOnly ? ts.TypeReferenceSerializationKind.ObjectType : ts.TypeReferenceSerializationKind.Unknown;
+            }
+            else if (type.flags & 3 /* AnyOrUnknown */) {
+                return ts.TypeReferenceSerializationKind.ObjectType;
+            }
+            else if (isTypeAssignableToKind(type, 16384 /* Void */ | 98304 /* Nullable */ | 131072 /* Never */)) {
+                return ts.TypeReferenceSerializationKind.VoidNullableOrNeverType;
+            }
+            else if (isTypeAssignableToKind(type, 528 /* BooleanLike */)) {
+                return ts.TypeReferenceSerializationKind.BooleanType;
+            }
+            else if (isTypeAssignableToKind(type, 296 /* NumberLike */)) {
+                return ts.TypeReferenceSerializationKind.NumberLikeType;
+            }
+            else if (isTypeAssignableToKind(type, 2112 /* BigIntLike */)) {
+                return ts.TypeReferenceSerializationKind.BigIntLikeType;
+            }
+            else if (isTypeAssignableToKind(type, 402653316 /* StringLike */)) {
+                return ts.TypeReferenceSerializationKind.StringLikeType;
+            }
+            else if (isTupleType(type)) {
+                return ts.TypeReferenceSerializationKind.ArrayLikeType;
+            }
+            else if (isTypeAssignableToKind(type, 12288 /* ESSymbolLike */)) {
+                return ts.TypeReferenceSerializationKind.ESSymbolType;
+            }
+            else if (isFunctionType(type)) {
+                return ts.TypeReferenceSerializationKind.TypeWithCallSignature;
+            }
+            else if (isArrayType(type)) {
+                return ts.TypeReferenceSerializationKind.ArrayLikeType;
+            }
+            else {
+                return ts.TypeReferenceSerializationKind.ObjectType;
             }
-            ts.Debug.assert(text === cooked, "Expected argument 'text' to be the normalized (i.e. 'cooked') version of argument 'rawText'.");
-            node.rawText = rawText;
-        }
-        return node;
-    }
-    function createTemplateHead(text, rawText) {
-        var node = createTemplateLiteralLikeNode(15 /* TemplateHead */, text, rawText);
-        node.text = text;
-        return node;
-    }
-    ts.createTemplateHead = createTemplateHead;
-    function createTemplateMiddle(text, rawText) {
-        var node = createTemplateLiteralLikeNode(16 /* TemplateMiddle */, text, rawText);
-        node.text = text;
-        return node;
-    }
-    ts.createTemplateMiddle = createTemplateMiddle;
-    function createTemplateTail(text, rawText) {
-        var node = createTemplateLiteralLikeNode(17 /* TemplateTail */, text, rawText);
-        node.text = text;
-        return node;
-    }
-    ts.createTemplateTail = createTemplateTail;
-    function createNoSubstitutionTemplateLiteral(text, rawText) {
-        var node = createTemplateLiteralLikeNode(14 /* NoSubstitutionTemplateLiteral */, text, rawText);
-        return node;
-    }
-    ts.createNoSubstitutionTemplateLiteral = createNoSubstitutionTemplateLiteral;
-    function createYield(asteriskTokenOrExpression, expression) {
-        var asteriskToken = asteriskTokenOrExpression && asteriskTokenOrExpression.kind === 41 /* AsteriskToken */ ? asteriskTokenOrExpression : undefined;
-        expression = asteriskTokenOrExpression && asteriskTokenOrExpression.kind !== 41 /* AsteriskToken */ ? asteriskTokenOrExpression : expression;
-        var node = createSynthesizedNode(212 /* YieldExpression */);
-        node.asteriskToken = asteriskToken;
-        node.expression = expression && ts.parenthesizeExpressionForList(expression);
-        return node;
-    }
-    ts.createYield = createYield;
-    function updateYield(node, asteriskToken, expression) {
-        return node.expression !== expression
-            || node.asteriskToken !== asteriskToken
-            ? updateNode(createYield(asteriskToken, expression), node)
-            : node;
-    }
-    ts.updateYield = updateYield;
-    function createSpread(expression) {
-        var node = createSynthesizedNode(213 /* SpreadElement */);
-        node.expression = ts.parenthesizeExpressionForList(expression);
-        return node;
-    }
-    ts.createSpread = createSpread;
-    function updateSpread(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createSpread(expression), node)
-            : node;
-    }
-    ts.updateSpread = updateSpread;
-    function createClassExpression(modifiers, name, typeParameters, heritageClauses, members) {
-        var node = createSynthesizedNode(214 /* ClassExpression */);
-        node.decorators = undefined;
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.heritageClauses = asNodeArray(heritageClauses);
-        node.members = createNodeArray(members);
-        return node;
-    }
-    ts.createClassExpression = createClassExpression;
-    function updateClassExpression(node, modifiers, name, typeParameters, heritageClauses, members) {
-        return node.modifiers !== modifiers
-            || node.name !== name
-            || node.typeParameters !== typeParameters
-            || node.heritageClauses !== heritageClauses
-            || node.members !== members
-            ? updateNode(createClassExpression(modifiers, name, typeParameters, heritageClauses, members), node)
-            : node;
-    }
-    ts.updateClassExpression = updateClassExpression;
-    function createOmittedExpression() {
-        return createSynthesizedNode(215 /* OmittedExpression */);
-    }
-    ts.createOmittedExpression = createOmittedExpression;
-    function createExpressionWithTypeArguments(typeArguments, expression) {
-        var node = createSynthesizedNode(216 /* ExpressionWithTypeArguments */);
-        node.expression = ts.parenthesizeForAccess(expression);
-        node.typeArguments = asNodeArray(typeArguments);
-        return node;
-    }
-    ts.createExpressionWithTypeArguments = createExpressionWithTypeArguments;
-    function updateExpressionWithTypeArguments(node, typeArguments, expression) {
-        return node.typeArguments !== typeArguments
-            || node.expression !== expression
-            ? updateNode(createExpressionWithTypeArguments(typeArguments, expression), node)
-            : node;
-    }
-    ts.updateExpressionWithTypeArguments = updateExpressionWithTypeArguments;
-    function createAsExpression(expression, type) {
-        var node = createSynthesizedNode(217 /* AsExpression */);
-        node.expression = expression;
-        node.type = type;
-        return node;
-    }
-    ts.createAsExpression = createAsExpression;
-    function updateAsExpression(node, expression, type) {
-        return node.expression !== expression
-            || node.type !== type
-            ? updateNode(createAsExpression(expression, type), node)
-            : node;
-    }
-    ts.updateAsExpression = updateAsExpression;
-    function createNonNullExpression(expression) {
-        var node = createSynthesizedNode(218 /* NonNullExpression */);
-        node.expression = ts.parenthesizeForAccess(expression);
-        return node;
-    }
-    ts.createNonNullExpression = createNonNullExpression;
-    function updateNonNullExpression(node, expression) {
-        if (ts.isNonNullChain(node)) {
-            return updateNonNullChain(node, expression);
         }
-        return node.expression !== expression
-            ? updateNode(createNonNullExpression(expression), node)
-            : node;
-    }
-    ts.updateNonNullExpression = updateNonNullExpression;
-    function createNonNullChain(expression) {
-        var node = createSynthesizedNode(218 /* NonNullExpression */);
-        node.flags |= 32 /* OptionalChain */;
-        node.expression = ts.parenthesizeForAccess(expression);
-        return node;
-    }
-    ts.createNonNullChain = createNonNullChain;
-    function updateNonNullChain(node, expression) {
-        ts.Debug.assert(!!(node.flags & 32 /* OptionalChain */), "Cannot update a NonNullExpression using updateNonNullChain. Use updateNonNullExpression instead.");
-        return node.expression !== expression
-            ? updateNode(createNonNullChain(expression), node)
-            : node;
-    }
-    ts.updateNonNullChain = updateNonNullChain;
-    function createMetaProperty(keywordToken, name) {
-        var node = createSynthesizedNode(219 /* MetaProperty */);
-        node.keywordToken = keywordToken;
-        node.name = name;
-        return node;
-    }
-    ts.createMetaProperty = createMetaProperty;
-    function updateMetaProperty(node, name) {
-        return node.name !== name
-            ? updateNode(createMetaProperty(node.keywordToken, name), node)
-            : node;
-    }
-    ts.updateMetaProperty = updateMetaProperty;
-    // Misc
-    function createTemplateSpan(expression, literal) {
-        var node = createSynthesizedNode(221 /* TemplateSpan */);
-        node.expression = expression;
-        node.literal = literal;
-        return node;
-    }
-    ts.createTemplateSpan = createTemplateSpan;
-    function updateTemplateSpan(node, expression, literal) {
-        return node.expression !== expression
-            || node.literal !== literal
-            ? updateNode(createTemplateSpan(expression, literal), node)
-            : node;
-    }
-    ts.updateTemplateSpan = updateTemplateSpan;
-    function createSemicolonClassElement() {
-        return createSynthesizedNode(222 /* SemicolonClassElement */);
-    }
-    ts.createSemicolonClassElement = createSemicolonClassElement;
-    // Element
-    function createBlock(statements, multiLine) {
-        var block = createSynthesizedNode(223 /* Block */);
-        block.statements = createNodeArray(statements);
-        if (multiLine)
-            block.multiLine = multiLine;
-        return block;
-    }
-    ts.createBlock = createBlock;
-    function updateBlock(node, statements) {
-        return node.statements !== statements
-            ? updateNode(createBlock(statements, node.multiLine), node)
-            : node;
-    }
-    ts.updateBlock = updateBlock;
-    function createVariableStatement(modifiers, declarationList) {
-        var node = createSynthesizedNode(225 /* VariableStatement */);
-        node.decorators = undefined;
-        node.modifiers = asNodeArray(modifiers);
-        node.declarationList = ts.isArray(declarationList) ? createVariableDeclarationList(declarationList) : declarationList;
-        return node;
-    }
-    ts.createVariableStatement = createVariableStatement;
-    function updateVariableStatement(node, modifiers, declarationList) {
-        return node.modifiers !== modifiers
-            || node.declarationList !== declarationList
-            ? updateNode(createVariableStatement(modifiers, declarationList), node)
-            : node;
-    }
-    ts.updateVariableStatement = updateVariableStatement;
-    function createEmptyStatement() {
-        return createSynthesizedNode(224 /* EmptyStatement */);
-    }
-    ts.createEmptyStatement = createEmptyStatement;
-    function createExpressionStatement(expression) {
-        var node = createSynthesizedNode(226 /* ExpressionStatement */);
-        node.expression = ts.parenthesizeExpressionForExpressionStatement(expression);
-        return node;
-    }
-    ts.createExpressionStatement = createExpressionStatement;
-    function updateExpressionStatement(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createExpressionStatement(expression), node)
-            : node;
-    }
-    ts.updateExpressionStatement = updateExpressionStatement;
-    /** @deprecated Use `createExpressionStatement` instead.  */
-    ts.createStatement = createExpressionStatement;
-    /** @deprecated Use `updateExpressionStatement` instead.  */
-    ts.updateStatement = updateExpressionStatement;
-    function createIf(expression, thenStatement, elseStatement) {
-        var node = createSynthesizedNode(227 /* IfStatement */);
-        node.expression = expression;
-        node.thenStatement = asEmbeddedStatement(thenStatement);
-        node.elseStatement = asEmbeddedStatement(elseStatement);
-        return node;
-    }
-    ts.createIf = createIf;
-    function updateIf(node, expression, thenStatement, elseStatement) {
-        return node.expression !== expression
-            || node.thenStatement !== thenStatement
-            || node.elseStatement !== elseStatement
-            ? updateNode(createIf(expression, thenStatement, elseStatement), node)
-            : node;
-    }
-    ts.updateIf = updateIf;
-    function createDo(statement, expression) {
-        var node = createSynthesizedNode(228 /* DoStatement */);
-        node.statement = asEmbeddedStatement(statement);
-        node.expression = expression;
-        return node;
-    }
-    ts.createDo = createDo;
-    function updateDo(node, statement, expression) {
-        return node.statement !== statement
-            || node.expression !== expression
-            ? updateNode(createDo(statement, expression), node)
-            : node;
-    }
-    ts.updateDo = updateDo;
-    function createWhile(expression, statement) {
-        var node = createSynthesizedNode(229 /* WhileStatement */);
-        node.expression = expression;
-        node.statement = asEmbeddedStatement(statement);
-        return node;
-    }
-    ts.createWhile = createWhile;
-    function updateWhile(node, expression, statement) {
-        return node.expression !== expression
-            || node.statement !== statement
-            ? updateNode(createWhile(expression, statement), node)
-            : node;
-    }
-    ts.updateWhile = updateWhile;
-    function createFor(initializer, condition, incrementor, statement) {
-        var node = createSynthesizedNode(230 /* ForStatement */);
-        node.initializer = initializer;
-        node.condition = condition;
-        node.incrementor = incrementor;
-        node.statement = asEmbeddedStatement(statement);
-        return node;
-    }
-    ts.createFor = createFor;
-    function updateFor(node, initializer, condition, incrementor, statement) {
-        return node.initializer !== initializer
-            || node.condition !== condition
-            || node.incrementor !== incrementor
-            || node.statement !== statement
-            ? updateNode(createFor(initializer, condition, incrementor, statement), node)
-            : node;
-    }
-    ts.updateFor = updateFor;
-    function createForIn(initializer, expression, statement) {
-        var node = createSynthesizedNode(231 /* ForInStatement */);
-        node.initializer = initializer;
-        node.expression = expression;
-        node.statement = asEmbeddedStatement(statement);
-        return node;
-    }
-    ts.createForIn = createForIn;
-    function updateForIn(node, initializer, expression, statement) {
-        return node.initializer !== initializer
-            || node.expression !== expression
-            || node.statement !== statement
-            ? updateNode(createForIn(initializer, expression, statement), node)
-            : node;
-    }
-    ts.updateForIn = updateForIn;
-    function createForOf(awaitModifier, initializer, expression, statement) {
-        var node = createSynthesizedNode(232 /* ForOfStatement */);
-        node.awaitModifier = awaitModifier;
-        node.initializer = initializer;
-        node.expression = ts.isCommaSequence(expression) ? createParen(expression) : expression;
-        node.statement = asEmbeddedStatement(statement);
-        return node;
-    }
-    ts.createForOf = createForOf;
-    function updateForOf(node, awaitModifier, initializer, expression, statement) {
-        return node.awaitModifier !== awaitModifier
-            || node.initializer !== initializer
-            || node.expression !== expression
-            || node.statement !== statement
-            ? updateNode(createForOf(awaitModifier, initializer, expression, statement), node)
-            : node;
-    }
-    ts.updateForOf = updateForOf;
-    function createContinue(label) {
-        var node = createSynthesizedNode(233 /* ContinueStatement */);
-        node.label = asName(label);
-        return node;
-    }
-    ts.createContinue = createContinue;
-    function updateContinue(node, label) {
-        return node.label !== label
-            ? updateNode(createContinue(label), node)
-            : node;
-    }
-    ts.updateContinue = updateContinue;
-    function createBreak(label) {
-        var node = createSynthesizedNode(234 /* BreakStatement */);
-        node.label = asName(label);
-        return node;
-    }
-    ts.createBreak = createBreak;
-    function updateBreak(node, label) {
-        return node.label !== label
-            ? updateNode(createBreak(label), node)
-            : node;
-    }
-    ts.updateBreak = updateBreak;
-    function createReturn(expression) {
-        var node = createSynthesizedNode(235 /* ReturnStatement */);
-        node.expression = expression;
-        return node;
-    }
-    ts.createReturn = createReturn;
-    function updateReturn(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createReturn(expression), node)
-            : node;
-    }
-    ts.updateReturn = updateReturn;
-    function createWith(expression, statement) {
-        var node = createSynthesizedNode(236 /* WithStatement */);
-        node.expression = expression;
-        node.statement = asEmbeddedStatement(statement);
-        return node;
-    }
-    ts.createWith = createWith;
-    function updateWith(node, expression, statement) {
-        return node.expression !== expression
-            || node.statement !== statement
-            ? updateNode(createWith(expression, statement), node)
-            : node;
-    }
-    ts.updateWith = updateWith;
-    function createSwitch(expression, caseBlock) {
-        var node = createSynthesizedNode(237 /* SwitchStatement */);
-        node.expression = ts.parenthesizeExpressionForList(expression);
-        node.caseBlock = caseBlock;
-        return node;
-    }
-    ts.createSwitch = createSwitch;
-    function updateSwitch(node, expression, caseBlock) {
-        return node.expression !== expression
-            || node.caseBlock !== caseBlock
-            ? updateNode(createSwitch(expression, caseBlock), node)
-            : node;
-    }
-    ts.updateSwitch = updateSwitch;
-    function createLabel(label, statement) {
-        var node = createSynthesizedNode(238 /* LabeledStatement */);
-        node.label = asName(label);
-        node.statement = asEmbeddedStatement(statement);
-        return node;
-    }
-    ts.createLabel = createLabel;
-    function updateLabel(node, label, statement) {
-        return node.label !== label
-            || node.statement !== statement
-            ? updateNode(createLabel(label, statement), node)
-            : node;
-    }
-    ts.updateLabel = updateLabel;
-    function createThrow(expression) {
-        var node = createSynthesizedNode(239 /* ThrowStatement */);
-        node.expression = expression;
-        return node;
-    }
-    ts.createThrow = createThrow;
-    function updateThrow(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createThrow(expression), node)
-            : node;
-    }
-    ts.updateThrow = updateThrow;
-    function createTry(tryBlock, catchClause, finallyBlock) {
-        var node = createSynthesizedNode(240 /* TryStatement */);
-        node.tryBlock = tryBlock;
-        node.catchClause = catchClause;
-        node.finallyBlock = finallyBlock;
-        return node;
-    }
-    ts.createTry = createTry;
-    function updateTry(node, tryBlock, catchClause, finallyBlock) {
-        return node.tryBlock !== tryBlock
-            || node.catchClause !== catchClause
-            || node.finallyBlock !== finallyBlock
-            ? updateNode(createTry(tryBlock, catchClause, finallyBlock), node)
-            : node;
-    }
-    ts.updateTry = updateTry;
-    function createDebuggerStatement() {
-        return createSynthesizedNode(241 /* DebuggerStatement */);
-    }
-    ts.createDebuggerStatement = createDebuggerStatement;
-    function createVariableDeclaration(name, type, initializer) {
-        /* Internally, one should probably use createTypeScriptVariableDeclaration instead and handle definite assignment assertions */
-        var node = createSynthesizedNode(242 /* VariableDeclaration */);
-        node.name = asName(name);
-        node.type = type;
-        node.initializer = initializer !== undefined ? ts.parenthesizeExpressionForList(initializer) : undefined;
-        return node;
-    }
-    ts.createVariableDeclaration = createVariableDeclaration;
-    function updateVariableDeclaration(node, name, type, initializer) {
-        /* Internally, one should probably use updateTypeScriptVariableDeclaration instead and handle definite assignment assertions */
-        return node.name !== name
-            || node.type !== type
-            || node.initializer !== initializer
-            ? updateNode(createVariableDeclaration(name, type, initializer), node)
-            : node;
-    }
-    ts.updateVariableDeclaration = updateVariableDeclaration;
-    /* @internal */
-    function createTypeScriptVariableDeclaration(name, exclaimationToken, type, initializer) {
-        var node = createSynthesizedNode(242 /* VariableDeclaration */);
-        node.name = asName(name);
-        node.type = type;
-        node.initializer = initializer !== undefined ? ts.parenthesizeExpressionForList(initializer) : undefined;
-        node.exclamationToken = exclaimationToken;
-        return node;
-    }
-    ts.createTypeScriptVariableDeclaration = createTypeScriptVariableDeclaration;
-    /* @internal */
-    function updateTypeScriptVariableDeclaration(node, name, exclaimationToken, type, initializer) {
-        return node.name !== name
-            || node.type !== type
-            || node.initializer !== initializer
-            || node.exclamationToken !== exclaimationToken
-            ? updateNode(createTypeScriptVariableDeclaration(name, exclaimationToken, type, initializer), node)
-            : node;
-    }
-    ts.updateTypeScriptVariableDeclaration = updateTypeScriptVariableDeclaration;
-    function createVariableDeclarationList(declarations, flags) {
-        if (flags === void 0) { flags = 0 /* None */; }
-        var node = createSynthesizedNode(243 /* VariableDeclarationList */);
-        node.flags |= flags & 3 /* BlockScoped */;
-        node.declarations = createNodeArray(declarations);
-        return node;
-    }
-    ts.createVariableDeclarationList = createVariableDeclarationList;
-    function updateVariableDeclarationList(node, declarations) {
-        return node.declarations !== declarations
-            ? updateNode(createVariableDeclarationList(declarations, node.flags), node)
-            : node;
-    }
-    ts.updateVariableDeclarationList = updateVariableDeclarationList;
-    function createFunctionDeclaration(decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
-        var node = createSynthesizedNode(244 /* FunctionDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.asteriskToken = asteriskToken;
-        node.name = asName(name);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.parameters = createNodeArray(parameters);
-        node.type = type;
-        node.body = body;
-        return node;
-    }
-    ts.createFunctionDeclaration = createFunctionDeclaration;
-    function updateFunctionDeclaration(node, decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.asteriskToken !== asteriskToken
-            || node.name !== name
-            || node.typeParameters !== typeParameters
-            || node.parameters !== parameters
-            || node.type !== type
-            || node.body !== body
-            ? updateNode(createFunctionDeclaration(decorators, modifiers, asteriskToken, name, typeParameters, parameters, type, body), node)
-            : node;
-    }
-    ts.updateFunctionDeclaration = updateFunctionDeclaration;
-    /* @internal */
-    function updateFunctionLikeBody(declaration, body) {
-        switch (declaration.kind) {
-            case 244 /* FunctionDeclaration */:
-                return createFunctionDeclaration(declaration.decorators, declaration.modifiers, declaration.asteriskToken, declaration.name, declaration.typeParameters, declaration.parameters, declaration.type, body);
-            case 161 /* MethodDeclaration */:
-                return createMethod(declaration.decorators, declaration.modifiers, declaration.asteriskToken, declaration.name, declaration.questionToken, declaration.typeParameters, declaration.parameters, declaration.type, body);
-            case 163 /* GetAccessor */:
-                return createGetAccessor(declaration.decorators, declaration.modifiers, declaration.name, declaration.parameters, declaration.type, body);
-            case 164 /* SetAccessor */:
-                return createSetAccessor(declaration.decorators, declaration.modifiers, declaration.name, declaration.parameters, body);
-            case 162 /* Constructor */:
-                return createConstructor(declaration.decorators, declaration.modifiers, declaration.parameters, body);
-            case 201 /* FunctionExpression */:
-                return createFunctionExpression(declaration.modifiers, declaration.asteriskToken, declaration.name, declaration.typeParameters, declaration.parameters, declaration.type, body);
-            case 202 /* ArrowFunction */:
-                return createArrowFunction(declaration.modifiers, declaration.typeParameters, declaration.parameters, declaration.type, declaration.equalsGreaterThanToken, body);
-        }
-    }
-    ts.updateFunctionLikeBody = updateFunctionLikeBody;
-    function createClassDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members) {
-        var node = createSynthesizedNode(245 /* ClassDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.heritageClauses = asNodeArray(heritageClauses);
-        node.members = createNodeArray(members);
-        return node;
-    }
-    ts.createClassDeclaration = createClassDeclaration;
-    function updateClassDeclaration(node, decorators, modifiers, name, typeParameters, heritageClauses, members) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.typeParameters !== typeParameters
-            || node.heritageClauses !== heritageClauses
-            || node.members !== members
-            ? updateNode(createClassDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members), node)
-            : node;
-    }
-    ts.updateClassDeclaration = updateClassDeclaration;
-    function createInterfaceDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members) {
-        var node = createSynthesizedNode(246 /* InterfaceDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.heritageClauses = asNodeArray(heritageClauses);
-        node.members = createNodeArray(members);
-        return node;
-    }
-    ts.createInterfaceDeclaration = createInterfaceDeclaration;
-    function updateInterfaceDeclaration(node, decorators, modifiers, name, typeParameters, heritageClauses, members) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.typeParameters !== typeParameters
-            || node.heritageClauses !== heritageClauses
-            || node.members !== members
-            ? updateNode(createInterfaceDeclaration(decorators, modifiers, name, typeParameters, heritageClauses, members), node)
-            : node;
-    }
-    ts.updateInterfaceDeclaration = updateInterfaceDeclaration;
-    function createTypeAliasDeclaration(decorators, modifiers, name, typeParameters, type) {
-        var node = createSynthesizedNode(247 /* TypeAliasDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.typeParameters = asNodeArray(typeParameters);
-        node.type = type;
-        return node;
-    }
-    ts.createTypeAliasDeclaration = createTypeAliasDeclaration;
-    function updateTypeAliasDeclaration(node, decorators, modifiers, name, typeParameters, type) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.typeParameters !== typeParameters
-            || node.type !== type
-            ? updateNode(createTypeAliasDeclaration(decorators, modifiers, name, typeParameters, type), node)
-            : node;
-    }
-    ts.updateTypeAliasDeclaration = updateTypeAliasDeclaration;
-    function createEnumDeclaration(decorators, modifiers, name, members) {
-        var node = createSynthesizedNode(248 /* EnumDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.members = createNodeArray(members);
-        return node;
-    }
-    ts.createEnumDeclaration = createEnumDeclaration;
-    function updateEnumDeclaration(node, decorators, modifiers, name, members) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.members !== members
-            ? updateNode(createEnumDeclaration(decorators, modifiers, name, members), node)
-            : node;
-    }
-    ts.updateEnumDeclaration = updateEnumDeclaration;
-    function createModuleDeclaration(decorators, modifiers, name, body, flags) {
-        if (flags === void 0) { flags = 0 /* None */; }
-        var node = createSynthesizedNode(249 /* ModuleDeclaration */);
-        node.flags |= flags & (16 /* Namespace */ | 4 /* NestedNamespace */ | 1024 /* GlobalAugmentation */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = name;
-        node.body = body;
-        return node;
-    }
-    ts.createModuleDeclaration = createModuleDeclaration;
-    function updateModuleDeclaration(node, decorators, modifiers, name, body) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.body !== body
-            ? updateNode(createModuleDeclaration(decorators, modifiers, name, body, node.flags), node)
-            : node;
-    }
-    ts.updateModuleDeclaration = updateModuleDeclaration;
-    function createModuleBlock(statements) {
-        var node = createSynthesizedNode(250 /* ModuleBlock */);
-        node.statements = createNodeArray(statements);
-        return node;
-    }
-    ts.createModuleBlock = createModuleBlock;
-    function updateModuleBlock(node, statements) {
-        return node.statements !== statements
-            ? updateNode(createModuleBlock(statements), node)
-            : node;
-    }
-    ts.updateModuleBlock = updateModuleBlock;
-    function createCaseBlock(clauses) {
-        var node = createSynthesizedNode(251 /* CaseBlock */);
-        node.clauses = createNodeArray(clauses);
-        return node;
-    }
-    ts.createCaseBlock = createCaseBlock;
-    function updateCaseBlock(node, clauses) {
-        return node.clauses !== clauses
-            ? updateNode(createCaseBlock(clauses), node)
-            : node;
-    }
-    ts.updateCaseBlock = updateCaseBlock;
-    function createNamespaceExportDeclaration(name) {
-        var node = createSynthesizedNode(252 /* NamespaceExportDeclaration */);
-        node.name = asName(name);
-        return node;
-    }
-    ts.createNamespaceExportDeclaration = createNamespaceExportDeclaration;
-    function updateNamespaceExportDeclaration(node, name) {
-        return node.name !== name
-            ? updateNode(createNamespaceExportDeclaration(name), node)
-            : node;
-    }
-    ts.updateNamespaceExportDeclaration = updateNamespaceExportDeclaration;
-    function createImportEqualsDeclaration(decorators, modifiers, name, moduleReference) {
-        var node = createSynthesizedNode(253 /* ImportEqualsDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.name = asName(name);
-        node.moduleReference = moduleReference;
-        return node;
-    }
-    ts.createImportEqualsDeclaration = createImportEqualsDeclaration;
-    function updateImportEqualsDeclaration(node, decorators, modifiers, name, moduleReference) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.name !== name
-            || node.moduleReference !== moduleReference
-            ? updateNode(createImportEqualsDeclaration(decorators, modifiers, name, moduleReference), node)
-            : node;
-    }
-    ts.updateImportEqualsDeclaration = updateImportEqualsDeclaration;
-    function createImportDeclaration(decorators, modifiers, importClause, moduleSpecifier) {
-        var node = createSynthesizedNode(254 /* ImportDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.importClause = importClause;
-        node.moduleSpecifier = moduleSpecifier;
-        return node;
-    }
-    ts.createImportDeclaration = createImportDeclaration;
-    function updateImportDeclaration(node, decorators, modifiers, importClause, moduleSpecifier) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.importClause !== importClause
-            || node.moduleSpecifier !== moduleSpecifier
-            ? updateNode(createImportDeclaration(decorators, modifiers, importClause, moduleSpecifier), node)
-            : node;
-    }
-    ts.updateImportDeclaration = updateImportDeclaration;
-    function createImportClause(name, namedBindings, isTypeOnly) {
-        if (isTypeOnly === void 0) { isTypeOnly = false; }
-        var node = createSynthesizedNode(255 /* ImportClause */);
-        node.name = name;
-        node.namedBindings = namedBindings;
-        node.isTypeOnly = isTypeOnly;
-        return node;
-    }
-    ts.createImportClause = createImportClause;
-    function updateImportClause(node, name, namedBindings, isTypeOnly) {
-        return node.name !== name
-            || node.namedBindings !== namedBindings
-            || node.isTypeOnly !== isTypeOnly
-            ? updateNode(createImportClause(name, namedBindings, isTypeOnly), node)
-            : node;
-    }
-    ts.updateImportClause = updateImportClause;
-    function createNamespaceImport(name) {
-        var node = createSynthesizedNode(256 /* NamespaceImport */);
-        node.name = name;
-        return node;
-    }
-    ts.createNamespaceImport = createNamespaceImport;
-    function createNamespaceExport(name) {
-        var node = createSynthesizedNode(262 /* NamespaceExport */);
-        node.name = name;
-        return node;
-    }
-    ts.createNamespaceExport = createNamespaceExport;
-    function updateNamespaceImport(node, name) {
-        return node.name !== name
-            ? updateNode(createNamespaceImport(name), node)
-            : node;
-    }
-    ts.updateNamespaceImport = updateNamespaceImport;
-    function updateNamespaceExport(node, name) {
-        return node.name !== name
-            ? updateNode(createNamespaceExport(name), node)
-            : node;
-    }
-    ts.updateNamespaceExport = updateNamespaceExport;
-    function createNamedImports(elements) {
-        var node = createSynthesizedNode(257 /* NamedImports */);
-        node.elements = createNodeArray(elements);
-        return node;
-    }
-    ts.createNamedImports = createNamedImports;
-    function updateNamedImports(node, elements) {
-        return node.elements !== elements
-            ? updateNode(createNamedImports(elements), node)
-            : node;
-    }
-    ts.updateNamedImports = updateNamedImports;
-    function createImportSpecifier(propertyName, name) {
-        var node = createSynthesizedNode(258 /* ImportSpecifier */);
-        node.propertyName = propertyName;
-        node.name = name;
-        return node;
-    }
-    ts.createImportSpecifier = createImportSpecifier;
-    function updateImportSpecifier(node, propertyName, name) {
-        return node.propertyName !== propertyName
-            || node.name !== name
-            ? updateNode(createImportSpecifier(propertyName, name), node)
-            : node;
-    }
-    ts.updateImportSpecifier = updateImportSpecifier;
-    function createExportAssignment(decorators, modifiers, isExportEquals, expression) {
-        var node = createSynthesizedNode(259 /* ExportAssignment */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.isExportEquals = isExportEquals;
-        node.expression = isExportEquals ? ts.parenthesizeBinaryOperand(62 /* EqualsToken */, expression, /*isLeftSideOfBinary*/ false, /*leftOperand*/ undefined) : ts.parenthesizeDefaultExpression(expression);
-        return node;
-    }
-    ts.createExportAssignment = createExportAssignment;
-    function updateExportAssignment(node, decorators, modifiers, expression) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.expression !== expression
-            ? updateNode(createExportAssignment(decorators, modifiers, node.isExportEquals, expression), node)
-            : node;
-    }
-    ts.updateExportAssignment = updateExportAssignment;
-    function createExportDeclaration(decorators, modifiers, exportClause, moduleSpecifier, isTypeOnly) {
-        if (isTypeOnly === void 0) { isTypeOnly = false; }
-        var node = createSynthesizedNode(260 /* ExportDeclaration */);
-        node.decorators = asNodeArray(decorators);
-        node.modifiers = asNodeArray(modifiers);
-        node.isTypeOnly = isTypeOnly;
-        node.exportClause = exportClause;
-        node.moduleSpecifier = moduleSpecifier;
-        return node;
-    }
-    ts.createExportDeclaration = createExportDeclaration;
-    function updateExportDeclaration(node, decorators, modifiers, exportClause, moduleSpecifier, isTypeOnly) {
-        return node.decorators !== decorators
-            || node.modifiers !== modifiers
-            || node.isTypeOnly !== isTypeOnly
-            || node.exportClause !== exportClause
-            || node.moduleSpecifier !== moduleSpecifier
-            ? updateNode(createExportDeclaration(decorators, modifiers, exportClause, moduleSpecifier, isTypeOnly), node)
-            : node;
-    }
-    ts.updateExportDeclaration = updateExportDeclaration;
-    /* @internal */
-    function createEmptyExports() {
-        return createExportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, createNamedExports([]), /*moduleSpecifier*/ undefined);
-    }
-    ts.createEmptyExports = createEmptyExports;
-    function createNamedExports(elements) {
-        var node = createSynthesizedNode(261 /* NamedExports */);
-        node.elements = createNodeArray(elements);
-        return node;
-    }
-    ts.createNamedExports = createNamedExports;
-    function updateNamedExports(node, elements) {
-        return node.elements !== elements
-            ? updateNode(createNamedExports(elements), node)
-            : node;
-    }
-    ts.updateNamedExports = updateNamedExports;
-    function createExportSpecifier(propertyName, name) {
-        var node = createSynthesizedNode(263 /* ExportSpecifier */);
-        node.propertyName = asName(propertyName);
-        node.name = asName(name);
-        return node;
-    }
-    ts.createExportSpecifier = createExportSpecifier;
-    function updateExportSpecifier(node, propertyName, name) {
-        return node.propertyName !== propertyName
-            || node.name !== name
-            ? updateNode(createExportSpecifier(propertyName, name), node)
-            : node;
-    }
-    ts.updateExportSpecifier = updateExportSpecifier;
-    // Module references
-    function createExternalModuleReference(expression) {
-        var node = createSynthesizedNode(265 /* ExternalModuleReference */);
-        node.expression = expression;
-        return node;
-    }
-    ts.createExternalModuleReference = createExternalModuleReference;
-    function updateExternalModuleReference(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createExternalModuleReference(expression), node)
-            : node;
-    }
-    ts.updateExternalModuleReference = updateExternalModuleReference;
-    // JSDoc
-    /* @internal */
-    function createJSDocTypeExpression(type) {
-        var node = createSynthesizedNode(294 /* JSDocTypeExpression */);
-        node.type = type;
-        return node;
-    }
-    ts.createJSDocTypeExpression = createJSDocTypeExpression;
-    /* @internal */
-    function createJSDocTypeTag(typeExpression, comment) {
-        var tag = createJSDocTag(320 /* JSDocTypeTag */, "type");
-        tag.typeExpression = typeExpression;
-        tag.comment = comment;
-        return tag;
-    }
-    ts.createJSDocTypeTag = createJSDocTypeTag;
-    /* @internal */
-    function createJSDocReturnTag(typeExpression, comment) {
-        var tag = createJSDocTag(318 /* JSDocReturnTag */, "returns");
-        tag.typeExpression = typeExpression;
-        tag.comment = comment;
-        return tag;
-    }
-    ts.createJSDocReturnTag = createJSDocReturnTag;
-    /** @internal */
-    function createJSDocThisTag(typeExpression) {
-        var tag = createJSDocTag(319 /* JSDocThisTag */, "this");
-        tag.typeExpression = typeExpression;
-        return tag;
-    }
-    ts.createJSDocThisTag = createJSDocThisTag;
-    /* @internal */
-    function createJSDocParamTag(name, isBracketed, typeExpression, comment) {
-        var tag = createJSDocTag(317 /* JSDocParameterTag */, "param");
-        tag.typeExpression = typeExpression;
-        tag.name = name;
-        tag.isBracketed = isBracketed;
-        tag.comment = comment;
-        return tag;
-    }
-    ts.createJSDocParamTag = createJSDocParamTag;
-    /* @internal */
-    function createJSDocClassTag() {
-        return createJSDocTag(310 /* JSDocClassTag */, "class");
-    }
-    ts.createJSDocClassTag = createJSDocClassTag;
-    /* @internal */
-    function createJSDocComment(comment, tags) {
-        var node = createSynthesizedNode(303 /* JSDocComment */);
-        node.comment = comment;
-        node.tags = tags;
-        return node;
-    }
-    ts.createJSDocComment = createJSDocComment;
-    /* @internal */
-    function createJSDocTag(kind, tagName) {
-        var node = createSynthesizedNode(kind);
-        node.tagName = createIdentifier(tagName);
-        return node;
-    }
-    // JSX
-    function createJsxElement(openingElement, children, closingElement) {
-        var node = createSynthesizedNode(266 /* JsxElement */);
-        node.openingElement = openingElement;
-        node.children = createNodeArray(children);
-        node.closingElement = closingElement;
-        return node;
-    }
-    ts.createJsxElement = createJsxElement;
-    function updateJsxElement(node, openingElement, children, closingElement) {
-        return node.openingElement !== openingElement
-            || node.children !== children
-            || node.closingElement !== closingElement
-            ? updateNode(createJsxElement(openingElement, children, closingElement), node)
-            : node;
-    }
-    ts.updateJsxElement = updateJsxElement;
-    function createJsxSelfClosingElement(tagName, typeArguments, attributes) {
-        var node = createSynthesizedNode(267 /* JsxSelfClosingElement */);
-        node.tagName = tagName;
-        node.typeArguments = asNodeArray(typeArguments);
-        node.attributes = attributes;
-        return node;
-    }
-    ts.createJsxSelfClosingElement = createJsxSelfClosingElement;
-    function updateJsxSelfClosingElement(node, tagName, typeArguments, attributes) {
-        return node.tagName !== tagName
-            || node.typeArguments !== typeArguments
-            || node.attributes !== attributes
-            ? updateNode(createJsxSelfClosingElement(tagName, typeArguments, attributes), node)
-            : node;
-    }
-    ts.updateJsxSelfClosingElement = updateJsxSelfClosingElement;
-    function createJsxOpeningElement(tagName, typeArguments, attributes) {
-        var node = createSynthesizedNode(268 /* JsxOpeningElement */);
-        node.tagName = tagName;
-        node.typeArguments = asNodeArray(typeArguments);
-        node.attributes = attributes;
-        return node;
-    }
-    ts.createJsxOpeningElement = createJsxOpeningElement;
-    function updateJsxOpeningElement(node, tagName, typeArguments, attributes) {
-        return node.tagName !== tagName
-            || node.typeArguments !== typeArguments
-            || node.attributes !== attributes
-            ? updateNode(createJsxOpeningElement(tagName, typeArguments, attributes), node)
-            : node;
-    }
-    ts.updateJsxOpeningElement = updateJsxOpeningElement;
-    function createJsxClosingElement(tagName) {
-        var node = createSynthesizedNode(269 /* JsxClosingElement */);
-        node.tagName = tagName;
-        return node;
-    }
-    ts.createJsxClosingElement = createJsxClosingElement;
-    function updateJsxClosingElement(node, tagName) {
-        return node.tagName !== tagName
-            ? updateNode(createJsxClosingElement(tagName), node)
-            : node;
-    }
-    ts.updateJsxClosingElement = updateJsxClosingElement;
-    function createJsxFragment(openingFragment, children, closingFragment) {
-        var node = createSynthesizedNode(270 /* JsxFragment */);
-        node.openingFragment = openingFragment;
-        node.children = createNodeArray(children);
-        node.closingFragment = closingFragment;
-        return node;
-    }
-    ts.createJsxFragment = createJsxFragment;
-    function createJsxText(text, containsOnlyTriviaWhiteSpaces) {
-        var node = createSynthesizedNode(11 /* JsxText */);
-        node.text = text;
-        node.containsOnlyTriviaWhiteSpaces = !!containsOnlyTriviaWhiteSpaces;
-        return node;
-    }
-    ts.createJsxText = createJsxText;
-    function updateJsxText(node, text, containsOnlyTriviaWhiteSpaces) {
-        return node.text !== text
-            || node.containsOnlyTriviaWhiteSpaces !== containsOnlyTriviaWhiteSpaces
-            ? updateNode(createJsxText(text, containsOnlyTriviaWhiteSpaces), node)
-            : node;
-    }
-    ts.updateJsxText = updateJsxText;
-    function createJsxOpeningFragment() {
-        return createSynthesizedNode(271 /* JsxOpeningFragment */);
-    }
-    ts.createJsxOpeningFragment = createJsxOpeningFragment;
-    function createJsxJsxClosingFragment() {
-        return createSynthesizedNode(272 /* JsxClosingFragment */);
-    }
-    ts.createJsxJsxClosingFragment = createJsxJsxClosingFragment;
-    function updateJsxFragment(node, openingFragment, children, closingFragment) {
-        return node.openingFragment !== openingFragment
-            || node.children !== children
-            || node.closingFragment !== closingFragment
-            ? updateNode(createJsxFragment(openingFragment, children, closingFragment), node)
-            : node;
-    }
-    ts.updateJsxFragment = updateJsxFragment;
-    function createJsxAttribute(name, initializer) {
-        var node = createSynthesizedNode(273 /* JsxAttribute */);
-        node.name = name;
-        node.initializer = initializer;
-        return node;
-    }
-    ts.createJsxAttribute = createJsxAttribute;
-    function updateJsxAttribute(node, name, initializer) {
-        return node.name !== name
-            || node.initializer !== initializer
-            ? updateNode(createJsxAttribute(name, initializer), node)
-            : node;
-    }
-    ts.updateJsxAttribute = updateJsxAttribute;
-    function createJsxAttributes(properties) {
-        var node = createSynthesizedNode(274 /* JsxAttributes */);
-        node.properties = createNodeArray(properties);
-        return node;
-    }
-    ts.createJsxAttributes = createJsxAttributes;
-    function updateJsxAttributes(node, properties) {
-        return node.properties !== properties
-            ? updateNode(createJsxAttributes(properties), node)
-            : node;
-    }
-    ts.updateJsxAttributes = updateJsxAttributes;
-    function createJsxSpreadAttribute(expression) {
-        var node = createSynthesizedNode(275 /* JsxSpreadAttribute */);
-        node.expression = expression;
-        return node;
-    }
-    ts.createJsxSpreadAttribute = createJsxSpreadAttribute;
-    function updateJsxSpreadAttribute(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createJsxSpreadAttribute(expression), node)
-            : node;
-    }
-    ts.updateJsxSpreadAttribute = updateJsxSpreadAttribute;
-    function createJsxExpression(dotDotDotToken, expression) {
-        var node = createSynthesizedNode(276 /* JsxExpression */);
-        node.dotDotDotToken = dotDotDotToken;
-        node.expression = expression;
-        return node;
-    }
-    ts.createJsxExpression = createJsxExpression;
-    function updateJsxExpression(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createJsxExpression(node.dotDotDotToken, expression), node)
-            : node;
-    }
-    ts.updateJsxExpression = updateJsxExpression;
-    // Clauses
-    function createCaseClause(expression, statements) {
-        var node = createSynthesizedNode(277 /* CaseClause */);
-        node.expression = ts.parenthesizeExpressionForList(expression);
-        node.statements = createNodeArray(statements);
-        return node;
-    }
-    ts.createCaseClause = createCaseClause;
-    function updateCaseClause(node, expression, statements) {
-        return node.expression !== expression
-            || node.statements !== statements
-            ? updateNode(createCaseClause(expression, statements), node)
-            : node;
-    }
-    ts.updateCaseClause = updateCaseClause;
-    function createDefaultClause(statements) {
-        var node = createSynthesizedNode(278 /* DefaultClause */);
-        node.statements = createNodeArray(statements);
-        return node;
-    }
-    ts.createDefaultClause = createDefaultClause;
-    function updateDefaultClause(node, statements) {
-        return node.statements !== statements
-            ? updateNode(createDefaultClause(statements), node)
-            : node;
-    }
-    ts.updateDefaultClause = updateDefaultClause;
-    function createHeritageClause(token, types) {
-        var node = createSynthesizedNode(279 /* HeritageClause */);
-        node.token = token;
-        node.types = createNodeArray(types);
-        return node;
-    }
-    ts.createHeritageClause = createHeritageClause;
-    function updateHeritageClause(node, types) {
-        return node.types !== types
-            ? updateNode(createHeritageClause(node.token, types), node)
-            : node;
-    }
-    ts.updateHeritageClause = updateHeritageClause;
-    function createCatchClause(variableDeclaration, block) {
-        var node = createSynthesizedNode(280 /* CatchClause */);
-        node.variableDeclaration = ts.isString(variableDeclaration) ? createVariableDeclaration(variableDeclaration) : variableDeclaration;
-        node.block = block;
-        return node;
-    }
-    ts.createCatchClause = createCatchClause;
-    function updateCatchClause(node, variableDeclaration, block) {
-        return node.variableDeclaration !== variableDeclaration
-            || node.block !== block
-            ? updateNode(createCatchClause(variableDeclaration, block), node)
-            : node;
-    }
-    ts.updateCatchClause = updateCatchClause;
-    // Property assignments
-    function createPropertyAssignment(name, initializer) {
-        var node = createSynthesizedNode(281 /* PropertyAssignment */);
-        node.name = asName(name);
-        node.questionToken = undefined;
-        node.initializer = ts.parenthesizeExpressionForList(initializer);
-        return node;
-    }
-    ts.createPropertyAssignment = createPropertyAssignment;
-    function updatePropertyAssignment(node, name, initializer) {
-        return node.name !== name
-            || node.initializer !== initializer
-            ? updateNode(createPropertyAssignment(name, initializer), node)
-            : node;
-    }
-    ts.updatePropertyAssignment = updatePropertyAssignment;
-    function createShorthandPropertyAssignment(name, objectAssignmentInitializer) {
-        var node = createSynthesizedNode(282 /* ShorthandPropertyAssignment */);
-        node.name = asName(name);
-        node.objectAssignmentInitializer = objectAssignmentInitializer !== undefined ? ts.parenthesizeExpressionForList(objectAssignmentInitializer) : undefined;
-        return node;
-    }
-    ts.createShorthandPropertyAssignment = createShorthandPropertyAssignment;
-    function updateShorthandPropertyAssignment(node, name, objectAssignmentInitializer) {
-        return node.name !== name
-            || node.objectAssignmentInitializer !== objectAssignmentInitializer
-            ? updateNode(createShorthandPropertyAssignment(name, objectAssignmentInitializer), node)
-            : node;
-    }
-    ts.updateShorthandPropertyAssignment = updateShorthandPropertyAssignment;
-    function createSpreadAssignment(expression) {
-        var node = createSynthesizedNode(283 /* SpreadAssignment */);
-        node.expression = ts.parenthesizeExpressionForList(expression);
-        return node;
-    }
-    ts.createSpreadAssignment = createSpreadAssignment;
-    function updateSpreadAssignment(node, expression) {
-        return node.expression !== expression
-            ? updateNode(createSpreadAssignment(expression), node)
-            : node;
-    }
-    ts.updateSpreadAssignment = updateSpreadAssignment;
-    // Enum
-    function createEnumMember(name, initializer) {
-        var node = createSynthesizedNode(284 /* EnumMember */);
-        node.name = asName(name);
-        node.initializer = initializer && ts.parenthesizeExpressionForList(initializer);
-        return node;
-    }
-    ts.createEnumMember = createEnumMember;
-    function updateEnumMember(node, name, initializer) {
-        return node.name !== name
-            || node.initializer !== initializer
-            ? updateNode(createEnumMember(name, initializer), node)
-            : node;
-    }
-    ts.updateEnumMember = updateEnumMember;
-    // Top-level nodes
-    function updateSourceFileNode(node, statements, isDeclarationFile, referencedFiles, typeReferences, hasNoDefaultLib, libReferences) {
-        if (node.statements !== statements ||
-            (isDeclarationFile !== undefined && node.isDeclarationFile !== isDeclarationFile) ||
-            (referencedFiles !== undefined && node.referencedFiles !== referencedFiles) ||
-            (typeReferences !== undefined && node.typeReferenceDirectives !== typeReferences) ||
-            (libReferences !== undefined && node.libReferenceDirectives !== libReferences) ||
-            (hasNoDefaultLib !== undefined && node.hasNoDefaultLib !== hasNoDefaultLib)) {
-            var updated = createSynthesizedNode(290 /* SourceFile */);
-            updated.flags |= node.flags;
-            updated.statements = createNodeArray(statements);
-            updated.endOfFileToken = node.endOfFileToken;
-            updated.fileName = node.fileName;
-            updated.path = node.path;
-            updated.text = node.text;
-            updated.isDeclarationFile = isDeclarationFile === undefined ? node.isDeclarationFile : isDeclarationFile;
-            updated.referencedFiles = referencedFiles === undefined ? node.referencedFiles : referencedFiles;
-            updated.typeReferenceDirectives = typeReferences === undefined ? node.typeReferenceDirectives : typeReferences;
-            updated.hasNoDefaultLib = hasNoDefaultLib === undefined ? node.hasNoDefaultLib : hasNoDefaultLib;
-            updated.libReferenceDirectives = libReferences === undefined ? node.libReferenceDirectives : libReferences;
-            if (node.amdDependencies !== undefined)
-                updated.amdDependencies = node.amdDependencies;
-            if (node.moduleName !== undefined)
-                updated.moduleName = node.moduleName;
-            if (node.languageVariant !== undefined)
-                updated.languageVariant = node.languageVariant;
-            if (node.renamedDependencies !== undefined)
-                updated.renamedDependencies = node.renamedDependencies;
-            if (node.languageVersion !== undefined)
-                updated.languageVersion = node.languageVersion;
-            if (node.scriptKind !== undefined)
-                updated.scriptKind = node.scriptKind;
-            if (node.externalModuleIndicator !== undefined)
-                updated.externalModuleIndicator = node.externalModuleIndicator;
-            if (node.commonJsModuleIndicator !== undefined)
-                updated.commonJsModuleIndicator = node.commonJsModuleIndicator;
-            if (node.identifiers !== undefined)
-                updated.identifiers = node.identifiers;
-            if (node.nodeCount !== undefined)
-                updated.nodeCount = node.nodeCount;
-            if (node.identifierCount !== undefined)
-                updated.identifierCount = node.identifierCount;
-            if (node.symbolCount !== undefined)
-                updated.symbolCount = node.symbolCount;
-            if (node.parseDiagnostics !== undefined)
-                updated.parseDiagnostics = node.parseDiagnostics;
-            if (node.bindDiagnostics !== undefined)
-                updated.bindDiagnostics = node.bindDiagnostics;
-            if (node.bindSuggestionDiagnostics !== undefined)
-                updated.bindSuggestionDiagnostics = node.bindSuggestionDiagnostics;
-            if (node.lineMap !== undefined)
-                updated.lineMap = node.lineMap;
-            if (node.classifiableNames !== undefined)
-                updated.classifiableNames = node.classifiableNames;
-            if (node.resolvedModules !== undefined)
-                updated.resolvedModules = node.resolvedModules;
-            if (node.resolvedTypeReferenceDirectiveNames !== undefined)
-                updated.resolvedTypeReferenceDirectiveNames = node.resolvedTypeReferenceDirectiveNames;
-            if (node.imports !== undefined)
-                updated.imports = node.imports;
-            if (node.moduleAugmentations !== undefined)
-                updated.moduleAugmentations = node.moduleAugmentations;
-            if (node.pragmas !== undefined)
-                updated.pragmas = node.pragmas;
-            if (node.localJsxFactory !== undefined)
-                updated.localJsxFactory = node.localJsxFactory;
-            if (node.localJsxNamespace !== undefined)
-                updated.localJsxNamespace = node.localJsxNamespace;
-            return updateNode(updated, node);
+        function createTypeOfDeclaration(declarationIn, enclosingDeclaration, flags, tracker, addUndefined) {
+            var declaration = ts.getParseTreeNode(declarationIn, ts.isVariableLikeOrAccessor);
+            if (!declaration) {
+                return ts.factory.createToken(128 /* AnyKeyword */);
+            }
+            // Get type of the symbol if this is the valid symbol otherwise get type at location
+            var symbol = getSymbolOfNode(declaration);
+            var type = symbol && !(symbol.flags & (2048 /* TypeLiteral */ | 131072 /* Signature */))
+                ? getWidenedLiteralType(getTypeOfSymbol(symbol))
+                : errorType;
+            if (type.flags & 8192 /* UniqueESSymbol */ &&
+                type.symbol === symbol) {
+                flags |= 1048576 /* AllowUniqueESSymbolType */;
+            }
+            if (addUndefined) {
+                type = getOptionalType(type);
+            }
+            return nodeBuilder.typeToTypeNode(type, enclosingDeclaration, flags | 1024 /* MultilineObjectLiterals */, tracker);
         }
-        return node;
-    }
-    ts.updateSourceFileNode = updateSourceFileNode;
-    /**
-     * Creates a shallow, memberwise clone of a node for mutation.
-     */
-    function getMutableClone(node) {
-        var clone = getSynthesizedClone(node);
-        clone.pos = node.pos;
-        clone.end = node.end;
-        clone.parent = node.parent;
-        return clone;
-    }
-    ts.getMutableClone = getMutableClone;
-    // Transformation nodes
-    /**
-     * Creates a synthetic statement to act as a placeholder for a not-emitted statement in
-     * order to preserve comments.
-     *
-     * @param original The original statement.
-     */
-    function createNotEmittedStatement(original) {
-        var node = createSynthesizedNode(325 /* NotEmittedStatement */);
-        node.original = original;
-        setTextRange(node, original);
-        return node;
-    }
-    ts.createNotEmittedStatement = createNotEmittedStatement;
-    /**
-     * Creates a synthetic element to act as a placeholder for the end of an emitted declaration in
-     * order to properly emit exports.
-     */
-    /* @internal */
-    function createEndOfDeclarationMarker(original) {
-        var node = createSynthesizedNode(329 /* EndOfDeclarationMarker */);
-        node.emitNode = {};
-        node.original = original;
-        return node;
-    }
-    ts.createEndOfDeclarationMarker = createEndOfDeclarationMarker;
-    /**
-     * Creates a synthetic element to act as a placeholder for the beginning of a merged declaration in
-     * order to properly emit exports.
-     */
-    /* @internal */
-    function createMergeDeclarationMarker(original) {
-        var node = createSynthesizedNode(328 /* MergeDeclarationMarker */);
-        node.emitNode = {};
-        node.original = original;
-        return node;
-    }
-    ts.createMergeDeclarationMarker = createMergeDeclarationMarker;
-    /**
-     * Creates a synthetic expression to act as a placeholder for a not-emitted expression in
-     * order to preserve comments or sourcemap positions.
-     *
-     * @param expression The inner expression to emit.
-     * @param original The original outer expression.
-     * @param location The location for the expression. Defaults to the positions from "original" if provided.
-     */
-    function createPartiallyEmittedExpression(expression, original) {
-        var node = createSynthesizedNode(326 /* PartiallyEmittedExpression */);
-        node.expression = expression;
-        node.original = original;
-        setTextRange(node, original);
-        return node;
-    }
-    ts.createPartiallyEmittedExpression = createPartiallyEmittedExpression;
-    function updatePartiallyEmittedExpression(node, expression) {
-        if (node.expression !== expression) {
-            return updateNode(createPartiallyEmittedExpression(expression, node.original), node);
+        function createReturnTypeOfSignatureDeclaration(signatureDeclarationIn, enclosingDeclaration, flags, tracker) {
+            var signatureDeclaration = ts.getParseTreeNode(signatureDeclarationIn, ts.isFunctionLike);
+            if (!signatureDeclaration) {
+                return ts.factory.createToken(128 /* AnyKeyword */);
+            }
+            var signature = getSignatureFromDeclaration(signatureDeclaration);
+            return nodeBuilder.typeToTypeNode(getReturnTypeOfSignature(signature), enclosingDeclaration, flags | 1024 /* MultilineObjectLiterals */, tracker);
         }
-        return node;
-    }
-    ts.updatePartiallyEmittedExpression = updatePartiallyEmittedExpression;
-    function flattenCommaElements(node) {
-        if (ts.nodeIsSynthesized(node) && !ts.isParseTreeNode(node) && !node.original && !node.emitNode && !node.id) {
-            if (node.kind === 327 /* CommaListExpression */) {
-                return node.elements;
+        function createTypeOfExpression(exprIn, enclosingDeclaration, flags, tracker) {
+            var expr = ts.getParseTreeNode(exprIn, ts.isExpression);
+            if (!expr) {
+                return ts.factory.createToken(128 /* AnyKeyword */);
+            }
+            var type = getWidenedType(getRegularTypeOfExpression(expr));
+            return nodeBuilder.typeToTypeNode(type, enclosingDeclaration, flags | 1024 /* MultilineObjectLiterals */, tracker);
+        }
+        function hasGlobalName(name) {
+            return globals.has(ts.escapeLeadingUnderscores(name));
+        }
+        function getReferencedValueSymbol(reference, startInDeclarationContainer) {
+            var resolvedSymbol = getNodeLinks(reference).resolvedSymbol;
+            if (resolvedSymbol) {
+                return resolvedSymbol;
             }
-            if (ts.isBinaryExpression(node) && node.operatorToken.kind === 27 /* CommaToken */) {
-                return [node.left, node.right];
+            var location = reference;
+            if (startInDeclarationContainer) {
+                // When resolving the name of a declaration as a value, we need to start resolution
+                // at a point outside of the declaration.
+                var parent = reference.parent;
+                if (ts.isDeclaration(parent) && reference === parent.name) {
+                    location = getDeclarationContainer(parent);
+                }
             }
+            return resolveName(location, reference.escapedText, 111551 /* Value */ | 1048576 /* ExportValue */ | 2097152 /* Alias */, /*nodeNotFoundMessage*/ undefined, /*nameArg*/ undefined, /*isUse*/ true);
         }
-        return node;
-    }
-    function createCommaList(elements) {
-        var node = createSynthesizedNode(327 /* CommaListExpression */);
-        node.elements = createNodeArray(ts.sameFlatMap(elements, flattenCommaElements));
-        return node;
-    }
-    ts.createCommaList = createCommaList;
-    function updateCommaList(node, elements) {
-        return node.elements !== elements
-            ? updateNode(createCommaList(elements), node)
-            : node;
-    }
-    ts.updateCommaList = updateCommaList;
-    /* @internal */
-    function createSyntheticReferenceExpression(expression, thisArg) {
-        var node = createSynthesizedNode(330 /* SyntheticReferenceExpression */);
-        node.expression = expression;
-        node.thisArg = thisArg;
-        return node;
-    }
-    ts.createSyntheticReferenceExpression = createSyntheticReferenceExpression;
-    /* @internal */
-    function updateSyntheticReferenceExpression(node, expression, thisArg) {
-        return node.expression !== expression
-            || node.thisArg !== thisArg
-            ? updateNode(createSyntheticReferenceExpression(expression, thisArg), node)
-            : node;
-    }
-    ts.updateSyntheticReferenceExpression = updateSyntheticReferenceExpression;
-    function createBundle(sourceFiles, prepends) {
-        if (prepends === void 0) { prepends = ts.emptyArray; }
-        var node = ts.createNode(291 /* Bundle */);
-        node.prepends = prepends;
-        node.sourceFiles = sourceFiles;
-        return node;
-    }
-    ts.createBundle = createBundle;
-    var allUnscopedEmitHelpers;
-    function getAllUnscopedEmitHelpers() {
-        return allUnscopedEmitHelpers || (allUnscopedEmitHelpers = ts.arrayToMap([
-            ts.valuesHelper,
-            ts.readHelper,
-            ts.spreadHelper,
-            ts.spreadArraysHelper,
-            ts.restHelper,
-            ts.decorateHelper,
-            ts.metadataHelper,
-            ts.paramHelper,
-            ts.awaiterHelper,
-            ts.assignHelper,
-            ts.awaitHelper,
-            ts.asyncGeneratorHelper,
-            ts.asyncDelegator,
-            ts.asyncValues,
-            ts.extendsHelper,
-            ts.templateObjectHelper,
-            ts.generatorHelper,
-            ts.importStarHelper,
-            ts.importDefaultHelper,
-            ts.classPrivateFieldGetHelper,
-            ts.classPrivateFieldSetHelper,
-            ts.createBindingHelper,
-            ts.setModuleDefaultHelper
-        ], function (helper) { return helper.name; }));
-    }
-    function createUnparsedSource() {
-        var node = ts.createNode(292 /* UnparsedSource */);
-        node.prologues = ts.emptyArray;
-        node.referencedFiles = ts.emptyArray;
-        node.libReferenceDirectives = ts.emptyArray;
-        node.getLineAndCharacterOfPosition = function (pos) { return ts.getLineAndCharacterOfPosition(node, pos); };
-        return node;
-    }
-    function createUnparsedSourceFile(textOrInputFiles, mapPathOrType, mapTextOrStripInternal) {
-        var node = createUnparsedSource();
-        var stripInternal;
-        var bundleFileInfo;
-        if (!ts.isString(textOrInputFiles)) {
-            ts.Debug.assert(mapPathOrType === "js" || mapPathOrType === "dts");
-            node.fileName = (mapPathOrType === "js" ? textOrInputFiles.javascriptPath : textOrInputFiles.declarationPath) || "";
-            node.sourceMapPath = mapPathOrType === "js" ? textOrInputFiles.javascriptMapPath : textOrInputFiles.declarationMapPath;
-            Object.defineProperties(node, {
-                text: { get: function () { return mapPathOrType === "js" ? textOrInputFiles.javascriptText : textOrInputFiles.declarationText; } },
-                sourceMapText: { get: function () { return mapPathOrType === "js" ? textOrInputFiles.javascriptMapText : textOrInputFiles.declarationMapText; } },
-            });
-            if (textOrInputFiles.buildInfo && textOrInputFiles.buildInfo.bundle) {
-                node.oldFileOfCurrentEmit = textOrInputFiles.oldFileOfCurrentEmit;
-                ts.Debug.assert(mapTextOrStripInternal === undefined || typeof mapTextOrStripInternal === "boolean");
-                stripInternal = mapTextOrStripInternal;
-                bundleFileInfo = mapPathOrType === "js" ? textOrInputFiles.buildInfo.bundle.js : textOrInputFiles.buildInfo.bundle.dts;
-                if (node.oldFileOfCurrentEmit) {
-                    parseOldFileOfCurrentEmit(node, ts.Debug.checkDefined(bundleFileInfo));
-                    return node;
+        function getReferencedValueDeclaration(referenceIn) {
+            if (!ts.isGeneratedIdentifier(referenceIn)) {
+                var reference = ts.getParseTreeNode(referenceIn, ts.isIdentifier);
+                if (reference) {
+                    var symbol = getReferencedValueSymbol(reference);
+                    if (symbol) {
+                        return getExportSymbolOfValueSymbolIfExported(symbol).valueDeclaration;
+                    }
                 }
             }
+            return undefined;
         }
-        else {
-            node.fileName = "";
-            node.text = textOrInputFiles;
-            node.sourceMapPath = mapPathOrType;
-            node.sourceMapText = mapTextOrStripInternal;
+        function isLiteralConstDeclaration(node) {
+            if (ts.isDeclarationReadonly(node) || ts.isVariableDeclaration(node) && ts.isVarConst(node)) {
+                return isFreshLiteralType(getTypeOfSymbol(getSymbolOfNode(node)));
+            }
+            return false;
         }
-        ts.Debug.assert(!node.oldFileOfCurrentEmit);
-        parseUnparsedSourceFile(node, bundleFileInfo, stripInternal);
-        return node;
-    }
-    ts.createUnparsedSourceFile = createUnparsedSourceFile;
-    function parseUnparsedSourceFile(node, bundleFileInfo, stripInternal) {
-        var prologues;
-        var helpers;
-        var referencedFiles;
-        var typeReferenceDirectives;
-        var libReferenceDirectives;
-        var texts;
-        for (var _i = 0, _a = bundleFileInfo ? bundleFileInfo.sections : ts.emptyArray; _i < _a.length; _i++) {
-            var section = _a[_i];
-            switch (section.kind) {
-                case "prologue" /* Prologue */:
-                    (prologues || (prologues = [])).push(createUnparsedNode(section, node));
-                    break;
-                case "emitHelpers" /* EmitHelpers */:
-                    (helpers || (helpers = [])).push(getAllUnscopedEmitHelpers().get(section.data));
-                    break;
-                case "no-default-lib" /* NoDefaultLib */:
-                    node.hasNoDefaultLib = true;
-                    break;
-                case "reference" /* Reference */:
-                    (referencedFiles || (referencedFiles = [])).push({ pos: -1, end: -1, fileName: section.data });
-                    break;
-                case "type" /* Type */:
-                    (typeReferenceDirectives || (typeReferenceDirectives = [])).push(section.data);
-                    break;
-                case "lib" /* Lib */:
-                    (libReferenceDirectives || (libReferenceDirectives = [])).push({ pos: -1, end: -1, fileName: section.data });
-                    break;
-                case "prepend" /* Prepend */:
-                    var prependNode = createUnparsedNode(section, node);
-                    var prependTexts = void 0;
-                    for (var _b = 0, _c = section.texts; _b < _c.length; _b++) {
-                        var text = _c[_b];
-                        if (!stripInternal || text.kind !== "internal" /* Internal */) {
-                            (prependTexts || (prependTexts = [])).push(createUnparsedNode(text, node));
+        function literalTypeToNode(type, enclosing, tracker) {
+            var enumResult = type.flags & 1024 /* EnumLiteral */ ? nodeBuilder.symbolToExpression(type.symbol, 111551 /* Value */, enclosing, /*flags*/ undefined, tracker)
+                : type === trueType ? ts.factory.createTrue() : type === falseType && ts.factory.createFalse();
+            if (enumResult)
+                return enumResult;
+            var literalValue = type.value;
+            return typeof literalValue === "object" ? ts.factory.createBigIntLiteral(literalValue) :
+                typeof literalValue === "number" ? ts.factory.createNumericLiteral(literalValue) :
+                    ts.factory.createStringLiteral(literalValue);
+        }
+        function createLiteralConstValue(node, tracker) {
+            var type = getTypeOfSymbol(getSymbolOfNode(node));
+            return literalTypeToNode(type, node, tracker);
+        }
+        function getJsxFactoryEntity(location) {
+            return location ? (getJsxNamespace(location), (ts.getSourceFileOfNode(location).localJsxFactory || _jsxFactoryEntity)) : _jsxFactoryEntity;
+        }
+        function getJsxFragmentFactoryEntity(location) {
+            if (location) {
+                var file = ts.getSourceFileOfNode(location);
+                if (file) {
+                    if (file.localJsxFragmentFactory) {
+                        return file.localJsxFragmentFactory;
+                    }
+                    var jsxFragPragmas = file.pragmas.get("jsxfrag");
+                    var jsxFragPragma = ts.isArray(jsxFragPragmas) ? jsxFragPragmas[0] : jsxFragPragmas;
+                    if (jsxFragPragma) {
+                        file.localJsxFragmentFactory = ts.parseIsolatedEntityName(jsxFragPragma.arguments.factory, languageVersion);
+                        return file.localJsxFragmentFactory;
+                    }
+                }
+            }
+            if (compilerOptions.jsxFragmentFactory) {
+                return ts.parseIsolatedEntityName(compilerOptions.jsxFragmentFactory, languageVersion);
+            }
+        }
+        function createResolver() {
+            // this variable and functions that use it are deliberately moved here from the outer scope
+            // to avoid scope pollution
+            var resolvedTypeReferenceDirectives = host.getResolvedTypeReferenceDirectives();
+            var fileToDirective;
+            if (resolvedTypeReferenceDirectives) {
+                // populate reverse mapping: file path -> type reference directive that was resolved to this file
+                fileToDirective = new ts.Map();
+                resolvedTypeReferenceDirectives.forEach(function (resolvedDirective, key) {
+                    if (!resolvedDirective || !resolvedDirective.resolvedFileName) {
+                        return;
+                    }
+                    var file = host.getSourceFile(resolvedDirective.resolvedFileName);
+                    if (file) {
+                        // Add the transitive closure of path references loaded by this file (as long as they are not)
+                        // part of an existing type reference.
+                        addReferencedFilesToTypeDirective(file, key);
+                    }
+                });
+            }
+            return {
+                getReferencedExportContainer: getReferencedExportContainer,
+                getReferencedImportDeclaration: getReferencedImportDeclaration,
+                getReferencedDeclarationWithCollidingName: getReferencedDeclarationWithCollidingName,
+                isDeclarationWithCollidingName: isDeclarationWithCollidingName,
+                isValueAliasDeclaration: function (nodeIn) {
+                    var node = ts.getParseTreeNode(nodeIn);
+                    // Synthesized nodes are always treated like values.
+                    return node ? isValueAliasDeclaration(node) : true;
+                },
+                hasGlobalName: hasGlobalName,
+                isReferencedAliasDeclaration: function (nodeIn, checkChildren) {
+                    var node = ts.getParseTreeNode(nodeIn);
+                    // Synthesized nodes are always treated as referenced.
+                    return node ? isReferencedAliasDeclaration(node, checkChildren) : true;
+                },
+                getNodeCheckFlags: function (nodeIn) {
+                    var node = ts.getParseTreeNode(nodeIn);
+                    return node ? getNodeCheckFlags(node) : 0;
+                },
+                isTopLevelValueImportEqualsWithEntityName: isTopLevelValueImportEqualsWithEntityName,
+                isDeclarationVisible: isDeclarationVisible,
+                isImplementationOfOverload: isImplementationOfOverload,
+                isRequiredInitializedParameter: isRequiredInitializedParameter,
+                isOptionalUninitializedParameterProperty: isOptionalUninitializedParameterProperty,
+                isExpandoFunctionDeclaration: isExpandoFunctionDeclaration,
+                getPropertiesOfContainerFunction: getPropertiesOfContainerFunction,
+                createTypeOfDeclaration: createTypeOfDeclaration,
+                createReturnTypeOfSignatureDeclaration: createReturnTypeOfSignatureDeclaration,
+                createTypeOfExpression: createTypeOfExpression,
+                createLiteralConstValue: createLiteralConstValue,
+                isSymbolAccessible: isSymbolAccessible,
+                isEntityNameVisible: isEntityNameVisible,
+                getConstantValue: function (nodeIn) {
+                    var node = ts.getParseTreeNode(nodeIn, canHaveConstantValue);
+                    return node ? getConstantValue(node) : undefined;
+                },
+                collectLinkedAliases: collectLinkedAliases,
+                getReferencedValueDeclaration: getReferencedValueDeclaration,
+                getTypeReferenceSerializationKind: getTypeReferenceSerializationKind,
+                isOptionalParameter: isOptionalParameter,
+                moduleExportsSomeValue: moduleExportsSomeValue,
+                isArgumentsLocalBinding: isArgumentsLocalBinding,
+                getExternalModuleFileFromDeclaration: getExternalModuleFileFromDeclaration,
+                getTypeReferenceDirectivesForEntityName: getTypeReferenceDirectivesForEntityName,
+                getTypeReferenceDirectivesForSymbol: getTypeReferenceDirectivesForSymbol,
+                isLiteralConstDeclaration: isLiteralConstDeclaration,
+                isLateBound: function (nodeIn) {
+                    var node = ts.getParseTreeNode(nodeIn, ts.isDeclaration);
+                    var symbol = node && getSymbolOfNode(node);
+                    return !!(symbol && ts.getCheckFlags(symbol) & 4096 /* Late */);
+                },
+                getJsxFactoryEntity: getJsxFactoryEntity,
+                getJsxFragmentFactoryEntity: getJsxFragmentFactoryEntity,
+                getAllAccessorDeclarations: function (accessor) {
+                    accessor = ts.getParseTreeNode(accessor, ts.isGetOrSetAccessorDeclaration); // TODO: GH#18217
+                    var otherKind = accessor.kind === 168 /* SetAccessor */ ? 167 /* GetAccessor */ : 168 /* SetAccessor */;
+                    var otherAccessor = ts.getDeclarationOfKind(getSymbolOfNode(accessor), otherKind);
+                    var firstAccessor = otherAccessor && (otherAccessor.pos < accessor.pos) ? otherAccessor : accessor;
+                    var secondAccessor = otherAccessor && (otherAccessor.pos < accessor.pos) ? accessor : otherAccessor;
+                    var setAccessor = accessor.kind === 168 /* SetAccessor */ ? accessor : otherAccessor;
+                    var getAccessor = accessor.kind === 167 /* GetAccessor */ ? accessor : otherAccessor;
+                    return {
+                        firstAccessor: firstAccessor,
+                        secondAccessor: secondAccessor,
+                        setAccessor: setAccessor,
+                        getAccessor: getAccessor
+                    };
+                },
+                getSymbolOfExternalModuleSpecifier: function (moduleName) { return resolveExternalModuleNameWorker(moduleName, moduleName, /*moduleNotFoundError*/ undefined); },
+                isBindingCapturedByNode: function (node, decl) {
+                    var parseNode = ts.getParseTreeNode(node);
+                    var parseDecl = ts.getParseTreeNode(decl);
+                    return !!parseNode && !!parseDecl && (ts.isVariableDeclaration(parseDecl) || ts.isBindingElement(parseDecl)) && isBindingCapturedByNode(parseNode, parseDecl);
+                },
+                getDeclarationStatementsForSourceFile: function (node, flags, tracker, bundled) {
+                    var n = ts.getParseTreeNode(node);
+                    ts.Debug.assert(n && n.kind === 297 /* SourceFile */, "Non-sourcefile node passed into getDeclarationsForSourceFile");
+                    var sym = getSymbolOfNode(node);
+                    if (!sym) {
+                        return !node.locals ? [] : nodeBuilder.symbolTableToDeclarationStatements(node.locals, node, flags, tracker, bundled);
+                    }
+                    return !sym.exports ? [] : nodeBuilder.symbolTableToDeclarationStatements(sym.exports, node, flags, tracker, bundled);
+                },
+                isImportRequiredByAugmentation: isImportRequiredByAugmentation,
+            };
+            function isImportRequiredByAugmentation(node) {
+                var file = ts.getSourceFileOfNode(node);
+                if (!file.symbol)
+                    return false;
+                var importTarget = getExternalModuleFileFromDeclaration(node);
+                if (!importTarget)
+                    return false;
+                if (importTarget === file)
+                    return false;
+                var exports = getExportsOfModule(file.symbol);
+                for (var _i = 0, _a = ts.arrayFrom(exports.values()); _i < _a.length; _i++) {
+                    var s = _a[_i];
+                    if (s.mergeId) {
+                        var merged = getMergedSymbol(s);
+                        for (var _b = 0, _c = merged.declarations; _b < _c.length; _b++) {
+                            var d = _c[_b];
+                            var declFile = ts.getSourceFileOfNode(d);
+                            if (declFile === importTarget) {
+                                return true;
+                            }
                         }
                     }
-                    prependNode.texts = prependTexts || ts.emptyArray;
-                    (texts || (texts = [])).push(prependNode);
-                    break;
-                case "internal" /* Internal */:
-                    if (stripInternal) {
-                        if (!texts)
-                            texts = [];
+                }
+                return false;
+            }
+            function isInHeritageClause(node) {
+                return node.parent && node.parent.kind === 223 /* ExpressionWithTypeArguments */ && node.parent.parent && node.parent.parent.kind === 286 /* HeritageClause */;
+            }
+            // defined here to avoid outer scope pollution
+            function getTypeReferenceDirectivesForEntityName(node) {
+                // program does not have any files with type reference directives - bail out
+                if (!fileToDirective) {
+                    return undefined;
+                }
+                // property access can only be used as values, or types when within an expression with type arguments inside a heritage clause
+                // qualified names can only be used as types\namespaces
+                // identifiers are treated as values only if they appear in type queries
+                var meaning = 788968 /* Type */ | 1920 /* Namespace */;
+                if ((node.kind === 78 /* Identifier */ && isInTypeQuery(node)) || (node.kind === 201 /* PropertyAccessExpression */ && !isInHeritageClause(node))) {
+                    meaning = 111551 /* Value */ | 1048576 /* ExportValue */;
+                }
+                var symbol = resolveEntityName(node, meaning, /*ignoreErrors*/ true);
+                return symbol && symbol !== unknownSymbol ? getTypeReferenceDirectivesForSymbol(symbol, meaning) : undefined;
+            }
+            // defined here to avoid outer scope pollution
+            function getTypeReferenceDirectivesForSymbol(symbol, meaning) {
+                // program does not have any files with type reference directives - bail out
+                if (!fileToDirective) {
+                    return undefined;
+                }
+                if (!isSymbolFromTypeDeclarationFile(symbol)) {
+                    return undefined;
+                }
+                // check what declarations in the symbol can contribute to the target meaning
+                var typeReferenceDirectives;
+                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                    var decl = _a[_i];
+                    // check meaning of the local symbol to see if declaration needs to be analyzed further
+                    if (decl.symbol && decl.symbol.flags & meaning) {
+                        var file = ts.getSourceFileOfNode(decl);
+                        var typeReferenceDirective = fileToDirective.get(file.path);
+                        if (typeReferenceDirective) {
+                            (typeReferenceDirectives || (typeReferenceDirectives = [])).push(typeReferenceDirective);
+                        }
+                        else {
+                            // found at least one entry that does not originate from type reference directive
+                            return undefined;
+                        }
+                    }
+                }
+                return typeReferenceDirectives;
+            }
+            function isSymbolFromTypeDeclarationFile(symbol) {
+                // bail out if symbol does not have associated declarations (i.e. this is transient symbol created for property in binding pattern)
+                if (!symbol.declarations) {
+                    return false;
+                }
+                // walk the parent chain for symbols to make sure that top level parent symbol is in the global scope
+                // external modules cannot define or contribute to type declaration files
+                var current = symbol;
+                while (true) {
+                    var parent = getParentOfSymbol(current);
+                    if (parent) {
+                        current = parent;
+                    }
+                    else {
                         break;
                     }
-                // falls through
-                case "text" /* Text */:
-                    (texts || (texts = [])).push(createUnparsedNode(section, node));
-                    break;
-                default:
-                    ts.Debug.assertNever(section);
+                }
+                if (current.valueDeclaration && current.valueDeclaration.kind === 297 /* SourceFile */ && current.flags & 512 /* ValueModule */) {
+                    return false;
+                }
+                // check that at least one declaration of top level symbol originates from type declaration file
+                for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
+                    var decl = _a[_i];
+                    var file = ts.getSourceFileOfNode(decl);
+                    if (fileToDirective.has(file.path)) {
+                        return true;
+                    }
+                }
+                return false;
             }
-        }
-        node.prologues = prologues || ts.emptyArray;
-        node.helpers = helpers;
-        node.referencedFiles = referencedFiles || ts.emptyArray;
-        node.typeReferenceDirectives = typeReferenceDirectives;
-        node.libReferenceDirectives = libReferenceDirectives || ts.emptyArray;
-        node.texts = texts || [createUnparsedNode({ kind: "text" /* Text */, pos: 0, end: node.text.length }, node)];
-    }
-    function parseOldFileOfCurrentEmit(node, bundleFileInfo) {
-        ts.Debug.assert(!!node.oldFileOfCurrentEmit);
-        var texts;
-        var syntheticReferences;
-        for (var _i = 0, _a = bundleFileInfo.sections; _i < _a.length; _i++) {
-            var section = _a[_i];
-            switch (section.kind) {
-                case "internal" /* Internal */:
-                case "text" /* Text */:
-                    (texts || (texts = [])).push(createUnparsedNode(section, node));
-                    break;
-                case "no-default-lib" /* NoDefaultLib */:
-                case "reference" /* Reference */:
-                case "type" /* Type */:
-                case "lib" /* Lib */:
-                    (syntheticReferences || (syntheticReferences = [])).push(createUnparsedSyntheticReference(section, node));
-                    break;
-                // Ignore
-                case "prologue" /* Prologue */:
-                case "emitHelpers" /* EmitHelpers */:
-                case "prepend" /* Prepend */:
-                    break;
-                default:
-                    ts.Debug.assertNever(section);
+            function addReferencedFilesToTypeDirective(file, key) {
+                if (fileToDirective.has(file.path))
+                    return;
+                fileToDirective.set(file.path, key);
+                for (var _i = 0, _a = file.referencedFiles; _i < _a.length; _i++) {
+                    var fileName = _a[_i].fileName;
+                    var resolvedFile = ts.resolveTripleslashReference(fileName, file.fileName);
+                    var referencedFile = host.getSourceFile(resolvedFile);
+                    if (referencedFile) {
+                        addReferencedFilesToTypeDirective(referencedFile, key);
+                    }
+                }
             }
         }
-        node.texts = texts || ts.emptyArray;
-        node.helpers = ts.map(bundleFileInfo.sources && bundleFileInfo.sources.helpers, function (name) { return getAllUnscopedEmitHelpers().get(name); });
-        node.syntheticReferences = syntheticReferences;
-        return node;
-    }
-    function mapBundleFileSectionKindToSyntaxKind(kind) {
-        switch (kind) {
-            case "prologue" /* Prologue */: return 285 /* UnparsedPrologue */;
-            case "prepend" /* Prepend */: return 286 /* UnparsedPrepend */;
-            case "internal" /* Internal */: return 288 /* UnparsedInternalText */;
-            case "text" /* Text */: return 287 /* UnparsedText */;
-            case "emitHelpers" /* EmitHelpers */:
-            case "no-default-lib" /* NoDefaultLib */:
-            case "reference" /* Reference */:
-            case "type" /* Type */:
-            case "lib" /* Lib */:
-                return ts.Debug.fail("BundleFileSectionKind: " + kind + " not yet mapped to SyntaxKind");
-            default:
-                return ts.Debug.assertNever(kind);
+        function getExternalModuleFileFromDeclaration(declaration) {
+            var specifier = declaration.kind === 256 /* ModuleDeclaration */ ? ts.tryCast(declaration.name, ts.isStringLiteral) : ts.getExternalModuleName(declaration);
+            var moduleSymbol = resolveExternalModuleNameWorker(specifier, specifier, /*moduleNotFoundError*/ undefined); // TODO: GH#18217
+            if (!moduleSymbol) {
+                return undefined;
+            }
+            return ts.getDeclarationOfKind(moduleSymbol, 297 /* SourceFile */);
         }
-    }
-    function createUnparsedNode(section, parent) {
-        var node = ts.createNode(mapBundleFileSectionKindToSyntaxKind(section.kind), section.pos, section.end);
-        node.parent = parent;
-        node.data = section.data;
-        return node;
-    }
-    function createUnparsedSyntheticReference(section, parent) {
-        var node = ts.createNode(289 /* UnparsedSyntheticReference */, section.pos, section.end);
-        node.parent = parent;
-        node.data = section.data;
-        node.section = section;
-        return node;
-    }
-    function createInputFiles(javascriptTextOrReadFileText, declarationTextOrJavascriptPath, javascriptMapPath, javascriptMapTextOrDeclarationPath, declarationMapPath, declarationMapTextOrBuildInfoPath, javascriptPath, declarationPath, buildInfoPath, buildInfo, oldFileOfCurrentEmit) {
-        var node = ts.createNode(293 /* InputFiles */);
-        if (!ts.isString(javascriptTextOrReadFileText)) {
-            var cache_1 = ts.createMap();
-            var textGetter_1 = function (path) {
-                if (path === undefined)
-                    return undefined;
-                var value = cache_1.get(path);
-                if (value === undefined) {
-                    value = javascriptTextOrReadFileText(path);
-                    cache_1.set(path, value !== undefined ? value : false);
+        function initializeTypeChecker() {
+            // Bind all source files and propagate errors
+            for (var _i = 0, _a = host.getSourceFiles(); _i < _a.length; _i++) {
+                var file = _a[_i];
+                ts.bindSourceFile(file, compilerOptions);
+            }
+            amalgamatedDuplicates = new ts.Map();
+            // Initialize global symbol table
+            var augmentations;
+            for (var _b = 0, _c = host.getSourceFiles(); _b < _c.length; _b++) {
+                var file = _c[_b];
+                if (file.redirectInfo) {
+                    continue;
                 }
-                return value !== false ? value : undefined;
-            };
-            var definedTextGetter_1 = function (path) {
-                var result = textGetter_1(path);
-                return result !== undefined ? result : "/* Input file " + path + " was missing */\r\n";
-            };
-            var buildInfo_1;
-            var getAndCacheBuildInfo_1 = function (getText) {
-                if (buildInfo_1 === undefined) {
-                    var result = getText();
-                    buildInfo_1 = result !== undefined ? ts.getBuildInfo(result) : false;
+                if (!ts.isExternalOrCommonJsModule(file)) {
+                    // It is an error for a non-external-module (i.e. script) to declare its own `globalThis`.
+                    // We can't use `builtinGlobals` for this due to synthetic expando-namespace generation in JS files.
+                    var fileGlobalThisSymbol = file.locals.get("globalThis");
+                    if (fileGlobalThisSymbol) {
+                        for (var _d = 0, _e = fileGlobalThisSymbol.declarations; _d < _e.length; _d++) {
+                            var declaration = _e[_d];
+                            diagnostics.add(ts.createDiagnosticForNode(declaration, ts.Diagnostics.Declaration_name_conflicts_with_built_in_global_identifier_0, "globalThis"));
+                        }
+                    }
+                    mergeSymbolTable(globals, file.locals);
+                }
+                if (file.jsGlobalAugmentations) {
+                    mergeSymbolTable(globals, file.jsGlobalAugmentations);
+                }
+                if (file.patternAmbientModules && file.patternAmbientModules.length) {
+                    patternAmbientModules = ts.concatenate(patternAmbientModules, file.patternAmbientModules);
+                }
+                if (file.moduleAugmentations.length) {
+                    (augmentations || (augmentations = [])).push(file.moduleAugmentations);
+                }
+                if (file.symbol && file.symbol.globalExports) {
+                    // Merge in UMD exports with first-in-wins semantics (see #9771)
+                    var source = file.symbol.globalExports;
+                    source.forEach(function (sourceSymbol, id) {
+                        if (!globals.has(id)) {
+                            globals.set(id, sourceSymbol);
+                        }
+                    });
+                }
+            }
+            // We do global augmentations separately from module augmentations (and before creating global types) because they
+            //  1. Affect global types. We won't have the correct global types until global augmentations are merged. Also,
+            //  2. Module augmentation instantiation requires creating the type of a module, which, in turn, can require
+            //       checking for an export or property on the module (if export=) which, in turn, can fall back to the
+            //       apparent type of the module - either globalObjectType or globalFunctionType - which wouldn't exist if we
+            //       did module augmentations prior to finalizing the global types.
+            if (augmentations) {
+                // merge _global_ module augmentations.
+                // this needs to be done after global symbol table is initialized to make sure that all ambient modules are indexed
+                for (var _f = 0, augmentations_1 = augmentations; _f < augmentations_1.length; _f++) {
+                    var list = augmentations_1[_f];
+                    for (var _g = 0, list_1 = list; _g < list_1.length; _g++) {
+                        var augmentation = list_1[_g];
+                        if (!ts.isGlobalScopeAugmentation(augmentation.parent))
+                            continue;
+                        mergeModuleAugmentation(augmentation);
+                    }
+                }
+            }
+            // Setup global builtins
+            addToSymbolTable(globals, builtinGlobals, ts.Diagnostics.Declaration_name_conflicts_with_built_in_global_identifier_0);
+            getSymbolLinks(undefinedSymbol).type = undefinedWideningType;
+            getSymbolLinks(argumentsSymbol).type = getGlobalType("IArguments", /*arity*/ 0, /*reportErrors*/ true);
+            getSymbolLinks(unknownSymbol).type = errorType;
+            getSymbolLinks(globalThisSymbol).type = createObjectType(16 /* Anonymous */, globalThisSymbol);
+            // Initialize special types
+            globalArrayType = getGlobalType("Array", /*arity*/ 1, /*reportErrors*/ true);
+            globalObjectType = getGlobalType("Object", /*arity*/ 0, /*reportErrors*/ true);
+            globalFunctionType = getGlobalType("Function", /*arity*/ 0, /*reportErrors*/ true);
+            globalCallableFunctionType = strictBindCallApply && getGlobalType("CallableFunction", /*arity*/ 0, /*reportErrors*/ true) || globalFunctionType;
+            globalNewableFunctionType = strictBindCallApply && getGlobalType("NewableFunction", /*arity*/ 0, /*reportErrors*/ true) || globalFunctionType;
+            globalStringType = getGlobalType("String", /*arity*/ 0, /*reportErrors*/ true);
+            globalNumberType = getGlobalType("Number", /*arity*/ 0, /*reportErrors*/ true);
+            globalBooleanType = getGlobalType("Boolean", /*arity*/ 0, /*reportErrors*/ true);
+            globalRegExpType = getGlobalType("RegExp", /*arity*/ 0, /*reportErrors*/ true);
+            anyArrayType = createArrayType(anyType);
+            autoArrayType = createArrayType(autoType);
+            if (autoArrayType === emptyObjectType) {
+                // autoArrayType is used as a marker, so even if global Array type is not defined, it needs to be a unique type
+                autoArrayType = createAnonymousType(undefined, emptySymbols, ts.emptyArray, ts.emptyArray, undefined, undefined);
+            }
+            globalReadonlyArrayType = getGlobalTypeOrUndefined("ReadonlyArray", /*arity*/ 1) || globalArrayType;
+            anyReadonlyArrayType = globalReadonlyArrayType ? createTypeFromGenericGlobalType(globalReadonlyArrayType, [anyType]) : anyArrayType;
+            globalThisType = getGlobalTypeOrUndefined("ThisType", /*arity*/ 1);
+            if (augmentations) {
+                // merge _nonglobal_ module augmentations.
+                // this needs to be done after global symbol table is initialized to make sure that all ambient modules are indexed
+                for (var _h = 0, augmentations_2 = augmentations; _h < augmentations_2.length; _h++) {
+                    var list = augmentations_2[_h];
+                    for (var _j = 0, list_2 = list; _j < list_2.length; _j++) {
+                        var augmentation = list_2[_j];
+                        if (ts.isGlobalScopeAugmentation(augmentation.parent))
+                            continue;
+                        mergeModuleAugmentation(augmentation);
+                    }
+                }
+            }
+            amalgamatedDuplicates.forEach(function (_a) {
+                var firstFile = _a.firstFile, secondFile = _a.secondFile, conflictingSymbols = _a.conflictingSymbols;
+                // If not many things conflict, issue individual errors
+                if (conflictingSymbols.size < 8) {
+                    conflictingSymbols.forEach(function (_a, symbolName) {
+                        var isBlockScoped = _a.isBlockScoped, firstFileLocations = _a.firstFileLocations, secondFileLocations = _a.secondFileLocations;
+                        var message = isBlockScoped ? ts.Diagnostics.Cannot_redeclare_block_scoped_variable_0 : ts.Diagnostics.Duplicate_identifier_0;
+                        for (var _i = 0, firstFileLocations_1 = firstFileLocations; _i < firstFileLocations_1.length; _i++) {
+                            var node = firstFileLocations_1[_i];
+                            addDuplicateDeclarationError(node, message, symbolName, secondFileLocations);
+                        }
+                        for (var _b = 0, secondFileLocations_1 = secondFileLocations; _b < secondFileLocations_1.length; _b++) {
+                            var node = secondFileLocations_1[_b];
+                            addDuplicateDeclarationError(node, message, symbolName, firstFileLocations);
+                        }
+                    });
+                }
+                else {
+                    // Otherwise issue top-level error since the files appear very identical in terms of what they contain
+                    var list = ts.arrayFrom(conflictingSymbols.keys()).join(", ");
+                    diagnostics.add(ts.addRelatedInfo(ts.createDiagnosticForNode(firstFile, ts.Diagnostics.Definitions_of_the_following_identifiers_conflict_with_those_in_another_file_Colon_0, list), ts.createDiagnosticForNode(secondFile, ts.Diagnostics.Conflicts_are_in_this_file)));
+                    diagnostics.add(ts.addRelatedInfo(ts.createDiagnosticForNode(secondFile, ts.Diagnostics.Definitions_of_the_following_identifiers_conflict_with_those_in_another_file_Colon_0, list), ts.createDiagnosticForNode(firstFile, ts.Diagnostics.Conflicts_are_in_this_file)));
                 }
-                return buildInfo_1 || undefined;
-            };
-            node.javascriptPath = declarationTextOrJavascriptPath;
-            node.javascriptMapPath = javascriptMapPath;
-            node.declarationPath = ts.Debug.checkDefined(javascriptMapTextOrDeclarationPath);
-            node.declarationMapPath = declarationMapPath;
-            node.buildInfoPath = declarationMapTextOrBuildInfoPath;
-            Object.defineProperties(node, {
-                javascriptText: { get: function () { return definedTextGetter_1(declarationTextOrJavascriptPath); } },
-                javascriptMapText: { get: function () { return textGetter_1(javascriptMapPath); } },
-                declarationText: { get: function () { return definedTextGetter_1(ts.Debug.checkDefined(javascriptMapTextOrDeclarationPath)); } },
-                declarationMapText: { get: function () { return textGetter_1(declarationMapPath); } },
-                buildInfo: { get: function () { return getAndCacheBuildInfo_1(function () { return textGetter_1(declarationMapTextOrBuildInfoPath); }); } }
             });
+            amalgamatedDuplicates = undefined;
         }
-        else {
-            node.javascriptText = javascriptTextOrReadFileText;
-            node.javascriptMapPath = javascriptMapPath;
-            node.javascriptMapText = javascriptMapTextOrDeclarationPath;
-            node.declarationText = declarationTextOrJavascriptPath;
-            node.declarationMapPath = declarationMapPath;
-            node.declarationMapText = declarationMapTextOrBuildInfoPath;
-            node.javascriptPath = javascriptPath;
-            node.declarationPath = declarationPath;
-            node.buildInfoPath = buildInfoPath;
-            node.buildInfo = buildInfo;
-            node.oldFileOfCurrentEmit = oldFileOfCurrentEmit;
+        function checkExternalEmitHelpers(location, helpers) {
+            if ((requestedExternalEmitHelpers & helpers) !== helpers && compilerOptions.importHelpers) {
+                var sourceFile = ts.getSourceFileOfNode(location);
+                if (ts.isEffectiveExternalModule(sourceFile, compilerOptions) && !(location.flags & 8388608 /* Ambient */)) {
+                    var helpersModule = resolveHelpersModule(sourceFile, location);
+                    if (helpersModule !== unknownSymbol) {
+                        var uncheckedHelpers = helpers & ~requestedExternalEmitHelpers;
+                        for (var helper = 1 /* FirstEmitHelper */; helper <= 4194304 /* LastEmitHelper */; helper <<= 1) {
+                            if (uncheckedHelpers & helper) {
+                                var name = getHelperName(helper);
+                                var symbol = getSymbol(helpersModule.exports, ts.escapeLeadingUnderscores(name), 111551 /* Value */);
+                                if (!symbol) {
+                                    error(location, ts.Diagnostics.This_syntax_requires_an_imported_helper_named_1_which_does_not_exist_in_0_Consider_upgrading_your_version_of_0, ts.externalHelpersModuleNameText, name);
+                                }
+                            }
+                        }
+                    }
+                    requestedExternalEmitHelpers |= helpers;
+                }
+            }
         }
-        return node;
-    }
-    ts.createInputFiles = createInputFiles;
-    function updateBundle(node, sourceFiles, prepends) {
-        if (prepends === void 0) { prepends = ts.emptyArray; }
-        if (node.sourceFiles !== sourceFiles || node.prepends !== prepends) {
-            return createBundle(sourceFiles, prepends);
+        function getHelperName(helper) {
+            switch (helper) {
+                case 1 /* Extends */: return "__extends";
+                case 2 /* Assign */: return "__assign";
+                case 4 /* Rest */: return "__rest";
+                case 8 /* Decorate */: return "__decorate";
+                case 16 /* Metadata */: return "__metadata";
+                case 32 /* Param */: return "__param";
+                case 64 /* Awaiter */: return "__awaiter";
+                case 128 /* Generator */: return "__generator";
+                case 256 /* Values */: return "__values";
+                case 512 /* Read */: return "__read";
+                case 1024 /* Spread */: return "__spread";
+                case 2048 /* SpreadArrays */: return "__spreadArrays";
+                case 4096 /* Await */: return "__await";
+                case 8192 /* AsyncGenerator */: return "__asyncGenerator";
+                case 16384 /* AsyncDelegator */: return "__asyncDelegator";
+                case 32768 /* AsyncValues */: return "__asyncValues";
+                case 65536 /* ExportStar */: return "__exportStar";
+                case 131072 /* ImportStar */: return "__importStar";
+                case 262144 /* ImportDefault */: return "__importDefault";
+                case 524288 /* MakeTemplateObject */: return "__makeTemplateObject";
+                case 1048576 /* ClassPrivateFieldGet */: return "__classPrivateFieldGet";
+                case 2097152 /* ClassPrivateFieldSet */: return "__classPrivateFieldSet";
+                case 4194304 /* CreateBinding */: return "__createBinding";
+                default: return ts.Debug.fail("Unrecognized helper");
+            }
         }
-        return node;
-    }
-    ts.updateBundle = updateBundle;
-    function createImmediatelyInvokedFunctionExpression(statements, param, paramValue) {
-        return createCall(createFunctionExpression(
-        /*modifiers*/ undefined, 
-        /*asteriskToken*/ undefined, 
-        /*name*/ undefined, 
-        /*typeParameters*/ undefined, 
-        /*parameters*/ param ? [param] : [], 
-        /*type*/ undefined, createBlock(statements, /*multiLine*/ true)), 
-        /*typeArguments*/ undefined, 
-        /*argumentsArray*/ paramValue ? [paramValue] : []);
-    }
-    ts.createImmediatelyInvokedFunctionExpression = createImmediatelyInvokedFunctionExpression;
-    function createImmediatelyInvokedArrowFunction(statements, param, paramValue) {
-        return createCall(createArrowFunction(
-        /*modifiers*/ undefined, 
-        /*typeParameters*/ undefined, 
-        /*parameters*/ param ? [param] : [], 
-        /*type*/ undefined, 
-        /*equalsGreaterThanToken*/ undefined, createBlock(statements, /*multiLine*/ true)), 
-        /*typeArguments*/ undefined, 
-        /*argumentsArray*/ paramValue ? [paramValue] : []);
-    }
-    ts.createImmediatelyInvokedArrowFunction = createImmediatelyInvokedArrowFunction;
-    function createComma(left, right) {
-        return createBinary(left, 27 /* CommaToken */, right);
-    }
-    ts.createComma = createComma;
-    function createLessThan(left, right) {
-        return createBinary(left, 29 /* LessThanToken */, right);
-    }
-    ts.createLessThan = createLessThan;
-    function createAssignment(left, right) {
-        return createBinary(left, 62 /* EqualsToken */, right);
-    }
-    ts.createAssignment = createAssignment;
-    function createStrictEquality(left, right) {
-        return createBinary(left, 36 /* EqualsEqualsEqualsToken */, right);
-    }
-    ts.createStrictEquality = createStrictEquality;
-    function createStrictInequality(left, right) {
-        return createBinary(left, 37 /* ExclamationEqualsEqualsToken */, right);
-    }
-    ts.createStrictInequality = createStrictInequality;
-    function createAdd(left, right) {
-        return createBinary(left, 39 /* PlusToken */, right);
-    }
-    ts.createAdd = createAdd;
-    function createSubtract(left, right) {
-        return createBinary(left, 40 /* MinusToken */, right);
-    }
-    ts.createSubtract = createSubtract;
-    function createPostfixIncrement(operand) {
-        return createPostfix(operand, 45 /* PlusPlusToken */);
-    }
-    ts.createPostfixIncrement = createPostfixIncrement;
-    function createLogicalAnd(left, right) {
-        return createBinary(left, 55 /* AmpersandAmpersandToken */, right);
-    }
-    ts.createLogicalAnd = createLogicalAnd;
-    function createLogicalOr(left, right) {
-        return createBinary(left, 56 /* BarBarToken */, right);
-    }
-    ts.createLogicalOr = createLogicalOr;
-    function createNullishCoalesce(left, right) {
-        return createBinary(left, 60 /* QuestionQuestionToken */, right);
-    }
-    ts.createNullishCoalesce = createNullishCoalesce;
-    function createLogicalNot(operand) {
-        return createPrefix(53 /* ExclamationToken */, operand);
-    }
-    ts.createLogicalNot = createLogicalNot;
-    function createVoidZero() {
-        return createVoid(createLiteral(0));
-    }
-    ts.createVoidZero = createVoidZero;
-    function createExportDefault(expression) {
-        return createExportAssignment(/*decorators*/ undefined, /*modifiers*/ undefined, /*isExportEquals*/ false, expression);
-    }
-    ts.createExportDefault = createExportDefault;
-    function createExternalModuleExport(exportName) {
-        return createExportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, createNamedExports([createExportSpecifier(/*propertyName*/ undefined, exportName)]));
-    }
-    ts.createExternalModuleExport = createExternalModuleExport;
-    // Utilities
-    function asName(name) {
-        return ts.isString(name) ? createIdentifier(name) : name;
-    }
-    function asExpression(value) {
-        return typeof value === "string" ? createStringLiteral(value) :
-            typeof value === "number" ? createNumericLiteral("" + value) :
-                typeof value === "boolean" ? value ? createTrue() : createFalse() :
-                    value;
-    }
-    function asNodeArray(array) {
-        return array ? createNodeArray(array) : undefined;
-    }
-    function asToken(value) {
-        return typeof value === "number" ? createToken(value) : value;
-    }
-    function asEmbeddedStatement(statement) {
-        return statement && ts.isNotEmittedStatement(statement) ? setTextRange(setOriginalNode(createEmptyStatement(), statement), statement) : statement;
-    }
-    /**
-     * Clears any EmitNode entries from parse-tree nodes.
-     * @param sourceFile A source file.
-     */
-    function disposeEmitNodes(sourceFile) {
-        // During transformation we may need to annotate a parse tree node with transient
-        // transformation properties. As parse tree nodes live longer than transformation
-        // nodes, we need to make sure we reclaim any memory allocated for custom ranges
-        // from these nodes to ensure we do not hold onto entire subtrees just for position
-        // information. We also need to reset these nodes to a pre-transformation state
-        // for incremental parsing scenarios so that we do not impact later emit.
-        sourceFile = ts.getSourceFileOfNode(ts.getParseTreeNode(sourceFile));
-        var emitNode = sourceFile && sourceFile.emitNode;
-        var annotatedNodes = emitNode && emitNode.annotatedNodes;
-        if (annotatedNodes) {
-            for (var _i = 0, annotatedNodes_1 = annotatedNodes; _i < annotatedNodes_1.length; _i++) {
-                var node = annotatedNodes_1[_i];
-                node.emitNode = undefined;
+        function resolveHelpersModule(node, errorNode) {
+            if (!externalHelpersModule) {
+                externalHelpersModule = resolveExternalModule(node, ts.externalHelpersModuleNameText, ts.Diagnostics.This_syntax_requires_an_imported_helper_but_module_0_cannot_be_found, errorNode) || unknownSymbol;
             }
+            return externalHelpersModule;
         }
-    }
-    ts.disposeEmitNodes = disposeEmitNodes;
-    /**
-     * Associates a node with the current transformation, initializing
-     * various transient transformation properties.
-     */
-    /* @internal */
-    function getOrCreateEmitNode(node) {
-        if (!node.emitNode) {
-            if (ts.isParseTreeNode(node)) {
-                // To avoid holding onto transformation artifacts, we keep track of any
-                // parse tree node we are annotating. This allows us to clean them up after
-                // all transformations have completed.
-                if (node.kind === 290 /* SourceFile */) {
-                    return node.emitNode = { annotatedNodes: [node] };
+        // GRAMMAR CHECKING
+        function checkGrammarDecoratorsAndModifiers(node) {
+            return checkGrammarDecorators(node) || checkGrammarModifiers(node);
+        }
+        function checkGrammarDecorators(node) {
+            if (!node.decorators) {
+                return false;
+            }
+            if (!ts.nodeCanBeDecorated(node, node.parent, node.parent.parent)) {
+                if (node.kind === 165 /* MethodDeclaration */ && !ts.nodeIsPresent(node.body)) {
+                    return grammarErrorOnFirstToken(node, ts.Diagnostics.A_decorator_can_only_decorate_a_method_implementation_not_an_overload);
+                }
+                else {
+                    return grammarErrorOnFirstToken(node, ts.Diagnostics.Decorators_are_not_valid_here);
                 }
-                var sourceFile = ts.getSourceFileOfNode(ts.getParseTreeNode(ts.getSourceFileOfNode(node)));
-                getOrCreateEmitNode(sourceFile).annotatedNodes.push(node);
             }
-            node.emitNode = {};
-        }
-        return node.emitNode;
-    }
-    ts.getOrCreateEmitNode = getOrCreateEmitNode;
-    /**
-     * Sets `EmitFlags.NoComments` on a node and removes any leading and trailing synthetic comments.
-     * @internal
-     */
-    function removeAllComments(node) {
-        var emitNode = getOrCreateEmitNode(node);
-        emitNode.flags |= 1536 /* NoComments */;
-        emitNode.leadingComments = undefined;
-        emitNode.trailingComments = undefined;
-        return node;
-    }
-    ts.removeAllComments = removeAllComments;
-    function setTextRange(range, location) {
-        if (location) {
-            range.pos = location.pos;
-            range.end = location.end;
+            else if (node.kind === 167 /* GetAccessor */ || node.kind === 168 /* SetAccessor */) {
+                var accessors = ts.getAllAccessorDeclarations(node.parent.members, node);
+                if (accessors.firstAccessor.decorators && node === accessors.secondAccessor) {
+                    return grammarErrorOnFirstToken(node, ts.Diagnostics.Decorators_cannot_be_applied_to_multiple_get_Slashset_accessors_of_the_same_name);
+                }
+            }
+            return false;
         }
-        return range;
-    }
-    ts.setTextRange = setTextRange;
-    /**
-     * Sets flags that control emit behavior of a node.
-     */
-    function setEmitFlags(node, emitFlags) {
-        getOrCreateEmitNode(node).flags = emitFlags;
-        return node;
-    }
-    ts.setEmitFlags = setEmitFlags;
-    /**
-     * Sets flags that control emit behavior of a node.
-     */
-    /* @internal */
-    function addEmitFlags(node, emitFlags) {
-        var emitNode = getOrCreateEmitNode(node);
-        emitNode.flags = emitNode.flags | emitFlags;
-        return node;
-    }
-    ts.addEmitFlags = addEmitFlags;
-    /**
-     * Gets a custom text range to use when emitting source maps.
-     */
-    function getSourceMapRange(node) {
-        var emitNode = node.emitNode;
-        return (emitNode && emitNode.sourceMapRange) || node;
-    }
-    ts.getSourceMapRange = getSourceMapRange;
-    /**
-     * Sets a custom text range to use when emitting source maps.
-     */
-    function setSourceMapRange(node, range) {
-        getOrCreateEmitNode(node).sourceMapRange = range;
-        return node;
-    }
-    ts.setSourceMapRange = setSourceMapRange;
-    var SourceMapSource;
-    /**
-     * Create an external source map source file reference
-     */
-    function createSourceMapSource(fileName, text, skipTrivia) {
-        return new (SourceMapSource || (SourceMapSource = ts.objectAllocator.getSourceMapSourceConstructor()))(fileName, text, skipTrivia);
-    }
-    ts.createSourceMapSource = createSourceMapSource;
-    /**
-     * Gets the TextRange to use for source maps for a token of a node.
-     */
-    function getTokenSourceMapRange(node, token) {
-        var emitNode = node.emitNode;
-        var tokenSourceMapRanges = emitNode && emitNode.tokenSourceMapRanges;
-        return tokenSourceMapRanges && tokenSourceMapRanges[token];
-    }
-    ts.getTokenSourceMapRange = getTokenSourceMapRange;
-    /**
-     * Sets the TextRange to use for source maps for a token of a node.
-     */
-    function setTokenSourceMapRange(node, token, range) {
-        var emitNode = getOrCreateEmitNode(node);
-        var tokenSourceMapRanges = emitNode.tokenSourceMapRanges || (emitNode.tokenSourceMapRanges = []);
-        tokenSourceMapRanges[token] = range;
-        return node;
-    }
-    ts.setTokenSourceMapRange = setTokenSourceMapRange;
-    /**
-     * Gets a custom text range to use when emitting comments.
-     */
-    /*@internal*/
-    function getStartsOnNewLine(node) {
-        var emitNode = node.emitNode;
-        return emitNode && emitNode.startsOnNewLine;
-    }
-    ts.getStartsOnNewLine = getStartsOnNewLine;
-    /**
-     * Sets a custom text range to use when emitting comments.
-     */
-    /*@internal*/
-    function setStartsOnNewLine(node, newLine) {
-        getOrCreateEmitNode(node).startsOnNewLine = newLine;
-        return node;
-    }
-    ts.setStartsOnNewLine = setStartsOnNewLine;
-    /**
-     * Gets a custom text range to use when emitting comments.
-     */
-    function getCommentRange(node) {
-        var emitNode = node.emitNode;
-        return (emitNode && emitNode.commentRange) || node;
-    }
-    ts.getCommentRange = getCommentRange;
-    /**
-     * Sets a custom text range to use when emitting comments.
-     */
-    function setCommentRange(node, range) {
-        getOrCreateEmitNode(node).commentRange = range;
-        return node;
-    }
-    ts.setCommentRange = setCommentRange;
-    function getSyntheticLeadingComments(node) {
-        var emitNode = node.emitNode;
-        return emitNode && emitNode.leadingComments;
-    }
-    ts.getSyntheticLeadingComments = getSyntheticLeadingComments;
-    function setSyntheticLeadingComments(node, comments) {
-        getOrCreateEmitNode(node).leadingComments = comments;
-        return node;
-    }
-    ts.setSyntheticLeadingComments = setSyntheticLeadingComments;
-    function addSyntheticLeadingComment(node, kind, text, hasTrailingNewLine) {
-        return setSyntheticLeadingComments(node, ts.append(getSyntheticLeadingComments(node), { kind: kind, pos: -1, end: -1, hasTrailingNewLine: hasTrailingNewLine, text: text }));
-    }
-    ts.addSyntheticLeadingComment = addSyntheticLeadingComment;
-    function getSyntheticTrailingComments(node) {
-        var emitNode = node.emitNode;
-        return emitNode && emitNode.trailingComments;
-    }
-    ts.getSyntheticTrailingComments = getSyntheticTrailingComments;
-    function setSyntheticTrailingComments(node, comments) {
-        getOrCreateEmitNode(node).trailingComments = comments;
-        return node;
-    }
-    ts.setSyntheticTrailingComments = setSyntheticTrailingComments;
-    function addSyntheticTrailingComment(node, kind, text, hasTrailingNewLine) {
-        return setSyntheticTrailingComments(node, ts.append(getSyntheticTrailingComments(node), { kind: kind, pos: -1, end: -1, hasTrailingNewLine: hasTrailingNewLine, text: text }));
-    }
-    ts.addSyntheticTrailingComment = addSyntheticTrailingComment;
-    function moveSyntheticComments(node, original) {
-        setSyntheticLeadingComments(node, getSyntheticLeadingComments(original));
-        setSyntheticTrailingComments(node, getSyntheticTrailingComments(original));
-        var emit = getOrCreateEmitNode(original);
-        emit.leadingComments = undefined;
-        emit.trailingComments = undefined;
-        return node;
-    }
-    ts.moveSyntheticComments = moveSyntheticComments;
-    /** @internal */
-    function ignoreSourceNewlines(node) {
-        getOrCreateEmitNode(node).flags |= 134217728 /* IgnoreSourceNewlines */;
-        return node;
-    }
-    ts.ignoreSourceNewlines = ignoreSourceNewlines;
-    /**
-     * Gets the constant value to emit for an expression.
-     */
-    function getConstantValue(node) {
-        var emitNode = node.emitNode;
-        return emitNode && emitNode.constantValue;
-    }
-    ts.getConstantValue = getConstantValue;
-    /**
-     * Sets the constant value to emit for an expression.
-     */
-    function setConstantValue(node, value) {
-        var emitNode = getOrCreateEmitNode(node);
-        emitNode.constantValue = value;
-        return node;
-    }
-    ts.setConstantValue = setConstantValue;
-    /**
-     * Adds an EmitHelper to a node.
-     */
-    function addEmitHelper(node, helper) {
-        var emitNode = getOrCreateEmitNode(node);
-        emitNode.helpers = ts.append(emitNode.helpers, helper);
-        return node;
-    }
-    ts.addEmitHelper = addEmitHelper;
-    /**
-     * Add EmitHelpers to a node.
-     */
-    function addEmitHelpers(node, helpers) {
-        if (ts.some(helpers)) {
-            var emitNode = getOrCreateEmitNode(node);
-            for (var _i = 0, helpers_1 = helpers; _i < helpers_1.length; _i++) {
-                var helper = helpers_1[_i];
-                emitNode.helpers = ts.appendIfUnique(emitNode.helpers, helper);
+        function checkGrammarModifiers(node) {
+            var quickResult = reportObviousModifierErrors(node);
+            if (quickResult !== undefined) {
+                return quickResult;
+            }
+            var lastStatic, lastDeclare, lastAsync, lastReadonly;
+            var flags = 0 /* None */;
+            for (var _i = 0, _a = node.modifiers; _i < _a.length; _i++) {
+                var modifier = _a[_i];
+                if (modifier.kind !== 142 /* ReadonlyKeyword */) {
+                    if (node.kind === 162 /* PropertySignature */ || node.kind === 164 /* MethodSignature */) {
+                        return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_type_member, ts.tokenToString(modifier.kind));
+                    }
+                    if (node.kind === 171 /* IndexSignature */) {
+                        return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_an_index_signature, ts.tokenToString(modifier.kind));
+                    }
+                }
+                switch (modifier.kind) {
+                    case 84 /* ConstKeyword */:
+                        if (node.kind !== 255 /* EnumDeclaration */) {
+                            return grammarErrorOnNode(node, ts.Diagnostics.A_class_member_cannot_have_the_0_keyword, ts.tokenToString(84 /* ConstKeyword */));
+                        }
+                        break;
+                    case 122 /* PublicKeyword */:
+                    case 121 /* ProtectedKeyword */:
+                    case 120 /* PrivateKeyword */:
+                        var text = visibilityToString(ts.modifierToFlag(modifier.kind));
+                        if (flags & 28 /* AccessibilityModifier */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics.Accessibility_modifier_already_seen);
+                        }
+                        else if (flags & 32 /* Static */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "static");
+                        }
+                        else if (flags & 64 /* Readonly */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "readonly");
+                        }
+                        else if (flags & 256 /* Async */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "async");
+                        }
+                        else if (node.parent.kind === 257 /* ModuleBlock */ || node.parent.kind === 297 /* SourceFile */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_module_or_namespace_element, text);
+                        }
+                        else if (flags & 128 /* Abstract */) {
+                            if (modifier.kind === 120 /* PrivateKeyword */) {
+                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, text, "abstract");
+                            }
+                            else {
+                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, text, "abstract");
+                            }
+                        }
+                        else if (ts.isPrivateIdentifierPropertyDeclaration(node)) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics.An_accessibility_modifier_cannot_be_used_with_a_private_identifier);
+                        }
+                        flags |= ts.modifierToFlag(modifier.kind);
+                        break;
+                    case 123 /* StaticKeyword */:
+                        if (flags & 32 /* Static */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "static");
+                        }
+                        else if (flags & 64 /* Readonly */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "static", "readonly");
+                        }
+                        else if (flags & 256 /* Async */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "static", "async");
+                        }
+                        else if (node.parent.kind === 257 /* ModuleBlock */ || node.parent.kind === 297 /* SourceFile */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_module_or_namespace_element, "static");
+                        }
+                        else if (node.kind === 160 /* Parameter */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "static");
+                        }
+                        else if (flags & 128 /* Abstract */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "static", "abstract");
+                        }
+                        else if (ts.isPrivateIdentifierPropertyDeclaration(node)) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_a_private_identifier, "static");
+                        }
+                        flags |= 32 /* Static */;
+                        lastStatic = modifier;
+                        break;
+                    case 142 /* ReadonlyKeyword */:
+                        if (flags & 64 /* Readonly */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "readonly");
+                        }
+                        else if (node.kind !== 163 /* PropertyDeclaration */ && node.kind !== 162 /* PropertySignature */ && node.kind !== 171 /* IndexSignature */ && node.kind !== 160 /* Parameter */) {
+                            // If node.kind === SyntaxKind.Parameter, checkParameter report an error if it's not a parameter property.
+                            return grammarErrorOnNode(modifier, ts.Diagnostics.readonly_modifier_can_only_appear_on_a_property_declaration_or_index_signature);
+                        }
+                        flags |= 64 /* Readonly */;
+                        lastReadonly = modifier;
+                        break;
+                    case 92 /* ExportKeyword */:
+                        if (flags & 1 /* Export */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "export");
+                        }
+                        else if (flags & 2 /* Ambient */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "export", "declare");
+                        }
+                        else if (flags & 128 /* Abstract */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "export", "abstract");
+                        }
+                        else if (flags & 256 /* Async */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_must_precede_1_modifier, "export", "async");
+                        }
+                        else if (ts.isClassLike(node.parent)) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_class_elements_of_this_kind, "export");
+                        }
+                        else if (node.kind === 160 /* Parameter */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "export");
+                        }
+                        flags |= 1 /* Export */;
+                        break;
+                    case 87 /* DefaultKeyword */:
+                        var container = node.parent.kind === 297 /* SourceFile */ ? node.parent : node.parent.parent;
+                        if (container.kind === 256 /* ModuleDeclaration */ && !ts.isAmbientModule(container)) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics.A_default_export_can_only_be_used_in_an_ECMAScript_style_module);
+                        }
+                        flags |= 512 /* Default */;
+                        break;
+                    case 133 /* DeclareKeyword */:
+                        if (flags & 2 /* Ambient */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "declare");
+                        }
+                        else if (flags & 256 /* Async */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_in_an_ambient_context, "async");
+                        }
+                        else if (ts.isClassLike(node.parent) && !ts.isPropertyDeclaration(node)) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_class_elements_of_this_kind, "declare");
+                        }
+                        else if (node.kind === 160 /* Parameter */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "declare");
+                        }
+                        else if ((node.parent.flags & 8388608 /* Ambient */) && node.parent.kind === 257 /* ModuleBlock */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics.A_declare_modifier_cannot_be_used_in_an_already_ambient_context);
+                        }
+                        else if (ts.isPrivateIdentifierPropertyDeclaration(node)) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_a_private_identifier, "declare");
+                        }
+                        flags |= 2 /* Ambient */;
+                        lastDeclare = modifier;
+                        break;
+                    case 125 /* AbstractKeyword */:
+                        if (flags & 128 /* Abstract */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "abstract");
+                        }
+                        if (node.kind !== 252 /* ClassDeclaration */) {
+                            if (node.kind !== 165 /* MethodDeclaration */ &&
+                                node.kind !== 163 /* PropertyDeclaration */ &&
+                                node.kind !== 167 /* GetAccessor */ &&
+                                node.kind !== 168 /* SetAccessor */) {
+                                return grammarErrorOnNode(modifier, ts.Diagnostics.abstract_modifier_can_only_appear_on_a_class_method_or_property_declaration);
+                            }
+                            if (!(node.parent.kind === 252 /* ClassDeclaration */ && ts.hasSyntacticModifier(node.parent, 128 /* Abstract */))) {
+                                return grammarErrorOnNode(modifier, ts.Diagnostics.Abstract_methods_can_only_appear_within_an_abstract_class);
+                            }
+                            if (flags & 32 /* Static */) {
+                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "static", "abstract");
+                            }
+                            if (flags & 8 /* Private */) {
+                                return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "private", "abstract");
+                            }
+                            if (flags & 256 /* Async */ && lastAsync) {
+                                return grammarErrorOnNode(lastAsync, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "async", "abstract");
+                            }
+                        }
+                        if (ts.isNamedDeclaration(node) && node.name.kind === 79 /* PrivateIdentifier */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_a_private_identifier, "abstract");
+                        }
+                        flags |= 128 /* Abstract */;
+                        break;
+                    case 129 /* AsyncKeyword */:
+                        if (flags & 256 /* Async */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_already_seen, "async");
+                        }
+                        else if (flags & 2 /* Ambient */ || node.parent.flags & 8388608 /* Ambient */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_in_an_ambient_context, "async");
+                        }
+                        else if (node.kind === 160 /* Parameter */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_appear_on_a_parameter, "async");
+                        }
+                        if (flags & 128 /* Abstract */) {
+                            return grammarErrorOnNode(modifier, ts.Diagnostics._0_modifier_cannot_be_used_with_1_modifier, "async", "abstract");
+                        }
+                        flags |= 256 /* Async */;
+                        lastAsync = modifier;
+                        break;
+                }
             }
-        }
-        return node;
-    }
-    ts.addEmitHelpers = addEmitHelpers;
-    /**
-     * Removes an EmitHelper from a node.
-     */
-    function removeEmitHelper(node, helper) {
-        var emitNode = node.emitNode;
-        if (emitNode) {
-            var helpers = emitNode.helpers;
-            if (helpers) {
-                return ts.orderedRemoveItem(helpers, helper);
+            if (node.kind === 166 /* Constructor */) {
+                if (flags & 32 /* Static */) {
+                    return grammarErrorOnNode(lastStatic, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "static");
+                }
+                if (flags & 128 /* Abstract */) {
+                    return grammarErrorOnNode(lastStatic, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "abstract"); // TODO: GH#18217
+                }
+                else if (flags & 256 /* Async */) {
+                    return grammarErrorOnNode(lastAsync, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "async");
+                }
+                else if (flags & 64 /* Readonly */) {
+                    return grammarErrorOnNode(lastReadonly, ts.Diagnostics._0_modifier_cannot_appear_on_a_constructor_declaration, "readonly");
+                }
+                return false;
             }
-        }
-        return false;
-    }
-    ts.removeEmitHelper = removeEmitHelper;
-    /**
-     * Gets the EmitHelpers of a node.
-     */
-    function getEmitHelpers(node) {
-        var emitNode = node.emitNode;
-        return emitNode && emitNode.helpers;
-    }
-    ts.getEmitHelpers = getEmitHelpers;
-    /**
-     * Moves matching emit helpers from a source node to a target node.
-     */
-    function moveEmitHelpers(source, target, predicate) {
-        var sourceEmitNode = source.emitNode;
-        var sourceEmitHelpers = sourceEmitNode && sourceEmitNode.helpers;
-        if (!ts.some(sourceEmitHelpers))
-            return;
-        var targetEmitNode = getOrCreateEmitNode(target);
-        var helpersRemoved = 0;
-        for (var i = 0; i < sourceEmitHelpers.length; i++) {
-            var helper = sourceEmitHelpers[i];
-            if (predicate(helper)) {
-                helpersRemoved++;
-                targetEmitNode.helpers = ts.appendIfUnique(targetEmitNode.helpers, helper);
+            else if ((node.kind === 261 /* ImportDeclaration */ || node.kind === 260 /* ImportEqualsDeclaration */) && flags & 2 /* Ambient */) {
+                return grammarErrorOnNode(lastDeclare, ts.Diagnostics.A_0_modifier_cannot_be_used_with_an_import_declaration, "declare");
             }
-            else if (helpersRemoved > 0) {
-                sourceEmitHelpers[i - helpersRemoved] = helper;
+            else if (node.kind === 160 /* Parameter */ && (flags & 92 /* ParameterPropertyModifier */) && ts.isBindingPattern(node.name)) {
+                return grammarErrorOnNode(node, ts.Diagnostics.A_parameter_property_may_not_be_declared_using_a_binding_pattern);
+            }
+            else if (node.kind === 160 /* Parameter */ && (flags & 92 /* ParameterPropertyModifier */) && node.dotDotDotToken) {
+                return grammarErrorOnNode(node, ts.Diagnostics.A_parameter_property_cannot_be_declared_using_a_rest_parameter);
+            }
+            if (flags & 256 /* Async */) {
+                return checkGrammarAsyncModifier(node, lastAsync);
             }
+            return false;
         }
-        if (helpersRemoved > 0) {
-            sourceEmitHelpers.length -= helpersRemoved;
+        /**
+         * true | false: Early return this value from checkGrammarModifiers.
+         * undefined: Need to do full checking on the modifiers.
+         */
+        function reportObviousModifierErrors(node) {
+            return !node.modifiers
+                ? false
+                : shouldReportBadModifier(node)
+                    ? grammarErrorOnFirstToken(node, ts.Diagnostics.Modifiers_cannot_appear_here)
+                    : undefined;
         }
-    }
-    ts.moveEmitHelpers = moveEmitHelpers;
-    /* @internal */
-    function compareEmitHelpers(x, y) {
-        if (x === y)
-            return 0 /* EqualTo */;
-        if (x.priority === y.priority)
-            return 0 /* EqualTo */;
-        if (x.priority === undefined)
-            return 1 /* GreaterThan */;
-        if (y.priority === undefined)
-            return -1 /* LessThan */;
-        return ts.compareValues(x.priority, y.priority);
-    }
-    ts.compareEmitHelpers = compareEmitHelpers;
-    function setOriginalNode(node, original) {
-        node.original = original;
-        if (original) {
-            var emitNode = original.emitNode;
-            if (emitNode)
-                node.emitNode = mergeEmitNode(emitNode, node.emitNode);
+        function shouldReportBadModifier(node) {
+            switch (node.kind) {
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 166 /* Constructor */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 171 /* IndexSignature */:
+                case 256 /* ModuleDeclaration */:
+                case 261 /* ImportDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
+                case 267 /* ExportDeclaration */:
+                case 266 /* ExportAssignment */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 160 /* Parameter */:
+                    return false;
+                default:
+                    if (node.parent.kind === 257 /* ModuleBlock */ || node.parent.kind === 297 /* SourceFile */) {
+                        return false;
+                    }
+                    switch (node.kind) {
+                        case 251 /* FunctionDeclaration */:
+                            return nodeHasAnyModifiersExcept(node, 129 /* AsyncKeyword */);
+                        case 252 /* ClassDeclaration */:
+                            return nodeHasAnyModifiersExcept(node, 125 /* AbstractKeyword */);
+                        case 253 /* InterfaceDeclaration */:
+                        case 232 /* VariableStatement */:
+                        case 254 /* TypeAliasDeclaration */:
+                            return true;
+                        case 255 /* EnumDeclaration */:
+                            return nodeHasAnyModifiersExcept(node, 84 /* ConstKeyword */);
+                        default:
+                            ts.Debug.fail();
+                            return false;
+                    }
+            }
         }
-        return node;
-    }
-    ts.setOriginalNode = setOriginalNode;
-    function mergeEmitNode(sourceEmitNode, destEmitNode) {
-        var flags = sourceEmitNode.flags, leadingComments = sourceEmitNode.leadingComments, trailingComments = sourceEmitNode.trailingComments, commentRange = sourceEmitNode.commentRange, sourceMapRange = sourceEmitNode.sourceMapRange, tokenSourceMapRanges = sourceEmitNode.tokenSourceMapRanges, constantValue = sourceEmitNode.constantValue, helpers = sourceEmitNode.helpers, startsOnNewLine = sourceEmitNode.startsOnNewLine;
-        if (!destEmitNode)
-            destEmitNode = {};
-        // We are using `.slice()` here in case `destEmitNode.leadingComments` is pushed to later.
-        if (leadingComments)
-            destEmitNode.leadingComments = ts.addRange(leadingComments.slice(), destEmitNode.leadingComments);
-        if (trailingComments)
-            destEmitNode.trailingComments = ts.addRange(trailingComments.slice(), destEmitNode.trailingComments);
-        if (flags)
-            destEmitNode.flags = flags;
-        if (commentRange)
-            destEmitNode.commentRange = commentRange;
-        if (sourceMapRange)
-            destEmitNode.sourceMapRange = sourceMapRange;
-        if (tokenSourceMapRanges)
-            destEmitNode.tokenSourceMapRanges = mergeTokenSourceMapRanges(tokenSourceMapRanges, destEmitNode.tokenSourceMapRanges);
-        if (constantValue !== undefined)
-            destEmitNode.constantValue = constantValue;
-        if (helpers)
-            destEmitNode.helpers = ts.addRange(destEmitNode.helpers, helpers);
-        if (startsOnNewLine !== undefined)
-            destEmitNode.startsOnNewLine = startsOnNewLine;
-        return destEmitNode;
-    }
-    function mergeTokenSourceMapRanges(sourceRanges, destRanges) {
-        if (!destRanges)
-            destRanges = [];
-        for (var key in sourceRanges) {
-            destRanges[key] = sourceRanges[key];
+        function nodeHasAnyModifiersExcept(node, allowedModifier) {
+            return node.modifiers.length > 1 || node.modifiers[0].kind !== allowedModifier;
         }
-        return destRanges;
-    }
-})(ts || (ts = {}));
-/* @internal */
-var ts;
-(function (ts) {
-    ts.nullTransformationContext = {
-        enableEmitNotification: ts.noop,
-        enableSubstitution: ts.noop,
-        endLexicalEnvironment: ts.returnUndefined,
-        getCompilerOptions: function () { return ({}); },
-        getEmitHost: ts.notImplemented,
-        getEmitResolver: ts.notImplemented,
-        setLexicalEnvironmentFlags: ts.noop,
-        getLexicalEnvironmentFlags: function () { return 0; },
-        hoistFunctionDeclaration: ts.noop,
-        hoistVariableDeclaration: ts.noop,
-        addInitializationStatement: ts.noop,
-        isEmitNotificationEnabled: ts.notImplemented,
-        isSubstitutionEnabled: ts.notImplemented,
-        onEmitNode: ts.noop,
-        onSubstituteNode: ts.notImplemented,
-        readEmitHelpers: ts.notImplemented,
-        requestEmitHelper: ts.noop,
-        resumeLexicalEnvironment: ts.noop,
-        startLexicalEnvironment: ts.noop,
-        suspendLexicalEnvironment: ts.noop,
-        addDiagnostic: ts.noop,
-    };
-    function createTypeCheck(value, tag) {
-        return tag === "undefined"
-            ? ts.createStrictEquality(value, ts.createVoidZero())
-            : ts.createStrictEquality(ts.createTypeOf(value), ts.createLiteral(tag));
-    }
-    ts.createTypeCheck = createTypeCheck;
-    function createMemberAccessForPropertyName(target, memberName, location) {
-        if (ts.isComputedPropertyName(memberName)) {
-            return ts.setTextRange(ts.createElementAccess(target, memberName.expression), location);
+        function checkGrammarAsyncModifier(node, asyncModifier) {
+            switch (node.kind) {
+                case 165 /* MethodDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                    return false;
+            }
+            return grammarErrorOnNode(asyncModifier, ts.Diagnostics._0_modifier_cannot_be_used_here, "async");
         }
-        else {
-            var expression = ts.setTextRange((ts.isIdentifier(memberName) || ts.isPrivateIdentifier(memberName))
-                ? ts.createPropertyAccess(target, memberName)
-                : ts.createElementAccess(target, memberName), memberName);
-            ts.getOrCreateEmitNode(expression).flags |= 64 /* NoNestedSourceMaps */;
-            return expression;
+        function checkGrammarForDisallowedTrailingComma(list, diag) {
+            if (diag === void 0) { diag = ts.Diagnostics.Trailing_comma_not_allowed; }
+            if (list && list.hasTrailingComma) {
+                return grammarErrorAtPos(list[0], list.end - ",".length, ",".length, diag);
+            }
+            return false;
         }
-    }
-    ts.createMemberAccessForPropertyName = createMemberAccessForPropertyName;
-    function createFunctionCall(func, thisArg, argumentsList, location) {
-        return ts.setTextRange(ts.createCall(ts.createPropertyAccess(func, "call"), 
-        /*typeArguments*/ undefined, __spreadArrays([
-            thisArg
-        ], argumentsList)), location);
-    }
-    ts.createFunctionCall = createFunctionCall;
-    function createFunctionApply(func, thisArg, argumentsExpression, location) {
-        return ts.setTextRange(ts.createCall(ts.createPropertyAccess(func, "apply"), 
-        /*typeArguments*/ undefined, [
-            thisArg,
-            argumentsExpression
-        ]), location);
-    }
-    ts.createFunctionApply = createFunctionApply;
-    function createArraySlice(array, start) {
-        var argumentsList = [];
-        if (start !== undefined) {
-            argumentsList.push(typeof start === "number" ? ts.createLiteral(start) : start);
-        }
-        return ts.createCall(ts.createPropertyAccess(array, "slice"), /*typeArguments*/ undefined, argumentsList);
-    }
-    ts.createArraySlice = createArraySlice;
-    function createArrayConcat(array, values) {
-        return ts.createCall(ts.createPropertyAccess(array, "concat"), 
-        /*typeArguments*/ undefined, values);
-    }
-    ts.createArrayConcat = createArrayConcat;
-    function createMathPow(left, right, location) {
-        return ts.setTextRange(ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Math"), "pow"), 
-        /*typeArguments*/ undefined, [left, right]), location);
-    }
-    ts.createMathPow = createMathPow;
-    function createReactNamespace(reactNamespace, parent) {
-        // To ensure the emit resolver can properly resolve the namespace, we need to
-        // treat this identifier as if it were a source tree node by clearing the `Synthesized`
-        // flag and setting a parent node.
-        var react = ts.createIdentifier(reactNamespace || "React");
-        react.flags &= ~8 /* Synthesized */;
-        // Set the parent that is in parse tree
-        // this makes sure that parent chain is intact for checker to traverse complete scope tree
-        react.parent = ts.getParseTreeNode(parent);
-        return react;
-    }
-    function createJsxFactoryExpressionFromEntityName(jsxFactory, parent) {
-        if (ts.isQualifiedName(jsxFactory)) {
-            var left = createJsxFactoryExpressionFromEntityName(jsxFactory.left, parent);
-            var right = ts.createIdentifier(ts.idText(jsxFactory.right));
-            right.escapedText = jsxFactory.right.escapedText;
-            return ts.createPropertyAccess(left, right);
+        function checkGrammarTypeParameterList(typeParameters, file) {
+            if (typeParameters && typeParameters.length === 0) {
+                var start = typeParameters.pos - "<".length;
+                var end = ts.skipTrivia(file.text, typeParameters.end) + ">".length;
+                return grammarErrorAtPos(file, start, end - start, ts.Diagnostics.Type_parameter_list_cannot_be_empty);
+            }
+            return false;
         }
-        else {
-            return createReactNamespace(ts.idText(jsxFactory), parent);
+        function checkGrammarParameterList(parameters) {
+            var seenOptionalParameter = false;
+            var parameterCount = parameters.length;
+            for (var i = 0; i < parameterCount; i++) {
+                var parameter = parameters[i];
+                if (parameter.dotDotDotToken) {
+                    if (i !== (parameterCount - 1)) {
+                        return grammarErrorOnNode(parameter.dotDotDotToken, ts.Diagnostics.A_rest_parameter_must_be_last_in_a_parameter_list);
+                    }
+                    if (!(parameter.flags & 8388608 /* Ambient */)) { // Allow `...foo,` in ambient declarations; see GH#23070
+                        checkGrammarForDisallowedTrailingComma(parameters, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
+                    }
+                    if (parameter.questionToken) {
+                        return grammarErrorOnNode(parameter.questionToken, ts.Diagnostics.A_rest_parameter_cannot_be_optional);
+                    }
+                    if (parameter.initializer) {
+                        return grammarErrorOnNode(parameter.name, ts.Diagnostics.A_rest_parameter_cannot_have_an_initializer);
+                    }
+                }
+                else if (isOptionalParameter(parameter)) {
+                    seenOptionalParameter = true;
+                    if (parameter.questionToken && parameter.initializer) {
+                        return grammarErrorOnNode(parameter.name, ts.Diagnostics.Parameter_cannot_have_question_mark_and_initializer);
+                    }
+                }
+                else if (seenOptionalParameter && !parameter.initializer) {
+                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.A_required_parameter_cannot_follow_an_optional_parameter);
+                }
+            }
         }
-    }
-    function createJsxFactoryExpression(jsxFactoryEntity, reactNamespace, parent) {
-        return jsxFactoryEntity ?
-            createJsxFactoryExpressionFromEntityName(jsxFactoryEntity, parent) :
-            ts.createPropertyAccess(createReactNamespace(reactNamespace, parent), "createElement");
-    }
-    function createExpressionForJsxElement(jsxFactoryEntity, reactNamespace, tagName, props, children, parentElement, location) {
-        var argumentsList = [tagName];
-        if (props) {
-            argumentsList.push(props);
+        function getNonSimpleParameters(parameters) {
+            return ts.filter(parameters, function (parameter) { return !!parameter.initializer || ts.isBindingPattern(parameter.name) || ts.isRestParameter(parameter); });
         }
-        if (children && children.length > 0) {
-            if (!props) {
-                argumentsList.push(ts.createNull());
-            }
-            if (children.length > 1) {
-                for (var _i = 0, children_2 = children; _i < children_2.length; _i++) {
-                    var child = children_2[_i];
-                    startOnNewLine(child);
-                    argumentsList.push(child);
+        function checkGrammarForUseStrictSimpleParameterList(node) {
+            if (languageVersion >= 3 /* ES2016 */) {
+                var useStrictDirective_1 = node.body && ts.isBlock(node.body) && ts.findUseStrictPrologue(node.body.statements);
+                if (useStrictDirective_1) {
+                    var nonSimpleParameters = getNonSimpleParameters(node.parameters);
+                    if (ts.length(nonSimpleParameters)) {
+                        ts.forEach(nonSimpleParameters, function (parameter) {
+                            ts.addRelatedInfo(error(parameter, ts.Diagnostics.This_parameter_is_not_allowed_with_use_strict_directive), ts.createDiagnosticForNode(useStrictDirective_1, ts.Diagnostics.use_strict_directive_used_here));
+                        });
+                        var diagnostics_2 = nonSimpleParameters.map(function (parameter, index) { return (index === 0 ? ts.createDiagnosticForNode(parameter, ts.Diagnostics.Non_simple_parameter_declared_here) : ts.createDiagnosticForNode(parameter, ts.Diagnostics.and_here)); });
+                        ts.addRelatedInfo.apply(void 0, __spreadArrays([error(useStrictDirective_1, ts.Diagnostics.use_strict_directive_cannot_be_used_with_non_simple_parameter_list)], diagnostics_2));
+                        return true;
+                    }
                 }
             }
-            else {
-                argumentsList.push(children[0]);
+            return false;
+        }
+        function checkGrammarFunctionLikeDeclaration(node) {
+            // Prevent cascading error by short-circuit
+            var file = ts.getSourceFileOfNode(node);
+            return checkGrammarDecoratorsAndModifiers(node) ||
+                checkGrammarTypeParameterList(node.typeParameters, file) ||
+                checkGrammarParameterList(node.parameters) ||
+                checkGrammarArrowFunction(node, file) ||
+                (ts.isFunctionLikeDeclaration(node) && checkGrammarForUseStrictSimpleParameterList(node));
+        }
+        function checkGrammarClassLikeDeclaration(node) {
+            var file = ts.getSourceFileOfNode(node);
+            return checkGrammarClassDeclarationHeritageClauses(node) ||
+                checkGrammarTypeParameterList(node.typeParameters, file);
+        }
+        function checkGrammarArrowFunction(node, file) {
+            if (!ts.isArrowFunction(node)) {
+                return false;
             }
+            var equalsGreaterThanToken = node.equalsGreaterThanToken;
+            var startLine = ts.getLineAndCharacterOfPosition(file, equalsGreaterThanToken.pos).line;
+            var endLine = ts.getLineAndCharacterOfPosition(file, equalsGreaterThanToken.end).line;
+            return startLine !== endLine && grammarErrorOnNode(equalsGreaterThanToken, ts.Diagnostics.Line_terminator_not_permitted_before_arrow);
         }
-        return ts.setTextRange(ts.createCall(createJsxFactoryExpression(jsxFactoryEntity, reactNamespace, parentElement), 
-        /*typeArguments*/ undefined, argumentsList), location);
-    }
-    ts.createExpressionForJsxElement = createExpressionForJsxElement;
-    function createExpressionForJsxFragment(jsxFactoryEntity, reactNamespace, children, parentElement, location) {
-        var tagName = ts.createPropertyAccess(createReactNamespace(reactNamespace, parentElement), "Fragment");
-        var argumentsList = [tagName];
-        argumentsList.push(ts.createNull());
-        if (children && children.length > 0) {
-            if (children.length > 1) {
-                for (var _i = 0, children_3 = children; _i < children_3.length; _i++) {
-                    var child = children_3[_i];
-                    startOnNewLine(child);
-                    argumentsList.push(child);
+        function checkGrammarIndexSignatureParameters(node) {
+            var parameter = node.parameters[0];
+            if (node.parameters.length !== 1) {
+                if (parameter) {
+                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_must_have_exactly_one_parameter);
+                }
+                else {
+                    return grammarErrorOnNode(node, ts.Diagnostics.An_index_signature_must_have_exactly_one_parameter);
                 }
             }
-            else {
-                argumentsList.push(children[0]);
+            checkGrammarForDisallowedTrailingComma(node.parameters, ts.Diagnostics.An_index_signature_cannot_have_a_trailing_comma);
+            if (parameter.dotDotDotToken) {
+                return grammarErrorOnNode(parameter.dotDotDotToken, ts.Diagnostics.An_index_signature_cannot_have_a_rest_parameter);
             }
+            if (ts.hasEffectiveModifiers(parameter)) {
+                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_cannot_have_an_accessibility_modifier);
+            }
+            if (parameter.questionToken) {
+                return grammarErrorOnNode(parameter.questionToken, ts.Diagnostics.An_index_signature_parameter_cannot_have_a_question_mark);
+            }
+            if (parameter.initializer) {
+                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_cannot_have_an_initializer);
+            }
+            if (!parameter.type) {
+                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_must_have_a_type_annotation);
+            }
+            if (parameter.type.kind !== 147 /* StringKeyword */ && parameter.type.kind !== 144 /* NumberKeyword */) {
+                var type = getTypeFromTypeNode(parameter.type);
+                if (type.flags & 4 /* String */ || type.flags & 8 /* Number */) {
+                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_type_cannot_be_a_type_alias_Consider_writing_0_Colon_1_Colon_2_instead, ts.getTextOfNode(parameter.name), typeToString(type), typeToString(node.type ? getTypeFromTypeNode(node.type) : anyType));
+                }
+                if (type.flags & 1048576 /* Union */ && allTypesAssignableToKind(type, 384 /* StringOrNumberLiteral */, /*strict*/ true)) {
+                    return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_type_cannot_be_a_union_type_Consider_using_a_mapped_object_type_instead);
+                }
+                return grammarErrorOnNode(parameter.name, ts.Diagnostics.An_index_signature_parameter_type_must_be_either_string_or_number);
+            }
+            if (!node.type) {
+                return grammarErrorOnNode(node, ts.Diagnostics.An_index_signature_must_have_a_type_annotation);
+            }
+            return false;
         }
-        return ts.setTextRange(ts.createCall(createJsxFactoryExpression(jsxFactoryEntity, reactNamespace, parentElement), 
-        /*typeArguments*/ undefined, argumentsList), location);
-    }
-    ts.createExpressionForJsxFragment = createExpressionForJsxFragment;
-    // Helpers
-    /**
-     * Gets an identifier for the name of an *unscoped* emit helper.
-     */
-    function getUnscopedHelperName(name) {
-        return ts.setEmitFlags(ts.createIdentifier(name), 4096 /* HelperName */ | 2 /* AdviseOnEmitNode */);
-    }
-    ts.getUnscopedHelperName = getUnscopedHelperName;
-    ts.valuesHelper = {
-        name: "typescript:values",
-        importName: "__values",
-        scoped: false,
-        text: "\n            var __values = (this && this.__values) || function(o) {\n                var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\n                if (m) return m.call(o);\n                if (o && typeof o.length === \"number\") return {\n                    next: function () {\n                        if (o && i >= o.length) o = void 0;\n                        return { value: o && o[i++], done: !o };\n                    }\n                };\n                throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\n            };"
-    };
-    function createValuesHelper(context, expression, location) {
-        context.requestEmitHelper(ts.valuesHelper);
-        return ts.setTextRange(ts.createCall(getUnscopedHelperName("__values"), 
-        /*typeArguments*/ undefined, [expression]), location);
-    }
-    ts.createValuesHelper = createValuesHelper;
-    ts.readHelper = {
-        name: "typescript:read",
-        importName: "__read",
-        scoped: false,
-        text: "\n            var __read = (this && this.__read) || function (o, n) {\n                var m = typeof Symbol === \"function\" && o[Symbol.iterator];\n                if (!m) return o;\n                var i = m.call(o), r, ar = [], e;\n                try {\n                    while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\n                }\n                catch (error) { e = { error: error }; }\n                finally {\n                    try {\n                        if (r && !r.done && (m = i[\"return\"])) m.call(i);\n                    }\n                    finally { if (e) throw e.error; }\n                }\n                return ar;\n            };"
-    };
-    function createReadHelper(context, iteratorRecord, count, location) {
-        context.requestEmitHelper(ts.readHelper);
-        return ts.setTextRange(ts.createCall(getUnscopedHelperName("__read"), 
-        /*typeArguments*/ undefined, count !== undefined
-            ? [iteratorRecord, ts.createLiteral(count)]
-            : [iteratorRecord]), location);
-    }
-    ts.createReadHelper = createReadHelper;
-    ts.spreadHelper = {
-        name: "typescript:spread",
-        importName: "__spread",
-        scoped: false,
-        dependencies: [ts.readHelper],
-        text: "\n            var __spread = (this && this.__spread) || function () {\n                for (var ar = [], i = 0; i < arguments.length; i++) ar = ar.concat(__read(arguments[i]));\n                return ar;\n            };"
-    };
-    function createSpreadHelper(context, argumentList, location) {
-        context.requestEmitHelper(ts.spreadHelper);
-        return ts.setTextRange(ts.createCall(getUnscopedHelperName("__spread"), 
-        /*typeArguments*/ undefined, argumentList), location);
-    }
-    ts.createSpreadHelper = createSpreadHelper;
-    ts.spreadArraysHelper = {
-        name: "typescript:spreadArrays",
-        importName: "__spreadArrays",
-        scoped: false,
-        text: "\n            var __spreadArrays = (this && this.__spreadArrays) || function () {\n                for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\n                for (var r = Array(s), k = 0, i = 0; i < il; i++)\n                    for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\n                        r[k] = a[j];\n                return r;\n            };"
-    };
-    function createSpreadArraysHelper(context, argumentList, location) {
-        context.requestEmitHelper(ts.spreadArraysHelper);
-        return ts.setTextRange(ts.createCall(getUnscopedHelperName("__spreadArrays"), 
-        /*typeArguments*/ undefined, argumentList), location);
-    }
-    ts.createSpreadArraysHelper = createSpreadArraysHelper;
-    // Utilities
-    function createForOfBindingStatement(node, boundValue) {
-        if (ts.isVariableDeclarationList(node)) {
-            var firstDeclaration = ts.first(node.declarations);
-            var updatedDeclaration = ts.updateVariableDeclaration(firstDeclaration, firstDeclaration.name, 
-            /*typeNode*/ undefined, boundValue);
-            return ts.setTextRange(ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.updateVariableDeclarationList(node, [updatedDeclaration])), 
-            /*location*/ node);
-        }
-        else {
-            var updatedExpression = ts.setTextRange(ts.createAssignment(node, boundValue), /*location*/ node);
-            return ts.setTextRange(ts.createStatement(updatedExpression), /*location*/ node);
-        }
-    }
-    ts.createForOfBindingStatement = createForOfBindingStatement;
-    function insertLeadingStatement(dest, source) {
-        if (ts.isBlock(dest)) {
-            return ts.updateBlock(dest, ts.setTextRange(ts.createNodeArray(__spreadArrays([source], dest.statements)), dest.statements));
+        function checkGrammarIndexSignature(node) {
+            // Prevent cascading error by short-circuit
+            return checkGrammarDecoratorsAndModifiers(node) || checkGrammarIndexSignatureParameters(node);
         }
-        else {
-            return ts.createBlock(ts.createNodeArray([dest, source]), /*multiLine*/ true);
+        function checkGrammarForAtLeastOneTypeArgument(node, typeArguments) {
+            if (typeArguments && typeArguments.length === 0) {
+                var sourceFile = ts.getSourceFileOfNode(node);
+                var start = typeArguments.pos - "<".length;
+                var end = ts.skipTrivia(sourceFile.text, typeArguments.end) + ">".length;
+                return grammarErrorAtPos(sourceFile, start, end - start, ts.Diagnostics.Type_argument_list_cannot_be_empty);
+            }
+            return false;
         }
-    }
-    ts.insertLeadingStatement = insertLeadingStatement;
-    function restoreEnclosingLabel(node, outermostLabeledStatement, afterRestoreLabelCallback) {
-        if (!outermostLabeledStatement) {
-            return node;
+        function checkGrammarTypeArguments(node, typeArguments) {
+            return checkGrammarForDisallowedTrailingComma(typeArguments) ||
+                checkGrammarForAtLeastOneTypeArgument(node, typeArguments);
         }
-        var updated = ts.updateLabel(outermostLabeledStatement, outermostLabeledStatement.label, outermostLabeledStatement.statement.kind === 238 /* LabeledStatement */
-            ? restoreEnclosingLabel(node, outermostLabeledStatement.statement)
-            : node);
-        if (afterRestoreLabelCallback) {
-            afterRestoreLabelCallback(outermostLabeledStatement);
+        function checkGrammarTaggedTemplateChain(node) {
+            if (node.questionDotToken || node.flags & 32 /* OptionalChain */) {
+                return grammarErrorOnNode(node.template, ts.Diagnostics.Tagged_template_expressions_are_not_permitted_in_an_optional_chain);
+            }
+            return false;
         }
-        return updated;
-    }
-    ts.restoreEnclosingLabel = restoreEnclosingLabel;
-    function shouldBeCapturedInTempVariable(node, cacheIdentifiers) {
-        var target = ts.skipParentheses(node);
-        switch (target.kind) {
-            case 75 /* Identifier */:
-                return cacheIdentifiers;
-            case 104 /* ThisKeyword */:
-            case 8 /* NumericLiteral */:
-            case 9 /* BigIntLiteral */:
-            case 10 /* StringLiteral */:
-                return false;
-            case 192 /* ArrayLiteralExpression */:
-                var elements = target.elements;
-                if (elements.length === 0) {
-                    return false;
+        function checkGrammarForOmittedArgument(args) {
+            if (args) {
+                for (var _i = 0, args_4 = args; _i < args_4.length; _i++) {
+                    var arg = args_4[_i];
+                    if (arg.kind === 222 /* OmittedExpression */) {
+                        return grammarErrorAtPos(arg, arg.pos, 0, ts.Diagnostics.Argument_expression_expected);
+                    }
                 }
-                return true;
-            case 193 /* ObjectLiteralExpression */:
-                return target.properties.length > 0;
-            default:
-                return true;
+            }
+            return false;
         }
-    }
-    function createCallBinding(expression, recordTempVariable, languageVersion, cacheIdentifiers) {
-        if (cacheIdentifiers === void 0) { cacheIdentifiers = false; }
-        var callee = skipOuterExpressions(expression, 15 /* All */);
-        var thisArg;
-        var target;
-        if (ts.isSuperProperty(callee)) {
-            thisArg = ts.createThis();
-            target = callee;
+        function checkGrammarArguments(args) {
+            return checkGrammarForOmittedArgument(args);
         }
-        else if (callee.kind === 102 /* SuperKeyword */) {
-            thisArg = ts.createThis();
-            target = languageVersion < 2 /* ES2015 */
-                ? ts.setTextRange(ts.createIdentifier("_super"), callee)
-                : callee;
+        function checkGrammarHeritageClause(node) {
+            var types = node.types;
+            if (checkGrammarForDisallowedTrailingComma(types)) {
+                return true;
+            }
+            if (types && types.length === 0) {
+                var listType = ts.tokenToString(node.token);
+                return grammarErrorAtPos(node, types.pos, 0, ts.Diagnostics._0_list_cannot_be_empty, listType);
+            }
+            return ts.some(types, checkGrammarExpressionWithTypeArguments);
         }
-        else if (ts.getEmitFlags(callee) & 4096 /* HelperName */) {
-            thisArg = ts.createVoidZero();
-            target = parenthesizeForAccess(callee);
+        function checkGrammarExpressionWithTypeArguments(node) {
+            return checkGrammarTypeArguments(node, node.typeArguments);
         }
-        else {
-            switch (callee.kind) {
-                case 194 /* PropertyAccessExpression */: {
-                    if (shouldBeCapturedInTempVariable(callee.expression, cacheIdentifiers)) {
-                        // for `a.b()` target is `(_a = a).b` and thisArg is `_a`
-                        thisArg = ts.createTempVariable(recordTempVariable);
-                        target = ts.createPropertyAccess(ts.setTextRange(ts.createAssignment(thisArg, callee.expression), callee.expression), callee.name);
-                        ts.setTextRange(target, callee);
+        function checkGrammarClassDeclarationHeritageClauses(node) {
+            var seenExtendsClause = false;
+            var seenImplementsClause = false;
+            if (!checkGrammarDecoratorsAndModifiers(node) && node.heritageClauses) {
+                for (var _i = 0, _a = node.heritageClauses; _i < _a.length; _i++) {
+                    var heritageClause = _a[_i];
+                    if (heritageClause.token === 93 /* ExtendsKeyword */) {
+                        if (seenExtendsClause) {
+                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.extends_clause_already_seen);
+                        }
+                        if (seenImplementsClause) {
+                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.extends_clause_must_precede_implements_clause);
+                        }
+                        if (heritageClause.types.length > 1) {
+                            return grammarErrorOnFirstToken(heritageClause.types[1], ts.Diagnostics.Classes_can_only_extend_a_single_class);
+                        }
+                        seenExtendsClause = true;
                     }
                     else {
-                        thisArg = callee.expression;
-                        target = callee;
+                        ts.Debug.assert(heritageClause.token === 116 /* ImplementsKeyword */);
+                        if (seenImplementsClause) {
+                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.implements_clause_already_seen);
+                        }
+                        seenImplementsClause = true;
                     }
-                    break;
+                    // Grammar checking heritageClause inside class declaration
+                    checkGrammarHeritageClause(heritageClause);
                 }
-                case 195 /* ElementAccessExpression */: {
-                    if (shouldBeCapturedInTempVariable(callee.expression, cacheIdentifiers)) {
-                        // for `a[b]()` target is `(_a = a)[b]` and thisArg is `_a`
-                        thisArg = ts.createTempVariable(recordTempVariable);
-                        target = ts.createElementAccess(ts.setTextRange(ts.createAssignment(thisArg, callee.expression), callee.expression), callee.argumentExpression);
-                        ts.setTextRange(target, callee);
+            }
+        }
+        function checkGrammarInterfaceDeclaration(node) {
+            var seenExtendsClause = false;
+            if (node.heritageClauses) {
+                for (var _i = 0, _a = node.heritageClauses; _i < _a.length; _i++) {
+                    var heritageClause = _a[_i];
+                    if (heritageClause.token === 93 /* ExtendsKeyword */) {
+                        if (seenExtendsClause) {
+                            return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.extends_clause_already_seen);
+                        }
+                        seenExtendsClause = true;
                     }
                     else {
-                        thisArg = callee.expression;
-                        target = callee;
+                        ts.Debug.assert(heritageClause.token === 116 /* ImplementsKeyword */);
+                        return grammarErrorOnFirstToken(heritageClause, ts.Diagnostics.Interface_declaration_cannot_have_implements_clause);
                     }
-                    break;
-                }
-                default: {
-                    // for `a()` target is `a` and thisArg is `void 0`
-                    thisArg = ts.createVoidZero();
-                    target = parenthesizeForAccess(expression);
-                    break;
+                    // Grammar checking heritageClause inside class declaration
+                    checkGrammarHeritageClause(heritageClause);
                 }
             }
+            return false;
         }
-        return { target: target, thisArg: thisArg };
-    }
-    ts.createCallBinding = createCallBinding;
-    function inlineExpressions(expressions) {
-        // Avoid deeply nested comma expressions as traversing them during emit can result in "Maximum call
-        // stack size exceeded" errors.
-        return expressions.length > 10
-            ? ts.createCommaList(expressions)
-            : ts.reduceLeft(expressions, ts.createComma);
-    }
-    ts.inlineExpressions = inlineExpressions;
-    function createExpressionFromEntityName(node) {
-        if (ts.isQualifiedName(node)) {
-            var left = createExpressionFromEntityName(node.left);
-            var right = ts.getMutableClone(node.right);
-            return ts.setTextRange(ts.createPropertyAccess(left, right), node);
+        function checkGrammarComputedPropertyName(node) {
+            // If node is not a computedPropertyName, just skip the grammar checking
+            if (node.kind !== 158 /* ComputedPropertyName */) {
+                return false;
+            }
+            var computedPropertyName = node;
+            if (computedPropertyName.expression.kind === 216 /* BinaryExpression */ && computedPropertyName.expression.operatorToken.kind === 27 /* CommaToken */) {
+                return grammarErrorOnNode(computedPropertyName.expression, ts.Diagnostics.A_comma_expression_is_not_allowed_in_a_computed_property_name);
+            }
+            return false;
         }
-        else {
-            return ts.getMutableClone(node);
+        function checkGrammarForGenerator(node) {
+            if (node.asteriskToken) {
+                ts.Debug.assert(node.kind === 251 /* FunctionDeclaration */ ||
+                    node.kind === 208 /* FunctionExpression */ ||
+                    node.kind === 165 /* MethodDeclaration */);
+                if (node.flags & 8388608 /* Ambient */) {
+                    return grammarErrorOnNode(node.asteriskToken, ts.Diagnostics.Generators_are_not_allowed_in_an_ambient_context);
+                }
+                if (!node.body) {
+                    return grammarErrorOnNode(node.asteriskToken, ts.Diagnostics.An_overload_signature_cannot_be_declared_as_a_generator);
+                }
+            }
         }
-    }
-    ts.createExpressionFromEntityName = createExpressionFromEntityName;
-    function createExpressionForPropertyName(memberName) {
-        if (ts.isIdentifier(memberName)) {
-            return ts.createLiteral(memberName);
+        function checkGrammarForInvalidQuestionMark(questionToken, message) {
+            return !!questionToken && grammarErrorOnNode(questionToken, message);
         }
-        else if (ts.isComputedPropertyName(memberName)) {
-            return ts.getMutableClone(memberName.expression);
+        function checkGrammarForInvalidExclamationToken(exclamationToken, message) {
+            return !!exclamationToken && grammarErrorOnNode(exclamationToken, message);
         }
-        else {
-            return ts.getMutableClone(memberName);
+        function checkGrammarObjectLiteralExpression(node, inDestructuring) {
+            var seen = new ts.Map();
+            for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
+                var prop = _a[_i];
+                if (prop.kind === 290 /* SpreadAssignment */) {
+                    if (inDestructuring) {
+                        // a rest property cannot be destructured any further
+                        var expression = ts.skipParentheses(prop.expression);
+                        if (ts.isArrayLiteralExpression(expression) || ts.isObjectLiteralExpression(expression)) {
+                            return grammarErrorOnNode(prop.expression, ts.Diagnostics.A_rest_element_cannot_contain_a_binding_pattern);
+                        }
+                    }
+                    continue;
+                }
+                var name = prop.name;
+                if (name.kind === 158 /* ComputedPropertyName */) {
+                    // If the name is not a ComputedPropertyName, the grammar checking will skip it
+                    checkGrammarComputedPropertyName(name);
+                }
+                if (prop.kind === 289 /* ShorthandPropertyAssignment */ && !inDestructuring && prop.objectAssignmentInitializer) {
+                    // having objectAssignmentInitializer is only valid in ObjectAssignmentPattern
+                    // outside of destructuring it is a syntax error
+                    return grammarErrorOnNode(prop.equalsToken, ts.Diagnostics.Did_you_mean_to_use_a_Colon_An_can_only_follow_a_property_name_when_the_containing_object_literal_is_part_of_a_destructuring_pattern);
+                }
+                if (name.kind === 79 /* PrivateIdentifier */) {
+                    return grammarErrorOnNode(name, ts.Diagnostics.Private_identifiers_are_not_allowed_outside_class_bodies);
+                }
+                // Modifiers are never allowed on properties except for 'async' on a method declaration
+                if (prop.modifiers) {
+                    // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
+                    for (var _b = 0, _c = prop.modifiers; _b < _c.length; _b++) { // TODO: GH#19955
+                        var mod = _c[_b];
+                        if (mod.kind !== 129 /* AsyncKeyword */ || prop.kind !== 165 /* MethodDeclaration */) {
+                            grammarErrorOnNode(mod, ts.Diagnostics._0_modifier_cannot_be_used_here, ts.getTextOfNode(mod));
+                        }
+                    }
+                }
+                // ECMA-262 11.1.5 Object Initializer
+                // If previous is not undefined then throw a SyntaxError exception if any of the following conditions are true
+                // a.This production is contained in strict code and IsDataDescriptor(previous) is true and
+                // IsDataDescriptor(propId.descriptor) is true.
+                //    b.IsDataDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true.
+                //    c.IsAccessorDescriptor(previous) is true and IsDataDescriptor(propId.descriptor) is true.
+                //    d.IsAccessorDescriptor(previous) is true and IsAccessorDescriptor(propId.descriptor) is true
+                // and either both previous and propId.descriptor have[[Get]] fields or both previous and propId.descriptor have[[Set]] fields
+                var currentKind = void 0;
+                switch (prop.kind) {
+                    case 289 /* ShorthandPropertyAssignment */:
+                        checkGrammarForInvalidExclamationToken(prop.exclamationToken, ts.Diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context);
+                    // falls through
+                    case 288 /* PropertyAssignment */:
+                        // Grammar checking for computedPropertyName and shorthandPropertyAssignment
+                        checkGrammarForInvalidQuestionMark(prop.questionToken, ts.Diagnostics.An_object_member_cannot_be_declared_optional);
+                        if (name.kind === 8 /* NumericLiteral */) {
+                            checkGrammarNumericLiteral(name);
+                        }
+                        currentKind = 4 /* PropertyAssignment */;
+                        break;
+                    case 165 /* MethodDeclaration */:
+                        currentKind = 8 /* Method */;
+                        break;
+                    case 167 /* GetAccessor */:
+                        currentKind = 1 /* GetAccessor */;
+                        break;
+                    case 168 /* SetAccessor */:
+                        currentKind = 2 /* SetAccessor */;
+                        break;
+                    default:
+                        throw ts.Debug.assertNever(prop, "Unexpected syntax kind:" + prop.kind);
+                }
+                if (!inDestructuring) {
+                    var effectiveName = ts.getPropertyNameForPropertyNameNode(name);
+                    if (effectiveName === undefined) {
+                        continue;
+                    }
+                    var existingKind = seen.get(effectiveName);
+                    if (!existingKind) {
+                        seen.set(effectiveName, currentKind);
+                    }
+                    else {
+                        if ((currentKind & 12 /* PropertyAssignmentOrMethod */) && (existingKind & 12 /* PropertyAssignmentOrMethod */)) {
+                            grammarErrorOnNode(name, ts.Diagnostics.Duplicate_identifier_0, ts.getTextOfNode(name));
+                        }
+                        else if ((currentKind & 3 /* GetOrSetAccessor */) && (existingKind & 3 /* GetOrSetAccessor */)) {
+                            if (existingKind !== 3 /* GetOrSetAccessor */ && currentKind !== existingKind) {
+                                seen.set(effectiveName, currentKind | existingKind);
+                            }
+                            else {
+                                return grammarErrorOnNode(name, ts.Diagnostics.An_object_literal_cannot_have_multiple_get_Slashset_accessors_with_the_same_name);
+                            }
+                        }
+                        else {
+                            return grammarErrorOnNode(name, ts.Diagnostics.An_object_literal_cannot_have_property_and_accessor_with_the_same_name);
+                        }
+                    }
+                }
+            }
         }
-    }
-    ts.createExpressionForPropertyName = createExpressionForPropertyName;
-    function createExpressionForObjectLiteralElementLike(node, property, receiver) {
-        if (property.name && ts.isPrivateIdentifier(property.name)) {
-            ts.Debug.failBadSyntaxKind(property.name, "Private identifiers are not allowed in object literals.");
+        function checkGrammarJsxElement(node) {
+            checkGrammarTypeArguments(node, node.typeArguments);
+            var seen = new ts.Map();
+            for (var _i = 0, _a = node.attributes.properties; _i < _a.length; _i++) {
+                var attr = _a[_i];
+                if (attr.kind === 282 /* JsxSpreadAttribute */) {
+                    continue;
+                }
+                var name = attr.name, initializer = attr.initializer;
+                if (!seen.get(name.escapedText)) {
+                    seen.set(name.escapedText, true);
+                }
+                else {
+                    return grammarErrorOnNode(name, ts.Diagnostics.JSX_elements_cannot_have_multiple_attributes_with_the_same_name);
+                }
+                if (initializer && initializer.kind === 283 /* JsxExpression */ && !initializer.expression) {
+                    return grammarErrorOnNode(initializer, ts.Diagnostics.JSX_attributes_must_only_be_assigned_a_non_empty_expression);
+                }
+            }
         }
-        switch (property.kind) {
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-                return createExpressionForAccessorDeclaration(node.properties, property, receiver, !!node.multiLine);
-            case 281 /* PropertyAssignment */:
-                return createExpressionForPropertyAssignment(property, receiver);
-            case 282 /* ShorthandPropertyAssignment */:
-                return createExpressionForShorthandPropertyAssignment(property, receiver);
-            case 161 /* MethodDeclaration */:
-                return createExpressionForMethodDeclaration(property, receiver);
+        function checkGrammarJsxExpression(node) {
+            if (node.expression && ts.isCommaSequence(node.expression)) {
+                return grammarErrorOnNode(node.expression, ts.Diagnostics.JSX_expressions_may_not_use_the_comma_operator_Did_you_mean_to_write_an_array);
+            }
         }
-    }
-    ts.createExpressionForObjectLiteralElementLike = createExpressionForObjectLiteralElementLike;
-    function createExpressionForAccessorDeclaration(properties, property, receiver, multiLine) {
-        var _a = ts.getAllAccessorDeclarations(properties, property), firstAccessor = _a.firstAccessor, getAccessor = _a.getAccessor, setAccessor = _a.setAccessor;
-        if (property === firstAccessor) {
-            var properties_7 = [];
-            if (getAccessor) {
-                var getterFunction = ts.createFunctionExpression(getAccessor.modifiers, 
-                /*asteriskToken*/ undefined, 
-                /*name*/ undefined, 
-                /*typeParameters*/ undefined, getAccessor.parameters, 
-                /*type*/ undefined, getAccessor.body // TODO: GH#18217
-                );
-                ts.setTextRange(getterFunction, getAccessor);
-                ts.setOriginalNode(getterFunction, getAccessor);
-                var getter = ts.createPropertyAssignment("get", getterFunction);
-                properties_7.push(getter);
+        function checkGrammarForInOrForOfStatement(forInOrOfStatement) {
+            if (checkGrammarStatementInAmbientContext(forInOrOfStatement)) {
+                return true;
             }
-            if (setAccessor) {
-                var setterFunction = ts.createFunctionExpression(setAccessor.modifiers, 
-                /*asteriskToken*/ undefined, 
-                /*name*/ undefined, 
-                /*typeParameters*/ undefined, setAccessor.parameters, 
-                /*type*/ undefined, setAccessor.body // TODO: GH#18217
-                );
-                ts.setTextRange(setterFunction, setAccessor);
-                ts.setOriginalNode(setterFunction, setAccessor);
-                var setter = ts.createPropertyAssignment("set", setterFunction);
-                properties_7.push(setter);
-            }
-            properties_7.push(ts.createPropertyAssignment("enumerable", getAccessor || setAccessor ? ts.createFalse() : ts.createTrue()));
-            properties_7.push(ts.createPropertyAssignment("configurable", ts.createTrue()));
-            var expression = ts.setTextRange(ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Object"), "defineProperty"), 
-            /*typeArguments*/ undefined, [
-                receiver,
-                createExpressionForPropertyName(property.name),
-                ts.createObjectLiteral(properties_7, multiLine)
-            ]), 
-            /*location*/ firstAccessor);
-            return ts.aggregateTransformFlags(expression);
+            if (forInOrOfStatement.kind === 239 /* ForOfStatement */ && forInOrOfStatement.awaitModifier) {
+                if ((forInOrOfStatement.flags & 32768 /* AwaitContext */) === 0 /* None */) {
+                    // use of 'for-await-of' in non-async function
+                    var sourceFile = ts.getSourceFileOfNode(forInOrOfStatement);
+                    if (!hasParseDiagnostics(sourceFile)) {
+                        var diagnostic = ts.createDiagnosticForNode(forInOrOfStatement.awaitModifier, ts.Diagnostics.A_for_await_of_statement_is_only_allowed_within_an_async_function_or_async_generator);
+                        var func = ts.getContainingFunction(forInOrOfStatement);
+                        if (func && func.kind !== 166 /* Constructor */) {
+                            ts.Debug.assert((ts.getFunctionFlags(func) & 2 /* Async */) === 0, "Enclosing function should never be an async function.");
+                            var relatedInfo = ts.createDiagnosticForNode(func, ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async);
+                            ts.addRelatedInfo(diagnostic, relatedInfo);
+                        }
+                        diagnostics.add(diagnostic);
+                        return true;
+                    }
+                    return false;
+                }
+            }
+            if (forInOrOfStatement.initializer.kind === 250 /* VariableDeclarationList */) {
+                var variableList = forInOrOfStatement.initializer;
+                if (!checkGrammarVariableDeclarationList(variableList)) {
+                    var declarations = variableList.declarations;
+                    // declarations.length can be zero if there is an error in variable declaration in for-of or for-in
+                    // See http://www.ecma-international.org/ecma-262/6.0/#sec-for-in-and-for-of-statements for details
+                    // For example:
+                    //      var let = 10;
+                    //      for (let of [1,2,3]) {} // this is invalid ES6 syntax
+                    //      for (let in [1,2,3]) {} // this is invalid ES6 syntax
+                    // We will then want to skip on grammar checking on variableList declaration
+                    if (!declarations.length) {
+                        return false;
+                    }
+                    if (declarations.length > 1) {
+                        var diagnostic = forInOrOfStatement.kind === 238 /* ForInStatement */
+                            ? ts.Diagnostics.Only_a_single_variable_declaration_is_allowed_in_a_for_in_statement
+                            : ts.Diagnostics.Only_a_single_variable_declaration_is_allowed_in_a_for_of_statement;
+                        return grammarErrorOnFirstToken(variableList.declarations[1], diagnostic);
+                    }
+                    var firstDeclaration = declarations[0];
+                    if (firstDeclaration.initializer) {
+                        var diagnostic = forInOrOfStatement.kind === 238 /* ForInStatement */
+                            ? ts.Diagnostics.The_variable_declaration_of_a_for_in_statement_cannot_have_an_initializer
+                            : ts.Diagnostics.The_variable_declaration_of_a_for_of_statement_cannot_have_an_initializer;
+                        return grammarErrorOnNode(firstDeclaration.name, diagnostic);
+                    }
+                    if (firstDeclaration.type) {
+                        var diagnostic = forInOrOfStatement.kind === 238 /* ForInStatement */
+                            ? ts.Diagnostics.The_left_hand_side_of_a_for_in_statement_cannot_use_a_type_annotation
+                            : ts.Diagnostics.The_left_hand_side_of_a_for_of_statement_cannot_use_a_type_annotation;
+                        return grammarErrorOnNode(firstDeclaration, diagnostic);
+                    }
+                }
+            }
+            return false;
         }
-        return undefined;
-    }
-    function createExpressionForPropertyAssignment(property, receiver) {
-        return ts.aggregateTransformFlags(ts.setOriginalNode(ts.setTextRange(ts.createAssignment(createMemberAccessForPropertyName(receiver, property.name, /*location*/ property.name), property.initializer), property), property));
-    }
-    function createExpressionForShorthandPropertyAssignment(property, receiver) {
-        return ts.aggregateTransformFlags(ts.setOriginalNode(ts.setTextRange(ts.createAssignment(createMemberAccessForPropertyName(receiver, property.name, /*location*/ property.name), ts.getSynthesizedClone(property.name)), 
-        /*location*/ property), 
-        /*original*/ property));
-    }
-    function createExpressionForMethodDeclaration(method, receiver) {
-        return ts.aggregateTransformFlags(ts.setOriginalNode(ts.setTextRange(ts.createAssignment(createMemberAccessForPropertyName(receiver, method.name, /*location*/ method.name), ts.setOriginalNode(ts.setTextRange(ts.createFunctionExpression(method.modifiers, method.asteriskToken, 
-        /*name*/ undefined, 
-        /*typeParameters*/ undefined, method.parameters, 
-        /*type*/ undefined, method.body // TODO: GH#18217
-        ), 
-        /*location*/ method), 
-        /*original*/ method)), 
-        /*location*/ method), 
-        /*original*/ method));
-    }
-    /**
-     * Gets the internal name of a declaration. This is primarily used for declarations that can be
-     * referred to by name in the body of an ES5 class function body. An internal name will *never*
-     * be prefixed with an module or namespace export modifier like "exports." when emitted as an
-     * expression. An internal name will also *never* be renamed due to a collision with a block
-     * scoped variable.
-     *
-     * @param node The declaration.
-     * @param allowComments A value indicating whether comments may be emitted for the name.
-     * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
-     */
-    function getInternalName(node, allowComments, allowSourceMaps) {
-        return getName(node, allowComments, allowSourceMaps, 16384 /* LocalName */ | 32768 /* InternalName */);
-    }
-    ts.getInternalName = getInternalName;
-    /**
-     * Gets whether an identifier should only be referred to by its internal name.
-     */
-    function isInternalName(node) {
-        return (ts.getEmitFlags(node) & 32768 /* InternalName */) !== 0;
-    }
-    ts.isInternalName = isInternalName;
-    /**
-     * Gets the local name of a declaration. This is primarily used for declarations that can be
-     * referred to by name in the declaration's immediate scope (classes, enums, namespaces). A
-     * local name will *never* be prefixed with an module or namespace export modifier like
-     * "exports." when emitted as an expression.
-     *
-     * @param node The declaration.
-     * @param allowComments A value indicating whether comments may be emitted for the name.
-     * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
-     */
-    function getLocalName(node, allowComments, allowSourceMaps) {
-        return getName(node, allowComments, allowSourceMaps, 16384 /* LocalName */);
-    }
-    ts.getLocalName = getLocalName;
-    /**
-     * Gets whether an identifier should only be referred to by its local name.
-     */
-    function isLocalName(node) {
-        return (ts.getEmitFlags(node) & 16384 /* LocalName */) !== 0;
-    }
-    ts.isLocalName = isLocalName;
-    /**
-     * Gets the export name of a declaration. This is primarily used for declarations that can be
-     * referred to by name in the declaration's immediate scope (classes, enums, namespaces). An
-     * export name will *always* be prefixed with an module or namespace export modifier like
-     * `"exports."` when emitted as an expression if the name points to an exported symbol.
-     *
-     * @param node The declaration.
-     * @param allowComments A value indicating whether comments may be emitted for the name.
-     * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
-     */
-    function getExportName(node, allowComments, allowSourceMaps) {
-        return getName(node, allowComments, allowSourceMaps, 8192 /* ExportName */);
-    }
-    ts.getExportName = getExportName;
-    /**
-     * Gets whether an identifier should only be referred to by its export representation if the
-     * name points to an exported symbol.
-     */
-    function isExportName(node) {
-        return (ts.getEmitFlags(node) & 8192 /* ExportName */) !== 0;
-    }
-    ts.isExportName = isExportName;
-    /**
-     * Gets the name of a declaration for use in declarations.
-     *
-     * @param node The declaration.
-     * @param allowComments A value indicating whether comments may be emitted for the name.
-     * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
-     */
-    function getDeclarationName(node, allowComments, allowSourceMaps) {
-        return getName(node, allowComments, allowSourceMaps);
-    }
-    ts.getDeclarationName = getDeclarationName;
-    function getName(node, allowComments, allowSourceMaps, emitFlags) {
-        if (emitFlags === void 0) { emitFlags = 0; }
-        var nodeName = ts.getNameOfDeclaration(node);
-        if (nodeName && ts.isIdentifier(nodeName) && !ts.isGeneratedIdentifier(nodeName)) {
-            var name = ts.getMutableClone(nodeName);
-            emitFlags |= ts.getEmitFlags(nodeName);
-            if (!allowSourceMaps)
-                emitFlags |= 48 /* NoSourceMap */;
-            if (!allowComments)
-                emitFlags |= 1536 /* NoComments */;
-            if (emitFlags)
-                ts.setEmitFlags(name, emitFlags);
-            return name;
+        function checkGrammarAccessor(accessor) {
+            if (!(accessor.flags & 8388608 /* Ambient */)) {
+                if (languageVersion < 1 /* ES5 */) {
+                    return grammarErrorOnNode(accessor.name, ts.Diagnostics.Accessors_are_only_available_when_targeting_ECMAScript_5_and_higher);
+                }
+                if (accessor.body === undefined && !ts.hasSyntacticModifier(accessor, 128 /* Abstract */)) {
+                    return grammarErrorAtPos(accessor, accessor.end - 1, ";".length, ts.Diagnostics._0_expected, "{");
+                }
+            }
+            if (accessor.body && ts.hasSyntacticModifier(accessor, 128 /* Abstract */)) {
+                return grammarErrorOnNode(accessor, ts.Diagnostics.An_abstract_accessor_cannot_have_an_implementation);
+            }
+            if (accessor.typeParameters) {
+                return grammarErrorOnNode(accessor.name, ts.Diagnostics.An_accessor_cannot_have_type_parameters);
+            }
+            if (!doesAccessorHaveCorrectParameterCount(accessor)) {
+                return grammarErrorOnNode(accessor.name, accessor.kind === 167 /* GetAccessor */ ?
+                    ts.Diagnostics.A_get_accessor_cannot_have_parameters :
+                    ts.Diagnostics.A_set_accessor_must_have_exactly_one_parameter);
+            }
+            if (accessor.kind === 168 /* SetAccessor */) {
+                if (accessor.type) {
+                    return grammarErrorOnNode(accessor.name, ts.Diagnostics.A_set_accessor_cannot_have_a_return_type_annotation);
+                }
+                var parameter = ts.Debug.checkDefined(ts.getSetAccessorValueParameter(accessor), "Return value does not match parameter count assertion.");
+                if (parameter.dotDotDotToken) {
+                    return grammarErrorOnNode(parameter.dotDotDotToken, ts.Diagnostics.A_set_accessor_cannot_have_rest_parameter);
+                }
+                if (parameter.questionToken) {
+                    return grammarErrorOnNode(parameter.questionToken, ts.Diagnostics.A_set_accessor_cannot_have_an_optional_parameter);
+                }
+                if (parameter.initializer) {
+                    return grammarErrorOnNode(accessor.name, ts.Diagnostics.A_set_accessor_parameter_cannot_have_an_initializer);
+                }
+            }
+            return false;
         }
-        return ts.getGeneratedNameForNode(node);
-    }
-    /**
-     * Gets the exported name of a declaration for use in expressions.
-     *
-     * An exported name will *always* be prefixed with an module or namespace export modifier like
-     * "exports." if the name points to an exported symbol.
-     *
-     * @param ns The namespace identifier.
-     * @param node The declaration.
-     * @param allowComments A value indicating whether comments may be emitted for the name.
-     * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
-     */
-    function getExternalModuleOrNamespaceExportName(ns, node, allowComments, allowSourceMaps) {
-        if (ns && ts.hasModifier(node, 1 /* Export */)) {
-            return getNamespaceMemberName(ns, getName(node), allowComments, allowSourceMaps);
+        /** Does the accessor have the right number of parameters?
+         * A get accessor has no parameters or a single `this` parameter.
+         * A set accessor has one parameter or a `this` parameter and one more parameter.
+         */
+        function doesAccessorHaveCorrectParameterCount(accessor) {
+            return getAccessorThisParameter(accessor) || accessor.parameters.length === (accessor.kind === 167 /* GetAccessor */ ? 0 : 1);
         }
-        return getExportName(node, allowComments, allowSourceMaps);
-    }
-    ts.getExternalModuleOrNamespaceExportName = getExternalModuleOrNamespaceExportName;
-    /**
-     * Gets a namespace-qualified name for use in expressions.
-     *
-     * @param ns The namespace identifier.
-     * @param name The name.
-     * @param allowComments A value indicating whether comments may be emitted for the name.
-     * @param allowSourceMaps A value indicating whether source maps may be emitted for the name.
-     */
-    function getNamespaceMemberName(ns, name, allowComments, allowSourceMaps) {
-        var qualifiedName = ts.createPropertyAccess(ns, ts.nodeIsSynthesized(name) ? name : ts.getSynthesizedClone(name));
-        ts.setTextRange(qualifiedName, name);
-        var emitFlags = 0;
-        if (!allowSourceMaps)
-            emitFlags |= 48 /* NoSourceMap */;
-        if (!allowComments)
-            emitFlags |= 1536 /* NoComments */;
-        if (emitFlags)
-            ts.setEmitFlags(qualifiedName, emitFlags);
-        return qualifiedName;
-    }
-    ts.getNamespaceMemberName = getNamespaceMemberName;
-    function convertToFunctionBody(node, multiLine) {
-        return ts.isBlock(node) ? node : ts.setTextRange(ts.createBlock([ts.setTextRange(ts.createReturn(node), node)], multiLine), node);
-    }
-    ts.convertToFunctionBody = convertToFunctionBody;
-    function convertFunctionDeclarationToExpression(node) {
-        if (!node.body)
-            return ts.Debug.fail();
-        var updated = ts.createFunctionExpression(node.modifiers, node.asteriskToken, node.name, node.typeParameters, node.parameters, node.type, node.body);
-        ts.setOriginalNode(updated, node);
-        ts.setTextRange(updated, node);
-        if (ts.getStartsOnNewLine(node)) {
-            ts.setStartsOnNewLine(updated, /*newLine*/ true);
+        function getAccessorThisParameter(accessor) {
+            if (accessor.parameters.length === (accessor.kind === 167 /* GetAccessor */ ? 1 : 2)) {
+                return ts.getThisParameter(accessor);
+            }
         }
-        ts.aggregateTransformFlags(updated);
-        return updated;
-    }
-    ts.convertFunctionDeclarationToExpression = convertFunctionDeclarationToExpression;
-    function isUseStrictPrologue(node) {
-        return ts.isStringLiteral(node.expression) && node.expression.text === "use strict";
-    }
-    /**
-     * Add any necessary prologue-directives into target statement-array.
-     * The function needs to be called during each transformation step.
-     * This function needs to be called whenever we transform the statement
-     * list of a source file, namespace, or function-like body.
-     *
-     * @param target: result statements array
-     * @param source: origin statements array
-     * @param ensureUseStrict: boolean determining whether the function need to add prologue-directives
-     * @param visitor: Optional callback used to visit any custom prologue directives.
-     */
-    function addPrologue(target, source, ensureUseStrict, visitor) {
-        var offset = addStandardPrologue(target, source, ensureUseStrict);
-        return addCustomPrologue(target, source, offset, visitor);
-    }
-    ts.addPrologue = addPrologue;
-    /**
-     * Add just the standard (string-expression) prologue-directives into target statement-array.
-     * The function needs to be called during each transformation step.
-     * This function needs to be called whenever we transform the statement
-     * list of a source file, namespace, or function-like body.
-     */
-    function addStandardPrologue(target, source, ensureUseStrict) {
-        ts.Debug.assert(target.length === 0, "Prologue directives should be at the first statement in the target statements array");
-        var foundUseStrict = false;
-        var statementOffset = 0;
-        var numStatements = source.length;
-        while (statementOffset < numStatements) {
-            var statement = source[statementOffset];
-            if (ts.isPrologueDirective(statement)) {
-                if (isUseStrictPrologue(statement)) {
-                    foundUseStrict = true;
+        function checkGrammarTypeOperatorNode(node) {
+            if (node.operator === 151 /* UniqueKeyword */) {
+                if (node.type.kind !== 148 /* SymbolKeyword */) {
+                    return grammarErrorOnNode(node.type, ts.Diagnostics._0_expected, ts.tokenToString(148 /* SymbolKeyword */));
+                }
+                var parent = ts.walkUpParenthesizedTypes(node.parent);
+                if (ts.isInJSFile(parent) && ts.isJSDocTypeExpression(parent)) {
+                    parent = parent.parent;
+                    if (ts.isJSDocTypeTag(parent)) {
+                        // walk up past JSDoc comment node
+                        parent = parent.parent.parent;
+                    }
+                }
+                switch (parent.kind) {
+                    case 249 /* VariableDeclaration */:
+                        var decl = parent;
+                        if (decl.name.kind !== 78 /* Identifier */) {
+                            return grammarErrorOnNode(node, ts.Diagnostics.unique_symbol_types_may_not_be_used_on_a_variable_declaration_with_a_binding_name);
+                        }
+                        if (!ts.isVariableDeclarationInVariableStatement(decl)) {
+                            return grammarErrorOnNode(node, ts.Diagnostics.unique_symbol_types_are_only_allowed_on_variables_in_a_variable_statement);
+                        }
+                        if (!(decl.parent.flags & 2 /* Const */)) {
+                            return grammarErrorOnNode(parent.name, ts.Diagnostics.A_variable_whose_type_is_a_unique_symbol_type_must_be_const);
+                        }
+                        break;
+                    case 163 /* PropertyDeclaration */:
+                        if (!ts.hasSyntacticModifier(parent, 32 /* Static */) ||
+                            !ts.hasEffectiveModifier(parent, 64 /* Readonly */)) {
+                            return grammarErrorOnNode(parent.name, ts.Diagnostics.A_property_of_a_class_whose_type_is_a_unique_symbol_type_must_be_both_static_and_readonly);
+                        }
+                        break;
+                    case 162 /* PropertySignature */:
+                        if (!ts.hasSyntacticModifier(parent, 64 /* Readonly */)) {
+                            return grammarErrorOnNode(parent.name, ts.Diagnostics.A_property_of_an_interface_or_type_literal_whose_type_is_a_unique_symbol_type_must_be_readonly);
+                        }
+                        break;
+                    default:
+                        return grammarErrorOnNode(node, ts.Diagnostics.unique_symbol_types_are_not_allowed_here);
                 }
-                target.push(statement);
             }
-            else {
-                break;
+            else if (node.operator === 142 /* ReadonlyKeyword */) {
+                if (node.type.kind !== 178 /* ArrayType */ && node.type.kind !== 179 /* TupleType */) {
+                    return grammarErrorOnFirstToken(node, ts.Diagnostics.readonly_type_modifier_is_only_permitted_on_array_and_tuple_literal_types, ts.tokenToString(148 /* SymbolKeyword */));
+                }
             }
-            statementOffset++;
         }
-        if (ensureUseStrict && !foundUseStrict) {
-            target.push(startOnNewLine(ts.createStatement(ts.createLiteral("use strict"))));
+        function checkGrammarForInvalidDynamicName(node, message) {
+            if (isNonBindableDynamicName(node)) {
+                return grammarErrorOnNode(node, message);
+            }
         }
-        return statementOffset;
-    }
-    ts.addStandardPrologue = addStandardPrologue;
-    function addCustomPrologue(target, source, statementOffset, visitor, filter) {
-        if (filter === void 0) { filter = ts.returnTrue; }
-        var numStatements = source.length;
-        while (statementOffset !== undefined && statementOffset < numStatements) {
-            var statement = source[statementOffset];
-            if (ts.getEmitFlags(statement) & 1048576 /* CustomPrologue */ && filter(statement)) {
-                ts.append(target, visitor ? ts.visitNode(statement, visitor, ts.isStatement) : statement);
+        function checkGrammarMethod(node) {
+            if (checkGrammarFunctionLikeDeclaration(node)) {
+                return true;
             }
-            else {
-                break;
+            if (node.kind === 165 /* MethodDeclaration */) {
+                if (node.parent.kind === 200 /* ObjectLiteralExpression */) {
+                    // We only disallow modifier on a method declaration if it is a property of object-literal-expression
+                    if (node.modifiers && !(node.modifiers.length === 1 && ts.first(node.modifiers).kind === 129 /* AsyncKeyword */)) {
+                        return grammarErrorOnFirstToken(node, ts.Diagnostics.Modifiers_cannot_appear_here);
+                    }
+                    else if (checkGrammarForInvalidQuestionMark(node.questionToken, ts.Diagnostics.An_object_member_cannot_be_declared_optional)) {
+                        return true;
+                    }
+                    else if (checkGrammarForInvalidExclamationToken(node.exclamationToken, ts.Diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context)) {
+                        return true;
+                    }
+                    else if (node.body === undefined) {
+                        return grammarErrorAtPos(node, node.end - 1, ";".length, ts.Diagnostics._0_expected, "{");
+                    }
+                }
+                if (checkGrammarForGenerator(node)) {
+                    return true;
+                }
+            }
+            if (ts.isClassLike(node.parent)) {
+                // Technically, computed properties in ambient contexts is disallowed
+                // for property declarations and accessors too, not just methods.
+                // However, property declarations disallow computed names in general,
+                // and accessors are not allowed in ambient contexts in general,
+                // so this error only really matters for methods.
+                if (node.flags & 8388608 /* Ambient */) {
+                    return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_an_ambient_context_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
+                }
+                else if (node.kind === 165 /* MethodDeclaration */ && !node.body) {
+                    return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_method_overload_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
+                }
+            }
+            else if (node.parent.kind === 253 /* InterfaceDeclaration */) {
+                return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_an_interface_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
+            }
+            else if (node.parent.kind === 177 /* TypeLiteral */) {
+                return checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_type_literal_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type);
             }
-            statementOffset++;
         }
-        return statementOffset;
-    }
-    ts.addCustomPrologue = addCustomPrologue;
-    function findUseStrictPrologue(statements) {
-        for (var _i = 0, statements_4 = statements; _i < statements_4.length; _i++) {
-            var statement = statements_4[_i];
-            if (ts.isPrologueDirective(statement)) {
-                if (isUseStrictPrologue(statement)) {
-                    return statement;
+        function checkGrammarBreakOrContinueStatement(node) {
+            var current = node;
+            while (current) {
+                if (ts.isFunctionLike(current)) {
+                    return grammarErrorOnNode(node, ts.Diagnostics.Jump_target_cannot_cross_function_boundary);
+                }
+                switch (current.kind) {
+                    case 245 /* LabeledStatement */:
+                        if (node.label && current.label.escapedText === node.label.escapedText) {
+                            // found matching label - verify that label usage is correct
+                            // continue can only target labels that are on iteration statements
+                            var isMisplacedContinueLabel = node.kind === 240 /* ContinueStatement */
+                                && !ts.isIterationStatement(current.statement, /*lookInLabeledStatement*/ true);
+                            if (isMisplacedContinueLabel) {
+                                return grammarErrorOnNode(node, ts.Diagnostics.A_continue_statement_can_only_jump_to_a_label_of_an_enclosing_iteration_statement);
+                            }
+                            return false;
+                        }
+                        break;
+                    case 244 /* SwitchStatement */:
+                        if (node.kind === 241 /* BreakStatement */ && !node.label) {
+                            // unlabeled break within switch statement - ok
+                            return false;
+                        }
+                        break;
+                    default:
+                        if (ts.isIterationStatement(current, /*lookInLabeledStatement*/ false) && !node.label) {
+                            // unlabeled break or continue within iteration statement - ok
+                            return false;
+                        }
+                        break;
                 }
+                current = current.parent;
+            }
+            if (node.label) {
+                var message = node.kind === 241 /* BreakStatement */
+                    ? ts.Diagnostics.A_break_statement_can_only_jump_to_a_label_of_an_enclosing_statement
+                    : ts.Diagnostics.A_continue_statement_can_only_jump_to_a_label_of_an_enclosing_iteration_statement;
+                return grammarErrorOnNode(node, message);
             }
             else {
-                break;
+                var message = node.kind === 241 /* BreakStatement */
+                    ? ts.Diagnostics.A_break_statement_can_only_be_used_within_an_enclosing_iteration_or_switch_statement
+                    : ts.Diagnostics.A_continue_statement_can_only_be_used_within_an_enclosing_iteration_statement;
+                return grammarErrorOnNode(node, message);
             }
         }
-        return undefined;
-    }
-    ts.findUseStrictPrologue = findUseStrictPrologue;
-    function startsWithUseStrict(statements) {
-        var firstStatement = ts.firstOrUndefined(statements);
-        return firstStatement !== undefined
-            && ts.isPrologueDirective(firstStatement)
-            && isUseStrictPrologue(firstStatement);
-    }
-    ts.startsWithUseStrict = startsWithUseStrict;
-    /**
-     * Ensures "use strict" directive is added
-     *
-     * @param statements An array of statements
-     */
-    function ensureUseStrict(statements) {
-        var foundUseStrict = findUseStrictPrologue(statements);
-        if (!foundUseStrict) {
-            return ts.setTextRange(ts.createNodeArray(__spreadArrays([
-                startOnNewLine(ts.createStatement(ts.createLiteral("use strict")))
-            ], statements)), statements);
+        function checkGrammarBindingElement(node) {
+            if (node.dotDotDotToken) {
+                var elements = node.parent.elements;
+                if (node !== ts.last(elements)) {
+                    return grammarErrorOnNode(node, ts.Diagnostics.A_rest_element_must_be_last_in_a_destructuring_pattern);
+                }
+                checkGrammarForDisallowedTrailingComma(elements, ts.Diagnostics.A_rest_parameter_or_binding_pattern_may_not_have_a_trailing_comma);
+                if (node.propertyName) {
+                    return grammarErrorOnNode(node.name, ts.Diagnostics.A_rest_element_cannot_have_a_property_name);
+                }
+            }
+            if (node.dotDotDotToken && node.initializer) {
+                // Error on equals token which immediately precedes the initializer
+                return grammarErrorAtPos(node, node.initializer.pos - 1, 1, ts.Diagnostics.A_rest_element_cannot_have_an_initializer);
+            }
         }
-        return statements;
-    }
-    ts.ensureUseStrict = ensureUseStrict;
-    /**
-     * Wraps the operand to a BinaryExpression in parentheses if they are needed to preserve the intended
-     * order of operations.
-     *
-     * @param binaryOperator The operator for the BinaryExpression.
-     * @param operand The operand for the BinaryExpression.
-     * @param isLeftSideOfBinary A value indicating whether the operand is the left side of the
-     *                           BinaryExpression.
-     */
-    function parenthesizeBinaryOperand(binaryOperator, operand, isLeftSideOfBinary, leftOperand) {
-        var skipped = ts.skipPartiallyEmittedExpressions(operand);
-        // If the resulting expression is already parenthesized, we do not need to do any further processing.
-        if (skipped.kind === 200 /* ParenthesizedExpression */) {
-            return operand;
+        function isStringOrNumberLiteralExpression(expr) {
+            return ts.isStringOrNumericLiteralLike(expr) ||
+                expr.kind === 214 /* PrefixUnaryExpression */ && expr.operator === 40 /* MinusToken */ &&
+                    expr.operand.kind === 8 /* NumericLiteral */;
         }
-        return binaryOperandNeedsParentheses(binaryOperator, operand, isLeftSideOfBinary, leftOperand)
-            ? ts.createParen(operand)
-            : operand;
-    }
-    ts.parenthesizeBinaryOperand = parenthesizeBinaryOperand;
-    /**
-     * Determines whether the operand to a BinaryExpression needs to be parenthesized.
-     *
-     * @param binaryOperator The operator for the BinaryExpression.
-     * @param operand The operand for the BinaryExpression.
-     * @param isLeftSideOfBinary A value indicating whether the operand is the left side of the
-     *                           BinaryExpression.
-     */
-    function binaryOperandNeedsParentheses(binaryOperator, operand, isLeftSideOfBinary, leftOperand) {
-        // If the operand has lower precedence, then it needs to be parenthesized to preserve the
-        // intent of the expression. For example, if the operand is `a + b` and the operator is
-        // `*`, then we need to parenthesize the operand to preserve the intended order of
-        // operations: `(a + b) * x`.
-        //
-        // If the operand has higher precedence, then it does not need to be parenthesized. For
-        // example, if the operand is `a * b` and the operator is `+`, then we do not need to
-        // parenthesize to preserve the intended order of operations: `a * b + x`.
-        //
-        // If the operand has the same precedence, then we need to check the associativity of
-        // the operator based on whether this is the left or right operand of the expression.
-        //
-        // For example, if `a / d` is on the right of operator `*`, we need to parenthesize
-        // to preserve the intended order of operations: `x * (a / d)`
-        //
-        // If `a ** d` is on the left of operator `**`, we need to parenthesize to preserve
-        // the intended order of operations: `(a ** b) ** c`
-        var binaryOperatorPrecedence = ts.getOperatorPrecedence(209 /* BinaryExpression */, binaryOperator);
-        var binaryOperatorAssociativity = ts.getOperatorAssociativity(209 /* BinaryExpression */, binaryOperator);
-        var emittedOperand = ts.skipPartiallyEmittedExpressions(operand);
-        if (!isLeftSideOfBinary && operand.kind === 202 /* ArrowFunction */ && binaryOperatorPrecedence > 3) {
-            // We need to parenthesize arrow functions on the right side to avoid it being
-            // parsed as parenthesized expression: `a && (() => {})`
-            return true;
+        function isBigIntLiteralExpression(expr) {
+            return expr.kind === 9 /* BigIntLiteral */ ||
+                expr.kind === 214 /* PrefixUnaryExpression */ && expr.operator === 40 /* MinusToken */ &&
+                    expr.operand.kind === 9 /* BigIntLiteral */;
+        }
+        function isSimpleLiteralEnumReference(expr) {
+            if ((ts.isPropertyAccessExpression(expr) || (ts.isElementAccessExpression(expr) && isStringOrNumberLiteralExpression(expr.argumentExpression))) &&
+                ts.isEntityNameExpression(expr.expression)) {
+                return !!(checkExpressionCached(expr).flags & 1024 /* EnumLiteral */);
+            }
+        }
+        function checkAmbientInitializer(node) {
+            var initializer = node.initializer;
+            if (initializer) {
+                var isInvalidInitializer = !(isStringOrNumberLiteralExpression(initializer) ||
+                    isSimpleLiteralEnumReference(initializer) ||
+                    initializer.kind === 109 /* TrueKeyword */ || initializer.kind === 94 /* FalseKeyword */ ||
+                    isBigIntLiteralExpression(initializer));
+                var isConstOrReadonly = ts.isDeclarationReadonly(node) || ts.isVariableDeclaration(node) && ts.isVarConst(node);
+                if (isConstOrReadonly && !node.type) {
+                    if (isInvalidInitializer) {
+                        return grammarErrorOnNode(initializer, ts.Diagnostics.A_const_initializer_in_an_ambient_context_must_be_a_string_or_numeric_literal_or_literal_enum_reference);
+                    }
+                }
+                else {
+                    return grammarErrorOnNode(initializer, ts.Diagnostics.Initializers_are_not_allowed_in_ambient_contexts);
+                }
+                if (!isConstOrReadonly || isInvalidInitializer) {
+                    return grammarErrorOnNode(initializer, ts.Diagnostics.Initializers_are_not_allowed_in_ambient_contexts);
+                }
+            }
+        }
+        function checkGrammarVariableDeclaration(node) {
+            if (node.parent.parent.kind !== 238 /* ForInStatement */ && node.parent.parent.kind !== 239 /* ForOfStatement */) {
+                if (node.flags & 8388608 /* Ambient */) {
+                    checkAmbientInitializer(node);
+                }
+                else if (!node.initializer) {
+                    if (ts.isBindingPattern(node.name) && !ts.isBindingPattern(node.parent)) {
+                        return grammarErrorOnNode(node, ts.Diagnostics.A_destructuring_declaration_must_have_an_initializer);
+                    }
+                    if (ts.isVarConst(node)) {
+                        return grammarErrorOnNode(node, ts.Diagnostics.const_declarations_must_be_initialized);
+                    }
+                }
+            }
+            if (node.exclamationToken && (node.parent.parent.kind !== 232 /* VariableStatement */ || !node.type || node.initializer || node.flags & 8388608 /* Ambient */)) {
+                var message = node.initializer
+                    ? ts.Diagnostics.Declarations_with_initializers_cannot_also_have_definite_assignment_assertions
+                    : ts.Diagnostics.Declarations_with_definite_assignment_assertions_must_also_have_type_annotations;
+                return grammarErrorOnNode(node.exclamationToken, message);
+            }
+            var moduleKind = ts.getEmitModuleKind(compilerOptions);
+            if (moduleKind < ts.ModuleKind.ES2015 && moduleKind !== ts.ModuleKind.System &&
+                !(node.parent.parent.flags & 8388608 /* Ambient */) && ts.hasSyntacticModifier(node.parent.parent, 1 /* Export */)) {
+                checkESModuleMarker(node.name);
+            }
+            var checkLetConstNames = (ts.isLet(node) || ts.isVarConst(node));
+            // 1. LexicalDeclaration : LetOrConst BindingList ;
+            // It is a Syntax Error if the BoundNames of BindingList contains "let".
+            // 2. ForDeclaration: ForDeclaration : LetOrConst ForBinding
+            // It is a Syntax Error if the BoundNames of ForDeclaration contains "let".
+            // It is a SyntaxError if a VariableDeclaration or VariableDeclarationNoIn occurs within strict code
+            // and its Identifier is eval or arguments
+            return checkLetConstNames && checkGrammarNameInLetOrConstDeclarations(node.name);
         }
-        var operandPrecedence = ts.getExpressionPrecedence(emittedOperand);
-        switch (ts.compareValues(operandPrecedence, binaryOperatorPrecedence)) {
-            case -1 /* LessThan */:
-                // If the operand is the right side of a right-associative binary operation
-                // and is a yield expression, then we do not need parentheses.
-                if (!isLeftSideOfBinary
-                    && binaryOperatorAssociativity === 1 /* Right */
-                    && operand.kind === 212 /* YieldExpression */) {
-                    return false;
-                }
-                return true;
-            case 1 /* GreaterThan */:
-                return false;
-            case 0 /* EqualTo */:
-                if (isLeftSideOfBinary) {
-                    // No need to parenthesize the left operand when the binary operator is
-                    // left associative:
-                    //  (a*b)/x    -> a*b/x
-                    //  (a**b)/x   -> a**b/x
-                    //
-                    // Parentheses are needed for the left operand when the binary operator is
-                    // right associative:
-                    //  (a/b)**x   -> (a/b)**x
-                    //  (a**b)**x  -> (a**b)**x
-                    return binaryOperatorAssociativity === 1 /* Right */;
+        function checkESModuleMarker(name) {
+            if (name.kind === 78 /* Identifier */) {
+                if (ts.idText(name) === "__esModule") {
+                    return grammarErrorOnNodeSkippedOn("noEmit", name, ts.Diagnostics.Identifier_expected_esModule_is_reserved_as_an_exported_marker_when_transforming_ECMAScript_modules);
                 }
-                else {
-                    if (ts.isBinaryExpression(emittedOperand)
-                        && emittedOperand.operatorToken.kind === binaryOperator) {
-                        // No need to parenthesize the right operand when the binary operator and
-                        // operand are the same and one of the following:
-                        //  x*(a*b)     => x*a*b
-                        //  x|(a|b)     => x|a|b
-                        //  x&(a&b)     => x&a&b
-                        //  x^(a^b)     => x^a^b
-                        if (operatorHasAssociativeProperty(binaryOperator)) {
-                            return false;
-                        }
-                        // No need to parenthesize the right operand when the binary operator
-                        // is plus (+) if both the left and right operands consist solely of either
-                        // literals of the same kind or binary plus (+) expressions for literals of
-                        // the same kind (recursively).
-                        //  "a"+(1+2)       => "a"+(1+2)
-                        //  "a"+("b"+"c")   => "a"+"b"+"c"
-                        if (binaryOperator === 39 /* PlusToken */) {
-                            var leftKind = leftOperand ? getLiteralKindOfBinaryPlusOperand(leftOperand) : 0 /* Unknown */;
-                            if (ts.isLiteralKind(leftKind) && leftKind === getLiteralKindOfBinaryPlusOperand(emittedOperand)) {
-                                return false;
-                            }
-                        }
+            }
+            else {
+                var elements = name.elements;
+                for (var _i = 0, elements_1 = elements; _i < elements_1.length; _i++) {
+                    var element = elements_1[_i];
+                    if (!ts.isOmittedExpression(element)) {
+                        return checkESModuleMarker(element.name);
                     }
-                    // No need to parenthesize the right operand when the operand is right
-                    // associative:
-                    //  x/(a**b)    -> x/a**b
-                    //  x**(a**b)   -> x**a**b
-                    //
-                    // Parentheses are needed for the right operand when the operand is left
-                    // associative:
-                    //  x/(a*b)     -> x/(a*b)
-                    //  x**(a/b)    -> x**(a/b)
-                    var operandAssociativity = ts.getExpressionAssociativity(emittedOperand);
-                    return operandAssociativity === 0 /* Left */;
                 }
+            }
+            return false;
         }
-    }
-    /**
-     * Determines whether a binary operator is mathematically associative.
-     *
-     * @param binaryOperator The binary operator.
-     */
-    function operatorHasAssociativeProperty(binaryOperator) {
-        // The following operators are associative in JavaScript:
-        //  (a*b)*c     -> a*(b*c)  -> a*b*c
-        //  (a|b)|c     -> a|(b|c)  -> a|b|c
-        //  (a&b)&c     -> a&(b&c)  -> a&b&c
-        //  (a^b)^c     -> a^(b^c)  -> a^b^c
-        //
-        // While addition is associative in mathematics, JavaScript's `+` is not
-        // guaranteed to be associative as it is overloaded with string concatenation.
-        return binaryOperator === 41 /* AsteriskToken */
-            || binaryOperator === 51 /* BarToken */
-            || binaryOperator === 50 /* AmpersandToken */
-            || binaryOperator === 52 /* CaretToken */;
-    }
-    /**
-     * This function determines whether an expression consists of a homogeneous set of
-     * literal expressions or binary plus expressions that all share the same literal kind.
-     * It is used to determine whether the right-hand operand of a binary plus expression can be
-     * emitted without parentheses.
-     */
-    function getLiteralKindOfBinaryPlusOperand(node) {
-        node = ts.skipPartiallyEmittedExpressions(node);
-        if (ts.isLiteralKind(node.kind)) {
-            return node.kind;
-        }
-        if (node.kind === 209 /* BinaryExpression */ && node.operatorToken.kind === 39 /* PlusToken */) {
-            if (node.cachedLiteralKind !== undefined) {
-                return node.cachedLiteralKind;
-            }
-            var leftKind = getLiteralKindOfBinaryPlusOperand(node.left);
-            var literalKind = ts.isLiteralKind(leftKind) && leftKind === getLiteralKindOfBinaryPlusOperand(node.right) ? leftKind :
-                0 /* Unknown */;
-            node.cachedLiteralKind = literalKind;
-            return literalKind;
-        }
-        return 0 /* Unknown */;
-    }
-    function parenthesizeForConditionalHead(condition) {
-        var conditionalPrecedence = ts.getOperatorPrecedence(210 /* ConditionalExpression */, 57 /* QuestionToken */);
-        var emittedCondition = ts.skipPartiallyEmittedExpressions(condition);
-        var conditionPrecedence = ts.getExpressionPrecedence(emittedCondition);
-        if (ts.compareValues(conditionPrecedence, conditionalPrecedence) !== 1 /* GreaterThan */) {
-            return ts.createParen(condition);
-        }
-        return condition;
-    }
-    ts.parenthesizeForConditionalHead = parenthesizeForConditionalHead;
-    function parenthesizeSubexpressionOfConditionalExpression(e) {
-        // per ES grammar both 'whenTrue' and 'whenFalse' parts of conditional expression are assignment expressions
-        // so in case when comma expression is introduced as a part of previous transformations
-        // if should be wrapped in parens since comma operator has the lowest precedence
-        var emittedExpression = ts.skipPartiallyEmittedExpressions(e);
-        return isCommaSequence(emittedExpression)
-            ? ts.createParen(e)
-            : e;
-    }
-    ts.parenthesizeSubexpressionOfConditionalExpression = parenthesizeSubexpressionOfConditionalExpression;
-    /**
-     *  [Per the spec](https://tc39.github.io/ecma262/#prod-ExportDeclaration), `export default` accepts _AssigmentExpression_ but
-     *  has a lookahead restriction for `function`, `async function`, and `class`.
-     *
-     * Basically, that means we need to parenthesize in the following cases:
-     *
-     * - BinaryExpression of CommaToken
-     * - CommaList (synthetic list of multiple comma expressions)
-     * - FunctionExpression
-     * - ClassExpression
-     */
-    function parenthesizeDefaultExpression(e) {
-        var check = ts.skipPartiallyEmittedExpressions(e);
-        var needsParens = isCommaSequence(check);
-        if (!needsParens) {
-            switch (getLeftmostExpression(check, /*stopAtCallExpression*/ false).kind) {
-                case 214 /* ClassExpression */:
-                case 201 /* FunctionExpression */:
-                    needsParens = true;
+        function checkGrammarNameInLetOrConstDeclarations(name) {
+            if (name.kind === 78 /* Identifier */) {
+                if (name.originalKeywordKind === 118 /* LetKeyword */) {
+                    return grammarErrorOnNode(name, ts.Diagnostics.let_is_not_allowed_to_be_used_as_a_name_in_let_or_const_declarations);
+                }
+            }
+            else {
+                var elements = name.elements;
+                for (var _i = 0, elements_2 = elements; _i < elements_2.length; _i++) {
+                    var element = elements_2[_i];
+                    if (!ts.isOmittedExpression(element)) {
+                        checkGrammarNameInLetOrConstDeclarations(element.name);
+                    }
+                }
             }
+            return false;
         }
-        return needsParens ? ts.createParen(e) : e;
-    }
-    ts.parenthesizeDefaultExpression = parenthesizeDefaultExpression;
-    /**
-     * Wraps an expression in parentheses if it is needed in order to use the expression
-     * as the expression of a NewExpression node.
-     *
-     * @param expression The Expression node.
-     */
-    function parenthesizeForNew(expression) {
-        var leftmostExpr = getLeftmostExpression(expression, /*stopAtCallExpressions*/ true);
-        switch (leftmostExpr.kind) {
-            case 196 /* CallExpression */:
-                return ts.createParen(expression);
-            case 197 /* NewExpression */:
-                return !leftmostExpr.arguments
-                    ? ts.createParen(expression)
-                    : expression;
+        function checkGrammarVariableDeclarationList(declarationList) {
+            var declarations = declarationList.declarations;
+            if (checkGrammarForDisallowedTrailingComma(declarationList.declarations)) {
+                return true;
+            }
+            if (!declarationList.declarations.length) {
+                return grammarErrorAtPos(declarationList, declarations.pos, declarations.end - declarations.pos, ts.Diagnostics.Variable_declaration_list_cannot_be_empty);
+            }
+            return false;
         }
-        return parenthesizeForAccess(expression);
-    }
-    ts.parenthesizeForNew = parenthesizeForNew;
-    /**
-     * Wraps an expression in parentheses if it is needed in order to use the expression for
-     * property or element access.
-     *
-     * @param expr The expression node.
-     */
-    function parenthesizeForAccess(expression) {
-        // isLeftHandSideExpression is almost the correct criterion for when it is not necessary
-        // to parenthesize the expression before a dot. The known exception is:
-        //
-        //    NewExpression:
-        //       new C.x        -> not the same as (new C).x
-        //
-        var emittedExpression = ts.skipPartiallyEmittedExpressions(expression);
-        if (ts.isLeftHandSideExpression(emittedExpression)
-            && (emittedExpression.kind !== 197 /* NewExpression */ || emittedExpression.arguments)) {
-            return expression;
+        function allowLetAndConstDeclarations(parent) {
+            switch (parent.kind) {
+                case 234 /* IfStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
+                case 243 /* WithStatement */:
+                case 237 /* ForStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                    return false;
+                case 245 /* LabeledStatement */:
+                    return allowLetAndConstDeclarations(parent.parent);
+            }
+            return true;
         }
-        return ts.setTextRange(ts.createParen(expression), expression);
-    }
-    ts.parenthesizeForAccess = parenthesizeForAccess;
-    function parenthesizePostfixOperand(operand) {
-        return ts.isLeftHandSideExpression(operand)
-            ? operand
-            : ts.setTextRange(ts.createParen(operand), operand);
-    }
-    ts.parenthesizePostfixOperand = parenthesizePostfixOperand;
-    function parenthesizePrefixOperand(operand) {
-        return ts.isUnaryExpression(operand)
-            ? operand
-            : ts.setTextRange(ts.createParen(operand), operand);
-    }
-    ts.parenthesizePrefixOperand = parenthesizePrefixOperand;
-    function parenthesizeListElements(elements) {
-        var result;
-        for (var i = 0; i < elements.length; i++) {
-            var element = parenthesizeExpressionForList(elements[i]);
-            if (result !== undefined || element !== elements[i]) {
-                if (result === undefined) {
-                    result = elements.slice(0, i);
+        function checkGrammarForDisallowedLetOrConstStatement(node) {
+            if (!allowLetAndConstDeclarations(node.parent)) {
+                if (ts.isLet(node.declarationList)) {
+                    return grammarErrorOnNode(node, ts.Diagnostics.let_declarations_can_only_be_declared_inside_a_block);
+                }
+                else if (ts.isVarConst(node.declarationList)) {
+                    return grammarErrorOnNode(node, ts.Diagnostics.const_declarations_can_only_be_declared_inside_a_block);
                 }
-                result.push(element);
             }
         }
-        if (result !== undefined) {
-            return ts.setTextRange(ts.createNodeArray(result, elements.hasTrailingComma), elements);
-        }
-        return elements;
-    }
-    ts.parenthesizeListElements = parenthesizeListElements;
-    function parenthesizeExpressionForList(expression) {
-        var emittedExpression = ts.skipPartiallyEmittedExpressions(expression);
-        var expressionPrecedence = ts.getExpressionPrecedence(emittedExpression);
-        var commaPrecedence = ts.getOperatorPrecedence(209 /* BinaryExpression */, 27 /* CommaToken */);
-        return expressionPrecedence > commaPrecedence
-            ? expression
-            : ts.setTextRange(ts.createParen(expression), expression);
-    }
-    ts.parenthesizeExpressionForList = parenthesizeExpressionForList;
-    function parenthesizeExpressionForExpressionStatement(expression) {
-        var emittedExpression = ts.skipPartiallyEmittedExpressions(expression);
-        if (ts.isCallExpression(emittedExpression)) {
-            var callee = emittedExpression.expression;
-            var kind = ts.skipPartiallyEmittedExpressions(callee).kind;
-            if (kind === 201 /* FunctionExpression */ || kind === 202 /* ArrowFunction */) {
-                var mutableCall = ts.getMutableClone(emittedExpression);
-                mutableCall.expression = ts.setTextRange(ts.createParen(callee), callee);
-                return recreateOuterExpressions(expression, mutableCall, 8 /* PartiallyEmittedExpressions */);
-            }
-        }
-        var leftmostExpressionKind = getLeftmostExpression(emittedExpression, /*stopAtCallExpressions*/ false).kind;
-        if (leftmostExpressionKind === 193 /* ObjectLiteralExpression */ || leftmostExpressionKind === 201 /* FunctionExpression */) {
-            return ts.setTextRange(ts.createParen(expression), expression);
-        }
-        return expression;
-    }
-    ts.parenthesizeExpressionForExpressionStatement = parenthesizeExpressionForExpressionStatement;
-    function parenthesizeConditionalTypeMember(member) {
-        return member.kind === 180 /* ConditionalType */ ? ts.createParenthesizedType(member) : member;
-    }
-    ts.parenthesizeConditionalTypeMember = parenthesizeConditionalTypeMember;
-    function parenthesizeElementTypeMember(member) {
-        switch (member.kind) {
-            case 178 /* UnionType */:
-            case 179 /* IntersectionType */:
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
-                return ts.createParenthesizedType(member);
-        }
-        return parenthesizeConditionalTypeMember(member);
-    }
-    ts.parenthesizeElementTypeMember = parenthesizeElementTypeMember;
-    function parenthesizeArrayTypeMember(member) {
-        switch (member.kind) {
-            case 172 /* TypeQuery */:
-            case 184 /* TypeOperator */:
-            case 181 /* InferType */:
-                return ts.createParenthesizedType(member);
-        }
-        return parenthesizeElementTypeMember(member);
-    }
-    ts.parenthesizeArrayTypeMember = parenthesizeArrayTypeMember;
-    function parenthesizeElementTypeMembers(members) {
-        return ts.createNodeArray(ts.sameMap(members, parenthesizeElementTypeMember));
-    }
-    ts.parenthesizeElementTypeMembers = parenthesizeElementTypeMembers;
-    function parenthesizeTypeParameters(typeParameters) {
-        if (ts.some(typeParameters)) {
-            var params = [];
-            for (var i = 0; i < typeParameters.length; ++i) {
-                var entry = typeParameters[i];
-                params.push(i === 0 && ts.isFunctionOrConstructorTypeNode(entry) && entry.typeParameters ?
-                    ts.createParenthesizedType(entry) :
-                    entry);
-            }
-            return ts.createNodeArray(params);
-        }
-    }
-    ts.parenthesizeTypeParameters = parenthesizeTypeParameters;
-    function getLeftmostExpression(node, stopAtCallExpressions) {
-        while (true) {
-            switch (node.kind) {
-                case 208 /* PostfixUnaryExpression */:
-                    node = node.operand;
-                    continue;
-                case 209 /* BinaryExpression */:
-                    node = node.left;
-                    continue;
-                case 210 /* ConditionalExpression */:
-                    node = node.condition;
-                    continue;
-                case 198 /* TaggedTemplateExpression */:
-                    node = node.tag;
-                    continue;
-                case 196 /* CallExpression */:
-                    if (stopAtCallExpressions) {
-                        return node;
+        function checkGrammarMetaProperty(node) {
+            var escapedText = node.name.escapedText;
+            switch (node.keywordToken) {
+                case 102 /* NewKeyword */:
+                    if (escapedText !== "target") {
+                        return grammarErrorOnNode(node.name, ts.Diagnostics._0_is_not_a_valid_meta_property_for_keyword_1_Did_you_mean_2, node.name.escapedText, ts.tokenToString(node.keywordToken), "target");
                     }
-                // falls through
-                case 217 /* AsExpression */:
-                case 195 /* ElementAccessExpression */:
-                case 194 /* PropertyAccessExpression */:
-                case 218 /* NonNullExpression */:
-                case 326 /* PartiallyEmittedExpression */:
-                    node = node.expression;
-                    continue;
+                    break;
+                case 99 /* ImportKeyword */:
+                    if (escapedText !== "meta") {
+                        return grammarErrorOnNode(node.name, ts.Diagnostics._0_is_not_a_valid_meta_property_for_keyword_1_Did_you_mean_2, node.name.escapedText, ts.tokenToString(node.keywordToken), "meta");
+                    }
+                    break;
             }
-            return node;
         }
-    }
-    ts.getLeftmostExpression = getLeftmostExpression;
-    function parenthesizeConciseBody(body) {
-        if (!ts.isBlock(body) && (isCommaSequence(body) || getLeftmostExpression(body, /*stopAtCallExpressions*/ false).kind === 193 /* ObjectLiteralExpression */)) {
-            return ts.setTextRange(ts.createParen(body), body);
+        function hasParseDiagnostics(sourceFile) {
+            return sourceFile.parseDiagnostics.length > 0;
         }
-        return body;
-    }
-    ts.parenthesizeConciseBody = parenthesizeConciseBody;
-    function isCommaSequence(node) {
-        return node.kind === 209 /* BinaryExpression */ && node.operatorToken.kind === 27 /* CommaToken */ ||
-            node.kind === 327 /* CommaListExpression */;
-    }
-    ts.isCommaSequence = isCommaSequence;
-    var OuterExpressionKinds;
-    (function (OuterExpressionKinds) {
-        OuterExpressionKinds[OuterExpressionKinds["Parentheses"] = 1] = "Parentheses";
-        OuterExpressionKinds[OuterExpressionKinds["TypeAssertions"] = 2] = "TypeAssertions";
-        OuterExpressionKinds[OuterExpressionKinds["NonNullAssertions"] = 4] = "NonNullAssertions";
-        OuterExpressionKinds[OuterExpressionKinds["PartiallyEmittedExpressions"] = 8] = "PartiallyEmittedExpressions";
-        OuterExpressionKinds[OuterExpressionKinds["Assertions"] = 6] = "Assertions";
-        OuterExpressionKinds[OuterExpressionKinds["All"] = 15] = "All";
-    })(OuterExpressionKinds = ts.OuterExpressionKinds || (ts.OuterExpressionKinds = {}));
-    function isOuterExpression(node, kinds) {
-        if (kinds === void 0) { kinds = 15 /* All */; }
-        switch (node.kind) {
-            case 200 /* ParenthesizedExpression */:
-                return (kinds & 1 /* Parentheses */) !== 0;
-            case 199 /* TypeAssertionExpression */:
-            case 217 /* AsExpression */:
-                return (kinds & 2 /* TypeAssertions */) !== 0;
-            case 218 /* NonNullExpression */:
-                return (kinds & 4 /* NonNullAssertions */) !== 0;
-            case 326 /* PartiallyEmittedExpression */:
-                return (kinds & 8 /* PartiallyEmittedExpressions */) !== 0;
+        function grammarErrorOnFirstToken(node, message, arg0, arg1, arg2) {
+            var sourceFile = ts.getSourceFileOfNode(node);
+            if (!hasParseDiagnostics(sourceFile)) {
+                var span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
+                diagnostics.add(ts.createFileDiagnostic(sourceFile, span.start, span.length, message, arg0, arg1, arg2));
+                return true;
+            }
+            return false;
         }
-        return false;
-    }
-    ts.isOuterExpression = isOuterExpression;
-    function skipOuterExpressions(node, kinds) {
-        if (kinds === void 0) { kinds = 15 /* All */; }
-        while (isOuterExpression(node, kinds)) {
-            node = node.expression;
+        function grammarErrorAtPos(nodeForSourceFile, start, length, message, arg0, arg1, arg2) {
+            var sourceFile = ts.getSourceFileOfNode(nodeForSourceFile);
+            if (!hasParseDiagnostics(sourceFile)) {
+                diagnostics.add(ts.createFileDiagnostic(sourceFile, start, length, message, arg0, arg1, arg2));
+                return true;
+            }
+            return false;
         }
-        return node;
-    }
-    ts.skipOuterExpressions = skipOuterExpressions;
-    function skipAssertions(node) {
-        return skipOuterExpressions(node, 6 /* Assertions */);
-    }
-    ts.skipAssertions = skipAssertions;
-    function updateOuterExpression(outerExpression, expression) {
-        switch (outerExpression.kind) {
-            case 200 /* ParenthesizedExpression */: return ts.updateParen(outerExpression, expression);
-            case 199 /* TypeAssertionExpression */: return ts.updateTypeAssertion(outerExpression, outerExpression.type, expression);
-            case 217 /* AsExpression */: return ts.updateAsExpression(outerExpression, expression, outerExpression.type);
-            case 218 /* NonNullExpression */: return ts.updateNonNullExpression(outerExpression, expression);
-            case 326 /* PartiallyEmittedExpression */: return ts.updatePartiallyEmittedExpression(outerExpression, expression);
+        function grammarErrorOnNodeSkippedOn(key, node, message, arg0, arg1, arg2) {
+            var sourceFile = ts.getSourceFileOfNode(node);
+            if (!hasParseDiagnostics(sourceFile)) {
+                errorSkippedOn(key, node, message, arg0, arg1, arg2);
+                return true;
+            }
+            return false;
         }
-    }
-    /**
-     * Determines whether a node is a parenthesized expression that can be ignored when recreating outer expressions.
-     *
-     * A parenthesized expression can be ignored when all of the following are true:
-     *
-     * - It's `pos` and `end` are not -1
-     * - It does not have a custom source map range
-     * - It does not have a custom comment range
-     * - It does not have synthetic leading or trailing comments
-     *
-     * If an outermost parenthesized expression is ignored, but the containing expression requires a parentheses around
-     * the expression to maintain precedence, a new parenthesized expression should be created automatically when
-     * the containing expression is created/updated.
-     */
-    function isIgnorableParen(node) {
-        return node.kind === 200 /* ParenthesizedExpression */
-            && ts.nodeIsSynthesized(node)
-            && ts.nodeIsSynthesized(ts.getSourceMapRange(node))
-            && ts.nodeIsSynthesized(ts.getCommentRange(node))
-            && !ts.some(ts.getSyntheticLeadingComments(node))
-            && !ts.some(ts.getSyntheticTrailingComments(node));
-    }
-    function recreateOuterExpressions(outerExpression, innerExpression, kinds) {
-        if (kinds === void 0) { kinds = 15 /* All */; }
-        if (outerExpression && isOuterExpression(outerExpression, kinds) && !isIgnorableParen(outerExpression)) {
-            return updateOuterExpression(outerExpression, recreateOuterExpressions(outerExpression.expression, innerExpression));
+        function grammarErrorOnNode(node, message, arg0, arg1, arg2) {
+            var sourceFile = ts.getSourceFileOfNode(node);
+            if (!hasParseDiagnostics(sourceFile)) {
+                diagnostics.add(ts.createDiagnosticForNode(node, message, arg0, arg1, arg2));
+                return true;
+            }
+            return false;
         }
-        return innerExpression;
-    }
-    ts.recreateOuterExpressions = recreateOuterExpressions;
-    function startOnNewLine(node) {
-        return ts.setStartsOnNewLine(node, /*newLine*/ true);
-    }
-    ts.startOnNewLine = startOnNewLine;
-    function getExternalHelpersModuleName(node) {
-        var parseNode = ts.getOriginalNode(node, ts.isSourceFile);
-        var emitNode = parseNode && parseNode.emitNode;
-        return emitNode && emitNode.externalHelpersModuleName;
-    }
-    ts.getExternalHelpersModuleName = getExternalHelpersModuleName;
-    function hasRecordedExternalHelpers(sourceFile) {
-        var parseNode = ts.getOriginalNode(sourceFile, ts.isSourceFile);
-        var emitNode = parseNode && parseNode.emitNode;
-        return !!emitNode && (!!emitNode.externalHelpersModuleName || !!emitNode.externalHelpers);
-    }
-    ts.hasRecordedExternalHelpers = hasRecordedExternalHelpers;
-    function createExternalHelpersImportDeclarationIfNeeded(sourceFile, compilerOptions, hasExportStarsToExportValues, hasImportStar, hasImportDefault) {
-        if (compilerOptions.importHelpers && ts.isEffectiveExternalModule(sourceFile, compilerOptions)) {
-            var namedBindings = void 0;
-            var moduleKind = ts.getEmitModuleKind(compilerOptions);
-            if (moduleKind >= ts.ModuleKind.ES2015 && moduleKind <= ts.ModuleKind.ESNext) {
-                // use named imports
-                var helpers = ts.getEmitHelpers(sourceFile);
-                if (helpers) {
-                    var helperNames = [];
-                    for (var _i = 0, helpers_2 = helpers; _i < helpers_2.length; _i++) {
-                        var helper = helpers_2[_i];
-                        if (!helper.scoped) {
-                            var importName = helper.importName;
-                            if (importName) {
-                                ts.pushIfUnique(helperNames, importName);
-                            }
-                        }
-                    }
-                    if (ts.some(helperNames)) {
-                        helperNames.sort(ts.compareStringsCaseSensitive);
-                        // Alias the imports if the names are used somewhere in the file.
-                        // NOTE: We don't need to care about global import collisions as this is a module.
-                        namedBindings = ts.createNamedImports(ts.map(helperNames, function (name) { return ts.isFileLevelUniqueName(sourceFile, name)
-                            ? ts.createImportSpecifier(/*propertyName*/ undefined, ts.createIdentifier(name))
-                            : ts.createImportSpecifier(ts.createIdentifier(name), getUnscopedHelperName(name)); }));
-                        var parseNode = ts.getOriginalNode(sourceFile, ts.isSourceFile);
-                        var emitNode = ts.getOrCreateEmitNode(parseNode);
-                        emitNode.externalHelpers = true;
-                    }
+        function checkGrammarConstructorTypeParameters(node) {
+            var jsdocTypeParameters = ts.isInJSFile(node) ? ts.getJSDocTypeParameterDeclarations(node) : undefined;
+            var range = node.typeParameters || jsdocTypeParameters && ts.firstOrUndefined(jsdocTypeParameters);
+            if (range) {
+                var pos = range.pos === range.end ? range.pos : ts.skipTrivia(ts.getSourceFileOfNode(node).text, range.pos);
+                return grammarErrorAtPos(node, pos, range.end - pos, ts.Diagnostics.Type_parameters_cannot_appear_on_a_constructor_declaration);
+            }
+        }
+        function checkGrammarConstructorTypeAnnotation(node) {
+            var type = ts.getEffectiveReturnTypeNode(node);
+            if (type) {
+                return grammarErrorOnNode(type, ts.Diagnostics.Type_annotation_cannot_appear_on_a_constructor_declaration);
+            }
+        }
+        function checkGrammarProperty(node) {
+            if (ts.isClassLike(node.parent)) {
+                if (ts.isStringLiteral(node.name) && node.name.text === "constructor") {
+                    return grammarErrorOnNode(node.name, ts.Diagnostics.Classes_may_not_have_a_field_named_constructor);
+                }
+                if (checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_class_property_declaration_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type)) {
+                    return true;
+                }
+                if (languageVersion < 2 /* ES2015 */ && ts.isPrivateIdentifier(node.name)) {
+                    return grammarErrorOnNode(node.name, ts.Diagnostics.Private_identifiers_are_only_available_when_targeting_ECMAScript_2015_and_higher);
                 }
             }
-            else {
-                // use a namespace import
-                var externalHelpersModuleName = getOrCreateExternalHelpersModuleNameIfNeeded(sourceFile, compilerOptions, hasExportStarsToExportValues, hasImportStar || hasImportDefault);
-                if (externalHelpersModuleName) {
-                    namedBindings = ts.createNamespaceImport(externalHelpersModuleName);
+            else if (node.parent.kind === 253 /* InterfaceDeclaration */) {
+                if (checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_an_interface_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type)) {
+                    return true;
+                }
+                if (node.initializer) {
+                    return grammarErrorOnNode(node.initializer, ts.Diagnostics.An_interface_property_cannot_have_an_initializer);
                 }
             }
-            if (namedBindings) {
-                var externalHelpersImportDeclaration = ts.createImportDeclaration(
-                /*decorators*/ undefined, 
-                /*modifiers*/ undefined, ts.createImportClause(/*name*/ undefined, namedBindings), ts.createLiteral(ts.externalHelpersModuleNameText));
-                ts.addEmitFlags(externalHelpersImportDeclaration, 67108864 /* NeverApplyImportHelper */);
-                return externalHelpersImportDeclaration;
+            else if (node.parent.kind === 177 /* TypeLiteral */) {
+                if (checkGrammarForInvalidDynamicName(node.name, ts.Diagnostics.A_computed_property_name_in_a_type_literal_must_refer_to_an_expression_whose_type_is_a_literal_type_or_a_unique_symbol_type)) {
+                    return true;
+                }
+                if (node.initializer) {
+                    return grammarErrorOnNode(node.initializer, ts.Diagnostics.A_type_literal_property_cannot_have_an_initializer);
+                }
+            }
+            if (node.flags & 8388608 /* Ambient */) {
+                checkAmbientInitializer(node);
+            }
+            if (ts.isPropertyDeclaration(node) && node.exclamationToken && (!ts.isClassLike(node.parent) || !node.type || node.initializer ||
+                node.flags & 8388608 /* Ambient */ || ts.hasSyntacticModifier(node, 32 /* Static */ | 128 /* Abstract */))) {
+                var message = node.initializer
+                    ? ts.Diagnostics.Declarations_with_initializers_cannot_also_have_definite_assignment_assertions
+                    : !node.type
+                        ? ts.Diagnostics.Declarations_with_definite_assignment_assertions_must_also_have_type_annotations
+                        : ts.Diagnostics.A_definite_assignment_assertion_is_not_permitted_in_this_context;
+                return grammarErrorOnNode(node.exclamationToken, message);
             }
         }
-    }
-    ts.createExternalHelpersImportDeclarationIfNeeded = createExternalHelpersImportDeclarationIfNeeded;
-    function getOrCreateExternalHelpersModuleNameIfNeeded(node, compilerOptions, hasExportStarsToExportValues, hasImportStarOrImportDefault) {
-        if (compilerOptions.importHelpers && ts.isEffectiveExternalModule(node, compilerOptions)) {
-            var externalHelpersModuleName = getExternalHelpersModuleName(node);
-            if (externalHelpersModuleName) {
-                return externalHelpersModuleName;
+        function checkGrammarTopLevelElementForRequiredDeclareModifier(node) {
+            // A declare modifier is required for any top level .d.ts declaration except export=, export default, export as namespace
+            // interfaces and imports categories:
+            //
+            //  DeclarationElement:
+            //     ExportAssignment
+            //     export_opt   InterfaceDeclaration
+            //     export_opt   TypeAliasDeclaration
+            //     export_opt   ImportDeclaration
+            //     export_opt   ExternalImportDeclaration
+            //     export_opt   AmbientDeclaration
+            //
+            // TODO: The spec needs to be amended to reflect this grammar.
+            if (node.kind === 253 /* InterfaceDeclaration */ ||
+                node.kind === 254 /* TypeAliasDeclaration */ ||
+                node.kind === 261 /* ImportDeclaration */ ||
+                node.kind === 260 /* ImportEqualsDeclaration */ ||
+                node.kind === 267 /* ExportDeclaration */ ||
+                node.kind === 266 /* ExportAssignment */ ||
+                node.kind === 259 /* NamespaceExportDeclaration */ ||
+                ts.hasSyntacticModifier(node, 2 /* Ambient */ | 1 /* Export */ | 512 /* Default */)) {
+                return false;
             }
-            var moduleKind = ts.getEmitModuleKind(compilerOptions);
-            var create = (hasExportStarsToExportValues || (compilerOptions.esModuleInterop && hasImportStarOrImportDefault))
-                && moduleKind !== ts.ModuleKind.System
-                && moduleKind < ts.ModuleKind.ES2015;
-            if (!create) {
-                var helpers = ts.getEmitHelpers(node);
-                if (helpers) {
-                    for (var _i = 0, helpers_3 = helpers; _i < helpers_3.length; _i++) {
-                        var helper = helpers_3[_i];
-                        if (!helper.scoped) {
-                            create = true;
-                            break;
-                        }
+            return grammarErrorOnFirstToken(node, ts.Diagnostics.Top_level_declarations_in_d_ts_files_must_start_with_either_a_declare_or_export_modifier);
+        }
+        function checkGrammarTopLevelElementsForRequiredDeclareModifier(file) {
+            for (var _i = 0, _a = file.statements; _i < _a.length; _i++) {
+                var decl = _a[_i];
+                if (ts.isDeclaration(decl) || decl.kind === 232 /* VariableStatement */) {
+                    if (checkGrammarTopLevelElementForRequiredDeclareModifier(decl)) {
+                        return true;
                     }
                 }
             }
-            if (create) {
-                var parseNode = ts.getOriginalNode(node, ts.isSourceFile);
-                var emitNode = ts.getOrCreateEmitNode(parseNode);
-                return emitNode.externalHelpersModuleName || (emitNode.externalHelpersModuleName = ts.createUniqueName(ts.externalHelpersModuleNameText));
-            }
-        }
-    }
-    ts.getOrCreateExternalHelpersModuleNameIfNeeded = getOrCreateExternalHelpersModuleNameIfNeeded;
-    /**
-     * Get the name of that target module from an import or export declaration
-     */
-    function getLocalNameForExternalImport(node, sourceFile) {
-        var namespaceDeclaration = ts.getNamespaceDeclarationNode(node);
-        if (namespaceDeclaration && !ts.isDefaultImport(node)) {
-            var name = namespaceDeclaration.name;
-            return ts.isGeneratedIdentifier(name) ? name : ts.createIdentifier(ts.getSourceTextOfNodeFromSourceFile(sourceFile, name) || ts.idText(name));
-        }
-        if (node.kind === 254 /* ImportDeclaration */ && node.importClause) {
-            return ts.getGeneratedNameForNode(node);
-        }
-        if (node.kind === 260 /* ExportDeclaration */ && node.moduleSpecifier) {
-            return ts.getGeneratedNameForNode(node);
-        }
-        return undefined;
-    }
-    ts.getLocalNameForExternalImport = getLocalNameForExternalImport;
-    /**
-     * Get the name of a target module from an import/export declaration as should be written in the emitted output.
-     * The emitted output name can be different from the input if:
-     *  1. The module has a /// <amd-module name="<new name>" />
-     *  2. --out or --outFile is used, making the name relative to the rootDir
-     *  3- The containing SourceFile has an entry in renamedDependencies for the import as requested by some module loaders (e.g. System).
-     * Otherwise, a new StringLiteral node representing the module name will be returned.
-     */
-    function getExternalModuleNameLiteral(importNode, sourceFile, host, resolver, compilerOptions) {
-        var moduleName = ts.getExternalModuleName(importNode); // TODO: GH#18217
-        if (moduleName.kind === 10 /* StringLiteral */) {
-            return tryGetModuleNameFromDeclaration(importNode, host, resolver, compilerOptions)
-                || tryRenameExternalModule(moduleName, sourceFile)
-                || ts.getSynthesizedClone(moduleName);
-        }
-        return undefined;
-    }
-    ts.getExternalModuleNameLiteral = getExternalModuleNameLiteral;
-    /**
-     * Some bundlers (SystemJS builder) sometimes want to rename dependencies.
-     * Here we check if alternative name was provided for a given moduleName and return it if possible.
-     */
-    function tryRenameExternalModule(moduleName, sourceFile) {
-        var rename = sourceFile.renamedDependencies && sourceFile.renamedDependencies.get(moduleName.text);
-        return rename && ts.createLiteral(rename);
-    }
-    /**
-     * Get the name of a module as should be written in the emitted output.
-     * The emitted output name can be different from the input if:
-     *  1. The module has a /// <amd-module name="<new name>" />
-     *  2. --out or --outFile is used, making the name relative to the rootDir
-     * Otherwise, a new StringLiteral node representing the module name will be returned.
-     */
-    function tryGetModuleNameFromFile(file, host, options) {
-        if (!file) {
-            return undefined;
+            return false;
         }
-        if (file.moduleName) {
-            return ts.createLiteral(file.moduleName);
+        function checkGrammarSourceFile(node) {
+            return !!(node.flags & 8388608 /* Ambient */) && checkGrammarTopLevelElementsForRequiredDeclareModifier(node);
         }
-        if (!file.isDeclarationFile && (options.out || options.outFile)) {
-            return ts.createLiteral(ts.getExternalModuleNameFromPath(host, file.fileName));
+        function checkGrammarStatementInAmbientContext(node) {
+            if (node.flags & 8388608 /* Ambient */) {
+                // Find containing block which is either Block, ModuleBlock, SourceFile
+                var links = getNodeLinks(node);
+                if (!links.hasReportedStatementInAmbientContext && (ts.isFunctionLike(node.parent) || ts.isAccessor(node.parent))) {
+                    return getNodeLinks(node).hasReportedStatementInAmbientContext = grammarErrorOnFirstToken(node, ts.Diagnostics.An_implementation_cannot_be_declared_in_ambient_contexts);
+                }
+                // We are either parented by another statement, or some sort of block.
+                // If we're in a block, we only want to really report an error once
+                // to prevent noisiness.  So use a bit on the block to indicate if
+                // this has already been reported, and don't report if it has.
+                //
+                if (node.parent.kind === 230 /* Block */ || node.parent.kind === 257 /* ModuleBlock */ || node.parent.kind === 297 /* SourceFile */) {
+                    var links_2 = getNodeLinks(node.parent);
+                    // Check if the containing block ever report this error
+                    if (!links_2.hasReportedStatementInAmbientContext) {
+                        return links_2.hasReportedStatementInAmbientContext = grammarErrorOnFirstToken(node, ts.Diagnostics.Statements_are_not_allowed_in_ambient_contexts);
+                    }
+                }
+                else {
+                    // We must be parented by a statement.  If so, there's no need
+                    // to report the error as our parent will have already done it.
+                    // Debug.assert(isStatement(node.parent));
+                }
+            }
+            return false;
         }
-        return undefined;
-    }
-    ts.tryGetModuleNameFromFile = tryGetModuleNameFromFile;
-    function tryGetModuleNameFromDeclaration(declaration, host, resolver, compilerOptions) {
-        return tryGetModuleNameFromFile(resolver.getExternalModuleFileFromDeclaration(declaration), host, compilerOptions);
-    }
-    /**
-     * Gets the initializer of an BindingOrAssignmentElement.
-     */
-    function getInitializerOfBindingOrAssignmentElement(bindingElement) {
-        if (ts.isDeclarationBindingElement(bindingElement)) {
-            // `1` in `let { a = 1 } = ...`
-            // `1` in `let { a: b = 1 } = ...`
-            // `1` in `let { a: {b} = 1 } = ...`
-            // `1` in `let { a: [b] = 1 } = ...`
-            // `1` in `let [a = 1] = ...`
-            // `1` in `let [{a} = 1] = ...`
-            // `1` in `let [[a] = 1] = ...`
-            return bindingElement.initializer;
+        function checkGrammarNumericLiteral(node) {
+            // Grammar checking
+            if (node.numericLiteralFlags & 32 /* Octal */) {
+                var diagnosticMessage = void 0;
+                if (languageVersion >= 1 /* ES5 */) {
+                    diagnosticMessage = ts.Diagnostics.Octal_literals_are_not_available_when_targeting_ECMAScript_5_and_higher_Use_the_syntax_0;
+                }
+                else if (ts.isChildOfNodeWithKind(node, 191 /* LiteralType */)) {
+                    diagnosticMessage = ts.Diagnostics.Octal_literal_types_must_use_ES2015_syntax_Use_the_syntax_0;
+                }
+                else if (ts.isChildOfNodeWithKind(node, 291 /* EnumMember */)) {
+                    diagnosticMessage = ts.Diagnostics.Octal_literals_are_not_allowed_in_enums_members_initializer_Use_the_syntax_0;
+                }
+                if (diagnosticMessage) {
+                    var withMinus = ts.isPrefixUnaryExpression(node.parent) && node.parent.operator === 40 /* MinusToken */;
+                    var literal = (withMinus ? "-" : "") + "0o" + node.text;
+                    return grammarErrorOnNode(withMinus ? node.parent : node, diagnosticMessage, literal);
+                }
+            }
+            // Realism (size) checking
+            checkNumericLiteralValueSize(node);
+            return false;
         }
-        if (ts.isPropertyAssignment(bindingElement)) {
-            // `1` in `({ a: b = 1 } = ...)`
-            // `1` in `({ a: {b} = 1 } = ...)`
-            // `1` in `({ a: [b] = 1 } = ...)`
-            var initializer = bindingElement.initializer;
-            return ts.isAssignmentExpression(initializer, /*excludeCompoundAssignment*/ true)
-                ? initializer.right
-                : undefined;
+        function checkNumericLiteralValueSize(node) {
+            // Scientific notation (e.g. 2e54 and 1e00000000010) can't be converted to bigint
+            // Literals with 15 or fewer characters aren't long enough to reach past 2^53 - 1
+            // Fractional numbers (e.g. 9000000000000000.001) are inherently imprecise anyway
+            if (node.numericLiteralFlags & 16 /* Scientific */ || node.text.length <= 15 || node.text.indexOf(".") !== -1) {
+                return;
+            }
+            // We can't rely on the runtime to accurately store and compare extremely large numeric values
+            // Even for internal use, we use getTextOfNode: https://github.com/microsoft/TypeScript/issues/33298
+            // Thus, if the runtime claims a too-large number is lower than Number.MAX_SAFE_INTEGER,
+            // it's likely addition operations on it will fail too
+            var apparentValue = +ts.getTextOfNode(node);
+            if (apparentValue <= Math.pow(2, 53) - 1 && apparentValue + 1 > apparentValue) {
+                return;
+            }
+            addErrorOrSuggestion(/*isError*/ false, ts.createDiagnosticForNode(node, ts.Diagnostics.Numeric_literals_with_absolute_values_equal_to_2_53_or_greater_are_too_large_to_be_represented_accurately_as_integers));
         }
-        if (ts.isShorthandPropertyAssignment(bindingElement)) {
-            // `1` in `({ a = 1 } = ...)`
-            return bindingElement.objectAssignmentInitializer;
+        function checkGrammarBigIntLiteral(node) {
+            var literalType = ts.isLiteralTypeNode(node.parent) ||
+                ts.isPrefixUnaryExpression(node.parent) && ts.isLiteralTypeNode(node.parent.parent);
+            if (!literalType) {
+                if (languageVersion < 7 /* ES2020 */) {
+                    if (grammarErrorOnNode(node, ts.Diagnostics.BigInt_literals_are_not_available_when_targeting_lower_than_ES2020)) {
+                        return true;
+                    }
+                }
+            }
+            return false;
         }
-        if (ts.isAssignmentExpression(bindingElement, /*excludeCompoundAssignment*/ true)) {
-            // `1` in `[a = 1] = ...`
-            // `1` in `[{a} = 1] = ...`
-            // `1` in `[[a] = 1] = ...`
-            return bindingElement.right;
+        function grammarErrorAfterFirstToken(node, message, arg0, arg1, arg2) {
+            var sourceFile = ts.getSourceFileOfNode(node);
+            if (!hasParseDiagnostics(sourceFile)) {
+                var span = ts.getSpanOfTokenAtPosition(sourceFile, node.pos);
+                diagnostics.add(ts.createFileDiagnostic(sourceFile, ts.textSpanEnd(span), /*length*/ 0, message, arg0, arg1, arg2));
+                return true;
+            }
+            return false;
         }
-        if (ts.isSpreadElement(bindingElement)) {
-            // Recovery consistent with existing emit.
-            return getInitializerOfBindingOrAssignmentElement(bindingElement.expression);
+        function getAmbientModules() {
+            if (!ambientModulesCache) {
+                ambientModulesCache = [];
+                globals.forEach(function (global, sym) {
+                    // No need to `unescapeLeadingUnderscores`, an escaped symbol is never an ambient module.
+                    if (ambientModuleSymbolRegex.test(sym)) {
+                        ambientModulesCache.push(global);
+                    }
+                });
+            }
+            return ambientModulesCache;
         }
-    }
-    ts.getInitializerOfBindingOrAssignmentElement = getInitializerOfBindingOrAssignmentElement;
-    /**
-     * Gets the name of an BindingOrAssignmentElement.
-     */
-    function getTargetOfBindingOrAssignmentElement(bindingElement) {
-        if (ts.isDeclarationBindingElement(bindingElement)) {
-            // `a` in `let { a } = ...`
-            // `a` in `let { a = 1 } = ...`
-            // `b` in `let { a: b } = ...`
-            // `b` in `let { a: b = 1 } = ...`
-            // `a` in `let { ...a } = ...`
-            // `{b}` in `let { a: {b} } = ...`
-            // `{b}` in `let { a: {b} = 1 } = ...`
-            // `[b]` in `let { a: [b] } = ...`
-            // `[b]` in `let { a: [b] = 1 } = ...`
-            // `a` in `let [a] = ...`
-            // `a` in `let [a = 1] = ...`
-            // `a` in `let [...a] = ...`
-            // `{a}` in `let [{a}] = ...`
-            // `{a}` in `let [{a} = 1] = ...`
-            // `[a]` in `let [[a]] = ...`
-            // `[a]` in `let [[a] = 1] = ...`
-            return bindingElement.name;
+        function checkGrammarImportClause(node) {
+            if (node.isTypeOnly && node.name && node.namedBindings) {
+                return grammarErrorOnNode(node, ts.Diagnostics.A_type_only_import_can_specify_a_default_import_or_named_bindings_but_not_both);
+            }
+            return false;
         }
-        if (ts.isObjectLiteralElementLike(bindingElement)) {
-            switch (bindingElement.kind) {
-                case 281 /* PropertyAssignment */:
-                    // `b` in `({ a: b } = ...)`
-                    // `b` in `({ a: b = 1 } = ...)`
-                    // `{b}` in `({ a: {b} } = ...)`
-                    // `{b}` in `({ a: {b} = 1 } = ...)`
-                    // `[b]` in `({ a: [b] } = ...)`
-                    // `[b]` in `({ a: [b] = 1 } = ...)`
-                    // `b.c` in `({ a: b.c } = ...)`
-                    // `b.c` in `({ a: b.c = 1 } = ...)`
-                    // `b[0]` in `({ a: b[0] } = ...)`
-                    // `b[0]` in `({ a: b[0] = 1 } = ...)`
-                    return getTargetOfBindingOrAssignmentElement(bindingElement.initializer);
-                case 282 /* ShorthandPropertyAssignment */:
-                    // `a` in `({ a } = ...)`
-                    // `a` in `({ a = 1 } = ...)`
-                    return bindingElement.name;
-                case 283 /* SpreadAssignment */:
-                    // `a` in `({ ...a } = ...)`
-                    return getTargetOfBindingOrAssignmentElement(bindingElement.expression);
+        function checkGrammarImportCallExpression(node) {
+            if (moduleKind === ts.ModuleKind.ES2015) {
+                return grammarErrorOnNode(node, ts.Diagnostics.Dynamic_imports_are_only_supported_when_the_module_flag_is_set_to_es2020_esnext_commonjs_amd_system_or_umd);
             }
-            // no target
-            return undefined;
+            if (node.typeArguments) {
+                return grammarErrorOnNode(node, ts.Diagnostics.Dynamic_import_cannot_have_type_arguments);
+            }
+            var nodeArguments = node.arguments;
+            if (nodeArguments.length !== 1) {
+                return grammarErrorOnNode(node, ts.Diagnostics.Dynamic_import_must_have_one_specifier_as_an_argument);
+            }
+            checkGrammarForDisallowedTrailingComma(nodeArguments);
+            // see: parseArgumentOrArrayLiteralElement...we use this function which parse arguments of callExpression to parse specifier for dynamic import.
+            // parseArgumentOrArrayLiteralElement allows spread element to be in an argument list which is not allowed as specifier in dynamic import.
+            if (ts.isSpreadElement(nodeArguments[0])) {
+                return grammarErrorOnNode(nodeArguments[0], ts.Diagnostics.Specifier_of_dynamic_import_cannot_be_spread_element);
+            }
+            return false;
         }
-        if (ts.isAssignmentExpression(bindingElement, /*excludeCompoundAssignment*/ true)) {
-            // `a` in `[a = 1] = ...`
-            // `{a}` in `[{a} = 1] = ...`
-            // `[a]` in `[[a] = 1] = ...`
-            // `a.b` in `[a.b = 1] = ...`
-            // `a[0]` in `[a[0] = 1] = ...`
-            return getTargetOfBindingOrAssignmentElement(bindingElement.left);
+        function findMatchingTypeReferenceOrTypeAliasReference(source, unionTarget) {
+            var sourceObjectFlags = ts.getObjectFlags(source);
+            if (sourceObjectFlags & (4 /* Reference */ | 16 /* Anonymous */) && unionTarget.flags & 1048576 /* Union */) {
+                return ts.find(unionTarget.types, function (target) {
+                    if (target.flags & 524288 /* Object */) {
+                        var overlapObjFlags = sourceObjectFlags & ts.getObjectFlags(target);
+                        if (overlapObjFlags & 4 /* Reference */) {
+                            return source.target === target.target;
+                        }
+                        if (overlapObjFlags & 16 /* Anonymous */) {
+                            return !!source.aliasSymbol && source.aliasSymbol === target.aliasSymbol;
+                        }
+                    }
+                    return false;
+                });
+            }
         }
-        if (ts.isSpreadElement(bindingElement)) {
-            // `a` in `[...a] = ...`
-            return getTargetOfBindingOrAssignmentElement(bindingElement.expression);
+        function findBestTypeForObjectLiteral(source, unionTarget) {
+            if (ts.getObjectFlags(source) & 128 /* ObjectLiteral */ && forEachType(unionTarget, isArrayLikeType)) {
+                return ts.find(unionTarget.types, function (t) { return !isArrayLikeType(t); });
+            }
         }
-        // `a` in `[a] = ...`
-        // `{a}` in `[{a}] = ...`
-        // `[a]` in `[[a]] = ...`
-        // `a.b` in `[a.b] = ...`
-        // `a[0]` in `[a[0]] = ...`
-        return bindingElement;
-    }
-    ts.getTargetOfBindingOrAssignmentElement = getTargetOfBindingOrAssignmentElement;
-    /**
-     * Determines whether an BindingOrAssignmentElement is a rest element.
-     */
-    function getRestIndicatorOfBindingOrAssignmentElement(bindingElement) {
-        switch (bindingElement.kind) {
-            case 156 /* Parameter */:
-            case 191 /* BindingElement */:
-                // `...` in `let [...a] = ...`
-                return bindingElement.dotDotDotToken;
-            case 213 /* SpreadElement */:
-            case 283 /* SpreadAssignment */:
-                // `...` in `[...a] = ...`
-                return bindingElement;
+        function findBestTypeForInvokable(source, unionTarget) {
+            var signatureKind = 0 /* Call */;
+            var hasSignatures = getSignaturesOfType(source, signatureKind).length > 0 ||
+                (signatureKind = 1 /* Construct */, getSignaturesOfType(source, signatureKind).length > 0);
+            if (hasSignatures) {
+                return ts.find(unionTarget.types, function (t) { return getSignaturesOfType(t, signatureKind).length > 0; });
+            }
         }
-        return undefined;
-    }
-    ts.getRestIndicatorOfBindingOrAssignmentElement = getRestIndicatorOfBindingOrAssignmentElement;
-    /**
-     * Gets the property name of a BindingOrAssignmentElement
-     */
-    function getPropertyNameOfBindingOrAssignmentElement(bindingElement) {
-        var propertyName = tryGetPropertyNameOfBindingOrAssignmentElement(bindingElement);
-        ts.Debug.assert(!!propertyName || ts.isSpreadAssignment(bindingElement), "Invalid property name for binding element.");
-        return propertyName;
-    }
-    ts.getPropertyNameOfBindingOrAssignmentElement = getPropertyNameOfBindingOrAssignmentElement;
-    function tryGetPropertyNameOfBindingOrAssignmentElement(bindingElement) {
-        switch (bindingElement.kind) {
-            case 191 /* BindingElement */:
-                // `a` in `let { a: b } = ...`
-                // `[a]` in `let { [a]: b } = ...`
-                // `"a"` in `let { "a": b } = ...`
-                // `1` in `let { 1: b } = ...`
-                if (bindingElement.propertyName) {
-                    var propertyName = bindingElement.propertyName;
-                    if (ts.isPrivateIdentifier(propertyName)) {
-                        return ts.Debug.failBadSyntaxKind(propertyName);
-                    }
-                    return ts.isComputedPropertyName(propertyName) && isStringOrNumericLiteral(propertyName.expression)
-                        ? propertyName.expression
-                        : propertyName;
+        function findMostOverlappyType(source, unionTarget) {
+            var bestMatch;
+            var matchingCount = 0;
+            for (var _i = 0, _a = unionTarget.types; _i < _a.length; _i++) {
+                var target = _a[_i];
+                var overlap = getIntersectionType([getIndexType(source), getIndexType(target)]);
+                if (overlap.flags & 4194304 /* Index */) {
+                    // perfect overlap of keys
+                    bestMatch = target;
+                    matchingCount = Infinity;
                 }
-                break;
-            case 281 /* PropertyAssignment */:
-                // `a` in `({ a: b } = ...)`
-                // `[a]` in `({ [a]: b } = ...)`
-                // `"a"` in `({ "a": b } = ...)`
-                // `1` in `({ 1: b } = ...)`
-                if (bindingElement.name) {
-                    var propertyName = bindingElement.name;
-                    if (ts.isPrivateIdentifier(propertyName)) {
-                        return ts.Debug.failBadSyntaxKind(propertyName);
+                else if (overlap.flags & 1048576 /* Union */) {
+                    // We only want to account for literal types otherwise.
+                    // If we have a union of index types, it seems likely that we
+                    // needed to elaborate between two generic mapped types anyway.
+                    var len = ts.length(ts.filter(overlap.types, isUnitType));
+                    if (len >= matchingCount) {
+                        bestMatch = target;
+                        matchingCount = len;
                     }
-                    return ts.isComputedPropertyName(propertyName) && isStringOrNumericLiteral(propertyName.expression)
-                        ? propertyName.expression
-                        : propertyName;
                 }
-                break;
-            case 283 /* SpreadAssignment */:
-                // `a` in `({ ...a } = ...)`
-                if (bindingElement.name && ts.isPrivateIdentifier(bindingElement.name)) {
-                    return ts.Debug.failBadSyntaxKind(bindingElement.name);
+                else if (isUnitType(overlap) && 1 >= matchingCount) {
+                    bestMatch = target;
+                    matchingCount = 1;
                 }
-                return bindingElement.name;
-        }
-        var target = getTargetOfBindingOrAssignmentElement(bindingElement);
-        if (target && ts.isPropertyName(target)) {
-            return target;
-        }
-    }
-    ts.tryGetPropertyNameOfBindingOrAssignmentElement = tryGetPropertyNameOfBindingOrAssignmentElement;
-    function isStringOrNumericLiteral(node) {
-        var kind = node.kind;
-        return kind === 10 /* StringLiteral */
-            || kind === 8 /* NumericLiteral */;
-    }
-    /**
-     * Gets the elements of a BindingOrAssignmentPattern
-     */
-    function getElementsOfBindingOrAssignmentPattern(name) {
-        switch (name.kind) {
-            case 189 /* ObjectBindingPattern */:
-            case 190 /* ArrayBindingPattern */:
-            case 192 /* ArrayLiteralExpression */:
-                // `a` in `{a}`
-                // `a` in `[a]`
-                return name.elements;
-            case 193 /* ObjectLiteralExpression */:
-                // `a` in `{a}`
-                return name.properties;
-        }
-    }
-    ts.getElementsOfBindingOrAssignmentPattern = getElementsOfBindingOrAssignmentPattern;
-    function convertToArrayAssignmentElement(element) {
-        if (ts.isBindingElement(element)) {
-            if (element.dotDotDotToken) {
-                ts.Debug.assertNode(element.name, ts.isIdentifier);
-                return ts.setOriginalNode(ts.setTextRange(ts.createSpread(element.name), element), element);
             }
-            var expression = convertToAssignmentElementTarget(element.name);
-            return element.initializer
-                ? ts.setOriginalNode(ts.setTextRange(ts.createAssignment(expression, element.initializer), element), element)
-                : expression;
+            return bestMatch;
         }
-        ts.Debug.assertNode(element, ts.isExpression);
-        return element;
-    }
-    ts.convertToArrayAssignmentElement = convertToArrayAssignmentElement;
-    function convertToObjectAssignmentElement(element) {
-        if (ts.isBindingElement(element)) {
-            if (element.dotDotDotToken) {
-                ts.Debug.assertNode(element.name, ts.isIdentifier);
-                return ts.setOriginalNode(ts.setTextRange(ts.createSpreadAssignment(element.name), element), element);
+        function filterPrimitivesIfContainsNonPrimitive(type) {
+            if (maybeTypeOfKind(type, 67108864 /* NonPrimitive */)) {
+                var result = filterType(type, function (t) { return !(t.flags & 131068 /* Primitive */); });
+                if (!(result.flags & 131072 /* Never */)) {
+                    return result;
+                }
             }
-            if (element.propertyName) {
-                var expression = convertToAssignmentElementTarget(element.name);
-                return ts.setOriginalNode(ts.setTextRange(ts.createPropertyAssignment(element.propertyName, element.initializer ? ts.createAssignment(expression, element.initializer) : expression), element), element);
+            return type;
+        }
+        // Keep this up-to-date with the same logic within `getApparentTypeOfContextualType`, since they should behave similarly
+        function findMatchingDiscriminantType(source, target, isRelatedTo, skipPartial) {
+            if (target.flags & 1048576 /* Union */ && source.flags & (2097152 /* Intersection */ | 524288 /* Object */)) {
+                var sourceProperties = getPropertiesOfType(source);
+                if (sourceProperties) {
+                    var sourcePropertiesFiltered = findDiscriminantProperties(sourceProperties, target);
+                    if (sourcePropertiesFiltered) {
+                        return discriminateTypeByDiscriminableItems(target, ts.map(sourcePropertiesFiltered, function (p) { return [function () { return getTypeOfSymbol(p); }, p.escapedName]; }), isRelatedTo, /*defaultValue*/ undefined, skipPartial);
+                    }
+                }
             }
-            ts.Debug.assertNode(element.name, ts.isIdentifier);
-            return ts.setOriginalNode(ts.setTextRange(ts.createShorthandPropertyAssignment(element.name, element.initializer), element), element);
+            return undefined;
         }
-        ts.Debug.assertNode(element, ts.isObjectLiteralElementLike);
-        return element;
     }
-    ts.convertToObjectAssignmentElement = convertToObjectAssignmentElement;
-    function convertToAssignmentPattern(node) {
-        switch (node.kind) {
-            case 190 /* ArrayBindingPattern */:
-            case 192 /* ArrayLiteralExpression */:
-                return convertToArrayAssignmentPattern(node);
-            case 189 /* ObjectBindingPattern */:
-            case 193 /* ObjectLiteralExpression */:
-                return convertToObjectAssignmentPattern(node);
-        }
+    ts.createTypeChecker = createTypeChecker;
+    function isNotAccessor(declaration) {
+        // Accessors check for their own matching duplicates, and in contexts where they are valid, there are already duplicate identifier checks
+        return !ts.isAccessor(declaration);
     }
-    ts.convertToAssignmentPattern = convertToAssignmentPattern;
-    function convertToObjectAssignmentPattern(node) {
-        if (ts.isObjectBindingPattern(node)) {
-            return ts.setOriginalNode(ts.setTextRange(ts.createObjectLiteral(ts.map(node.elements, convertToObjectAssignmentElement)), node), node);
+    function isNotOverload(declaration) {
+        return (declaration.kind !== 251 /* FunctionDeclaration */ && declaration.kind !== 165 /* MethodDeclaration */) ||
+            !!declaration.body;
+    }
+    /** Like 'isDeclarationName', but returns true for LHS of `import { x as y }` or `export { x as y }`. */
+    function isDeclarationNameOrImportPropertyName(name) {
+        switch (name.parent.kind) {
+            case 265 /* ImportSpecifier */:
+            case 270 /* ExportSpecifier */:
+                return ts.isIdentifier(name);
+            default:
+                return ts.isDeclarationName(name);
         }
-        ts.Debug.assertNode(node, ts.isObjectLiteralExpression);
-        return node;
     }
-    ts.convertToObjectAssignmentPattern = convertToObjectAssignmentPattern;
-    function convertToArrayAssignmentPattern(node) {
-        if (ts.isArrayBindingPattern(node)) {
-            return ts.setOriginalNode(ts.setTextRange(ts.createArrayLiteral(ts.map(node.elements, convertToArrayAssignmentElement)), node), node);
+    function isSomeImportDeclaration(decl) {
+        switch (decl.kind) {
+            case 262 /* ImportClause */: // For default import
+            case 260 /* ImportEqualsDeclaration */:
+            case 263 /* NamespaceImport */:
+            case 265 /* ImportSpecifier */: // For rename import `x as y`
+                return true;
+            case 78 /* Identifier */:
+                // For regular import, `decl` is an Identifier under the ImportSpecifier.
+                return decl.parent.kind === 265 /* ImportSpecifier */;
+            default:
+                return false;
         }
-        ts.Debug.assertNode(node, ts.isArrayLiteralExpression);
-        return node;
     }
-    ts.convertToArrayAssignmentPattern = convertToArrayAssignmentPattern;
-    function convertToAssignmentElementTarget(node) {
-        if (ts.isBindingPattern(node)) {
-            return convertToAssignmentPattern(node);
+    var JsxNames;
+    (function (JsxNames) {
+        JsxNames.JSX = "JSX";
+        JsxNames.IntrinsicElements = "IntrinsicElements";
+        JsxNames.ElementClass = "ElementClass";
+        JsxNames.ElementAttributesPropertyNameContainer = "ElementAttributesProperty"; // TODO: Deprecate and remove support
+        JsxNames.ElementChildrenAttributeNameContainer = "ElementChildrenAttribute";
+        JsxNames.Element = "Element";
+        JsxNames.IntrinsicAttributes = "IntrinsicAttributes";
+        JsxNames.IntrinsicClassAttributes = "IntrinsicClassAttributes";
+        JsxNames.LibraryManagedAttributes = "LibraryManagedAttributes";
+    })(JsxNames || (JsxNames = {}));
+    function getIterationTypesKeyFromIterationTypeKind(typeKind) {
+        switch (typeKind) {
+            case 0 /* Yield */: return "yieldType";
+            case 1 /* Return */: return "returnType";
+            case 2 /* Next */: return "nextType";
         }
-        ts.Debug.assertNode(node, ts.isExpression);
-        return node;
     }
-    ts.convertToAssignmentElementTarget = convertToAssignmentElementTarget;
+    function signatureHasRestParameter(s) {
+        return !!(s.flags & 1 /* HasRestParameter */);
+    }
+    ts.signatureHasRestParameter = signatureHasRestParameter;
+    function signatureHasLiteralTypes(s) {
+        return !!(s.flags & 2 /* HasLiteralTypes */);
+    }
+    ts.signatureHasLiteralTypes = signatureHasLiteralTypes;
 })(ts || (ts = {}));
 var ts;
 (function (ts) {
@@ -74771,7 +80326,6 @@ var ts;
         if (node === undefined || visitor === undefined) {
             return node;
         }
-        ts.aggregateTransformFlags(node);
         var visited = visitor(node);
         if (visited === node) {
             return node;
@@ -74787,7 +80341,6 @@ var ts;
             visitedNode = visited;
         }
         ts.Debug.assertNode(visitedNode, test);
-        ts.aggregateTransformFlags(visitedNode);
         return visitedNode;
     }
     ts.visitNode = visitNode;
@@ -74813,53 +80366,63 @@ var ts;
         if (count === undefined || count > length - start) {
             count = length - start;
         }
+        var hasTrailingComma;
+        var pos = -1;
+        var end = -1;
         if (start > 0 || count < length) {
             // If we are not visiting all of the original nodes, we must always create a new array.
             // Since this is a fragment of a node array, we do not copy over the previous location
             // and will only copy over `hasTrailingComma` if we are including the last element.
-            updated = ts.createNodeArray([], /*hasTrailingComma*/ nodes.hasTrailingComma && start + count === length);
+            updated = [];
+            hasTrailingComma = nodes.hasTrailingComma && start + count === length;
         }
         // Visit each original node.
         for (var i = 0; i < count; i++) {
             var node = nodes[i + start];
-            ts.aggregateTransformFlags(node);
             var visited = node !== undefined ? visitor(node) : undefined;
             if (updated !== undefined || visited === undefined || visited !== node) {
                 if (updated === undefined) {
                     // Ensure we have a copy of `nodes`, up to the current index.
-                    updated = ts.createNodeArray(nodes.slice(0, i), nodes.hasTrailingComma);
-                    ts.setTextRange(updated, nodes);
+                    updated = nodes.slice(0, i);
+                    hasTrailingComma = nodes.hasTrailingComma;
+                    pos = nodes.pos;
+                    end = nodes.end;
                 }
                 if (visited) {
                     if (ts.isArray(visited)) {
                         for (var _i = 0, visited_1 = visited; _i < visited_1.length; _i++) {
                             var visitedNode = visited_1[_i];
-                            ts.Debug.assertNode(visitedNode, test);
-                            ts.aggregateTransformFlags(visitedNode);
+                            void ts.Debug.assertNode(visitedNode, test);
                             updated.push(visitedNode);
                         }
                     }
                     else {
-                        ts.Debug.assertNode(visited, test);
-                        ts.aggregateTransformFlags(visited);
+                        void ts.Debug.assertNode(visited, test);
                         updated.push(visited);
                     }
                 }
             }
         }
-        return updated || nodes;
+        if (updated) {
+            // TODO(rbuckton): Remove dependency on `ts.factory` in favor of a provided factory.
+            var updatedArray = ts.factory.createNodeArray(updated, hasTrailingComma);
+            ts.setTextRangePosEnd(updatedArray, pos, end);
+            return updatedArray;
+        }
+        return nodes;
     }
     ts.visitNodes = visitNodes;
     /**
      * Starts a new lexical environment and visits a statement list, ending the lexical environment
      * and merging hoisted declarations upon completion.
      */
-    function visitLexicalEnvironment(statements, visitor, context, start, ensureUseStrict) {
+    function visitLexicalEnvironment(statements, visitor, context, start, ensureUseStrict, nodesVisitor) {
+        if (nodesVisitor === void 0) { nodesVisitor = visitNodes; }
         context.startLexicalEnvironment();
-        statements = visitNodes(statements, visitor, ts.isStatement, start);
+        statements = nodesVisitor(statements, visitor, ts.isStatement, start);
         if (ensureUseStrict)
-            statements = ts.ensureUseStrict(statements); // eslint-disable-line @typescript-eslint/no-unnecessary-qualifier
-        return ts.mergeLexicalEnvironment(statements, context.endLexicalEnvironment());
+            statements = context.factory.ensureUseStrict(statements);
+        return ts.factory.mergeLexicalEnvironment(statements, context.endLexicalEnvironment());
     }
     ts.visitLexicalEnvironment = visitLexicalEnvironment;
     function visitParameterList(nodes, visitor, context, nodesVisitor) {
@@ -74897,7 +80460,7 @@ var ts;
             }
         }
         if (result) {
-            return ts.setTextRange(ts.createNodeArray(result, parameters.hasTrailingComma), parameters);
+            return ts.setTextRange(context.factory.createNodeArray(result, parameters.hasTrailingComma), parameters);
         }
         return parameters;
     }
@@ -74910,325 +80473,345 @@ var ts;
                     parameter;
     }
     function addDefaultValueAssignmentForBindingPattern(parameter, context) {
-        context.addInitializationStatement(ts.createVariableStatement(
-        /*modifiers*/ undefined, ts.createVariableDeclarationList([
-            ts.createVariableDeclaration(parameter.name, parameter.type, parameter.initializer ?
-                ts.createConditional(ts.createStrictEquality(ts.getGeneratedNameForNode(parameter), ts.createVoidZero()), parameter.initializer, ts.getGeneratedNameForNode(parameter)) :
-                ts.getGeneratedNameForNode(parameter)),
+        var factory = context.factory;
+        context.addInitializationStatement(factory.createVariableStatement(
+        /*modifiers*/ undefined, factory.createVariableDeclarationList([
+            factory.createVariableDeclaration(parameter.name, 
+            /*exclamationToken*/ undefined, parameter.type, parameter.initializer ?
+                factory.createConditionalExpression(factory.createStrictEquality(factory.getGeneratedNameForNode(parameter), factory.createVoidZero()), 
+                /*questionToken*/ undefined, parameter.initializer, 
+                /*colonToken*/ undefined, factory.getGeneratedNameForNode(parameter)) :
+                factory.getGeneratedNameForNode(parameter)),
         ])));
-        return ts.updateParameter(parameter, parameter.decorators, parameter.modifiers, parameter.dotDotDotToken, ts.getGeneratedNameForNode(parameter), parameter.questionToken, parameter.type, 
+        return factory.updateParameterDeclaration(parameter, parameter.decorators, parameter.modifiers, parameter.dotDotDotToken, factory.getGeneratedNameForNode(parameter), parameter.questionToken, parameter.type, 
         /*initializer*/ undefined);
     }
     function addDefaultValueAssignmentForInitializer(parameter, name, initializer, context) {
-        context.addInitializationStatement(ts.createIf(ts.createTypeCheck(ts.getSynthesizedClone(name), "undefined"), ts.setEmitFlags(ts.setTextRange(ts.createBlock([
-            ts.createExpressionStatement(ts.setEmitFlags(ts.setTextRange(ts.createAssignment(ts.setEmitFlags(ts.getMutableClone(name), 48 /* NoSourceMap */), ts.setEmitFlags(initializer, 48 /* NoSourceMap */ | ts.getEmitFlags(initializer) | 1536 /* NoComments */)), parameter), 1536 /* NoComments */))
+        var factory = context.factory;
+        context.addInitializationStatement(factory.createIfStatement(factory.createTypeCheck(factory.cloneNode(name), "undefined"), ts.setEmitFlags(ts.setTextRange(factory.createBlock([
+            factory.createExpressionStatement(ts.setEmitFlags(ts.setTextRange(factory.createAssignment(ts.setEmitFlags(factory.cloneNode(name), 48 /* NoSourceMap */), ts.setEmitFlags(initializer, 48 /* NoSourceMap */ | ts.getEmitFlags(initializer) | 1536 /* NoComments */)), parameter), 1536 /* NoComments */))
         ]), parameter), 1 /* SingleLine */ | 32 /* NoTrailingSourceMap */ | 384 /* NoTokenSourceMaps */ | 1536 /* NoComments */)));
-        return ts.updateParameter(parameter, parameter.decorators, parameter.modifiers, parameter.dotDotDotToken, parameter.name, parameter.questionToken, parameter.type, 
+        return factory.updateParameterDeclaration(parameter, parameter.decorators, parameter.modifiers, parameter.dotDotDotToken, parameter.name, parameter.questionToken, parameter.type, 
         /*initializer*/ undefined);
     }
-    function visitFunctionBody(node, visitor, context) {
+    function visitFunctionBody(node, visitor, context, nodeVisitor) {
+        if (nodeVisitor === void 0) { nodeVisitor = visitNode; }
         context.resumeLexicalEnvironment();
-        var updated = visitNode(node, visitor, ts.isConciseBody);
+        var updated = nodeVisitor(node, visitor, ts.isConciseBody);
         var declarations = context.endLexicalEnvironment();
         if (ts.some(declarations)) {
-            var block = ts.convertToFunctionBody(updated);
-            var statements = ts.mergeLexicalEnvironment(block.statements, declarations);
-            return ts.updateBlock(block, statements);
+            if (!updated) {
+                return context.factory.createBlock(declarations);
+            }
+            var block = context.factory.converters.convertToFunctionBlock(updated);
+            var statements = ts.factory.mergeLexicalEnvironment(block.statements, declarations);
+            return context.factory.updateBlock(block, statements);
         }
         return updated;
     }
     ts.visitFunctionBody = visitFunctionBody;
-    function visitEachChild(node, visitor, context, nodesVisitor, tokenVisitor) {
+    function visitEachChild(node, visitor, context, nodesVisitor, tokenVisitor, nodeVisitor) {
         if (nodesVisitor === void 0) { nodesVisitor = visitNodes; }
+        if (nodeVisitor === void 0) { nodeVisitor = visitNode; }
         if (node === undefined) {
             return undefined;
         }
         var kind = node.kind;
         // No need to visit nodes with no children.
-        if ((kind > 0 /* FirstToken */ && kind <= 152 /* LastToken */) || kind === 183 /* ThisType */) {
+        if ((kind > 0 /* FirstToken */ && kind <= 156 /* LastToken */) || kind === 187 /* ThisType */) {
             return node;
         }
+        var factory = context.factory;
         switch (kind) {
             // Names
-            case 75 /* Identifier */:
-                return ts.updateIdentifier(node, nodesVisitor(node.typeArguments, visitor, isTypeNodeOrTypeParameterDeclaration));
-            case 153 /* QualifiedName */:
-                return ts.updateQualifiedName(node, visitNode(node.left, visitor, ts.isEntityName), visitNode(node.right, visitor, ts.isIdentifier));
-            case 154 /* ComputedPropertyName */:
-                return ts.updateComputedPropertyName(node, visitNode(node.expression, visitor, ts.isExpression));
+            case 78 /* Identifier */:
+                return factory.updateIdentifier(node, nodesVisitor(node.typeArguments, visitor, isTypeNodeOrTypeParameterDeclaration));
+            case 157 /* QualifiedName */:
+                return factory.updateQualifiedName(node, nodeVisitor(node.left, visitor, ts.isEntityName), nodeVisitor(node.right, visitor, ts.isIdentifier));
+            case 158 /* ComputedPropertyName */:
+                return factory.updateComputedPropertyName(node, nodeVisitor(node.expression, visitor, ts.isExpression));
             // Signature elements
-            case 155 /* TypeParameter */:
-                return ts.updateTypeParameterDeclaration(node, visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.constraint, visitor, ts.isTypeNode), visitNode(node.default, visitor, ts.isTypeNode));
-            case 156 /* Parameter */:
-                return ts.updateParameter(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.dotDotDotToken, tokenVisitor, ts.isToken), visitNode(node.name, visitor, ts.isBindingName), visitNode(node.questionToken, tokenVisitor, ts.isToken), visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.initializer, visitor, ts.isExpression));
-            case 157 /* Decorator */:
-                return ts.updateDecorator(node, visitNode(node.expression, visitor, ts.isExpression));
+            case 159 /* TypeParameter */:
+                return factory.updateTypeParameterDeclaration(node, nodeVisitor(node.name, visitor, ts.isIdentifier), nodeVisitor(node.constraint, visitor, ts.isTypeNode), nodeVisitor(node.default, visitor, ts.isTypeNode));
+            case 160 /* Parameter */:
+                return factory.updateParameterDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.dotDotDotToken, tokenVisitor, ts.isToken), nodeVisitor(node.name, visitor, ts.isBindingName), nodeVisitor(node.questionToken, tokenVisitor, ts.isToken), nodeVisitor(node.type, visitor, ts.isTypeNode), nodeVisitor(node.initializer, visitor, ts.isExpression));
+            case 161 /* Decorator */:
+                return factory.updateDecorator(node, nodeVisitor(node.expression, visitor, ts.isExpression));
             // Type elements
-            case 158 /* PropertySignature */:
-                return ts.updatePropertySignature(node, nodesVisitor(node.modifiers, visitor, ts.isToken), visitNode(node.name, visitor, ts.isPropertyName), visitNode(node.questionToken, tokenVisitor, ts.isToken), visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.initializer, visitor, ts.isExpression));
-            case 159 /* PropertyDeclaration */:
-                return ts.updateProperty(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isPropertyName), 
+            case 162 /* PropertySignature */:
+                return factory.updatePropertySignature(node, nodesVisitor(node.modifiers, visitor, ts.isToken), nodeVisitor(node.name, visitor, ts.isPropertyName), nodeVisitor(node.questionToken, tokenVisitor, ts.isToken), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 163 /* PropertyDeclaration */:
+                return factory.updatePropertyDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isPropertyName), 
                 // QuestionToken and ExclamationToken is uniqued in Property Declaration and the signature of 'updateProperty' is that too
-                visitNode(node.questionToken || node.exclamationToken, tokenVisitor, ts.isToken), visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.initializer, visitor, ts.isExpression));
-            case 160 /* MethodSignature */:
-                return ts.updateMethodSignature(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.name, visitor, ts.isPropertyName), visitNode(node.questionToken, tokenVisitor, ts.isToken));
-            case 161 /* MethodDeclaration */:
-                return ts.updateMethod(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.asteriskToken, tokenVisitor, ts.isToken), visitNode(node.name, visitor, ts.isPropertyName), visitNode(node.questionToken, tokenVisitor, ts.isToken), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitNode(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context));
-            case 162 /* Constructor */:
-                return ts.updateConstructor(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitFunctionBody(node.body, visitor, context));
-            case 163 /* GetAccessor */:
-                return ts.updateGetAccessor(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isPropertyName), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitNode(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context));
-            case 164 /* SetAccessor */:
-                return ts.updateSetAccessor(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isPropertyName), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitFunctionBody(node.body, visitor, context));
-            case 165 /* CallSignature */:
-                return ts.updateCallSignature(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode));
-            case 166 /* ConstructSignature */:
-                return ts.updateConstructSignature(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode));
-            case 167 /* IndexSignature */:
-                return ts.updateIndexSignature(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode));
+                nodeVisitor(node.questionToken || node.exclamationToken, tokenVisitor, ts.isToken), nodeVisitor(node.type, visitor, ts.isTypeNode), nodeVisitor(node.initializer, visitor, ts.isExpression));
+            case 164 /* MethodSignature */:
+                return factory.updateMethodSignature(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isPropertyName), nodeVisitor(node.questionToken, tokenVisitor, ts.isToken), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 165 /* MethodDeclaration */:
+                return factory.updateMethodDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.asteriskToken, tokenVisitor, ts.isToken), nodeVisitor(node.name, visitor, ts.isPropertyName), nodeVisitor(node.questionToken, tokenVisitor, ts.isToken), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), nodeVisitor(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 166 /* Constructor */:
+                return factory.updateConstructorDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 167 /* GetAccessor */:
+                return factory.updateGetAccessorDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isPropertyName), visitParameterList(node.parameters, visitor, context, nodesVisitor), nodeVisitor(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 168 /* SetAccessor */:
+                return factory.updateSetAccessorDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isPropertyName), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 169 /* CallSignature */:
+                return factory.updateCallSignature(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 170 /* ConstructSignature */:
+                return factory.updateConstructSignature(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 171 /* IndexSignature */:
+                return factory.updateIndexSignature(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
             // Types
-            case 168 /* TypePredicate */:
-                return ts.updateTypePredicateNodeWithModifier(node, visitNode(node.assertsModifier, visitor), visitNode(node.parameterName, visitor), visitNode(node.type, visitor, ts.isTypeNode));
-            case 169 /* TypeReference */:
-                return ts.updateTypeReferenceNode(node, visitNode(node.typeName, visitor, ts.isEntityName), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode));
-            case 170 /* FunctionType */:
-                return ts.updateFunctionTypeNode(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode));
-            case 171 /* ConstructorType */:
-                return ts.updateConstructorTypeNode(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode));
-            case 172 /* TypeQuery */:
-                return ts.updateTypeQueryNode(node, visitNode(node.exprName, visitor, ts.isEntityName));
-            case 173 /* TypeLiteral */:
-                return ts.updateTypeLiteralNode(node, nodesVisitor(node.members, visitor, ts.isTypeElement));
-            case 174 /* ArrayType */:
-                return ts.updateArrayTypeNode(node, visitNode(node.elementType, visitor, ts.isTypeNode));
-            case 175 /* TupleType */:
-                return ts.updateTupleTypeNode(node, nodesVisitor(node.elementTypes, visitor, ts.isTypeNode));
-            case 176 /* OptionalType */:
-                return ts.updateOptionalTypeNode(node, visitNode(node.type, visitor, ts.isTypeNode));
-            case 177 /* RestType */:
-                return ts.updateRestTypeNode(node, visitNode(node.type, visitor, ts.isTypeNode));
-            case 178 /* UnionType */:
-                return ts.updateUnionTypeNode(node, nodesVisitor(node.types, visitor, ts.isTypeNode));
-            case 179 /* IntersectionType */:
-                return ts.updateIntersectionTypeNode(node, nodesVisitor(node.types, visitor, ts.isTypeNode));
-            case 180 /* ConditionalType */:
-                return ts.updateConditionalTypeNode(node, visitNode(node.checkType, visitor, ts.isTypeNode), visitNode(node.extendsType, visitor, ts.isTypeNode), visitNode(node.trueType, visitor, ts.isTypeNode), visitNode(node.falseType, visitor, ts.isTypeNode));
-            case 181 /* InferType */:
-                return ts.updateInferTypeNode(node, visitNode(node.typeParameter, visitor, ts.isTypeParameterDeclaration));
-            case 188 /* ImportType */:
-                return ts.updateImportTypeNode(node, visitNode(node.argument, visitor, ts.isTypeNode), visitNode(node.qualifier, visitor, ts.isEntityName), visitNodes(node.typeArguments, visitor, ts.isTypeNode), node.isTypeOf);
-            case 182 /* ParenthesizedType */:
-                return ts.updateParenthesizedType(node, visitNode(node.type, visitor, ts.isTypeNode));
-            case 184 /* TypeOperator */:
-                return ts.updateTypeOperatorNode(node, visitNode(node.type, visitor, ts.isTypeNode));
-            case 185 /* IndexedAccessType */:
-                return ts.updateIndexedAccessTypeNode(node, visitNode(node.objectType, visitor, ts.isTypeNode), visitNode(node.indexType, visitor, ts.isTypeNode));
-            case 186 /* MappedType */:
-                return ts.updateMappedTypeNode(node, visitNode(node.readonlyToken, tokenVisitor, ts.isToken), visitNode(node.typeParameter, visitor, ts.isTypeParameterDeclaration), visitNode(node.questionToken, tokenVisitor, ts.isToken), visitNode(node.type, visitor, ts.isTypeNode));
-            case 187 /* LiteralType */:
-                return ts.updateLiteralTypeNode(node, visitNode(node.literal, visitor, ts.isExpression));
+            case 172 /* TypePredicate */:
+                return factory.updateTypePredicateNode(node, nodeVisitor(node.assertsModifier, visitor), nodeVisitor(node.parameterName, visitor), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 173 /* TypeReference */:
+                return factory.updateTypeReferenceNode(node, nodeVisitor(node.typeName, visitor, ts.isEntityName), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode));
+            case 174 /* FunctionType */:
+                return factory.updateFunctionTypeNode(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 175 /* ConstructorType */:
+                return factory.updateConstructorTypeNode(node, nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.parameters, visitor, ts.isParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 176 /* TypeQuery */:
+                return factory.updateTypeQueryNode(node, nodeVisitor(node.exprName, visitor, ts.isEntityName));
+            case 177 /* TypeLiteral */:
+                return factory.updateTypeLiteralNode(node, nodesVisitor(node.members, visitor, ts.isTypeElement));
+            case 178 /* ArrayType */:
+                return factory.updateArrayTypeNode(node, nodeVisitor(node.elementType, visitor, ts.isTypeNode));
+            case 179 /* TupleType */:
+                return factory.updateTupleTypeNode(node, nodesVisitor(node.elements, visitor, ts.isTypeNode));
+            case 180 /* OptionalType */:
+                return factory.updateOptionalTypeNode(node, nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 181 /* RestType */:
+                return factory.updateRestTypeNode(node, nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 182 /* UnionType */:
+                return factory.updateUnionTypeNode(node, nodesVisitor(node.types, visitor, ts.isTypeNode));
+            case 183 /* IntersectionType */:
+                return factory.updateIntersectionTypeNode(node, nodesVisitor(node.types, visitor, ts.isTypeNode));
+            case 184 /* ConditionalType */:
+                return factory.updateConditionalTypeNode(node, nodeVisitor(node.checkType, visitor, ts.isTypeNode), nodeVisitor(node.extendsType, visitor, ts.isTypeNode), nodeVisitor(node.trueType, visitor, ts.isTypeNode), nodeVisitor(node.falseType, visitor, ts.isTypeNode));
+            case 185 /* InferType */:
+                return factory.updateInferTypeNode(node, nodeVisitor(node.typeParameter, visitor, ts.isTypeParameterDeclaration));
+            case 195 /* ImportType */:
+                return factory.updateImportTypeNode(node, nodeVisitor(node.argument, visitor, ts.isTypeNode), nodeVisitor(node.qualifier, visitor, ts.isEntityName), visitNodes(node.typeArguments, visitor, ts.isTypeNode), node.isTypeOf);
+            case 192 /* NamedTupleMember */:
+                return factory.updateNamedTupleMember(node, visitNode(node.dotDotDotToken, visitor, ts.isToken), visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.questionToken, visitor, ts.isToken), visitNode(node.type, visitor, ts.isTypeNode));
+            case 186 /* ParenthesizedType */:
+                return factory.updateParenthesizedType(node, nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 188 /* TypeOperator */:
+                return factory.updateTypeOperatorNode(node, nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 189 /* IndexedAccessType */:
+                return factory.updateIndexedAccessTypeNode(node, nodeVisitor(node.objectType, visitor, ts.isTypeNode), nodeVisitor(node.indexType, visitor, ts.isTypeNode));
+            case 190 /* MappedType */:
+                return factory.updateMappedTypeNode(node, nodeVisitor(node.readonlyToken, tokenVisitor, ts.isToken), nodeVisitor(node.typeParameter, visitor, ts.isTypeParameterDeclaration), nodeVisitor(node.nameType, visitor, ts.isTypeNode), nodeVisitor(node.questionToken, tokenVisitor, ts.isToken), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 191 /* LiteralType */:
+                return factory.updateLiteralTypeNode(node, nodeVisitor(node.literal, visitor, ts.isExpression));
+            case 193 /* TemplateLiteralType */:
+                return factory.updateTemplateLiteralType(node, nodeVisitor(node.head, visitor, ts.isTemplateHead), nodesVisitor(node.templateSpans, visitor, ts.isTemplateLiteralTypeSpan));
+            case 194 /* TemplateLiteralTypeSpan */:
+                return factory.updateTemplateLiteralTypeSpan(node, nodeVisitor(node.type, visitor, ts.isTypeNode), nodeVisitor(node.literal, visitor, ts.isTemplateMiddleOrTemplateTail));
             // Binding patterns
-            case 189 /* ObjectBindingPattern */:
-                return ts.updateObjectBindingPattern(node, nodesVisitor(node.elements, visitor, ts.isBindingElement));
-            case 190 /* ArrayBindingPattern */:
-                return ts.updateArrayBindingPattern(node, nodesVisitor(node.elements, visitor, ts.isArrayBindingElement));
-            case 191 /* BindingElement */:
-                return ts.updateBindingElement(node, visitNode(node.dotDotDotToken, tokenVisitor, ts.isToken), visitNode(node.propertyName, visitor, ts.isPropertyName), visitNode(node.name, visitor, ts.isBindingName), visitNode(node.initializer, visitor, ts.isExpression));
+            case 196 /* ObjectBindingPattern */:
+                return factory.updateObjectBindingPattern(node, nodesVisitor(node.elements, visitor, ts.isBindingElement));
+            case 197 /* ArrayBindingPattern */:
+                return factory.updateArrayBindingPattern(node, nodesVisitor(node.elements, visitor, ts.isArrayBindingElement));
+            case 198 /* BindingElement */:
+                return factory.updateBindingElement(node, nodeVisitor(node.dotDotDotToken, tokenVisitor, ts.isToken), nodeVisitor(node.propertyName, visitor, ts.isPropertyName), nodeVisitor(node.name, visitor, ts.isBindingName), nodeVisitor(node.initializer, visitor, ts.isExpression));
             // Expression
-            case 192 /* ArrayLiteralExpression */:
-                return ts.updateArrayLiteral(node, nodesVisitor(node.elements, visitor, ts.isExpression));
-            case 193 /* ObjectLiteralExpression */:
-                return ts.updateObjectLiteral(node, nodesVisitor(node.properties, visitor, ts.isObjectLiteralElementLike));
-            case 194 /* PropertyAccessExpression */:
+            case 199 /* ArrayLiteralExpression */:
+                return factory.updateArrayLiteralExpression(node, nodesVisitor(node.elements, visitor, ts.isExpression));
+            case 200 /* ObjectLiteralExpression */:
+                return factory.updateObjectLiteralExpression(node, nodesVisitor(node.properties, visitor, ts.isObjectLiteralElementLike));
+            case 201 /* PropertyAccessExpression */:
                 if (node.flags & 32 /* OptionalChain */) {
-                    return ts.updatePropertyAccessChain(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.questionDotToken, tokenVisitor, ts.isToken), visitNode(node.name, visitor, ts.isIdentifier));
+                    return factory.updatePropertyAccessChain(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.questionDotToken, tokenVisitor, ts.isToken), nodeVisitor(node.name, visitor, ts.isIdentifier));
                 }
-                return ts.updatePropertyAccess(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.name, visitor, ts.isIdentifierOrPrivateIdentifier));
-            case 195 /* ElementAccessExpression */:
+                return factory.updatePropertyAccessExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.name, visitor, ts.isIdentifierOrPrivateIdentifier));
+            case 202 /* ElementAccessExpression */:
                 if (node.flags & 32 /* OptionalChain */) {
-                    return ts.updateElementAccessChain(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.questionDotToken, tokenVisitor, ts.isToken), visitNode(node.argumentExpression, visitor, ts.isExpression));
+                    return factory.updateElementAccessChain(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.questionDotToken, tokenVisitor, ts.isToken), nodeVisitor(node.argumentExpression, visitor, ts.isExpression));
                 }
-                return ts.updateElementAccess(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.argumentExpression, visitor, ts.isExpression));
-            case 196 /* CallExpression */:
+                return factory.updateElementAccessExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.argumentExpression, visitor, ts.isExpression));
+            case 203 /* CallExpression */:
                 if (node.flags & 32 /* OptionalChain */) {
-                    return ts.updateCallChain(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.questionDotToken, tokenVisitor, ts.isToken), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodesVisitor(node.arguments, visitor, ts.isExpression));
-                }
-                return ts.updateCall(node, visitNode(node.expression, visitor, ts.isExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodesVisitor(node.arguments, visitor, ts.isExpression));
-            case 197 /* NewExpression */:
-                return ts.updateNew(node, visitNode(node.expression, visitor, ts.isExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodesVisitor(node.arguments, visitor, ts.isExpression));
-            case 198 /* TaggedTemplateExpression */:
-                return ts.updateTaggedTemplate(node, visitNode(node.tag, visitor, ts.isExpression), visitNodes(node.typeArguments, visitor, ts.isExpression), visitNode(node.template, visitor, ts.isTemplateLiteral));
-            case 199 /* TypeAssertionExpression */:
-                return ts.updateTypeAssertion(node, visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.expression, visitor, ts.isExpression));
-            case 200 /* ParenthesizedExpression */:
-                return ts.updateParen(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 201 /* FunctionExpression */:
-                return ts.updateFunctionExpression(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.asteriskToken, tokenVisitor, ts.isToken), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitNode(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context));
-            case 202 /* ArrowFunction */:
-                return ts.updateArrowFunction(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.equalsGreaterThanToken, tokenVisitor, ts.isToken), visitFunctionBody(node.body, visitor, context));
-            case 203 /* DeleteExpression */:
-                return ts.updateDelete(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 204 /* TypeOfExpression */:
-                return ts.updateTypeOf(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 205 /* VoidExpression */:
-                return ts.updateVoid(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 206 /* AwaitExpression */:
-                return ts.updateAwait(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 207 /* PrefixUnaryExpression */:
-                return ts.updatePrefix(node, visitNode(node.operand, visitor, ts.isExpression));
-            case 208 /* PostfixUnaryExpression */:
-                return ts.updatePostfix(node, visitNode(node.operand, visitor, ts.isExpression));
-            case 209 /* BinaryExpression */:
-                return ts.updateBinary(node, visitNode(node.left, visitor, ts.isExpression), visitNode(node.right, visitor, ts.isExpression), visitNode(node.operatorToken, tokenVisitor, ts.isToken));
-            case 210 /* ConditionalExpression */:
-                return ts.updateConditional(node, visitNode(node.condition, visitor, ts.isExpression), visitNode(node.questionToken, tokenVisitor, ts.isToken), visitNode(node.whenTrue, visitor, ts.isExpression), visitNode(node.colonToken, tokenVisitor, ts.isToken), visitNode(node.whenFalse, visitor, ts.isExpression));
-            case 211 /* TemplateExpression */:
-                return ts.updateTemplateExpression(node, visitNode(node.head, visitor, ts.isTemplateHead), nodesVisitor(node.templateSpans, visitor, ts.isTemplateSpan));
-            case 212 /* YieldExpression */:
-                return ts.updateYield(node, visitNode(node.asteriskToken, tokenVisitor, ts.isToken), visitNode(node.expression, visitor, ts.isExpression));
-            case 213 /* SpreadElement */:
-                return ts.updateSpread(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 214 /* ClassExpression */:
-                return ts.updateClassExpression(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.heritageClauses, visitor, ts.isHeritageClause), nodesVisitor(node.members, visitor, ts.isClassElement));
-            case 216 /* ExpressionWithTypeArguments */:
-                return ts.updateExpressionWithTypeArguments(node, nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), visitNode(node.expression, visitor, ts.isExpression));
-            case 217 /* AsExpression */:
-                return ts.updateAsExpression(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.type, visitor, ts.isTypeNode));
-            case 218 /* NonNullExpression */:
-                return ts.updateNonNullExpression(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 219 /* MetaProperty */:
-                return ts.updateMetaProperty(node, visitNode(node.name, visitor, ts.isIdentifier));
+                    return factory.updateCallChain(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.questionDotToken, tokenVisitor, ts.isToken), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodesVisitor(node.arguments, visitor, ts.isExpression));
+                }
+                return factory.updateCallExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodesVisitor(node.arguments, visitor, ts.isExpression));
+            case 204 /* NewExpression */:
+                return factory.updateNewExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodesVisitor(node.arguments, visitor, ts.isExpression));
+            case 205 /* TaggedTemplateExpression */:
+                return factory.updateTaggedTemplateExpression(node, nodeVisitor(node.tag, visitor, ts.isExpression), visitNodes(node.typeArguments, visitor, ts.isExpression), nodeVisitor(node.template, visitor, ts.isTemplateLiteral));
+            case 206 /* TypeAssertionExpression */:
+                return factory.updateTypeAssertion(node, nodeVisitor(node.type, visitor, ts.isTypeNode), nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 207 /* ParenthesizedExpression */:
+                return factory.updateParenthesizedExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 208 /* FunctionExpression */:
+                return factory.updateFunctionExpression(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.asteriskToken, tokenVisitor, ts.isToken), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), nodeVisitor(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 209 /* ArrowFunction */:
+                return factory.updateArrowFunction(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), nodeVisitor(node.type, visitor, ts.isTypeNode), nodeVisitor(node.equalsGreaterThanToken, tokenVisitor, ts.isToken), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 210 /* DeleteExpression */:
+                return factory.updateDeleteExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 211 /* TypeOfExpression */:
+                return factory.updateTypeOfExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 212 /* VoidExpression */:
+                return factory.updateVoidExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 213 /* AwaitExpression */:
+                return factory.updateAwaitExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 214 /* PrefixUnaryExpression */:
+                return factory.updatePrefixUnaryExpression(node, nodeVisitor(node.operand, visitor, ts.isExpression));
+            case 215 /* PostfixUnaryExpression */:
+                return factory.updatePostfixUnaryExpression(node, nodeVisitor(node.operand, visitor, ts.isExpression));
+            case 216 /* BinaryExpression */:
+                return factory.updateBinaryExpression(node, nodeVisitor(node.left, visitor, ts.isExpression), nodeVisitor(node.operatorToken, tokenVisitor, ts.isToken), nodeVisitor(node.right, visitor, ts.isExpression));
+            case 217 /* ConditionalExpression */:
+                return factory.updateConditionalExpression(node, nodeVisitor(node.condition, visitor, ts.isExpression), nodeVisitor(node.questionToken, tokenVisitor, ts.isToken), nodeVisitor(node.whenTrue, visitor, ts.isExpression), nodeVisitor(node.colonToken, tokenVisitor, ts.isToken), nodeVisitor(node.whenFalse, visitor, ts.isExpression));
+            case 218 /* TemplateExpression */:
+                return factory.updateTemplateExpression(node, nodeVisitor(node.head, visitor, ts.isTemplateHead), nodesVisitor(node.templateSpans, visitor, ts.isTemplateSpan));
+            case 219 /* YieldExpression */:
+                return factory.updateYieldExpression(node, nodeVisitor(node.asteriskToken, tokenVisitor, ts.isToken), nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 220 /* SpreadElement */:
+                return factory.updateSpreadElement(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 221 /* ClassExpression */:
+                return factory.updateClassExpression(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.heritageClauses, visitor, ts.isHeritageClause), nodesVisitor(node.members, visitor, ts.isClassElement));
+            case 223 /* ExpressionWithTypeArguments */:
+                return factory.updateExpressionWithTypeArguments(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode));
+            case 224 /* AsExpression */:
+                return factory.updateAsExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 225 /* NonNullExpression */:
+                if (node.flags & 32 /* OptionalChain */) {
+                    return factory.updateNonNullChain(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+                }
+                return factory.updateNonNullExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 226 /* MetaProperty */:
+                return factory.updateMetaProperty(node, nodeVisitor(node.name, visitor, ts.isIdentifier));
             // Misc
-            case 221 /* TemplateSpan */:
-                return ts.updateTemplateSpan(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.literal, visitor, ts.isTemplateMiddleOrTemplateTail));
+            case 228 /* TemplateSpan */:
+                return factory.updateTemplateSpan(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.literal, visitor, ts.isTemplateMiddleOrTemplateTail));
             // Element
-            case 223 /* Block */:
-                return ts.updateBlock(node, nodesVisitor(node.statements, visitor, ts.isStatement));
-            case 225 /* VariableStatement */:
-                return ts.updateVariableStatement(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.declarationList, visitor, ts.isVariableDeclarationList));
-            case 226 /* ExpressionStatement */:
-                return ts.updateExpressionStatement(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 227 /* IfStatement */:
-                return ts.updateIf(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.thenStatement, visitor, ts.isStatement, ts.liftToBlock), visitNode(node.elseStatement, visitor, ts.isStatement, ts.liftToBlock));
-            case 228 /* DoStatement */:
-                return ts.updateDo(node, visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock), visitNode(node.expression, visitor, ts.isExpression));
-            case 229 /* WhileStatement */:
-                return ts.updateWhile(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-            case 230 /* ForStatement */:
-                return ts.updateFor(node, visitNode(node.initializer, visitor, ts.isForInitializer), visitNode(node.condition, visitor, ts.isExpression), visitNode(node.incrementor, visitor, ts.isExpression), visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-            case 231 /* ForInStatement */:
-                return ts.updateForIn(node, visitNode(node.initializer, visitor, ts.isForInitializer), visitNode(node.expression, visitor, ts.isExpression), visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-            case 232 /* ForOfStatement */:
-                return ts.updateForOf(node, visitNode(node.awaitModifier, tokenVisitor, ts.isToken), visitNode(node.initializer, visitor, ts.isForInitializer), visitNode(node.expression, visitor, ts.isExpression), visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-            case 233 /* ContinueStatement */:
-                return ts.updateContinue(node, visitNode(node.label, visitor, ts.isIdentifier));
-            case 234 /* BreakStatement */:
-                return ts.updateBreak(node, visitNode(node.label, visitor, ts.isIdentifier));
-            case 235 /* ReturnStatement */:
-                return ts.updateReturn(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 236 /* WithStatement */:
-                return ts.updateWith(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-            case 237 /* SwitchStatement */:
-                return ts.updateSwitch(node, visitNode(node.expression, visitor, ts.isExpression), visitNode(node.caseBlock, visitor, ts.isCaseBlock));
-            case 238 /* LabeledStatement */:
-                return ts.updateLabel(node, visitNode(node.label, visitor, ts.isIdentifier), visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-            case 239 /* ThrowStatement */:
-                return ts.updateThrow(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 240 /* TryStatement */:
-                return ts.updateTry(node, visitNode(node.tryBlock, visitor, ts.isBlock), visitNode(node.catchClause, visitor, ts.isCatchClause), visitNode(node.finallyBlock, visitor, ts.isBlock));
-            case 242 /* VariableDeclaration */:
-                return ts.updateTypeScriptVariableDeclaration(node, visitNode(node.name, visitor, ts.isBindingName), visitNode(node.exclamationToken, tokenVisitor, ts.isToken), visitNode(node.type, visitor, ts.isTypeNode), visitNode(node.initializer, visitor, ts.isExpression));
-            case 243 /* VariableDeclarationList */:
-                return ts.updateVariableDeclarationList(node, nodesVisitor(node.declarations, visitor, ts.isVariableDeclaration));
-            case 244 /* FunctionDeclaration */:
-                return ts.updateFunctionDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.asteriskToken, tokenVisitor, ts.isToken), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), visitNode(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context));
-            case 245 /* ClassDeclaration */:
-                return ts.updateClassDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.heritageClauses, visitor, ts.isHeritageClause), nodesVisitor(node.members, visitor, ts.isClassElement));
-            case 246 /* InterfaceDeclaration */:
-                return ts.updateInterfaceDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.heritageClauses, visitor, ts.isHeritageClause), nodesVisitor(node.members, visitor, ts.isTypeElement));
-            case 247 /* TypeAliasDeclaration */:
-                return ts.updateTypeAliasDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitNode(node.type, visitor, ts.isTypeNode));
-            case 248 /* EnumDeclaration */:
-                return ts.updateEnumDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), nodesVisitor(node.members, visitor, ts.isEnumMember));
-            case 249 /* ModuleDeclaration */:
-                return ts.updateModuleDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.body, visitor, ts.isModuleBody));
-            case 250 /* ModuleBlock */:
-                return ts.updateModuleBlock(node, nodesVisitor(node.statements, visitor, ts.isStatement));
-            case 251 /* CaseBlock */:
-                return ts.updateCaseBlock(node, nodesVisitor(node.clauses, visitor, ts.isCaseOrDefaultClause));
-            case 252 /* NamespaceExportDeclaration */:
-                return ts.updateNamespaceExportDeclaration(node, visitNode(node.name, visitor, ts.isIdentifier));
-            case 253 /* ImportEqualsDeclaration */:
-                return ts.updateImportEqualsDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.moduleReference, visitor, ts.isModuleReference));
-            case 254 /* ImportDeclaration */:
-                return ts.updateImportDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.importClause, visitor, ts.isImportClause), visitNode(node.moduleSpecifier, visitor, ts.isExpression));
-            case 255 /* ImportClause */:
-                return ts.updateImportClause(node, visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.namedBindings, visitor, ts.isNamedImportBindings), node.isTypeOnly);
-            case 256 /* NamespaceImport */:
-                return ts.updateNamespaceImport(node, visitNode(node.name, visitor, ts.isIdentifier));
-            case 262 /* NamespaceExport */:
-                return ts.updateNamespaceExport(node, visitNode(node.name, visitor, ts.isIdentifier));
-            case 257 /* NamedImports */:
-                return ts.updateNamedImports(node, nodesVisitor(node.elements, visitor, ts.isImportSpecifier));
-            case 258 /* ImportSpecifier */:
-                return ts.updateImportSpecifier(node, visitNode(node.propertyName, visitor, ts.isIdentifier), visitNode(node.name, visitor, ts.isIdentifier));
-            case 259 /* ExportAssignment */:
-                return ts.updateExportAssignment(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.expression, visitor, ts.isExpression));
-            case 260 /* ExportDeclaration */:
-                return ts.updateExportDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), visitNode(node.exportClause, visitor, ts.isNamedExportBindings), visitNode(node.moduleSpecifier, visitor, ts.isExpression), node.isTypeOnly);
-            case 261 /* NamedExports */:
-                return ts.updateNamedExports(node, nodesVisitor(node.elements, visitor, ts.isExportSpecifier));
-            case 263 /* ExportSpecifier */:
-                return ts.updateExportSpecifier(node, visitNode(node.propertyName, visitor, ts.isIdentifier), visitNode(node.name, visitor, ts.isIdentifier));
+            case 230 /* Block */:
+                return factory.updateBlock(node, nodesVisitor(node.statements, visitor, ts.isStatement));
+            case 232 /* VariableStatement */:
+                return factory.updateVariableStatement(node, nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.declarationList, visitor, ts.isVariableDeclarationList));
+            case 233 /* ExpressionStatement */:
+                return factory.updateExpressionStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 234 /* IfStatement */:
+                return factory.updateIfStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.thenStatement, visitor, ts.isStatement, factory.liftToBlock), nodeVisitor(node.elseStatement, visitor, ts.isStatement, factory.liftToBlock));
+            case 235 /* DoStatement */:
+                return factory.updateDoStatement(node, nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock), nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 236 /* WhileStatement */:
+                return factory.updateWhileStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+            case 237 /* ForStatement */:
+                return factory.updateForStatement(node, nodeVisitor(node.initializer, visitor, ts.isForInitializer), nodeVisitor(node.condition, visitor, ts.isExpression), nodeVisitor(node.incrementor, visitor, ts.isExpression), nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+            case 238 /* ForInStatement */:
+                return factory.updateForInStatement(node, nodeVisitor(node.initializer, visitor, ts.isForInitializer), nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+            case 239 /* ForOfStatement */:
+                return factory.updateForOfStatement(node, nodeVisitor(node.awaitModifier, tokenVisitor, ts.isToken), nodeVisitor(node.initializer, visitor, ts.isForInitializer), nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+            case 240 /* ContinueStatement */:
+                return factory.updateContinueStatement(node, nodeVisitor(node.label, visitor, ts.isIdentifier));
+            case 241 /* BreakStatement */:
+                return factory.updateBreakStatement(node, nodeVisitor(node.label, visitor, ts.isIdentifier));
+            case 242 /* ReturnStatement */:
+                return factory.updateReturnStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 243 /* WithStatement */:
+                return factory.updateWithStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+            case 244 /* SwitchStatement */:
+                return factory.updateSwitchStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodeVisitor(node.caseBlock, visitor, ts.isCaseBlock));
+            case 245 /* LabeledStatement */:
+                return factory.updateLabeledStatement(node, nodeVisitor(node.label, visitor, ts.isIdentifier), nodeVisitor(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+            case 246 /* ThrowStatement */:
+                return factory.updateThrowStatement(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 247 /* TryStatement */:
+                return factory.updateTryStatement(node, nodeVisitor(node.tryBlock, visitor, ts.isBlock), nodeVisitor(node.catchClause, visitor, ts.isCatchClause), nodeVisitor(node.finallyBlock, visitor, ts.isBlock));
+            case 249 /* VariableDeclaration */:
+                return factory.updateVariableDeclaration(node, nodeVisitor(node.name, visitor, ts.isBindingName), nodeVisitor(node.exclamationToken, tokenVisitor, ts.isToken), nodeVisitor(node.type, visitor, ts.isTypeNode), nodeVisitor(node.initializer, visitor, ts.isExpression));
+            case 250 /* VariableDeclarationList */:
+                return factory.updateVariableDeclarationList(node, nodesVisitor(node.declarations, visitor, ts.isVariableDeclaration));
+            case 251 /* FunctionDeclaration */:
+                return factory.updateFunctionDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.asteriskToken, tokenVisitor, ts.isToken), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), visitParameterList(node.parameters, visitor, context, nodesVisitor), nodeVisitor(node.type, visitor, ts.isTypeNode), visitFunctionBody(node.body, visitor, context, nodeVisitor));
+            case 252 /* ClassDeclaration */:
+                return factory.updateClassDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.heritageClauses, visitor, ts.isHeritageClause), nodesVisitor(node.members, visitor, ts.isClassElement));
+            case 253 /* InterfaceDeclaration */:
+                return factory.updateInterfaceDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodesVisitor(node.heritageClauses, visitor, ts.isHeritageClause), nodesVisitor(node.members, visitor, ts.isTypeElement));
+            case 254 /* TypeAliasDeclaration */:
+                return factory.updateTypeAliasDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.typeParameters, visitor, ts.isTypeParameterDeclaration), nodeVisitor(node.type, visitor, ts.isTypeNode));
+            case 255 /* EnumDeclaration */:
+                return factory.updateEnumDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodesVisitor(node.members, visitor, ts.isEnumMember));
+            case 256 /* ModuleDeclaration */:
+                return factory.updateModuleDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodeVisitor(node.body, visitor, ts.isModuleBody));
+            case 257 /* ModuleBlock */:
+                return factory.updateModuleBlock(node, nodesVisitor(node.statements, visitor, ts.isStatement));
+            case 258 /* CaseBlock */:
+                return factory.updateCaseBlock(node, nodesVisitor(node.clauses, visitor, ts.isCaseOrDefaultClause));
+            case 259 /* NamespaceExportDeclaration */:
+                return factory.updateNamespaceExportDeclaration(node, nodeVisitor(node.name, visitor, ts.isIdentifier));
+            case 260 /* ImportEqualsDeclaration */:
+                return factory.updateImportEqualsDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.name, visitor, ts.isIdentifier), nodeVisitor(node.moduleReference, visitor, ts.isModuleReference));
+            case 261 /* ImportDeclaration */:
+                return factory.updateImportDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.importClause, visitor, ts.isImportClause), nodeVisitor(node.moduleSpecifier, visitor, ts.isExpression));
+            case 262 /* ImportClause */:
+                return factory.updateImportClause(node, node.isTypeOnly, nodeVisitor(node.name, visitor, ts.isIdentifier), nodeVisitor(node.namedBindings, visitor, ts.isNamedImportBindings));
+            case 263 /* NamespaceImport */:
+                return factory.updateNamespaceImport(node, nodeVisitor(node.name, visitor, ts.isIdentifier));
+            case 269 /* NamespaceExport */:
+                return factory.updateNamespaceExport(node, nodeVisitor(node.name, visitor, ts.isIdentifier));
+            case 264 /* NamedImports */:
+                return factory.updateNamedImports(node, nodesVisitor(node.elements, visitor, ts.isImportSpecifier));
+            case 265 /* ImportSpecifier */:
+                return factory.updateImportSpecifier(node, nodeVisitor(node.propertyName, visitor, ts.isIdentifier), nodeVisitor(node.name, visitor, ts.isIdentifier));
+            case 266 /* ExportAssignment */:
+                return factory.updateExportAssignment(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 267 /* ExportDeclaration */:
+                return factory.updateExportDeclaration(node, nodesVisitor(node.decorators, visitor, ts.isDecorator), nodesVisitor(node.modifiers, visitor, ts.isModifier), node.isTypeOnly, nodeVisitor(node.exportClause, visitor, ts.isNamedExportBindings), nodeVisitor(node.moduleSpecifier, visitor, ts.isExpression));
+            case 268 /* NamedExports */:
+                return factory.updateNamedExports(node, nodesVisitor(node.elements, visitor, ts.isExportSpecifier));
+            case 270 /* ExportSpecifier */:
+                return factory.updateExportSpecifier(node, nodeVisitor(node.propertyName, visitor, ts.isIdentifier), nodeVisitor(node.name, visitor, ts.isIdentifier));
             // Module references
-            case 265 /* ExternalModuleReference */:
-                return ts.updateExternalModuleReference(node, visitNode(node.expression, visitor, ts.isExpression));
+            case 272 /* ExternalModuleReference */:
+                return factory.updateExternalModuleReference(node, nodeVisitor(node.expression, visitor, ts.isExpression));
             // JSX
-            case 266 /* JsxElement */:
-                return ts.updateJsxElement(node, visitNode(node.openingElement, visitor, ts.isJsxOpeningElement), nodesVisitor(node.children, visitor, ts.isJsxChild), visitNode(node.closingElement, visitor, ts.isJsxClosingElement));
-            case 267 /* JsxSelfClosingElement */:
-                return ts.updateJsxSelfClosingElement(node, visitNode(node.tagName, visitor, ts.isJsxTagNameExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), visitNode(node.attributes, visitor, ts.isJsxAttributes));
-            case 268 /* JsxOpeningElement */:
-                return ts.updateJsxOpeningElement(node, visitNode(node.tagName, visitor, ts.isJsxTagNameExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), visitNode(node.attributes, visitor, ts.isJsxAttributes));
-            case 269 /* JsxClosingElement */:
-                return ts.updateJsxClosingElement(node, visitNode(node.tagName, visitor, ts.isJsxTagNameExpression));
-            case 270 /* JsxFragment */:
-                return ts.updateJsxFragment(node, visitNode(node.openingFragment, visitor, ts.isJsxOpeningFragment), nodesVisitor(node.children, visitor, ts.isJsxChild), visitNode(node.closingFragment, visitor, ts.isJsxClosingFragment));
-            case 273 /* JsxAttribute */:
-                return ts.updateJsxAttribute(node, visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.initializer, visitor, ts.isStringLiteralOrJsxExpression));
-            case 274 /* JsxAttributes */:
-                return ts.updateJsxAttributes(node, nodesVisitor(node.properties, visitor, ts.isJsxAttributeLike));
-            case 275 /* JsxSpreadAttribute */:
-                return ts.updateJsxSpreadAttribute(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 276 /* JsxExpression */:
-                return ts.updateJsxExpression(node, visitNode(node.expression, visitor, ts.isExpression));
+            case 273 /* JsxElement */:
+                return factory.updateJsxElement(node, nodeVisitor(node.openingElement, visitor, ts.isJsxOpeningElement), nodesVisitor(node.children, visitor, ts.isJsxChild), nodeVisitor(node.closingElement, visitor, ts.isJsxClosingElement));
+            case 274 /* JsxSelfClosingElement */:
+                return factory.updateJsxSelfClosingElement(node, nodeVisitor(node.tagName, visitor, ts.isJsxTagNameExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodeVisitor(node.attributes, visitor, ts.isJsxAttributes));
+            case 275 /* JsxOpeningElement */:
+                return factory.updateJsxOpeningElement(node, nodeVisitor(node.tagName, visitor, ts.isJsxTagNameExpression), nodesVisitor(node.typeArguments, visitor, ts.isTypeNode), nodeVisitor(node.attributes, visitor, ts.isJsxAttributes));
+            case 276 /* JsxClosingElement */:
+                return factory.updateJsxClosingElement(node, nodeVisitor(node.tagName, visitor, ts.isJsxTagNameExpression));
+            case 277 /* JsxFragment */:
+                return factory.updateJsxFragment(node, nodeVisitor(node.openingFragment, visitor, ts.isJsxOpeningFragment), nodesVisitor(node.children, visitor, ts.isJsxChild), nodeVisitor(node.closingFragment, visitor, ts.isJsxClosingFragment));
+            case 280 /* JsxAttribute */:
+                return factory.updateJsxAttribute(node, nodeVisitor(node.name, visitor, ts.isIdentifier), nodeVisitor(node.initializer, visitor, ts.isStringLiteralOrJsxExpression));
+            case 281 /* JsxAttributes */:
+                return factory.updateJsxAttributes(node, nodesVisitor(node.properties, visitor, ts.isJsxAttributeLike));
+            case 282 /* JsxSpreadAttribute */:
+                return factory.updateJsxSpreadAttribute(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 283 /* JsxExpression */:
+                return factory.updateJsxExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
             // Clauses
-            case 277 /* CaseClause */:
-                return ts.updateCaseClause(node, visitNode(node.expression, visitor, ts.isExpression), nodesVisitor(node.statements, visitor, ts.isStatement));
-            case 278 /* DefaultClause */:
-                return ts.updateDefaultClause(node, nodesVisitor(node.statements, visitor, ts.isStatement));
-            case 279 /* HeritageClause */:
-                return ts.updateHeritageClause(node, nodesVisitor(node.types, visitor, ts.isExpressionWithTypeArguments));
-            case 280 /* CatchClause */:
-                return ts.updateCatchClause(node, visitNode(node.variableDeclaration, visitor, ts.isVariableDeclaration), visitNode(node.block, visitor, ts.isBlock));
+            case 284 /* CaseClause */:
+                return factory.updateCaseClause(node, nodeVisitor(node.expression, visitor, ts.isExpression), nodesVisitor(node.statements, visitor, ts.isStatement));
+            case 285 /* DefaultClause */:
+                return factory.updateDefaultClause(node, nodesVisitor(node.statements, visitor, ts.isStatement));
+            case 286 /* HeritageClause */:
+                return factory.updateHeritageClause(node, nodesVisitor(node.types, visitor, ts.isExpressionWithTypeArguments));
+            case 287 /* CatchClause */:
+                return factory.updateCatchClause(node, nodeVisitor(node.variableDeclaration, visitor, ts.isVariableDeclaration), nodeVisitor(node.block, visitor, ts.isBlock));
             // Property assignments
-            case 281 /* PropertyAssignment */:
-                return ts.updatePropertyAssignment(node, visitNode(node.name, visitor, ts.isPropertyName), visitNode(node.initializer, visitor, ts.isExpression));
-            case 282 /* ShorthandPropertyAssignment */:
-                return ts.updateShorthandPropertyAssignment(node, visitNode(node.name, visitor, ts.isIdentifier), visitNode(node.objectAssignmentInitializer, visitor, ts.isExpression));
-            case 283 /* SpreadAssignment */:
-                return ts.updateSpreadAssignment(node, visitNode(node.expression, visitor, ts.isExpression));
+            case 288 /* PropertyAssignment */:
+                return factory.updatePropertyAssignment(node, nodeVisitor(node.name, visitor, ts.isPropertyName), nodeVisitor(node.initializer, visitor, ts.isExpression));
+            case 289 /* ShorthandPropertyAssignment */:
+                return factory.updateShorthandPropertyAssignment(node, nodeVisitor(node.name, visitor, ts.isIdentifier), nodeVisitor(node.objectAssignmentInitializer, visitor, ts.isExpression));
+            case 290 /* SpreadAssignment */:
+                return factory.updateSpreadAssignment(node, nodeVisitor(node.expression, visitor, ts.isExpression));
             // Enum
-            case 284 /* EnumMember */:
-                return ts.updateEnumMember(node, visitNode(node.name, visitor, ts.isPropertyName), visitNode(node.initializer, visitor, ts.isExpression));
+            case 291 /* EnumMember */:
+                return factory.updateEnumMember(node, nodeVisitor(node.name, visitor, ts.isPropertyName), nodeVisitor(node.initializer, visitor, ts.isExpression));
             // Top-level nodes
-            case 290 /* SourceFile */:
-                return ts.updateSourceFileNode(node, visitLexicalEnvironment(node.statements, visitor, context));
+            case 297 /* SourceFile */:
+                return factory.updateSourceFile(node, visitLexicalEnvironment(node.statements, visitor, context));
             // Transformation nodes
-            case 326 /* PartiallyEmittedExpression */:
-                return ts.updatePartiallyEmittedExpression(node, visitNode(node.expression, visitor, ts.isExpression));
-            case 327 /* CommaListExpression */:
-                return ts.updateCommaList(node, nodesVisitor(node.elements, visitor, ts.isExpression));
+            case 336 /* PartiallyEmittedExpression */:
+                return factory.updatePartiallyEmittedExpression(node, nodeVisitor(node.expression, visitor, ts.isExpression));
+            case 337 /* CommaListExpression */:
+                return factory.updateCommaListExpression(node, nodesVisitor(node.elements, visitor, ts.isExpression));
             default:
                 // No need to visit nodes with no children.
                 return node;
@@ -75247,602 +80830,6 @@ var ts;
 })(ts || (ts = {}));
 /* @internal */
 var ts;
-(function (ts) {
-    function reduceNode(node, f, initial) {
-        return node ? f(initial, node) : initial;
-    }
-    function reduceNodeArray(nodes, f, initial) {
-        return nodes ? f(initial, nodes) : initial;
-    }
-    /**
-     * Similar to `reduceLeft`, performs a reduction against each child of a node.
-     * NOTE: Unlike `forEachChild`, this does *not* visit every node.
-     *
-     * @param node The node containing the children to reduce.
-     * @param initial The initial value to supply to the reduction.
-     * @param f The callback function
-     */
-    function reduceEachChild(node, initial, cbNode, cbNodeArray) {
-        if (node === undefined) {
-            return initial;
-        }
-        var reduceNodes = cbNodeArray ? reduceNodeArray : ts.reduceLeft;
-        var cbNodes = cbNodeArray || cbNode;
-        var kind = node.kind;
-        // No need to visit nodes with no children.
-        if ((kind > 0 /* FirstToken */ && kind <= 152 /* LastToken */)) {
-            return initial;
-        }
-        // We do not yet support types.
-        if ((kind >= 168 /* TypePredicate */ && kind <= 187 /* LiteralType */)) {
-            return initial;
-        }
-        var result = initial;
-        switch (node.kind) {
-            // Leaf nodes
-            case 222 /* SemicolonClassElement */:
-            case 224 /* EmptyStatement */:
-            case 215 /* OmittedExpression */:
-            case 241 /* DebuggerStatement */:
-            case 325 /* NotEmittedStatement */:
-                // No need to visit nodes with no children.
-                break;
-            // Names
-            case 153 /* QualifiedName */:
-                result = reduceNode(node.left, cbNode, result);
-                result = reduceNode(node.right, cbNode, result);
-                break;
-            case 154 /* ComputedPropertyName */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            // Signature elements
-            case 156 /* Parameter */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            case 157 /* Decorator */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            // Type member
-            case 158 /* PropertySignature */:
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.questionToken, cbNode, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            case 159 /* PropertyDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            case 161 /* MethodDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.typeParameters, cbNodes, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 162 /* Constructor */:
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 163 /* GetAccessor */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 164 /* SetAccessor */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            // Binding patterns
-            case 189 /* ObjectBindingPattern */:
-            case 190 /* ArrayBindingPattern */:
-                result = reduceNodes(node.elements, cbNodes, result);
-                break;
-            case 191 /* BindingElement */:
-                result = reduceNode(node.propertyName, cbNode, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            // Expression
-            case 192 /* ArrayLiteralExpression */:
-                result = reduceNodes(node.elements, cbNodes, result);
-                break;
-            case 193 /* ObjectLiteralExpression */:
-                result = reduceNodes(node.properties, cbNodes, result);
-                break;
-            case 194 /* PropertyAccessExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.name, cbNode, result);
-                break;
-            case 195 /* ElementAccessExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.argumentExpression, cbNode, result);
-                break;
-            case 196 /* CallExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNodes(node.typeArguments, cbNodes, result);
-                result = reduceNodes(node.arguments, cbNodes, result);
-                break;
-            case 197 /* NewExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNodes(node.typeArguments, cbNodes, result);
-                result = reduceNodes(node.arguments, cbNodes, result);
-                break;
-            case 198 /* TaggedTemplateExpression */:
-                result = reduceNode(node.tag, cbNode, result);
-                result = reduceNodes(node.typeArguments, cbNodes, result);
-                result = reduceNode(node.template, cbNode, result);
-                break;
-            case 199 /* TypeAssertionExpression */:
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 201 /* FunctionExpression */:
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.typeParameters, cbNodes, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 202 /* ArrowFunction */:
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNodes(node.typeParameters, cbNodes, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 200 /* ParenthesizedExpression */:
-            case 203 /* DeleteExpression */:
-            case 204 /* TypeOfExpression */:
-            case 205 /* VoidExpression */:
-            case 206 /* AwaitExpression */:
-            case 212 /* YieldExpression */:
-            case 213 /* SpreadElement */:
-            case 218 /* NonNullExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 207 /* PrefixUnaryExpression */:
-            case 208 /* PostfixUnaryExpression */:
-                result = reduceNode(node.operand, cbNode, result);
-                break;
-            case 209 /* BinaryExpression */:
-                result = reduceNode(node.left, cbNode, result);
-                result = reduceNode(node.right, cbNode, result);
-                break;
-            case 210 /* ConditionalExpression */:
-                result = reduceNode(node.condition, cbNode, result);
-                result = reduceNode(node.whenTrue, cbNode, result);
-                result = reduceNode(node.whenFalse, cbNode, result);
-                break;
-            case 211 /* TemplateExpression */:
-                result = reduceNode(node.head, cbNode, result);
-                result = reduceNodes(node.templateSpans, cbNodes, result);
-                break;
-            case 214 /* ClassExpression */:
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.typeParameters, cbNodes, result);
-                result = reduceNodes(node.heritageClauses, cbNodes, result);
-                result = reduceNodes(node.members, cbNodes, result);
-                break;
-            case 216 /* ExpressionWithTypeArguments */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNodes(node.typeArguments, cbNodes, result);
-                break;
-            case 217 /* AsExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.type, cbNode, result);
-                break;
-            // Misc
-            case 221 /* TemplateSpan */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.literal, cbNode, result);
-                break;
-            // Element
-            case 223 /* Block */:
-                result = reduceNodes(node.statements, cbNodes, result);
-                break;
-            case 225 /* VariableStatement */:
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.declarationList, cbNode, result);
-                break;
-            case 226 /* ExpressionStatement */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 227 /* IfStatement */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.thenStatement, cbNode, result);
-                result = reduceNode(node.elseStatement, cbNode, result);
-                break;
-            case 228 /* DoStatement */:
-                result = reduceNode(node.statement, cbNode, result);
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 229 /* WhileStatement */:
-            case 236 /* WithStatement */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.statement, cbNode, result);
-                break;
-            case 230 /* ForStatement */:
-                result = reduceNode(node.initializer, cbNode, result);
-                result = reduceNode(node.condition, cbNode, result);
-                result = reduceNode(node.incrementor, cbNode, result);
-                result = reduceNode(node.statement, cbNode, result);
-                break;
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
-                result = reduceNode(node.initializer, cbNode, result);
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.statement, cbNode, result);
-                break;
-            case 235 /* ReturnStatement */:
-            case 239 /* ThrowStatement */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 237 /* SwitchStatement */:
-                result = reduceNode(node.expression, cbNode, result);
-                result = reduceNode(node.caseBlock, cbNode, result);
-                break;
-            case 238 /* LabeledStatement */:
-                result = reduceNode(node.label, cbNode, result);
-                result = reduceNode(node.statement, cbNode, result);
-                break;
-            case 240 /* TryStatement */:
-                result = reduceNode(node.tryBlock, cbNode, result);
-                result = reduceNode(node.catchClause, cbNode, result);
-                result = reduceNode(node.finallyBlock, cbNode, result);
-                break;
-            case 242 /* VariableDeclaration */:
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            case 243 /* VariableDeclarationList */:
-                result = reduceNodes(node.declarations, cbNodes, result);
-                break;
-            case 244 /* FunctionDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.typeParameters, cbNodes, result);
-                result = reduceNodes(node.parameters, cbNodes, result);
-                result = reduceNode(node.type, cbNode, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 245 /* ClassDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.typeParameters, cbNodes, result);
-                result = reduceNodes(node.heritageClauses, cbNodes, result);
-                result = reduceNodes(node.members, cbNodes, result);
-                break;
-            case 248 /* EnumDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNodes(node.members, cbNodes, result);
-                break;
-            case 249 /* ModuleDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.body, cbNode, result);
-                break;
-            case 250 /* ModuleBlock */:
-                result = reduceNodes(node.statements, cbNodes, result);
-                break;
-            case 251 /* CaseBlock */:
-                result = reduceNodes(node.clauses, cbNodes, result);
-                break;
-            case 253 /* ImportEqualsDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.moduleReference, cbNode, result);
-                break;
-            case 254 /* ImportDeclaration */:
-                result = reduceNodes(node.decorators, cbNodes, result);
-                result = reduceNodes(node.modifiers, cbNodes, result);
-                result = reduceNode(node.importClause, cbNode, result);
-                result = reduceNode(node.moduleSpecifier, cbNode, result);
-                break;
-            case 255 /* ImportClause */:
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.namedBindings, cbNode, result);
-                break;
-            case 256 /* NamespaceImport */:
-                result = reduceNode(node.name, cbNode, result);
-                break;
-            case 262 /* NamespaceExport */:
-                result = reduceNode(node.name, cbNode, result);
-                break;
-            case 257 /* NamedImports */:
-            case 261 /* NamedExports */:
-                result = reduceNodes(node.elements, cbNodes, result);
-                break;
-            case 258 /* ImportSpecifier */:
-            case 263 /* ExportSpecifier */:
-                result = reduceNode(node.propertyName, cbNode, result);
-                result = reduceNode(node.name, cbNode, result);
-                break;
-            case 259 /* ExportAssignment */:
-                result = ts.reduceLeft(node.decorators, cbNode, result);
-                result = ts.reduceLeft(node.modifiers, cbNode, result);
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 260 /* ExportDeclaration */:
-                result = ts.reduceLeft(node.decorators, cbNode, result);
-                result = ts.reduceLeft(node.modifiers, cbNode, result);
-                result = reduceNode(node.exportClause, cbNode, result);
-                result = reduceNode(node.moduleSpecifier, cbNode, result);
-                break;
-            // Module references
-            case 265 /* ExternalModuleReference */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            // JSX
-            case 266 /* JsxElement */:
-                result = reduceNode(node.openingElement, cbNode, result);
-                result = ts.reduceLeft(node.children, cbNode, result);
-                result = reduceNode(node.closingElement, cbNode, result);
-                break;
-            case 270 /* JsxFragment */:
-                result = reduceNode(node.openingFragment, cbNode, result);
-                result = ts.reduceLeft(node.children, cbNode, result);
-                result = reduceNode(node.closingFragment, cbNode, result);
-                break;
-            case 267 /* JsxSelfClosingElement */:
-            case 268 /* JsxOpeningElement */:
-                result = reduceNode(node.tagName, cbNode, result);
-                result = reduceNodes(node.typeArguments, cbNode, result);
-                result = reduceNode(node.attributes, cbNode, result);
-                break;
-            case 274 /* JsxAttributes */:
-                result = reduceNodes(node.properties, cbNodes, result);
-                break;
-            case 269 /* JsxClosingElement */:
-                result = reduceNode(node.tagName, cbNode, result);
-                break;
-            case 273 /* JsxAttribute */:
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            case 275 /* JsxSpreadAttribute */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 276 /* JsxExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            // Clauses
-            case 277 /* CaseClause */:
-                result = reduceNode(node.expression, cbNode, result);
-            // falls through
-            case 278 /* DefaultClause */:
-                result = reduceNodes(node.statements, cbNodes, result);
-                break;
-            case 279 /* HeritageClause */:
-                result = reduceNodes(node.types, cbNodes, result);
-                break;
-            case 280 /* CatchClause */:
-                result = reduceNode(node.variableDeclaration, cbNode, result);
-                result = reduceNode(node.block, cbNode, result);
-                break;
-            // Property assignments
-            case 281 /* PropertyAssignment */:
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            case 282 /* ShorthandPropertyAssignment */:
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.objectAssignmentInitializer, cbNode, result);
-                break;
-            case 283 /* SpreadAssignment */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            // Enum
-            case 284 /* EnumMember */:
-                result = reduceNode(node.name, cbNode, result);
-                result = reduceNode(node.initializer, cbNode, result);
-                break;
-            // Top-level nodes
-            case 290 /* SourceFile */:
-                result = reduceNodes(node.statements, cbNodes, result);
-                break;
-            // Transformation nodes
-            case 326 /* PartiallyEmittedExpression */:
-                result = reduceNode(node.expression, cbNode, result);
-                break;
-            case 327 /* CommaListExpression */:
-                result = reduceNodes(node.elements, cbNodes, result);
-                break;
-            default:
-                break;
-        }
-        return result;
-    }
-    ts.reduceEachChild = reduceEachChild;
-    function findSpanEnd(array, test, start) {
-        var i = start;
-        while (i < array.length && test(array[i])) {
-            i++;
-        }
-        return i;
-    }
-    function mergeLexicalEnvironment(statements, declarations) {
-        if (!ts.some(declarations)) {
-            return statements;
-        }
-        // When we merge new lexical statements into an existing statement list, we merge them in the following manner:
-        //
-        // Given:
-        //
-        // | Left                               | Right                               |
-        // |------------------------------------|-------------------------------------|
-        // | [standard prologues (left)]        | [standard prologues (right)]        |
-        // | [hoisted functions (left)]         | [hoisted functions (right)]         |
-        // | [hoisted variables (left)]         | [hoisted variables (right)]         |
-        // | [lexical init statements (left)]   | [lexical init statements (right)]   |
-        // | [other statements (left)]          |                                     |
-        //
-        // The resulting statement list will be:
-        //
-        // | Result                              |
-        // |-------------------------------------|
-        // | [standard prologues (right)]        |
-        // | [standard prologues (left)]         |
-        // | [hoisted functions (right)]         |
-        // | [hoisted functions (left)]          |
-        // | [hoisted variables (right)]         |
-        // | [hoisted variables (left)]          |
-        // | [lexical init statements (right)]   |
-        // | [lexical init statements (left)]    |
-        // | [other statements (left)]           |
-        //
-        // NOTE: It is expected that new lexical init statements must be evaluated before existing lexical init statements,
-        // as the prior transformation may depend on the evaluation of the lexical init statements to be in the correct state.
-        // find standard prologues on left in the following order: standard directives, hoisted functions, hoisted variables, other custom
-        var leftStandardPrologueEnd = findSpanEnd(statements, ts.isPrologueDirective, 0);
-        var leftHoistedFunctionsEnd = findSpanEnd(statements, ts.isHoistedFunction, leftStandardPrologueEnd);
-        var leftHoistedVariablesEnd = findSpanEnd(statements, ts.isHoistedVariableStatement, leftHoistedFunctionsEnd);
-        // find standard prologues on right in the following order: standard directives, hoisted functions, hoisted variables, other custom
-        var rightStandardPrologueEnd = findSpanEnd(declarations, ts.isPrologueDirective, 0);
-        var rightHoistedFunctionsEnd = findSpanEnd(declarations, ts.isHoistedFunction, rightStandardPrologueEnd);
-        var rightHoistedVariablesEnd = findSpanEnd(declarations, ts.isHoistedVariableStatement, rightHoistedFunctionsEnd);
-        var rightCustomPrologueEnd = findSpanEnd(declarations, ts.isCustomPrologue, rightHoistedVariablesEnd);
-        ts.Debug.assert(rightCustomPrologueEnd === declarations.length, "Expected declarations to be valid standard or custom prologues");
-        // splice prologues from the right into the left. We do this in reverse order
-        // so that we don't need to recompute the index on the left when we insert items.
-        var left = ts.isNodeArray(statements) ? statements.slice() : statements;
-        // splice other custom prologues from right into left
-        if (rightCustomPrologueEnd > rightHoistedVariablesEnd) {
-            left.splice.apply(left, __spreadArrays([leftHoistedVariablesEnd, 0], declarations.slice(rightHoistedVariablesEnd, rightCustomPrologueEnd)));
-        }
-        // splice hoisted variables from right into left
-        if (rightHoistedVariablesEnd > rightHoistedFunctionsEnd) {
-            left.splice.apply(left, __spreadArrays([leftHoistedFunctionsEnd, 0], declarations.slice(rightHoistedFunctionsEnd, rightHoistedVariablesEnd)));
-        }
-        // splice hoisted functions from right into left
-        if (rightHoistedFunctionsEnd > rightStandardPrologueEnd) {
-            left.splice.apply(left, __spreadArrays([leftStandardPrologueEnd, 0], declarations.slice(rightStandardPrologueEnd, rightHoistedFunctionsEnd)));
-        }
-        // splice standard prologues from right into left (that are not already in left)
-        if (rightStandardPrologueEnd > 0) {
-            if (leftStandardPrologueEnd === 0) {
-                left.splice.apply(left, __spreadArrays([0, 0], declarations.slice(0, rightStandardPrologueEnd)));
-            }
-            else {
-                var leftPrologues = ts.createMap();
-                for (var i = 0; i < leftStandardPrologueEnd; i++) {
-                    var leftPrologue = statements[i];
-                    leftPrologues.set(leftPrologue.expression.text, true);
-                }
-                for (var i = rightStandardPrologueEnd - 1; i >= 0; i--) {
-                    var rightPrologue = declarations[i];
-                    if (!leftPrologues.has(rightPrologue.expression.text)) {
-                        left.unshift(rightPrologue);
-                    }
-                }
-            }
-        }
-        if (ts.isNodeArray(statements)) {
-            return ts.setTextRange(ts.createNodeArray(left, statements.hasTrailingComma), statements);
-        }
-        return statements;
-    }
-    ts.mergeLexicalEnvironment = mergeLexicalEnvironment;
-    /**
-     * Lifts a NodeArray containing only Statement nodes to a block.
-     *
-     * @param nodes The NodeArray.
-     */
-    function liftToBlock(nodes) {
-        ts.Debug.assert(ts.every(nodes, ts.isStatement), "Cannot lift nodes to a Block.");
-        return ts.singleOrUndefined(nodes) || ts.createBlock(nodes);
-    }
-    ts.liftToBlock = liftToBlock;
-    /**
-     * Aggregates the TransformFlags for a Node and its subtree.
-     */
-    function aggregateTransformFlags(node) {
-        aggregateTransformFlagsForNode(node);
-        return node;
-    }
-    ts.aggregateTransformFlags = aggregateTransformFlags;
-    /**
-     * Aggregates the TransformFlags for a Node and its subtree. The flags for the subtree are
-     * computed first, then the transform flags for the current node are computed from the subtree
-     * flags and the state of the current node. Finally, the transform flags of the node are
-     * returned, excluding any flags that should not be included in its parent node's subtree
-     * flags.
-     */
-    function aggregateTransformFlagsForNode(node) {
-        if (node === undefined) {
-            return 0 /* None */;
-        }
-        if (node.transformFlags & 536870912 /* HasComputedFlags */) {
-            return node.transformFlags & ~ts.getTransformFlagsSubtreeExclusions(node.kind);
-        }
-        var subtreeFlags = aggregateTransformFlagsForSubtree(node);
-        return ts.computeTransformFlagsForNode(node, subtreeFlags);
-    }
-    function aggregateTransformFlagsForNodeArray(nodes) {
-        if (nodes === undefined) {
-            return 0 /* None */;
-        }
-        var subtreeFlags = 0 /* None */;
-        var nodeArrayFlags = 0 /* None */;
-        for (var _i = 0, nodes_3 = nodes; _i < nodes_3.length; _i++) {
-            var node = nodes_3[_i];
-            subtreeFlags |= aggregateTransformFlagsForNode(node);
-            nodeArrayFlags |= node.transformFlags & ~536870912 /* HasComputedFlags */;
-        }
-        nodes.transformFlags = nodeArrayFlags | 536870912 /* HasComputedFlags */;
-        return subtreeFlags;
-    }
-    /**
-     * Aggregates the transform flags for the subtree of a node.
-     */
-    function aggregateTransformFlagsForSubtree(node) {
-        // We do not transform ambient declarations or types, so there is no need to
-        // recursively aggregate transform flags.
-        if (ts.hasModifier(node, 2 /* Ambient */) || (ts.isTypeNode(node) && node.kind !== 216 /* ExpressionWithTypeArguments */)) {
-            return 0 /* None */;
-        }
-        // Aggregate the transform flags of each child.
-        return reduceEachChild(node, 0 /* None */, aggregateTransformFlagsForChildNode, aggregateTransformFlagsForChildNodes);
-    }
-    /**
-     * Aggregates the TransformFlags of a child node with the TransformFlags of its
-     * siblings.
-     */
-    function aggregateTransformFlagsForChildNode(transformFlags, node) {
-        return transformFlags | aggregateTransformFlagsForNode(node);
-    }
-    function aggregateTransformFlagsForChildNodes(transformFlags, nodes) {
-        return transformFlags | aggregateTransformFlagsForNodeArray(nodes);
-    }
-})(ts || (ts = {}));
-/* @internal */
-var ts;
 (function (ts) {
     function createSourceMapGenerator(host, file, sourceRoot, sourcesDirectoryPath, generatorOptions) {
         var _a = generatorOptions.extendedDiagnostics
@@ -75851,7 +80838,7 @@ var ts;
         // Current source map file and its index in the sources list
         var rawSources = [];
         var sources = [];
-        var sourceToSourceIndexMap = ts.createMap();
+        var sourceToSourceIndexMap = new ts.Map();
         var sourcesContent;
         var names = [];
         var nameToNameIndexMap;
@@ -75914,7 +80901,7 @@ var ts;
         function addName(name) {
             enter();
             if (!nameToNameIndexMap)
-                nameToNameIndexMap = ts.createMap();
+                nameToNameIndexMap = new ts.Map();
             var nameIndex = nameToNameIndexMap.get(name);
             if (nameIndex === undefined) {
                 nameIndex = names.length;
@@ -76365,7 +81352,7 @@ var ts;
         var generatedAbsoluteFilePath = ts.getNormalizedAbsolutePath(map.file, mapDirectory);
         var generatedFile = host.getSourceFileLike(generatedAbsoluteFilePath);
         var sourceFileAbsolutePaths = map.sources.map(function (source) { return ts.getNormalizedAbsolutePath(source, sourceRoot); });
-        var sourceToSourceIndexMap = ts.createMapFromEntries(sourceFileAbsolutePaths.map(function (source, i) { return [host.getCanonicalFileName(source), i]; }));
+        var sourceToSourceIndexMap = new ts.Map(sourceFileAbsolutePaths.map(function (source, i) { return [host.getCanonicalFileName(source), i]; }));
         var decodedMappings;
         var generatedMappings;
         var sourceMappings;
@@ -76495,13 +81482,13 @@ var ts;
     function isNamedDefaultReference(e) {
         return e.propertyName !== undefined && e.propertyName.escapedText === "default" /* Default */;
     }
-    function chainBundle(transformSourceFile) {
+    function chainBundle(context, transformSourceFile) {
         return transformSourceFileOrBundle;
         function transformSourceFileOrBundle(node) {
-            return node.kind === 290 /* SourceFile */ ? transformSourceFile(node) : transformBundle(node);
+            return node.kind === 297 /* SourceFile */ ? transformSourceFile(node) : transformBundle(node);
         }
         function transformBundle(node) {
-            return ts.createBundle(ts.map(node.sourceFiles, transformSourceFile), node.prepends);
+            return context.factory.createBundle(ts.map(node.sourceFiles, transformSourceFile), node.prepends);
         }
     }
     ts.chainBundle = chainBundle;
@@ -76535,11 +81522,11 @@ var ts;
         return !getImportNeedsImportStarHelper(node) && (ts.isDefaultImport(node) || (!!node.importClause && ts.isNamedImports(node.importClause.namedBindings) && containsDefaultReference(node.importClause.namedBindings))); // TODO: GH#18217
     }
     ts.getImportNeedsImportDefaultHelper = getImportNeedsImportDefaultHelper;
-    function collectExternalModuleInfo(sourceFile, resolver, compilerOptions) {
+    function collectExternalModuleInfo(context, sourceFile, resolver, compilerOptions) {
         var externalImports = [];
         var exportSpecifiers = ts.createMultiMap();
         var exportedBindings = [];
-        var uniqueExports = ts.createMap();
+        var uniqueExports = new ts.Map();
         var exportedNames;
         var hasExportDefault = false;
         var exportEquals;
@@ -76549,7 +81536,7 @@ var ts;
         for (var _i = 0, _a = sourceFile.statements; _i < _a.length; _i++) {
             var node = _a[_i];
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     // import "mod"
                     // import x from "mod"
                     // import * as x from "mod"
@@ -76562,13 +81549,13 @@ var ts;
                         hasImportDefault = true;
                     }
                     break;
-                case 253 /* ImportEqualsDeclaration */:
-                    if (node.moduleReference.kind === 265 /* ExternalModuleReference */) {
+                case 260 /* ImportEqualsDeclaration */:
+                    if (node.moduleReference.kind === 272 /* ExternalModuleReference */) {
                         // import x = require("mod")
                         externalImports.push(node);
                     }
                     break;
-                case 260 /* ExportDeclaration */:
+                case 267 /* ExportDeclaration */:
                     if (node.moduleSpecifier) {
                         if (!node.exportClause) {
                             // export * from "mod"
@@ -76579,46 +81566,46 @@ var ts;
                             // export * as ns from "mod"
                             // export { x, y } from "mod"
                             externalImports.push(node);
+                            if (ts.isNamedExports(node.exportClause)) {
+                                addExportedNamesForExportDeclaration(node);
+                            }
+                            else {
+                                var name = node.exportClause.name;
+                                if (!uniqueExports.get(ts.idText(name))) {
+                                    multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(node), name);
+                                    uniqueExports.set(ts.idText(name), true);
+                                    exportedNames = ts.append(exportedNames, name);
+                                }
+                                // we use the same helpers for `export * as ns` as we do for `import * as ns`
+                                hasImportStar = true;
+                            }
                         }
                     }
                     else {
                         // export { x, y }
-                        for (var _b = 0, _c = ts.cast(node.exportClause, ts.isNamedExports).elements; _b < _c.length; _b++) {
-                            var specifier = _c[_b];
-                            if (!uniqueExports.get(ts.idText(specifier.name))) {
-                                var name = specifier.propertyName || specifier.name;
-                                exportSpecifiers.add(ts.idText(name), specifier);
-                                var decl = resolver.getReferencedImportDeclaration(name)
-                                    || resolver.getReferencedValueDeclaration(name);
-                                if (decl) {
-                                    multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(decl), specifier.name);
-                                }
-                                uniqueExports.set(ts.idText(specifier.name), true);
-                                exportedNames = ts.append(exportedNames, specifier.name);
-                            }
-                        }
+                        addExportedNamesForExportDeclaration(node);
                     }
                     break;
-                case 259 /* ExportAssignment */:
+                case 266 /* ExportAssignment */:
                     if (node.isExportEquals && !exportEquals) {
                         // export = x
                         exportEquals = node;
                     }
                     break;
-                case 225 /* VariableStatement */:
-                    if (ts.hasModifier(node, 1 /* Export */)) {
-                        for (var _d = 0, _e = node.declarationList.declarations; _d < _e.length; _d++) {
-                            var decl = _e[_d];
+                case 232 /* VariableStatement */:
+                    if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                        for (var _b = 0, _c = node.declarationList.declarations; _b < _c.length; _b++) {
+                            var decl = _c[_b];
                             exportedNames = collectExportedVariableInfo(decl, uniqueExports, exportedNames);
                         }
                     }
                     break;
-                case 244 /* FunctionDeclaration */:
-                    if (ts.hasModifier(node, 1 /* Export */)) {
-                        if (ts.hasModifier(node, 512 /* Default */)) {
+                case 251 /* FunctionDeclaration */:
+                    if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                        if (ts.hasSyntacticModifier(node, 512 /* Default */)) {
                             // export default function() { }
                             if (!hasExportDefault) {
-                                multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(node), ts.getDeclarationName(node));
+                                multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(node), context.factory.getDeclarationName(node));
                                 hasExportDefault = true;
                             }
                         }
@@ -76633,12 +81620,12 @@ var ts;
                         }
                     }
                     break;
-                case 245 /* ClassDeclaration */:
-                    if (ts.hasModifier(node, 1 /* Export */)) {
-                        if (ts.hasModifier(node, 512 /* Default */)) {
+                case 252 /* ClassDeclaration */:
+                    if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                        if (ts.hasSyntacticModifier(node, 512 /* Default */)) {
                             // export default class { }
                             if (!hasExportDefault) {
-                                multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(node), ts.getDeclarationName(node));
+                                multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(node), context.factory.getDeclarationName(node));
                                 hasExportDefault = true;
                             }
                         }
@@ -76655,11 +81642,29 @@ var ts;
                     break;
             }
         }
-        var externalHelpersImportDeclaration = ts.createExternalHelpersImportDeclarationIfNeeded(sourceFile, compilerOptions, hasExportStarsToExportValues, hasImportStar, hasImportDefault);
+        var externalHelpersImportDeclaration = ts.createExternalHelpersImportDeclarationIfNeeded(context.factory, context.getEmitHelperFactory(), sourceFile, compilerOptions, hasExportStarsToExportValues, hasImportStar, hasImportDefault);
         if (externalHelpersImportDeclaration) {
             externalImports.unshift(externalHelpersImportDeclaration);
         }
         return { externalImports: externalImports, exportSpecifiers: exportSpecifiers, exportEquals: exportEquals, hasExportStarsToExportValues: hasExportStarsToExportValues, exportedBindings: exportedBindings, exportedNames: exportedNames, externalHelpersImportDeclaration: externalHelpersImportDeclaration };
+        function addExportedNamesForExportDeclaration(node) {
+            for (var _i = 0, _a = ts.cast(node.exportClause, ts.isNamedExports).elements; _i < _a.length; _i++) {
+                var specifier = _a[_i];
+                if (!uniqueExports.get(ts.idText(specifier.name))) {
+                    var name = specifier.propertyName || specifier.name;
+                    if (!node.moduleSpecifier) {
+                        exportSpecifiers.add(ts.idText(name), specifier);
+                    }
+                    var decl = resolver.getReferencedImportDeclaration(name)
+                        || resolver.getReferencedValueDeclaration(name);
+                    if (decl) {
+                        multiMapSparseArrayAdd(exportedBindings, getOriginalNodeId(decl), specifier.name);
+                    }
+                    uniqueExports.set(ts.idText(specifier.name), true);
+                    exportedNames = ts.append(exportedNames, specifier.name);
+                }
+            }
+        }
     }
     ts.collectExternalModuleInfo = collectExternalModuleInfo;
     function collectExportedVariableInfo(decl, uniqueExports, exportedNames) {
@@ -76715,7 +81720,7 @@ var ts;
     ts.isSimpleInlineableExpression = isSimpleInlineableExpression;
     function isCompoundAssignment(kind) {
         return kind >= 63 /* FirstCompoundAssignment */
-            && kind <= 74 /* LastCompoundAssignment */;
+            && kind <= 77 /* LastCompoundAssignment */;
     }
     ts.isCompoundAssignment = isCompoundAssignment;
     function getNonAssignmentOperatorForCompoundAssignment(kind) {
@@ -76731,7 +81736,10 @@ var ts;
             case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */: return 49 /* GreaterThanGreaterThanGreaterThanToken */;
             case 72 /* AmpersandEqualsToken */: return 50 /* AmpersandToken */;
             case 73 /* BarEqualsToken */: return 51 /* BarToken */;
-            case 74 /* CaretEqualsToken */: return 52 /* CaretToken */;
+            case 77 /* CaretEqualsToken */: return 52 /* CaretToken */;
+            case 74 /* BarBarEqualsToken */: return 56 /* BarBarToken */;
+            case 75 /* AmpersandAmpersandEqualsToken */: return 55 /* AmpersandAmpersandToken */;
+            case 76 /* QuestionQuestionEqualsToken */: return 60 /* QuestionQuestionToken */;
         }
     }
     ts.getNonAssignmentOperatorForCompoundAssignment = getNonAssignmentOperatorForCompoundAssignment;
@@ -76743,11 +81751,11 @@ var ts;
      * @param visitor The visitor to apply to each node added to the result array.
      * @returns index of the statement that follows super call
      */
-    function addPrologueDirectivesAndInitialSuperCall(ctor, result, visitor) {
+    function addPrologueDirectivesAndInitialSuperCall(factory, ctor, result, visitor) {
         if (ctor.body) {
             var statements = ctor.body.statements;
             // add prologue directives to the list (if any)
-            var index = ts.addPrologue(result, statements, /*ensureUseStrict*/ false, visitor);
+            var index = factory.copyPrologue(statements, result, /*ensureUseStrict*/ false, visitor);
             if (index === statements.length) {
                 // list contains nothing but prologue directives (or empty) - exit
                 return index;
@@ -76764,32 +81772,6 @@ var ts;
         return 0;
     }
     ts.addPrologueDirectivesAndInitialSuperCall = addPrologueDirectivesAndInitialSuperCall;
-    /**
-     * @param input Template string input strings
-     * @param args Names which need to be made file-level unique
-     */
-    function helperString(input) {
-        var args = [];
-        for (var _i = 1; _i < arguments.length; _i++) {
-            args[_i - 1] = arguments[_i];
-        }
-        return function (uniqueName) {
-            var result = "";
-            for (var i = 0; i < args.length; i++) {
-                result += input[i];
-                result += uniqueName(args[i]);
-            }
-            result += input[input.length - 1];
-            return result;
-        };
-    }
-    ts.helperString = helperString;
-    /**
-     * Gets all the static or all the instance property declarations of a class
-     *
-     * @param node The class node.
-     * @param isStatic A value indicating whether to get properties from the static or instance side of the class.
-     */
     function getProperties(node, requireInitializer, isStatic) {
         return ts.filter(node.members, function (m) { return isInitializedOrStaticProperty(m, requireInitializer, isStatic); });
     }
@@ -76812,7 +81794,7 @@ var ts;
      * @param isStatic A value indicating whether the member should be a static or instance member.
      */
     function isInitializedProperty(member) {
-        return member.kind === 159 /* PropertyDeclaration */
+        return member.kind === 163 /* PropertyDeclaration */
             && member.initializer !== undefined;
     }
     ts.isInitializedProperty = isInitializedProperty;
@@ -76859,8 +81841,8 @@ var ts;
             hoistTempVariables: true,
             emitExpression: emitExpression,
             emitBindingOrAssignment: emitBindingOrAssignment,
-            createArrayBindingOrAssignmentPattern: makeArrayAssignmentPattern,
-            createObjectBindingOrAssignmentPattern: makeObjectAssignmentPattern,
+            createArrayBindingOrAssignmentPattern: function (elements) { return makeArrayAssignmentPattern(context.factory, elements); },
+            createObjectBindingOrAssignmentPattern: function (elements) { return makeObjectAssignmentPattern(context.factory, elements); },
             createArrayBindingOrAssignmentElement: makeAssignmentElement,
             visitor: visitor
         };
@@ -76898,16 +81880,15 @@ var ts;
             }
             expressions.push(value);
         }
-        return ts.aggregateTransformFlags(ts.inlineExpressions(expressions)) || ts.createOmittedExpression();
+        return context.factory.inlineExpressions(expressions) || context.factory.createOmittedExpression();
         function emitExpression(expression) {
-            ts.aggregateTransformFlags(expression);
             expressions = ts.append(expressions, expression);
         }
         function emitBindingOrAssignment(target, value, location, original) {
             ts.Debug.assertNode(target, createAssignmentCallback ? ts.isIdentifier : ts.isExpression);
             var expression = createAssignmentCallback
                 ? createAssignmentCallback(target, value, location)
-                : ts.setTextRange(ts.createAssignment(ts.visitNode(target, visitor, ts.isExpression), value), location);
+                : ts.setTextRange(context.factory.createAssignment(ts.visitNode(target, visitor, ts.isExpression), value), location);
             expression.original = original;
             emitExpression(expression);
         }
@@ -76967,9 +81948,9 @@ var ts;
             hoistTempVariables: hoistTempVariables,
             emitExpression: emitExpression,
             emitBindingOrAssignment: emitBindingOrAssignment,
-            createArrayBindingOrAssignmentPattern: makeArrayBindingPattern,
-            createObjectBindingOrAssignmentPattern: makeObjectBindingPattern,
-            createArrayBindingOrAssignmentElement: makeBindingElement,
+            createArrayBindingOrAssignmentPattern: function (elements) { return makeArrayBindingPattern(context.factory, elements); },
+            createObjectBindingOrAssignmentPattern: function (elements) { return makeObjectBindingPattern(context.factory, elements); },
+            createArrayBindingOrAssignmentElement: function (name) { return makeBindingElement(context.factory, name); },
             visitor: visitor
         };
         if (ts.isVariableDeclaration(node)) {
@@ -76979,32 +81960,32 @@ var ts;
                 // If the right-hand value of the assignment is also an assignment target then
                 // we need to cache the right-hand value.
                 initializer = ensureIdentifier(flattenContext, initializer, /*reuseIdentifierExpressions*/ false, initializer);
-                node = ts.updateVariableDeclaration(node, node.name, node.type, initializer);
+                node = context.factory.updateVariableDeclaration(node, node.name, /*exclamationToken*/ undefined, /*type*/ undefined, initializer);
             }
         }
         flattenBindingOrAssignmentElement(flattenContext, node, rval, node, skipInitializer);
         if (pendingExpressions) {
-            var temp = ts.createTempVariable(/*recordTempVariable*/ undefined);
+            var temp = context.factory.createTempVariable(/*recordTempVariable*/ undefined);
             if (hoistTempVariables) {
-                var value = ts.inlineExpressions(pendingExpressions);
+                var value = context.factory.inlineExpressions(pendingExpressions);
                 pendingExpressions = undefined;
                 emitBindingOrAssignment(temp, value, /*location*/ undefined, /*original*/ undefined);
             }
             else {
                 context.hoistVariableDeclaration(temp);
                 var pendingDeclaration = ts.last(pendingDeclarations);
-                pendingDeclaration.pendingExpressions = ts.append(pendingDeclaration.pendingExpressions, ts.createAssignment(temp, pendingDeclaration.value));
+                pendingDeclaration.pendingExpressions = ts.append(pendingDeclaration.pendingExpressions, context.factory.createAssignment(temp, pendingDeclaration.value));
                 ts.addRange(pendingDeclaration.pendingExpressions, pendingExpressions);
                 pendingDeclaration.value = temp;
             }
         }
         for (var _i = 0, pendingDeclarations_1 = pendingDeclarations; _i < pendingDeclarations_1.length; _i++) {
             var _a = pendingDeclarations_1[_i], pendingExpressions_1 = _a.pendingExpressions, name = _a.name, value = _a.value, location = _a.location, original = _a.original;
-            var variable = ts.createVariableDeclaration(name, 
-            /*type*/ undefined, pendingExpressions_1 ? ts.inlineExpressions(ts.append(pendingExpressions_1, value)) : value);
+            var variable = context.factory.createVariableDeclaration(name, 
+            /*exclamationToken*/ undefined, 
+            /*type*/ undefined, pendingExpressions_1 ? context.factory.inlineExpressions(ts.append(pendingExpressions_1, value)) : value);
             variable.original = original;
             ts.setTextRange(variable, location);
-            ts.aggregateTransformFlags(variable);
             declarations.push(variable);
         }
         return declarations;
@@ -77014,7 +81995,7 @@ var ts;
         function emitBindingOrAssignment(target, value, location, original) {
             ts.Debug.assertNode(target, ts.isBindingName);
             if (pendingExpressions) {
-                value = ts.inlineExpressions(ts.append(pendingExpressions, value));
+                value = context.factory.inlineExpressions(ts.append(pendingExpressions, value));
                 pendingExpressions = undefined;
             }
             pendingDeclarations.push({ pendingExpressions: pendingExpressions, name: target, value: value, location: location, original: original });
@@ -77032,18 +82013,27 @@ var ts;
      * for the element.
      */
     function flattenBindingOrAssignmentElement(flattenContext, element, value, location, skipInitializer) {
+        var bindingTarget = ts.getTargetOfBindingOrAssignmentElement(element); // TODO: GH#18217
         if (!skipInitializer) {
             var initializer = ts.visitNode(ts.getInitializerOfBindingOrAssignmentElement(element), flattenContext.visitor, ts.isExpression);
             if (initializer) {
                 // Combine value and initializer
-                value = value ? createDefaultValueCheck(flattenContext, value, initializer, location) : initializer;
+                if (value) {
+                    value = createDefaultValueCheck(flattenContext, value, initializer, location);
+                    // If 'value' is not a simple expression, it could contain side-effecting code that should evaluate before an object or array binding pattern.
+                    if (!ts.isSimpleInlineableExpression(initializer) && ts.isBindingOrAssignmentPattern(bindingTarget)) {
+                        value = ensureIdentifier(flattenContext, value, /*reuseIdentifierExpressions*/ true, location);
+                    }
+                }
+                else {
+                    value = initializer;
+                }
             }
             else if (!value) {
                 // Use 'void 0' in absence of value and initializer
-                value = ts.createVoidZero();
+                value = flattenContext.context.factory.createVoidZero();
             }
         }
-        var bindingTarget = ts.getTargetOfBindingOrAssignmentElement(element); // TODO: GH#18217
         if (ts.isObjectBindingOrAssignmentPattern(bindingTarget)) {
             flattenObjectBindingOrAssignmentPattern(flattenContext, element, bindingTarget, value, location);
         }
@@ -77103,7 +82093,7 @@ var ts;
                     flattenContext.emitBindingOrAssignment(flattenContext.createObjectBindingOrAssignmentPattern(bindingElements), value, location, pattern);
                     bindingElements = undefined;
                 }
-                var rhsValue = createRestCall(flattenContext.context, value, elements, computedTempVariables, pattern); // TODO: GH#18217
+                var rhsValue = flattenContext.context.getEmitHelperFactory().createRestHelper(value, elements, computedTempVariables, pattern);
                 flattenBindingOrAssignmentElement(flattenContext, element, rhsValue, element);
             }
         }
@@ -77125,9 +82115,9 @@ var ts;
         var numElements = elements.length;
         if (flattenContext.level < 1 /* ObjectRest */ && flattenContext.downlevelIteration) {
             // Read the elements of the iterable into an array
-            value = ensureIdentifier(flattenContext, ts.createReadHelper(flattenContext.context, value, numElements > 0 && ts.getRestIndicatorOfBindingOrAssignmentElement(elements[numElements - 1])
+            value = ensureIdentifier(flattenContext, ts.setTextRange(flattenContext.context.getEmitHelperFactory().createReadHelper(value, numElements > 0 && ts.getRestIndicatorOfBindingOrAssignmentElement(elements[numElements - 1])
                 ? undefined
-                : numElements, location), 
+                : numElements), location), 
             /*reuseIdentifierExpressions*/ false, location);
         }
         else if (numElements !== 1 && (flattenContext.level < 1 /* ObjectRest */ || numElements === 0)
@@ -77148,8 +82138,9 @@ var ts;
             if (flattenContext.level >= 1 /* ObjectRest */) {
                 // If an array pattern contains an ObjectRest, we must cache the result so that we
                 // can perform the ObjectRest destructuring in a different declaration
-                if (element.transformFlags & 16384 /* ContainsObjectRestOrSpread */) {
-                    var temp = ts.createTempVariable(/*recordTempVariable*/ undefined);
+                if (element.transformFlags & 16384 /* ContainsObjectRestOrSpread */ || flattenContext.hasTransformedPriorElement && !isSimpleBindingOrAssignmentElement(element)) {
+                    flattenContext.hasTransformedPriorElement = true;
+                    var temp = flattenContext.context.factory.createTempVariable(/*recordTempVariable*/ undefined);
                     if (flattenContext.hoistTempVariables) {
                         flattenContext.context.hoistVariableDeclaration(temp);
                     }
@@ -77164,11 +82155,11 @@ var ts;
                 continue;
             }
             else if (!ts.getRestIndicatorOfBindingOrAssignmentElement(element)) {
-                var rhsValue = ts.createElementAccess(value, i);
+                var rhsValue = flattenContext.context.factory.createElementAccessExpression(value, i);
                 flattenBindingOrAssignmentElement(flattenContext, element, rhsValue, /*location*/ element);
             }
             else if (i === numElements - 1) {
-                var rhsValue = ts.createArraySlice(value, i);
+                var rhsValue = flattenContext.context.factory.createArraySliceCall(value, i);
                 flattenBindingOrAssignmentElement(flattenContext, element, rhsValue, /*location*/ element);
             }
         }
@@ -77182,6 +82173,20 @@ var ts;
             }
         }
     }
+    function isSimpleBindingOrAssignmentElement(element) {
+        var target = ts.getTargetOfBindingOrAssignmentElement(element);
+        if (!target || ts.isOmittedExpression(target))
+            return true;
+        var propertyName = ts.tryGetPropertyNameOfBindingOrAssignmentElement(element);
+        if (propertyName && !ts.isPropertyNameLiteral(propertyName))
+            return false;
+        var initializer = ts.getInitializerOfBindingOrAssignmentElement(element);
+        if (initializer && !ts.isSimpleInlineableExpression(initializer))
+            return false;
+        if (ts.isBindingOrAssignmentPattern(target))
+            return ts.every(ts.getElementsOfBindingOrAssignmentPattern(target), isSimpleBindingOrAssignmentElement);
+        return ts.isIdentifier(target);
+    }
     /**
      * Creates an expression used to provide a default value if a value is `undefined` at runtime.
      *
@@ -77192,7 +82197,7 @@ var ts;
      */
     function createDefaultValueCheck(flattenContext, value, defaultValue, location) {
         value = ensureIdentifier(flattenContext, value, /*reuseIdentifierExpressions*/ true, location);
-        return ts.createConditional(ts.createTypeCheck(value, "undefined"), defaultValue, value);
+        return flattenContext.context.factory.createConditionalExpression(flattenContext.context.factory.createTypeCheck(value, "undefined"), /*questionToken*/ undefined, defaultValue, /*colonToken*/ undefined, value);
     }
     /**
      * Creates either a PropertyAccessExpression or an ElementAccessExpression for the
@@ -77207,16 +82212,15 @@ var ts;
     function createDestructuringPropertyAccess(flattenContext, value, propertyName) {
         if (ts.isComputedPropertyName(propertyName)) {
             var argumentExpression = ensureIdentifier(flattenContext, ts.visitNode(propertyName.expression, flattenContext.visitor), /*reuseIdentifierExpressions*/ false, /*location*/ propertyName);
-            return ts.createElementAccess(value, argumentExpression);
+            return flattenContext.context.factory.createElementAccessExpression(value, argumentExpression);
         }
         else if (ts.isStringOrNumericLiteralLike(propertyName)) {
-            var argumentExpression = ts.getSynthesizedClone(propertyName);
-            argumentExpression.text = argumentExpression.text;
-            return ts.createElementAccess(value, argumentExpression);
+            var argumentExpression = ts.factory.cloneNode(propertyName);
+            return flattenContext.context.factory.createElementAccessExpression(value, argumentExpression);
         }
         else {
-            var name = ts.createIdentifier(ts.idText(propertyName));
-            return ts.createPropertyAccess(value, name);
+            var name = flattenContext.context.factory.createIdentifier(ts.idText(propertyName));
+            return flattenContext.context.factory.createPropertyAccessExpression(value, name);
         }
     }
     /**
@@ -77235,10 +82239,10 @@ var ts;
             return value;
         }
         else {
-            var temp = ts.createTempVariable(/*recordTempVariable*/ undefined);
+            var temp = flattenContext.context.factory.createTempVariable(/*recordTempVariable*/ undefined);
             if (flattenContext.hoistTempVariables) {
                 flattenContext.context.hoistVariableDeclaration(temp);
-                flattenContext.emitExpression(ts.setTextRange(ts.createAssignment(temp, value), location));
+                flattenContext.emitExpression(ts.setTextRange(flattenContext.context.factory.createAssignment(temp, value), location));
             }
             else {
                 flattenContext.emitBindingOrAssignment(temp, value, location, /*original*/ undefined);
@@ -77246,59 +82250,26 @@ var ts;
             return temp;
         }
     }
-    function makeArrayBindingPattern(elements) {
+    function makeArrayBindingPattern(factory, elements) {
         ts.Debug.assertEachNode(elements, ts.isArrayBindingElement);
-        return ts.createArrayBindingPattern(elements);
+        return factory.createArrayBindingPattern(elements);
     }
-    function makeArrayAssignmentPattern(elements) {
-        return ts.createArrayLiteral(ts.map(elements, ts.convertToArrayAssignmentElement));
+    function makeArrayAssignmentPattern(factory, elements) {
+        return factory.createArrayLiteralExpression(ts.map(elements, factory.converters.convertToArrayAssignmentElement));
     }
-    function makeObjectBindingPattern(elements) {
+    function makeObjectBindingPattern(factory, elements) {
         ts.Debug.assertEachNode(elements, ts.isBindingElement);
-        return ts.createObjectBindingPattern(elements);
+        return factory.createObjectBindingPattern(elements);
     }
-    function makeObjectAssignmentPattern(elements) {
-        return ts.createObjectLiteral(ts.map(elements, ts.convertToObjectAssignmentElement));
+    function makeObjectAssignmentPattern(factory, elements) {
+        return factory.createObjectLiteralExpression(ts.map(elements, factory.converters.convertToObjectAssignmentElement));
     }
-    function makeBindingElement(name) {
-        return ts.createBindingElement(/*dotDotDotToken*/ undefined, /*propertyName*/ undefined, name);
+    function makeBindingElement(factory, name) {
+        return factory.createBindingElement(/*dotDotDotToken*/ undefined, /*propertyName*/ undefined, name);
     }
     function makeAssignmentElement(name) {
         return name;
     }
-    ts.restHelper = {
-        name: "typescript:rest",
-        importName: "__rest",
-        scoped: false,
-        text: "\n            var __rest = (this && this.__rest) || function (s, e) {\n                var t = {};\n                for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\n                    t[p] = s[p];\n                if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\n                    for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\n                        if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\n                            t[p[i]] = s[p[i]];\n                    }\n                return t;\n            };"
-    };
-    /** Given value: o, propName: p, pattern: { a, b, ...p } from the original statement
-     * `{ a, b, ...p } = o`, create `p = __rest(o, ["a", "b"]);`
-     */
-    function createRestCall(context, value, elements, computedTempVariables, location) {
-        context.requestEmitHelper(ts.restHelper);
-        var propertyNames = [];
-        var computedTempVariableOffset = 0;
-        for (var i = 0; i < elements.length - 1; i++) {
-            var propertyName = ts.getPropertyNameOfBindingOrAssignmentElement(elements[i]);
-            if (propertyName) {
-                if (ts.isComputedPropertyName(propertyName)) {
-                    var temp = computedTempVariables[computedTempVariableOffset];
-                    computedTempVariableOffset++;
-                    // typeof _tmp === "symbol" ? _tmp : _tmp + ""
-                    propertyNames.push(ts.createConditional(ts.createTypeCheck(temp, "symbol"), temp, ts.createAdd(temp, ts.createLiteral(""))));
-                }
-                else {
-                    propertyNames.push(ts.createLiteral(propertyName));
-                }
-            }
-        }
-        return ts.createCall(ts.getUnscopedHelperName("__rest"), 
-        /*typeArguments*/ undefined, [
-            value,
-            ts.setTextRange(ts.createArrayLiteral(propertyNames), location)
-        ]);
-    }
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -77318,8 +82289,9 @@ var ts;
         var cookedStrings = [];
         var rawStrings = [];
         var template = node.template;
-        if (level === ProcessLevel.LiftRestriction && !ts.hasInvalidEscape(template))
-            return node;
+        if (level === ProcessLevel.LiftRestriction && !ts.hasInvalidEscape(template)) {
+            return ts.visitEachChild(node, visitor, context);
+        }
         if (ts.isNoSubstitutionTemplateLiteral(template)) {
             cookedStrings.push(createTemplateCooked(template));
             rawStrings.push(getRawLiteral(template, currentSourceFile));
@@ -77334,23 +82306,23 @@ var ts;
                 templateArguments.push(ts.visitNode(templateSpan.expression, visitor, ts.isExpression));
             }
         }
-        var helperCall = createTemplateObjectHelper(context, ts.createArrayLiteral(cookedStrings), ts.createArrayLiteral(rawStrings));
+        var helperCall = context.getEmitHelperFactory().createTemplateObjectHelper(ts.factory.createArrayLiteralExpression(cookedStrings), ts.factory.createArrayLiteralExpression(rawStrings));
         // Create a variable to cache the template object if we're in a module.
         // Do not do this in the global scope, as any variable we currently generate could conflict with
         // variables from outside of the current compilation. In the future, we can revisit this behavior.
         if (ts.isExternalModule(currentSourceFile)) {
-            var tempVar = ts.createUniqueName("templateObject");
+            var tempVar = ts.factory.createUniqueName("templateObject");
             recordTaggedTemplateString(tempVar);
-            templateArguments[0] = ts.createLogicalOr(tempVar, ts.createAssignment(tempVar, helperCall));
+            templateArguments[0] = ts.factory.createLogicalOr(tempVar, ts.factory.createAssignment(tempVar, helperCall));
         }
         else {
             templateArguments[0] = helperCall;
         }
-        return ts.createCall(tag, /*typeArguments*/ undefined, templateArguments);
+        return ts.factory.createCallExpression(tag, /*typeArguments*/ undefined, templateArguments);
     }
     ts.processTaggedTemplateExpression = processTaggedTemplateExpression;
     function createTemplateCooked(template) {
-        return template.templateFlags ? ts.createIdentifier("undefined") : ts.createLiteral(template.text);
+        return template.templateFlags ? ts.factory.createVoidZero() : ts.factory.createStringLiteral(template.text);
     }
     /**
      * Creates an ES5 compatible literal from an ES6 template literal.
@@ -77375,23 +82347,8 @@ var ts;
         // ES6 Spec 11.8.6.1 - Static Semantics of TV's and TRV's
         // <CR><LF> and <CR> LineTerminatorSequences are normalized to <LF> for both TV and TRV.
         text = text.replace(/\r\n?/g, "\n");
-        return ts.setTextRange(ts.createLiteral(text), node);
-    }
-    function createTemplateObjectHelper(context, cooked, raw) {
-        context.requestEmitHelper(ts.templateObjectHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__makeTemplateObject"), 
-        /*typeArguments*/ undefined, [
-            cooked,
-            raw
-        ]);
+        return ts.setTextRange(ts.factory.createStringLiteral(text), node);
     }
-    ts.templateObjectHelper = {
-        name: "typescript:makeTemplateObject",
-        importName: "__makeTemplateObject",
-        scoped: false,
-        priority: 0,
-        text: "\n            var __makeTemplateObject = (this && this.__makeTemplateObject) || function (cooked, raw) {\n                if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\n                return cooked;\n            };"
-    };
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -77426,7 +82383,7 @@ var ts;
         ClassFacts[ClassFacts["IsExported"] = 56] = "IsExported";
     })(ClassFacts || (ClassFacts = {}));
     function transformTypeScript(context) {
-        var startLexicalEnvironment = context.startLexicalEnvironment, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory, startLexicalEnvironment = context.startLexicalEnvironment, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var resolver = context.getEmitResolver();
         var compilerOptions = context.getCompilerOptions();
         var strictNullChecks = ts.getStrictOptionValue(compilerOptions, "strictNullChecks");
@@ -77439,8 +82396,8 @@ var ts;
         context.onEmitNode = onEmitNode;
         context.onSubstituteNode = onSubstituteNode;
         // Enable substitution for property/element access to emit const enum values.
-        context.enableSubstitution(194 /* PropertyAccessExpression */);
-        context.enableSubstitution(195 /* ElementAccessExpression */);
+        context.enableSubstitution(201 /* PropertyAccessExpression */);
+        context.enableSubstitution(202 /* ElementAccessExpression */);
         // These variables contain state that changes as we descend into the tree.
         var currentSourceFile;
         var currentNamespace;
@@ -77466,14 +82423,14 @@ var ts;
         var applicableSubstitutions;
         return transformSourceFileOrBundle;
         function transformSourceFileOrBundle(node) {
-            if (node.kind === 291 /* Bundle */) {
+            if (node.kind === 298 /* Bundle */) {
                 return transformBundle(node);
             }
             return transformSourceFile(node);
         }
         function transformBundle(node) {
-            return ts.createBundle(node.sourceFiles.map(transformSourceFile), ts.mapDefined(node.prepends, function (prepend) {
-                if (prepend.kind === 293 /* InputFiles */) {
+            return factory.createBundle(node.sourceFiles.map(transformSourceFile), ts.mapDefined(node.prepends, function (prepend) {
+                if (prepend.kind === 300 /* InputFiles */) {
                     return ts.createUnparsedSourceFile(prepend, "js");
                 }
                 return prepend;
@@ -77524,17 +82481,17 @@ var ts;
          */
         function onBeforeVisitNode(node) {
             switch (node.kind) {
-                case 290 /* SourceFile */:
-                case 251 /* CaseBlock */:
-                case 250 /* ModuleBlock */:
-                case 223 /* Block */:
+                case 297 /* SourceFile */:
+                case 258 /* CaseBlock */:
+                case 257 /* ModuleBlock */:
+                case 230 /* Block */:
                     currentLexicalScope = node;
                     currentNameScope = undefined;
                     currentScopeFirstDeclarationsOfName = undefined;
                     break;
-                case 245 /* ClassDeclaration */:
-                case 244 /* FunctionDeclaration */:
-                    if (ts.hasModifier(node, 2 /* Ambient */)) {
+                case 252 /* ClassDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                    if (ts.hasSyntacticModifier(node, 2 /* Ambient */)) {
                         break;
                     }
                     // Record these declarations provided that they have a name.
@@ -77545,7 +82502,7 @@ var ts;
                         // These nodes should always have names unless they are default-exports;
                         // however, class declaration parsing allows for undefined names, so syntactically invalid
                         // programs may also have an undefined name.
-                        ts.Debug.assert(node.kind === 245 /* ClassDeclaration */ || ts.hasModifier(node, 512 /* Default */));
+                        ts.Debug.assert(node.kind === 252 /* ClassDeclaration */ || ts.hasSyntacticModifier(node, 512 /* Default */));
                     }
                     if (ts.isClassDeclaration(node)) {
                         // XXX: should probably also cover interfaces and type aliases that can have type variables?
@@ -77588,16 +82545,16 @@ var ts;
          */
         function sourceElementVisitorWorker(node) {
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
-                case 253 /* ImportEqualsDeclaration */:
-                case 259 /* ExportAssignment */:
-                case 260 /* ExportDeclaration */:
-                    return visitEllidableStatement(node);
+                case 261 /* ImportDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
+                case 266 /* ExportAssignment */:
+                case 267 /* ExportDeclaration */:
+                    return visitElidableStatement(node);
                 default:
                     return visitorWorker(node);
             }
         }
-        function visitEllidableStatement(node) {
+        function visitElidableStatement(node) {
             var parsed = ts.getParseTreeNode(node);
             if (parsed !== node) {
                 // If the node has been transformed by a `before` transformer, perform no ellision on it
@@ -77612,13 +82569,13 @@ var ts;
                 return node;
             }
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     return visitImportDeclaration(node);
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return visitImportEqualsDeclaration(node);
-                case 259 /* ExportAssignment */:
+                case 266 /* ExportAssignment */:
                     return visitExportAssignment(node);
-                case 260 /* ExportDeclaration */:
+                case 267 /* ExportDeclaration */:
                     return visitExportDeclaration(node);
                 default:
                     ts.Debug.fail("Unhandled ellided statement");
@@ -77638,15 +82595,15 @@ var ts;
          * @param node The node to visit.
          */
         function namespaceElementVisitorWorker(node) {
-            if (node.kind === 260 /* ExportDeclaration */ ||
-                node.kind === 254 /* ImportDeclaration */ ||
-                node.kind === 255 /* ImportClause */ ||
-                (node.kind === 253 /* ImportEqualsDeclaration */ &&
-                    node.moduleReference.kind === 265 /* ExternalModuleReference */)) {
+            if (node.kind === 267 /* ExportDeclaration */ ||
+                node.kind === 261 /* ImportDeclaration */ ||
+                node.kind === 262 /* ImportClause */ ||
+                (node.kind === 260 /* ImportEqualsDeclaration */ &&
+                    node.moduleReference.kind === 272 /* ExternalModuleReference */)) {
                 // do not emit ES6 imports and exports since they are illegal inside a namespace
                 return undefined;
             }
-            else if (node.transformFlags & 1 /* ContainsTypeScript */ || ts.hasModifier(node, 1 /* Export */)) {
+            else if (node.transformFlags & 1 /* ContainsTypeScript */ || ts.hasSyntacticModifier(node, 1 /* Export */)) {
                 return visitTypeScript(node);
             }
             return node;
@@ -77666,19 +82623,19 @@ var ts;
          */
         function classElementVisitorWorker(node) {
             switch (node.kind) {
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     return visitConstructor(node);
-                case 159 /* PropertyDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     // Property declarations are not TypeScript syntax, but they must be visited
                     // for the decorator transformation.
                     return visitPropertyDeclaration(node);
-                case 167 /* IndexSignature */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 161 /* MethodDeclaration */:
+                case 171 /* IndexSignature */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 165 /* MethodDeclaration */:
                     // Fallback to the default visit behavior.
                     return visitorWorker(node);
-                case 222 /* SemicolonClassElement */:
+                case 229 /* SemicolonClassElement */:
                     return node;
                 default:
                     return ts.Debug.failBadSyntaxKind(node);
@@ -77688,7 +82645,7 @@ var ts;
             if (ts.modifierToFlag(node.kind) & 2270 /* TypeScriptModifier */) {
                 return undefined;
             }
-            else if (currentNamespace && node.kind === 89 /* ExportKeyword */) {
+            else if (currentNamespace && node.kind === 92 /* ExportKeyword */) {
                 return undefined;
             }
             return node;
@@ -77699,77 +82656,77 @@ var ts;
          * @param node The node to visit.
          */
         function visitTypeScript(node) {
-            if (ts.isStatement(node) && ts.hasModifier(node, 2 /* Ambient */)) {
+            if (ts.isStatement(node) && ts.hasSyntacticModifier(node, 2 /* Ambient */)) {
                 // TypeScript ambient declarations are elided, but some comments may be preserved.
                 // See the implementation of `getLeadingComments` in comments.ts for more details.
-                return ts.createNotEmittedStatement(node);
+                return factory.createNotEmittedStatement(node);
             }
             switch (node.kind) {
-                case 89 /* ExportKeyword */:
-                case 84 /* DefaultKeyword */:
+                case 92 /* ExportKeyword */:
+                case 87 /* DefaultKeyword */:
                     // ES6 export and default modifiers are elided when inside a namespace.
                     return currentNamespace ? undefined : node;
-                case 119 /* PublicKeyword */:
-                case 117 /* PrivateKeyword */:
-                case 118 /* ProtectedKeyword */:
-                case 122 /* AbstractKeyword */:
-                case 81 /* ConstKeyword */:
-                case 130 /* DeclareKeyword */:
-                case 138 /* ReadonlyKeyword */:
+                case 122 /* PublicKeyword */:
+                case 120 /* PrivateKeyword */:
+                case 121 /* ProtectedKeyword */:
+                case 125 /* AbstractKeyword */:
+                case 84 /* ConstKeyword */:
+                case 133 /* DeclareKeyword */:
+                case 142 /* ReadonlyKeyword */:
                 // TypeScript accessibility and readonly modifiers are elided
                 // falls through
-                case 174 /* ArrayType */:
-                case 175 /* TupleType */:
-                case 176 /* OptionalType */:
-                case 177 /* RestType */:
-                case 173 /* TypeLiteral */:
-                case 168 /* TypePredicate */:
-                case 155 /* TypeParameter */:
-                case 125 /* AnyKeyword */:
-                case 148 /* UnknownKeyword */:
-                case 128 /* BooleanKeyword */:
-                case 143 /* StringKeyword */:
-                case 140 /* NumberKeyword */:
-                case 137 /* NeverKeyword */:
-                case 110 /* VoidKeyword */:
-                case 144 /* SymbolKeyword */:
-                case 171 /* ConstructorType */:
-                case 170 /* FunctionType */:
-                case 172 /* TypeQuery */:
-                case 169 /* TypeReference */:
-                case 178 /* UnionType */:
-                case 179 /* IntersectionType */:
-                case 180 /* ConditionalType */:
-                case 182 /* ParenthesizedType */:
-                case 183 /* ThisType */:
-                case 184 /* TypeOperator */:
-                case 185 /* IndexedAccessType */:
-                case 186 /* MappedType */:
-                case 187 /* LiteralType */:
+                case 178 /* ArrayType */:
+                case 179 /* TupleType */:
+                case 180 /* OptionalType */:
+                case 181 /* RestType */:
+                case 177 /* TypeLiteral */:
+                case 172 /* TypePredicate */:
+                case 159 /* TypeParameter */:
+                case 128 /* AnyKeyword */:
+                case 152 /* UnknownKeyword */:
+                case 131 /* BooleanKeyword */:
+                case 147 /* StringKeyword */:
+                case 144 /* NumberKeyword */:
+                case 141 /* NeverKeyword */:
+                case 113 /* VoidKeyword */:
+                case 148 /* SymbolKeyword */:
+                case 175 /* ConstructorType */:
+                case 174 /* FunctionType */:
+                case 176 /* TypeQuery */:
+                case 173 /* TypeReference */:
+                case 182 /* UnionType */:
+                case 183 /* IntersectionType */:
+                case 184 /* ConditionalType */:
+                case 186 /* ParenthesizedType */:
+                case 187 /* ThisType */:
+                case 188 /* TypeOperator */:
+                case 189 /* IndexedAccessType */:
+                case 190 /* MappedType */:
+                case 191 /* LiteralType */:
                 // TypeScript type nodes are elided.
                 // falls through
-                case 167 /* IndexSignature */:
+                case 171 /* IndexSignature */:
                 // TypeScript index signatures are elided.
                 // falls through
-                case 157 /* Decorator */:
+                case 161 /* Decorator */:
                 // TypeScript decorators are elided. They will be emitted as part of visitClassDeclaration.
                 // falls through
-                case 247 /* TypeAliasDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
                     // TypeScript type-only declarations are elided.
                     return undefined;
-                case 159 /* PropertyDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     // TypeScript property declarations are elided. However their names are still visited, and can potentially be retained if they could have sideeffects
                     return visitPropertyDeclaration(node);
-                case 252 /* NamespaceExportDeclaration */:
+                case 259 /* NamespaceExportDeclaration */:
                     // TypeScript namespace export declarations are elided.
                     return undefined;
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     return visitConstructor(node);
-                case 246 /* InterfaceDeclaration */:
+                case 253 /* InterfaceDeclaration */:
                     // TypeScript interfaces are elided, but some comments may be preserved.
                     // See the implementation of `getLeadingComments` in comments.ts for more details.
-                    return ts.createNotEmittedStatement(node);
-                case 245 /* ClassDeclaration */:
+                    return factory.createNotEmittedStatement(node);
+                case 252 /* ClassDeclaration */:
                     // This may be a class declaration with TypeScript syntax extensions.
                     //
                     // TypeScript class syntax extensions include:
@@ -77779,7 +82736,7 @@ var ts;
                     // - index signatures
                     // - method overload signatures
                     return visitClassDeclaration(node);
-                case 214 /* ClassExpression */:
+                case 221 /* ClassExpression */:
                     // This may be a class expression with TypeScript syntax extensions.
                     //
                     // TypeScript class syntax extensions include:
@@ -77789,35 +82746,35 @@ var ts;
                     // - index signatures
                     // - method overload signatures
                     return visitClassExpression(node);
-                case 279 /* HeritageClause */:
+                case 286 /* HeritageClause */:
                     // This may be a heritage clause with TypeScript syntax extensions.
                     //
                     // TypeScript heritage clause extensions include:
                     // - `implements` clause
                     return visitHeritageClause(node);
-                case 216 /* ExpressionWithTypeArguments */:
+                case 223 /* ExpressionWithTypeArguments */:
                     // TypeScript supports type arguments on an expression in an `extends` heritage clause.
                     return visitExpressionWithTypeArguments(node);
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     // TypeScript method declarations may have decorators, modifiers
                     // or type annotations.
                     return visitMethodDeclaration(node);
-                case 163 /* GetAccessor */:
+                case 167 /* GetAccessor */:
                     // Get Accessors can have TypeScript modifiers, decorators, and type annotations.
                     return visitGetAccessor(node);
-                case 164 /* SetAccessor */:
+                case 168 /* SetAccessor */:
                     // Set Accessors can have TypeScript modifiers and type annotations.
                     return visitSetAccessor(node);
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     // Typescript function declarations can have modifiers, decorators, and type annotations.
                     return visitFunctionDeclaration(node);
-                case 201 /* FunctionExpression */:
+                case 208 /* FunctionExpression */:
                     // TypeScript function expressions can have modifiers and type annotations.
                     return visitFunctionExpression(node);
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     // TypeScript arrow functions can have modifiers and type annotations.
                     return visitArrowFunction(node);
-                case 156 /* Parameter */:
+                case 160 /* Parameter */:
                     // This may be a parameter declaration with TypeScript syntax extensions.
                     //
                     // TypeScript parameter declaration syntax extensions include:
@@ -77827,40 +82784,40 @@ var ts;
                     // - type annotations
                     // - this parameters
                     return visitParameter(node);
-                case 200 /* ParenthesizedExpression */:
+                case 207 /* ParenthesizedExpression */:
                     // ParenthesizedExpressions are TypeScript if their expression is a
                     // TypeAssertion or AsExpression
                     return visitParenthesizedExpression(node);
-                case 199 /* TypeAssertionExpression */:
-                case 217 /* AsExpression */:
+                case 206 /* TypeAssertionExpression */:
+                case 224 /* AsExpression */:
                     // TypeScript type assertions are removed, but their subtrees are preserved.
                     return visitAssertionExpression(node);
-                case 196 /* CallExpression */:
+                case 203 /* CallExpression */:
                     return visitCallExpression(node);
-                case 197 /* NewExpression */:
+                case 204 /* NewExpression */:
                     return visitNewExpression(node);
-                case 198 /* TaggedTemplateExpression */:
+                case 205 /* TaggedTemplateExpression */:
                     return visitTaggedTemplateExpression(node);
-                case 218 /* NonNullExpression */:
+                case 225 /* NonNullExpression */:
                     // TypeScript non-null expressions are removed, but their subtrees are preserved.
                     return visitNonNullExpression(node);
-                case 248 /* EnumDeclaration */:
+                case 255 /* EnumDeclaration */:
                     // TypeScript enum declarations do not exist in ES6 and must be rewritten.
                     return visitEnumDeclaration(node);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     // TypeScript namespace exports for variable statements must be transformed.
                     return visitVariableStatement(node);
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return visitVariableDeclaration(node);
-                case 249 /* ModuleDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     // TypeScript namespace declarations must be transformed.
                     return visitModuleDeclaration(node);
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     // TypeScript namespace or external module import.
                     return visitImportEqualsDeclaration(node);
-                case 267 /* JsxSelfClosingElement */:
+                case 274 /* JsxSelfClosingElement */:
                     return visitJsxSelfClosingElement(node);
-                case 268 /* JsxOpeningElement */:
+                case 275 /* JsxOpeningElement */:
                     return visitJsxJsxOpeningElement(node);
                 default:
                     // node contains some other TypeScript syntax
@@ -77871,7 +82828,7 @@ var ts;
             var alwaysStrict = ts.getStrictOptionValue(compilerOptions, "alwaysStrict") &&
                 !(ts.isExternalModule(node) && moduleKind >= ts.ModuleKind.ES2015) &&
                 !ts.isJsonSourceFile(node);
-            return ts.updateSourceFileNode(node, ts.visitLexicalEnvironment(node.statements, sourceElementVisitor, context, /*start*/ 0, alwaysStrict));
+            return factory.updateSourceFile(node, ts.visitLexicalEnvironment(node.statements, sourceElementVisitor, context, /*start*/ 0, alwaysStrict));
         }
         /**
          * Tests whether we should emit a __decorate call for a class declaration.
@@ -77897,7 +82854,7 @@ var ts;
             if (ts.some(staticProperties))
                 facts |= 1 /* HasStaticInitializedProperties */;
             var extendsClauseElement = ts.getEffectiveBaseTypeNode(node);
-            if (extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 100 /* NullKeyword */)
+            if (extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 103 /* NullKeyword */)
                 facts |= 64 /* IsDerivedClass */;
             if (shouldEmitDecorateCallForClass(node))
                 facts |= 2 /* HasConstructorDecorators */;
@@ -77923,7 +82880,7 @@ var ts;
                 || ts.some(node.members, hasTypeScriptClassSyntax);
         }
         function visitClassDeclaration(node) {
-            if (!isClassLikeDeclarationWithTypeScriptSyntax(node) && !(currentNamespace && ts.hasModifier(node, 1 /* Export */))) {
+            if (!isClassLikeDeclarationWithTypeScriptSyntax(node) && !(currentNamespace && ts.hasSyntacticModifier(node, 1 /* Export */))) {
                 return ts.visitEachChild(node, visitor, context);
             }
             var staticProperties = ts.getProperties(node, /*requireInitializer*/ true, /*isStatic*/ true);
@@ -77931,7 +82888,7 @@ var ts;
             if (facts & 128 /* UseImmediatelyInvokedFunctionExpression */) {
                 context.startLexicalEnvironment();
             }
-            var name = node.name || (facts & 5 /* NeedsName */ ? ts.getGeneratedNameForNode(node) : undefined);
+            var name = node.name || (facts & 5 /* NeedsName */ ? factory.getGeneratedNameForNode(node) : undefined);
             var classStatement = facts & 2 /* HasConstructorDecorators */
                 ? createClassDeclarationHeadWithDecorators(node, name)
                 : createClassDeclarationHeadWithoutDecorators(node, name, facts);
@@ -77953,22 +82910,23 @@ var ts;
                 //  }();
                 //
                 var closingBraceLocation = ts.createTokenRange(ts.skipTrivia(currentSourceFile.text, node.members.end), 19 /* CloseBraceToken */);
-                var localName = ts.getInternalName(node);
+                var localName = factory.getInternalName(node);
                 // The following partially-emitted expression exists purely to align our sourcemap
                 // emit with the original emitter.
-                var outer = ts.createPartiallyEmittedExpression(localName);
-                outer.end = closingBraceLocation.end;
+                var outer = factory.createPartiallyEmittedExpression(localName);
+                ts.setTextRangeEnd(outer, closingBraceLocation.end);
                 ts.setEmitFlags(outer, 1536 /* NoComments */);
-                var statement = ts.createReturn(outer);
-                statement.pos = closingBraceLocation.pos;
+                var statement = factory.createReturnStatement(outer);
+                ts.setTextRangePos(statement, closingBraceLocation.pos);
                 ts.setEmitFlags(statement, 1536 /* NoComments */ | 384 /* NoTokenSourceMaps */);
                 statements.push(statement);
                 ts.insertStatementsAfterStandardPrologue(statements, context.endLexicalEnvironment());
-                var iife = ts.createImmediatelyInvokedArrowFunction(statements);
+                var iife = factory.createImmediatelyInvokedArrowFunction(statements);
                 ts.setEmitFlags(iife, 33554432 /* TypeScriptClassWrapper */);
-                var varStatement = ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                    ts.createVariableDeclaration(ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ false), 
+                var varStatement = factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                    factory.createVariableDeclaration(factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ false), 
+                    /*exclamationToken*/ undefined, 
                     /*type*/ undefined, iife)
                 ]));
                 ts.setOriginalNode(varStatement, node);
@@ -77985,15 +82943,15 @@ var ts;
             }
             else if (facts & 128 /* UseImmediatelyInvokedFunctionExpression */ || facts & 2 /* HasConstructorDecorators */) {
                 if (facts & 32 /* IsDefaultExternalExport */) {
-                    statements.push(ts.createExportDefault(ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true)));
+                    statements.push(factory.createExportDefault(factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true)));
                 }
                 else if (facts & 16 /* IsNamedExternalExport */) {
-                    statements.push(ts.createExternalModuleExport(ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true)));
+                    statements.push(factory.createExternalModuleExport(factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true)));
                 }
             }
             if (statements.length > 1) {
                 // Add a DeclarationMarker as a marker for the end of the declaration
-                statements.push(ts.createEndOfDeclarationMarker(node));
+                statements.push(factory.createEndOfDeclarationMarker(node));
                 ts.setEmitFlags(classStatement, ts.getEmitFlags(classStatement) | 4194304 /* HasEndOfDeclarationMarker */);
             }
             return ts.singleOrMany(statements);
@@ -78013,7 +82971,7 @@ var ts;
             var modifiers = !(facts & 128 /* UseImmediatelyInvokedFunctionExpression */)
                 ? ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier)
                 : undefined;
-            var classDeclaration = ts.createClassDeclaration(
+            var classDeclaration = factory.createClassDeclaration(
             /*decorators*/ undefined, modifiers, name, 
             /*typeParameters*/ undefined, ts.visitNodes(node.heritageClauses, visitor, ts.isHeritageClause), transformClassMembers(node));
             // To better align with the old emitter, we should not emit a trailing source map
@@ -78022,7 +82980,6 @@ var ts;
             if (facts & 1 /* HasStaticInitializedProperties */) {
                 emitFlags |= 32 /* NoTrailingSourceMap */;
             }
-            ts.aggregateTransformFlags(classDeclaration);
             ts.setTextRange(classDeclaration, node);
             ts.setOriginalNode(classDeclaration, node);
             ts.setEmitFlags(classDeclaration, emitFlags);
@@ -78120,22 +83077,22 @@ var ts;
             //
             var location = ts.moveRangePastDecorators(node);
             var classAlias = getClassAliasIfNeeded(node);
-            var declName = ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
+            var declName = factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
             //  ... = class ${name} ${heritageClauses} {
             //      ${members}
             //  }
             var heritageClauses = ts.visitNodes(node.heritageClauses, visitor, ts.isHeritageClause);
             var members = transformClassMembers(node);
-            var classExpression = ts.createClassExpression(/*modifiers*/ undefined, name, /*typeParameters*/ undefined, heritageClauses, members);
-            ts.aggregateTransformFlags(classExpression);
+            var classExpression = factory.createClassExpression(/*decorators*/ undefined, /*modifiers*/ undefined, name, /*typeParameters*/ undefined, heritageClauses, members);
             ts.setOriginalNode(classExpression, node);
             ts.setTextRange(classExpression, location);
             //  let ${name} = ${classExpression} where name is either declaredName if the class doesn't contain self-reference
             //                                         or decoratedClassAlias if the class contain self-reference.
-            var statement = ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(declName, 
-                /*type*/ undefined, classAlias ? ts.createAssignment(classAlias, classExpression) : classExpression)
+            var statement = factory.createVariableStatement(
+            /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(declName, 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, classAlias ? factory.createAssignment(classAlias, classExpression) : classExpression)
             ], 1 /* Let */));
             ts.setOriginalNode(statement, node);
             ts.setTextRange(statement, location);
@@ -78146,10 +83103,10 @@ var ts;
             if (!isClassLikeDeclarationWithTypeScriptSyntax(node)) {
                 return ts.visitEachChild(node, visitor, context);
             }
-            var classExpression = ts.createClassExpression(
+            var classExpression = factory.createClassExpression(
+            /*decorators*/ undefined, 
             /*modifiers*/ undefined, node.name, 
             /*typeParameters*/ undefined, ts.visitNodes(node.heritageClauses, visitor, ts.isHeritageClause), transformClassMembers(node));
-            ts.aggregateTransformFlags(classExpression);
             ts.setOriginalNode(classExpression, node);
             ts.setTextRange(classExpression, node);
             return classExpression;
@@ -78168,17 +83125,17 @@ var ts;
                 for (var _i = 0, parametersWithPropertyAssignments_1 = parametersWithPropertyAssignments; _i < parametersWithPropertyAssignments_1.length; _i++) {
                     var parameter = parametersWithPropertyAssignments_1[_i];
                     if (ts.isIdentifier(parameter.name)) {
-                        members.push(ts.setOriginalNode(ts.aggregateTransformFlags(ts.createProperty(
+                        members.push(ts.setOriginalNode(factory.createPropertyDeclaration(
                         /*decorators*/ undefined, 
                         /*modifiers*/ undefined, parameter.name, 
                         /*questionOrExclamationToken*/ undefined, 
                         /*type*/ undefined, 
-                        /*initializer*/ undefined)), parameter));
+                        /*initializer*/ undefined), parameter));
                     }
                 }
             }
             ts.addRange(members, ts.visitNodes(node.members, classElementVisitor, ts.isClassElement));
-            return ts.setTextRange(ts.createNodeArray(members), /*location*/ node.members);
+            return ts.setTextRange(factory.createNodeArray(members), /*location*/ node.members);
         }
         /**
          * Gets either the static or instance members of a class that are decorated, or have
@@ -78217,7 +83174,7 @@ var ts;
          */
         function isDecoratedClassElement(member, isStatic, parent) {
             return ts.nodeOrChildIsDecorated(member, parent)
-                && isStatic === ts.hasModifier(member, 32 /* Static */);
+                && isStatic === ts.hasSyntacticModifier(member, 32 /* Static */);
         }
         /**
          * Gets an array of arrays of decorators for the parameters of a function-like node.
@@ -78269,12 +83226,12 @@ var ts;
          */
         function getAllDecoratorsOfClassElement(node, member) {
             switch (member.kind) {
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return getAllDecoratorsOfAccessors(node, member);
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     return getAllDecoratorsOfMethod(member);
-                case 159 /* PropertyDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     return getAllDecoratorsOfProperty(member);
                 default:
                     return undefined;
@@ -78427,15 +83384,16 @@ var ts;
             var prefix = getClassMemberPrefix(node, member);
             var memberName = getExpressionForPropertyName(member, /*generateNameForComputedPropertyName*/ true);
             var descriptor = languageVersion > 0 /* ES3 */
-                ? member.kind === 159 /* PropertyDeclaration */
+                ? member.kind === 163 /* PropertyDeclaration */
                     // We emit `void 0` here to indicate to `__decorate` that it can invoke `Object.defineProperty` directly, but that it
                     // should not invoke `Object.getOwnPropertyDescriptor`.
-                    ? ts.createVoidZero()
+                    ? factory.createVoidZero()
                     // We emit `null` here to indicate to `__decorate` that it can invoke `Object.getOwnPropertyDescriptor` directly.
                     // We have this extra argument here so that we can inject an explicit property descriptor at a later date.
-                    : ts.createNull()
+                    : factory.createNull()
                 : undefined;
-            var helper = createDecorateHelper(context, decoratorExpressions, prefix, memberName, descriptor, ts.moveRangePastDecorators(member));
+            var helper = emitHelpers().createDecorateHelper(decoratorExpressions, prefix, memberName, descriptor);
+            ts.setTextRange(helper, ts.moveRangePastDecorators(member));
             ts.setEmitFlags(helper, 1536 /* NoComments */);
             return helper;
         }
@@ -78447,7 +83405,7 @@ var ts;
         function addConstructorDecorationStatement(statements, node) {
             var expression = generateConstructorDecorationExpression(node);
             if (expression) {
-                statements.push(ts.setOriginalNode(ts.createExpressionStatement(expression), node));
+                statements.push(ts.setOriginalNode(factory.createExpressionStatement(expression), node));
             }
         }
         /**
@@ -78462,9 +83420,9 @@ var ts;
                 return undefined;
             }
             var classAlias = classAliases && classAliases[ts.getOriginalNodeId(node)];
-            var localName = ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
-            var decorate = createDecorateHelper(context, decoratorExpressions, localName);
-            var expression = ts.createAssignment(localName, classAlias ? ts.createAssignment(classAlias, decorate) : decorate);
+            var localName = factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
+            var decorate = emitHelpers().createDecorateHelper(decoratorExpressions, localName);
+            var expression = factory.createAssignment(localName, classAlias ? factory.createAssignment(classAlias, decorate) : decorate);
             ts.setEmitFlags(expression, 1536 /* NoComments */);
             ts.setSourceMapRange(expression, ts.moveRangePastDecorators(node));
             return expression;
@@ -78489,8 +83447,8 @@ var ts;
                 expressions = [];
                 for (var _i = 0, decorators_1 = decorators; _i < decorators_1.length; _i++) {
                     var decorator = decorators_1[_i];
-                    var helper = createParamHelper(context, transformDecorator(decorator), parameterOffset, 
-                    /*location*/ decorator.expression);
+                    var helper = emitHelpers().createParamHelper(transformDecorator(decorator), parameterOffset);
+                    ts.setTextRange(helper, decorator.expression);
                     ts.setEmitFlags(helper, 1536 /* NoComments */);
                     expressions.push(helper);
                 }
@@ -78514,13 +83472,13 @@ var ts;
         function addOldTypeMetadata(node, container, decoratorExpressions) {
             if (compilerOptions.emitDecoratorMetadata) {
                 if (shouldAddTypeMetadata(node)) {
-                    decoratorExpressions.push(createMetadataHelper(context, "design:type", serializeTypeOfNode(node)));
+                    decoratorExpressions.push(emitHelpers().createMetadataHelper("design:type", serializeTypeOfNode(node)));
                 }
                 if (shouldAddParamTypesMetadata(node)) {
-                    decoratorExpressions.push(createMetadataHelper(context, "design:paramtypes", serializeParameterTypesOfNode(node, container)));
+                    decoratorExpressions.push(emitHelpers().createMetadataHelper("design:paramtypes", serializeParameterTypesOfNode(node, container)));
                 }
                 if (shouldAddReturnTypeMetadata(node)) {
-                    decoratorExpressions.push(createMetadataHelper(context, "design:returntype", serializeReturnTypeOfNode(node)));
+                    decoratorExpressions.push(emitHelpers().createMetadataHelper("design:returntype", serializeReturnTypeOfNode(node)));
                 }
             }
         }
@@ -78528,16 +83486,16 @@ var ts;
             if (compilerOptions.emitDecoratorMetadata) {
                 var properties = void 0;
                 if (shouldAddTypeMetadata(node)) {
-                    (properties || (properties = [])).push(ts.createPropertyAssignment("type", ts.createArrowFunction(/*modifiers*/ undefined, /*typeParameters*/ undefined, [], /*type*/ undefined, ts.createToken(38 /* EqualsGreaterThanToken */), serializeTypeOfNode(node))));
+                    (properties || (properties = [])).push(factory.createPropertyAssignment("type", factory.createArrowFunction(/*modifiers*/ undefined, /*typeParameters*/ undefined, [], /*type*/ undefined, factory.createToken(38 /* EqualsGreaterThanToken */), serializeTypeOfNode(node))));
                 }
                 if (shouldAddParamTypesMetadata(node)) {
-                    (properties || (properties = [])).push(ts.createPropertyAssignment("paramTypes", ts.createArrowFunction(/*modifiers*/ undefined, /*typeParameters*/ undefined, [], /*type*/ undefined, ts.createToken(38 /* EqualsGreaterThanToken */), serializeParameterTypesOfNode(node, container))));
+                    (properties || (properties = [])).push(factory.createPropertyAssignment("paramTypes", factory.createArrowFunction(/*modifiers*/ undefined, /*typeParameters*/ undefined, [], /*type*/ undefined, factory.createToken(38 /* EqualsGreaterThanToken */), serializeParameterTypesOfNode(node, container))));
                 }
                 if (shouldAddReturnTypeMetadata(node)) {
-                    (properties || (properties = [])).push(ts.createPropertyAssignment("returnType", ts.createArrowFunction(/*modifiers*/ undefined, /*typeParameters*/ undefined, [], /*type*/ undefined, ts.createToken(38 /* EqualsGreaterThanToken */), serializeReturnTypeOfNode(node))));
+                    (properties || (properties = [])).push(factory.createPropertyAssignment("returnType", factory.createArrowFunction(/*modifiers*/ undefined, /*typeParameters*/ undefined, [], /*type*/ undefined, factory.createToken(38 /* EqualsGreaterThanToken */), serializeReturnTypeOfNode(node))));
                 }
                 if (properties) {
-                    decoratorExpressions.push(createMetadataHelper(context, "design:typeinfo", ts.createObjectLiteral(properties, /*multiLine*/ true)));
+                    decoratorExpressions.push(emitHelpers().createMetadataHelper("design:typeinfo", factory.createObjectLiteralExpression(properties, /*multiLine*/ true)));
                 }
             }
         }
@@ -78550,10 +83508,10 @@ var ts;
          */
         function shouldAddTypeMetadata(node) {
             var kind = node.kind;
-            return kind === 161 /* MethodDeclaration */
-                || kind === 163 /* GetAccessor */
-                || kind === 164 /* SetAccessor */
-                || kind === 159 /* PropertyDeclaration */;
+            return kind === 165 /* MethodDeclaration */
+                || kind === 167 /* GetAccessor */
+                || kind === 168 /* SetAccessor */
+                || kind === 163 /* PropertyDeclaration */;
         }
         /**
          * Determines whether to emit the "design:returntype" metadata based on the node's kind.
@@ -78563,7 +83521,7 @@ var ts;
          * @param node The node to test.
          */
         function shouldAddReturnTypeMetadata(node) {
-            return node.kind === 161 /* MethodDeclaration */;
+            return node.kind === 165 /* MethodDeclaration */;
         }
         /**
          * Determines whether to emit the "design:paramtypes" metadata based on the node's kind.
@@ -78574,12 +83532,12 @@ var ts;
          */
         function shouldAddParamTypesMetadata(node) {
             switch (node.kind) {
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
                     return ts.getFirstConstructorWithBody(node) !== undefined;
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return true;
             }
             return false;
@@ -78596,18 +83554,18 @@ var ts;
          */
         function serializeTypeOfNode(node) {
             switch (node.kind) {
-                case 159 /* PropertyDeclaration */:
-                case 156 /* Parameter */:
+                case 163 /* PropertyDeclaration */:
+                case 160 /* Parameter */:
                     return serializeTypeNode(node.type);
-                case 164 /* SetAccessor */:
-                case 163 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 167 /* GetAccessor */:
                     return serializeTypeNode(getAccessorTypeNode(node));
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 161 /* MethodDeclaration */:
-                    return ts.createIdentifier("Function");
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 165 /* MethodDeclaration */:
+                    return factory.createIdentifier("Function");
                 default:
-                    return ts.createVoidZero();
+                    return factory.createVoidZero();
             }
         }
         /**
@@ -78638,10 +83596,10 @@ var ts;
                     }
                 }
             }
-            return ts.createArrayLiteral(expressions);
+            return factory.createArrayLiteralExpression(expressions);
         }
         function getParametersOfDecoratedDeclaration(node, container) {
-            if (container && node.kind === 163 /* GetAccessor */) {
+            if (container && node.kind === 167 /* GetAccessor */) {
                 var setAccessor = ts.getAllAccessorDeclarations(container.members, node).setAccessor;
                 if (setAccessor) {
                     return setAccessor.parameters;
@@ -78659,9 +83617,9 @@ var ts;
                 return serializeTypeNode(node.type);
             }
             else if (ts.isAsyncFunction(node)) {
-                return ts.createIdentifier("Promise");
+                return factory.createIdentifier("Promise");
             }
-            return ts.createVoidZero();
+            return factory.createVoidZero();
         }
         /**
          * Serializes a type node for use with decorator type metadata.
@@ -78683,102 +83641,104 @@ var ts;
          */
         function serializeTypeNode(node) {
             if (node === undefined) {
-                return ts.createIdentifier("Object");
+                return factory.createIdentifier("Object");
             }
             switch (node.kind) {
-                case 110 /* VoidKeyword */:
-                case 146 /* UndefinedKeyword */:
-                case 100 /* NullKeyword */:
-                case 137 /* NeverKeyword */:
-                    return ts.createVoidZero();
-                case 182 /* ParenthesizedType */:
+                case 113 /* VoidKeyword */:
+                case 150 /* UndefinedKeyword */:
+                case 141 /* NeverKeyword */:
+                    return factory.createVoidZero();
+                case 186 /* ParenthesizedType */:
                     return serializeTypeNode(node.type);
-                case 170 /* FunctionType */:
-                case 171 /* ConstructorType */:
-                    return ts.createIdentifier("Function");
-                case 174 /* ArrayType */:
-                case 175 /* TupleType */:
-                    return ts.createIdentifier("Array");
-                case 168 /* TypePredicate */:
-                case 128 /* BooleanKeyword */:
-                    return ts.createIdentifier("Boolean");
-                case 143 /* StringKeyword */:
-                    return ts.createIdentifier("String");
-                case 141 /* ObjectKeyword */:
-                    return ts.createIdentifier("Object");
-                case 187 /* LiteralType */:
+                case 174 /* FunctionType */:
+                case 175 /* ConstructorType */:
+                    return factory.createIdentifier("Function");
+                case 178 /* ArrayType */:
+                case 179 /* TupleType */:
+                    return factory.createIdentifier("Array");
+                case 172 /* TypePredicate */:
+                case 131 /* BooleanKeyword */:
+                    return factory.createIdentifier("Boolean");
+                case 147 /* StringKeyword */:
+                    return factory.createIdentifier("String");
+                case 145 /* ObjectKeyword */:
+                    return factory.createIdentifier("Object");
+                case 191 /* LiteralType */:
                     switch (node.literal.kind) {
                         case 10 /* StringLiteral */:
-                            return ts.createIdentifier("String");
-                        case 207 /* PrefixUnaryExpression */:
+                        case 14 /* NoSubstitutionTemplateLiteral */:
+                            return factory.createIdentifier("String");
+                        case 214 /* PrefixUnaryExpression */:
                         case 8 /* NumericLiteral */:
-                            return ts.createIdentifier("Number");
+                            return factory.createIdentifier("Number");
                         case 9 /* BigIntLiteral */:
                             return getGlobalBigIntNameWithFallback();
-                        case 106 /* TrueKeyword */:
-                        case 91 /* FalseKeyword */:
-                            return ts.createIdentifier("Boolean");
+                        case 109 /* TrueKeyword */:
+                        case 94 /* FalseKeyword */:
+                            return factory.createIdentifier("Boolean");
+                        case 103 /* NullKeyword */:
+                            return factory.createVoidZero();
                         default:
                             return ts.Debug.failBadSyntaxKind(node.literal);
                     }
-                case 140 /* NumberKeyword */:
-                    return ts.createIdentifier("Number");
-                case 151 /* BigIntKeyword */:
+                case 144 /* NumberKeyword */:
+                    return factory.createIdentifier("Number");
+                case 155 /* BigIntKeyword */:
                     return getGlobalBigIntNameWithFallback();
-                case 144 /* SymbolKeyword */:
+                case 148 /* SymbolKeyword */:
                     return languageVersion < 2 /* ES2015 */
                         ? getGlobalSymbolNameWithFallback()
-                        : ts.createIdentifier("Symbol");
-                case 169 /* TypeReference */:
+                        : factory.createIdentifier("Symbol");
+                case 173 /* TypeReference */:
                     return serializeTypeReferenceNode(node);
-                case 179 /* IntersectionType */:
-                case 178 /* UnionType */:
+                case 183 /* IntersectionType */:
+                case 182 /* UnionType */:
                     return serializeTypeList(node.types);
-                case 180 /* ConditionalType */:
+                case 184 /* ConditionalType */:
                     return serializeTypeList([node.trueType, node.falseType]);
-                case 184 /* TypeOperator */:
-                    if (node.operator === 138 /* ReadonlyKeyword */) {
+                case 188 /* TypeOperator */:
+                    if (node.operator === 142 /* ReadonlyKeyword */) {
                         return serializeTypeNode(node.type);
                     }
                     break;
-                case 172 /* TypeQuery */:
-                case 185 /* IndexedAccessType */:
-                case 186 /* MappedType */:
-                case 173 /* TypeLiteral */:
-                case 125 /* AnyKeyword */:
-                case 148 /* UnknownKeyword */:
-                case 183 /* ThisType */:
-                case 188 /* ImportType */:
+                case 176 /* TypeQuery */:
+                case 189 /* IndexedAccessType */:
+                case 190 /* MappedType */:
+                case 177 /* TypeLiteral */:
+                case 128 /* AnyKeyword */:
+                case 152 /* UnknownKeyword */:
+                case 187 /* ThisType */:
+                case 195 /* ImportType */:
                     break;
                 // handle JSDoc types from an invalid parse
-                case 295 /* JSDocAllType */:
-                case 296 /* JSDocUnknownType */:
-                case 300 /* JSDocFunctionType */:
-                case 301 /* JSDocVariadicType */:
-                case 302 /* JSDocNamepathType */:
+                case 303 /* JSDocAllType */:
+                case 304 /* JSDocUnknownType */:
+                case 308 /* JSDocFunctionType */:
+                case 309 /* JSDocVariadicType */:
+                case 310 /* JSDocNamepathType */:
                     break;
-                case 297 /* JSDocNullableType */:
-                case 298 /* JSDocNonNullableType */:
-                case 299 /* JSDocOptionalType */:
+                case 305 /* JSDocNullableType */:
+                case 306 /* JSDocNonNullableType */:
+                case 307 /* JSDocOptionalType */:
                     return serializeTypeNode(node.type);
                 default:
                     return ts.Debug.failBadSyntaxKind(node);
             }
-            return ts.createIdentifier("Object");
+            return factory.createIdentifier("Object");
         }
         function serializeTypeList(types) {
             // Note when updating logic here also update getEntityNameForDecoratorMetadata
             // so that aliases can be marked as referenced
             var serializedUnion;
-            for (var _i = 0, types_21 = types; _i < types_21.length; _i++) {
-                var typeNode = types_21[_i];
-                while (typeNode.kind === 182 /* ParenthesizedType */) {
+            for (var _i = 0, types_22 = types; _i < types_22.length; _i++) {
+                var typeNode = types_22[_i];
+                while (typeNode.kind === 186 /* ParenthesizedType */) {
                     typeNode = typeNode.type; // Skip parens if need be
                 }
-                if (typeNode.kind === 137 /* NeverKeyword */) {
+                if (typeNode.kind === 141 /* NeverKeyword */) {
                     continue; // Always elide `never` from the union/intersection if possible
                 }
-                if (!strictNullChecks && (typeNode.kind === 100 /* NullKeyword */ || typeNode.kind === 146 /* UndefinedKeyword */)) {
+                if (!strictNullChecks && (typeNode.kind === 191 /* LiteralType */ && typeNode.literal.kind === 103 /* NullKeyword */ || typeNode.kind === 150 /* UndefinedKeyword */)) {
                     continue; // Elide null and undefined from unions for metadata, just like what we did prior to the implementation of strict null checks
                 }
                 var serializedIndividual = serializeTypeNode(typeNode);
@@ -78793,7 +83753,7 @@ var ts;
                     if (!ts.isIdentifier(serializedUnion) ||
                         !ts.isIdentifier(serializedIndividual) ||
                         serializedUnion.escapedText !== serializedIndividual.escapedText) {
-                        return ts.createIdentifier("Object");
+                        return factory.createIdentifier("Object");
                     }
                 }
                 else {
@@ -78802,7 +83762,7 @@ var ts;
                 }
             }
             // If we were able to find common type, use it
-            return serializedUnion || ts.createVoidZero(); // Fallback is only hit if all union constituients are null/undefined/never
+            return serializedUnion || factory.createVoidZero(); // Fallback is only hit if all union constituients are null/undefined/never
         }
         /**
          * Serializes a TypeReferenceNode to an appropriate JS constructor value for use with
@@ -78816,41 +83776,43 @@ var ts;
                 case ts.TypeReferenceSerializationKind.Unknown:
                     // From conditional type type reference that cannot be resolved is Similar to any or unknown
                     if (ts.findAncestor(node, function (n) { return n.parent && ts.isConditionalTypeNode(n.parent) && (n.parent.trueType === n || n.parent.falseType === n); })) {
-                        return ts.createIdentifier("Object");
+                        return factory.createIdentifier("Object");
                     }
                     var serialized = serializeEntityNameAsExpressionFallback(node.typeName);
-                    var temp = ts.createTempVariable(hoistVariableDeclaration);
-                    return ts.createConditional(ts.createTypeCheck(ts.createAssignment(temp, serialized), "function"), temp, ts.createIdentifier("Object"));
+                    var temp = factory.createTempVariable(hoistVariableDeclaration);
+                    return factory.createConditionalExpression(factory.createTypeCheck(factory.createAssignment(temp, serialized), "function"), 
+                    /*questionToken*/ undefined, temp, 
+                    /*colonToken*/ undefined, factory.createIdentifier("Object"));
                 case ts.TypeReferenceSerializationKind.TypeWithConstructSignatureAndValue:
                     return serializeEntityNameAsExpression(node.typeName);
                 case ts.TypeReferenceSerializationKind.VoidNullableOrNeverType:
-                    return ts.createVoidZero();
+                    return factory.createVoidZero();
                 case ts.TypeReferenceSerializationKind.BigIntLikeType:
                     return getGlobalBigIntNameWithFallback();
                 case ts.TypeReferenceSerializationKind.BooleanType:
-                    return ts.createIdentifier("Boolean");
+                    return factory.createIdentifier("Boolean");
                 case ts.TypeReferenceSerializationKind.NumberLikeType:
-                    return ts.createIdentifier("Number");
+                    return factory.createIdentifier("Number");
                 case ts.TypeReferenceSerializationKind.StringLikeType:
-                    return ts.createIdentifier("String");
+                    return factory.createIdentifier("String");
                 case ts.TypeReferenceSerializationKind.ArrayLikeType:
-                    return ts.createIdentifier("Array");
+                    return factory.createIdentifier("Array");
                 case ts.TypeReferenceSerializationKind.ESSymbolType:
                     return languageVersion < 2 /* ES2015 */
                         ? getGlobalSymbolNameWithFallback()
-                        : ts.createIdentifier("Symbol");
+                        : factory.createIdentifier("Symbol");
                 case ts.TypeReferenceSerializationKind.TypeWithCallSignature:
-                    return ts.createIdentifier("Function");
+                    return factory.createIdentifier("Function");
                 case ts.TypeReferenceSerializationKind.Promise:
-                    return ts.createIdentifier("Promise");
+                    return factory.createIdentifier("Promise");
                 case ts.TypeReferenceSerializationKind.ObjectType:
-                    return ts.createIdentifier("Object");
+                    return factory.createIdentifier("Object");
                 default:
                     return ts.Debug.assertNever(kind);
             }
         }
         function createCheckedValue(left, right) {
-            return ts.createLogicalAnd(ts.createStrictInequality(ts.createTypeOf(left), ts.createLiteral("undefined")), right);
+            return factory.createLogicalAnd(factory.createStrictInequality(factory.createTypeOfExpression(left), factory.createStringLiteral("undefined")), right);
         }
         /**
          * Serializes an entity name which may not exist at runtime, but whose access shouldn't throw
@@ -78858,19 +83820,19 @@ var ts;
          * @param node The entity name to serialize.
          */
         function serializeEntityNameAsExpressionFallback(node) {
-            if (node.kind === 75 /* Identifier */) {
+            if (node.kind === 78 /* Identifier */) {
                 // A -> typeof A !== undefined && A
                 var copied = serializeEntityNameAsExpression(node);
                 return createCheckedValue(copied, copied);
             }
-            if (node.left.kind === 75 /* Identifier */) {
+            if (node.left.kind === 78 /* Identifier */) {
                 // A.B -> typeof A !== undefined && A.B
                 return createCheckedValue(serializeEntityNameAsExpression(node.left), serializeEntityNameAsExpression(node));
             }
             // A.B.C -> typeof A !== undefined && (_a = A.B) !== void 0 && _a.C
             var left = serializeEntityNameAsExpressionFallback(node.left);
-            var temp = ts.createTempVariable(hoistVariableDeclaration);
-            return ts.createLogicalAnd(ts.createLogicalAnd(left.left, ts.createStrictInequality(ts.createAssignment(temp, left.right), ts.createVoidZero())), ts.createPropertyAccess(temp, node.right));
+            var temp = factory.createTempVariable(hoistVariableDeclaration);
+            return factory.createLogicalAnd(factory.createLogicalAnd(left.left, factory.createStrictInequality(factory.createAssignment(temp, left.right), factory.createVoidZero())), factory.createPropertyAccessExpression(temp, node.right));
         }
         /**
          * Serializes an entity name as an expression for decorator type metadata.
@@ -78879,15 +83841,14 @@ var ts;
          */
         function serializeEntityNameAsExpression(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     // Create a clone of the name with a new parent, and treat it as if it were
                     // a source tree node for the purposes of the checker.
-                    var name = ts.getMutableClone(node);
-                    name.flags &= ~8 /* Synthesized */;
+                    var name = ts.setParent(ts.setTextRange(ts.parseNodeFactory.cloneNode(node), node), node.parent);
                     name.original = undefined;
-                    name.parent = ts.getParseTreeNode(currentLexicalScope); // ensure the parent is set to a parse tree node.
+                    ts.setParent(name, ts.getParseTreeNode(currentLexicalScope)); // ensure the parent is set to a parse tree node.
                     return name;
-                case 153 /* QualifiedName */:
+                case 157 /* QualifiedName */:
                     return serializeQualifiedNameAsExpression(node);
             }
         }
@@ -78899,14 +83860,16 @@ var ts;
          *                    qualified name at runtime.
          */
         function serializeQualifiedNameAsExpression(node) {
-            return ts.createPropertyAccess(serializeEntityNameAsExpression(node.left), node.right);
+            return factory.createPropertyAccessExpression(serializeEntityNameAsExpression(node.left), node.right);
         }
         /**
          * Gets an expression that points to the global "Symbol" constructor at runtime if it is
          * available.
          */
         function getGlobalSymbolNameWithFallback() {
-            return ts.createConditional(ts.createTypeCheck(ts.createIdentifier("Symbol"), "function"), ts.createIdentifier("Symbol"), ts.createIdentifier("Object"));
+            return factory.createConditionalExpression(factory.createTypeCheck(factory.createIdentifier("Symbol"), "function"), 
+            /*questionToken*/ undefined, factory.createIdentifier("Symbol"), 
+            /*colonToken*/ undefined, factory.createIdentifier("Object"));
         }
         /**
          * Gets an expression that points to the global "BigInt" constructor at runtime if it is
@@ -78914,8 +83877,10 @@ var ts;
          */
         function getGlobalBigIntNameWithFallback() {
             return languageVersion < 99 /* ESNext */
-                ? ts.createConditional(ts.createTypeCheck(ts.createIdentifier("BigInt"), "function"), ts.createIdentifier("BigInt"), ts.createIdentifier("Object"))
-                : ts.createIdentifier("BigInt");
+                ? factory.createConditionalExpression(factory.createTypeCheck(factory.createIdentifier("BigInt"), "function"), 
+                /*questionToken*/ undefined, factory.createIdentifier("BigInt"), 
+                /*colonToken*/ undefined, factory.createIdentifier("Object"))
+                : factory.createIdentifier("BigInt");
         }
         /**
          * Gets an expression that represents a property name (for decorated properties or enums).
@@ -78926,18 +83891,18 @@ var ts;
         function getExpressionForPropertyName(member, generateNameForComputedPropertyName) {
             var name = member.name;
             if (ts.isPrivateIdentifier(name)) {
-                return ts.createIdentifier("");
+                return factory.createIdentifier("");
             }
             else if (ts.isComputedPropertyName(name)) {
                 return generateNameForComputedPropertyName && !ts.isSimpleInlineableExpression(name.expression)
-                    ? ts.getGeneratedNameForNode(name)
+                    ? factory.getGeneratedNameForNode(name)
                     : name.expression;
             }
             else if (ts.isIdentifier(name)) {
-                return ts.createLiteral(ts.idText(name));
+                return factory.createStringLiteral(ts.idText(name));
             }
             else {
-                return ts.getSynthesizedClone(name);
+                return factory.cloneNode(name);
             }
         }
         /**
@@ -78957,9 +83922,9 @@ var ts;
                 var expression = ts.visitNode(name.expression, visitor, ts.isExpression);
                 var innerExpression = ts.skipPartiallyEmittedExpressions(expression);
                 if (!ts.isSimpleInlineableExpression(innerExpression)) {
-                    var generatedName = ts.getGeneratedNameForNode(name);
+                    var generatedName = factory.getGeneratedNameForNode(name);
                     hoistVariableDeclaration(generatedName);
-                    return ts.updateComputedPropertyName(name, ts.createAssignment(generatedName, expression));
+                    return factory.updateComputedPropertyName(name, factory.createAssignment(generatedName, expression));
                 }
             }
             return ts.visitNode(name, visitor, ts.isPropertyName);
@@ -78974,7 +83939,7 @@ var ts;
          * @param node The HeritageClause to transform.
          */
         function visitHeritageClause(node) {
-            if (node.token === 113 /* ImplementsKeyword */) {
+            if (node.token === 116 /* ImplementsKeyword */) {
                 // implements clauses are elided
                 return undefined;
             }
@@ -78989,8 +83954,8 @@ var ts;
          * @param node The ExpressionWithTypeArguments to transform.
          */
         function visitExpressionWithTypeArguments(node) {
-            return ts.updateExpressionWithTypeArguments(node
-            /*typeArguments*/ undefined, ts.visitNode(node.expression, visitor, ts.isLeftHandSideExpression));
+            return factory.updateExpressionWithTypeArguments(node, ts.visitNode(node.expression, visitor, ts.isLeftHandSideExpression)
+            /*typeArguments*/ undefined);
         }
         /**
          * Determines whether to emit a function-like declaration. We should not emit the
@@ -79005,7 +83970,7 @@ var ts;
             if (node.flags & 8388608 /* Ambient */) {
                 return undefined;
             }
-            var updated = ts.updateProperty(node, 
+            var updated = factory.updatePropertyDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, visitor, ts.isModifier), visitPropertyNameOfClassElement(node), 
             /*questionOrExclamationToken*/ undefined, 
             /*type*/ undefined, ts.visitNode(node.initializer, visitor));
@@ -79021,7 +83986,7 @@ var ts;
             if (!shouldEmitFunctionLikeDeclaration(node)) {
                 return undefined;
             }
-            return ts.updateConstructor(node, 
+            return factory.updateConstructorDeclaration(node, 
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, ts.visitParameterList(node.parameters, visitor, context), transformConstructorBody(node.body, node));
         }
@@ -79034,7 +83999,7 @@ var ts;
             var statements = [];
             var indexOfFirstStatement = 0;
             resumeLexicalEnvironment();
-            indexOfFirstStatement = ts.addPrologueDirectivesAndInitialSuperCall(constructor, statements, visitor);
+            indexOfFirstStatement = ts.addPrologueDirectivesAndInitialSuperCall(factory, constructor, statements, visitor);
             // Add parameters with property assignments. Transforms this:
             //
             //  constructor (public x, public y) {
@@ -79051,8 +84016,8 @@ var ts;
             // Add the existing statements, skipping the initial super call.
             ts.addRange(statements, ts.visitNodes(body.statements, visitor, ts.isStatement, indexOfFirstStatement));
             // End the lexical environment.
-            statements = ts.mergeLexicalEnvironment(statements, endLexicalEnvironment());
-            var block = ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), body.statements), /*multiLine*/ true);
+            statements = factory.mergeLexicalEnvironment(statements, endLexicalEnvironment());
+            var block = factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), body.statements), /*multiLine*/ true);
             ts.setTextRange(block, /*location*/ body);
             ts.setOriginalNode(block, body);
             return block;
@@ -79067,17 +84032,19 @@ var ts;
             if (!ts.isIdentifier(name)) {
                 return undefined;
             }
-            var propertyName = ts.getMutableClone(name);
+            // TODO(rbuckton): Does this need to be parented?
+            var propertyName = ts.setParent(ts.setTextRange(factory.cloneNode(name), name), name.parent);
             ts.setEmitFlags(propertyName, 1536 /* NoComments */ | 48 /* NoSourceMap */);
-            var localName = ts.getMutableClone(name);
+            // TODO(rbuckton): Does this need to be parented?
+            var localName = ts.setParent(ts.setTextRange(factory.cloneNode(name), name), name.parent);
             ts.setEmitFlags(localName, 1536 /* NoComments */);
-            return ts.startOnNewLine(ts.removeAllComments(ts.setTextRange(ts.setOriginalNode(ts.createExpressionStatement(ts.createAssignment(ts.setTextRange(ts.createPropertyAccess(ts.createThis(), propertyName), node.name), localName)), node), ts.moveRangePos(node, -1))));
+            return ts.startOnNewLine(ts.removeAllComments(ts.setTextRange(ts.setOriginalNode(factory.createExpressionStatement(factory.createAssignment(ts.setTextRange(factory.createPropertyAccessExpression(factory.createThis(), propertyName), node.name), localName)), node), ts.moveRangePos(node, -1))));
         }
         function visitMethodDeclaration(node) {
             if (!shouldEmitFunctionLikeDeclaration(node)) {
                 return undefined;
             }
-            var updated = ts.updateMethod(node, 
+            var updated = factory.updateMethodDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, visitPropertyNameOfClassElement(node), 
             /*questionToken*/ undefined, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
@@ -79097,15 +84064,15 @@ var ts;
          * @param node The declaration node.
          */
         function shouldEmitAccessorDeclaration(node) {
-            return !(ts.nodeIsMissing(node.body) && ts.hasModifier(node, 128 /* Abstract */));
+            return !(ts.nodeIsMissing(node.body) && ts.hasSyntacticModifier(node, 128 /* Abstract */));
         }
         function visitGetAccessor(node) {
             if (!shouldEmitAccessorDeclaration(node)) {
                 return undefined;
             }
-            var updated = ts.updateGetAccessor(node, 
+            var updated = factory.updateGetAccessorDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), visitPropertyNameOfClassElement(node), ts.visitParameterList(node.parameters, visitor, context), 
-            /*type*/ undefined, ts.visitFunctionBody(node.body, visitor, context) || ts.createBlock([]));
+            /*type*/ undefined, ts.visitFunctionBody(node.body, visitor, context) || factory.createBlock([]));
             if (updated !== node) {
                 // While we emit the source map for the node after skipping decorators and modifiers,
                 // we need to emit the comments for the original range.
@@ -79118,8 +84085,8 @@ var ts;
             if (!shouldEmitAccessorDeclaration(node)) {
                 return undefined;
             }
-            var updated = ts.updateSetAccessor(node, 
-            /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), visitPropertyNameOfClassElement(node), ts.visitParameterList(node.parameters, visitor, context), ts.visitFunctionBody(node.body, visitor, context) || ts.createBlock([]));
+            var updated = factory.updateSetAccessorDeclaration(node, 
+            /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), visitPropertyNameOfClassElement(node), ts.visitParameterList(node.parameters, visitor, context), ts.visitFunctionBody(node.body, visitor, context) || factory.createBlock([]));
             if (updated !== node) {
                 // While we emit the source map for the node after skipping decorators and modifiers,
                 // we need to emit the comments for the original range.
@@ -79130,12 +84097,12 @@ var ts;
         }
         function visitFunctionDeclaration(node) {
             if (!shouldEmitFunctionLikeDeclaration(node)) {
-                return ts.createNotEmittedStatement(node);
+                return factory.createNotEmittedStatement(node);
             }
-            var updated = ts.updateFunctionDeclaration(node, 
+            var updated = factory.updateFunctionDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, node.name, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
-            /*type*/ undefined, ts.visitFunctionBody(node.body, visitor, context) || ts.createBlock([]));
+            /*type*/ undefined, ts.visitFunctionBody(node.body, visitor, context) || factory.createBlock([]));
             if (isExportOfNamespace(node)) {
                 var statements = [updated];
                 addExportMemberAssignment(statements, node);
@@ -79145,15 +84112,15 @@ var ts;
         }
         function visitFunctionExpression(node) {
             if (!shouldEmitFunctionLikeDeclaration(node)) {
-                return ts.createOmittedExpression();
+                return factory.createOmittedExpression();
             }
-            var updated = ts.updateFunctionExpression(node, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, node.name, 
+            var updated = factory.updateFunctionExpression(node, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, node.name, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
-            /*type*/ undefined, ts.visitFunctionBody(node.body, visitor, context) || ts.createBlock([]));
+            /*type*/ undefined, ts.visitFunctionBody(node.body, visitor, context) || factory.createBlock([]));
             return updated;
         }
         function visitArrowFunction(node) {
-            var updated = ts.updateArrowFunction(node, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), 
+            var updated = factory.updateArrowFunction(node, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
             /*type*/ undefined, node.equalsGreaterThanToken, ts.visitFunctionBody(node.body, visitor, context));
             return updated;
@@ -79162,7 +84129,7 @@ var ts;
             if (ts.parameterIsThisKeyword(node)) {
                 return undefined;
             }
-            var updated = ts.updateParameter(node, 
+            var updated = factory.updateParameterDeclaration(node, 
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, node.dotDotDotToken, ts.visitNode(node.name, visitor, ts.isBindingName), 
             /*questionToken*/ undefined, 
@@ -79184,7 +84151,7 @@ var ts;
                     // elide statement if there are no initialized variables.
                     return undefined;
                 }
-                return ts.setTextRange(ts.createExpressionStatement(ts.inlineExpressions(ts.map(variables, transformInitializedVariable))), node);
+                return ts.setTextRange(factory.createExpressionStatement(factory.inlineExpressions(ts.map(variables, transformInitializedVariable))), node);
             }
             else {
                 return ts.visitEachChild(node, visitor, context);
@@ -79197,13 +84164,13 @@ var ts;
                 /*needsValue*/ false, createNamespaceExportExpression);
             }
             else {
-                return ts.setTextRange(ts.createAssignment(getNamespaceMemberNameWithSourceMapsAndWithoutComments(name), ts.visitNode(node.initializer, visitor, ts.isExpression)), 
+                return ts.setTextRange(factory.createAssignment(getNamespaceMemberNameWithSourceMapsAndWithoutComments(name), ts.visitNode(node.initializer, visitor, ts.isExpression)), 
                 /*location*/ node);
             }
         }
         function visitVariableDeclaration(node) {
-            return ts.updateTypeScriptVariableDeclaration(node, ts.visitNode(node.name, visitor, ts.isBindingName), 
-            /*exclaimationToken*/ undefined, 
+            return factory.updateVariableDeclaration(node, ts.visitNode(node.name, visitor, ts.isBindingName), 
+            /*exclamationToken*/ undefined, 
             /*type*/ undefined, ts.visitNode(node.initializer, visitor, ts.isExpression));
         }
         function visitParenthesizedExpression(node) {
@@ -79228,38 +84195,38 @@ var ts;
                 // HOWEVER - if there are leading comments on the expression itself, to handle ASI
                 // correctly for return and throw, we must keep the parenthesis
                 if (ts.length(ts.getLeadingCommentRangesOfNode(expression, currentSourceFile))) {
-                    return ts.updateParen(node, expression);
+                    return factory.updateParenthesizedExpression(node, expression);
                 }
-                return ts.createPartiallyEmittedExpression(expression, node);
+                return factory.createPartiallyEmittedExpression(expression, node);
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitAssertionExpression(node) {
             var expression = ts.visitNode(node.expression, visitor, ts.isExpression);
-            return ts.createPartiallyEmittedExpression(expression, node);
+            return factory.createPartiallyEmittedExpression(expression, node);
         }
         function visitNonNullExpression(node) {
             var expression = ts.visitNode(node.expression, visitor, ts.isLeftHandSideExpression);
-            return ts.createPartiallyEmittedExpression(expression, node);
+            return factory.createPartiallyEmittedExpression(expression, node);
         }
         function visitCallExpression(node) {
-            return ts.updateCall(node, ts.visitNode(node.expression, visitor, ts.isExpression), 
+            return factory.updateCallExpression(node, ts.visitNode(node.expression, visitor, ts.isExpression), 
             /*typeArguments*/ undefined, ts.visitNodes(node.arguments, visitor, ts.isExpression));
         }
         function visitNewExpression(node) {
-            return ts.updateNew(node, ts.visitNode(node.expression, visitor, ts.isExpression), 
+            return factory.updateNewExpression(node, ts.visitNode(node.expression, visitor, ts.isExpression), 
             /*typeArguments*/ undefined, ts.visitNodes(node.arguments, visitor, ts.isExpression));
         }
         function visitTaggedTemplateExpression(node) {
-            return ts.updateTaggedTemplate(node, ts.visitNode(node.tag, visitor, ts.isExpression), 
+            return factory.updateTaggedTemplateExpression(node, ts.visitNode(node.tag, visitor, ts.isExpression), 
             /*typeArguments*/ undefined, ts.visitNode(node.template, visitor, ts.isExpression));
         }
         function visitJsxSelfClosingElement(node) {
-            return ts.updateJsxSelfClosingElement(node, ts.visitNode(node.tagName, visitor, ts.isJsxTagNameExpression), 
+            return factory.updateJsxSelfClosingElement(node, ts.visitNode(node.tagName, visitor, ts.isJsxTagNameExpression), 
             /*typeArguments*/ undefined, ts.visitNode(node.attributes, visitor, ts.isJsxAttributes));
         }
         function visitJsxJsxOpeningElement(node) {
-            return ts.updateJsxOpeningElement(node, ts.visitNode(node.tagName, visitor, ts.isJsxTagNameExpression), 
+            return factory.updateJsxOpeningElement(node, ts.visitNode(node.tagName, visitor, ts.isJsxTagNameExpression), 
             /*typeArguments*/ undefined, ts.visitNode(node.attributes, visitor, ts.isJsxAttributes));
         }
         /**
@@ -79281,7 +84248,7 @@ var ts;
          */
         function visitEnumDeclaration(node) {
             if (!shouldEmitEnumDeclaration(node)) {
-                return ts.createNotEmittedStatement(node);
+                return factory.createNotEmittedStatement(node);
             }
             var statements = [];
             // We request to be advised when the printer is about to print this node. This allows
@@ -79302,27 +84269,27 @@ var ts;
             // `containerName` is the expression used inside of the enum for assignments.
             var containerName = getNamespaceContainerName(node);
             // `exportName` is the expression used within this node's container for any exported references.
-            var exportName = ts.hasModifier(node, 1 /* Export */)
-                ? ts.getExternalModuleOrNamespaceExportName(currentNamespaceContainerName, node, /*allowComments*/ false, /*allowSourceMaps*/ true)
-                : ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
+            var exportName = ts.hasSyntacticModifier(node, 1 /* Export */)
+                ? factory.getExternalModuleOrNamespaceExportName(currentNamespaceContainerName, node, /*allowComments*/ false, /*allowSourceMaps*/ true)
+                : factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
             //  x || (x = {})
             //  exports.x || (exports.x = {})
-            var moduleArg = ts.createLogicalOr(exportName, ts.createAssignment(exportName, ts.createObjectLiteral()));
+            var moduleArg = factory.createLogicalOr(exportName, factory.createAssignment(exportName, factory.createObjectLiteralExpression()));
             if (hasNamespaceQualifiedExportName(node)) {
                 // `localName` is the expression used within this node's containing scope for any local references.
-                var localName = ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
+                var localName = factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
                 //  x = (exports.x || (exports.x = {}))
-                moduleArg = ts.createAssignment(localName, moduleArg);
+                moduleArg = factory.createAssignment(localName, moduleArg);
             }
             //  (function (x) {
             //      x[x["y"] = 0] = "y";
             //      ...
             //  })(x || (x = {}));
-            var enumStatement = ts.createExpressionStatement(ts.createCall(ts.createFunctionExpression(
+            var enumStatement = factory.createExpressionStatement(factory.createCallExpression(factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
-            /*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, parameterName)], 
+            /*typeParameters*/ undefined, [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, parameterName)], 
             /*type*/ undefined, transformEnumBody(node, containerName)), 
             /*typeArguments*/ undefined, [moduleArg]));
             ts.setOriginalNode(enumStatement, node);
@@ -79336,7 +84303,7 @@ var ts;
             statements.push(enumStatement);
             // Add a DeclarationMarker for the enum to preserve trailing comments and mark
             // the end of the declaration.
-            statements.push(ts.createEndOfDeclarationMarker(node));
+            statements.push(factory.createEndOfDeclarationMarker(node));
             return statements;
         }
         /**
@@ -79353,7 +84320,7 @@ var ts;
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
             ts.addRange(statements, members);
             currentNamespaceContainerName = savedCurrentNamespaceLocalName;
-            return ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), /*location*/ node.members), 
+            return factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), /*location*/ node.members), 
             /*multiLine*/ true);
         }
         /**
@@ -79367,11 +84334,11 @@ var ts;
             // old emitter always generate 'expression' part of the name as-is.
             var name = getExpressionForPropertyName(member, /*generateNameForComputedPropertyName*/ false);
             var valueExpression = transformEnumMemberDeclarationValue(member);
-            var innerAssignment = ts.createAssignment(ts.createElementAccess(currentNamespaceContainerName, name), valueExpression);
+            var innerAssignment = factory.createAssignment(factory.createElementAccessExpression(currentNamespaceContainerName, name), valueExpression);
             var outerAssignment = valueExpression.kind === 10 /* StringLiteral */ ?
                 innerAssignment :
-                ts.createAssignment(ts.createElementAccess(currentNamespaceContainerName, innerAssignment), name);
-            return ts.setTextRange(ts.createExpressionStatement(ts.setTextRange(outerAssignment, member)), member);
+                factory.createAssignment(factory.createElementAccessExpression(currentNamespaceContainerName, innerAssignment), name);
+            return ts.setTextRange(factory.createExpressionStatement(ts.setTextRange(outerAssignment, member)), member);
         }
         /**
          * Transforms the value of an enum member.
@@ -79381,7 +84348,7 @@ var ts;
         function transformEnumMemberDeclarationValue(member) {
             var value = resolver.getConstantValue(member);
             if (value !== undefined) {
-                return ts.createLiteral(value);
+                return typeof value === "string" ? factory.createStringLiteral(value) : factory.createNumericLiteral(value);
             }
             else {
                 enableSubstitutionForNonQualifiedEnumMembers();
@@ -79389,7 +84356,7 @@ var ts;
                     return ts.visitNode(member.initializer, visitor, ts.isExpression);
                 }
                 else {
-                    return ts.createVoidZero();
+                    return factory.createVoidZero();
                 }
             }
         }
@@ -79424,7 +84391,7 @@ var ts;
          */
         function recordEmittedDeclarationInScope(node) {
             if (!currentScopeFirstDeclarationsOfName) {
-                currentScopeFirstDeclarationsOfName = ts.createUnderscoreEscapedMap();
+                currentScopeFirstDeclarationsOfName = new ts.Map();
             }
             var name = declaredNameInScope(node);
             if (!currentScopeFirstDeclarationsOfName.has(name)) {
@@ -79453,14 +84420,14 @@ var ts;
             // Emit a variable statement for the module. We emit top-level enums as a `var`
             // declaration to avoid static errors in global scripts scripts due to redeclaration.
             // enums in any other scope are emitted as a `let` declaration.
-            var statement = ts.createVariableStatement(ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true))
-            ], currentLexicalScope.kind === 290 /* SourceFile */ ? 0 /* None */ : 1 /* Let */));
+            var statement = factory.createVariableStatement(ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true))
+            ], currentLexicalScope.kind === 297 /* SourceFile */ ? 0 /* None */ : 1 /* Let */));
             ts.setOriginalNode(statement, node);
             recordEmittedDeclarationInScope(node);
             if (isFirstEmittedDeclarationInScope(node)) {
                 // Adjust the source map emit to match the old emitter.
-                if (node.kind === 248 /* EnumDeclaration */) {
+                if (node.kind === 255 /* EnumDeclaration */) {
                     ts.setSourceMapRange(statement.declarationList, node);
                 }
                 else {
@@ -79494,7 +84461,7 @@ var ts;
                 // declaration we do not emit a leading variable declaration. To preserve the
                 // begin/end semantics of the declararation and to properly handle exports
                 // we wrap the leading variable declaration in a `MergeDeclarationMarker`.
-                var mergeMarker = ts.createMergeDeclarationMarker(statement);
+                var mergeMarker = factory.createMergeDeclarationMarker(statement);
                 ts.setEmitFlags(mergeMarker, 1536 /* NoComments */ | 4194304 /* HasEndOfDeclarationMarker */);
                 statements.push(mergeMarker);
                 return false;
@@ -79509,7 +84476,7 @@ var ts;
          */
         function visitModuleDeclaration(node) {
             if (!shouldEmitModuleDeclaration(node)) {
-                return ts.createNotEmittedStatement(node);
+                return factory.createNotEmittedStatement(node);
             }
             ts.Debug.assertNode(node.name, ts.isIdentifier, "A TypeScript namespace should have an Identifier name.");
             enableSubstitutionForNamespaceExports();
@@ -79532,26 +84499,26 @@ var ts;
             // `containerName` is the expression used inside of the namespace for exports.
             var containerName = getNamespaceContainerName(node);
             // `exportName` is the expression used within this node's container for any exported references.
-            var exportName = ts.hasModifier(node, 1 /* Export */)
-                ? ts.getExternalModuleOrNamespaceExportName(currentNamespaceContainerName, node, /*allowComments*/ false, /*allowSourceMaps*/ true)
-                : ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
+            var exportName = ts.hasSyntacticModifier(node, 1 /* Export */)
+                ? factory.getExternalModuleOrNamespaceExportName(currentNamespaceContainerName, node, /*allowComments*/ false, /*allowSourceMaps*/ true)
+                : factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
             //  x || (x = {})
             //  exports.x || (exports.x = {})
-            var moduleArg = ts.createLogicalOr(exportName, ts.createAssignment(exportName, ts.createObjectLiteral()));
+            var moduleArg = factory.createLogicalOr(exportName, factory.createAssignment(exportName, factory.createObjectLiteralExpression()));
             if (hasNamespaceQualifiedExportName(node)) {
                 // `localName` is the expression used within this node's containing scope for any local references.
-                var localName = ts.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
+                var localName = factory.getLocalName(node, /*allowComments*/ false, /*allowSourceMaps*/ true);
                 //  x = (exports.x || (exports.x = {}))
-                moduleArg = ts.createAssignment(localName, moduleArg);
+                moduleArg = factory.createAssignment(localName, moduleArg);
             }
             //  (function (x_1) {
             //      x_1.y = ...;
             //  })(x || (x = {}));
-            var moduleStatement = ts.createExpressionStatement(ts.createCall(ts.createFunctionExpression(
+            var moduleStatement = factory.createExpressionStatement(factory.createCallExpression(factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
-            /*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, parameterName)], 
+            /*typeParameters*/ undefined, [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, parameterName)], 
             /*type*/ undefined, transformModuleBody(node, containerName)), 
             /*typeArguments*/ undefined, [moduleArg]));
             ts.setOriginalNode(moduleStatement, node);
@@ -79565,7 +84532,7 @@ var ts;
             statements.push(moduleStatement);
             // Add a DeclarationMarker for the namespace to preserve trailing comments and mark
             // the end of the declaration.
-            statements.push(ts.createEndOfDeclarationMarker(node));
+            statements.push(factory.createEndOfDeclarationMarker(node));
             return statements;
         }
         /**
@@ -79585,7 +84552,7 @@ var ts;
             var statementsLocation;
             var blockLocation;
             if (node.body) {
-                if (node.body.kind === 250 /* ModuleBlock */) {
+                if (node.body.kind === 257 /* ModuleBlock */) {
                     saveStateAndInvoke(node.body, function (body) { return ts.addRange(statements, ts.visitNodes(body.statements, namespaceElementVisitor, ts.isStatement)); });
                     statementsLocation = node.body.statements;
                     blockLocation = node.body;
@@ -79608,7 +84575,7 @@ var ts;
             currentNamespaceContainerName = savedCurrentNamespaceContainerName;
             currentNamespace = savedCurrentNamespace;
             currentScopeFirstDeclarationsOfName = savedCurrentScopeFirstDeclarationsOfName;
-            var block = ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), 
+            var block = factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), 
             /*location*/ statementsLocation), 
             /*multiLine*/ true);
             ts.setTextRange(block, blockLocation);
@@ -79632,13 +84599,13 @@ var ts;
             //     })(hi = hello.hi || (hello.hi = {}));
             // })(hello || (hello = {}));
             // We only want to emit comment on the namespace which contains block body itself, not the containing namespaces.
-            if (!node.body || node.body.kind !== 250 /* ModuleBlock */) {
+            if (!node.body || node.body.kind !== 257 /* ModuleBlock */) {
                 ts.setEmitFlags(block, ts.getEmitFlags(block) | 1536 /* NoComments */);
             }
             return block;
         }
         function getInnerMostModuleDeclarationFromDottedModule(moduleDeclaration) {
-            if (moduleDeclaration.body.kind === 249 /* ModuleDeclaration */) {
+            if (moduleDeclaration.body.kind === 256 /* ModuleDeclaration */) {
                 var recursiveInnerModule = getInnerMostModuleDeclarationFromDottedModule(moduleDeclaration.body);
                 return recursiveInnerModule || moduleDeclaration.body;
             }
@@ -79663,7 +84630,7 @@ var ts;
             return importClause ||
                 compilerOptions.importsNotUsedAsValues === 1 /* Preserve */ ||
                 compilerOptions.importsNotUsedAsValues === 2 /* Error */
-                ? ts.updateImportDeclaration(node, 
+                ? factory.updateImportDeclaration(node, 
                 /*decorators*/ undefined, 
                 /*modifiers*/ undefined, importClause, node.moduleSpecifier)
                 : undefined;
@@ -79680,7 +84647,7 @@ var ts;
             // Elide the import clause if we elide both its name and its named bindings.
             var name = resolver.isReferencedAliasDeclaration(node) ? node.name : undefined;
             var namedBindings = ts.visitNode(node.namedBindings, visitNamedImportBindings, ts.isNamedImportBindings);
-            return (name || namedBindings) ? ts.updateImportClause(node, name, namedBindings, /*isTypeOnly*/ false) : undefined;
+            return (name || namedBindings) ? factory.updateImportClause(node, /*isTypeOnly*/ false, name, namedBindings) : undefined;
         }
         /**
          * Visits named import bindings, eliding it if it is not referenced.
@@ -79688,14 +84655,14 @@ var ts;
          * @param node The named import bindings node.
          */
         function visitNamedImportBindings(node) {
-            if (node.kind === 256 /* NamespaceImport */) {
+            if (node.kind === 263 /* NamespaceImport */) {
                 // Elide a namespace import if it is not referenced.
                 return resolver.isReferencedAliasDeclaration(node) ? node : undefined;
             }
             else {
                 // Elide named imports if all of its import specifiers are elided.
                 var elements = ts.visitNodes(node.elements, visitImportSpecifier, ts.isImportSpecifier);
-                return ts.some(elements) ? ts.updateNamedImports(node, elements) : undefined;
+                return ts.some(elements) ? factory.updateNamedImports(node, elements) : undefined;
             }
         }
         /**
@@ -79742,9 +84709,9 @@ var ts;
             // Elide the export declaration if all of its named exports are elided.
             var exportClause = ts.visitNode(node.exportClause, visitNamedExportBindings, ts.isNamedExportBindings);
             return exportClause
-                ? ts.updateExportDeclaration(node, 
+                ? factory.updateExportDeclaration(node, 
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, exportClause, node.moduleSpecifier, node.isTypeOnly)
+                /*modifiers*/ undefined, node.isTypeOnly, exportClause, node.moduleSpecifier)
                 : undefined;
         }
         /**
@@ -79756,10 +84723,10 @@ var ts;
         function visitNamedExports(node) {
             // Elide the named exports if all of its export specifiers were elided.
             var elements = ts.visitNodes(node.elements, visitExportSpecifier, ts.isExportSpecifier);
-            return ts.some(elements) ? ts.updateNamedExports(node, elements) : undefined;
+            return ts.some(elements) ? factory.updateNamedExports(node, elements) : undefined;
         }
         function visitNamespaceExports(node) {
-            return ts.updateNamespaceExport(node, ts.visitNode(node.name, visitor, ts.isIdentifier));
+            return factory.updateNamespaceExport(node, ts.visitNode(node.name, visitor, ts.isIdentifier));
         }
         function visitNamedExportBindings(node) {
             return ts.isNamespaceExport(node) ? visitNamespaceExports(node) : visitNamedExports(node);
@@ -79796,7 +84763,7 @@ var ts;
                 var isReferenced = resolver.isReferencedAliasDeclaration(node);
                 // If the alias is unreferenced but we want to keep the import, replace with 'import "mod"'.
                 if (!isReferenced && compilerOptions.importsNotUsedAsValues === 1 /* Preserve */) {
-                    return ts.setOriginalNode(ts.setTextRange(ts.createImportDeclaration(
+                    return ts.setOriginalNode(ts.setTextRange(factory.createImportDeclaration(
                     /*decorators*/ undefined, 
                     /*modifiers*/ undefined, 
                     /*importClause*/ undefined, node.moduleReference.expression), node), node);
@@ -79806,13 +84773,14 @@ var ts;
             if (!shouldEmitImportEqualsDeclaration(node)) {
                 return undefined;
             }
-            var moduleReference = ts.createExpressionFromEntityName(node.moduleReference);
+            var moduleReference = ts.createExpressionFromEntityName(factory, node.moduleReference);
             ts.setEmitFlags(moduleReference, 1536 /* NoComments */ | 2048 /* NoNestedComments */);
             if (isNamedExternalModuleExport(node) || !isExportOfNamespace(node)) {
                 //  export var ${name} = ${moduleReference};
                 //  var ${name} = ${moduleReference};
-                return ts.setOriginalNode(ts.setTextRange(ts.createVariableStatement(ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), ts.createVariableDeclarationList([
-                    ts.setOriginalNode(ts.createVariableDeclaration(node.name, 
+                return ts.setOriginalNode(ts.setTextRange(factory.createVariableStatement(ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), factory.createVariableDeclarationList([
+                    ts.setOriginalNode(factory.createVariableDeclaration(node.name, 
+                    /*exclamationToken*/ undefined, 
                     /*type*/ undefined, moduleReference), node)
                 ])), node), node);
             }
@@ -79827,7 +84795,7 @@ var ts;
          * @param node The node to test.
          */
         function isExportOfNamespace(node) {
-            return currentNamespace !== undefined && ts.hasModifier(node, 1 /* Export */);
+            return currentNamespace !== undefined && ts.hasSyntacticModifier(node, 1 /* Export */);
         }
         /**
          * Gets a value indicating whether the node is exported from an external module.
@@ -79835,7 +84803,7 @@ var ts;
          * @param node The node to test.
          */
         function isExternalModuleExport(node) {
-            return currentNamespace === undefined && ts.hasModifier(node, 1 /* Export */);
+            return currentNamespace === undefined && ts.hasSyntacticModifier(node, 1 /* Export */);
         }
         /**
          * Gets a value indicating whether the node is a named export from an external module.
@@ -79844,7 +84812,7 @@ var ts;
          */
         function isNamedExternalModuleExport(node) {
             return isExternalModuleExport(node)
-                && !ts.hasModifier(node, 512 /* Default */);
+                && !ts.hasSyntacticModifier(node, 512 /* Default */);
         }
         /**
          * Gets a value indicating whether the node is the default export of an external module.
@@ -79853,35 +84821,35 @@ var ts;
          */
         function isDefaultExternalModuleExport(node) {
             return isExternalModuleExport(node)
-                && ts.hasModifier(node, 512 /* Default */);
+                && ts.hasSyntacticModifier(node, 512 /* Default */);
         }
         /**
          * Creates a statement for the provided expression. This is used in calls to `map`.
          */
         function expressionToStatement(expression) {
-            return ts.createExpressionStatement(expression);
+            return factory.createExpressionStatement(expression);
         }
         function addExportMemberAssignment(statements, node) {
-            var expression = ts.createAssignment(ts.getExternalModuleOrNamespaceExportName(currentNamespaceContainerName, node, /*allowComments*/ false, /*allowSourceMaps*/ true), ts.getLocalName(node));
+            var expression = factory.createAssignment(factory.getExternalModuleOrNamespaceExportName(currentNamespaceContainerName, node, /*allowComments*/ false, /*allowSourceMaps*/ true), factory.getLocalName(node));
             ts.setSourceMapRange(expression, ts.createRange(node.name ? node.name.pos : node.pos, node.end));
-            var statement = ts.createExpressionStatement(expression);
+            var statement = factory.createExpressionStatement(expression);
             ts.setSourceMapRange(statement, ts.createRange(-1, node.end));
             statements.push(statement);
         }
         function createNamespaceExport(exportName, exportValue, location) {
-            return ts.setTextRange(ts.createExpressionStatement(ts.createAssignment(ts.getNamespaceMemberName(currentNamespaceContainerName, exportName, /*allowComments*/ false, /*allowSourceMaps*/ true), exportValue)), location);
+            return ts.setTextRange(factory.createExpressionStatement(factory.createAssignment(factory.getNamespaceMemberName(currentNamespaceContainerName, exportName, /*allowComments*/ false, /*allowSourceMaps*/ true), exportValue)), location);
         }
         function createNamespaceExportExpression(exportName, exportValue, location) {
-            return ts.setTextRange(ts.createAssignment(getNamespaceMemberNameWithSourceMapsAndWithoutComments(exportName), exportValue), location);
+            return ts.setTextRange(factory.createAssignment(getNamespaceMemberNameWithSourceMapsAndWithoutComments(exportName), exportValue), location);
         }
         function getNamespaceMemberNameWithSourceMapsAndWithoutComments(name) {
-            return ts.getNamespaceMemberName(currentNamespaceContainerName, name, /*allowComments*/ false, /*allowSourceMaps*/ true);
+            return factory.getNamespaceMemberName(currentNamespaceContainerName, name, /*allowComments*/ false, /*allowSourceMaps*/ true);
         }
         /**
          * Gets the declaration name used inside of a namespace or enum.
          */
         function getNamespaceParameterName(node) {
-            var name = ts.getGeneratedNameForNode(node);
+            var name = factory.getGeneratedNameForNode(node);
             ts.setSourceMapRange(name, node.name);
             return name;
         }
@@ -79890,7 +84858,7 @@ var ts;
          * of its declaration.
          */
         function getNamespaceContainerName(node) {
-            return ts.getGeneratedNameForNode(node);
+            return factory.getGeneratedNameForNode(node);
         }
         /**
          * Gets a local alias for a class declaration if it is a decorated class with an internal
@@ -79900,24 +84868,24 @@ var ts;
         function getClassAliasIfNeeded(node) {
             if (resolver.getNodeCheckFlags(node) & 16777216 /* ClassWithConstructorReference */) {
                 enableSubstitutionForClassAliases();
-                var classAlias = ts.createUniqueName(node.name && !ts.isGeneratedIdentifier(node.name) ? ts.idText(node.name) : "default");
+                var classAlias = factory.createUniqueName(node.name && !ts.isGeneratedIdentifier(node.name) ? ts.idText(node.name) : "default");
                 classAliases[ts.getOriginalNodeId(node)] = classAlias;
                 hoistVariableDeclaration(classAlias);
                 return classAlias;
             }
         }
         function getClassPrototype(node) {
-            return ts.createPropertyAccess(ts.getDeclarationName(node), "prototype");
+            return factory.createPropertyAccessExpression(factory.getDeclarationName(node), "prototype");
         }
         function getClassMemberPrefix(node, member) {
-            return ts.hasModifier(member, 32 /* Static */)
-                ? ts.getDeclarationName(node)
+            return ts.hasSyntacticModifier(member, 32 /* Static */)
+                ? factory.getDeclarationName(node)
                 : getClassPrototype(node);
         }
         function enableSubstitutionForNonQualifiedEnumMembers() {
             if ((enabledSubstitutions & 8 /* NonQualifiedEnumMembers */) === 0) {
                 enabledSubstitutions |= 8 /* NonQualifiedEnumMembers */;
-                context.enableSubstitution(75 /* Identifier */);
+                context.enableSubstitution(78 /* Identifier */);
             }
         }
         function enableSubstitutionForClassAliases() {
@@ -79925,7 +84893,7 @@ var ts;
                 enabledSubstitutions |= 1 /* ClassAliases */;
                 // We need to enable substitutions for identifiers. This allows us to
                 // substitute class names inside of a class declaration.
-                context.enableSubstitution(75 /* Identifier */);
+                context.enableSubstitution(78 /* Identifier */);
                 // Keep track of class aliases.
                 classAliases = [];
             }
@@ -79935,17 +84903,17 @@ var ts;
                 enabledSubstitutions |= 2 /* NamespaceExports */;
                 // We need to enable substitutions for identifiers and shorthand property assignments. This allows us to
                 // substitute the names of exported members of a namespace.
-                context.enableSubstitution(75 /* Identifier */);
-                context.enableSubstitution(282 /* ShorthandPropertyAssignment */);
+                context.enableSubstitution(78 /* Identifier */);
+                context.enableSubstitution(289 /* ShorthandPropertyAssignment */);
                 // We need to be notified when entering and exiting namespaces.
-                context.enableEmitNotification(249 /* ModuleDeclaration */);
+                context.enableEmitNotification(256 /* ModuleDeclaration */);
             }
         }
         function isTransformedModuleDeclaration(node) {
-            return ts.getOriginalNode(node).kind === 249 /* ModuleDeclaration */;
+            return ts.getOriginalNode(node).kind === 256 /* ModuleDeclaration */;
         }
         function isTransformedEnumDeclaration(node) {
-            return ts.getOriginalNode(node).kind === 248 /* EnumDeclaration */;
+            return ts.getOriginalNode(node).kind === 255 /* EnumDeclaration */;
         }
         /**
          * Hook for node emit.
@@ -79994,21 +84962,21 @@ var ts;
                     // A shorthand property with an assignment initializer is probably part of a
                     // destructuring assignment
                     if (node.objectAssignmentInitializer) {
-                        var initializer = ts.createAssignment(exportedName, node.objectAssignmentInitializer);
-                        return ts.setTextRange(ts.createPropertyAssignment(name, initializer), node);
+                        var initializer = factory.createAssignment(exportedName, node.objectAssignmentInitializer);
+                        return ts.setTextRange(factory.createPropertyAssignment(name, initializer), node);
                     }
-                    return ts.setTextRange(ts.createPropertyAssignment(name, exportedName), node);
+                    return ts.setTextRange(factory.createPropertyAssignment(name, exportedName), node);
                 }
             }
             return node;
         }
         function substituteExpression(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return substituteExpressionIdentifier(node);
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     return substitutePropertyAccessExpression(node);
-                case 195 /* ElementAccessExpression */:
+                case 202 /* ElementAccessExpression */:
                     return substituteElementAccessExpression(node);
             }
             return node;
@@ -80030,7 +84998,7 @@ var ts;
                     if (declaration) {
                         var classAlias = classAliases[declaration.id]; // TODO: GH#18217
                         if (classAlias) {
-                            var clone_1 = ts.getSynthesizedClone(classAlias);
+                            var clone_1 = factory.cloneNode(classAlias);
                             ts.setSourceMapRange(clone_1, node);
                             ts.setCommentRange(clone_1, node);
                             return clone_1;
@@ -80046,11 +85014,11 @@ var ts;
                 // If we are nested within a namespace declaration, we may need to qualifiy
                 // an identifier that is exported from a merged namespace.
                 var container = resolver.getReferencedExportContainer(node, /*prefixLocals*/ false);
-                if (container && container.kind !== 290 /* SourceFile */) {
-                    var substitute = (applicableSubstitutions & 2 /* NamespaceExports */ && container.kind === 249 /* ModuleDeclaration */) ||
-                        (applicableSubstitutions & 8 /* NonQualifiedEnumMembers */ && container.kind === 248 /* EnumDeclaration */);
+                if (container && container.kind !== 297 /* SourceFile */) {
+                    var substitute = (applicableSubstitutions & 2 /* NamespaceExports */ && container.kind === 256 /* ModuleDeclaration */) ||
+                        (applicableSubstitutions & 8 /* NonQualifiedEnumMembers */ && container.kind === 255 /* EnumDeclaration */);
                     if (substitute) {
-                        return ts.setTextRange(ts.createPropertyAccess(ts.getGeneratedNameForNode(container), node), 
+                        return ts.setTextRange(factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(container), node), 
                         /*location*/ node);
                     }
                 }
@@ -80068,7 +85036,7 @@ var ts;
             if (constantValue !== undefined) {
                 // track the constant value on the node for the printer in needsDotDotForPropertyAccess
                 ts.setConstantValue(node, constantValue);
-                var substitute = ts.createLiteral(constantValue);
+                var substitute = typeof constantValue === "string" ? factory.createStringLiteral(constantValue) : factory.createNumericLiteral(constantValue);
                 if (!compilerOptions.removeComments) {
                     var originalNode = ts.getOriginalNode(node, ts.isAccessExpression);
                     var propertyName = ts.isPropertyAccessExpression(originalNode)
@@ -80088,57 +85056,6 @@ var ts;
         }
     }
     ts.transformTypeScript = transformTypeScript;
-    function createDecorateHelper(context, decoratorExpressions, target, memberName, descriptor, location) {
-        var argumentsArray = [];
-        argumentsArray.push(ts.createArrayLiteral(decoratorExpressions, /*multiLine*/ true));
-        argumentsArray.push(target);
-        if (memberName) {
-            argumentsArray.push(memberName);
-            if (descriptor) {
-                argumentsArray.push(descriptor);
-            }
-        }
-        context.requestEmitHelper(ts.decorateHelper);
-        return ts.setTextRange(ts.createCall(ts.getUnscopedHelperName("__decorate"), 
-        /*typeArguments*/ undefined, argumentsArray), location);
-    }
-    ts.decorateHelper = {
-        name: "typescript:decorate",
-        importName: "__decorate",
-        scoped: false,
-        priority: 2,
-        text: "\n            var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {\n                var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\n                if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\n                else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\n                return c > 3 && r && Object.defineProperty(target, key, r), r;\n            };"
-    };
-    function createMetadataHelper(context, metadataKey, metadataValue) {
-        context.requestEmitHelper(ts.metadataHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__metadata"), 
-        /*typeArguments*/ undefined, [
-            ts.createLiteral(metadataKey),
-            metadataValue
-        ]);
-    }
-    ts.metadataHelper = {
-        name: "typescript:metadata",
-        importName: "__metadata",
-        scoped: false,
-        priority: 3,
-        text: "\n            var __metadata = (this && this.__metadata) || function (k, v) {\n                if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(k, v);\n            };"
-    };
-    function createParamHelper(context, expression, parameterOffset, location) {
-        context.requestEmitHelper(ts.paramHelper);
-        return ts.setTextRange(ts.createCall(ts.getUnscopedHelperName("__param"), 
-        /*typeArguments*/ undefined, [
-            ts.createLiteral(parameterOffset),
-            expression
-        ]), location);
-    }
-    ts.paramHelper = {
-        name: "typescript:param",
-        importName: "__param",
-        scoped: false,
-        priority: 4,
-        text: "\n            var __param = (this && this.__param) || function (paramIndex, decorator) {\n                return function (target, key) { decorator(target, key, paramIndex); }\n            };"
-    };
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -80163,7 +85080,7 @@ var ts;
      * When --useDefineForClassFields is on, this transforms to ECMAScript semantics, with Object.defineProperty.
      */
     function transformClassFields(context) {
-        var hoistVariableDeclaration = context.hoistVariableDeclaration, endLexicalEnvironment = context.endLexicalEnvironment, resumeLexicalEnvironment = context.resumeLexicalEnvironment;
+        var factory = context.factory, hoistVariableDeclaration = context.hoistVariableDeclaration, endLexicalEnvironment = context.endLexicalEnvironment, resumeLexicalEnvironment = context.resumeLexicalEnvironment;
         var resolver = context.getEmitResolver();
         var compilerOptions = context.getCompilerOptions();
         var languageVersion = ts.getEmitScriptTarget(compilerOptions);
@@ -80184,7 +85101,7 @@ var ts;
         var pendingStatements;
         var privateIdentifierEnvironmentStack = [];
         var currentPrivateIdentifierEnvironment;
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             var options = context.getCompilerOptions();
             if (node.isDeclarationFile
@@ -80199,40 +85116,40 @@ var ts;
             if (!(node.transformFlags & 4194304 /* ContainsClassFields */))
                 return node;
             switch (node.kind) {
-                case 214 /* ClassExpression */:
-                case 245 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
                     return visitClassLike(node);
-                case 159 /* PropertyDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     return visitPropertyDeclaration(node);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return visitVariableStatement(node);
-                case 154 /* ComputedPropertyName */:
+                case 158 /* ComputedPropertyName */:
                     return visitComputedPropertyName(node);
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     return visitPropertyAccessExpression(node);
-                case 207 /* PrefixUnaryExpression */:
+                case 214 /* PrefixUnaryExpression */:
                     return visitPrefixUnaryExpression(node);
-                case 208 /* PostfixUnaryExpression */:
+                case 215 /* PostfixUnaryExpression */:
                     return visitPostfixUnaryExpression(node, /*valueIsDiscarded*/ false);
-                case 196 /* CallExpression */:
+                case 203 /* CallExpression */:
                     return visitCallExpression(node);
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return visitBinaryExpression(node);
-                case 76 /* PrivateIdentifier */:
+                case 79 /* PrivateIdentifier */:
                     return visitPrivateIdentifier(node);
-                case 226 /* ExpressionStatement */:
+                case 233 /* ExpressionStatement */:
                     return visitExpressionStatement(node);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return visitForStatement(node);
-                case 198 /* TaggedTemplateExpression */:
+                case 205 /* TaggedTemplateExpression */:
                     return visitTaggedTemplateExpression(node);
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitorDestructuringTarget(node) {
             switch (node.kind) {
-                case 193 /* ObjectLiteralExpression */:
-                case 192 /* ArrayLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
+                case 199 /* ArrayLiteralExpression */:
                     return visitAssignmentPattern(node);
                 default:
                     return visitor(node);
@@ -80246,7 +85163,7 @@ var ts;
             if (!shouldTransformPrivateFields) {
                 return node;
             }
-            return ts.setOriginalNode(ts.createIdentifier(""), node);
+            return ts.setOriginalNode(factory.createIdentifier(""), node);
         }
         /**
          * Visits the members of a class that has fields.
@@ -80255,20 +85172,20 @@ var ts;
          */
         function classElementVisitor(node) {
             switch (node.kind) {
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     // Constructors for classes using class fields are transformed in
                     // `visitClassDeclaration` or `visitClassExpression`.
                     return undefined;
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 161 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 165 /* MethodDeclaration */:
                     // Visit the name of the member (if it's a computed property name).
                     return ts.visitEachChild(node, classElementVisitor, context);
-                case 159 /* PropertyDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     return visitPropertyDeclaration(node);
-                case 154 /* ComputedPropertyName */:
+                case 158 /* ComputedPropertyName */:
                     return visitComputedPropertyName(node);
-                case 222 /* SemicolonClassElement */:
+                case 229 /* SemicolonClassElement */:
                     return node;
                 default:
                     return visitor(node);
@@ -80289,7 +85206,7 @@ var ts;
                 var expressions = pendingExpressions;
                 expressions.push(name.expression);
                 pendingExpressions = [];
-                node = ts.updateComputedPropertyName(node, ts.inlineExpressions(expressions));
+                node = factory.updateComputedPropertyName(node, factory.inlineExpressions(expressions));
             }
             return node;
         }
@@ -80297,7 +85214,7 @@ var ts;
             ts.Debug.assert(!ts.some(node.decorators));
             if (!shouldTransformPrivateFields && ts.isPrivateIdentifier(node.name)) {
                 // Initializer is elided as the field is initialized in transformConstructor.
-                return ts.updateProperty(node, 
+                return factory.updatePropertyDeclaration(node, 
                 /*decorators*/ undefined, ts.visitNodes(node.modifiers, visitor, ts.isModifier), node.name, 
                 /*questionOrExclamationToken*/ undefined, 
                 /*type*/ undefined, 
@@ -80308,7 +85225,7 @@ var ts;
             // the property name to the temporary variable.
             var expr = getPropertyNameExpressionIfNeeded(node.name, !!node.initializer || !!context.getCompilerOptions().useDefineForClassFields);
             if (expr && !ts.isSimpleInlineableExpression(expr)) {
-                (pendingExpressions || (pendingExpressions = [])).push(expr);
+                getPendingExpressions().push(expr);
             }
             return undefined;
         }
@@ -80316,7 +85233,7 @@ var ts;
             receiver = ts.visitNode(receiver, visitor, ts.isExpression);
             switch (info.placement) {
                 case 0 /* InstanceField */:
-                    return createClassPrivateFieldGetHelper(context, ts.nodeIsSynthesized(receiver) ? receiver : ts.getSynthesizedClone(receiver), info.weakMapName);
+                    return context.getEmitHelperFactory().createClassPrivateFieldGetHelper(ts.nodeIsSynthesized(receiver) ? receiver : factory.cloneNode(receiver), info.weakMapName);
                 default: return ts.Debug.fail("Unexpected private identifier placement");
             }
         }
@@ -80338,8 +85255,8 @@ var ts;
                 if (operator && (info = accessPrivateIdentifier(node.operand.name))) {
                     var receiver = ts.visitNode(node.operand.expression, visitor, ts.isExpression);
                     var _a = createCopiableReceiverExpr(receiver), readExpression = _a.readExpression, initializeExpression = _a.initializeExpression;
-                    var existingValue = ts.createPrefix(39 /* PlusToken */, createPrivateIdentifierAccess(info, readExpression));
-                    return ts.setOriginalNode(createPrivateIdentifierAssignment(info, initializeExpression || readExpression, ts.createBinary(existingValue, operator, ts.createLiteral(1)), 62 /* EqualsToken */), node);
+                    var existingValue = factory.createPrefixUnaryExpression(39 /* PlusToken */, createPrivateIdentifierAccess(info, readExpression));
+                    return ts.setOriginalNode(createPrivateIdentifierAssignment(info, initializeExpression || readExpression, factory.createBinaryExpression(existingValue, operator, factory.createNumericLiteral(1)), 62 /* EqualsToken */), node);
                 }
             }
             return ts.visitEachChild(node, visitor, context);
@@ -80353,11 +85270,11 @@ var ts;
                 if (operator && (info = accessPrivateIdentifier(node.operand.name))) {
                     var receiver = ts.visitNode(node.operand.expression, visitor, ts.isExpression);
                     var _a = createCopiableReceiverExpr(receiver), readExpression = _a.readExpression, initializeExpression = _a.initializeExpression;
-                    var existingValue = ts.createPrefix(39 /* PlusToken */, createPrivateIdentifierAccess(info, readExpression));
+                    var existingValue = factory.createPrefixUnaryExpression(39 /* PlusToken */, createPrivateIdentifierAccess(info, readExpression));
                     // Create a temporary variable to store the value returned by the expression.
-                    var returnValue = valueIsDiscarded ? undefined : ts.createTempVariable(hoistVariableDeclaration);
-                    return ts.setOriginalNode(ts.inlineExpressions(ts.compact([
-                        createPrivateIdentifierAssignment(info, initializeExpression || readExpression, ts.createBinary(returnValue ? ts.createAssignment(returnValue, existingValue) : existingValue, operator, ts.createLiteral(1)), 62 /* EqualsToken */),
+                    var returnValue = valueIsDiscarded ? undefined : factory.createTempVariable(hoistVariableDeclaration);
+                    return ts.setOriginalNode(factory.inlineExpressions(ts.compact([
+                        createPrivateIdentifierAssignment(info, initializeExpression || readExpression, factory.createBinaryExpression(returnValue ? factory.createAssignment(returnValue, existingValue) : existingValue, operator, factory.createNumericLiteral(1)), 62 /* EqualsToken */),
                         returnValue
                     ])), node);
                 }
@@ -80366,30 +85283,30 @@ var ts;
         }
         function visitForStatement(node) {
             if (node.incrementor && ts.isPostfixUnaryExpression(node.incrementor)) {
-                return ts.updateFor(node, ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), visitPostfixUnaryExpression(node.incrementor, /*valueIsDiscarded*/ true), ts.visitNode(node.statement, visitor, ts.isStatement));
+                return factory.updateForStatement(node, ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), visitPostfixUnaryExpression(node.incrementor, /*valueIsDiscarded*/ true), ts.visitNode(node.statement, visitor, ts.isStatement));
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitExpressionStatement(node) {
             if (ts.isPostfixUnaryExpression(node.expression)) {
-                return ts.updateExpressionStatement(node, visitPostfixUnaryExpression(node.expression, /*valueIsDiscarded*/ true));
+                return factory.updateExpressionStatement(node, visitPostfixUnaryExpression(node.expression, /*valueIsDiscarded*/ true));
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function createCopiableReceiverExpr(receiver) {
-            var clone = ts.nodeIsSynthesized(receiver) ? receiver : ts.getSynthesizedClone(receiver);
+            var clone = ts.nodeIsSynthesized(receiver) ? receiver : factory.cloneNode(receiver);
             if (ts.isSimpleInlineableExpression(receiver)) {
                 return { readExpression: clone, initializeExpression: undefined };
             }
-            var readExpression = ts.createTempVariable(hoistVariableDeclaration);
-            var initializeExpression = ts.createAssignment(readExpression, clone);
+            var readExpression = factory.createTempVariable(hoistVariableDeclaration);
+            var initializeExpression = factory.createAssignment(readExpression, clone);
             return { readExpression: readExpression, initializeExpression: initializeExpression };
         }
         function visitCallExpression(node) {
             if (shouldTransformPrivateFields && ts.isPrivateIdentifierPropertyAccessExpression(node.expression)) {
                 // Transform call expressions of private names to properly bind the `this` parameter.
-                var _a = ts.createCallBinding(node.expression, hoistVariableDeclaration, languageVersion), thisArg = _a.thisArg, target = _a.target;
-                return ts.updateCall(node, ts.createPropertyAccess(ts.visitNode(target, visitor), "call"), 
+                var _a = factory.createCallBinding(node.expression, hoistVariableDeclaration, languageVersion), thisArg = _a.thisArg, target = _a.target;
+                return factory.updateCallExpression(node, factory.createPropertyAccessExpression(ts.visitNode(target, visitor), "call"), 
                 /*typeArguments*/ undefined, __spreadArrays([ts.visitNode(thisArg, visitor, ts.isExpression)], ts.visitNodes(node.arguments, visitor, ts.isExpression)));
             }
             return ts.visitEachChild(node, visitor, context);
@@ -80397,9 +85314,10 @@ var ts;
         function visitTaggedTemplateExpression(node) {
             if (shouldTransformPrivateFields && ts.isPrivateIdentifierPropertyAccessExpression(node.tag)) {
                 // Bind the `this` correctly for tagged template literals when the tag is a private identifier property access.
-                var _a = ts.createCallBinding(node.tag, hoistVariableDeclaration, languageVersion), thisArg = _a.thisArg, target = _a.target;
-                return ts.updateTaggedTemplate(node, ts.createCall(ts.createPropertyAccess(ts.visitNode(target, visitor), "bind"), 
-                /*typeArguments*/ undefined, [ts.visitNode(thisArg, visitor, ts.isExpression)]), ts.visitNode(node.template, visitor, ts.isTemplateLiteral));
+                var _a = factory.createCallBinding(node.tag, hoistVariableDeclaration, languageVersion), thisArg = _a.thisArg, target = _a.target;
+                return factory.updateTaggedTemplateExpression(node, factory.createCallExpression(factory.createPropertyAccessExpression(ts.visitNode(target, visitor), "bind"), 
+                /*typeArguments*/ undefined, [ts.visitNode(thisArg, visitor, ts.isExpression)]), 
+                /*typeArguments*/ undefined, ts.visitNode(node.template, visitor, ts.isTemplateLiteral));
             }
             return ts.visitEachChild(node, visitor, context);
         }
@@ -80408,9 +85326,9 @@ var ts;
                 if (ts.isDestructuringAssignment(node)) {
                     var savedPendingExpressions = pendingExpressions;
                     pendingExpressions = undefined;
-                    node = ts.updateBinary(node, ts.visitNode(node.left, visitorDestructuringTarget), ts.visitNode(node.right, visitor), node.operatorToken);
+                    node = factory.updateBinaryExpression(node, ts.visitNode(node.left, visitorDestructuringTarget), node.operatorToken, ts.visitNode(node.right, visitor));
                     var expr = ts.some(pendingExpressions) ?
-                        ts.inlineExpressions(ts.compact(__spreadArrays(pendingExpressions, [node]))) :
+                        factory.inlineExpressions(ts.compact(__spreadArrays(pendingExpressions, [node]))) :
                         node;
                     pendingExpressions = savedPendingExpressions;
                     return expr;
@@ -80437,10 +85355,10 @@ var ts;
             right = ts.visitNode(right, visitor, ts.isExpression);
             if (ts.isCompoundAssignment(operator)) {
                 var _a = createCopiableReceiverExpr(receiver), readExpression = _a.readExpression, initializeExpression = _a.initializeExpression;
-                return createClassPrivateFieldSetHelper(context, initializeExpression || readExpression, info.weakMapName, ts.createBinary(createClassPrivateFieldGetHelper(context, readExpression, info.weakMapName), ts.getNonAssignmentOperatorForCompoundAssignment(operator), right));
+                return context.getEmitHelperFactory().createClassPrivateFieldSetHelper(initializeExpression || readExpression, info.weakMapName, factory.createBinaryExpression(context.getEmitHelperFactory().createClassPrivateFieldGetHelper(readExpression, info.weakMapName), ts.getNonAssignmentOperatorForCompoundAssignment(operator), right));
             }
             else {
-                return createClassPrivateFieldSetHelper(context, receiver, info.weakMapName, right);
+                return context.getEmitHelperFactory().createClassPrivateFieldSetHelper(receiver, info.weakMapName, right);
             }
         }
         /**
@@ -80469,15 +85387,15 @@ var ts;
                 return ts.visitEachChild(node, visitor, context);
             }
             var extendsClauseElement = ts.getEffectiveBaseTypeNode(node);
-            var isDerivedClass = !!(extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 100 /* NullKeyword */);
+            var isDerivedClass = !!(extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 103 /* NullKeyword */);
             var statements = [
-                ts.updateClassDeclaration(node, 
+                factory.updateClassDeclaration(node, 
                 /*decorators*/ undefined, node.modifiers, node.name, 
                 /*typeParameters*/ undefined, ts.visitNodes(node.heritageClauses, visitor, ts.isHeritageClause), transformClassMembers(node, isDerivedClass))
             ];
             // Write any pending expressions from elided or moved computed property names
             if (ts.some(pendingExpressions)) {
-                statements.push(ts.createExpressionStatement(ts.inlineExpressions(pendingExpressions)));
+                statements.push(factory.createExpressionStatement(factory.inlineExpressions(pendingExpressions)));
             }
             // Emit static property assignment. Because classDeclaration is lexically evaluated,
             // it is safe to emit static property assignment after classDeclaration
@@ -80486,7 +85404,7 @@ var ts;
             //                                  a lexical declaration such as a LexicalDeclaration or a ClassDeclaration.
             var staticProperties = ts.getProperties(node, /*requireInitializer*/ true, /*isStatic*/ true);
             if (ts.some(staticProperties)) {
-                addPropertyStatements(statements, staticProperties, ts.getInternalName(node));
+                addPropertyStatements(statements, staticProperties, factory.getInternalName(node));
             }
             return statements;
         }
@@ -80504,41 +85422,41 @@ var ts;
             var isDecoratedClassDeclaration = ts.isClassDeclaration(ts.getOriginalNode(node));
             var staticProperties = ts.getProperties(node, /*requireInitializer*/ true, /*isStatic*/ true);
             var extendsClauseElement = ts.getEffectiveBaseTypeNode(node);
-            var isDerivedClass = !!(extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 100 /* NullKeyword */);
-            var classExpression = ts.updateClassExpression(node, node.modifiers, node.name, 
+            var isDerivedClass = !!(extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 103 /* NullKeyword */);
+            var classExpression = factory.updateClassExpression(node, ts.visitNodes(node.decorators, visitor, ts.isDecorator), node.modifiers, node.name, 
             /*typeParameters*/ undefined, ts.visitNodes(node.heritageClauses, visitor, ts.isHeritageClause), transformClassMembers(node, isDerivedClass));
             if (ts.some(staticProperties) || ts.some(pendingExpressions)) {
                 if (isDecoratedClassDeclaration) {
                     ts.Debug.assertIsDefined(pendingStatements, "Decorated classes transformed by TypeScript are expected to be within a variable declaration.");
                     // Write any pending expressions from elided or moved computed property names
                     if (pendingStatements && pendingExpressions && ts.some(pendingExpressions)) {
-                        pendingStatements.push(ts.createExpressionStatement(ts.inlineExpressions(pendingExpressions)));
+                        pendingStatements.push(factory.createExpressionStatement(factory.inlineExpressions(pendingExpressions)));
                     }
                     if (pendingStatements && ts.some(staticProperties)) {
-                        addPropertyStatements(pendingStatements, staticProperties, ts.getInternalName(node));
+                        addPropertyStatements(pendingStatements, staticProperties, factory.getInternalName(node));
                     }
                     return classExpression;
                 }
                 else {
                     var expressions = [];
                     var isClassWithConstructorReference = resolver.getNodeCheckFlags(node) & 16777216 /* ClassWithConstructorReference */;
-                    var temp = ts.createTempVariable(hoistVariableDeclaration, !!isClassWithConstructorReference);
+                    var temp = factory.createTempVariable(hoistVariableDeclaration, !!isClassWithConstructorReference);
                     if (isClassWithConstructorReference) {
                         // record an alias as the class name is not in scope for statics.
                         enableSubstitutionForClassAliases();
-                        var alias = ts.getSynthesizedClone(temp);
+                        var alias = factory.cloneNode(temp);
                         alias.autoGenerateFlags &= ~8 /* ReservedInNestedScopes */;
                         classAliases[ts.getOriginalNodeId(node)] = alias;
                     }
                     // To preserve the behavior of the old emitter, we explicitly indent
                     // the body of a class with static initializers.
                     ts.setEmitFlags(classExpression, 65536 /* Indented */ | ts.getEmitFlags(classExpression));
-                    expressions.push(ts.startOnNewLine(ts.createAssignment(temp, classExpression)));
+                    expressions.push(ts.startOnNewLine(factory.createAssignment(temp, classExpression)));
                     // Add any pending expressions leftover from elided or relocated computed property names
                     ts.addRange(expressions, ts.map(pendingExpressions, ts.startOnNewLine));
                     ts.addRange(expressions, generateInitializedPropertyExpressions(staticProperties, temp));
                     expressions.push(ts.startOnNewLine(temp));
-                    return ts.inlineExpressions(expressions);
+                    return factory.inlineExpressions(expressions);
                 }
             }
             return classExpression;
@@ -80559,7 +85477,7 @@ var ts;
                 members.push(constructor);
             }
             ts.addRange(members, ts.visitNodes(node.members, classElementVisitor, ts.isClassElement));
-            return ts.setTextRange(ts.createNodeArray(members), /*location*/ node.members);
+            return ts.setTextRange(factory.createNodeArray(members), /*location*/ node.members);
         }
         function isPropertyDeclarationThatRequiresConstructorStatement(member) {
             if (!ts.isPropertyDeclaration(member) || ts.hasStaticModifier(member)) {
@@ -80583,7 +85501,7 @@ var ts;
             if (!body) {
                 return undefined;
             }
-            return ts.startOnNewLine(ts.setOriginalNode(ts.setTextRange(ts.createConstructor(
+            return ts.startOnNewLine(ts.setOriginalNode(ts.setTextRange(factory.createConstructorDeclaration(
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, parameters !== null && parameters !== void 0 ? parameters : [], body), constructor || node), constructor));
         }
@@ -80605,11 +85523,11 @@ var ts;
                 //
                 //  super(...arguments);
                 //
-                statements.push(ts.createExpressionStatement(ts.createCall(ts.createSuper(), 
-                /*typeArguments*/ undefined, [ts.createSpread(ts.createIdentifier("arguments"))])));
+                statements.push(factory.createExpressionStatement(factory.createCallExpression(factory.createSuper(), 
+                /*typeArguments*/ undefined, [factory.createSpreadElement(factory.createIdentifier("arguments"))])));
             }
             if (constructor) {
-                indexOfFirstStatement = ts.addPrologueDirectivesAndInitialSuperCall(constructor, statements, visitor);
+                indexOfFirstStatement = ts.addPrologueDirectivesAndInitialSuperCall(factory, constructor, statements, visitor);
             }
             // Add the property initializers. Transforms this:
             //
@@ -80633,13 +85551,13 @@ var ts;
                     indexOfFirstStatement = afterParameterProperties;
                 }
             }
-            addPropertyStatements(statements, properties, ts.createThis());
+            addPropertyStatements(statements, properties, factory.createThis());
             // Add existing statements, skipping the initial super call.
             if (constructor) {
                 ts.addRange(statements, ts.visitNodes(constructor.body.statements, visitor, ts.isStatement, indexOfFirstStatement));
             }
-            statements = ts.mergeLexicalEnvironment(statements, endLexicalEnvironment());
-            return ts.setTextRange(ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), 
+            statements = factory.mergeLexicalEnvironment(statements, endLexicalEnvironment());
+            return ts.setTextRange(factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), 
             /*location*/ constructor ? constructor.body.statements : node.members), 
             /*multiLine*/ true), 
             /*location*/ constructor ? constructor.body : undefined);
@@ -80651,13 +85569,13 @@ var ts;
          * @param receiver The receiver on which each property should be assigned.
          */
         function addPropertyStatements(statements, properties, receiver) {
-            for (var _i = 0, properties_8 = properties; _i < properties_8.length; _i++) {
-                var property = properties_8[_i];
+            for (var _i = 0, properties_7 = properties; _i < properties_7.length; _i++) {
+                var property = properties_7[_i];
                 var expression = transformProperty(property, receiver);
                 if (!expression) {
                     continue;
                 }
-                var statement = ts.createExpressionStatement(expression);
+                var statement = factory.createExpressionStatement(expression);
                 ts.setSourceMapRange(statement, ts.moveRangePastModifiers(property));
                 ts.setCommentRange(statement, property);
                 ts.setOriginalNode(statement, property);
@@ -80672,8 +85590,8 @@ var ts;
          */
         function generateInitializedPropertyExpressions(properties, receiver) {
             var expressions = [];
-            for (var _i = 0, properties_9 = properties; _i < properties_9.length; _i++) {
-                var property = properties_9[_i];
+            for (var _i = 0, properties_8 = properties; _i < properties_8.length; _i++) {
+                var property = properties_8[_i];
                 var expression = transformProperty(property, receiver);
                 if (!expression) {
                     continue;
@@ -80693,10 +85611,11 @@ var ts;
          * @param receiver The object receiving the property assignment.
          */
         function transformProperty(property, receiver) {
+            var _a;
             // We generate a name here in order to reuse the value cached by the relocated computed name expression (which uses the same generated name)
             var emitAssignment = !context.getCompilerOptions().useDefineForClassFields;
             var propertyName = ts.isComputedPropertyName(property.name) && !ts.isSimpleInlineableExpression(property.name.expression)
-                ? ts.updateComputedPropertyName(property.name, ts.getGeneratedNameForNode(property.name))
+                ? factory.updateComputedPropertyName(property.name, factory.getGeneratedNameForNode(property.name))
                 : property.name;
             if (shouldTransformPrivateFields && ts.isPrivateIdentifier(propertyName)) {
                 var privateIdentifierInfo = accessPrivateIdentifier(propertyName);
@@ -80718,19 +85637,18 @@ var ts;
                 return undefined;
             }
             var propertyOriginalNode = ts.getOriginalNode(property);
-            var initializer = property.initializer || emitAssignment ? ts.visitNode(property.initializer, visitor, ts.isExpression)
-                : ts.isParameterPropertyDeclaration(propertyOriginalNode, propertyOriginalNode.parent) && ts.isIdentifier(propertyName) ? propertyName
-                    : ts.createVoidZero();
+            var initializer = property.initializer || emitAssignment ? (_a = ts.visitNode(property.initializer, visitor, ts.isExpression)) !== null && _a !== void 0 ? _a : factory.createVoidZero() : ts.isParameterPropertyDeclaration(propertyOriginalNode, propertyOriginalNode.parent) && ts.isIdentifier(propertyName) ? propertyName
+                : factory.createVoidZero();
             if (emitAssignment || ts.isPrivateIdentifier(propertyName)) {
-                var memberAccess = ts.createMemberAccessForPropertyName(receiver, propertyName, /*location*/ propertyName);
-                return ts.createAssignment(memberAccess, initializer);
+                var memberAccess = ts.createMemberAccessForPropertyName(factory, receiver, propertyName, /*location*/ propertyName);
+                return factory.createAssignment(memberAccess, initializer);
             }
             else {
                 var name = ts.isComputedPropertyName(propertyName) ? propertyName.expression
-                    : ts.isIdentifier(propertyName) ? ts.createStringLiteral(ts.unescapeLeadingUnderscores(propertyName.escapedText))
+                    : ts.isIdentifier(propertyName) ? factory.createStringLiteral(ts.unescapeLeadingUnderscores(propertyName.escapedText))
                         : propertyName;
-                var descriptor = ts.createPropertyDescriptor({ value: initializer, configurable: true, writable: true, enumerable: true });
-                return ts.createObjectDefinePropertyCall(receiver, name, descriptor);
+                var descriptor = factory.createPropertyDescriptor({ value: initializer, configurable: true, writable: true, enumerable: true });
+                return factory.createObjectDefinePropertyCall(receiver, name, descriptor);
             }
         }
         function enableSubstitutionForClassAliases() {
@@ -80738,7 +85656,7 @@ var ts;
                 enabledSubstitutions |= 1 /* ClassAliases */;
                 // We need to enable substitutions for identifiers. This allows us to
                 // substitute class names inside of a class declaration.
-                context.enableSubstitution(75 /* Identifier */);
+                context.enableSubstitution(78 /* Identifier */);
                 // Keep track of class aliases.
                 classAliases = [];
             }
@@ -80758,7 +85676,7 @@ var ts;
         }
         function substituteExpression(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return substituteExpressionIdentifier(node);
             }
             return node;
@@ -80778,7 +85696,7 @@ var ts;
                     if (declaration) {
                         var classAlias = classAliases[declaration.id]; // TODO: GH#18217
                         if (classAlias) {
-                            var clone_2 = ts.getSynthesizedClone(classAlias);
+                            var clone_2 = factory.cloneNode(classAlias);
                             ts.setSourceMapRange(clone_2, node);
                             ts.setCommentRange(clone_2, node);
                             return clone_2;
@@ -80800,9 +85718,9 @@ var ts;
                 var inlinable = ts.isSimpleInlineableExpression(innerExpression);
                 var alreadyTransformed = ts.isAssignmentExpression(innerExpression) && ts.isGeneratedIdentifier(innerExpression.left);
                 if (!alreadyTransformed && !inlinable && shouldHoist) {
-                    var generatedName = ts.getGeneratedNameForNode(name);
+                    var generatedName = factory.getGeneratedNameForNode(name);
                     hoistVariableDeclaration(generatedName);
-                    return ts.createAssignment(generatedName, expression);
+                    return factory.createAssignment(generatedName, expression);
                 }
                 return (inlinable || ts.isIdentifier(innerExpression)) ? undefined : expression;
             }
@@ -80814,14 +85732,18 @@ var ts;
         function endPrivateIdentifierEnvironment() {
             currentPrivateIdentifierEnvironment = privateIdentifierEnvironmentStack.pop();
         }
+        function getPrivateIdentifierEnvironment() {
+            return currentPrivateIdentifierEnvironment || (currentPrivateIdentifierEnvironment = new ts.Map());
+        }
+        function getPendingExpressions() {
+            return pendingExpressions || (pendingExpressions = []);
+        }
         function addPrivateIdentifierToEnvironment(name) {
             var text = ts.getTextOfPropertyName(name);
-            var weakMapName = ts.createOptimisticUniqueName("_" + text.substring(1));
-            weakMapName.autoGenerateFlags |= 8 /* ReservedInNestedScopes */;
+            var weakMapName = factory.createUniqueName("_" + text.substring(1), 16 /* Optimistic */ | 8 /* ReservedInNestedScopes */);
             hoistVariableDeclaration(weakMapName);
-            (currentPrivateIdentifierEnvironment || (currentPrivateIdentifierEnvironment = ts.createUnderscoreEscapedMap()))
-                .set(name.escapedText, { placement: 0 /* InstanceField */, weakMapName: weakMapName });
-            (pendingExpressions || (pendingExpressions = [])).push(ts.createAssignment(weakMapName, ts.createNew(ts.createIdentifier("WeakMap"), 
+            getPrivateIdentifierEnvironment().set(name.escapedText, { placement: 0 /* InstanceField */, weakMapName: weakMapName });
+            getPendingExpressions().push(factory.createAssignment(weakMapName, factory.createNewExpression(factory.createIdentifier("WeakMap"), 
             /*typeArguments*/ undefined, [])));
         }
         function accessPrivateIdentifier(name) {
@@ -80844,7 +85766,7 @@ var ts;
             return undefined;
         }
         function wrapPrivateIdentifierForDestructuringTarget(node) {
-            var parameter = ts.getGeneratedNameForNode(node);
+            var parameter = factory.getGeneratedNameForNode(node);
             var info = accessPrivateIdentifier(node.name);
             if (!info) {
                 return ts.visitEachChild(node, visitor, context);
@@ -80853,22 +85775,21 @@ var ts;
             // We cannot copy `this` or `super` into the function because they will be bound
             // differently inside the function.
             if (ts.isThisProperty(node) || ts.isSuperProperty(node) || !ts.isSimpleCopiableExpression(node.expression)) {
-                receiver = ts.createTempVariable(hoistVariableDeclaration);
-                receiver.autoGenerateFlags |= 8 /* ReservedInNestedScopes */;
-                (pendingExpressions || (pendingExpressions = [])).push(ts.createBinary(receiver, 62 /* EqualsToken */, node.expression));
+                receiver = factory.createTempVariable(hoistVariableDeclaration, /*reservedInNestedScopes*/ true);
+                getPendingExpressions().push(factory.createBinaryExpression(receiver, 62 /* EqualsToken */, node.expression));
             }
-            return ts.createPropertyAccess(
+            return factory.createPropertyAccessExpression(
             // Explicit parens required because of v8 regression (https://bugs.chromium.org/p/v8/issues/detail?id=9560)
-            ts.createParen(ts.createObjectLiteral([
-                ts.createSetAccessor(
+            factory.createParenthesizedExpression(factory.createObjectLiteralExpression([
+                factory.createSetAccessorDeclaration(
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, "value", [ts.createParameter(
+                /*modifiers*/ undefined, "value", [factory.createParameterDeclaration(
                     /*decorators*/ undefined, 
                     /*modifiers*/ undefined, 
                     /*dotDotDotToken*/ undefined, parameter, 
                     /*questionToken*/ undefined, 
                     /*type*/ undefined, 
-                    /*initializer*/ undefined)], ts.createBlock([ts.createExpressionStatement(createPrivateIdentifierAssignment(info, receiver, parameter, 62 /* EqualsToken */))]))
+                    /*initializer*/ undefined)], factory.createBlock([factory.createExpressionStatement(createPrivateIdentifierAssignment(info, receiver, parameter, 62 /* EqualsToken */))]))
             ])), "value");
         }
         function visitArrayAssignmentTarget(node) {
@@ -80876,10 +85797,10 @@ var ts;
             if (target && ts.isPrivateIdentifierPropertyAccessExpression(target)) {
                 var wrapped = wrapPrivateIdentifierForDestructuringTarget(target);
                 if (ts.isAssignmentExpression(node)) {
-                    return ts.updateBinary(node, wrapped, ts.visitNode(node.right, visitor, ts.isExpression), node.operatorToken);
+                    return factory.updateBinaryExpression(node, wrapped, node.operatorToken, ts.visitNode(node.right, visitor, ts.isExpression));
                 }
                 else if (ts.isSpreadElement(node)) {
-                    return ts.updateSpread(node, wrapped);
+                    return factory.updateSpreadElement(node, wrapped);
                 }
                 else {
                     return wrapped;
@@ -80893,9 +85814,9 @@ var ts;
                 if (target && ts.isPrivateIdentifierPropertyAccessExpression(target)) {
                     var initializer = ts.getInitializerOfBindingOrAssignmentElement(node);
                     var wrapped = wrapPrivateIdentifierForDestructuringTarget(target);
-                    return ts.updatePropertyAssignment(node, ts.visitNode(node.name, visitor), initializer ? ts.createAssignment(wrapped, ts.visitNode(initializer, visitor)) : wrapped);
+                    return factory.updatePropertyAssignment(node, ts.visitNode(node.name, visitor), initializer ? factory.createAssignment(wrapped, ts.visitNode(initializer, visitor)) : wrapped);
                 }
-                return ts.updatePropertyAssignment(node, ts.visitNode(node.name, visitor), ts.visitNode(node.initializer, visitorDestructuringTarget));
+                return factory.updatePropertyAssignment(node, ts.visitNode(node.name, visitor), ts.visitNode(node.initializer, visitorDestructuringTarget));
             }
             return ts.visitNode(node, visitor);
         }
@@ -80908,7 +85829,7 @@ var ts;
                 //
                 // Transformation:
                 // [ { set value(x) { this.#myProp = x; } }.value ] = [ "hello" ];
-                return ts.updateArrayLiteral(node, ts.visitNodes(node.elements, visitArrayAssignmentTarget, ts.isExpression));
+                return factory.updateArrayLiteralExpression(node, ts.visitNodes(node.elements, visitArrayAssignmentTarget, ts.isExpression));
             }
             else {
                 // Transforms private names in destructuring assignment object bindings.
@@ -80918,32 +85839,14 @@ var ts;
                 //
                 // Transformation:
                 // ({ stringProperty: { set value(x) { this.#myProp = x; } }.value }) = { stringProperty: "hello" };
-                return ts.updateObjectLiteral(node, ts.visitNodes(node.properties, visitObjectAssignmentTarget, ts.isObjectLiteralElementLike));
+                return factory.updateObjectLiteralExpression(node, ts.visitNodes(node.properties, visitObjectAssignmentTarget, ts.isObjectLiteralElementLike));
             }
         }
     }
     ts.transformClassFields = transformClassFields;
     function createPrivateInstanceFieldInitializer(receiver, initializer, weakMapName) {
-        return ts.createCall(ts.createPropertyAccess(weakMapName, "set"), 
-        /*typeArguments*/ undefined, [receiver, initializer || ts.createVoidZero()]);
-    }
-    ts.classPrivateFieldGetHelper = {
-        name: "typescript:classPrivateFieldGet",
-        scoped: false,
-        text: "\n            var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, privateMap) {\n                if (!privateMap.has(receiver)) {\n                    throw new TypeError(\"attempted to get private field on non-instance\");\n                }\n                return privateMap.get(receiver);\n            };"
-    };
-    function createClassPrivateFieldGetHelper(context, receiver, privateField) {
-        context.requestEmitHelper(ts.classPrivateFieldGetHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__classPrivateFieldGet"), /* typeArguments */ undefined, [receiver, privateField]);
-    }
-    ts.classPrivateFieldSetHelper = {
-        name: "typescript:classPrivateFieldSet",
-        scoped: false,
-        text: "\n            var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, privateMap, value) {\n                if (!privateMap.has(receiver)) {\n                    throw new TypeError(\"attempted to set private field on non-instance\");\n                }\n                privateMap.set(receiver, value);\n                return value;\n            };"
-    };
-    function createClassPrivateFieldSetHelper(context, receiver, privateField, value) {
-        context.requestEmitHelper(ts.classPrivateFieldSetHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__classPrivateFieldSet"), /* typeArguments */ undefined, [receiver, privateField, value]);
+        return ts.factory.createCallExpression(ts.factory.createPropertyAccessExpression(weakMapName, "set"), 
+        /*typeArguments*/ undefined, [receiver, initializer || ts.factory.createVoidZero()]);
     }
 })(ts || (ts = {}));
 /*@internal*/
@@ -80960,7 +85863,7 @@ var ts;
         ContextFlags[ContextFlags["HasLexicalThis"] = 2] = "HasLexicalThis";
     })(ContextFlags || (ContextFlags = {}));
     function transformES2017(context) {
-        var resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var resolver = context.getEmitResolver();
         var compilerOptions = context.getCompilerOptions();
         var languageVersion = ts.getEmitScriptTarget(compilerOptions);
@@ -80990,7 +85893,7 @@ var ts;
         // Set new transformation hooks.
         context.onEmitNode = onEmitNode;
         context.onSubstituteNode = onSubstituteNode;
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
@@ -81031,34 +85934,34 @@ var ts;
                 return node;
             }
             switch (node.kind) {
-                case 126 /* AsyncKeyword */:
+                case 129 /* AsyncKeyword */:
                     // ES2017 async modifier should be elided for targets < ES2017
                     return undefined;
-                case 206 /* AwaitExpression */:
+                case 213 /* AwaitExpression */:
                     return visitAwaitExpression(node);
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     return doWithContext(1 /* NonTopLevel */ | 2 /* HasLexicalThis */, visitMethodDeclaration, node);
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return doWithContext(1 /* NonTopLevel */ | 2 /* HasLexicalThis */, visitFunctionDeclaration, node);
-                case 201 /* FunctionExpression */:
+                case 208 /* FunctionExpression */:
                     return doWithContext(1 /* NonTopLevel */ | 2 /* HasLexicalThis */, visitFunctionExpression, node);
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     return doWithContext(1 /* NonTopLevel */, visitArrowFunction, node);
-                case 194 /* PropertyAccessExpression */:
-                    if (capturedSuperProperties && ts.isPropertyAccessExpression(node) && node.expression.kind === 102 /* SuperKeyword */) {
-                        capturedSuperProperties.set(node.name.escapedText, true);
+                case 201 /* PropertyAccessExpression */:
+                    if (capturedSuperProperties && ts.isPropertyAccessExpression(node) && node.expression.kind === 105 /* SuperKeyword */) {
+                        capturedSuperProperties.add(node.name.escapedText);
                     }
                     return ts.visitEachChild(node, visitor, context);
-                case 195 /* ElementAccessExpression */:
-                    if (capturedSuperProperties && node.expression.kind === 102 /* SuperKeyword */) {
+                case 202 /* ElementAccessExpression */:
+                    if (capturedSuperProperties && node.expression.kind === 105 /* SuperKeyword */) {
                         hasSuperElementAccess = true;
                     }
                     return ts.visitEachChild(node, visitor, context);
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 162 /* Constructor */:
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 166 /* Constructor */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
                     return doWithContext(1 /* NonTopLevel */ | 2 /* HasLexicalThis */, visitDefault, node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -81067,27 +85970,27 @@ var ts;
         function asyncBodyVisitor(node) {
             if (ts.isNodeWithPossibleHoistedDeclaration(node)) {
                 switch (node.kind) {
-                    case 225 /* VariableStatement */:
+                    case 232 /* VariableStatement */:
                         return visitVariableStatementInAsyncBody(node);
-                    case 230 /* ForStatement */:
+                    case 237 /* ForStatement */:
                         return visitForStatementInAsyncBody(node);
-                    case 231 /* ForInStatement */:
+                    case 238 /* ForInStatement */:
                         return visitForInStatementInAsyncBody(node);
-                    case 232 /* ForOfStatement */:
+                    case 239 /* ForOfStatement */:
                         return visitForOfStatementInAsyncBody(node);
-                    case 280 /* CatchClause */:
+                    case 287 /* CatchClause */:
                         return visitCatchClauseInAsyncBody(node);
-                    case 223 /* Block */:
-                    case 237 /* SwitchStatement */:
-                    case 251 /* CaseBlock */:
-                    case 277 /* CaseClause */:
-                    case 278 /* DefaultClause */:
-                    case 240 /* TryStatement */:
-                    case 228 /* DoStatement */:
-                    case 229 /* WhileStatement */:
-                    case 227 /* IfStatement */:
-                    case 236 /* WithStatement */:
-                    case 238 /* LabeledStatement */:
+                    case 230 /* Block */:
+                    case 244 /* SwitchStatement */:
+                    case 258 /* CaseBlock */:
+                    case 284 /* CaseClause */:
+                    case 285 /* DefaultClause */:
+                    case 247 /* TryStatement */:
+                    case 235 /* DoStatement */:
+                    case 236 /* WhileStatement */:
+                    case 234 /* IfStatement */:
+                    case 243 /* WithStatement */:
+                    case 245 /* LabeledStatement */:
                         return ts.visitEachChild(node, asyncBodyVisitor, context);
                     default:
                         return ts.Debug.assertNever(node, "Unhandled node.");
@@ -81096,14 +85999,14 @@ var ts;
             return visitor(node);
         }
         function visitCatchClauseInAsyncBody(node) {
-            var catchClauseNames = ts.createUnderscoreEscapedMap();
+            var catchClauseNames = new ts.Set();
             recordDeclarationName(node.variableDeclaration, catchClauseNames); // TODO: GH#18217
             // names declared in a catch variable are block scoped
             var catchClauseUnshadowedNames;
             catchClauseNames.forEach(function (_, escapedName) {
                 if (enclosingFunctionParameterNames.has(escapedName)) {
                     if (!catchClauseUnshadowedNames) {
-                        catchClauseUnshadowedNames = ts.cloneMap(enclosingFunctionParameterNames);
+                        catchClauseUnshadowedNames = new ts.Set(enclosingFunctionParameterNames);
                     }
                     catchClauseUnshadowedNames.delete(escapedName);
                 }
@@ -81122,25 +86025,25 @@ var ts;
         function visitVariableStatementInAsyncBody(node) {
             if (isVariableDeclarationListWithCollidingName(node.declarationList)) {
                 var expression = visitVariableDeclarationListWithCollidingNames(node.declarationList, /*hasReceiver*/ false);
-                return expression ? ts.createExpressionStatement(expression) : undefined;
+                return expression ? factory.createExpressionStatement(expression) : undefined;
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitForInStatementInAsyncBody(node) {
-            return ts.updateForIn(node, isVariableDeclarationListWithCollidingName(node.initializer)
+            return factory.updateForInStatement(node, isVariableDeclarationListWithCollidingName(node.initializer)
                 ? visitVariableDeclarationListWithCollidingNames(node.initializer, /*hasReceiver*/ true)
-                : ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), ts.visitNode(node.statement, asyncBodyVisitor, ts.isStatement, ts.liftToBlock));
+                : ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), ts.visitNode(node.statement, asyncBodyVisitor, ts.isStatement, factory.liftToBlock));
         }
         function visitForOfStatementInAsyncBody(node) {
-            return ts.updateForOf(node, ts.visitNode(node.awaitModifier, visitor, ts.isToken), isVariableDeclarationListWithCollidingName(node.initializer)
+            return factory.updateForOfStatement(node, ts.visitNode(node.awaitModifier, visitor, ts.isToken), isVariableDeclarationListWithCollidingName(node.initializer)
                 ? visitVariableDeclarationListWithCollidingNames(node.initializer, /*hasReceiver*/ true)
-                : ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), ts.visitNode(node.statement, asyncBodyVisitor, ts.isStatement, ts.liftToBlock));
+                : ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), ts.visitNode(node.statement, asyncBodyVisitor, ts.isStatement, factory.liftToBlock));
         }
         function visitForStatementInAsyncBody(node) {
             var initializer = node.initializer; // TODO: GH#18217
-            return ts.updateFor(node, isVariableDeclarationListWithCollidingName(initializer)
+            return factory.updateForStatement(node, isVariableDeclarationListWithCollidingName(initializer)
                 ? visitVariableDeclarationListWithCollidingNames(initializer, /*hasReceiver*/ false)
-                : ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitor, ts.isExpression), ts.visitNode(node.statement, asyncBodyVisitor, ts.isStatement, ts.liftToBlock));
+                : ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitor, ts.isExpression), ts.visitNode(node.statement, asyncBodyVisitor, ts.isStatement, factory.liftToBlock));
         }
         /**
          * Visits an AwaitExpression node.
@@ -81154,7 +86057,7 @@ var ts;
             if (inTopLevelContext()) {
                 return ts.visitEachChild(node, visitor, context);
             }
-            return ts.setOriginalNode(ts.setTextRange(ts.createYield(
+            return ts.setOriginalNode(ts.setTextRange(factory.createYieldExpression(
             /*asteriskToken*/ undefined, ts.visitNode(node.expression, visitor, ts.isExpression)), node), node);
         }
         /**
@@ -81166,7 +86069,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitMethodDeclaration(node) {
-            return ts.updateMethod(node, 
+            return factory.updateMethodDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, visitor, ts.isModifier), node.asteriskToken, node.name, 
             /*questionToken*/ undefined, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
@@ -81183,7 +86086,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitFunctionDeclaration(node) {
-            return ts.updateFunctionDeclaration(node, 
+            return factory.updateFunctionDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, visitor, ts.isModifier), node.asteriskToken, node.name, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
             /*type*/ undefined, ts.getFunctionFlags(node) & 2 /* Async */
@@ -81199,7 +86102,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitFunctionExpression(node) {
-            return ts.updateFunctionExpression(node, ts.visitNodes(node.modifiers, visitor, ts.isModifier), node.asteriskToken, node.name, 
+            return factory.updateFunctionExpression(node, ts.visitNodes(node.modifiers, visitor, ts.isModifier), node.asteriskToken, node.name, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
             /*type*/ undefined, ts.getFunctionFlags(node) & 2 /* Async */
                 ? transformAsyncFunctionBody(node)
@@ -81214,7 +86117,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitArrowFunction(node) {
-            return ts.updateArrowFunction(node, ts.visitNodes(node.modifiers, visitor, ts.isModifier), 
+            return factory.updateArrowFunction(node, ts.visitNodes(node.modifiers, visitor, ts.isModifier), 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
             /*type*/ undefined, node.equalsGreaterThanToken, ts.getFunctionFlags(node) & 2 /* Async */
                 ? transformAsyncFunctionBody(node)
@@ -81223,7 +86126,7 @@ var ts;
         function recordDeclarationName(_a, names) {
             var name = _a.name;
             if (ts.isIdentifier(name)) {
-                names.set(name.escapedText, true);
+                names.add(name.escapedText);
             }
             else {
                 for (var _i = 0, _b = name.elements; _i < _b.length; _i++) {
@@ -81245,11 +86148,11 @@ var ts;
             var variables = ts.getInitializedVariables(node);
             if (variables.length === 0) {
                 if (hasReceiver) {
-                    return ts.visitNode(ts.convertToAssignmentElementTarget(node.declarations[0].name), visitor, ts.isExpression);
+                    return ts.visitNode(factory.converters.convertToAssignmentElementTarget(node.declarations[0].name), visitor, ts.isExpression);
                 }
                 return undefined;
             }
-            return ts.inlineExpressions(ts.map(variables, transformInitializedVariable));
+            return factory.inlineExpressions(ts.map(variables, transformInitializedVariable));
         }
         function hoistVariableDeclarationList(node) {
             ts.forEach(node.declarations, hoistVariable);
@@ -81269,7 +86172,7 @@ var ts;
             }
         }
         function transformInitializedVariable(node) {
-            var converted = ts.setSourceMapRange(ts.createAssignment(ts.convertToAssignmentElementTarget(node.name), node.initializer), node);
+            var converted = ts.setSourceMapRange(factory.createAssignment(factory.converters.convertToAssignmentElementTarget(node.name), node.initializer), node);
             return ts.visitNode(converted, visitor, ts.isExpression);
         }
         function collidesWithParameterName(_a) {
@@ -81292,7 +86195,7 @@ var ts;
             var original = ts.getOriginalNode(node, ts.isFunctionLike);
             var nodeType = original.type;
             var promiseConstructor = languageVersion < 2 /* ES2015 */ ? getPromiseConstructor(nodeType) : undefined;
-            var isArrowFunction = node.kind === 202 /* ArrowFunction */;
+            var isArrowFunction = node.kind === 209 /* ArrowFunction */;
             var hasLexicalArguments = (resolver.getNodeCheckFlags(node) & 8192 /* CaptureArguments */) !== 0;
             // An async function is emit as an outer function that calls an inner
             // generator function. To preserve lexical bindings, we pass the current
@@ -81300,7 +86203,7 @@ var ts;
             // passed to `__awaiter` is executed inside of the callback to the
             // promise constructor.
             var savedEnclosingFunctionParameterNames = enclosingFunctionParameterNames;
-            enclosingFunctionParameterNames = ts.createUnderscoreEscapedMap();
+            enclosingFunctionParameterNames = new ts.Set();
             for (var _i = 0, _a = node.parameters; _i < _a.length; _i++) {
                 var parameter = _a[_i];
                 recordDeclarationName(parameter, enclosingFunctionParameterNames);
@@ -81308,27 +86211,27 @@ var ts;
             var savedCapturedSuperProperties = capturedSuperProperties;
             var savedHasSuperElementAccess = hasSuperElementAccess;
             if (!isArrowFunction) {
-                capturedSuperProperties = ts.createUnderscoreEscapedMap();
+                capturedSuperProperties = new ts.Set();
                 hasSuperElementAccess = false;
             }
             var result;
             if (!isArrowFunction) {
                 var statements = [];
-                var statementOffset = ts.addPrologue(statements, node.body.statements, /*ensureUseStrict*/ false, visitor);
-                statements.push(ts.createReturn(createAwaiterHelper(context, inHasLexicalThisContext(), hasLexicalArguments, promiseConstructor, transformAsyncFunctionBodyWorker(node.body, statementOffset))));
+                var statementOffset = factory.copyPrologue(node.body.statements, statements, /*ensureUseStrict*/ false, visitor);
+                statements.push(factory.createReturnStatement(emitHelpers().createAwaiterHelper(inHasLexicalThisContext(), hasLexicalArguments, promiseConstructor, transformAsyncFunctionBodyWorker(node.body, statementOffset))));
                 ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
                 // Minor optimization, emit `_super` helper to capture `super` access in an arrow.
                 // This step isn't needed if we eventually transform this to ES5.
                 var emitSuperHelpers = languageVersion >= 2 /* ES2015 */ && resolver.getNodeCheckFlags(node) & (4096 /* AsyncMethodWithSuperBinding */ | 2048 /* AsyncMethodWithSuper */);
                 if (emitSuperHelpers) {
                     enableSubstitutionForAsyncMethodsWithSuper();
-                    if (ts.hasEntries(capturedSuperProperties)) {
-                        var variableStatement = createSuperAccessVariableStatement(resolver, node, capturedSuperProperties);
+                    if (capturedSuperProperties.size) {
+                        var variableStatement = createSuperAccessVariableStatement(factory, resolver, node, capturedSuperProperties);
                         substitutedSuperAccessors[ts.getNodeId(variableStatement)] = true;
                         ts.insertStatementsAfterStandardPrologue(statements, [variableStatement]);
                     }
                 }
-                var block = ts.createBlock(statements, /*multiLine*/ true);
+                var block = factory.createBlock(statements, /*multiLine*/ true);
                 ts.setTextRange(block, node.body);
                 if (emitSuperHelpers && hasSuperElementAccess) {
                     // Emit helpers for super element access expressions (`super[x]`).
@@ -81342,11 +86245,11 @@ var ts;
                 result = block;
             }
             else {
-                var expression = createAwaiterHelper(context, inHasLexicalThisContext(), hasLexicalArguments, promiseConstructor, transformAsyncFunctionBodyWorker(node.body));
+                var expression = emitHelpers().createAwaiterHelper(inHasLexicalThisContext(), hasLexicalArguments, promiseConstructor, transformAsyncFunctionBodyWorker(node.body));
                 var declarations = endLexicalEnvironment();
                 if (ts.some(declarations)) {
-                    var block = ts.convertToFunctionBody(expression);
-                    result = ts.updateBlock(block, ts.setTextRange(ts.createNodeArray(ts.concatenate(declarations, block.statements)), block.statements));
+                    var block = factory.converters.convertToFunctionBlock(expression);
+                    result = factory.updateBlock(block, ts.setTextRange(factory.createNodeArray(ts.concatenate(declarations, block.statements)), block.statements));
                 }
                 else {
                     result = expression;
@@ -81361,10 +86264,10 @@ var ts;
         }
         function transformAsyncFunctionBodyWorker(body, start) {
             if (ts.isBlock(body)) {
-                return ts.updateBlock(body, ts.visitNodes(body.statements, asyncBodyVisitor, ts.isStatement, start));
+                return factory.updateBlock(body, ts.visitNodes(body.statements, asyncBodyVisitor, ts.isStatement, start));
             }
             else {
-                return ts.convertToFunctionBody(ts.visitNode(body, asyncBodyVisitor, ts.isConciseBody));
+                return factory.converters.convertToFunctionBlock(ts.visitNode(body, asyncBodyVisitor, ts.isConciseBody));
             }
         }
         function getPromiseConstructor(type) {
@@ -81383,17 +86286,17 @@ var ts;
                 enabledSubstitutions |= 1 /* AsyncMethodsWithSuper */;
                 // We need to enable substitutions for call, property access, and element access
                 // if we need to rewrite super calls.
-                context.enableSubstitution(196 /* CallExpression */);
-                context.enableSubstitution(194 /* PropertyAccessExpression */);
-                context.enableSubstitution(195 /* ElementAccessExpression */);
+                context.enableSubstitution(203 /* CallExpression */);
+                context.enableSubstitution(201 /* PropertyAccessExpression */);
+                context.enableSubstitution(202 /* ElementAccessExpression */);
                 // We need to be notified when entering and exiting declarations that bind super.
-                context.enableEmitNotification(245 /* ClassDeclaration */);
-                context.enableEmitNotification(161 /* MethodDeclaration */);
-                context.enableEmitNotification(163 /* GetAccessor */);
-                context.enableEmitNotification(164 /* SetAccessor */);
-                context.enableEmitNotification(162 /* Constructor */);
+                context.enableEmitNotification(252 /* ClassDeclaration */);
+                context.enableEmitNotification(165 /* MethodDeclaration */);
+                context.enableEmitNotification(167 /* GetAccessor */);
+                context.enableEmitNotification(168 /* SetAccessor */);
+                context.enableEmitNotification(166 /* Constructor */);
                 // We need to be notified when entering the generated accessor arrow functions.
-                context.enableEmitNotification(225 /* VariableStatement */);
+                context.enableEmitNotification(232 /* VariableStatement */);
             }
         }
         /**
@@ -81441,23 +86344,23 @@ var ts;
         }
         function substituteExpression(node) {
             switch (node.kind) {
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     return substitutePropertyAccessExpression(node);
-                case 195 /* ElementAccessExpression */:
+                case 202 /* ElementAccessExpression */:
                     return substituteElementAccessExpression(node);
-                case 196 /* CallExpression */:
+                case 203 /* CallExpression */:
                     return substituteCallExpression(node);
             }
             return node;
         }
         function substitutePropertyAccessExpression(node) {
-            if (node.expression.kind === 102 /* SuperKeyword */) {
-                return ts.setTextRange(ts.createPropertyAccess(ts.createFileLevelUniqueName("_super"), node.name), node);
+            if (node.expression.kind === 105 /* SuperKeyword */) {
+                return ts.setTextRange(factory.createPropertyAccessExpression(factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */), node.name), node);
             }
             return node;
         }
         function substituteElementAccessExpression(node) {
-            if (node.expression.kind === 102 /* SuperKeyword */) {
+            if (node.expression.kind === 105 /* SuperKeyword */) {
                 return createSuperElementAccessInAsyncMethod(node.argumentExpression, node);
             }
             return node;
@@ -81468,35 +86371,35 @@ var ts;
                 var argumentExpression = ts.isPropertyAccessExpression(expression)
                     ? substitutePropertyAccessExpression(expression)
                     : substituteElementAccessExpression(expression);
-                return ts.createCall(ts.createPropertyAccess(argumentExpression, "call"), 
+                return factory.createCallExpression(factory.createPropertyAccessExpression(argumentExpression, "call"), 
                 /*typeArguments*/ undefined, __spreadArrays([
-                    ts.createThis()
+                    factory.createThis()
                 ], node.arguments));
             }
             return node;
         }
         function isSuperContainer(node) {
             var kind = node.kind;
-            return kind === 245 /* ClassDeclaration */
-                || kind === 162 /* Constructor */
-                || kind === 161 /* MethodDeclaration */
-                || kind === 163 /* GetAccessor */
-                || kind === 164 /* SetAccessor */;
+            return kind === 252 /* ClassDeclaration */
+                || kind === 166 /* Constructor */
+                || kind === 165 /* MethodDeclaration */
+                || kind === 167 /* GetAccessor */
+                || kind === 168 /* SetAccessor */;
         }
         function createSuperElementAccessInAsyncMethod(argumentExpression, location) {
             if (enclosingSuperContainerFlags & 4096 /* AsyncMethodWithSuperBinding */) {
-                return ts.setTextRange(ts.createPropertyAccess(ts.createCall(ts.createFileLevelUniqueName("_superIndex"), 
+                return ts.setTextRange(factory.createPropertyAccessExpression(factory.createCallExpression(factory.createUniqueName("_superIndex", 16 /* Optimistic */ | 32 /* FileLevel */), 
                 /*typeArguments*/ undefined, [argumentExpression]), "value"), location);
             }
             else {
-                return ts.setTextRange(ts.createCall(ts.createFileLevelUniqueName("_superIndex"), 
+                return ts.setTextRange(factory.createCallExpression(factory.createUniqueName("_superIndex", 16 /* Optimistic */ | 32 /* FileLevel */), 
                 /*typeArguments*/ undefined, [argumentExpression]), location);
             }
         }
     }
     ts.transformES2017 = transformES2017;
     /** Creates a variable named `_super` with accessor properties for the given property names. */
-    function createSuperAccessVariableStatement(resolver, node, names) {
+    function createSuperAccessVariableStatement(factory, resolver, node, names) {
         // Create a variable declaration with a getter/setter (if binding) definition for each name:
         //   const _super = Object.create(null, { x: { get: () => super.x, set: (v) => super.x = v }, ... });
         var hasBinding = (resolver.getNodeCheckFlags(node) & 4096 /* AsyncMethodWithSuperBinding */) !== 0;
@@ -81504,18 +86407,18 @@ var ts;
         names.forEach(function (_, key) {
             var name = ts.unescapeLeadingUnderscores(key);
             var getterAndSetter = [];
-            getterAndSetter.push(ts.createPropertyAssignment("get", ts.createArrowFunction(
+            getterAndSetter.push(factory.createPropertyAssignment("get", factory.createArrowFunction(
             /* modifiers */ undefined, 
             /* typeParameters */ undefined, 
             /* parameters */ [], 
             /* type */ undefined, 
-            /* equalsGreaterThanToken */ undefined, ts.setEmitFlags(ts.createPropertyAccess(ts.setEmitFlags(ts.createSuper(), 4 /* NoSubstitution */), name), 4 /* NoSubstitution */))));
+            /* equalsGreaterThanToken */ undefined, ts.setEmitFlags(factory.createPropertyAccessExpression(ts.setEmitFlags(factory.createSuper(), 4 /* NoSubstitution */), name), 4 /* NoSubstitution */))));
             if (hasBinding) {
-                getterAndSetter.push(ts.createPropertyAssignment("set", ts.createArrowFunction(
+                getterAndSetter.push(factory.createPropertyAssignment("set", factory.createArrowFunction(
                 /* modifiers */ undefined, 
                 /* typeParameters */ undefined, 
                 /* parameters */ [
-                    ts.createParameter(
+                    factory.createParameterDeclaration(
                     /* decorators */ undefined, 
                     /* modifiers */ undefined, 
                     /* dotDotDotToken */ undefined, "v", 
@@ -81524,56 +86427,22 @@ var ts;
                     /* initializer */ undefined)
                 ], 
                 /* type */ undefined, 
-                /* equalsGreaterThanToken */ undefined, ts.createAssignment(ts.setEmitFlags(ts.createPropertyAccess(ts.setEmitFlags(ts.createSuper(), 4 /* NoSubstitution */), name), 4 /* NoSubstitution */), ts.createIdentifier("v")))));
+                /* equalsGreaterThanToken */ undefined, factory.createAssignment(ts.setEmitFlags(factory.createPropertyAccessExpression(ts.setEmitFlags(factory.createSuper(), 4 /* NoSubstitution */), name), 4 /* NoSubstitution */), factory.createIdentifier("v")))));
             }
-            accessors.push(ts.createPropertyAssignment(name, ts.createObjectLiteral(getterAndSetter)));
+            accessors.push(factory.createPropertyAssignment(name, factory.createObjectLiteralExpression(getterAndSetter)));
         });
-        return ts.createVariableStatement(
-        /* modifiers */ undefined, ts.createVariableDeclarationList([
-            ts.createVariableDeclaration(ts.createFileLevelUniqueName("_super"), 
-            /* type */ undefined, ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Object"), "create"), 
+        return factory.createVariableStatement(
+        /* modifiers */ undefined, factory.createVariableDeclarationList([
+            factory.createVariableDeclaration(factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */), 
+            /*exclamationToken*/ undefined, 
+            /* type */ undefined, factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("Object"), "create"), 
             /* typeArguments */ undefined, [
-                ts.createNull(),
-                ts.createObjectLiteral(accessors, /* multiline */ true)
+                factory.createNull(),
+                factory.createObjectLiteralExpression(accessors, /* multiline */ true)
             ]))
         ], 2 /* Const */));
     }
     ts.createSuperAccessVariableStatement = createSuperAccessVariableStatement;
-    ts.awaiterHelper = {
-        name: "typescript:awaiter",
-        importName: "__awaiter",
-        scoped: false,
-        priority: 5,
-        text: "\n            var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n                function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n                return new (P || (P = Promise))(function (resolve, reject) {\n                    function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n                    function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n                    function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n                    step((generator = generator.apply(thisArg, _arguments || [])).next());\n                });\n            };"
-    };
-    function createAwaiterHelper(context, hasLexicalThis, hasLexicalArguments, promiseConstructor, body) {
-        context.requestEmitHelper(ts.awaiterHelper);
-        var generatorFunc = ts.createFunctionExpression(
-        /*modifiers*/ undefined, ts.createToken(41 /* AsteriskToken */), 
-        /*name*/ undefined, 
-        /*typeParameters*/ undefined, 
-        /*parameters*/ [], 
-        /*type*/ undefined, body);
-        // Mark this node as originally an async function
-        (generatorFunc.emitNode || (generatorFunc.emitNode = {})).flags |= 262144 /* AsyncFunctionBody */ | 524288 /* ReuseTempVariableScope */;
-        return ts.createCall(ts.getUnscopedHelperName("__awaiter"), 
-        /*typeArguments*/ undefined, [
-            hasLexicalThis ? ts.createThis() : ts.createVoidZero(),
-            hasLexicalArguments ? ts.createIdentifier("arguments") : ts.createVoidZero(),
-            promiseConstructor ? ts.createExpressionFromEntityName(promiseConstructor) : ts.createVoidZero(),
-            generatorFunc
-        ]);
-    }
-    ts.asyncSuperHelper = {
-        name: "typescript:async-super",
-        scoped: true,
-        text: ts.helperString(__makeTemplateObject(["\n            const ", " = name => super[name];"], ["\n            const ", " = name => super[name];"]), "_superIndex")
-    };
-    ts.advancedAsyncSuperHelper = {
-        name: "typescript:advanced-async-super",
-        scoped: true,
-        text: ts.helperString(__makeTemplateObject(["\n            const ", " = (function (geti, seti) {\n                const cache = Object.create(null);\n                return name => cache[name] || (cache[name] = { get value() { return geti(name); }, set value(v) { seti(name, v); } });\n            })(name => super[name], (name, value) => super[name] = value);"], ["\n            const ", " = (function (geti, seti) {\n                const cache = Object.create(null);\n                return name => cache[name] || (cache[name] = { get value() { return geti(name); }, set value(v) { seti(name, v); } });\n            })(name => super[name], (name, value) => super[name] = value);"]), "_superIndex")
-    };
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -81608,7 +86477,7 @@ var ts;
         HierarchyFacts[HierarchyFacts["IterationStatementExcludes"] = 0] = "IterationStatementExcludes";
     })(HierarchyFacts || (HierarchyFacts = {}));
     function transformES2018(context) {
-        var resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var resolver = context.getEmitResolver();
         var compilerOptions = context.getCompilerOptions();
         var languageVersion = ts.getEmitScriptTarget(compilerOptions);
@@ -81629,7 +86498,7 @@ var ts;
         var hasSuperElementAccess;
         /** A set of node IDs for generated super accessors. */
         var substitutedSuperAccessors = [];
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         function affectsSubtree(excludeFacts, includeFacts) {
             return hierarchyFacts !== (hierarchyFacts & ~excludeFacts | includeFacts);
         }
@@ -81652,7 +86521,7 @@ var ts;
             hierarchyFacts = ancestorFacts;
         }
         function recordTaggedTemplateString(temp) {
-            taggedTemplateStringDeclarations = ts.append(taggedTemplateStringDeclarations, ts.createVariableDeclaration(temp));
+            taggedTemplateStringDeclarations = ts.append(taggedTemplateStringDeclarations, factory.createVariableDeclaration(temp));
         }
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
@@ -81666,13 +86535,13 @@ var ts;
             return visited;
         }
         function visitor(node) {
-            return visitorWorker(node, /*noDestructuringValue*/ false);
+            return visitorWorker(node, /*expressionResultIsUnused*/ false);
         }
-        function visitorNoDestructuringValue(node) {
-            return visitorWorker(node, /*noDestructuringValue*/ true);
+        function visitorWithUnusedExpressionResult(node) {
+            return visitorWorker(node, /*expressionResultIsUnused*/ true);
         }
         function visitorNoAsyncModifier(node) {
-            if (node.kind === 126 /* AsyncKeyword */) {
+            if (node.kind === 129 /* AsyncKeyword */) {
                 return undefined;
             }
             return node;
@@ -81689,73 +86558,79 @@ var ts;
         function visitDefault(node) {
             return ts.visitEachChild(node, visitor, context);
         }
-        function visitorWorker(node, noDestructuringValue) {
+        /**
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
+         */
+        function visitorWorker(node, expressionResultIsUnused) {
             if ((node.transformFlags & 32 /* ContainsES2018 */) === 0) {
                 return node;
             }
             switch (node.kind) {
-                case 206 /* AwaitExpression */:
+                case 213 /* AwaitExpression */:
                     return visitAwaitExpression(node);
-                case 212 /* YieldExpression */:
+                case 219 /* YieldExpression */:
                     return visitYieldExpression(node);
-                case 235 /* ReturnStatement */:
+                case 242 /* ReturnStatement */:
                     return visitReturnStatement(node);
-                case 238 /* LabeledStatement */:
+                case 245 /* LabeledStatement */:
                     return visitLabeledStatement(node);
-                case 193 /* ObjectLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
                     return visitObjectLiteralExpression(node);
-                case 209 /* BinaryExpression */:
-                    return visitBinaryExpression(node, noDestructuringValue);
-                case 280 /* CatchClause */:
+                case 216 /* BinaryExpression */:
+                    return visitBinaryExpression(node, expressionResultIsUnused);
+                case 337 /* CommaListExpression */:
+                    return visitCommaListExpression(node, expressionResultIsUnused);
+                case 287 /* CatchClause */:
                     return visitCatchClause(node);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return visitVariableStatement(node);
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return visitVariableDeclaration(node);
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
-                case 231 /* ForInStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
+                case 238 /* ForInStatement */:
                     return doWithHierarchyFacts(visitDefault, node, 0 /* IterationStatementExcludes */, 2 /* IterationStatementIncludes */);
-                case 232 /* ForOfStatement */:
+                case 239 /* ForOfStatement */:
                     return visitForOfStatement(node, /*outermostLabeledStatement*/ undefined);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return doWithHierarchyFacts(visitForStatement, node, 0 /* IterationStatementExcludes */, 2 /* IterationStatementIncludes */);
-                case 205 /* VoidExpression */:
+                case 212 /* VoidExpression */:
                     return visitVoidExpression(node);
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     return doWithHierarchyFacts(visitConstructorDeclaration, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     return doWithHierarchyFacts(visitMethodDeclaration, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
-                case 163 /* GetAccessor */:
+                case 167 /* GetAccessor */:
                     return doWithHierarchyFacts(visitGetAccessorDeclaration, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
-                case 164 /* SetAccessor */:
+                case 168 /* SetAccessor */:
                     return doWithHierarchyFacts(visitSetAccessorDeclaration, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return doWithHierarchyFacts(visitFunctionDeclaration, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
-                case 201 /* FunctionExpression */:
+                case 208 /* FunctionExpression */:
                     return doWithHierarchyFacts(visitFunctionExpression, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     return doWithHierarchyFacts(visitArrowFunction, node, 2 /* ArrowFunctionExcludes */, 0 /* ArrowFunctionIncludes */);
-                case 156 /* Parameter */:
+                case 160 /* Parameter */:
                     return visitParameter(node);
-                case 226 /* ExpressionStatement */:
+                case 233 /* ExpressionStatement */:
                     return visitExpressionStatement(node);
-                case 200 /* ParenthesizedExpression */:
-                    return visitParenthesizedExpression(node, noDestructuringValue);
-                case 198 /* TaggedTemplateExpression */:
+                case 207 /* ParenthesizedExpression */:
+                    return visitParenthesizedExpression(node, expressionResultIsUnused);
+                case 205 /* TaggedTemplateExpression */:
                     return visitTaggedTemplateExpression(node);
-                case 194 /* PropertyAccessExpression */:
-                    if (capturedSuperProperties && ts.isPropertyAccessExpression(node) && node.expression.kind === 102 /* SuperKeyword */) {
-                        capturedSuperProperties.set(node.name.escapedText, true);
+                case 201 /* PropertyAccessExpression */:
+                    if (capturedSuperProperties && ts.isPropertyAccessExpression(node) && node.expression.kind === 105 /* SuperKeyword */) {
+                        capturedSuperProperties.add(node.name.escapedText);
                     }
                     return ts.visitEachChild(node, visitor, context);
-                case 195 /* ElementAccessExpression */:
-                    if (capturedSuperProperties && node.expression.kind === 102 /* SuperKeyword */) {
+                case 202 /* ElementAccessExpression */:
+                    if (capturedSuperProperties && node.expression.kind === 105 /* SuperKeyword */) {
                         hasSuperElementAccess = true;
                     }
                     return ts.visitEachChild(node, visitor, context);
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
                     return doWithHierarchyFacts(visitDefault, node, 2 /* ClassOrFunctionExcludes */, 1 /* ClassOrFunctionIncludes */);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -81763,7 +86638,7 @@ var ts;
         }
         function visitAwaitExpression(node) {
             if (enclosingFunctionFlags & 2 /* Async */ && enclosingFunctionFlags & 1 /* Generator */) {
-                return ts.setOriginalNode(ts.setTextRange(ts.createYield(createAwaitHelper(context, ts.visitNode(node.expression, visitor, ts.isExpression))), 
+                return ts.setOriginalNode(ts.setTextRange(factory.createYieldExpression(/*asteriskToken*/ undefined, emitHelpers().createAwaitHelper(ts.visitNode(node.expression, visitor, ts.isExpression))), 
                 /*location*/ node), node);
             }
             return ts.visitEachChild(node, visitor, context);
@@ -81771,28 +86646,30 @@ var ts;
         function visitYieldExpression(node) {
             if (enclosingFunctionFlags & 2 /* Async */ && enclosingFunctionFlags & 1 /* Generator */) {
                 if (node.asteriskToken) {
-                    var expression = ts.visitNode(node.expression, visitor, ts.isExpression);
-                    return ts.setOriginalNode(ts.setTextRange(ts.createYield(createAwaitHelper(context, ts.updateYield(node, node.asteriskToken, createAsyncDelegatorHelper(context, createAsyncValuesHelper(context, expression, expression), expression)))), node), node);
+                    var expression = ts.visitNode(ts.Debug.assertDefined(node.expression), visitor, ts.isExpression);
+                    return ts.setOriginalNode(ts.setTextRange(factory.createYieldExpression(
+                    /*asteriskToken*/ undefined, emitHelpers().createAwaitHelper(factory.updateYieldExpression(node, node.asteriskToken, ts.setTextRange(emitHelpers().createAsyncDelegatorHelper(ts.setTextRange(emitHelpers().createAsyncValuesHelper(expression), expression)), expression)))), node), node);
                 }
-                return ts.setOriginalNode(ts.setTextRange(ts.createYield(createDownlevelAwait(node.expression
+                return ts.setOriginalNode(ts.setTextRange(factory.createYieldExpression(
+                /*asteriskToken*/ undefined, createDownlevelAwait(node.expression
                     ? ts.visitNode(node.expression, visitor, ts.isExpression)
-                    : ts.createVoidZero())), node), node);
+                    : factory.createVoidZero())), node), node);
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitReturnStatement(node) {
             if (enclosingFunctionFlags & 2 /* Async */ && enclosingFunctionFlags & 1 /* Generator */) {
-                return ts.updateReturn(node, createDownlevelAwait(node.expression ? ts.visitNode(node.expression, visitor, ts.isExpression) : ts.createVoidZero()));
+                return factory.updateReturnStatement(node, createDownlevelAwait(node.expression ? ts.visitNode(node.expression, visitor, ts.isExpression) : factory.createVoidZero()));
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitLabeledStatement(node) {
             if (enclosingFunctionFlags & 2 /* Async */) {
                 var statement = ts.unwrapInnermostStatementOfLabel(node);
-                if (statement.kind === 232 /* ForOfStatement */ && statement.awaitModifier) {
+                if (statement.kind === 239 /* ForOfStatement */ && statement.awaitModifier) {
                     return visitForOfStatement(statement, node);
                 }
-                return ts.restoreEnclosingLabel(ts.visitNode(statement, visitor, ts.isStatement, ts.liftToBlock), node);
+                return factory.restoreEnclosingLabel(ts.visitNode(statement, visitor, ts.isStatement, factory.liftToBlock), node);
             }
             return ts.visitEachChild(node, visitor, context);
         }
@@ -81801,22 +86678,22 @@ var ts;
             var objects = [];
             for (var _i = 0, elements_4 = elements; _i < elements_4.length; _i++) {
                 var e = elements_4[_i];
-                if (e.kind === 283 /* SpreadAssignment */) {
+                if (e.kind === 290 /* SpreadAssignment */) {
                     if (chunkObject) {
-                        objects.push(ts.createObjectLiteral(chunkObject));
+                        objects.push(factory.createObjectLiteralExpression(chunkObject));
                         chunkObject = undefined;
                     }
                     var target = e.expression;
                     objects.push(ts.visitNode(target, visitor, ts.isExpression));
                 }
                 else {
-                    chunkObject = ts.append(chunkObject, e.kind === 281 /* PropertyAssignment */
-                        ? ts.createPropertyAssignment(e.name, ts.visitNode(e.initializer, visitor, ts.isExpression))
+                    chunkObject = ts.append(chunkObject, e.kind === 288 /* PropertyAssignment */
+                        ? factory.createPropertyAssignment(e.name, ts.visitNode(e.initializer, visitor, ts.isExpression))
                         : ts.visitNode(e, visitor, ts.isObjectLiteralElementLike));
                 }
             }
             if (chunkObject) {
-                objects.push(ts.createObjectLiteral(chunkObject));
+                objects.push(factory.createObjectLiteralExpression(chunkObject));
             }
             return objects;
         }
@@ -81844,27 +86721,31 @@ var ts;
                 // If we translate the above to `__assign({}, k, l)`, the `l` will evaluate before `k` is spread and we
                 // end up with `{ a: 1, b: 2, c: 3 }`
                 var objects = chunkObjectLiteralElements(node.properties);
-                if (objects.length && objects[0].kind !== 193 /* ObjectLiteralExpression */) {
-                    objects.unshift(ts.createObjectLiteral());
+                if (objects.length && objects[0].kind !== 200 /* ObjectLiteralExpression */) {
+                    objects.unshift(factory.createObjectLiteralExpression());
                 }
                 var expression = objects[0];
                 if (objects.length > 1) {
                     for (var i = 1; i < objects.length; i++) {
-                        expression = createAssignHelper(context, [expression, objects[i]]);
+                        expression = emitHelpers().createAssignHelper([expression, objects[i]]);
                     }
                     return expression;
                 }
                 else {
-                    return createAssignHelper(context, objects);
+                    return emitHelpers().createAssignHelper(objects);
                 }
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitExpressionStatement(node) {
-            return ts.visitEachChild(node, visitorNoDestructuringValue, context);
+            return ts.visitEachChild(node, visitorWithUnusedExpressionResult, context);
         }
-        function visitParenthesizedExpression(node, noDestructuringValue) {
-            return ts.visitEachChild(node, noDestructuringValue ? visitorNoDestructuringValue : visitor, context);
+        /**
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
+         */
+        function visitParenthesizedExpression(node, expressionResultIsUnused) {
+            return ts.visitEachChild(node, expressionResultIsUnused ? visitorWithUnusedExpressionResult : visitor, context);
         }
         function visitSourceFile(node) {
             var ancestorFacts = enterSubtree(2 /* SourceFileExcludes */, ts.isEffectiveStrictModeSourceFile(node, compilerOptions) ?
@@ -81873,9 +86754,9 @@ var ts;
             exportedVariableStatement = false;
             var visited = ts.visitEachChild(node, visitor, context);
             var statement = ts.concatenate(visited.statements, taggedTemplateStringDeclarations && [
-                ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList(taggedTemplateStringDeclarations))
+                factory.createVariableStatement(/*modifiers*/ undefined, factory.createVariableDeclarationList(taggedTemplateStringDeclarations))
             ]);
-            var result = ts.updateSourceFileNode(visited, ts.setTextRange(ts.createNodeArray(statement), node.statements));
+            var result = factory.updateSourceFile(visited, ts.setTextRange(factory.createNodeArray(statement), node.statements));
             exitSubtree(ancestorFacts);
             return result;
         }
@@ -81886,35 +86767,57 @@ var ts;
          * Visits a BinaryExpression that contains a destructuring assignment.
          *
          * @param node A BinaryExpression node.
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
          */
-        function visitBinaryExpression(node, noDestructuringValue) {
+        function visitBinaryExpression(node, expressionResultIsUnused) {
             if (ts.isDestructuringAssignment(node) && node.left.transformFlags & 16384 /* ContainsObjectRestOrSpread */) {
-                return ts.flattenDestructuringAssignment(node, visitor, context, 1 /* ObjectRest */, !noDestructuringValue);
+                return ts.flattenDestructuringAssignment(node, visitor, context, 1 /* ObjectRest */, !expressionResultIsUnused);
             }
-            else if (node.operatorToken.kind === 27 /* CommaToken */) {
-                return ts.updateBinary(node, ts.visitNode(node.left, visitorNoDestructuringValue, ts.isExpression), ts.visitNode(node.right, noDestructuringValue ? visitorNoDestructuringValue : visitor, ts.isExpression));
+            if (node.operatorToken.kind === 27 /* CommaToken */) {
+                return factory.updateBinaryExpression(node, ts.visitNode(node.left, visitorWithUnusedExpressionResult, ts.isExpression), node.operatorToken, ts.visitNode(node.right, expressionResultIsUnused ? visitorWithUnusedExpressionResult : visitor, ts.isExpression));
             }
             return ts.visitEachChild(node, visitor, context);
         }
+        /**
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
+         */
+        function visitCommaListExpression(node, expressionResultIsUnused) {
+            if (expressionResultIsUnused) {
+                return ts.visitEachChild(node, visitorWithUnusedExpressionResult, context);
+            }
+            var result;
+            for (var i = 0; i < node.elements.length; i++) {
+                var element = node.elements[i];
+                var visited = ts.visitNode(element, i < node.elements.length - 1 ? visitorWithUnusedExpressionResult : visitor, ts.isExpression);
+                if (result || visited !== element) {
+                    result || (result = node.elements.slice(0, i));
+                    result.push(visited);
+                }
+            }
+            var elements = result ? ts.setTextRange(factory.createNodeArray(result), node.elements) : node.elements;
+            return factory.updateCommaListExpression(node, elements);
+        }
         function visitCatchClause(node) {
             if (node.variableDeclaration &&
                 ts.isBindingPattern(node.variableDeclaration.name) &&
                 node.variableDeclaration.name.transformFlags & 16384 /* ContainsObjectRestOrSpread */) {
-                var name = ts.getGeneratedNameForNode(node.variableDeclaration.name);
-                var updatedDecl = ts.updateVariableDeclaration(node.variableDeclaration, node.variableDeclaration.name, /*type*/ undefined, name);
+                var name = factory.getGeneratedNameForNode(node.variableDeclaration.name);
+                var updatedDecl = factory.updateVariableDeclaration(node.variableDeclaration, node.variableDeclaration.name, /*exclamationToken*/ undefined, /*type*/ undefined, name);
                 var visitedBindings = ts.flattenDestructuringBinding(updatedDecl, visitor, context, 1 /* ObjectRest */);
                 var block = ts.visitNode(node.block, visitor, ts.isBlock);
                 if (ts.some(visitedBindings)) {
-                    block = ts.updateBlock(block, __spreadArrays([
-                        ts.createVariableStatement(/*modifiers*/ undefined, visitedBindings)
+                    block = factory.updateBlock(block, __spreadArrays([
+                        factory.createVariableStatement(/*modifiers*/ undefined, visitedBindings)
                     ], block.statements));
                 }
-                return ts.updateCatchClause(node, ts.updateVariableDeclaration(node.variableDeclaration, name, /*type*/ undefined, /*initializer*/ undefined), block);
+                return factory.updateCatchClause(node, factory.updateVariableDeclaration(node.variableDeclaration, name, /*exclamationToken*/ undefined, /*type*/ undefined, /*initializer*/ undefined), block);
             }
             return ts.visitEachChild(node, visitor, context);
         }
         function visitVariableStatement(node) {
-            if (ts.hasModifier(node, 1 /* Export */)) {
+            if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
                 var savedExportedVariableStatement = exportedVariableStatement;
                 exportedVariableStatement = true;
                 var visited = ts.visitEachChild(node, visitor, context);
@@ -81947,10 +86850,10 @@ var ts;
             return ts.visitEachChild(node, visitor, context);
         }
         function visitForStatement(node) {
-            return ts.updateFor(node, ts.visitNode(node.initializer, visitorNoDestructuringValue, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitor, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement));
+            return factory.updateForStatement(node, ts.visitNode(node.initializer, visitorWithUnusedExpressionResult, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitorWithUnusedExpressionResult, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement));
         }
         function visitVoidExpression(node) {
-            return ts.visitEachChild(node, visitorNoDestructuringValue, context);
+            return ts.visitEachChild(node, visitorWithUnusedExpressionResult, context);
         }
         /**
          * Visits a ForOfStatement and converts it into a ES2015-compatible ForOfStatement.
@@ -81964,7 +86867,7 @@ var ts;
             }
             var result = node.awaitModifier ?
                 transformForAwaitOfStatement(node, outermostLabeledStatement, ancestorFacts) :
-                ts.restoreEnclosingLabel(ts.visitEachChild(node, visitor, context), outermostLabeledStatement);
+                factory.restoreEnclosingLabel(ts.visitEachChild(node, visitor, context), outermostLabeledStatement);
             exitSubtree(ancestorFacts);
             return result;
         }
@@ -81973,8 +86876,8 @@ var ts;
             if (ts.isVariableDeclarationList(initializerWithoutParens) || ts.isAssignmentPattern(initializerWithoutParens)) {
                 var bodyLocation = void 0;
                 var statementsLocation = void 0;
-                var temp = ts.createTempVariable(/*recordTempVariable*/ undefined);
-                var statements = [ts.createForOfBindingStatement(initializerWithoutParens, temp)];
+                var temp = factory.createTempVariable(/*recordTempVariable*/ undefined);
+                var statements = [ts.createForOfBindingStatement(factory, initializerWithoutParens, temp)];
                 if (ts.isBlock(node.statement)) {
                     ts.addRange(statements, node.statement.statements);
                     bodyLocation = node.statement;
@@ -81985,15 +86888,15 @@ var ts;
                     bodyLocation = node.statement;
                     statementsLocation = node.statement;
                 }
-                return ts.updateForOf(node, node.awaitModifier, ts.setTextRange(ts.createVariableDeclarationList([
-                    ts.setTextRange(ts.createVariableDeclaration(temp), node.initializer)
-                ], 1 /* Let */), node.initializer), node.expression, ts.setTextRange(ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), statementsLocation), 
+                return factory.updateForOfStatement(node, node.awaitModifier, ts.setTextRange(factory.createVariableDeclarationList([
+                    ts.setTextRange(factory.createVariableDeclaration(temp), node.initializer)
+                ], 1 /* Let */), node.initializer), node.expression, ts.setTextRange(factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), statementsLocation), 
                 /*multiLine*/ true), bodyLocation));
             }
             return node;
         }
         function convertForOfStatementHead(node, boundValue) {
-            var binding = ts.createForOfBindingStatement(node.initializer, boundValue);
+            var binding = ts.createForOfBindingStatement(factory, node.initializer, boundValue);
             var bodyLocation;
             var statementsLocation;
             var statements = [ts.visitNode(binding, visitor, ts.isStatement)];
@@ -82006,55 +86909,55 @@ var ts;
             else {
                 statements.push(statement);
             }
-            return ts.setEmitFlags(ts.setTextRange(ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), statementsLocation), 
+            return ts.setEmitFlags(ts.setTextRange(factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), statementsLocation), 
             /*multiLine*/ true), bodyLocation), 48 /* NoSourceMap */ | 384 /* NoTokenSourceMaps */);
         }
         function createDownlevelAwait(expression) {
             return enclosingFunctionFlags & 1 /* Generator */
-                ? ts.createYield(/*asteriskToken*/ undefined, createAwaitHelper(context, expression))
-                : ts.createAwait(expression);
+                ? factory.createYieldExpression(/*asteriskToken*/ undefined, emitHelpers().createAwaitHelper(expression))
+                : factory.createAwaitExpression(expression);
         }
         function transformForAwaitOfStatement(node, outermostLabeledStatement, ancestorFacts) {
             var expression = ts.visitNode(node.expression, visitor, ts.isExpression);
-            var iterator = ts.isIdentifier(expression) ? ts.getGeneratedNameForNode(expression) : ts.createTempVariable(/*recordTempVariable*/ undefined);
-            var result = ts.isIdentifier(expression) ? ts.getGeneratedNameForNode(iterator) : ts.createTempVariable(/*recordTempVariable*/ undefined);
-            var errorRecord = ts.createUniqueName("e");
-            var catchVariable = ts.getGeneratedNameForNode(errorRecord);
-            var returnMethod = ts.createTempVariable(/*recordTempVariable*/ undefined);
-            var callValues = createAsyncValuesHelper(context, expression, /*location*/ node.expression);
-            var callNext = ts.createCall(ts.createPropertyAccess(iterator, "next"), /*typeArguments*/ undefined, []);
-            var getDone = ts.createPropertyAccess(result, "done");
-            var getValue = ts.createPropertyAccess(result, "value");
-            var callReturn = ts.createFunctionCall(returnMethod, iterator, []);
+            var iterator = ts.isIdentifier(expression) ? factory.getGeneratedNameForNode(expression) : factory.createTempVariable(/*recordTempVariable*/ undefined);
+            var result = ts.isIdentifier(expression) ? factory.getGeneratedNameForNode(iterator) : factory.createTempVariable(/*recordTempVariable*/ undefined);
+            var errorRecord = factory.createUniqueName("e");
+            var catchVariable = factory.getGeneratedNameForNode(errorRecord);
+            var returnMethod = factory.createTempVariable(/*recordTempVariable*/ undefined);
+            var callValues = ts.setTextRange(emitHelpers().createAsyncValuesHelper(expression), node.expression);
+            var callNext = factory.createCallExpression(factory.createPropertyAccessExpression(iterator, "next"), /*typeArguments*/ undefined, []);
+            var getDone = factory.createPropertyAccessExpression(result, "done");
+            var getValue = factory.createPropertyAccessExpression(result, "value");
+            var callReturn = factory.createFunctionCallCall(returnMethod, iterator, []);
             hoistVariableDeclaration(errorRecord);
             hoistVariableDeclaration(returnMethod);
             // if we are enclosed in an outer loop ensure we reset 'errorRecord' per each iteration
             var initializer = ancestorFacts & 2 /* IterationContainer */ ?
-                ts.inlineExpressions([ts.createAssignment(errorRecord, ts.createVoidZero()), callValues]) :
+                factory.inlineExpressions([factory.createAssignment(errorRecord, factory.createVoidZero()), callValues]) :
                 callValues;
-            var forStatement = ts.setEmitFlags(ts.setTextRange(ts.createFor(
-            /*initializer*/ ts.setEmitFlags(ts.setTextRange(ts.createVariableDeclarationList([
-                ts.setTextRange(ts.createVariableDeclaration(iterator, /*type*/ undefined, initializer), node.expression),
-                ts.createVariableDeclaration(result)
+            var forStatement = ts.setEmitFlags(ts.setTextRange(factory.createForStatement(
+            /*initializer*/ ts.setEmitFlags(ts.setTextRange(factory.createVariableDeclarationList([
+                ts.setTextRange(factory.createVariableDeclaration(iterator, /*exclamationToken*/ undefined, /*type*/ undefined, initializer), node.expression),
+                factory.createVariableDeclaration(result)
             ]), node.expression), 2097152 /* NoHoisting */), 
-            /*condition*/ ts.createComma(ts.createAssignment(result, createDownlevelAwait(callNext)), ts.createLogicalNot(getDone)), 
+            /*condition*/ factory.createComma(factory.createAssignment(result, createDownlevelAwait(callNext)), factory.createLogicalNot(getDone)), 
             /*incrementor*/ undefined, 
             /*statement*/ convertForOfStatementHead(node, getValue)), 
             /*location*/ node), 256 /* NoTokenTrailingSourceMaps */);
-            return ts.createTry(ts.createBlock([
-                ts.restoreEnclosingLabel(forStatement, outermostLabeledStatement)
-            ]), ts.createCatchClause(ts.createVariableDeclaration(catchVariable), ts.setEmitFlags(ts.createBlock([
-                ts.createExpressionStatement(ts.createAssignment(errorRecord, ts.createObjectLiteral([
-                    ts.createPropertyAssignment("error", catchVariable)
+            return factory.createTryStatement(factory.createBlock([
+                factory.restoreEnclosingLabel(forStatement, outermostLabeledStatement)
+            ]), factory.createCatchClause(factory.createVariableDeclaration(catchVariable), ts.setEmitFlags(factory.createBlock([
+                factory.createExpressionStatement(factory.createAssignment(errorRecord, factory.createObjectLiteralExpression([
+                    factory.createPropertyAssignment("error", catchVariable)
                 ])))
-            ]), 1 /* SingleLine */)), ts.createBlock([
-                ts.createTry(
-                /*tryBlock*/ ts.createBlock([
-                    ts.setEmitFlags(ts.createIf(ts.createLogicalAnd(ts.createLogicalAnd(result, ts.createLogicalNot(getDone)), ts.createAssignment(returnMethod, ts.createPropertyAccess(iterator, "return"))), ts.createExpressionStatement(createDownlevelAwait(callReturn))), 1 /* SingleLine */)
+            ]), 1 /* SingleLine */)), factory.createBlock([
+                factory.createTryStatement(
+                /*tryBlock*/ factory.createBlock([
+                    ts.setEmitFlags(factory.createIfStatement(factory.createLogicalAnd(factory.createLogicalAnd(result, factory.createLogicalNot(getDone)), factory.createAssignment(returnMethod, factory.createPropertyAccessExpression(iterator, "return"))), factory.createExpressionStatement(createDownlevelAwait(callReturn))), 1 /* SingleLine */)
                 ]), 
                 /*catchClause*/ undefined, 
-                /*finallyBlock*/ ts.setEmitFlags(ts.createBlock([
-                    ts.setEmitFlags(ts.createIf(errorRecord, ts.createThrow(ts.createPropertyAccess(errorRecord, "error"))), 1 /* SingleLine */)
+                /*finallyBlock*/ ts.setEmitFlags(factory.createBlock([
+                    ts.setEmitFlags(factory.createIfStatement(errorRecord, factory.createThrowStatement(factory.createPropertyAccessExpression(errorRecord, "error"))), 1 /* SingleLine */)
                 ]), 1 /* SingleLine */))
             ]));
         }
@@ -82062,9 +86965,9 @@ var ts;
             if (node.transformFlags & 16384 /* ContainsObjectRestOrSpread */) {
                 // Binding patterns are converted into a generated name and are
                 // evaluated inside the function body.
-                return ts.updateParameter(node, 
+                return factory.updateParameterDeclaration(node, 
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, node.dotDotDotToken, ts.getGeneratedNameForNode(node), 
+                /*modifiers*/ undefined, node.dotDotDotToken, factory.getGeneratedNameForNode(node), 
                 /*questionToken*/ undefined, 
                 /*type*/ undefined, ts.visitNode(node.initializer, visitor, ts.isExpression));
             }
@@ -82073,7 +86976,7 @@ var ts;
         function visitConstructorDeclaration(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = 0 /* Normal */;
-            var updated = ts.updateConstructor(node, 
+            var updated = factory.updateConstructorDeclaration(node, 
             /*decorators*/ undefined, node.modifiers, ts.visitParameterList(node.parameters, visitor, context), transformFunctionBody(node));
             enclosingFunctionFlags = savedEnclosingFunctionFlags;
             return updated;
@@ -82081,7 +86984,7 @@ var ts;
         function visitGetAccessorDeclaration(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = 0 /* Normal */;
-            var updated = ts.updateGetAccessor(node, 
+            var updated = factory.updateGetAccessorDeclaration(node, 
             /*decorators*/ undefined, node.modifiers, ts.visitNode(node.name, visitor, ts.isPropertyName), ts.visitParameterList(node.parameters, visitor, context), 
             /*type*/ undefined, transformFunctionBody(node));
             enclosingFunctionFlags = savedEnclosingFunctionFlags;
@@ -82090,7 +86993,7 @@ var ts;
         function visitSetAccessorDeclaration(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = 0 /* Normal */;
-            var updated = ts.updateSetAccessor(node, 
+            var updated = factory.updateSetAccessorDeclaration(node, 
             /*decorators*/ undefined, node.modifiers, ts.visitNode(node.name, visitor, ts.isPropertyName), ts.visitParameterList(node.parameters, visitor, context), transformFunctionBody(node));
             enclosingFunctionFlags = savedEnclosingFunctionFlags;
             return updated;
@@ -82098,7 +87001,7 @@ var ts;
         function visitMethodDeclaration(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = ts.getFunctionFlags(node);
-            var updated = ts.updateMethod(node, 
+            var updated = factory.updateMethodDeclaration(node, 
             /*decorators*/ undefined, enclosingFunctionFlags & 1 /* Generator */
                 ? ts.visitNodes(node.modifiers, visitorNoAsyncModifier, ts.isModifier)
                 : node.modifiers, enclosingFunctionFlags & 2 /* Async */
@@ -82114,7 +87017,7 @@ var ts;
         function visitFunctionDeclaration(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = ts.getFunctionFlags(node);
-            var updated = ts.updateFunctionDeclaration(node, 
+            var updated = factory.updateFunctionDeclaration(node, 
             /*decorators*/ undefined, enclosingFunctionFlags & 1 /* Generator */
                 ? ts.visitNodes(node.modifiers, visitorNoAsyncModifier, ts.isModifier)
                 : node.modifiers, enclosingFunctionFlags & 2 /* Async */
@@ -82130,7 +87033,7 @@ var ts;
         function visitArrowFunction(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = ts.getFunctionFlags(node);
-            var updated = ts.updateArrowFunction(node, node.modifiers, 
+            var updated = factory.updateArrowFunction(node, node.modifiers, 
             /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
             /*type*/ undefined, node.equalsGreaterThanToken, transformFunctionBody(node));
             enclosingFunctionFlags = savedEnclosingFunctionFlags;
@@ -82139,7 +87042,7 @@ var ts;
         function visitFunctionExpression(node) {
             var savedEnclosingFunctionFlags = enclosingFunctionFlags;
             enclosingFunctionFlags = ts.getFunctionFlags(node);
-            var updated = ts.updateFunctionExpression(node, enclosingFunctionFlags & 1 /* Generator */
+            var updated = factory.updateFunctionExpression(node, enclosingFunctionFlags & 1 /* Generator */
                 ? ts.visitNodes(node.modifiers, visitorNoAsyncModifier, ts.isModifier)
                 : node.modifiers, enclosingFunctionFlags & 2 /* Async */
                 ? undefined
@@ -82154,29 +87057,29 @@ var ts;
         function transformAsyncGeneratorFunctionBody(node) {
             resumeLexicalEnvironment();
             var statements = [];
-            var statementOffset = ts.addPrologue(statements, node.body.statements, /*ensureUseStrict*/ false, visitor);
+            var statementOffset = factory.copyPrologue(node.body.statements, statements, /*ensureUseStrict*/ false, visitor);
             appendObjectRestAssignmentsIfNeeded(statements, node);
             var savedCapturedSuperProperties = capturedSuperProperties;
             var savedHasSuperElementAccess = hasSuperElementAccess;
-            capturedSuperProperties = ts.createUnderscoreEscapedMap();
+            capturedSuperProperties = new ts.Set();
             hasSuperElementAccess = false;
-            var returnStatement = ts.createReturn(createAsyncGeneratorHelper(context, ts.createFunctionExpression(
-            /*modifiers*/ undefined, ts.createToken(41 /* AsteriskToken */), node.name && ts.getGeneratedNameForNode(node.name), 
+            var returnStatement = factory.createReturnStatement(emitHelpers().createAsyncGeneratorHelper(factory.createFunctionExpression(
+            /*modifiers*/ undefined, factory.createToken(41 /* AsteriskToken */), node.name && factory.getGeneratedNameForNode(node.name), 
             /*typeParameters*/ undefined, 
             /*parameters*/ [], 
-            /*type*/ undefined, ts.updateBlock(node.body, ts.visitLexicalEnvironment(node.body.statements, visitor, context, statementOffset))), !!(hierarchyFacts & 1 /* HasLexicalThis */)));
+            /*type*/ undefined, factory.updateBlock(node.body, ts.visitLexicalEnvironment(node.body.statements, visitor, context, statementOffset))), !!(hierarchyFacts & 1 /* HasLexicalThis */)));
             // Minor optimization, emit `_super` helper to capture `super` access in an arrow.
             // This step isn't needed if we eventually transform this to ES5.
             var emitSuperHelpers = languageVersion >= 2 /* ES2015 */ && resolver.getNodeCheckFlags(node) & (4096 /* AsyncMethodWithSuperBinding */ | 2048 /* AsyncMethodWithSuper */);
             if (emitSuperHelpers) {
                 enableSubstitutionForAsyncMethodsWithSuper();
-                var variableStatement = ts.createSuperAccessVariableStatement(resolver, node, capturedSuperProperties);
+                var variableStatement = ts.createSuperAccessVariableStatement(factory, resolver, node, capturedSuperProperties);
                 substitutedSuperAccessors[ts.getNodeId(variableStatement)] = true;
                 ts.insertStatementsAfterStandardPrologue(statements, [variableStatement]);
             }
             statements.push(returnStatement);
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
-            var block = ts.updateBlock(node.body, statements);
+            var block = factory.updateBlock(node.body, statements);
             if (emitSuperHelpers && hasSuperElementAccess) {
                 if (resolver.getNodeCheckFlags(node) & 4096 /* AsyncMethodWithSuperBinding */) {
                     ts.addEmitHelper(block, ts.advancedAsyncSuperHelper);
@@ -82190,20 +87093,21 @@ var ts;
             return block;
         }
         function transformFunctionBody(node) {
+            var _a;
             resumeLexicalEnvironment();
             var statementOffset = 0;
             var statements = [];
-            var body = ts.visitNode(node.body, visitor, ts.isConciseBody);
+            var body = (_a = ts.visitNode(node.body, visitor, ts.isConciseBody)) !== null && _a !== void 0 ? _a : factory.createBlock([]);
             if (ts.isBlock(body)) {
-                statementOffset = ts.addPrologue(statements, body.statements, /*ensureUseStrict*/ false, visitor);
+                statementOffset = factory.copyPrologue(body.statements, statements, /*ensureUseStrict*/ false, visitor);
             }
             ts.addRange(statements, appendObjectRestAssignmentsIfNeeded(/*statements*/ undefined, node));
             var leadingStatements = endLexicalEnvironment();
             if (statementOffset > 0 || ts.some(statements) || ts.some(leadingStatements)) {
-                var block = ts.convertToFunctionBody(body, /*multiLine*/ true);
+                var block = factory.converters.convertToFunctionBlock(body, /*multiLine*/ true);
                 ts.insertStatementsAfterStandardPrologue(statements, leadingStatements);
                 ts.addRange(statements, block.statements.slice(statementOffset));
-                return ts.updateBlock(block, ts.setTextRange(ts.createNodeArray(statements), block.statements));
+                return factory.updateBlock(block, ts.setTextRange(factory.createNodeArray(statements), block.statements));
             }
             return body;
         }
@@ -82211,13 +87115,13 @@ var ts;
             for (var _i = 0, _a = node.parameters; _i < _a.length; _i++) {
                 var parameter = _a[_i];
                 if (parameter.transformFlags & 16384 /* ContainsObjectRestOrSpread */) {
-                    var temp = ts.getGeneratedNameForNode(parameter);
+                    var temp = factory.getGeneratedNameForNode(parameter);
                     var declarations = ts.flattenDestructuringBinding(parameter, visitor, context, 1 /* ObjectRest */, temp, 
                     /*doNotRecordTempVariablesInLine*/ false, 
                     /*skipInitializer*/ true);
                     if (ts.some(declarations)) {
-                        var statement = ts.createVariableStatement(
-                        /*modifiers*/ undefined, ts.createVariableDeclarationList(declarations));
+                        var statement = factory.createVariableStatement(
+                        /*modifiers*/ undefined, factory.createVariableDeclarationList(declarations));
                         ts.setEmitFlags(statement, 1048576 /* CustomPrologue */);
                         statements = ts.append(statements, statement);
                     }
@@ -82230,17 +87134,17 @@ var ts;
                 enabledSubstitutions |= 1 /* AsyncMethodsWithSuper */;
                 // We need to enable substitutions for call, property access, and element access
                 // if we need to rewrite super calls.
-                context.enableSubstitution(196 /* CallExpression */);
-                context.enableSubstitution(194 /* PropertyAccessExpression */);
-                context.enableSubstitution(195 /* ElementAccessExpression */);
+                context.enableSubstitution(203 /* CallExpression */);
+                context.enableSubstitution(201 /* PropertyAccessExpression */);
+                context.enableSubstitution(202 /* ElementAccessExpression */);
                 // We need to be notified when entering and exiting declarations that bind super.
-                context.enableEmitNotification(245 /* ClassDeclaration */);
-                context.enableEmitNotification(161 /* MethodDeclaration */);
-                context.enableEmitNotification(163 /* GetAccessor */);
-                context.enableEmitNotification(164 /* SetAccessor */);
-                context.enableEmitNotification(162 /* Constructor */);
+                context.enableEmitNotification(252 /* ClassDeclaration */);
+                context.enableEmitNotification(165 /* MethodDeclaration */);
+                context.enableEmitNotification(167 /* GetAccessor */);
+                context.enableEmitNotification(168 /* SetAccessor */);
+                context.enableEmitNotification(166 /* Constructor */);
                 // We need to be notified when entering the generated accessor arrow functions.
-                context.enableEmitNotification(225 /* VariableStatement */);
+                context.enableEmitNotification(232 /* VariableStatement */);
             }
         }
         /**
@@ -82288,23 +87192,23 @@ var ts;
         }
         function substituteExpression(node) {
             switch (node.kind) {
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     return substitutePropertyAccessExpression(node);
-                case 195 /* ElementAccessExpression */:
+                case 202 /* ElementAccessExpression */:
                     return substituteElementAccessExpression(node);
-                case 196 /* CallExpression */:
+                case 203 /* CallExpression */:
                     return substituteCallExpression(node);
             }
             return node;
         }
         function substitutePropertyAccessExpression(node) {
-            if (node.expression.kind === 102 /* SuperKeyword */) {
-                return ts.setTextRange(ts.createPropertyAccess(ts.createFileLevelUniqueName("_super"), node.name), node);
+            if (node.expression.kind === 105 /* SuperKeyword */) {
+                return ts.setTextRange(factory.createPropertyAccessExpression(factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */), node.name), node);
             }
             return node;
         }
         function substituteElementAccessExpression(node) {
-            if (node.expression.kind === 102 /* SuperKeyword */) {
+            if (node.expression.kind === 105 /* SuperKeyword */) {
                 return createSuperElementAccessInAsyncMethod(node.argumentExpression, node);
             }
             return node;
@@ -82315,106 +87219,40 @@ var ts;
                 var argumentExpression = ts.isPropertyAccessExpression(expression)
                     ? substitutePropertyAccessExpression(expression)
                     : substituteElementAccessExpression(expression);
-                return ts.createCall(ts.createPropertyAccess(argumentExpression, "call"), 
+                return factory.createCallExpression(factory.createPropertyAccessExpression(argumentExpression, "call"), 
                 /*typeArguments*/ undefined, __spreadArrays([
-                    ts.createThis()
+                    factory.createThis()
                 ], node.arguments));
             }
             return node;
         }
         function isSuperContainer(node) {
             var kind = node.kind;
-            return kind === 245 /* ClassDeclaration */
-                || kind === 162 /* Constructor */
-                || kind === 161 /* MethodDeclaration */
-                || kind === 163 /* GetAccessor */
-                || kind === 164 /* SetAccessor */;
+            return kind === 252 /* ClassDeclaration */
+                || kind === 166 /* Constructor */
+                || kind === 165 /* MethodDeclaration */
+                || kind === 167 /* GetAccessor */
+                || kind === 168 /* SetAccessor */;
         }
         function createSuperElementAccessInAsyncMethod(argumentExpression, location) {
             if (enclosingSuperContainerFlags & 4096 /* AsyncMethodWithSuperBinding */) {
-                return ts.setTextRange(ts.createPropertyAccess(ts.createCall(ts.createIdentifier("_superIndex"), 
+                return ts.setTextRange(factory.createPropertyAccessExpression(factory.createCallExpression(factory.createIdentifier("_superIndex"), 
                 /*typeArguments*/ undefined, [argumentExpression]), "value"), location);
             }
             else {
-                return ts.setTextRange(ts.createCall(ts.createIdentifier("_superIndex"), 
+                return ts.setTextRange(factory.createCallExpression(factory.createIdentifier("_superIndex"), 
                 /*typeArguments*/ undefined, [argumentExpression]), location);
             }
         }
     }
     ts.transformES2018 = transformES2018;
-    ts.assignHelper = {
-        name: "typescript:assign",
-        importName: "__assign",
-        scoped: false,
-        priority: 1,
-        text: "\n            var __assign = (this && this.__assign) || function () {\n                __assign = Object.assign || function(t) {\n                    for (var s, i = 1, n = arguments.length; i < n; i++) {\n                        s = arguments[i];\n                        for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n                            t[p] = s[p];\n                    }\n                    return t;\n                };\n                return __assign.apply(this, arguments);\n            };"
-    };
-    function createAssignHelper(context, attributesSegments) {
-        if (context.getCompilerOptions().target >= 2 /* ES2015 */) {
-            return ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Object"), "assign"), /*typeArguments*/ undefined, attributesSegments);
-        }
-        context.requestEmitHelper(ts.assignHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__assign"), 
-        /*typeArguments*/ undefined, attributesSegments);
-    }
-    ts.createAssignHelper = createAssignHelper;
-    ts.awaitHelper = {
-        name: "typescript:await",
-        importName: "__await",
-        scoped: false,
-        text: "\n            var __await = (this && this.__await) || function (v) { return this instanceof __await ? (this.v = v, this) : new __await(v); }"
-    };
-    function createAwaitHelper(context, expression) {
-        context.requestEmitHelper(ts.awaitHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__await"), /*typeArguments*/ undefined, [expression]);
-    }
-    ts.asyncGeneratorHelper = {
-        name: "typescript:asyncGenerator",
-        importName: "__asyncGenerator",
-        scoped: false,
-        dependencies: [ts.awaitHelper],
-        text: "\n            var __asyncGenerator = (this && this.__asyncGenerator) || function (thisArg, _arguments, generator) {\n                if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n                var g = generator.apply(thisArg, _arguments || []), i, q = [];\n                return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\n                function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\n                function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\n                function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\n                function fulfill(value) { resume(\"next\", value); }\n                function reject(value) { resume(\"throw\", value); }\n                function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\n            };"
-    };
-    function createAsyncGeneratorHelper(context, generatorFunc, hasLexicalThis) {
-        context.requestEmitHelper(ts.asyncGeneratorHelper);
-        // Mark this node as originally an async function
-        (generatorFunc.emitNode || (generatorFunc.emitNode = {})).flags |= 262144 /* AsyncFunctionBody */ | 524288 /* ReuseTempVariableScope */;
-        return ts.createCall(ts.getUnscopedHelperName("__asyncGenerator"), 
-        /*typeArguments*/ undefined, [
-            hasLexicalThis ? ts.createThis() : ts.createVoidZero(),
-            ts.createIdentifier("arguments"),
-            generatorFunc
-        ]);
-    }
-    ts.asyncDelegator = {
-        name: "typescript:asyncDelegator",
-        importName: "__asyncDelegator",
-        scoped: false,
-        dependencies: [ts.awaitHelper],
-        text: "\n            var __asyncDelegator = (this && this.__asyncDelegator) || function (o) {\n                var i, p;\n                return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\n                function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: n === \"return\" } : f ? f(v) : v; } : f; }\n            };"
-    };
-    function createAsyncDelegatorHelper(context, expression, location) {
-        context.requestEmitHelper(ts.asyncDelegator);
-        return ts.setTextRange(ts.createCall(ts.getUnscopedHelperName("__asyncDelegator"), 
-        /*typeArguments*/ undefined, [expression]), location);
-    }
-    ts.asyncValues = {
-        name: "typescript:asyncValues",
-        importName: "__asyncValues",
-        scoped: false,
-        text: "\n            var __asyncValues = (this && this.__asyncValues) || function (o) {\n                if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n                var m = o[Symbol.asyncIterator], i;\n                return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\n                function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\n                function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\n            };"
-    };
-    function createAsyncValuesHelper(context, expression, location) {
-        context.requestEmitHelper(ts.asyncValues);
-        return ts.setTextRange(ts.createCall(ts.getUnscopedHelperName("__asyncValues"), 
-        /*typeArguments*/ undefined, [expression]), location);
-    }
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
 (function (ts) {
     function transformES2019(context) {
-        return ts.chainBundle(transformSourceFile);
+        var factory = context.factory;
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
@@ -82426,7 +87264,7 @@ var ts;
                 return node;
             }
             switch (node.kind) {
-                case 280 /* CatchClause */:
+                case 287 /* CatchClause */:
                     return visitCatchClause(node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -82434,7 +87272,7 @@ var ts;
         }
         function visitCatchClause(node) {
             if (!node.variableDeclaration) {
-                return ts.updateCatchClause(node, ts.createVariableDeclaration(ts.createTempVariable(/*recordTempVariable*/ undefined)), ts.visitNode(node.block, visitor, ts.isBlock));
+                return factory.updateCatchClause(node, factory.createVariableDeclaration(factory.createTempVariable(/*recordTempVariable*/ undefined)), ts.visitNode(node.block, visitor, ts.isBlock));
             }
             return ts.visitEachChild(node, visitor, context);
         }
@@ -82445,8 +87283,8 @@ var ts;
 var ts;
 (function (ts) {
     function transformES2020(context) {
-        var hoistVariableDeclaration = context.hoistVariableDeclaration;
-        return ts.chainBundle(transformSourceFile);
+        var factory = context.factory, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
@@ -82458,21 +87296,21 @@ var ts;
                 return node;
             }
             switch (node.kind) {
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */:
-                case 196 /* CallExpression */:
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */:
+                case 203 /* CallExpression */:
                     if (node.flags & 32 /* OptionalChain */) {
                         var updated = visitOptionalExpression(node, /*captureThisArg*/ false, /*isDelete*/ false);
                         ts.Debug.assertNotNode(updated, ts.isSyntheticReference);
                         return updated;
                     }
                     return ts.visitEachChild(node, visitor, context);
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     if (node.operatorToken.kind === 60 /* QuestionQuestionToken */) {
                         return transformNullishCoalescingExpression(node);
                     }
                     return ts.visitEachChild(node, visitor, context);
-                case 203 /* DeleteExpression */:
+                case 210 /* DeleteExpression */:
                     return visitDeleteExpression(node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -82493,9 +87331,9 @@ var ts;
             if (ts.isSyntheticReference(expression)) {
                 // `(a.b)` -> { expression `((_a = a).b)`, thisArg: `_a` }
                 // `(a[b])` -> { expression `((_a = a)[b])`, thisArg: `_a` }
-                return ts.createSyntheticReferenceExpression(ts.updateParen(node, expression.expression), expression.thisArg);
+                return factory.createSyntheticReferenceExpression(factory.updateParenthesizedExpression(node, expression.expression), expression.thisArg);
             }
-            return ts.updateParen(node, expression);
+            return factory.updateParenthesizedExpression(node, expression);
         }
         function visitNonOptionalPropertyOrElementAccessExpression(node, captureThisArg, isDelete) {
             if (ts.isOptionalChain(node)) {
@@ -82506,19 +87344,19 @@ var ts;
             ts.Debug.assertNotNode(expression, ts.isSyntheticReference);
             var thisArg;
             if (captureThisArg) {
-                if (shouldCaptureInTempVariable(expression)) {
-                    thisArg = ts.createTempVariable(hoistVariableDeclaration);
-                    expression = ts.createAssignment(thisArg, expression);
+                if (!ts.isSimpleCopiableExpression(expression)) {
+                    thisArg = factory.createTempVariable(hoistVariableDeclaration);
+                    expression = factory.createAssignment(thisArg, expression);
                     // if (inParameterInitializer) tempVariableInParameter = true;
                 }
                 else {
                     thisArg = expression;
                 }
             }
-            expression = node.kind === 194 /* PropertyAccessExpression */
-                ? ts.updatePropertyAccess(node, expression, ts.visitNode(node.name, visitor, ts.isIdentifier))
-                : ts.updateElementAccess(node, expression, ts.visitNode(node.argumentExpression, visitor, ts.isExpression));
-            return thisArg ? ts.createSyntheticReferenceExpression(expression, thisArg) : expression;
+            expression = node.kind === 201 /* PropertyAccessExpression */
+                ? factory.updatePropertyAccessExpression(node, expression, ts.visitNode(node.name, visitor, ts.isIdentifier))
+                : factory.updateElementAccessExpression(node, expression, ts.visitNode(node.argumentExpression, visitor, ts.isExpression));
+            return thisArg ? factory.createSyntheticReferenceExpression(expression, thisArg) : expression;
         }
         function visitNonOptionalCallExpression(node, captureThisArg) {
             if (ts.isOptionalChain(node)) {
@@ -82529,10 +87367,10 @@ var ts;
         }
         function visitNonOptionalExpression(node, captureThisArg, isDelete) {
             switch (node.kind) {
-                case 200 /* ParenthesizedExpression */: return visitNonOptionalParenthesizedExpression(node, captureThisArg, isDelete);
-                case 194 /* PropertyAccessExpression */:
-                case 195 /* ElementAccessExpression */: return visitNonOptionalPropertyOrElementAccessExpression(node, captureThisArg, isDelete);
-                case 196 /* CallExpression */: return visitNonOptionalCallExpression(node, captureThisArg);
+                case 207 /* ParenthesizedExpression */: return visitNonOptionalParenthesizedExpression(node, captureThisArg, isDelete);
+                case 201 /* PropertyAccessExpression */:
+                case 202 /* ElementAccessExpression */: return visitNonOptionalPropertyOrElementAccessExpression(node, captureThisArg, isDelete);
+                case 203 /* CallExpression */: return visitNonOptionalCallExpression(node, captureThisArg);
                 default: return ts.visitNode(node, visitor, ts.isExpression);
             }
         }
@@ -82542,9 +87380,9 @@ var ts;
             var leftThisArg = ts.isSyntheticReference(left) ? left.thisArg : undefined;
             var leftExpression = ts.isSyntheticReference(left) ? left.expression : left;
             var capturedLeft = leftExpression;
-            if (shouldCaptureInTempVariable(leftExpression)) {
-                capturedLeft = ts.createTempVariable(hoistVariableDeclaration);
-                leftExpression = ts.createAssignment(capturedLeft, leftExpression);
+            if (!ts.isSimpleCopiableExpression(leftExpression)) {
+                capturedLeft = factory.createTempVariable(hoistVariableDeclaration);
+                leftExpression = factory.createAssignment(capturedLeft, leftExpression);
                 // if (inParameterInitializer) tempVariableInParameter = true;
             }
             var rightExpression = capturedLeft;
@@ -82552,28 +87390,28 @@ var ts;
             for (var i = 0; i < chain.length; i++) {
                 var segment = chain[i];
                 switch (segment.kind) {
-                    case 194 /* PropertyAccessExpression */:
-                    case 195 /* ElementAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
                         if (i === chain.length - 1 && captureThisArg) {
-                            if (shouldCaptureInTempVariable(rightExpression)) {
-                                thisArg = ts.createTempVariable(hoistVariableDeclaration);
-                                rightExpression = ts.createAssignment(thisArg, rightExpression);
+                            if (!ts.isSimpleCopiableExpression(rightExpression)) {
+                                thisArg = factory.createTempVariable(hoistVariableDeclaration);
+                                rightExpression = factory.createAssignment(thisArg, rightExpression);
                                 // if (inParameterInitializer) tempVariableInParameter = true;
                             }
                             else {
                                 thisArg = rightExpression;
                             }
                         }
-                        rightExpression = segment.kind === 194 /* PropertyAccessExpression */
-                            ? ts.createPropertyAccess(rightExpression, ts.visitNode(segment.name, visitor, ts.isIdentifier))
-                            : ts.createElementAccess(rightExpression, ts.visitNode(segment.argumentExpression, visitor, ts.isExpression));
+                        rightExpression = segment.kind === 201 /* PropertyAccessExpression */
+                            ? factory.createPropertyAccessExpression(rightExpression, ts.visitNode(segment.name, visitor, ts.isIdentifier))
+                            : factory.createElementAccessExpression(rightExpression, ts.visitNode(segment.argumentExpression, visitor, ts.isExpression));
                         break;
-                    case 196 /* CallExpression */:
+                    case 203 /* CallExpression */:
                         if (i === 0 && leftThisArg) {
-                            rightExpression = ts.createFunctionCall(rightExpression, leftThisArg.kind === 102 /* SuperKeyword */ ? ts.createThis() : leftThisArg, ts.visitNodes(segment.arguments, visitor, ts.isExpression));
+                            rightExpression = factory.createFunctionCallCall(rightExpression, leftThisArg.kind === 105 /* SuperKeyword */ ? factory.createThis() : leftThisArg, ts.visitNodes(segment.arguments, visitor, ts.isExpression));
                         }
                         else {
-                            rightExpression = ts.createCall(rightExpression, 
+                            rightExpression = factory.createCallExpression(rightExpression, 
                             /*typeArguments*/ undefined, ts.visitNodes(segment.arguments, visitor, ts.isExpression));
                         }
                         break;
@@ -82581,34 +87419,29 @@ var ts;
                 ts.setOriginalNode(rightExpression, segment);
             }
             var target = isDelete
-                ? ts.createConditional(createNotNullCondition(leftExpression, capturedLeft, /*invert*/ true), ts.createTrue(), ts.createDelete(rightExpression))
-                : ts.createConditional(createNotNullCondition(leftExpression, capturedLeft, /*invert*/ true), ts.createVoidZero(), rightExpression);
-            return thisArg ? ts.createSyntheticReferenceExpression(target, thisArg) : target;
+                ? factory.createConditionalExpression(createNotNullCondition(leftExpression, capturedLeft, /*invert*/ true), /*questionToken*/ undefined, factory.createTrue(), /*colonToken*/ undefined, factory.createDeleteExpression(rightExpression))
+                : factory.createConditionalExpression(createNotNullCondition(leftExpression, capturedLeft, /*invert*/ true), /*questionToken*/ undefined, factory.createVoidZero(), /*colonToken*/ undefined, rightExpression);
+            return thisArg ? factory.createSyntheticReferenceExpression(target, thisArg) : target;
         }
         function createNotNullCondition(left, right, invert) {
-            return ts.createBinary(ts.createBinary(left, ts.createToken(invert ? 36 /* EqualsEqualsEqualsToken */ : 37 /* ExclamationEqualsEqualsToken */), ts.createNull()), ts.createToken(invert ? 56 /* BarBarToken */ : 55 /* AmpersandAmpersandToken */), ts.createBinary(right, ts.createToken(invert ? 36 /* EqualsEqualsEqualsToken */ : 37 /* ExclamationEqualsEqualsToken */), ts.createVoidZero()));
+            return factory.createBinaryExpression(factory.createBinaryExpression(left, factory.createToken(invert ? 36 /* EqualsEqualsEqualsToken */ : 37 /* ExclamationEqualsEqualsToken */), factory.createNull()), factory.createToken(invert ? 56 /* BarBarToken */ : 55 /* AmpersandAmpersandToken */), factory.createBinaryExpression(right, factory.createToken(invert ? 36 /* EqualsEqualsEqualsToken */ : 37 /* ExclamationEqualsEqualsToken */), factory.createVoidZero()));
         }
         function transformNullishCoalescingExpression(node) {
             var left = ts.visitNode(node.left, visitor, ts.isExpression);
             var right = left;
-            if (shouldCaptureInTempVariable(left)) {
-                right = ts.createTempVariable(hoistVariableDeclaration);
-                left = ts.createAssignment(right, left);
+            if (!ts.isSimpleCopiableExpression(left)) {
+                right = factory.createTempVariable(hoistVariableDeclaration);
+                left = factory.createAssignment(right, left);
                 // if (inParameterInitializer) tempVariableInParameter = true;
             }
-            return ts.createConditional(createNotNullCondition(left, right), right, ts.visitNode(node.right, visitor, ts.isExpression));
-        }
-        function shouldCaptureInTempVariable(expression) {
-            // don't capture identifiers and `this` in a temporary variable
-            // `super` cannot be captured as it's no real variable
-            return !ts.isIdentifier(expression) &&
-                expression.kind !== 104 /* ThisKeyword */ &&
-                expression.kind !== 102 /* SuperKeyword */;
+            return factory.createConditionalExpression(createNotNullCondition(left, right), 
+            /*questionToken*/ undefined, right, 
+            /*colonToken*/ undefined, ts.visitNode(node.right, visitor, ts.isExpression));
         }
         function visitDeleteExpression(node) {
             return ts.isOptionalChain(ts.skipParentheses(node.expression))
                 ? ts.setOriginalNode(visitNonOptionalExpression(node.expression, /*captureThisArg*/ false, /*isDelete*/ true), node)
-                : ts.updateDelete(node, ts.visitNode(node.expression, visitor, ts.isExpression));
+                : factory.updateDeleteExpression(node, ts.visitNode(node.expression, visitor, ts.isExpression));
         }
     }
     ts.transformES2020 = transformES2020;
@@ -82617,7 +87450,8 @@ var ts;
 var ts;
 (function (ts) {
     function transformESNext(context) {
-        return ts.chainBundle(transformSourceFile);
+        var hoistVariableDeclaration = context.hoistVariableDeclaration, factory = context.factory;
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
@@ -82629,10 +87463,41 @@ var ts;
                 return node;
             }
             switch (node.kind) {
+                case 216 /* BinaryExpression */:
+                    var binaryExpression = node;
+                    if (ts.isLogicalOrCoalescingAssignmentExpression(binaryExpression)) {
+                        return transformLogicalAssignment(binaryExpression);
+                    }
+                // falls through
                 default:
                     return ts.visitEachChild(node, visitor, context);
             }
         }
+        function transformLogicalAssignment(binaryExpression) {
+            var operator = binaryExpression.operatorToken;
+            var nonAssignmentOperator = ts.getNonAssignmentOperatorForCompoundAssignment(operator.kind);
+            var left = ts.skipParentheses(ts.visitNode(binaryExpression.left, visitor, ts.isLeftHandSideExpression));
+            var assignmentTarget = left;
+            var right = ts.skipParentheses(ts.visitNode(binaryExpression.right, visitor, ts.isExpression));
+            if (ts.isAccessExpression(left)) {
+                var propertyAccessTargetSimpleCopiable = ts.isSimpleCopiableExpression(left.expression);
+                var propertyAccessTarget = propertyAccessTargetSimpleCopiable ? left.expression :
+                    factory.createTempVariable(hoistVariableDeclaration);
+                var propertyAccessTargetAssignment = propertyAccessTargetSimpleCopiable ? left.expression : factory.createAssignment(propertyAccessTarget, left.expression);
+                if (ts.isPropertyAccessExpression(left)) {
+                    assignmentTarget = factory.createPropertyAccessExpression(propertyAccessTarget, left.name);
+                    left = factory.createPropertyAccessExpression(propertyAccessTargetAssignment, left.name);
+                }
+                else {
+                    var elementAccessArgumentSimpleCopiable = ts.isSimpleCopiableExpression(left.argumentExpression);
+                    var elementAccessArgument = elementAccessArgumentSimpleCopiable ? left.argumentExpression :
+                        factory.createTempVariable(hoistVariableDeclaration);
+                    assignmentTarget = factory.createElementAccessExpression(propertyAccessTarget, elementAccessArgument);
+                    left = factory.createElementAccessExpression(propertyAccessTargetAssignment, elementAccessArgumentSimpleCopiable ? left.argumentExpression : factory.createAssignment(elementAccessArgument, left.argumentExpression));
+                }
+            }
+            return factory.createBinaryExpression(left, nonAssignmentOperator, factory.createParenthesizedExpression(factory.createAssignment(assignmentTarget, right)));
+        }
     }
     ts.transformESNext = transformESNext;
 })(ts || (ts = {}));
@@ -82640,9 +87505,52 @@ var ts;
 var ts;
 (function (ts) {
     function transformJsx(context) {
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory;
         var compilerOptions = context.getCompilerOptions();
         var currentSourceFile;
-        return ts.chainBundle(transformSourceFile);
+        var currentFileState;
+        return ts.chainBundle(context, transformSourceFile);
+        function getCurrentFileNameExpression() {
+            if (currentFileState.filenameDeclaration) {
+                return currentFileState.filenameDeclaration.name;
+            }
+            var declaration = factory.createVariableDeclaration(factory.createUniqueName("_jsxFileName", 16 /* Optimistic */ | 32 /* FileLevel */), /*exclaimationToken*/ undefined, /*type*/ undefined, factory.createStringLiteral(currentSourceFile.fileName));
+            currentFileState.filenameDeclaration = declaration;
+            return currentFileState.filenameDeclaration.name;
+        }
+        function getJsxFactoryCalleePrimitive(childrenLength) {
+            return compilerOptions.jsx === 5 /* ReactJSXDev */ ? "jsxDEV" : childrenLength > 1 ? "jsxs" : "jsx";
+        }
+        function getJsxFactoryCallee(childrenLength) {
+            var type = getJsxFactoryCalleePrimitive(childrenLength);
+            return getImplicitImportForName(type);
+        }
+        function getImplicitJsxFragmentReference() {
+            return getImplicitImportForName("Fragment");
+        }
+        function getImplicitImportForName(name) {
+            var _a, _b;
+            var importSource = name === "createElement"
+                ? currentFileState.importSpecifier
+                : ts.getJSXRuntimeImport(currentFileState.importSpecifier, compilerOptions);
+            var existing = (_b = (_a = currentFileState.utilizedImplicitRuntimeImports) === null || _a === void 0 ? void 0 : _a.get(importSource)) === null || _b === void 0 ? void 0 : _b.get(name);
+            if (existing) {
+                return existing.name;
+            }
+            if (!currentFileState.utilizedImplicitRuntimeImports) {
+                currentFileState.utilizedImplicitRuntimeImports = ts.createMap();
+            }
+            var specifierSourceImports = currentFileState.utilizedImplicitRuntimeImports.get(importSource);
+            if (!specifierSourceImports) {
+                specifierSourceImports = ts.createMap();
+                currentFileState.utilizedImplicitRuntimeImports.set(importSource, specifierSourceImports);
+            }
+            var generatedName = factory.createUniqueName("_" + name, 16 /* Optimistic */ | 32 /* FileLevel */ | 64 /* AllowNameSubstitution */);
+            var specifier = factory.createImportSpecifier(factory.createIdentifier(name), generatedName);
+            generatedName.generatedImportReference = specifier;
+            specifierSourceImports.set(name, specifier);
+            return generatedName;
+        }
         /**
          * Transform JSX-specific syntax in a SourceFile.
          *
@@ -82653,8 +87561,42 @@ var ts;
                 return node;
             }
             currentSourceFile = node;
+            currentFileState = {};
+            currentFileState.importSpecifier = ts.getJSXImplicitImportBase(compilerOptions, node);
             var visited = ts.visitEachChild(node, visitor, context);
             ts.addEmitHelpers(visited, context.readEmitHelpers());
+            var statements = visited.statements;
+            if (currentFileState.filenameDeclaration) {
+                statements = ts.insertStatementAfterCustomPrologue(statements.slice(), factory.createVariableStatement(/*modifiers*/ undefined, factory.createVariableDeclarationList([currentFileState.filenameDeclaration], 2 /* Const */)));
+            }
+            if (currentFileState.utilizedImplicitRuntimeImports) {
+                for (var _i = 0, _a = ts.arrayFrom(currentFileState.utilizedImplicitRuntimeImports.entries()); _i < _a.length; _i++) {
+                    var _b = _a[_i], importSource = _b[0], importSpecifiersMap = _b[1];
+                    if (ts.isExternalModule(node)) {
+                        // Add `import` statement
+                        var importStatement = factory.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, factory.createImportClause(/*typeOnly*/ false, /*name*/ undefined, factory.createNamedImports(ts.arrayFrom(importSpecifiersMap.values()))), factory.createStringLiteral(importSource));
+                        ts.setParentRecursive(importStatement, /*incremental*/ false);
+                        statements = ts.insertStatementAfterCustomPrologue(statements.slice(), importStatement);
+                    }
+                    else if (ts.isExternalOrCommonJsModule(node)) {
+                        // Add `require` statement
+                        var requireStatement = factory.createVariableStatement(/*modifiers*/ undefined, factory.createVariableDeclarationList([
+                            factory.createVariableDeclaration(factory.createObjectBindingPattern(ts.map(ts.arrayFrom(importSpecifiersMap.values()), function (s) { return factory.createBindingElement(/*dotdotdot*/ undefined, s.propertyName, s.name); })), 
+                            /*exclaimationToken*/ undefined, 
+                            /*type*/ undefined, factory.createCallExpression(factory.createIdentifier("require"), /*typeArguments*/ undefined, [factory.createStringLiteral(importSource)]))
+                        ], 2 /* Const */));
+                        ts.setParentRecursive(requireStatement, /*incremental*/ false);
+                        statements = ts.insertStatementAfterCustomPrologue(statements.slice(), requireStatement);
+                    }
+                    else {
+                        // Do nothing (script file) - consider an error in the checker?
+                    }
+                }
+            }
+            if (statements !== visited.statements) {
+                visited = factory.updateSourceFile(visited, statements);
+            }
+            currentFileState = undefined;
             return visited;
         }
         function visitor(node) {
@@ -82667,13 +87609,13 @@ var ts;
         }
         function visitorWorker(node) {
             switch (node.kind) {
-                case 266 /* JsxElement */:
+                case 273 /* JsxElement */:
                     return visitJsxElement(node, /*isChild*/ false);
-                case 267 /* JsxSelfClosingElement */:
+                case 274 /* JsxSelfClosingElement */:
                     return visitJsxSelfClosingElement(node, /*isChild*/ false);
-                case 270 /* JsxFragment */:
+                case 277 /* JsxFragment */:
                     return visitJsxFragment(node, /*isChild*/ false);
-                case 276 /* JsxExpression */:
+                case 283 /* JsxExpression */:
                     return visitJsxExpression(node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -82683,62 +87625,170 @@ var ts;
             switch (node.kind) {
                 case 11 /* JsxText */:
                     return visitJsxText(node);
-                case 276 /* JsxExpression */:
+                case 283 /* JsxExpression */:
                     return visitJsxExpression(node);
-                case 266 /* JsxElement */:
+                case 273 /* JsxElement */:
                     return visitJsxElement(node, /*isChild*/ true);
-                case 267 /* JsxSelfClosingElement */:
+                case 274 /* JsxSelfClosingElement */:
                     return visitJsxSelfClosingElement(node, /*isChild*/ true);
-                case 270 /* JsxFragment */:
+                case 277 /* JsxFragment */:
                     return visitJsxFragment(node, /*isChild*/ true);
                 default:
                     return ts.Debug.failBadSyntaxKind(node);
             }
         }
+        /**
+         * The react jsx/jsxs transform falls back to `createElement` when an explicit `key` argument comes after a spread
+         */
+        function hasKeyAfterPropsSpread(node) {
+            var spread = false;
+            for (var _i = 0, _a = node.attributes.properties; _i < _a.length; _i++) {
+                var elem = _a[_i];
+                if (ts.isJsxSpreadAttribute(elem)) {
+                    spread = true;
+                }
+                else if (spread && ts.isJsxAttribute(elem) && elem.name.escapedText === "key") {
+                    return true;
+                }
+            }
+            return false;
+        }
+        function shouldUseCreateElement(node) {
+            return currentFileState.importSpecifier === undefined || hasKeyAfterPropsSpread(node);
+        }
         function visitJsxElement(node, isChild) {
-            return visitJsxOpeningLikeElement(node.openingElement, node.children, isChild, /*location*/ node);
+            var tagTransform = shouldUseCreateElement(node.openingElement) ? visitJsxOpeningLikeElementCreateElement : visitJsxOpeningLikeElementJSX;
+            return tagTransform(node.openingElement, node.children, isChild, /*location*/ node);
         }
         function visitJsxSelfClosingElement(node, isChild) {
-            return visitJsxOpeningLikeElement(node, /*children*/ undefined, isChild, /*location*/ node);
+            var tagTransform = shouldUseCreateElement(node) ? visitJsxOpeningLikeElementCreateElement : visitJsxOpeningLikeElementJSX;
+            return tagTransform(node, /*children*/ undefined, isChild, /*location*/ node);
         }
         function visitJsxFragment(node, isChild) {
-            return visitJsxOpeningFragment(node.openingFragment, node.children, isChild, /*location*/ node);
+            var tagTransform = currentFileState.importSpecifier === undefined ? visitJsxOpeningFragmentCreateElement : visitJsxOpeningFragmentJSX;
+            return tagTransform(node.openingFragment, node.children, isChild, /*location*/ node);
+        }
+        function convertJsxChildrenToChildrenPropObject(children) {
+            var nonWhitespaceChildren = ts.getSemanticJsxChildren(children);
+            if (ts.length(nonWhitespaceChildren) === 1) {
+                var result_13 = transformJsxChildToExpression(nonWhitespaceChildren[0]);
+                return result_13 && factory.createObjectLiteralExpression([
+                    factory.createPropertyAssignment("children", result_13)
+                ]);
+            }
+            var result = ts.mapDefined(children, transformJsxChildToExpression);
+            return !result.length ? undefined : factory.createObjectLiteralExpression([
+                factory.createPropertyAssignment("children", factory.createArrayLiteralExpression(result))
+            ]);
         }
-        function visitJsxOpeningLikeElement(node, children, isChild, location) {
+        function visitJsxOpeningLikeElementJSX(node, children, isChild, location) {
+            var tagName = getTagName(node);
+            var objectProperties;
+            var keyAttr = ts.find(node.attributes.properties, function (p) { return !!p.name && ts.isIdentifier(p.name) && p.name.escapedText === "key"; });
+            var attrs = keyAttr ? ts.filter(node.attributes.properties, function (p) { return p !== keyAttr; }) : node.attributes.properties;
+            var segments = [];
+            if (attrs.length) {
+                // Map spans of JsxAttribute nodes into object literals and spans
+                // of JsxSpreadAttribute nodes into expressions.
+                segments = ts.flatten(ts.spanMap(attrs, ts.isJsxSpreadAttribute, function (attrs, isSpread) { return isSpread
+                    ? ts.map(attrs, transformJsxSpreadAttributeToExpression)
+                    : factory.createObjectLiteralExpression(ts.map(attrs, transformJsxAttributeToObjectLiteralElement)); }));
+                if (ts.isJsxSpreadAttribute(attrs[0])) {
+                    // We must always emit at least one object literal before a spread
+                    // argument.factory.createObjectLiteral
+                    segments.unshift(factory.createObjectLiteralExpression());
+                }
+            }
+            if (children && children.length) {
+                var result = convertJsxChildrenToChildrenPropObject(children);
+                if (result) {
+                    segments.push(result);
+                }
+            }
+            if (segments.length === 0) {
+                objectProperties = factory.createObjectLiteralExpression([]);
+                // When there are no attributes, React wants {}
+            }
+            else {
+                // Either emit one big object literal (no spread attribs), or
+                // a call to the __assign helper.
+                objectProperties = ts.singleOrUndefined(segments) || emitHelpers().createAssignHelper(segments);
+            }
+            return visitJsxOpeningLikeElementOrFragmentJSX(tagName, objectProperties, keyAttr, ts.length(ts.getSemanticJsxChildren(children || ts.emptyArray)), isChild, location);
+        }
+        function visitJsxOpeningLikeElementOrFragmentJSX(tagName, objectProperties, keyAttr, childrenLength, isChild, location) {
+            var args = [tagName, objectProperties, !keyAttr ? factory.createVoidZero() : transformJsxAttributeInitializer(keyAttr.initializer)];
+            if (compilerOptions.jsx === 5 /* ReactJSXDev */) {
+                var originalFile = ts.getOriginalNode(currentSourceFile);
+                if (originalFile && ts.isSourceFile(originalFile)) {
+                    // isStaticChildren development flag
+                    args.push(childrenLength > 1 ? factory.createTrue() : factory.createFalse());
+                    // __source development flag
+                    var lineCol = ts.getLineAndCharacterOfPosition(originalFile, location.pos);
+                    args.push(factory.createObjectLiteralExpression([
+                        factory.createPropertyAssignment("fileName", getCurrentFileNameExpression()),
+                        factory.createPropertyAssignment("lineNumber", factory.createNumericLiteral(lineCol.line + 1)),
+                        factory.createPropertyAssignment("columnNumber", factory.createNumericLiteral(lineCol.character + 1))
+                    ]));
+                    // __self development flag
+                    args.push(factory.createThis());
+                }
+            }
+            var element = ts.setTextRange(factory.createCallExpression(getJsxFactoryCallee(childrenLength), /*typeArguments*/ undefined, args), location);
+            if (isChild) {
+                ts.startOnNewLine(element);
+            }
+            return element;
+        }
+        function visitJsxOpeningLikeElementCreateElement(node, children, isChild, location) {
             var tagName = getTagName(node);
             var objectProperties;
             var attrs = node.attributes.properties;
             if (attrs.length === 0) {
+                objectProperties = factory.createNull();
                 // When there are no attributes, React wants "null"
-                objectProperties = ts.createNull();
             }
             else {
                 // Map spans of JsxAttribute nodes into object literals and spans
                 // of JsxSpreadAttribute nodes into expressions.
                 var segments = ts.flatten(ts.spanMap(attrs, ts.isJsxSpreadAttribute, function (attrs, isSpread) { return isSpread
                     ? ts.map(attrs, transformJsxSpreadAttributeToExpression)
-                    : ts.createObjectLiteral(ts.map(attrs, transformJsxAttributeToObjectLiteralElement)); }));
+                    : factory.createObjectLiteralExpression(ts.map(attrs, transformJsxAttributeToObjectLiteralElement)); }));
                 if (ts.isJsxSpreadAttribute(attrs[0])) {
                     // We must always emit at least one object literal before a spread
-                    // argument.
-                    segments.unshift(ts.createObjectLiteral());
+                    // argument.factory.createObjectLiteral
+                    segments.unshift(factory.createObjectLiteralExpression());
                 }
                 // Either emit one big object literal (no spread attribs), or
                 // a call to the __assign helper.
                 objectProperties = ts.singleOrUndefined(segments);
                 if (!objectProperties) {
-                    objectProperties = ts.createAssignHelper(context, segments);
+                    objectProperties = emitHelpers().createAssignHelper(segments);
                 }
             }
-            var element = ts.createExpressionForJsxElement(context.getEmitResolver().getJsxFactoryEntity(currentSourceFile), compilerOptions.reactNamespace, // TODO: GH#18217
-            tagName, objectProperties, ts.mapDefined(children, transformJsxChildToExpression), node, location);
+            var callee = currentFileState.importSpecifier === undefined
+                ? ts.createJsxFactoryExpression(factory, context.getEmitResolver().getJsxFactoryEntity(currentSourceFile), compilerOptions.reactNamespace, // TODO: GH#18217
+                node)
+                : getImplicitImportForName("createElement");
+            var element = ts.createExpressionForJsxElement(factory, callee, tagName, objectProperties, ts.mapDefined(children, transformJsxChildToExpression), location);
             if (isChild) {
                 ts.startOnNewLine(element);
             }
             return element;
         }
-        function visitJsxOpeningFragment(node, children, isChild, location) {
-            var element = ts.createExpressionForJsxFragment(context.getEmitResolver().getJsxFactoryEntity(currentSourceFile), compilerOptions.reactNamespace, // TODO: GH#18217
+        function visitJsxOpeningFragmentJSX(_node, children, isChild, location) {
+            var childrenProps;
+            if (children && children.length) {
+                var result = convertJsxChildrenToChildrenPropObject(children);
+                if (result) {
+                    childrenProps = result;
+                }
+            }
+            return visitJsxOpeningLikeElementOrFragmentJSX(getImplicitJsxFragmentReference(), childrenProps || factory.createObjectLiteralExpression([]), 
+            /*keyAttr*/ undefined, ts.length(ts.getSemanticJsxChildren(children)), isChild, location);
+        }
+        function visitJsxOpeningFragmentCreateElement(node, children, isChild, location) {
+            var element = ts.createExpressionForJsxFragment(factory, context.getEmitResolver().getJsxFactoryEntity(currentSourceFile), context.getEmitResolver().getJsxFragmentFactoryEntity(currentSourceFile), compilerOptions.reactNamespace, // TODO: GH#18217
             ts.mapDefined(children, transformJsxChildToExpression), node, location);
             if (isChild) {
                 ts.startOnNewLine(element);
@@ -82751,24 +87801,24 @@ var ts;
         function transformJsxAttributeToObjectLiteralElement(node) {
             var name = getAttributeName(node);
             var expression = transformJsxAttributeInitializer(node.initializer);
-            return ts.createPropertyAssignment(name, expression);
+            return factory.createPropertyAssignment(name, expression);
         }
         function transformJsxAttributeInitializer(node) {
             if (node === undefined) {
-                return ts.createTrue();
+                return factory.createTrue();
             }
             else if (node.kind === 10 /* StringLiteral */) {
                 // Always recreate the literal to escape any escape sequences or newlines which may be in the original jsx string and which
                 // Need to be escaped to be handled correctly in a normal string
-                var literal = ts.createLiteral(tryDecodeEntities(node.text) || node.text);
-                literal.singleQuote = node.singleQuote !== undefined ? node.singleQuote : !ts.isStringDoubleQuoted(node, currentSourceFile);
+                var singleQuote = node.singleQuote !== undefined ? node.singleQuote : !ts.isStringDoubleQuoted(node, currentSourceFile);
+                var literal = factory.createStringLiteral(tryDecodeEntities(node.text) || node.text, singleQuote);
                 return ts.setTextRange(literal, node);
             }
-            else if (node.kind === 276 /* JsxExpression */) {
+            else if (node.kind === 283 /* JsxExpression */) {
                 if (node.expression === undefined) {
-                    return ts.createTrue();
+                    return factory.createTrue();
                 }
-                return visitJsxExpression(node);
+                return ts.visitNode(node.expression, visitor, ts.isExpression);
             }
             else {
                 return ts.Debug.failBadSyntaxKind(node);
@@ -82776,7 +87826,7 @@ var ts;
         }
         function visitJsxText(node) {
             var fixed = fixupWhitespaceAndDecodeEntities(node.text);
-            return fixed === undefined ? undefined : ts.createLiteral(fixed);
+            return fixed === undefined ? undefined : factory.createStringLiteral(fixed);
         }
         /**
          * JSX trims whitespace at the end and beginning of lines, except that the
@@ -82858,16 +87908,16 @@ var ts;
             return decoded === text ? undefined : decoded;
         }
         function getTagName(node) {
-            if (node.kind === 266 /* JsxElement */) {
+            if (node.kind === 273 /* JsxElement */) {
                 return getTagName(node.openingElement);
             }
             else {
                 var name = node.tagName;
                 if (ts.isIdentifier(name) && ts.isIntrinsicJsxName(name.escapedText)) {
-                    return ts.createLiteral(ts.idText(name));
+                    return factory.createStringLiteral(ts.idText(name));
                 }
                 else {
-                    return ts.createExpressionFromEntityName(name);
+                    return ts.createExpressionFromEntityName(factory, name);
                 }
             }
         }
@@ -82883,7 +87933,7 @@ var ts;
                 return name;
             }
             else {
-                return ts.createLiteral(text);
+                return factory.createStringLiteral(text);
             }
         }
         function visitJsxExpression(node) {
@@ -82891,7 +87941,7 @@ var ts;
         }
     }
     ts.transformJsx = transformJsx;
-    var entities = ts.createMapFromTemplate({
+    var entities = new ts.Map(ts.getEntries({
         quot: 0x0022,
         amp: 0x0026,
         apos: 0x0027,
@@ -83145,14 +88195,14 @@ var ts;
         clubs: 0x2663,
         hearts: 0x2665,
         diams: 0x2666
-    });
+    }));
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
 (function (ts) {
     function transformES2016(context) {
-        var hoistVariableDeclaration = context.hoistVariableDeclaration;
-        return ts.chainBundle(transformSourceFile);
+        var factory = context.factory, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
@@ -83164,7 +88214,7 @@ var ts;
                 return node;
             }
             switch (node.kind) {
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return visitBinaryExpression(node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -83187,29 +88237,29 @@ var ts;
             var right = ts.visitNode(node.right, visitor, ts.isExpression);
             if (ts.isElementAccessExpression(left)) {
                 // Transforms `a[x] **= b` into `(_a = a)[_x = x] = Math.pow(_a[_x], b)`
-                var expressionTemp = ts.createTempVariable(hoistVariableDeclaration);
-                var argumentExpressionTemp = ts.createTempVariable(hoistVariableDeclaration);
-                target = ts.setTextRange(ts.createElementAccess(ts.setTextRange(ts.createAssignment(expressionTemp, left.expression), left.expression), ts.setTextRange(ts.createAssignment(argumentExpressionTemp, left.argumentExpression), left.argumentExpression)), left);
-                value = ts.setTextRange(ts.createElementAccess(expressionTemp, argumentExpressionTemp), left);
+                var expressionTemp = factory.createTempVariable(hoistVariableDeclaration);
+                var argumentExpressionTemp = factory.createTempVariable(hoistVariableDeclaration);
+                target = ts.setTextRange(factory.createElementAccessExpression(ts.setTextRange(factory.createAssignment(expressionTemp, left.expression), left.expression), ts.setTextRange(factory.createAssignment(argumentExpressionTemp, left.argumentExpression), left.argumentExpression)), left);
+                value = ts.setTextRange(factory.createElementAccessExpression(expressionTemp, argumentExpressionTemp), left);
             }
             else if (ts.isPropertyAccessExpression(left)) {
                 // Transforms `a.x **= b` into `(_a = a).x = Math.pow(_a.x, b)`
-                var expressionTemp = ts.createTempVariable(hoistVariableDeclaration);
-                target = ts.setTextRange(ts.createPropertyAccess(ts.setTextRange(ts.createAssignment(expressionTemp, left.expression), left.expression), left.name), left);
-                value = ts.setTextRange(ts.createPropertyAccess(expressionTemp, left.name), left);
+                var expressionTemp = factory.createTempVariable(hoistVariableDeclaration);
+                target = ts.setTextRange(factory.createPropertyAccessExpression(ts.setTextRange(factory.createAssignment(expressionTemp, left.expression), left.expression), left.name), left);
+                value = ts.setTextRange(factory.createPropertyAccessExpression(expressionTemp, left.name), left);
             }
             else {
                 // Transforms `a **= b` into `a = Math.pow(a, b)`
                 target = left;
                 value = left;
             }
-            return ts.setTextRange(ts.createAssignment(target, ts.createMathPow(value, right, /*location*/ node)), node);
+            return ts.setTextRange(factory.createAssignment(target, ts.setTextRange(factory.createGlobalMethodCall("Math", "pow", [value, right]), node)), node);
         }
         function visitExponentiationExpression(node) {
             // Transforms `a ** b` into `Math.pow(a, b)`
             var left = ts.visitNode(node.left, visitor, ts.isExpression);
             var right = ts.visitNode(node.right, visitor, ts.isExpression);
-            return ts.createMathPow(left, right, /*location*/ node);
+            return ts.setTextRange(factory.createGlobalMethodCall("Math", "pow", [left, right]), node);
         }
     }
     ts.transformES2016 = transformES2016;
@@ -83316,7 +88366,7 @@ var ts;
         HierarchyFacts[HierarchyFacts["FunctionSubtreeExcludes"] = 49152] = "FunctionSubtreeExcludes";
     })(HierarchyFacts || (HierarchyFacts = {}));
     function transformES2015(context) {
-        var startLexicalEnvironment = context.startLexicalEnvironment, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory, startLexicalEnvironment = context.startLexicalEnvironment, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var compilerOptions = context.getCompilerOptions();
         var resolver = context.getEmitResolver();
         var previousOnSubstituteNode = context.onSubstituteNode;
@@ -83328,7 +88378,7 @@ var ts;
         var hierarchyFacts;
         var taggedTemplateStringDeclarations;
         function recordTaggedTemplateString(temp) {
-            taggedTemplateStringDeclarations = ts.append(taggedTemplateStringDeclarations, ts.createVariableDeclaration(temp));
+            taggedTemplateStringDeclarations = ts.append(taggedTemplateStringDeclarations, factory.createVariableDeclaration(temp));
         }
         /**
          * Used to track if we are emitting body of the converted loop
@@ -83340,7 +88390,7 @@ var ts;
          * be reset.
          */
         var enabledSubstitutions;
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
@@ -83377,92 +88427,107 @@ var ts;
         }
         function isReturnVoidStatementInConstructorWithCapturedSuper(node) {
             return (hierarchyFacts & 8192 /* ConstructorWithCapturedSuper */) !== 0
-                && node.kind === 235 /* ReturnStatement */
+                && node.kind === 242 /* ReturnStatement */
                 && !node.expression;
         }
+        function isOrMayContainReturnCompletion(node) {
+            return node.transformFlags & 1048576 /* ContainsHoistedDeclarationOrCompletion */
+                && (ts.isReturnStatement(node)
+                    || ts.isIfStatement(node)
+                    || ts.isWithStatement(node)
+                    || ts.isSwitchStatement(node)
+                    || ts.isCaseBlock(node)
+                    || ts.isCaseClause(node)
+                    || ts.isDefaultClause(node)
+                    || ts.isTryStatement(node)
+                    || ts.isCatchClause(node)
+                    || ts.isLabeledStatement(node)
+                    || ts.isIterationStatement(node, /*lookInLabeledStatements*/ false)
+                    || ts.isBlock(node));
+        }
         function shouldVisitNode(node) {
             return (node.transformFlags & 256 /* ContainsES2015 */) !== 0
                 || convertedLoopState !== undefined
-                || (hierarchyFacts & 8192 /* ConstructorWithCapturedSuper */ && (ts.isStatement(node) || (node.kind === 223 /* Block */)))
+                || (hierarchyFacts & 8192 /* ConstructorWithCapturedSuper */ && isOrMayContainReturnCompletion(node))
                 || (ts.isIterationStatement(node, /*lookInLabeledStatements*/ false) && shouldConvertIterationStatement(node))
                 || (ts.getEmitFlags(node) & 33554432 /* TypeScriptClassWrapper */) !== 0;
         }
         function visitor(node) {
-            if (shouldVisitNode(node)) {
-                return visitJavaScript(node);
-            }
-            else {
-                return node;
-            }
+            return shouldVisitNode(node) ? visitorWorker(node, /*expressionResultIsUnused*/ false) : node;
+        }
+        function visitorWithUnusedExpressionResult(node) {
+            return shouldVisitNode(node) ? visitorWorker(node, /*expressionResultIsUnused*/ true) : node;
         }
         function callExpressionVisitor(node) {
-            if (node.kind === 102 /* SuperKeyword */) {
+            if (node.kind === 105 /* SuperKeyword */) {
                 return visitSuperKeyword(/*isExpressionOfCall*/ true);
             }
             return visitor(node);
         }
-        function visitJavaScript(node) {
+        function visitorWorker(node, expressionResultIsUnused) {
             switch (node.kind) {
-                case 120 /* StaticKeyword */:
+                case 123 /* StaticKeyword */:
                     return undefined; // elide static keyword
-                case 245 /* ClassDeclaration */:
+                case 252 /* ClassDeclaration */:
                     return visitClassDeclaration(node);
-                case 214 /* ClassExpression */:
+                case 221 /* ClassExpression */:
                     return visitClassExpression(node);
-                case 156 /* Parameter */:
+                case 160 /* Parameter */:
                     return visitParameter(node);
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return visitFunctionDeclaration(node);
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     return visitArrowFunction(node);
-                case 201 /* FunctionExpression */:
+                case 208 /* FunctionExpression */:
                     return visitFunctionExpression(node);
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return visitVariableDeclaration(node);
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return visitIdentifier(node);
-                case 243 /* VariableDeclarationList */:
+                case 250 /* VariableDeclarationList */:
                     return visitVariableDeclarationList(node);
-                case 237 /* SwitchStatement */:
+                case 244 /* SwitchStatement */:
                     return visitSwitchStatement(node);
-                case 251 /* CaseBlock */:
+                case 258 /* CaseBlock */:
                     return visitCaseBlock(node);
-                case 223 /* Block */:
+                case 230 /* Block */:
                     return visitBlock(node, /*isFunctionBody*/ false);
-                case 234 /* BreakStatement */:
-                case 233 /* ContinueStatement */:
+                case 241 /* BreakStatement */:
+                case 240 /* ContinueStatement */:
                     return visitBreakOrContinueStatement(node);
-                case 238 /* LabeledStatement */:
+                case 245 /* LabeledStatement */:
                     return visitLabeledStatement(node);
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
                     return visitDoOrWhileStatement(node, /*outermostLabeledStatement*/ undefined);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return visitForStatement(node, /*outermostLabeledStatement*/ undefined);
-                case 231 /* ForInStatement */:
+                case 238 /* ForInStatement */:
                     return visitForInStatement(node, /*outermostLabeledStatement*/ undefined);
-                case 232 /* ForOfStatement */:
+                case 239 /* ForOfStatement */:
                     return visitForOfStatement(node, /*outermostLabeledStatement*/ undefined);
-                case 226 /* ExpressionStatement */:
+                case 233 /* ExpressionStatement */:
                     return visitExpressionStatement(node);
-                case 193 /* ObjectLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
                     return visitObjectLiteralExpression(node);
-                case 280 /* CatchClause */:
+                case 287 /* CatchClause */:
                     return visitCatchClause(node);
-                case 282 /* ShorthandPropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
                     return visitShorthandPropertyAssignment(node);
-                case 154 /* ComputedPropertyName */:
+                case 158 /* ComputedPropertyName */:
                     return visitComputedPropertyName(node);
-                case 192 /* ArrayLiteralExpression */:
+                case 199 /* ArrayLiteralExpression */:
                     return visitArrayLiteralExpression(node);
-                case 196 /* CallExpression */:
+                case 203 /* CallExpression */:
                     return visitCallExpression(node);
-                case 197 /* NewExpression */:
+                case 204 /* NewExpression */:
                     return visitNewExpression(node);
-                case 200 /* ParenthesizedExpression */:
-                    return visitParenthesizedExpression(node, /*needsDestructuringValue*/ true);
-                case 209 /* BinaryExpression */:
-                    return visitBinaryExpression(node, /*needsDestructuringValue*/ true);
+                case 207 /* ParenthesizedExpression */:
+                    return visitParenthesizedExpression(node, expressionResultIsUnused);
+                case 216 /* BinaryExpression */:
+                    return visitBinaryExpression(node, expressionResultIsUnused);
+                case 337 /* CommaListExpression */:
+                    return visitCommaListExpression(node, expressionResultIsUnused);
                 case 14 /* NoSubstitutionTemplateLiteral */:
                 case 15 /* TemplateHead */:
                 case 16 /* TemplateMiddle */:
@@ -83472,29 +88537,31 @@ var ts;
                     return visitStringLiteral(node);
                 case 8 /* NumericLiteral */:
                     return visitNumericLiteral(node);
-                case 198 /* TaggedTemplateExpression */:
+                case 205 /* TaggedTemplateExpression */:
                     return visitTaggedTemplateExpression(node);
-                case 211 /* TemplateExpression */:
+                case 218 /* TemplateExpression */:
                     return visitTemplateExpression(node);
-                case 212 /* YieldExpression */:
+                case 219 /* YieldExpression */:
                     return visitYieldExpression(node);
-                case 213 /* SpreadElement */:
+                case 220 /* SpreadElement */:
                     return visitSpreadElement(node);
-                case 102 /* SuperKeyword */:
+                case 105 /* SuperKeyword */:
                     return visitSuperKeyword(/*isExpressionOfCall*/ false);
-                case 104 /* ThisKeyword */:
+                case 107 /* ThisKeyword */:
                     return visitThisKeyword(node);
-                case 219 /* MetaProperty */:
+                case 226 /* MetaProperty */:
                     return visitMetaProperty(node);
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     return visitMethodDeclaration(node);
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return visitAccessorDeclaration(node);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return visitVariableStatement(node);
-                case 235 /* ReturnStatement */:
+                case 242 /* ReturnStatement */:
                     return visitReturnStatement(node);
+                case 212 /* VoidExpression */:
+                    return visitVoidExpression(node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
             }
@@ -83504,16 +88571,15 @@ var ts;
             var prologue = [];
             var statements = [];
             startLexicalEnvironment();
-            var statementOffset = ts.addStandardPrologue(prologue, node.statements, /*ensureUseStrict*/ false);
-            statementOffset = ts.addCustomPrologue(prologue, node.statements, statementOffset, visitor);
+            var statementOffset = factory.copyPrologue(node.statements, prologue, /*ensureUseStrict*/ false, visitor);
             ts.addRange(statements, ts.visitNodes(node.statements, visitor, ts.isStatement, statementOffset));
             if (taggedTemplateStringDeclarations) {
-                statements.push(ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList(taggedTemplateStringDeclarations)));
+                statements.push(factory.createVariableStatement(/*modifiers*/ undefined, factory.createVariableDeclarationList(taggedTemplateStringDeclarations)));
             }
-            ts.mergeLexicalEnvironment(prologue, endLexicalEnvironment());
+            factory.mergeLexicalEnvironment(prologue, endLexicalEnvironment());
             insertCaptureThisForNodeIfNeeded(prologue, node);
             exitSubtree(ancestorFacts, 0 /* None */, 0 /* None */);
-            return ts.updateSourceFileNode(node, ts.setTextRange(ts.createNodeArray(ts.concatenate(prologue, statements)), node.statements));
+            return factory.updateSourceFile(node, ts.setTextRange(factory.createNodeArray(ts.concatenate(prologue, statements)), node.statements));
         }
         function visitSwitchStatement(node) {
             if (convertedLoopState !== undefined) {
@@ -83533,7 +88599,7 @@ var ts;
             return updated;
         }
         function returnCapturedThis(node) {
-            return ts.setOriginalNode(ts.createReturn(ts.createFileLevelUniqueName("_this")), node);
+            return ts.setOriginalNode(factory.createReturnStatement(factory.createUniqueName("_this", 16 /* Optimistic */ | 32 /* FileLevel */)), node);
         }
         function visitReturnStatement(node) {
             if (convertedLoopState) {
@@ -83541,10 +88607,10 @@ var ts;
                 if (isReturnVoidStatementInConstructorWithCapturedSuper(node)) {
                     node = returnCapturedThis(node);
                 }
-                return ts.createReturn(ts.createObjectLiteral([
-                    ts.createPropertyAssignment(ts.createIdentifier("value"), node.expression
+                return factory.createReturnStatement(factory.createObjectLiteralExpression([
+                    factory.createPropertyAssignment(factory.createIdentifier("value"), node.expression
                         ? ts.visitNode(node.expression, visitor, ts.isExpression)
-                        : ts.createVoidZero())
+                        : factory.createVoidZero())
                 ]));
             }
             else if (isReturnVoidStatementInConstructorWithCapturedSuper(node)) {
@@ -83562,21 +88628,21 @@ var ts;
                     convertedLoopState.containsLexicalThis = true;
                     return node;
                 }
-                return convertedLoopState.thisName || (convertedLoopState.thisName = ts.createUniqueName("this"));
+                return convertedLoopState.thisName || (convertedLoopState.thisName = factory.createUniqueName("this"));
             }
             return node;
         }
+        function visitVoidExpression(node) {
+            return ts.visitEachChild(node, visitorWithUnusedExpressionResult, context);
+        }
         function visitIdentifier(node) {
             if (!convertedLoopState) {
                 return node;
             }
-            if (ts.isGeneratedIdentifier(node)) {
-                return node;
+            if (resolver.isArgumentsLocalBinding(node)) {
+                return convertedLoopState.argumentsName || (convertedLoopState.argumentsName = factory.createUniqueName("arguments"));
             }
-            if (node.escapedText !== "arguments" || !resolver.isArgumentsLocalBinding(node)) {
-                return node;
-            }
-            return convertedLoopState.argumentsName || (convertedLoopState.argumentsName = ts.createUniqueName("arguments"));
+            return node;
         }
         function visitBreakOrContinueStatement(node) {
             if (convertedLoopState) {
@@ -83584,14 +88650,14 @@ var ts;
                 // it is possible if either
                 //   - break/continue is labeled and label is located inside the converted loop
                 //   - break/continue is non-labeled and located in non-converted loop/switch statement
-                var jump = node.kind === 234 /* BreakStatement */ ? 2 /* Break */ : 4 /* Continue */;
+                var jump = node.kind === 241 /* BreakStatement */ ? 2 /* Break */ : 4 /* Continue */;
                 var canUseBreakOrContinue = (node.label && convertedLoopState.labels && convertedLoopState.labels.get(ts.idText(node.label))) ||
                     (!node.label && (convertedLoopState.allowedNonLabeledJumps & jump));
                 if (!canUseBreakOrContinue) {
                     var labelMarker = void 0;
                     var label = node.label;
                     if (!label) {
-                        if (node.kind === 234 /* BreakStatement */) {
+                        if (node.kind === 241 /* BreakStatement */) {
                             convertedLoopState.nonLocalJumps |= 2 /* Break */;
                             labelMarker = "break";
                         }
@@ -83602,7 +88668,7 @@ var ts;
                         }
                     }
                     else {
-                        if (node.kind === 234 /* BreakStatement */) {
+                        if (node.kind === 241 /* BreakStatement */) {
                             labelMarker = "break-" + label.escapedText;
                             setLabeledJump(convertedLoopState, /*isBreak*/ true, ts.idText(label), labelMarker);
                         }
@@ -83611,7 +88677,7 @@ var ts;
                             setLabeledJump(convertedLoopState, /*isBreak*/ false, ts.idText(label), labelMarker);
                         }
                     }
-                    var returnExpression = ts.createLiteral(labelMarker);
+                    var returnExpression = factory.createStringLiteral(labelMarker);
                     if (convertedLoopState.loopOutParameters.length) {
                         var outParams = convertedLoopState.loopOutParameters;
                         var expr = void 0;
@@ -83621,12 +88687,12 @@ var ts;
                                 expr = copyExpr;
                             }
                             else {
-                                expr = ts.createBinary(expr, 27 /* CommaToken */, copyExpr);
+                                expr = factory.createBinaryExpression(expr, 27 /* CommaToken */, copyExpr);
                             }
                         }
-                        returnExpression = ts.createBinary(expr, 27 /* CommaToken */, returnExpression);
+                        returnExpression = factory.createBinaryExpression(expr, 27 /* CommaToken */, returnExpression);
                     }
-                    return ts.createReturn(returnExpression);
+                    return factory.createReturnStatement(returnExpression);
                 }
             }
             return ts.visitEachChild(node, visitor, context);
@@ -83646,27 +88712,28 @@ var ts;
             //          }
             //          return C;
             //      }());
-            var variable = ts.createVariableDeclaration(ts.getLocalName(node, /*allowComments*/ true), 
+            var variable = factory.createVariableDeclaration(factory.getLocalName(node, /*allowComments*/ true), 
+            /*exclamationToken*/ undefined, 
             /*type*/ undefined, transformClassLikeDeclarationToExpression(node));
             ts.setOriginalNode(variable, node);
             var statements = [];
-            var statement = ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([variable]));
+            var statement = factory.createVariableStatement(/*modifiers*/ undefined, factory.createVariableDeclarationList([variable]));
             ts.setOriginalNode(statement, node);
             ts.setTextRange(statement, node);
             ts.startOnNewLine(statement);
             statements.push(statement);
             // Add an `export default` statement for default exports (for `--target es5 --module es6`)
-            if (ts.hasModifier(node, 1 /* Export */)) {
-                var exportStatement = ts.hasModifier(node, 512 /* Default */)
-                    ? ts.createExportDefault(ts.getLocalName(node))
-                    : ts.createExternalModuleExport(ts.getLocalName(node));
+            if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                var exportStatement = ts.hasSyntacticModifier(node, 512 /* Default */)
+                    ? factory.createExportDefault(factory.getLocalName(node))
+                    : factory.createExternalModuleExport(factory.getLocalName(node));
                 ts.setOriginalNode(exportStatement, statement);
                 statements.push(exportStatement);
             }
             var emitFlags = ts.getEmitFlags(node);
             if ((emitFlags & 4194304 /* HasEndOfDeclarationMarker */) === 0) {
                 // Add a DeclarationMarker as a marker for the end of the declaration
-                statements.push(ts.createEndOfDeclarationMarker(node));
+                statements.push(factory.createEndOfDeclarationMarker(node));
                 ts.setEmitFlags(statement, emitFlags | 4194304 /* HasEndOfDeclarationMarker */);
             }
             return ts.singleOrMany(statements);
@@ -83720,11 +88787,11 @@ var ts;
                 enableSubstitutionsForBlockScopedBindings();
             }
             var extendsClauseElement = ts.getClassExtendsHeritageElement(node);
-            var classFunction = ts.createFunctionExpression(
+            var classFunction = factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
-            /*typeParameters*/ undefined, extendsClauseElement ? [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, ts.createFileLevelUniqueName("_super"))] : [], 
+            /*typeParameters*/ undefined, extendsClauseElement ? [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */))] : [], 
             /*type*/ undefined, transformClassBody(node, extendsClauseElement));
             // To preserve the behavior of the old emitter, we explicitly indent
             // the body of the function here if it was requested in an earlier
@@ -83732,13 +88799,13 @@ var ts;
             ts.setEmitFlags(classFunction, (ts.getEmitFlags(node) & 65536 /* Indented */) | 524288 /* ReuseTempVariableScope */);
             // "inner" and "outer" below are added purely to preserve source map locations from
             // the old emitter
-            var inner = ts.createPartiallyEmittedExpression(classFunction);
-            inner.end = node.end;
+            var inner = factory.createPartiallyEmittedExpression(classFunction);
+            ts.setTextRangeEnd(inner, node.end);
             ts.setEmitFlags(inner, 1536 /* NoComments */);
-            var outer = ts.createPartiallyEmittedExpression(inner);
-            outer.end = ts.skipTrivia(currentText, node.pos);
+            var outer = factory.createPartiallyEmittedExpression(inner);
+            ts.setTextRangeEnd(outer, ts.skipTrivia(currentText, node.pos));
             ts.setEmitFlags(outer, 1536 /* NoComments */);
-            var result = ts.createParen(ts.createCall(outer, 
+            var result = factory.createParenthesizedExpression(factory.createCallExpression(outer, 
             /*typeArguments*/ undefined, extendsClauseElement
                 ? [ts.visitNode(extendsClauseElement.expression, visitor, ts.isExpression)]
                 : []));
@@ -83753,24 +88820,25 @@ var ts;
          */
         function transformClassBody(node, extendsClauseElement) {
             var statements = [];
+            var name = factory.getInternalName(node);
+            var constructorLikeName = ts.isIdentifierANonContextualKeyword(name) ? factory.getGeneratedNameForNode(name) : name;
             startLexicalEnvironment();
             addExtendsHelperIfNeeded(statements, node, extendsClauseElement);
-            addConstructor(statements, node, extendsClauseElement);
+            addConstructor(statements, node, constructorLikeName, extendsClauseElement);
             addClassMembers(statements, node);
             // Create a synthetic text range for the return statement.
             var closingBraceLocation = ts.createTokenRange(ts.skipTrivia(currentText, node.members.end), 19 /* CloseBraceToken */);
-            var localName = ts.getInternalName(node);
             // The following partially-emitted expression exists purely to align our sourcemap
             // emit with the original emitter.
-            var outer = ts.createPartiallyEmittedExpression(localName);
-            outer.end = closingBraceLocation.end;
+            var outer = factory.createPartiallyEmittedExpression(constructorLikeName);
+            ts.setTextRangeEnd(outer, closingBraceLocation.end);
             ts.setEmitFlags(outer, 1536 /* NoComments */);
-            var statement = ts.createReturn(outer);
-            statement.pos = closingBraceLocation.pos;
+            var statement = factory.createReturnStatement(outer);
+            ts.setTextRangePos(statement, closingBraceLocation.pos);
             ts.setEmitFlags(statement, 1536 /* NoComments */ | 384 /* NoTokenSourceMaps */);
             statements.push(statement);
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
-            var block = ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), /*location*/ node.members), /*multiLine*/ true);
+            var block = factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), /*location*/ node.members), /*multiLine*/ true);
             ts.setEmitFlags(block, 1536 /* NoComments */);
             return block;
         }
@@ -83783,7 +88851,7 @@ var ts;
          */
         function addExtendsHelperIfNeeded(statements, node, extendsClauseElement) {
             if (extendsClauseElement) {
-                statements.push(ts.setTextRange(ts.createExpressionStatement(createExtendsHelper(context, ts.getInternalName(node))), 
+                statements.push(ts.setTextRange(factory.createExpressionStatement(emitHelpers().createExtendsHelper(factory.getInternalName(node))), 
                 /*location*/ extendsClauseElement));
             }
         }
@@ -83794,16 +88862,16 @@ var ts;
          * @param node The ClassExpression or ClassDeclaration node.
          * @param extendsClauseElement The expression for the class `extends` clause.
          */
-        function addConstructor(statements, node, extendsClauseElement) {
+        function addConstructor(statements, node, name, extendsClauseElement) {
             var savedConvertedLoopState = convertedLoopState;
             convertedLoopState = undefined;
             var ancestorFacts = enterSubtree(16278 /* ConstructorExcludes */, 73 /* ConstructorIncludes */);
             var constructor = ts.getFirstConstructorWithBody(node);
             var hasSynthesizedSuper = hasSynthesizedDefaultSuperCall(constructor, extendsClauseElement !== undefined);
-            var constructorFunction = ts.createFunctionDeclaration(
+            var constructorFunction = factory.createFunctionDeclaration(
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, 
-            /*asteriskToken*/ undefined, ts.getInternalName(node)
+            /*asteriskToken*/ undefined, name
             /*typeParameters*/ undefined, transformConstructorParameters(constructor, hasSynthesizedSuper), 
             /*type*/ undefined, transformConstructorBody(constructor, node, extendsClauseElement, hasSynthesizedSuper));
             ts.setTextRange(constructorFunction, constructor || node);
@@ -83836,14 +88904,14 @@ var ts;
             // If that's the case we can just immediately return the result of a 'super()' call.
             var statements = [];
             resumeLexicalEnvironment();
-            ts.mergeLexicalEnvironment(statements, endLexicalEnvironment());
+            factory.mergeLexicalEnvironment(statements, endLexicalEnvironment());
             if (isDerivedClass) {
                 // return _super !== null && _super.apply(this, arguments) || this;
-                statements.push(ts.createReturn(createDefaultSuperCallOrThis()));
+                statements.push(factory.createReturnStatement(createDefaultSuperCallOrThis()));
             }
-            var statementsArray = ts.createNodeArray(statements);
+            var statementsArray = factory.createNodeArray(statements);
             ts.setTextRange(statementsArray, node.members);
-            var block = ts.createBlock(statementsArray, /*multiLine*/ true);
+            var block = factory.createBlock(statementsArray, /*multiLine*/ true);
             ts.setTextRange(block, node);
             ts.setEmitFlags(block, 1536 /* NoComments */);
             return block;
@@ -83860,7 +88928,7 @@ var ts;
         function transformConstructorBody(constructor, node, extendsClauseElement, hasSynthesizedSuper) {
             // determine whether the class is known syntactically to be a derived class (e.g. a
             // class that extends a value that is not syntactically known to be `null`).
-            var isDerivedClass = !!extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 100 /* NullKeyword */;
+            var isDerivedClass = !!extendsClauseElement && ts.skipOuterExpressions(extendsClauseElement.expression).kind !== 103 /* NullKeyword */;
             // When the subclass does not have a constructor, we synthesize a *default* constructor using the following
             // representation:
             //
@@ -83887,11 +88955,11 @@ var ts;
             // The assumption is that no prior step in the pipeline has added any prologue directives.
             var statementOffset = 0;
             if (!hasSynthesizedSuper)
-                statementOffset = ts.addStandardPrologue(prologue, constructor.body.statements, /*ensureUseStrict*/ false);
+                statementOffset = factory.copyStandardPrologue(constructor.body.statements, prologue, /*ensureUseStrict*/ false);
             addDefaultValueAssignmentsIfNeeded(statements, constructor);
             addRestParameterIfNeeded(statements, constructor, hasSynthesizedSuper);
             if (!hasSynthesizedSuper)
-                statementOffset = ts.addCustomPrologue(statements, constructor.body.statements, statementOffset, visitor);
+                statementOffset = factory.copyCustomPrologue(constructor.body.statements, statements, statementOffset, visitor);
             // If the first statement is a call to `super()`, visit the statement directly
             var superCallExpression;
             if (hasSynthesizedSuper) {
@@ -83909,7 +88977,7 @@ var ts;
             }
             // visit the remaining statements
             ts.addRange(statements, ts.visitNodes(constructor.body.statements, visitor, ts.isStatement, /*start*/ statementOffset));
-            ts.mergeLexicalEnvironment(prologue, endLexicalEnvironment());
+            factory.mergeLexicalEnvironment(prologue, endLexicalEnvironment());
             insertCaptureNewTargetIfNeeded(prologue, constructor, /*copyOnWrite*/ false);
             if (isDerivedClass) {
                 if (superCallExpression && statementOffset === constructor.body.statements.length && !(constructor.body.transformFlags & 4096 /* ContainsLexicalThis */)) {
@@ -83933,7 +89001,7 @@ var ts;
                     // })(Base);
                     // ```
                     var superCall = ts.cast(ts.cast(superCallExpression, ts.isBinaryExpression).left, ts.isCallExpression);
-                    var returnStatement = ts.createReturn(superCallExpression);
+                    var returnStatement = factory.createReturnStatement(superCallExpression);
                     ts.setCommentRange(returnStatement, ts.getCommentRange(superCall));
                     ts.setEmitFlags(superCall, 1536 /* NoComments */);
                     statements.push(returnStatement);
@@ -83964,7 +89032,7 @@ var ts;
                     // `super()` at the top of the list of `statements` (after any pre-existing custom prologues).
                     insertCaptureThisForNode(statements, constructor, superCallExpression || createActualThis());
                     if (!isSufficientlyCoveredByReturnStatements(constructor.body)) {
-                        statements.push(ts.createReturn(ts.createFileLevelUniqueName("_this")));
+                        statements.push(factory.createReturnStatement(factory.createUniqueName("_this", 16 /* Optimistic */ | 32 /* FileLevel */)));
                     }
                 }
             }
@@ -83985,7 +89053,7 @@ var ts;
                 // ```
                 insertCaptureThisForNodeIfNeeded(prologue, constructor);
             }
-            var block = ts.createBlock(ts.setTextRange(ts.createNodeArray(ts.concatenate(prologue, statements)), 
+            var block = factory.createBlock(ts.setTextRange(factory.createNodeArray(ts.concatenate(prologue, statements)), 
             /*location*/ constructor.body.statements), 
             /*multiLine*/ true);
             ts.setTextRange(block, constructor.body);
@@ -83998,11 +89066,11 @@ var ts;
          */
         function isSufficientlyCoveredByReturnStatements(statement) {
             // A return statement is considered covered.
-            if (statement.kind === 235 /* ReturnStatement */) {
+            if (statement.kind === 242 /* ReturnStatement */) {
                 return true;
             }
             // An if-statement with two covered branches is covered.
-            else if (statement.kind === 227 /* IfStatement */) {
+            else if (statement.kind === 234 /* IfStatement */) {
                 var ifStatement = statement;
                 if (ifStatement.elseStatement) {
                     return isSufficientlyCoveredByReturnStatements(ifStatement.thenStatement) &&
@@ -84010,7 +89078,7 @@ var ts;
                 }
             }
             // A block is covered if it has a last statement which is covered.
-            else if (statement.kind === 223 /* Block */) {
+            else if (statement.kind === 230 /* Block */) {
                 var lastStatement = ts.lastOrUndefined(statement.statements);
                 if (lastStatement && isSufficientlyCoveredByReturnStatements(lastStatement)) {
                     return true;
@@ -84019,10 +89087,10 @@ var ts;
             return false;
         }
         function createActualThis() {
-            return ts.setEmitFlags(ts.createThis(), 4 /* NoSubstitution */);
+            return ts.setEmitFlags(factory.createThis(), 4 /* NoSubstitution */);
         }
         function createDefaultSuperCallOrThis() {
-            return ts.createLogicalOr(ts.createLogicalAnd(ts.createStrictInequality(ts.createFileLevelUniqueName("_super"), ts.createNull()), ts.createFunctionApply(ts.createFileLevelUniqueName("_super"), createActualThis(), ts.createIdentifier("arguments"))), createActualThis());
+            return factory.createLogicalOr(factory.createLogicalAnd(factory.createStrictInequality(factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */), factory.createNull()), factory.createFunctionApplyCall(factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */), createActualThis(), factory.createIdentifier("arguments"))), createActualThis());
         }
         /**
          * Visits a parameter declaration.
@@ -84037,10 +89105,10 @@ var ts;
             else if (ts.isBindingPattern(node.name)) {
                 // Binding patterns are converted into a generated name and are
                 // evaluated inside the function body.
-                return ts.setOriginalNode(ts.setTextRange(ts.createParameter(
+                return ts.setOriginalNode(ts.setTextRange(factory.createParameterDeclaration(
                 /*decorators*/ undefined, 
                 /*modifiers*/ undefined, 
-                /*dotDotDotToken*/ undefined, ts.getGeneratedNameForNode(node), 
+                /*dotDotDotToken*/ undefined, factory.getGeneratedNameForNode(node), 
                 /*questionToken*/ undefined, 
                 /*type*/ undefined, 
                 /*initializer*/ undefined), 
@@ -84049,7 +89117,7 @@ var ts;
             }
             else if (node.initializer) {
                 // Initializers are elided
-                return ts.setOriginalNode(ts.setTextRange(ts.createParameter(
+                return ts.setOriginalNode(ts.setTextRange(factory.createParameterDeclaration(
                 /*decorators*/ undefined, 
                 /*modifiers*/ undefined, 
                 /*dotDotDotToken*/ undefined, node.name, 
@@ -84110,12 +89178,12 @@ var ts;
             // we usually don't want to emit a var declaration; however, in the presence
             // of an initializer, we must emit that expression to preserve side effects.
             if (name.elements.length > 0) {
-                ts.insertStatementAfterCustomPrologue(statements, ts.setEmitFlags(ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList(ts.flattenDestructuringBinding(parameter, visitor, context, 0 /* All */, ts.getGeneratedNameForNode(parameter)))), 1048576 /* CustomPrologue */));
+                ts.insertStatementAfterCustomPrologue(statements, ts.setEmitFlags(factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList(ts.flattenDestructuringBinding(parameter, visitor, context, 0 /* All */, factory.getGeneratedNameForNode(parameter)))), 1048576 /* CustomPrologue */));
                 return true;
             }
             else if (initializer) {
-                ts.insertStatementAfterCustomPrologue(statements, ts.setEmitFlags(ts.createExpressionStatement(ts.createAssignment(ts.getGeneratedNameForNode(parameter), ts.visitNode(initializer, visitor, ts.isExpression))), 1048576 /* CustomPrologue */));
+                ts.insertStatementAfterCustomPrologue(statements, ts.setEmitFlags(factory.createExpressionStatement(factory.createAssignment(factory.getGeneratedNameForNode(parameter), ts.visitNode(initializer, visitor, ts.isExpression))), 1048576 /* CustomPrologue */));
                 return true;
             }
             return false;
@@ -84130,8 +89198,10 @@ var ts;
          */
         function insertDefaultValueAssignmentForInitializer(statements, parameter, name, initializer) {
             initializer = ts.visitNode(initializer, visitor, ts.isExpression);
-            var statement = ts.createIf(ts.createTypeCheck(ts.getSynthesizedClone(name), "undefined"), ts.setEmitFlags(ts.setTextRange(ts.createBlock([
-                ts.createExpressionStatement(ts.setEmitFlags(ts.setTextRange(ts.createAssignment(ts.setEmitFlags(ts.getMutableClone(name), 48 /* NoSourceMap */), ts.setEmitFlags(initializer, 48 /* NoSourceMap */ | ts.getEmitFlags(initializer) | 1536 /* NoComments */)), parameter), 1536 /* NoComments */))
+            var statement = factory.createIfStatement(factory.createTypeCheck(factory.cloneNode(name), "undefined"), ts.setEmitFlags(ts.setTextRange(factory.createBlock([
+                factory.createExpressionStatement(ts.setEmitFlags(ts.setTextRange(factory.createAssignment(
+                // TODO(rbuckton): Does this need to be parented?
+                ts.setEmitFlags(ts.setParent(ts.setTextRange(factory.cloneNode(name), name), name.parent), 48 /* NoSourceMap */), ts.setEmitFlags(initializer, 48 /* NoSourceMap */ | ts.getEmitFlags(initializer) | 1536 /* NoComments */)), parameter), 1536 /* NoComments */))
             ]), parameter), 1 /* SingleLine */ | 32 /* NoTrailingSourceMap */ | 384 /* NoTokenSourceMaps */ | 1536 /* NoComments */));
             ts.startOnNewLine(statement);
             ts.setTextRange(statement, parameter);
@@ -84165,37 +89235,39 @@ var ts;
                 return false;
             }
             // `declarationName` is the name of the local declaration for the parameter.
-            var declarationName = parameter.name.kind === 75 /* Identifier */ ? ts.getMutableClone(parameter.name) : ts.createTempVariable(/*recordTempVariable*/ undefined);
+            // TODO(rbuckton): Does this need to be parented?
+            var declarationName = parameter.name.kind === 78 /* Identifier */ ? ts.setParent(ts.setTextRange(factory.cloneNode(parameter.name), parameter.name), parameter.name.parent) : factory.createTempVariable(/*recordTempVariable*/ undefined);
             ts.setEmitFlags(declarationName, 48 /* NoSourceMap */);
             // `expressionName` is the name of the parameter used in expressions.
-            var expressionName = parameter.name.kind === 75 /* Identifier */ ? ts.getSynthesizedClone(parameter.name) : declarationName;
+            var expressionName = parameter.name.kind === 78 /* Identifier */ ? factory.cloneNode(parameter.name) : declarationName;
             var restIndex = node.parameters.length - 1;
-            var temp = ts.createLoopVariable();
+            var temp = factory.createLoopVariable();
             // var param = [];
-            prologueStatements.push(ts.setEmitFlags(ts.setTextRange(ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(declarationName, 
-                /*type*/ undefined, ts.createArrayLiteral([]))
+            prologueStatements.push(ts.setEmitFlags(ts.setTextRange(factory.createVariableStatement(
+            /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(declarationName, 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, factory.createArrayLiteralExpression([]))
             ])), 
             /*location*/ parameter), 1048576 /* CustomPrologue */));
             // for (var _i = restIndex; _i < arguments.length; _i++) {
             //   param[_i - restIndex] = arguments[_i];
             // }
-            var forStatement = ts.createFor(ts.setTextRange(ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(temp, /*type*/ undefined, ts.createLiteral(restIndex))
-            ]), parameter), ts.setTextRange(ts.createLessThan(temp, ts.createPropertyAccess(ts.createIdentifier("arguments"), "length")), parameter), ts.setTextRange(ts.createPostfixIncrement(temp), parameter), ts.createBlock([
-                ts.startOnNewLine(ts.setTextRange(ts.createExpressionStatement(ts.createAssignment(ts.createElementAccess(expressionName, restIndex === 0
+            var forStatement = factory.createForStatement(ts.setTextRange(factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(temp, /*exclamationToken*/ undefined, /*type*/ undefined, factory.createNumericLiteral(restIndex))
+            ]), parameter), ts.setTextRange(factory.createLessThan(temp, factory.createPropertyAccessExpression(factory.createIdentifier("arguments"), "length")), parameter), ts.setTextRange(factory.createPostfixIncrement(temp), parameter), factory.createBlock([
+                ts.startOnNewLine(ts.setTextRange(factory.createExpressionStatement(factory.createAssignment(factory.createElementAccessExpression(expressionName, restIndex === 0
                     ? temp
-                    : ts.createSubtract(temp, ts.createLiteral(restIndex))), ts.createElementAccess(ts.createIdentifier("arguments"), temp))), 
+                    : factory.createSubtract(temp, factory.createNumericLiteral(restIndex))), factory.createElementAccessExpression(factory.createIdentifier("arguments"), temp))), 
                 /*location*/ parameter))
             ]));
             ts.setEmitFlags(forStatement, 1048576 /* CustomPrologue */);
             ts.startOnNewLine(forStatement);
             prologueStatements.push(forStatement);
-            if (parameter.name.kind !== 75 /* Identifier */) {
+            if (parameter.name.kind !== 78 /* Identifier */) {
                 // do the actual destructuring of the rest parameter if necessary
-                prologueStatements.push(ts.setEmitFlags(ts.setTextRange(ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList(ts.flattenDestructuringBinding(parameter, visitor, context, 0 /* All */, expressionName))), parameter), 1048576 /* CustomPrologue */));
+                prologueStatements.push(ts.setEmitFlags(ts.setTextRange(factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList(ts.flattenDestructuringBinding(parameter, visitor, context, 0 /* All */, expressionName))), parameter), 1048576 /* CustomPrologue */));
             }
             ts.insertStatementsAfterCustomPrologue(statements, prologueStatements);
             return true;
@@ -84208,17 +89280,18 @@ var ts;
          * @param node A node.
          */
         function insertCaptureThisForNodeIfNeeded(statements, node) {
-            if (hierarchyFacts & 32768 /* CapturedLexicalThis */ && node.kind !== 202 /* ArrowFunction */) {
-                insertCaptureThisForNode(statements, node, ts.createThis());
+            if (hierarchyFacts & 32768 /* CapturedLexicalThis */ && node.kind !== 209 /* ArrowFunction */) {
+                insertCaptureThisForNode(statements, node, factory.createThis());
                 return true;
             }
             return false;
         }
         function insertCaptureThisForNode(statements, node, initializer) {
             enableSubstitutionsForCapturedThis();
-            var captureThisStatement = ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(ts.createFileLevelUniqueName("_this"), 
+            var captureThisStatement = factory.createVariableStatement(
+            /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(factory.createUniqueName("_this", 16 /* Optimistic */ | 32 /* FileLevel */), 
+                /*exclamationToken*/ undefined, 
                 /*type*/ undefined, initializer)
             ]));
             ts.setEmitFlags(captureThisStatement, 1536 /* NoComments */ | 1048576 /* CustomPrologue */);
@@ -84229,32 +89302,35 @@ var ts;
             if (hierarchyFacts & 16384 /* NewTarget */) {
                 var newTarget = void 0;
                 switch (node.kind) {
-                    case 202 /* ArrowFunction */:
+                    case 209 /* ArrowFunction */:
                         return statements;
-                    case 161 /* MethodDeclaration */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
+                    case 165 /* MethodDeclaration */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
                         // Methods and accessors cannot be constructors, so 'new.target' will
                         // always return 'undefined'.
-                        newTarget = ts.createVoidZero();
+                        newTarget = factory.createVoidZero();
                         break;
-                    case 162 /* Constructor */:
+                    case 166 /* Constructor */:
                         // Class constructors can only be called with `new`, so `this.constructor`
                         // should be relatively safe to use.
-                        newTarget = ts.createPropertyAccess(ts.setEmitFlags(ts.createThis(), 4 /* NoSubstitution */), "constructor");
+                        newTarget = factory.createPropertyAccessExpression(ts.setEmitFlags(factory.createThis(), 4 /* NoSubstitution */), "constructor");
                         break;
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
                         // Functions can be called or constructed, and may have a `this` due to
                         // being a member or when calling an imported function via `other_1.f()`.
-                        newTarget = ts.createConditional(ts.createLogicalAnd(ts.setEmitFlags(ts.createThis(), 4 /* NoSubstitution */), ts.createBinary(ts.setEmitFlags(ts.createThis(), 4 /* NoSubstitution */), 98 /* InstanceOfKeyword */, ts.getLocalName(node))), ts.createPropertyAccess(ts.setEmitFlags(ts.createThis(), 4 /* NoSubstitution */), "constructor"), ts.createVoidZero());
+                        newTarget = factory.createConditionalExpression(factory.createLogicalAnd(ts.setEmitFlags(factory.createThis(), 4 /* NoSubstitution */), factory.createBinaryExpression(ts.setEmitFlags(factory.createThis(), 4 /* NoSubstitution */), 101 /* InstanceOfKeyword */, factory.getLocalName(node))), 
+                        /*questionToken*/ undefined, factory.createPropertyAccessExpression(ts.setEmitFlags(factory.createThis(), 4 /* NoSubstitution */), "constructor"), 
+                        /*colonToken*/ undefined, factory.createVoidZero());
                         break;
                     default:
                         return ts.Debug.failBadSyntaxKind(node);
                 }
-                var captureNewTargetStatement = ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                    ts.createVariableDeclaration(ts.createFileLevelUniqueName("_newTarget"), 
+                var captureNewTargetStatement = factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                    factory.createVariableDeclaration(factory.createUniqueName("_newTarget", 16 /* Optimistic */ | 32 /* FileLevel */), 
+                    /*exclamationToken*/ undefined, 
                     /*type*/ undefined, newTarget)
                 ]));
                 ts.setEmitFlags(captureNewTargetStatement, 1536 /* NoComments */ | 1048576 /* CustomPrologue */);
@@ -84276,20 +89352,20 @@ var ts;
             for (var _i = 0, _a = node.members; _i < _a.length; _i++) {
                 var member = _a[_i];
                 switch (member.kind) {
-                    case 222 /* SemicolonClassElement */:
+                    case 229 /* SemicolonClassElement */:
                         statements.push(transformSemicolonClassElementToStatement(member));
                         break;
-                    case 161 /* MethodDeclaration */:
+                    case 165 /* MethodDeclaration */:
                         statements.push(transformClassMethodDeclarationToStatement(getClassMemberPrefix(node, member), member, node));
                         break;
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
                         var accessors = ts.getAllAccessorDeclarations(node.members, member);
                         if (member === accessors.firstAccessor) {
                             statements.push(transformAccessorsToStatement(getClassMemberPrefix(node, member), accessors, node));
                         }
                         break;
-                    case 162 /* Constructor */:
+                    case 166 /* Constructor */:
                         // Constructors are handled in visitClassExpression/visitClassDeclaration
                         break;
                     default:
@@ -84304,7 +89380,7 @@ var ts;
          * @param member The SemicolonClassElement node.
          */
         function transformSemicolonClassElementToStatement(member) {
-            return ts.setTextRange(ts.createEmptyStatement(), member);
+            return ts.setTextRange(factory.createEmptyStatement(), member);
         }
         /**
          * Transforms a MethodDeclaration into a statement for a class body function.
@@ -84320,17 +89396,17 @@ var ts;
             var e;
             if (!ts.isPrivateIdentifier(propertyName) && context.getCompilerOptions().useDefineForClassFields) {
                 var name = ts.isComputedPropertyName(propertyName) ? propertyName.expression
-                    : ts.isIdentifier(propertyName) ? ts.createStringLiteral(ts.unescapeLeadingUnderscores(propertyName.escapedText))
+                    : ts.isIdentifier(propertyName) ? factory.createStringLiteral(ts.unescapeLeadingUnderscores(propertyName.escapedText))
                         : propertyName;
-                e = ts.createObjectDefinePropertyCall(receiver, name, ts.createPropertyDescriptor({ value: memberFunction, enumerable: false, writable: true, configurable: true }));
+                e = factory.createObjectDefinePropertyCall(receiver, name, factory.createPropertyDescriptor({ value: memberFunction, enumerable: false, writable: true, configurable: true }));
             }
             else {
-                var memberName = ts.createMemberAccessForPropertyName(receiver, propertyName, /*location*/ member.name);
-                e = ts.createAssignment(memberName, memberFunction);
+                var memberName = ts.createMemberAccessForPropertyName(factory, receiver, propertyName, /*location*/ member.name);
+                e = factory.createAssignment(memberName, memberFunction);
             }
             ts.setEmitFlags(memberFunction, 1536 /* NoComments */);
             ts.setSourceMapRange(memberFunction, sourceMapRange);
-            var statement = ts.setTextRange(ts.createExpressionStatement(e), /*location*/ member);
+            var statement = ts.setTextRange(factory.createExpressionStatement(e), /*location*/ member);
             ts.setOriginalNode(statement, member);
             ts.setCommentRange(statement, commentRange);
             // The location for the statement is used to emit comments only.
@@ -84346,7 +89422,7 @@ var ts;
          * @param accessors The set of related get/set accessors.
          */
         function transformAccessorsToStatement(receiver, accessors, container) {
-            var statement = ts.createExpressionStatement(transformAccessorsToExpression(receiver, accessors, container, /*startsOnNewLine*/ false));
+            var statement = factory.createExpressionStatement(transformAccessorsToExpression(receiver, accessors, container, /*startsOnNewLine*/ false));
             // The location for the statement is used to emit source maps only.
             // No comments should be emitted for this statement to align with the
             // old emitter.
@@ -84364,14 +89440,15 @@ var ts;
             var firstAccessor = _a.firstAccessor, getAccessor = _a.getAccessor, setAccessor = _a.setAccessor;
             // To align with source maps in the old emitter, the receiver and property name
             // arguments are both mapped contiguously to the accessor name.
-            var target = ts.getMutableClone(receiver);
+            // TODO(rbuckton): Does this need to be parented?
+            var target = ts.setParent(ts.setTextRange(factory.cloneNode(receiver), receiver), receiver.parent);
             ts.setEmitFlags(target, 1536 /* NoComments */ | 32 /* NoTrailingSourceMap */);
             ts.setSourceMapRange(target, firstAccessor.name);
             var visitedAccessorName = ts.visitNode(firstAccessor.name, visitor, ts.isPropertyName);
             if (ts.isPrivateIdentifier(visitedAccessorName)) {
                 return ts.Debug.failBadSyntaxKind(visitedAccessorName, "Encountered unhandled private identifier while transforming ES2015.");
             }
-            var propertyName = ts.createExpressionForPropertyName(visitedAccessorName);
+            var propertyName = ts.createExpressionForPropertyName(factory, visitedAccessorName);
             ts.setEmitFlags(propertyName, 1536 /* NoComments */ | 16 /* NoLeadingSourceMap */);
             ts.setSourceMapRange(propertyName, firstAccessor.name);
             var properties = [];
@@ -84379,7 +89456,7 @@ var ts;
                 var getterFunction = transformFunctionLikeToExpression(getAccessor, /*location*/ undefined, /*name*/ undefined, container);
                 ts.setSourceMapRange(getterFunction, ts.getSourceMapRange(getAccessor));
                 ts.setEmitFlags(getterFunction, 512 /* NoLeadingComments */);
-                var getter = ts.createPropertyAssignment("get", getterFunction);
+                var getter = factory.createPropertyAssignment("get", getterFunction);
                 ts.setCommentRange(getter, ts.getCommentRange(getAccessor));
                 properties.push(getter);
             }
@@ -84387,16 +89464,16 @@ var ts;
                 var setterFunction = transformFunctionLikeToExpression(setAccessor, /*location*/ undefined, /*name*/ undefined, container);
                 ts.setSourceMapRange(setterFunction, ts.getSourceMapRange(setAccessor));
                 ts.setEmitFlags(setterFunction, 512 /* NoLeadingComments */);
-                var setter = ts.createPropertyAssignment("set", setterFunction);
+                var setter = factory.createPropertyAssignment("set", setterFunction);
                 ts.setCommentRange(setter, ts.getCommentRange(setAccessor));
                 properties.push(setter);
             }
-            properties.push(ts.createPropertyAssignment("enumerable", getAccessor || setAccessor ? ts.createFalse() : ts.createTrue()), ts.createPropertyAssignment("configurable", ts.createTrue()));
-            var call = ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Object"), "defineProperty"), 
+            properties.push(factory.createPropertyAssignment("enumerable", getAccessor || setAccessor ? factory.createFalse() : factory.createTrue()), factory.createPropertyAssignment("configurable", factory.createTrue()));
+            var call = factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("Object"), "defineProperty"), 
             /*typeArguments*/ undefined, [
                 target,
                 propertyName,
-                ts.createObjectLiteral(properties, /*multiLine*/ true)
+                factory.createObjectLiteralExpression(properties, /*multiLine*/ true)
             ]);
             if (startsOnNewLine) {
                 ts.startOnNewLine(call);
@@ -84415,7 +89492,7 @@ var ts;
             var savedConvertedLoopState = convertedLoopState;
             convertedLoopState = undefined;
             var ancestorFacts = enterSubtree(15232 /* ArrowFunctionExcludes */, 66 /* ArrowFunctionIncludes */);
-            var func = ts.createFunctionExpression(
+            var func = factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
@@ -84446,11 +89523,11 @@ var ts;
             var parameters = ts.visitParameterList(node.parameters, visitor, context);
             var body = transformFunctionBody(node);
             var name = hierarchyFacts & 16384 /* NewTarget */
-                ? ts.getLocalName(node)
+                ? factory.getLocalName(node)
                 : node.name;
             exitSubtree(ancestorFacts, 49152 /* FunctionSubtreeExcludes */, 0 /* None */);
             convertedLoopState = savedConvertedLoopState;
-            return ts.updateFunctionExpression(node, 
+            return factory.updateFunctionExpression(node, 
             /*modifiers*/ undefined, node.asteriskToken, name, 
             /*typeParameters*/ undefined, parameters, 
             /*type*/ undefined, body);
@@ -84467,11 +89544,11 @@ var ts;
             var parameters = ts.visitParameterList(node.parameters, visitor, context);
             var body = transformFunctionBody(node);
             var name = hierarchyFacts & 16384 /* NewTarget */
-                ? ts.getLocalName(node)
+                ? factory.getLocalName(node)
                 : node.name;
             exitSubtree(ancestorFacts, 49152 /* FunctionSubtreeExcludes */, 0 /* None */);
             convertedLoopState = savedConvertedLoopState;
-            return ts.updateFunctionDeclaration(node, 
+            return factory.updateFunctionDeclaration(node, 
             /*decorators*/ undefined, ts.visitNodes(node.modifiers, visitor, ts.isModifier), node.asteriskToken, name, 
             /*typeParameters*/ undefined, parameters, 
             /*type*/ undefined, body);
@@ -84486,17 +89563,17 @@ var ts;
         function transformFunctionLikeToExpression(node, location, name, container) {
             var savedConvertedLoopState = convertedLoopState;
             convertedLoopState = undefined;
-            var ancestorFacts = container && ts.isClassLike(container) && !ts.hasModifier(node, 32 /* Static */)
+            var ancestorFacts = container && ts.isClassLike(container) && !ts.hasSyntacticModifier(node, 32 /* Static */)
                 ? enterSubtree(16286 /* FunctionExcludes */, 65 /* FunctionIncludes */ | 8 /* NonStaticClassElement */)
                 : enterSubtree(16286 /* FunctionExcludes */, 65 /* FunctionIncludes */);
             var parameters = ts.visitParameterList(node.parameters, visitor, context);
             var body = transformFunctionBody(node);
-            if (hierarchyFacts & 16384 /* NewTarget */ && !name && (node.kind === 244 /* FunctionDeclaration */ || node.kind === 201 /* FunctionExpression */)) {
-                name = ts.getGeneratedNameForNode(node);
+            if (hierarchyFacts & 16384 /* NewTarget */ && !name && (node.kind === 251 /* FunctionDeclaration */ || node.kind === 208 /* FunctionExpression */)) {
+                name = factory.getGeneratedNameForNode(node);
             }
             exitSubtree(ancestorFacts, 49152 /* FunctionSubtreeExcludes */, 0 /* None */);
             convertedLoopState = savedConvertedLoopState;
-            return ts.setOriginalNode(ts.setTextRange(ts.createFunctionExpression(
+            return ts.setOriginalNode(ts.setTextRange(factory.createFunctionExpression(
             /*modifiers*/ undefined, node.asteriskToken, name, 
             /*typeParameters*/ undefined, parameters, 
             /*type*/ undefined, body), location), 
@@ -84520,15 +89597,15 @@ var ts;
             if (ts.isBlock(body)) {
                 // ensureUseStrict is false because no new prologue-directive should be added.
                 // addStandardPrologue will put already-existing directives at the beginning of the target statement-array
-                statementOffset = ts.addStandardPrologue(prologue, body.statements, /*ensureUseStrict*/ false);
-                statementOffset = ts.addCustomPrologue(statements, body.statements, statementOffset, visitor, ts.isHoistedFunction);
-                statementOffset = ts.addCustomPrologue(statements, body.statements, statementOffset, visitor, ts.isHoistedVariableStatement);
+                statementOffset = factory.copyStandardPrologue(body.statements, prologue, /*ensureUseStrict*/ false);
+                statementOffset = factory.copyCustomPrologue(body.statements, statements, statementOffset, visitor, ts.isHoistedFunction);
+                statementOffset = factory.copyCustomPrologue(body.statements, statements, statementOffset, visitor, ts.isHoistedVariableStatement);
             }
             multiLine = addDefaultValueAssignmentsIfNeeded(statements, node) || multiLine;
             multiLine = addRestParameterIfNeeded(statements, node, /*inConstructorWithSynthesizedSuper*/ false) || multiLine;
             if (ts.isBlock(body)) {
                 // addCustomPrologue puts already-existing directives at the beginning of the target statement-array
-                statementOffset = ts.addCustomPrologue(statements, body.statements, statementOffset, visitor);
+                statementOffset = factory.copyCustomPrologue(body.statements, statements, statementOffset, visitor);
                 statementsLocation = body.statements;
                 ts.addRange(statements, ts.visitNodes(body.statements, visitor, ts.isStatement, statementOffset));
                 // If the original body was a multi-line block, this must be a multi-line block.
@@ -84537,7 +89614,7 @@ var ts;
                 }
             }
             else {
-                ts.Debug.assert(node.kind === 202 /* ArrowFunction */);
+                ts.Debug.assert(node.kind === 209 /* ArrowFunction */);
                 // To align with the old emitter, we use a synthetic end position on the location
                 // for the statement list we synthesize when we down-level an arrow function with
                 // an expression function body. This prevents both comments and source maps from
@@ -84553,7 +89630,7 @@ var ts;
                     }
                 }
                 var expression = ts.visitNode(body, visitor, ts.isExpression);
-                var returnStatement = ts.createReturn(expression);
+                var returnStatement = factory.createReturnStatement(expression);
                 ts.setTextRange(returnStatement, body);
                 ts.moveSyntheticComments(returnStatement, body);
                 ts.setEmitFlags(returnStatement, 384 /* NoTokenSourceMaps */ | 32 /* NoTrailingSourceMap */ | 1024 /* NoTrailingComments */);
@@ -84562,7 +89639,7 @@ var ts;
                 // source map location for the close brace.
                 closeBraceLocation = body;
             }
-            ts.mergeLexicalEnvironment(prologue, endLexicalEnvironment());
+            factory.mergeLexicalEnvironment(prologue, endLexicalEnvironment());
             insertCaptureNewTargetIfNeeded(prologue, node, /*copyOnWrite*/ false);
             insertCaptureThisForNodeIfNeeded(prologue, node);
             // If we added any final generated statements, this must be a multi-line block
@@ -84574,7 +89651,7 @@ var ts;
                 // no changes were made, preserve the tree
                 return body;
             }
-            var block = ts.createBlock(ts.setTextRange(ts.createNodeArray(statements), statementsLocation), multiLine);
+            var block = factory.createBlock(ts.setTextRange(factory.createNodeArray(statements), statementsLocation), multiLine);
             ts.setTextRange(block, node.body);
             if (!multiLine && singleLine) {
                 ts.setEmitFlags(block, 1 /* SingleLine */);
@@ -84603,58 +89680,62 @@ var ts;
          * @param node An ExpressionStatement node.
          */
         function visitExpressionStatement(node) {
-            // If we are here it is most likely because our expression is a destructuring assignment.
-            switch (node.expression.kind) {
-                case 200 /* ParenthesizedExpression */:
-                    return ts.updateExpressionStatement(node, visitParenthesizedExpression(node.expression, /*needsDestructuringValue*/ false));
-                case 209 /* BinaryExpression */:
-                    return ts.updateExpressionStatement(node, visitBinaryExpression(node.expression, /*needsDestructuringValue*/ false));
-            }
-            return ts.visitEachChild(node, visitor, context);
+            return ts.visitEachChild(node, visitorWithUnusedExpressionResult, context);
         }
         /**
          * Visits a ParenthesizedExpression that may contain a destructuring assignment.
          *
          * @param node A ParenthesizedExpression node.
-         * @param needsDestructuringValue A value indicating whether we need to hold onto the rhs
-         *                                of a destructuring assignment.
-         */
-        function visitParenthesizedExpression(node, needsDestructuringValue) {
-            // If we are here it is most likely because our expression is a destructuring assignment.
-            if (!needsDestructuringValue) {
-                // By default we always emit the RHS at the end of a flattened destructuring
-                // expression. If we are in a state where we do not need the destructuring value,
-                // we pass that information along to the children that care about it.
-                switch (node.expression.kind) {
-                    case 200 /* ParenthesizedExpression */:
-                        return ts.updateParen(node, visitParenthesizedExpression(node.expression, /*needsDestructuringValue*/ false));
-                    case 209 /* BinaryExpression */:
-                        return ts.updateParen(node, visitBinaryExpression(node.expression, /*needsDestructuringValue*/ false));
-                }
-            }
-            return ts.visitEachChild(node, visitor, context);
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
+         */
+        function visitParenthesizedExpression(node, expressionResultIsUnused) {
+            return ts.visitEachChild(node, expressionResultIsUnused ? visitorWithUnusedExpressionResult : visitor, context);
         }
         /**
          * Visits a BinaryExpression that contains a destructuring assignment.
          *
          * @param node A BinaryExpression node.
-         * @param needsDestructuringValue A value indicating whether we need to hold onto the rhs
-         *                                of a destructuring assignment.
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
          */
-        function visitBinaryExpression(node, needsDestructuringValue) {
+        function visitBinaryExpression(node, expressionResultIsUnused) {
             // If we are here it is because this is a destructuring assignment.
             if (ts.isDestructuringAssignment(node)) {
-                return ts.flattenDestructuringAssignment(node, visitor, context, 0 /* All */, needsDestructuringValue);
+                return ts.flattenDestructuringAssignment(node, visitor, context, 0 /* All */, !expressionResultIsUnused);
+            }
+            if (node.operatorToken.kind === 27 /* CommaToken */) {
+                return factory.updateBinaryExpression(node, ts.visitNode(node.left, visitorWithUnusedExpressionResult, ts.isExpression), node.operatorToken, ts.visitNode(node.right, expressionResultIsUnused ? visitorWithUnusedExpressionResult : visitor, ts.isExpression));
             }
             return ts.visitEachChild(node, visitor, context);
         }
+        /**
+         * @param expressionResultIsUnused Indicates the result of an expression is unused by the parent node (i.e., the left side of a comma or the
+         * expression of an `ExpressionStatement`).
+         */
+        function visitCommaListExpression(node, expressionResultIsUnused) {
+            if (expressionResultIsUnused) {
+                return ts.visitEachChild(node, visitorWithUnusedExpressionResult, context);
+            }
+            var result;
+            for (var i = 0; i < node.elements.length; i++) {
+                var element = node.elements[i];
+                var visited = ts.visitNode(element, i < node.elements.length - 1 ? visitorWithUnusedExpressionResult : visitor, ts.isExpression);
+                if (result || visited !== element) {
+                    result || (result = node.elements.slice(0, i));
+                    result.push(visited);
+                }
+            }
+            var elements = result ? ts.setTextRange(factory.createNodeArray(result), node.elements) : node.elements;
+            return factory.updateCommaListExpression(node, elements);
+        }
         function isVariableStatementOfTypeScriptClassWrapper(node) {
             return node.declarationList.declarations.length === 1
                 && !!node.declarationList.declarations[0].initializer
                 && !!(ts.getEmitFlags(node.declarationList.declarations[0].initializer) & 33554432 /* TypeScriptClassWrapper */);
         }
         function visitVariableStatement(node) {
-            var ancestorFacts = enterSubtree(0 /* None */, ts.hasModifier(node, 1 /* Export */) ? 32 /* ExportedVariableStatement */ : 0 /* None */);
+            var ancestorFacts = enterSubtree(0 /* None */, ts.hasSyntacticModifier(node, 1 /* Export */) ? 32 /* ExportedVariableStatement */ : 0 /* None */);
             var updated;
             if (convertedLoopState && (node.declarationList.flags & 3 /* BlockScoped */) === 0 && !isVariableStatementOfTypeScriptClassWrapper(node)) {
                 // we are inside a converted loop - hoist variable declarations
@@ -84668,14 +89749,14 @@ var ts;
                             assignment = ts.flattenDestructuringAssignment(decl, visitor, context, 0 /* All */);
                         }
                         else {
-                            assignment = ts.createBinary(decl.name, 62 /* EqualsToken */, ts.visitNode(decl.initializer, visitor, ts.isExpression));
+                            assignment = factory.createBinaryExpression(decl.name, 62 /* EqualsToken */, ts.visitNode(decl.initializer, visitor, ts.isExpression));
                             ts.setTextRange(assignment, decl);
                         }
                         assignments = ts.append(assignments, assignment);
                     }
                 }
                 if (assignments) {
-                    updated = ts.setTextRange(ts.createExpressionStatement(ts.inlineExpressions(assignments)), node);
+                    updated = ts.setTextRange(factory.createExpressionStatement(factory.inlineExpressions(assignments)), node);
                 }
                 else {
                     // none of declarations has initializer - the entire variable statement can be deleted
@@ -84701,7 +89782,7 @@ var ts;
                 var declarations = ts.flatMap(node.declarations, node.flags & 1 /* Let */
                     ? visitVariableDeclarationInLetDeclarationList
                     : visitVariableDeclaration);
-                var declarationList = ts.createVariableDeclarationList(declarations);
+                var declarationList = factory.createVariableDeclarationList(declarations);
                 ts.setOriginalNode(declarationList, node);
                 ts.setTextRange(declarationList, node);
                 ts.setCommentRange(declarationList, node);
@@ -84801,9 +89882,7 @@ var ts;
                 return visitVariableDeclaration(node);
             }
             if (!node.initializer && shouldEmitExplicitInitializerForLetDeclaration(node)) {
-                var clone_3 = ts.getMutableClone(node);
-                clone_3.initializer = ts.createVoidZero();
-                return clone_3;
+                return factory.updateVariableDeclaration(node, node.name, /*exclamationToken*/ undefined, /*type*/ undefined, factory.createVoidZero());
             }
             return ts.visitEachChild(node, visitor, context);
         }
@@ -84833,23 +89912,23 @@ var ts;
         }
         function visitLabeledStatement(node) {
             if (convertedLoopState && !convertedLoopState.labels) {
-                convertedLoopState.labels = ts.createMap();
+                convertedLoopState.labels = new ts.Map();
             }
             var statement = ts.unwrapInnermostStatementOfLabel(node, convertedLoopState && recordLabel);
             return ts.isIterationStatement(statement, /*lookInLabeledStatements*/ false)
                 ? visitIterationStatement(statement, /*outermostLabeledStatement*/ node)
-                : ts.restoreEnclosingLabel(ts.visitNode(statement, visitor, ts.isStatement, ts.liftToBlock), node, convertedLoopState && resetLabel);
+                : factory.restoreEnclosingLabel(ts.visitNode(statement, visitor, ts.isStatement, factory.liftToBlock), node, convertedLoopState && resetLabel);
         }
         function visitIterationStatement(node, outermostLabeledStatement) {
             switch (node.kind) {
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
                     return visitDoOrWhileStatement(node, outermostLabeledStatement);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return visitForStatement(node, outermostLabeledStatement);
-                case 231 /* ForInStatement */:
+                case 238 /* ForInStatement */:
                     return visitForInStatement(node, outermostLabeledStatement);
-                case 232 /* ForOfStatement */:
+                case 239 /* ForOfStatement */:
                     return visitForOfStatement(node, outermostLabeledStatement);
             }
         }
@@ -84865,6 +89944,9 @@ var ts;
         function visitForStatement(node, outermostLabeledStatement) {
             return visitIterationStatementWithFacts(5056 /* ForStatementExcludes */, 3328 /* ForStatementIncludes */, node, outermostLabeledStatement);
         }
+        function visitEachChildOfForStatement(node) {
+            return factory.updateForStatement(node, ts.visitNode(node.initializer, visitorWithUnusedExpressionResult, ts.isForInitializer), ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitorWithUnusedExpressionResult, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+        }
         function visitForInStatement(node, outermostLabeledStatement) {
             return visitIterationStatementWithFacts(3008 /* ForInOrForOfStatementExcludes */, 5376 /* ForInOrForOfStatementIncludes */, node, outermostLabeledStatement);
         }
@@ -84883,20 +89965,21 @@ var ts;
                     // This works whether the declaration is a var, let, or const.
                     // It will use rhsIterationValue _a[_i] as the initializer.
                     var declarations = ts.flattenDestructuringBinding(firstOriginalDeclaration, visitor, context, 0 /* All */, boundValue);
-                    var declarationList = ts.setTextRange(ts.createVariableDeclarationList(declarations), node.initializer);
+                    var declarationList = ts.setTextRange(factory.createVariableDeclarationList(declarations), node.initializer);
                     ts.setOriginalNode(declarationList, node.initializer);
                     // Adjust the source map range for the first declaration to align with the old
                     // emitter.
                     ts.setSourceMapRange(declarationList, ts.createRange(declarations[0].pos, ts.last(declarations).end));
-                    statements.push(ts.createVariableStatement(
+                    statements.push(factory.createVariableStatement(
                     /*modifiers*/ undefined, declarationList));
                 }
                 else {
                     // The following call does not include the initializer, so we have
                     // to emit it separately.
-                    statements.push(ts.setTextRange(ts.createVariableStatement(
-                    /*modifiers*/ undefined, ts.setOriginalNode(ts.setTextRange(ts.createVariableDeclarationList([
-                        ts.createVariableDeclaration(firstOriginalDeclaration ? firstOriginalDeclaration.name : ts.createTempVariable(/*recordTempVariable*/ undefined), 
+                    statements.push(ts.setTextRange(factory.createVariableStatement(
+                    /*modifiers*/ undefined, ts.setOriginalNode(ts.setTextRange(factory.createVariableDeclarationList([
+                        factory.createVariableDeclaration(firstOriginalDeclaration ? firstOriginalDeclaration.name : factory.createTempVariable(/*recordTempVariable*/ undefined), 
+                        /*exclamationToken*/ undefined, 
                         /*type*/ undefined, boundValue)
                     ]), ts.moveRangePos(initializer, -1)), initializer)), ts.moveRangeEnd(initializer, -1)));
                 }
@@ -84904,23 +89987,22 @@ var ts;
             else {
                 // Initializer is an expression. Emit the expression in the body, so that it's
                 // evaluated on every iteration.
-                var assignment = ts.createAssignment(initializer, boundValue);
+                var assignment = factory.createAssignment(initializer, boundValue);
                 if (ts.isDestructuringAssignment(assignment)) {
-                    ts.aggregateTransformFlags(assignment);
-                    statements.push(ts.createExpressionStatement(visitBinaryExpression(assignment, /*needsDestructuringValue*/ false)));
+                    statements.push(factory.createExpressionStatement(visitBinaryExpression(assignment, /*expressionResultIsUnused*/ true)));
                 }
                 else {
-                    assignment.end = initializer.end;
-                    statements.push(ts.setTextRange(ts.createExpressionStatement(ts.visitNode(assignment, visitor, ts.isExpression)), ts.moveRangeEnd(initializer, -1)));
+                    ts.setTextRangeEnd(assignment, initializer.end);
+                    statements.push(ts.setTextRange(factory.createExpressionStatement(ts.visitNode(assignment, visitor, ts.isExpression)), ts.moveRangeEnd(initializer, -1)));
                 }
             }
             if (convertedLoopBodyStatements) {
                 return createSyntheticBlockForConvertedStatements(ts.addRange(statements, convertedLoopBodyStatements));
             }
             else {
-                var statement = ts.visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock);
+                var statement = ts.visitNode(node.statement, visitor, ts.isStatement, factory.liftToBlock);
                 if (ts.isBlock(statement)) {
-                    return ts.updateBlock(statement, ts.setTextRange(ts.createNodeArray(ts.concatenate(statements, statement.statements)), statement.statements));
+                    return factory.updateBlock(statement, ts.setTextRange(factory.createNodeArray(ts.concatenate(statements, statement.statements)), statement.statements));
                 }
                 else {
                     statements.push(statement);
@@ -84929,7 +90011,7 @@ var ts;
             }
         }
         function createSyntheticBlockForConvertedStatements(statements) {
-            return ts.setEmitFlags(ts.createBlock(ts.createNodeArray(statements), 
+            return ts.setEmitFlags(factory.createBlock(factory.createNodeArray(statements), 
             /*multiLine*/ true), 48 /* NoSourceMap */ | 384 /* NoTokenSourceMaps */);
         }
         function convertForOfStatementForArray(node, outermostLabeledStatement, convertedLoopBodyStatements) {
@@ -84959,62 +90041,62 @@ var ts;
             //     for (let v of arr) { }
             //
             // we don't want to emit a temporary variable for the RHS, just use it directly.
-            var counter = ts.createLoopVariable();
-            var rhsReference = ts.isIdentifier(expression) ? ts.getGeneratedNameForNode(expression) : ts.createTempVariable(/*recordTempVariable*/ undefined);
+            var counter = factory.createLoopVariable();
+            var rhsReference = ts.isIdentifier(expression) ? factory.getGeneratedNameForNode(expression) : factory.createTempVariable(/*recordTempVariable*/ undefined);
             // The old emitter does not emit source maps for the expression
             ts.setEmitFlags(expression, 48 /* NoSourceMap */ | ts.getEmitFlags(expression));
-            var forStatement = ts.setTextRange(ts.createFor(
-            /*initializer*/ ts.setEmitFlags(ts.setTextRange(ts.createVariableDeclarationList([
-                ts.setTextRange(ts.createVariableDeclaration(counter, /*type*/ undefined, ts.createLiteral(0)), ts.moveRangePos(node.expression, -1)),
-                ts.setTextRange(ts.createVariableDeclaration(rhsReference, /*type*/ undefined, expression), node.expression)
+            var forStatement = ts.setTextRange(factory.createForStatement(
+            /*initializer*/ ts.setEmitFlags(ts.setTextRange(factory.createVariableDeclarationList([
+                ts.setTextRange(factory.createVariableDeclaration(counter, /*exclamationToken*/ undefined, /*type*/ undefined, factory.createNumericLiteral(0)), ts.moveRangePos(node.expression, -1)),
+                ts.setTextRange(factory.createVariableDeclaration(rhsReference, /*exclamationToken*/ undefined, /*type*/ undefined, expression), node.expression)
             ]), node.expression), 2097152 /* NoHoisting */), 
-            /*condition*/ ts.setTextRange(ts.createLessThan(counter, ts.createPropertyAccess(rhsReference, "length")), node.expression), 
-            /*incrementor*/ ts.setTextRange(ts.createPostfixIncrement(counter), node.expression), 
-            /*statement*/ convertForOfStatementHead(node, ts.createElementAccess(rhsReference, counter), convertedLoopBodyStatements)), 
+            /*condition*/ ts.setTextRange(factory.createLessThan(counter, factory.createPropertyAccessExpression(rhsReference, "length")), node.expression), 
+            /*incrementor*/ ts.setTextRange(factory.createPostfixIncrement(counter), node.expression), 
+            /*statement*/ convertForOfStatementHead(node, factory.createElementAccessExpression(rhsReference, counter), convertedLoopBodyStatements)), 
             /*location*/ node);
             // Disable trailing source maps for the OpenParenToken to align source map emit with the old emitter.
             ts.setEmitFlags(forStatement, 256 /* NoTokenTrailingSourceMaps */);
             ts.setTextRange(forStatement, node);
-            return ts.restoreEnclosingLabel(forStatement, outermostLabeledStatement, convertedLoopState && resetLabel);
+            return factory.restoreEnclosingLabel(forStatement, outermostLabeledStatement, convertedLoopState && resetLabel);
         }
         function convertForOfStatementForIterable(node, outermostLabeledStatement, convertedLoopBodyStatements, ancestorFacts) {
             var expression = ts.visitNode(node.expression, visitor, ts.isExpression);
-            var iterator = ts.isIdentifier(expression) ? ts.getGeneratedNameForNode(expression) : ts.createTempVariable(/*recordTempVariable*/ undefined);
-            var result = ts.isIdentifier(expression) ? ts.getGeneratedNameForNode(iterator) : ts.createTempVariable(/*recordTempVariable*/ undefined);
-            var errorRecord = ts.createUniqueName("e");
-            var catchVariable = ts.getGeneratedNameForNode(errorRecord);
-            var returnMethod = ts.createTempVariable(/*recordTempVariable*/ undefined);
-            var values = ts.createValuesHelper(context, expression, node.expression);
-            var next = ts.createCall(ts.createPropertyAccess(iterator, "next"), /*typeArguments*/ undefined, []);
+            var iterator = ts.isIdentifier(expression) ? factory.getGeneratedNameForNode(expression) : factory.createTempVariable(/*recordTempVariable*/ undefined);
+            var result = ts.isIdentifier(expression) ? factory.getGeneratedNameForNode(iterator) : factory.createTempVariable(/*recordTempVariable*/ undefined);
+            var errorRecord = factory.createUniqueName("e");
+            var catchVariable = factory.getGeneratedNameForNode(errorRecord);
+            var returnMethod = factory.createTempVariable(/*recordTempVariable*/ undefined);
+            var values = ts.setTextRange(emitHelpers().createValuesHelper(expression), node.expression);
+            var next = factory.createCallExpression(factory.createPropertyAccessExpression(iterator, "next"), /*typeArguments*/ undefined, []);
             hoistVariableDeclaration(errorRecord);
             hoistVariableDeclaration(returnMethod);
             // if we are enclosed in an outer loop ensure we reset 'errorRecord' per each iteration
             var initializer = ancestorFacts & 1024 /* IterationContainer */
-                ? ts.inlineExpressions([ts.createAssignment(errorRecord, ts.createVoidZero()), values])
+                ? factory.inlineExpressions([factory.createAssignment(errorRecord, factory.createVoidZero()), values])
                 : values;
-            var forStatement = ts.setEmitFlags(ts.setTextRange(ts.createFor(
-            /*initializer*/ ts.setEmitFlags(ts.setTextRange(ts.createVariableDeclarationList([
-                ts.setTextRange(ts.createVariableDeclaration(iterator, /*type*/ undefined, initializer), node.expression),
-                ts.createVariableDeclaration(result, /*type*/ undefined, next)
+            var forStatement = ts.setEmitFlags(ts.setTextRange(factory.createForStatement(
+            /*initializer*/ ts.setEmitFlags(ts.setTextRange(factory.createVariableDeclarationList([
+                ts.setTextRange(factory.createVariableDeclaration(iterator, /*exclamationToken*/ undefined, /*type*/ undefined, initializer), node.expression),
+                factory.createVariableDeclaration(result, /*exclamationToken*/ undefined, /*type*/ undefined, next)
             ]), node.expression), 2097152 /* NoHoisting */), 
-            /*condition*/ ts.createLogicalNot(ts.createPropertyAccess(result, "done")), 
-            /*incrementor*/ ts.createAssignment(result, next), 
-            /*statement*/ convertForOfStatementHead(node, ts.createPropertyAccess(result, "value"), convertedLoopBodyStatements)), 
+            /*condition*/ factory.createLogicalNot(factory.createPropertyAccessExpression(result, "done")), 
+            /*incrementor*/ factory.createAssignment(result, next), 
+            /*statement*/ convertForOfStatementHead(node, factory.createPropertyAccessExpression(result, "value"), convertedLoopBodyStatements)), 
             /*location*/ node), 256 /* NoTokenTrailingSourceMaps */);
-            return ts.createTry(ts.createBlock([
-                ts.restoreEnclosingLabel(forStatement, outermostLabeledStatement, convertedLoopState && resetLabel)
-            ]), ts.createCatchClause(ts.createVariableDeclaration(catchVariable), ts.setEmitFlags(ts.createBlock([
-                ts.createExpressionStatement(ts.createAssignment(errorRecord, ts.createObjectLiteral([
-                    ts.createPropertyAssignment("error", catchVariable)
+            return factory.createTryStatement(factory.createBlock([
+                factory.restoreEnclosingLabel(forStatement, outermostLabeledStatement, convertedLoopState && resetLabel)
+            ]), factory.createCatchClause(factory.createVariableDeclaration(catchVariable), ts.setEmitFlags(factory.createBlock([
+                factory.createExpressionStatement(factory.createAssignment(errorRecord, factory.createObjectLiteralExpression([
+                    factory.createPropertyAssignment("error", catchVariable)
                 ])))
-            ]), 1 /* SingleLine */)), ts.createBlock([
-                ts.createTry(
-                /*tryBlock*/ ts.createBlock([
-                    ts.setEmitFlags(ts.createIf(ts.createLogicalAnd(ts.createLogicalAnd(result, ts.createLogicalNot(ts.createPropertyAccess(result, "done"))), ts.createAssignment(returnMethod, ts.createPropertyAccess(iterator, "return"))), ts.createExpressionStatement(ts.createFunctionCall(returnMethod, iterator, []))), 1 /* SingleLine */),
+            ]), 1 /* SingleLine */)), factory.createBlock([
+                factory.createTryStatement(
+                /*tryBlock*/ factory.createBlock([
+                    ts.setEmitFlags(factory.createIfStatement(factory.createLogicalAnd(factory.createLogicalAnd(result, factory.createLogicalNot(factory.createPropertyAccessExpression(result, "done"))), factory.createAssignment(returnMethod, factory.createPropertyAccessExpression(iterator, "return"))), factory.createExpressionStatement(factory.createFunctionCallCall(returnMethod, iterator, []))), 1 /* SingleLine */),
                 ]), 
                 /*catchClause*/ undefined, 
-                /*finallyBlock*/ ts.setEmitFlags(ts.createBlock([
-                    ts.setEmitFlags(ts.createIf(errorRecord, ts.createThrow(ts.createPropertyAccess(errorRecord, "error"))), 1 /* SingleLine */)
+                /*finallyBlock*/ ts.setEmitFlags(factory.createBlock([
+                    ts.setEmitFlags(factory.createIfStatement(errorRecord, factory.createThrowStatement(factory.createPropertyAccessExpression(errorRecord, "error"))), 1 /* SingleLine */)
                 ]), 1 /* SingleLine */))
             ]));
         }
@@ -85024,45 +90106,37 @@ var ts;
          * @param node An ObjectLiteralExpression node.
          */
         function visitObjectLiteralExpression(node) {
-            // We are here because a ComputedPropertyName was used somewhere in the expression.
             var properties = node.properties;
-            var numProperties = properties.length;
             // Find the first computed property.
             // Everything until that point can be emitted as part of the initial object literal.
-            var numInitialProperties = numProperties;
-            var numInitialPropertiesWithoutYield = numProperties;
-            for (var i = 0; i < numProperties; i++) {
+            var numInitialProperties = -1, hasComputed = false;
+            for (var i = 0; i < properties.length; i++) {
                 var property = properties[i];
-                if ((property.transformFlags & 262144 /* ContainsYield */ && hierarchyFacts & 4 /* AsyncFunctionBody */)
-                    && i < numInitialPropertiesWithoutYield) {
-                    numInitialPropertiesWithoutYield = i;
-                }
-                if (property.name.kind === 154 /* ComputedPropertyName */) {
+                if ((property.transformFlags & 262144 /* ContainsYield */ &&
+                    hierarchyFacts & 4 /* AsyncFunctionBody */)
+                    || (hasComputed = ts.Debug.checkDefined(property.name).kind === 158 /* ComputedPropertyName */)) {
                     numInitialProperties = i;
                     break;
                 }
             }
-            if (numInitialProperties !== numProperties) {
-                if (numInitialPropertiesWithoutYield < numInitialProperties) {
-                    numInitialProperties = numInitialPropertiesWithoutYield;
-                }
-                // For computed properties, we need to create a unique handle to the object
-                // literal so we can modify it without risking internal assignments tainting the object.
-                var temp = ts.createTempVariable(hoistVariableDeclaration);
-                // Write out the first non-computed properties, then emit the rest through indexing on the temp variable.
-                var expressions = [];
-                var assignment = ts.createAssignment(temp, ts.setEmitFlags(ts.createObjectLiteral(ts.visitNodes(properties, visitor, ts.isObjectLiteralElementLike, 0, numInitialProperties), node.multiLine), 65536 /* Indented */));
-                if (node.multiLine) {
-                    ts.startOnNewLine(assignment);
-                }
-                expressions.push(assignment);
-                addObjectLiteralMembers(expressions, node, temp, numInitialProperties);
-                // We need to clone the temporary identifier so that we can write it on a
-                // new line
-                expressions.push(node.multiLine ? ts.startOnNewLine(ts.getMutableClone(temp)) : temp);
-                return ts.inlineExpressions(expressions);
+            if (numInitialProperties < 0) {
+                return ts.visitEachChild(node, visitor, context);
             }
-            return ts.visitEachChild(node, visitor, context);
+            // For computed properties, we need to create a unique handle to the object
+            // literal so we can modify it without risking internal assignments tainting the object.
+            var temp = factory.createTempVariable(hoistVariableDeclaration);
+            // Write out the first non-computed properties, then emit the rest through indexing on the temp variable.
+            var expressions = [];
+            var assignment = factory.createAssignment(temp, ts.setEmitFlags(factory.createObjectLiteralExpression(ts.visitNodes(properties, visitor, ts.isObjectLiteralElementLike, 0, numInitialProperties), node.multiLine), hasComputed ? 65536 /* Indented */ : 0));
+            if (node.multiLine) {
+                ts.startOnNewLine(assignment);
+            }
+            expressions.push(assignment);
+            addObjectLiteralMembers(expressions, node, temp, numInitialProperties);
+            // We need to clone the temporary identifier so that we can write it on a
+            // new line
+            expressions.push(node.multiLine ? ts.startOnNewLine(ts.setParent(ts.setTextRange(factory.cloneNode(temp), temp), temp.parent)) : temp);
+            return factory.inlineExpressions(expressions);
         }
         function shouldConvertPartOfIterationStatement(node) {
             return (resolver.getNodeCheckFlags(node) & 131072 /* ContainsCapturedBlockScopeBinding */) !== 0;
@@ -85092,7 +90166,7 @@ var ts;
             }
             visit(node.name);
             function visit(node) {
-                if (node.kind === 75 /* Identifier */) {
+                if (node.kind === 78 /* Identifier */) {
                     state.hoistedLocalVariables.push(node);
                 }
                 else {
@@ -85116,7 +90190,7 @@ var ts;
                 }
                 var result = convert
                     ? convert(node, outermostLabeledStatement, /*convertedLoopBodyStatements*/ undefined, ancestorFacts)
-                    : ts.restoreEnclosingLabel(ts.visitEachChild(node, visitor, context), outermostLabeledStatement, convertedLoopState && resetLabel);
+                    : factory.restoreEnclosingLabel(ts.isForStatement(node) ? visitEachChildOfForStatement(node) : ts.visitEachChild(node, visitor, context), outermostLabeledStatement, convertedLoopState && resetLabel);
                 if (convertedLoopState) {
                     convertedLoopState.allowedNonLabeledJumps = saveAllowedNonLabeledJumps;
                 }
@@ -85143,55 +90217,53 @@ var ts;
                     loop = convert(node, outermostLabeledStatement, bodyFunction.part, ancestorFacts);
                 }
                 else {
-                    var clone_4 = convertIterationStatementCore(node, initializerFunction, ts.createBlock(bodyFunction.part, /*multiLine*/ true));
-                    ts.aggregateTransformFlags(clone_4);
-                    loop = ts.restoreEnclosingLabel(clone_4, outermostLabeledStatement, convertedLoopState && resetLabel);
+                    var clone_3 = convertIterationStatementCore(node, initializerFunction, factory.createBlock(bodyFunction.part, /*multiLine*/ true));
+                    loop = factory.restoreEnclosingLabel(clone_3, outermostLabeledStatement, convertedLoopState && resetLabel);
                 }
             }
             else {
-                var clone_5 = convertIterationStatementCore(node, initializerFunction, ts.visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
-                ts.aggregateTransformFlags(clone_5);
-                loop = ts.restoreEnclosingLabel(clone_5, outermostLabeledStatement, convertedLoopState && resetLabel);
+                var clone_4 = convertIterationStatementCore(node, initializerFunction, ts.visitNode(node.statement, visitor, ts.isStatement, factory.liftToBlock));
+                loop = factory.restoreEnclosingLabel(clone_4, outermostLabeledStatement, convertedLoopState && resetLabel);
             }
             statements.push(loop);
             return statements;
         }
         function convertIterationStatementCore(node, initializerFunction, convertedLoopBody) {
             switch (node.kind) {
-                case 230 /* ForStatement */: return convertForStatement(node, initializerFunction, convertedLoopBody);
-                case 231 /* ForInStatement */: return convertForInStatement(node, convertedLoopBody);
-                case 232 /* ForOfStatement */: return convertForOfStatement(node, convertedLoopBody);
-                case 228 /* DoStatement */: return convertDoStatement(node, convertedLoopBody);
-                case 229 /* WhileStatement */: return convertWhileStatement(node, convertedLoopBody);
+                case 237 /* ForStatement */: return convertForStatement(node, initializerFunction, convertedLoopBody);
+                case 238 /* ForInStatement */: return convertForInStatement(node, convertedLoopBody);
+                case 239 /* ForOfStatement */: return convertForOfStatement(node, convertedLoopBody);
+                case 235 /* DoStatement */: return convertDoStatement(node, convertedLoopBody);
+                case 236 /* WhileStatement */: return convertWhileStatement(node, convertedLoopBody);
                 default: return ts.Debug.failBadSyntaxKind(node, "IterationStatement expected");
             }
         }
         function convertForStatement(node, initializerFunction, convertedLoopBody) {
             var shouldConvertCondition = node.condition && shouldConvertPartOfIterationStatement(node.condition);
             var shouldConvertIncrementor = shouldConvertCondition || node.incrementor && shouldConvertPartOfIterationStatement(node.incrementor);
-            return ts.updateFor(node, ts.visitNode(initializerFunction ? initializerFunction.part : node.initializer, visitor, ts.isForInitializer), ts.visitNode(shouldConvertCondition ? undefined : node.condition, visitor, ts.isExpression), ts.visitNode(shouldConvertIncrementor ? undefined : node.incrementor, visitor, ts.isExpression), convertedLoopBody);
+            return factory.updateForStatement(node, ts.visitNode(initializerFunction ? initializerFunction.part : node.initializer, visitorWithUnusedExpressionResult, ts.isForInitializer), ts.visitNode(shouldConvertCondition ? undefined : node.condition, visitor, ts.isExpression), ts.visitNode(shouldConvertIncrementor ? undefined : node.incrementor, visitorWithUnusedExpressionResult, ts.isExpression), convertedLoopBody);
         }
         function convertForOfStatement(node, convertedLoopBody) {
-            return ts.updateForOf(node, 
+            return factory.updateForOfStatement(node, 
             /*awaitModifier*/ undefined, ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), convertedLoopBody);
         }
         function convertForInStatement(node, convertedLoopBody) {
-            return ts.updateForIn(node, ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), convertedLoopBody);
+            return factory.updateForInStatement(node, ts.visitNode(node.initializer, visitor, ts.isForInitializer), ts.visitNode(node.expression, visitor, ts.isExpression), convertedLoopBody);
         }
         function convertDoStatement(node, convertedLoopBody) {
-            return ts.updateDo(node, convertedLoopBody, ts.visitNode(node.expression, visitor, ts.isExpression));
+            return factory.updateDoStatement(node, convertedLoopBody, ts.visitNode(node.expression, visitor, ts.isExpression));
         }
         function convertWhileStatement(node, convertedLoopBody) {
-            return ts.updateWhile(node, ts.visitNode(node.expression, visitor, ts.isExpression), convertedLoopBody);
+            return factory.updateWhileStatement(node, ts.visitNode(node.expression, visitor, ts.isExpression), convertedLoopBody);
         }
         function createConvertedLoopState(node) {
             var loopInitializer;
             switch (node.kind) {
-                case 230 /* ForStatement */:
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
+                case 237 /* ForStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
                     var initializer = node.initializer;
-                    if (initializer && initializer.kind === 243 /* VariableDeclarationList */) {
+                    if (initializer && initializer.kind === 250 /* VariableDeclarationList */) {
                         loopInitializer = initializer;
                     }
                     break;
@@ -85240,8 +90312,9 @@ var ts;
                 }
                 else {
                     // this is top level converted loop and we need to create an alias for 'arguments' object
-                    (extraVariableDeclarations || (extraVariableDeclarations = [])).push(ts.createVariableDeclaration(state.argumentsName, 
-                    /*type*/ undefined, ts.createIdentifier("arguments")));
+                    (extraVariableDeclarations || (extraVariableDeclarations = [])).push(factory.createVariableDeclaration(state.argumentsName, 
+                    /*exclamationToken*/ undefined, 
+                    /*type*/ undefined, factory.createIdentifier("arguments")));
                 }
             }
             if (state.thisName) {
@@ -85255,8 +90328,9 @@ var ts;
                     // NOTE:
                     // if converted loops were all nested in arrow function then we'll always emit '_this' so convertedLoopState.thisName will not be set.
                     // If it is set this means that all nested loops are not nested in arrow function and it is safe to capture 'this'.
-                    (extraVariableDeclarations || (extraVariableDeclarations = [])).push(ts.createVariableDeclaration(state.thisName, 
-                    /*type*/ undefined, ts.createIdentifier("this")));
+                    (extraVariableDeclarations || (extraVariableDeclarations = [])).push(factory.createVariableDeclaration(state.thisName, 
+                    /*exclamationToken*/ undefined, 
+                    /*type*/ undefined, factory.createIdentifier("this")));
                 }
             }
             if (state.hoistedLocalVariables) {
@@ -85272,7 +90346,7 @@ var ts;
                     // hoist collected variable declarations
                     for (var _i = 0, _a = state.hoistedLocalVariables; _i < _a.length; _i++) {
                         var identifier = _a[_i];
-                        extraVariableDeclarations.push(ts.createVariableDeclaration(identifier));
+                        extraVariableDeclarations.push(factory.createVariableDeclaration(identifier));
                     }
                 }
             }
@@ -85283,23 +90357,23 @@ var ts;
                 }
                 for (var _b = 0, _c = state.loopOutParameters; _b < _c.length; _b++) {
                     var outParam = _c[_b];
-                    extraVariableDeclarations.push(ts.createVariableDeclaration(outParam.outParamName));
+                    extraVariableDeclarations.push(factory.createVariableDeclaration(outParam.outParamName));
                 }
             }
             if (state.conditionVariable) {
                 if (!extraVariableDeclarations) {
                     extraVariableDeclarations = [];
                 }
-                extraVariableDeclarations.push(ts.createVariableDeclaration(state.conditionVariable, /*type*/ undefined, ts.createFalse()));
+                extraVariableDeclarations.push(factory.createVariableDeclaration(state.conditionVariable, /*exclamationToken*/ undefined, /*type*/ undefined, factory.createFalse()));
             }
             // create variable statement to hold all introduced variable declarations
             if (extraVariableDeclarations) {
-                statements.push(ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList(extraVariableDeclarations)));
+                statements.push(factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList(extraVariableDeclarations)));
             }
         }
         function createOutVariable(p) {
-            return ts.createVariableDeclaration(p.originalName, /*type*/ undefined, p.outParamName);
+            return factory.createVariableDeclaration(p.originalName, /*exclamationToken*/ undefined, /*type*/ undefined, p.outParamName);
         }
         /**
          * Creates a `_loop_init` function for a `ForStatement` with a block-scoped initializer
@@ -85308,7 +90382,7 @@ var ts;
          * [13.7.4.8 RS: ForBodyEvaluation](https://tc39.github.io/ecma262/#sec-forbodyevaluation).
          */
         function createFunctionForInitializerOfForStatement(node, currentState) {
-            var functionName = ts.createUniqueName("_loop_init");
+            var functionName = factory.createUniqueName("_loop_init");
             var containsYield = (node.initializer.transformFlags & 262144 /* ContainsYield */) !== 0;
             var emitFlags = 0 /* None */;
             if (currentState.containsLexicalThis)
@@ -85316,7 +90390,7 @@ var ts;
             if (containsYield && hierarchyFacts & 4 /* AsyncFunctionBody */)
                 emitFlags |= 262144 /* AsyncFunctionBody */;
             var statements = [];
-            statements.push(ts.createVariableStatement(/*modifiers*/ undefined, node.initializer));
+            statements.push(factory.createVariableStatement(/*modifiers*/ undefined, node.initializer));
             copyOutParameters(currentState.loopOutParameters, 2 /* Initializer */, 1 /* ToOutParameter */, statements);
             // This transforms the following ES2015 syntax:
             //
@@ -85338,17 +90412,18 @@ var ts;
             //
             // Which prevents mutations to `i` in the per-iteration environment of the body
             // from affecting the initial value for `i` outside of the per-iteration environment.
-            var functionDeclaration = ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.setEmitFlags(ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(functionName, 
-                /*type*/ undefined, ts.setEmitFlags(ts.createFunctionExpression(
-                /*modifiers*/ undefined, containsYield ? ts.createToken(41 /* AsteriskToken */) : undefined, 
+            var functionDeclaration = factory.createVariableStatement(
+            /*modifiers*/ undefined, ts.setEmitFlags(factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(functionName, 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, ts.setEmitFlags(factory.createFunctionExpression(
+                /*modifiers*/ undefined, containsYield ? factory.createToken(41 /* AsteriskToken */) : undefined, 
                 /*name*/ undefined, 
                 /*typeParameters*/ undefined, 
                 /*parameters*/ undefined, 
-                /*type*/ undefined, ts.visitNode(ts.createBlock(statements, /*multiLine*/ true), visitor, ts.isBlock)), emitFlags))
+                /*type*/ undefined, ts.visitNode(factory.createBlock(statements, /*multiLine*/ true), visitor, ts.isBlock)), emitFlags))
             ]), 2097152 /* NoHoisting */));
-            var part = ts.createVariableDeclarationList(ts.map(currentState.loopOutParameters, createOutVariable));
+            var part = factory.createVariableDeclarationList(ts.map(currentState.loopOutParameters, createOutVariable));
             return { functionName: functionName, containsYield: containsYield, functionDeclaration: functionDeclaration, part: part };
         }
         /**
@@ -85358,9 +90433,9 @@ var ts;
          * [13.7.4.8 RS: ForBodyEvaluation](https://tc39.github.io/ecma262/#sec-forbodyevaluation).
          */
         function createFunctionForBodyOfIterationStatement(node, currentState, outerState) {
-            var functionName = ts.createUniqueName("_loop");
+            var functionName = factory.createUniqueName("_loop");
             startLexicalEnvironment();
-            var statement = ts.visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock);
+            var statement = ts.visitNode(node.statement, visitor, ts.isStatement, factory.liftToBlock);
             var lexicalEnvironment = endLexicalEnvironment();
             var statements = [];
             if (shouldConvertConditionOfForStatement(node) || shouldConvertIncrementorOfForStatement(node)) {
@@ -85401,10 +90476,15 @@ var ts;
                 // environment that is carried over to the next iteration of the loop. As a result,
                 // we must indicate whether this is the first evaluation of the loop body so that
                 // we only evaluate the incrementor on subsequent evaluations.
-                currentState.conditionVariable = ts.createUniqueName("inc");
-                statements.push(ts.createIf(currentState.conditionVariable, ts.createStatement(ts.visitNode(node.incrementor, visitor, ts.isExpression)), ts.createStatement(ts.createAssignment(currentState.conditionVariable, ts.createTrue()))));
+                currentState.conditionVariable = factory.createUniqueName("inc");
+                if (node.incrementor) {
+                    statements.push(factory.createIfStatement(currentState.conditionVariable, factory.createExpressionStatement(ts.visitNode(node.incrementor, visitor, ts.isExpression)), factory.createExpressionStatement(factory.createAssignment(currentState.conditionVariable, factory.createTrue()))));
+                }
+                else {
+                    statements.push(factory.createIfStatement(factory.createLogicalNot(currentState.conditionVariable), factory.createExpressionStatement(factory.createAssignment(currentState.conditionVariable, factory.createTrue()))));
+                }
                 if (shouldConvertConditionOfForStatement(node)) {
-                    statements.push(ts.createIf(ts.createPrefix(53 /* ExclamationToken */, ts.visitNode(node.condition, visitor, ts.isExpression)), ts.visitNode(ts.createBreak(), visitor, ts.isStatement)));
+                    statements.push(factory.createIfStatement(factory.createPrefixUnaryExpression(53 /* ExclamationToken */, ts.visitNode(node.condition, visitor, ts.isExpression)), ts.visitNode(factory.createBreakStatement(), visitor, ts.isStatement)));
                 }
             }
             if (ts.isBlock(statement)) {
@@ -85415,7 +90495,7 @@ var ts;
             }
             copyOutParameters(currentState.loopOutParameters, 1 /* Body */, 1 /* ToOutParameter */, statements);
             ts.insertStatementsAfterStandardPrologue(statements, lexicalEnvironment);
-            var loopBody = ts.createBlock(statements, /*multiLine*/ true);
+            var loopBody = factory.createBlock(statements, /*multiLine*/ true);
             if (ts.isBlock(statement))
                 ts.setOriginalNode(loopBody, statement);
             var containsYield = (node.statement.transformFlags & 262144 /* ContainsYield */) !== 0;
@@ -85438,11 +90518,12 @@ var ts;
             //  for (var i = 0; i < 2; i++) {
             //      _loop_1(i);
             //  }
-            var functionDeclaration = ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.setEmitFlags(ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(functionName, 
-                /*type*/ undefined, ts.setEmitFlags(ts.createFunctionExpression(
-                /*modifiers*/ undefined, containsYield ? ts.createToken(41 /* AsteriskToken */) : undefined, 
+            var functionDeclaration = factory.createVariableStatement(
+            /*modifiers*/ undefined, ts.setEmitFlags(factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(functionName, 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, ts.setEmitFlags(factory.createFunctionExpression(
+                /*modifiers*/ undefined, containsYield ? factory.createToken(41 /* AsteriskToken */) : undefined, 
                 /*name*/ undefined, 
                 /*typeParameters*/ undefined, currentState.loopParameters, 
                 /*type*/ undefined, loopBody), emitFlags))
@@ -85453,22 +90534,22 @@ var ts;
         function copyOutParameter(outParam, copyDirection) {
             var source = copyDirection === 0 /* ToOriginal */ ? outParam.outParamName : outParam.originalName;
             var target = copyDirection === 0 /* ToOriginal */ ? outParam.originalName : outParam.outParamName;
-            return ts.createBinary(target, 62 /* EqualsToken */, source);
+            return factory.createBinaryExpression(target, 62 /* EqualsToken */, source);
         }
         function copyOutParameters(outParams, partFlags, copyDirection, statements) {
             for (var _i = 0, outParams_1 = outParams; _i < outParams_1.length; _i++) {
                 var outParam = outParams_1[_i];
                 if (outParam.flags & partFlags) {
-                    statements.push(ts.createExpressionStatement(copyOutParameter(outParam, copyDirection)));
+                    statements.push(factory.createExpressionStatement(copyOutParameter(outParam, copyDirection)));
                 }
             }
         }
         function generateCallToConvertedLoopInitializer(initFunctionExpressionName, containsYield) {
-            var call = ts.createCall(initFunctionExpressionName, /*typeArguments*/ undefined, []);
+            var call = factory.createCallExpression(initFunctionExpressionName, /*typeArguments*/ undefined, []);
             var callResult = containsYield
-                ? ts.createYield(ts.createToken(41 /* AsteriskToken */), ts.setEmitFlags(call, 8388608 /* Iterator */))
+                ? factory.createYieldExpression(factory.createToken(41 /* AsteriskToken */), ts.setEmitFlags(call, 8388608 /* Iterator */))
                 : call;
-            return ts.createStatement(callResult);
+            return factory.createExpressionStatement(callResult);
         }
         function generateCallToConvertedLoop(loopFunctionExpressionName, state, outerState, containsYield) {
             var statements = [];
@@ -85478,39 +90559,39 @@ var ts;
             var isSimpleLoop = !(state.nonLocalJumps & ~4 /* Continue */) &&
                 !state.labeledNonLocalBreaks &&
                 !state.labeledNonLocalContinues;
-            var call = ts.createCall(loopFunctionExpressionName, /*typeArguments*/ undefined, ts.map(state.loopParameters, function (p) { return p.name; }));
+            var call = factory.createCallExpression(loopFunctionExpressionName, /*typeArguments*/ undefined, ts.map(state.loopParameters, function (p) { return p.name; }));
             var callResult = containsYield
-                ? ts.createYield(ts.createToken(41 /* AsteriskToken */), ts.setEmitFlags(call, 8388608 /* Iterator */))
+                ? factory.createYieldExpression(factory.createToken(41 /* AsteriskToken */), ts.setEmitFlags(call, 8388608 /* Iterator */))
                 : call;
             if (isSimpleLoop) {
-                statements.push(ts.createExpressionStatement(callResult));
+                statements.push(factory.createExpressionStatement(callResult));
                 copyOutParameters(state.loopOutParameters, 1 /* Body */, 0 /* ToOriginal */, statements);
             }
             else {
-                var loopResultName = ts.createUniqueName("state");
-                var stateVariable = ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(loopResultName, /*type*/ undefined, callResult)]));
+                var loopResultName = factory.createUniqueName("state");
+                var stateVariable = factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList([factory.createVariableDeclaration(loopResultName, /*exclamationToken*/ undefined, /*type*/ undefined, callResult)]));
                 statements.push(stateVariable);
                 copyOutParameters(state.loopOutParameters, 1 /* Body */, 0 /* ToOriginal */, statements);
                 if (state.nonLocalJumps & 8 /* Return */) {
                     var returnStatement = void 0;
                     if (outerState) {
                         outerState.nonLocalJumps |= 8 /* Return */;
-                        returnStatement = ts.createReturn(loopResultName);
+                        returnStatement = factory.createReturnStatement(loopResultName);
                     }
                     else {
-                        returnStatement = ts.createReturn(ts.createPropertyAccess(loopResultName, "value"));
+                        returnStatement = factory.createReturnStatement(factory.createPropertyAccessExpression(loopResultName, "value"));
                     }
-                    statements.push(ts.createIf(ts.createBinary(ts.createTypeOf(loopResultName), 36 /* EqualsEqualsEqualsToken */, ts.createLiteral("object")), returnStatement));
+                    statements.push(factory.createIfStatement(factory.createTypeCheck(loopResultName, "object"), returnStatement));
                 }
                 if (state.nonLocalJumps & 2 /* Break */) {
-                    statements.push(ts.createIf(ts.createBinary(loopResultName, 36 /* EqualsEqualsEqualsToken */, ts.createLiteral("break")), ts.createBreak()));
+                    statements.push(factory.createIfStatement(factory.createStrictEquality(loopResultName, factory.createStringLiteral("break")), factory.createBreakStatement()));
                 }
                 if (state.labeledNonLocalBreaks || state.labeledNonLocalContinues) {
                     var caseClauses = [];
                     processLabeledJumps(state.labeledNonLocalBreaks, /*isBreak*/ true, loopResultName, outerState, caseClauses);
                     processLabeledJumps(state.labeledNonLocalContinues, /*isBreak*/ false, loopResultName, outerState, caseClauses);
-                    statements.push(ts.createSwitch(loopResultName, ts.createCaseBlock(caseClauses)));
+                    statements.push(factory.createSwitchStatement(loopResultName, factory.createCaseBlock(caseClauses)));
                 }
             }
             return statements;
@@ -85518,13 +90599,13 @@ var ts;
         function setLabeledJump(state, isBreak, labelText, labelMarker) {
             if (isBreak) {
                 if (!state.labeledNonLocalBreaks) {
-                    state.labeledNonLocalBreaks = ts.createMap();
+                    state.labeledNonLocalBreaks = new ts.Map();
                 }
                 state.labeledNonLocalBreaks.set(labelText, labelMarker);
             }
             else {
                 if (!state.labeledNonLocalContinues) {
-                    state.labeledNonLocalContinues = ts.createMap();
+                    state.labeledNonLocalContinues = new ts.Map();
                 }
                 state.labeledNonLocalContinues.set(labelText, labelMarker);
             }
@@ -85539,14 +90620,14 @@ var ts;
                 // then emit labeled break\continue
                 // otherwise propagate pair 'label -> marker' to outer converted loop and emit 'return labelMarker' so outer loop can later decide what to do
                 if (!outerLoop || (outerLoop.labels && outerLoop.labels.get(labelText))) {
-                    var label = ts.createIdentifier(labelText);
-                    statements.push(isBreak ? ts.createBreak(label) : ts.createContinue(label));
+                    var label = factory.createIdentifier(labelText);
+                    statements.push(isBreak ? factory.createBreakStatement(label) : factory.createContinueStatement(label));
                 }
                 else {
                     setLabeledJump(outerLoop, isBreak, labelText, labelMarker);
-                    statements.push(ts.createReturn(loopResultName));
+                    statements.push(factory.createReturnStatement(loopResultName));
                 }
-                caseClauses.push(ts.createCaseClause(ts.createLiteral(labelMarker), statements));
+                caseClauses.push(factory.createCaseClause(factory.createStringLiteral(labelMarker), statements));
             });
         }
         function processLoopVariableDeclaration(container, decl, loopParameters, loopOutParameters, hasCapturedBindingsInForInitializer) {
@@ -85560,10 +90641,10 @@ var ts;
                 }
             }
             else {
-                loopParameters.push(ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, name));
+                loopParameters.push(factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, name));
                 var checkFlags = resolver.getNodeCheckFlags(decl);
                 if (checkFlags & 4194304 /* NeedsLoopOutParameter */ || hasCapturedBindingsInForInitializer) {
-                    var outParamName = ts.createUniqueName("out_" + ts.idText(name));
+                    var outParamName = factory.createUniqueName("out_" + ts.idText(name));
                     var flags = 0;
                     if (checkFlags & 4194304 /* NeedsLoopOutParameter */) {
                         flags |= 1 /* Body */;
@@ -85590,20 +90671,20 @@ var ts;
             for (var i = start; i < numProperties; i++) {
                 var property = properties[i];
                 switch (property.kind) {
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
                         var accessors = ts.getAllAccessorDeclarations(node.properties, property);
                         if (property === accessors.firstAccessor) {
                             expressions.push(transformAccessorsToExpression(receiver, accessors, node, !!node.multiLine));
                         }
                         break;
-                    case 161 /* MethodDeclaration */:
+                    case 165 /* MethodDeclaration */:
                         expressions.push(transformObjectLiteralMethodDeclarationToExpression(property, receiver, node, node.multiLine));
                         break;
-                    case 281 /* PropertyAssignment */:
+                    case 288 /* PropertyAssignment */:
                         expressions.push(transformPropertyAssignmentToExpression(property, receiver, node.multiLine));
                         break;
-                    case 282 /* ShorthandPropertyAssignment */:
+                    case 289 /* ShorthandPropertyAssignment */:
                         expressions.push(transformShorthandPropertyAssignmentToExpression(property, receiver, node.multiLine));
                         break;
                     default:
@@ -85620,7 +90701,7 @@ var ts;
          * @param receiver The receiver for the assignment.
          */
         function transformPropertyAssignmentToExpression(property, receiver, startsOnNewLine) {
-            var expression = ts.createAssignment(ts.createMemberAccessForPropertyName(receiver, ts.visitNode(property.name, visitor, ts.isPropertyName)), ts.visitNode(property.initializer, visitor, ts.isExpression));
+            var expression = factory.createAssignment(ts.createMemberAccessForPropertyName(factory, receiver, ts.visitNode(property.name, visitor, ts.isPropertyName)), ts.visitNode(property.initializer, visitor, ts.isExpression));
             ts.setTextRange(expression, property);
             if (startsOnNewLine) {
                 ts.startOnNewLine(expression);
@@ -85635,7 +90716,7 @@ var ts;
          * @param receiver The receiver for the assignment.
          */
         function transformShorthandPropertyAssignmentToExpression(property, receiver, startsOnNewLine) {
-            var expression = ts.createAssignment(ts.createMemberAccessForPropertyName(receiver, ts.visitNode(property.name, visitor, ts.isPropertyName)), ts.getSynthesizedClone(property.name));
+            var expression = factory.createAssignment(ts.createMemberAccessForPropertyName(factory, receiver, ts.visitNode(property.name, visitor, ts.isPropertyName)), factory.cloneNode(property.name));
             ts.setTextRange(expression, property);
             if (startsOnNewLine) {
                 ts.startOnNewLine(expression);
@@ -85650,7 +90731,7 @@ var ts;
          * @param receiver The receiver for the assignment.
          */
         function transformObjectLiteralMethodDeclarationToExpression(method, receiver, container, startsOnNewLine) {
-            var expression = ts.createAssignment(ts.createMemberAccessForPropertyName(receiver, ts.visitNode(method.name, visitor, ts.isPropertyName)), transformFunctionLikeToExpression(method, /*location*/ method, /*name*/ undefined, container));
+            var expression = factory.createAssignment(ts.createMemberAccessForPropertyName(factory, receiver, ts.visitNode(method.name, visitor, ts.isPropertyName)), transformFunctionLikeToExpression(method, /*location*/ method, /*name*/ undefined, container));
             ts.setTextRange(expression, method);
             if (startsOnNewLine) {
                 ts.startOnNewLine(expression);
@@ -85662,14 +90743,14 @@ var ts;
             var updated;
             ts.Debug.assert(!!node.variableDeclaration, "Catch clause variable should always be present when downleveling ES2015.");
             if (ts.isBindingPattern(node.variableDeclaration.name)) {
-                var temp = ts.createTempVariable(/*recordTempVariable*/ undefined);
-                var newVariableDeclaration = ts.createVariableDeclaration(temp);
+                var temp = factory.createTempVariable(/*recordTempVariable*/ undefined);
+                var newVariableDeclaration = factory.createVariableDeclaration(temp);
                 ts.setTextRange(newVariableDeclaration, node.variableDeclaration);
                 var vars = ts.flattenDestructuringBinding(node.variableDeclaration, visitor, context, 0 /* All */, temp);
-                var list = ts.createVariableDeclarationList(vars);
+                var list = factory.createVariableDeclarationList(vars);
                 ts.setTextRange(list, node.variableDeclaration);
-                var destructure = ts.createVariableStatement(/*modifiers*/ undefined, list);
-                updated = ts.updateCatchClause(node, newVariableDeclaration, addStatementToStartOfBlock(node.block, destructure));
+                var destructure = factory.createVariableStatement(/*modifiers*/ undefined, list);
+                updated = factory.updateCatchClause(node, newVariableDeclaration, addStatementToStartOfBlock(node.block, destructure));
             }
             else {
                 updated = ts.visitEachChild(node, visitor, context);
@@ -85679,7 +90760,7 @@ var ts;
         }
         function addStatementToStartOfBlock(block, statement) {
             var transformedStatements = ts.visitNodes(block.statements, visitor, ts.isStatement);
-            return ts.updateBlock(block, __spreadArrays([statement], transformedStatements));
+            return factory.updateBlock(block, __spreadArrays([statement], transformedStatements));
         }
         /**
          * Visits a MethodDeclaration of an ObjectLiteralExpression and transforms it into a
@@ -85694,7 +90775,7 @@ var ts;
             ts.Debug.assert(!ts.isComputedPropertyName(node.name));
             var functionExpression = transformFunctionLikeToExpression(node, /*location*/ ts.moveRangePos(node, -1), /*name*/ undefined, /*container*/ undefined);
             ts.setEmitFlags(functionExpression, 512 /* NoLeadingComments */ | ts.getEmitFlags(functionExpression));
-            return ts.setTextRange(ts.createPropertyAssignment(node.name, functionExpression), 
+            return ts.setTextRange(factory.createPropertyAssignment(node.name, functionExpression), 
             /*location*/ node);
         }
         /**
@@ -85710,11 +90791,11 @@ var ts;
             var updated;
             var parameters = ts.visitParameterList(node.parameters, visitor, context);
             var body = transformFunctionBody(node);
-            if (node.kind === 163 /* GetAccessor */) {
-                updated = ts.updateGetAccessor(node, node.decorators, node.modifiers, node.name, parameters, node.type, body);
+            if (node.kind === 167 /* GetAccessor */) {
+                updated = factory.updateGetAccessorDeclaration(node, node.decorators, node.modifiers, node.name, parameters, node.type, body);
             }
             else {
-                updated = ts.updateSetAccessor(node, node.decorators, node.modifiers, node.name, parameters, body);
+                updated = factory.updateSetAccessorDeclaration(node, node.decorators, node.modifiers, node.name, parameters, body);
             }
             exitSubtree(ancestorFacts, 49152 /* FunctionSubtreeExcludes */, 0 /* None */);
             convertedLoopState = savedConvertedLoopState;
@@ -85726,7 +90807,7 @@ var ts;
          * @param node A ShorthandPropertyAssignment node.
          */
         function visitShorthandPropertyAssignment(node) {
-            return ts.setTextRange(ts.createPropertyAssignment(node.name, ts.getSynthesizedClone(node.name)), 
+            return ts.setTextRange(factory.createPropertyAssignment(node.name, visitIdentifier(factory.cloneNode(node.name))), 
             /*location*/ node);
         }
         function visitComputedPropertyName(node) {
@@ -85763,12 +90844,12 @@ var ts;
                 return visitTypeScriptClassWrapper(node);
             }
             var expression = ts.skipOuterExpressions(node.expression);
-            if (expression.kind === 102 /* SuperKeyword */ ||
+            if (expression.kind === 105 /* SuperKeyword */ ||
                 ts.isSuperProperty(expression) ||
                 ts.some(node.arguments, ts.isSpreadElement)) {
                 return visitCallExpressionWithPotentialCapturedThisAssignment(node, /*assignToCapturedThis*/ true);
             }
-            return ts.updateCall(node, ts.visitNode(node.expression, callExpressionVisitor, ts.isExpression), 
+            return factory.updateCallExpression(node, ts.visitNode(node.expression, callExpressionVisitor, ts.isExpression), 
             /*typeArguments*/ undefined, ts.visitNodes(node.arguments, visitor, ts.isExpression));
         }
         function visitTypeScriptClassWrapper(node) {
@@ -85855,7 +90936,7 @@ var ts;
                 statements.push(funcStatements[classBodyStart]);
                 classBodyStart++;
                 // Add the class alias following the declaration.
-                statements.push(ts.createExpressionStatement(ts.createAssignment(aliasAssignment.left, ts.cast(variable.name, ts.isIdentifier))));
+                statements.push(factory.createExpressionStatement(factory.createAssignment(aliasAssignment.left, ts.cast(variable.name, ts.isIdentifier))));
             }
             // Find the trailing 'return' statement (4)
             while (!ts.isReturnStatement(ts.elementAt(funcStatements, classBodyEnd))) {
@@ -85876,12 +90957,12 @@ var ts;
             ts.addRange(statements, classStatements, /*start*/ 1);
             // Recreate any outer parentheses or partially-emitted expressions to preserve source map
             // and comment locations.
-            return ts.recreateOuterExpressions(node.expression, ts.recreateOuterExpressions(variable.initializer, ts.recreateOuterExpressions(aliasAssignment && aliasAssignment.right, ts.updateCall(call, ts.recreateOuterExpressions(call.expression, ts.updateFunctionExpression(func, 
+            return factory.restoreOuterExpressions(node.expression, factory.restoreOuterExpressions(variable.initializer, factory.restoreOuterExpressions(aliasAssignment && aliasAssignment.right, factory.updateCallExpression(call, factory.restoreOuterExpressions(call.expression, factory.updateFunctionExpression(func, 
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
             /*typeParameters*/ undefined, func.parameters, 
-            /*type*/ undefined, ts.updateBlock(func.body, statements))), 
+            /*type*/ undefined, factory.updateBlock(func.body, statements))), 
             /*typeArguments*/ undefined, call.arguments))));
         }
         function visitImmediateSuperCallInBody(node) {
@@ -85891,10 +90972,10 @@ var ts;
             // We are here either because SuperKeyword was used somewhere in the expression, or
             // because we contain a SpreadElementExpression.
             if (node.transformFlags & 8192 /* ContainsRestOrSpread */ ||
-                node.expression.kind === 102 /* SuperKeyword */ ||
+                node.expression.kind === 105 /* SuperKeyword */ ||
                 ts.isSuperProperty(ts.skipOuterExpressions(node.expression))) {
-                var _a = ts.createCallBinding(node.expression, hoistVariableDeclaration), target = _a.target, thisArg = _a.thisArg;
-                if (node.expression.kind === 102 /* SuperKeyword */) {
+                var _a = factory.createCallBinding(node.expression, hoistVariableDeclaration), target = _a.target, thisArg = _a.thisArg;
+                if (node.expression.kind === 105 /* SuperKeyword */) {
                     ts.setEmitFlags(thisArg, 4 /* NoSubstitution */);
                 }
                 var resultingCall = void 0;
@@ -85912,7 +90993,7 @@ var ts;
                     //      _super.apply(this, a.concat([b]))
                     //      _super.m.apply(this, a.concat([b]))
                     //      _super.prototype.m.apply(this, a.concat([b]))
-                    resultingCall = ts.createFunctionApply(ts.visitNode(target, callExpressionVisitor, ts.isExpression), node.expression.kind === 102 /* SuperKeyword */ ? thisArg : ts.visitNode(thisArg, visitor, ts.isExpression), transformAndSpreadElements(node.arguments, /*needsUniqueCopy*/ false, /*multiLine*/ false, /*hasTrailingComma*/ false));
+                    resultingCall = factory.createFunctionApplyCall(ts.visitNode(target, callExpressionVisitor, ts.isExpression), node.expression.kind === 105 /* SuperKeyword */ ? thisArg : ts.visitNode(thisArg, visitor, ts.isExpression), transformAndSpreadElements(node.arguments, /*needsUniqueCopy*/ false, /*multiLine*/ false, /*hasTrailingComma*/ false));
                 }
                 else {
                     // [source]
@@ -85924,13 +91005,12 @@ var ts;
                     //      _super.call(this, a)
                     //      _super.m.call(this, a)
                     //      _super.prototype.m.call(this, a)
-                    resultingCall = ts.createFunctionCall(ts.visitNode(target, callExpressionVisitor, ts.isExpression), node.expression.kind === 102 /* SuperKeyword */ ? thisArg : ts.visitNode(thisArg, visitor, ts.isExpression), ts.visitNodes(node.arguments, visitor, ts.isExpression), 
-                    /*location*/ node);
+                    resultingCall = ts.setTextRange(factory.createFunctionCallCall(ts.visitNode(target, callExpressionVisitor, ts.isExpression), node.expression.kind === 105 /* SuperKeyword */ ? thisArg : ts.visitNode(thisArg, visitor, ts.isExpression), ts.visitNodes(node.arguments, visitor, ts.isExpression)), node);
                 }
-                if (node.expression.kind === 102 /* SuperKeyword */) {
-                    var initializer = ts.createLogicalOr(resultingCall, createActualThis());
+                if (node.expression.kind === 105 /* SuperKeyword */) {
+                    var initializer = factory.createLogicalOr(resultingCall, createActualThis());
                     resultingCall = assignToCapturedThis
-                        ? ts.createAssignment(ts.createFileLevelUniqueName("_this"), initializer)
+                        ? factory.createAssignment(factory.createUniqueName("_this", 16 /* Optimistic */ | 32 /* FileLevel */), initializer)
                         : initializer;
                 }
                 return ts.setOriginalNode(resultingCall, node);
@@ -85950,8 +91030,8 @@ var ts;
                 //
                 // [output]
                 //      new ((_a = C).bind.apply(_a, [void 0].concat(a)))()
-                var _a = ts.createCallBinding(ts.createPropertyAccess(node.expression, "bind"), hoistVariableDeclaration), target = _a.target, thisArg = _a.thisArg;
-                return ts.createNew(ts.createFunctionApply(ts.visitNode(target, visitor, ts.isExpression), thisArg, transformAndSpreadElements(ts.createNodeArray(__spreadArrays([ts.createVoidZero()], node.arguments)), /*needsUniqueCopy*/ false, /*multiLine*/ false, /*hasTrailingComma*/ false)), 
+                var _a = factory.createCallBinding(factory.createPropertyAccessExpression(node.expression, "bind"), hoistVariableDeclaration), target = _a.target, thisArg = _a.thisArg;
+                return factory.createNewExpression(factory.createFunctionApplyCall(ts.visitNode(target, visitor, ts.isExpression), thisArg, transformAndSpreadElements(factory.createNodeArray(__spreadArrays([factory.createVoidZero()], node.arguments)), /*needsUniqueCopy*/ false, /*multiLine*/ false, /*hasTrailingComma*/ false)), 
                 /*typeArguments*/ undefined, []);
             }
             return ts.visitEachChild(node, visitor, context);
@@ -85985,7 +91065,7 @@ var ts;
                         return segments[0];
                     }
                 }
-                return ts.createSpreadHelper(context, segments);
+                return emitHelpers().createSpreadHelper(segments);
             }
             else {
                 if (segments.length === 1) {
@@ -85996,7 +91076,7 @@ var ts;
                         return segments[0];
                     }
                 }
-                return ts.createSpreadArraysHelper(context, segments);
+                return emitHelpers().createSpreadArraysHelper(segments);
             }
         }
         function isPackedElement(node) {
@@ -86020,7 +91100,7 @@ var ts;
             return ts.map(chunk, visitExpressionOfSpread);
         }
         function visitSpanOfNonSpreads(chunk, multiLine, hasTrailingComma) {
-            return ts.createArrayLiteral(ts.visitNodes(ts.createNodeArray(chunk, hasTrailingComma), visitor, ts.isExpression), multiLine);
+            return factory.createArrayLiteralExpression(ts.visitNodes(factory.createNodeArray(chunk, hasTrailingComma), visitor, ts.isExpression), multiLine);
         }
         function visitSpreadElement(node) {
             return ts.visitNode(node.expression, visitor, ts.isExpression);
@@ -86039,7 +91119,7 @@ var ts;
          * @param node A template literal.
          */
         function visitTemplateLiteral(node) {
-            return ts.setTextRange(ts.createLiteral(node.text), node);
+            return ts.setTextRange(factory.createStringLiteral(node.text), node);
         }
         /**
          * Visits a string literal with an extended unicode escape.
@@ -86048,7 +91128,7 @@ var ts;
          */
         function visitStringLiteral(node) {
             if (node.hasExtendedUnicodeEscape) {
-                return ts.setTextRange(ts.createLiteral(node.text), node);
+                return ts.setTextRange(factory.createStringLiteral(node.text), node);
             }
             return node;
         }
@@ -86059,7 +91139,7 @@ var ts;
          */
         function visitNumericLiteral(node) {
             if (node.numericLiteralFlags & 384 /* BinaryOrOctalSpecifier */) {
-                return ts.setTextRange(ts.createNumericLiteral(node.text), node);
+                return ts.setTextRange(factory.createNumericLiteral(node.text), node);
             }
             return node;
         }
@@ -86089,10 +91169,9 @@ var ts;
             //    ("abc" + 1) << (2 + "")
             // rather than
             //    "abc" + (1 << 2) + ""
-            var expression = ts.reduceLeft(expressions, ts.createAdd);
+            var expression = ts.reduceLeft(expressions, factory.createAdd);
             if (ts.nodeIsSynthesized(expression)) {
-                expression.pos = node.pos;
-                expression.end = node.end;
+                ts.setTextRange(expression, node);
             }
             return expression;
         }
@@ -86130,7 +91209,7 @@ var ts;
             if (!shouldAddTemplateHead(node)) {
                 return;
             }
-            expressions.push(ts.createLiteral(node.head.text));
+            expressions.push(factory.createStringLiteral(node.head.text));
         }
         /**
          * Visits and adds the template spans of a TemplateExpression to an array of expressions.
@@ -86147,7 +91226,7 @@ var ts;
                 // with the head will force the result up to this point to be a string.
                 // Emitting a '+ ""' has no semantic effect for middles and tails.
                 if (span.literal.text.length !== 0) {
-                    expressions.push(ts.createLiteral(span.literal.text));
+                    expressions.push(factory.createStringLiteral(span.literal.text));
                 }
             }
         }
@@ -86155,13 +91234,15 @@ var ts;
          * Visits the `super` keyword
          */
         function visitSuperKeyword(isExpressionOfCall) {
-            return hierarchyFacts & 8 /* NonStaticClassElement */ && !isExpressionOfCall ? ts.createPropertyAccess(ts.createFileLevelUniqueName("_super"), "prototype") :
-                ts.createFileLevelUniqueName("_super");
+            return hierarchyFacts & 8 /* NonStaticClassElement */
+                && !isExpressionOfCall
+                ? factory.createPropertyAccessExpression(factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */), "prototype")
+                : factory.createUniqueName("_super", 16 /* Optimistic */ | 32 /* FileLevel */);
         }
         function visitMetaProperty(node) {
-            if (node.keywordToken === 99 /* NewKeyword */ && node.name.escapedText === "target") {
+            if (node.keywordToken === 102 /* NewKeyword */ && node.name.escapedText === "target") {
                 hierarchyFacts |= 16384 /* NewTarget */;
-                return ts.createFileLevelUniqueName("_newTarget");
+                return factory.createUniqueName("_newTarget", 16 /* Optimistic */ | 32 /* FileLevel */);
             }
             return node;
         }
@@ -86191,7 +91272,7 @@ var ts;
         function enableSubstitutionsForBlockScopedBindings() {
             if ((enabledSubstitutions & 2 /* BlockScopedBindings */) === 0) {
                 enabledSubstitutions |= 2 /* BlockScopedBindings */;
-                context.enableSubstitution(75 /* Identifier */);
+                context.enableSubstitution(78 /* Identifier */);
             }
         }
         /**
@@ -86201,14 +91282,14 @@ var ts;
         function enableSubstitutionsForCapturedThis() {
             if ((enabledSubstitutions & 1 /* CapturedThis */) === 0) {
                 enabledSubstitutions |= 1 /* CapturedThis */;
-                context.enableSubstitution(104 /* ThisKeyword */);
-                context.enableEmitNotification(162 /* Constructor */);
-                context.enableEmitNotification(161 /* MethodDeclaration */);
-                context.enableEmitNotification(163 /* GetAccessor */);
-                context.enableEmitNotification(164 /* SetAccessor */);
-                context.enableEmitNotification(202 /* ArrowFunction */);
-                context.enableEmitNotification(201 /* FunctionExpression */);
-                context.enableEmitNotification(244 /* FunctionDeclaration */);
+                context.enableSubstitution(107 /* ThisKeyword */);
+                context.enableEmitNotification(166 /* Constructor */);
+                context.enableEmitNotification(165 /* MethodDeclaration */);
+                context.enableEmitNotification(167 /* GetAccessor */);
+                context.enableEmitNotification(168 /* SetAccessor */);
+                context.enableEmitNotification(209 /* ArrowFunction */);
+                context.enableEmitNotification(208 /* FunctionExpression */);
+                context.enableEmitNotification(251 /* FunctionDeclaration */);
             }
         }
         /**
@@ -86236,7 +91317,7 @@ var ts;
             if (enabledSubstitutions & 2 /* BlockScopedBindings */ && !ts.isInternalName(node)) {
                 var original = ts.getParseTreeNode(node, ts.isIdentifier);
                 if (original && isNameOfDeclarationWithCollidingName(original)) {
-                    return ts.setTextRange(ts.getGeneratedNameForNode(original), node);
+                    return ts.setTextRange(factory.getGeneratedNameForNode(original), node);
                 }
             }
             return node;
@@ -86249,10 +91330,10 @@ var ts;
          */
         function isNameOfDeclarationWithCollidingName(node) {
             switch (node.parent.kind) {
-                case 191 /* BindingElement */:
-                case 245 /* ClassDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 242 /* VariableDeclaration */:
+                case 198 /* BindingElement */:
+                case 252 /* ClassDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return node.parent.name === node
                         && resolver.isDeclarationWithCollidingName(node.parent);
             }
@@ -86265,9 +91346,9 @@ var ts;
          */
         function substituteExpression(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return substituteExpressionIdentifier(node);
-                case 104 /* ThisKeyword */:
+                case 107 /* ThisKeyword */:
                     return substituteThisKeyword(node);
             }
             return node;
@@ -86281,7 +91362,7 @@ var ts;
             if (enabledSubstitutions & 2 /* BlockScopedBindings */ && !ts.isInternalName(node)) {
                 var declaration = resolver.getReferencedDeclarationWithCollidingName(node);
                 if (declaration && !(ts.isClassLike(declaration) && isPartOfClassBody(declaration, node))) {
-                    return ts.setTextRange(ts.getGeneratedNameForNode(ts.getNameOfDeclaration(declaration)), node);
+                    return ts.setTextRange(factory.getGeneratedNameForNode(ts.getNameOfDeclaration(declaration)), node);
                 }
             }
             return node;
@@ -86317,14 +91398,14 @@ var ts;
         function substituteThisKeyword(node) {
             if (enabledSubstitutions & 1 /* CapturedThis */
                 && hierarchyFacts & 16 /* CapturesThis */) {
-                return ts.setTextRange(ts.createFileLevelUniqueName("_this"), node);
+                return ts.setTextRange(factory.createUniqueName("_this", 16 /* Optimistic */ | 32 /* FileLevel */), node);
             }
             return node;
         }
         function getClassMemberPrefix(node, member) {
-            return ts.hasModifier(member, 32 /* Static */)
-                ? ts.getInternalName(node)
-                : ts.createPropertyAccess(ts.getInternalName(node), "prototype");
+            return ts.hasSyntacticModifier(member, 32 /* Static */)
+                ? factory.getInternalName(node)
+                : factory.createPropertyAccessExpression(factory.getInternalName(node), "prototype");
         }
         function hasSynthesizedDefaultSuperCall(constructor, hasExtendsClause) {
             if (!constructor || !hasExtendsClause) {
@@ -86334,19 +91415,19 @@ var ts;
                 return false;
             }
             var statement = ts.firstOrUndefined(constructor.body.statements);
-            if (!statement || !ts.nodeIsSynthesized(statement) || statement.kind !== 226 /* ExpressionStatement */) {
+            if (!statement || !ts.nodeIsSynthesized(statement) || statement.kind !== 233 /* ExpressionStatement */) {
                 return false;
             }
             var statementExpression = statement.expression;
-            if (!ts.nodeIsSynthesized(statementExpression) || statementExpression.kind !== 196 /* CallExpression */) {
+            if (!ts.nodeIsSynthesized(statementExpression) || statementExpression.kind !== 203 /* CallExpression */) {
                 return false;
             }
             var callTarget = statementExpression.expression;
-            if (!ts.nodeIsSynthesized(callTarget) || callTarget.kind !== 102 /* SuperKeyword */) {
+            if (!ts.nodeIsSynthesized(callTarget) || callTarget.kind !== 105 /* SuperKeyword */) {
                 return false;
             }
             var callArgument = ts.singleOrUndefined(statementExpression.arguments);
-            if (!callArgument || !ts.nodeIsSynthesized(callArgument) || callArgument.kind !== 213 /* SpreadElement */) {
+            if (!callArgument || !ts.nodeIsSynthesized(callArgument) || callArgument.kind !== 220 /* SpreadElement */) {
                 return false;
             }
             var expression = callArgument.expression;
@@ -86354,21 +91435,6 @@ var ts;
         }
     }
     ts.transformES2015 = transformES2015;
-    function createExtendsHelper(context, name) {
-        context.requestEmitHelper(ts.extendsHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__extends"), 
-        /*typeArguments*/ undefined, [
-            name,
-            ts.createFileLevelUniqueName("_super")
-        ]);
-    }
-    ts.extendsHelper = {
-        name: "typescript:extends",
-        importName: "__extends",
-        scoped: false,
-        priority: 0,
-        text: "\n            var __extends = (this && this.__extends) || (function () {\n                var extendStatics = function (d, b) {\n                    extendStatics = Object.setPrototypeOf ||\n                        ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\n                        function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };\n                    return extendStatics(d, b);\n                };\n\n                return function (d, b) {\n                    extendStatics(d, b);\n                    function __() { this.constructor = d; }\n                    d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\n                };\n            })();"
-    };
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -86379,6 +91445,7 @@ var ts;
      * @param context Context and state information for the transformation.
      */
     function transformES5(context) {
+        var factory = context.factory;
         var compilerOptions = context.getCompilerOptions();
         // enable emit notification only if using --jsx preserve or react-native
         var previousOnEmitNode;
@@ -86386,16 +91453,16 @@ var ts;
         if (compilerOptions.jsx === 1 /* Preserve */ || compilerOptions.jsx === 3 /* ReactNative */) {
             previousOnEmitNode = context.onEmitNode;
             context.onEmitNode = onEmitNode;
-            context.enableEmitNotification(268 /* JsxOpeningElement */);
-            context.enableEmitNotification(269 /* JsxClosingElement */);
-            context.enableEmitNotification(267 /* JsxSelfClosingElement */);
+            context.enableEmitNotification(275 /* JsxOpeningElement */);
+            context.enableEmitNotification(276 /* JsxClosingElement */);
+            context.enableEmitNotification(274 /* JsxSelfClosingElement */);
             noSubstitution = [];
         }
         var previousOnSubstituteNode = context.onSubstituteNode;
         context.onSubstituteNode = onSubstituteNode;
-        context.enableSubstitution(194 /* PropertyAccessExpression */);
-        context.enableSubstitution(281 /* PropertyAssignment */);
-        return ts.chainBundle(transformSourceFile);
+        context.enableSubstitution(201 /* PropertyAccessExpression */);
+        context.enableSubstitution(288 /* PropertyAssignment */);
+        return ts.chainBundle(context, transformSourceFile);
         /**
          * Transforms an ES5 source file to ES3.
          *
@@ -86413,9 +91480,9 @@ var ts;
          */
         function onEmitNode(hint, node, emitCallback) {
             switch (node.kind) {
-                case 268 /* JsxOpeningElement */:
-                case 269 /* JsxClosingElement */:
-                case 267 /* JsxSelfClosingElement */:
+                case 275 /* JsxOpeningElement */:
+                case 276 /* JsxClosingElement */:
+                case 274 /* JsxSelfClosingElement */:
                     var tagName = node.tagName;
                     noSubstitution[ts.getOriginalNodeId(tagName)] = true;
                     break;
@@ -86452,7 +91519,7 @@ var ts;
             }
             var literalName = trySubstituteReservedName(node.name);
             if (literalName) {
-                return ts.setTextRange(ts.createElementAccess(node.expression, literalName), node);
+                return ts.setTextRange(factory.createElementAccessExpression(node.expression, literalName), node);
             }
             return node;
         }
@@ -86464,7 +91531,7 @@ var ts;
         function substitutePropertyAssignment(node) {
             var literalName = ts.isIdentifier(node.name) && trySubstituteReservedName(node.name);
             if (literalName) {
-                return ts.updatePropertyAssignment(node, literalName, node.initializer);
+                return factory.updatePropertyAssignment(node, literalName, node.initializer);
             }
             return node;
         }
@@ -86475,8 +91542,8 @@ var ts;
          */
         function trySubstituteReservedName(name) {
             var token = name.originalKeywordKind || (ts.nodeIsSynthesized(name) ? ts.stringToToken(ts.idText(name)) : undefined);
-            if (token !== undefined && token >= 77 /* FirstReservedWord */ && token <= 112 /* LastReservedWord */) {
-                return ts.setTextRange(ts.createLiteral(name), name);
+            if (token !== undefined && token >= 80 /* FirstReservedWord */ && token <= 115 /* LastReservedWord */) {
+                return ts.setTextRange(factory.createStringLiteralFromNode(name), name);
             }
             return undefined;
         }
@@ -86664,7 +91731,7 @@ var ts;
         }
     }
     function transformGenerators(context) {
-        var resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistFunctionDeclaration = context.hoistFunctionDeclaration, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory, resumeLexicalEnvironment = context.resumeLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistFunctionDeclaration = context.hoistFunctionDeclaration, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var compilerOptions = context.getCompilerOptions();
         var languageVersion = ts.getEmitScriptTarget(compilerOptions);
         var resolver = context.getEmitResolver();
@@ -86711,7 +91778,7 @@ var ts;
         var exceptionBlockStack; // A stack of containing exception blocks.
         var currentExceptionBlock; // The current exception block.
         var withBlockStack; // A stack containing `with` blocks.
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile || (node.transformFlags & 512 /* ContainsGenerator */) === 0) {
                 return node;
@@ -86750,13 +91817,13 @@ var ts;
          */
         function visitJavaScriptInStatementContainingYield(node) {
             switch (node.kind) {
-                case 228 /* DoStatement */:
+                case 235 /* DoStatement */:
                     return visitDoStatement(node);
-                case 229 /* WhileStatement */:
+                case 236 /* WhileStatement */:
                     return visitWhileStatement(node);
-                case 237 /* SwitchStatement */:
+                case 244 /* SwitchStatement */:
                     return visitSwitchStatement(node);
-                case 238 /* LabeledStatement */:
+                case 245 /* LabeledStatement */:
                     return visitLabeledStatement(node);
                 default:
                     return visitJavaScriptInGeneratorFunctionBody(node);
@@ -86769,24 +91836,24 @@ var ts;
          */
         function visitJavaScriptInGeneratorFunctionBody(node) {
             switch (node.kind) {
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return visitFunctionDeclaration(node);
-                case 201 /* FunctionExpression */:
+                case 208 /* FunctionExpression */:
                     return visitFunctionExpression(node);
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return visitAccessorDeclaration(node);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return visitVariableStatement(node);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return visitForStatement(node);
-                case 231 /* ForInStatement */:
+                case 238 /* ForInStatement */:
                     return visitForInStatement(node);
-                case 234 /* BreakStatement */:
+                case 241 /* BreakStatement */:
                     return visitBreakStatement(node);
-                case 233 /* ContinueStatement */:
+                case 240 /* ContinueStatement */:
                     return visitContinueStatement(node);
-                case 235 /* ReturnStatement */:
+                case 242 /* ReturnStatement */:
                     return visitReturnStatement(node);
                 default:
                     if (node.transformFlags & 262144 /* ContainsYield */) {
@@ -86807,21 +91874,23 @@ var ts;
          */
         function visitJavaScriptContainingYield(node) {
             switch (node.kind) {
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return visitBinaryExpression(node);
-                case 210 /* ConditionalExpression */:
+                case 337 /* CommaListExpression */:
+                    return visitCommaListExpression(node);
+                case 217 /* ConditionalExpression */:
                     return visitConditionalExpression(node);
-                case 212 /* YieldExpression */:
+                case 219 /* YieldExpression */:
                     return visitYieldExpression(node);
-                case 192 /* ArrayLiteralExpression */:
+                case 199 /* ArrayLiteralExpression */:
                     return visitArrayLiteralExpression(node);
-                case 193 /* ObjectLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
                     return visitObjectLiteralExpression(node);
-                case 195 /* ElementAccessExpression */:
+                case 202 /* ElementAccessExpression */:
                     return visitElementAccessExpression(node);
-                case 196 /* CallExpression */:
+                case 203 /* CallExpression */:
                     return visitCallExpression(node);
-                case 197 /* NewExpression */:
+                case 204 /* NewExpression */:
                     return visitNewExpression(node);
                 default:
                     return ts.visitEachChild(node, visitor, context);
@@ -86834,9 +91903,9 @@ var ts;
          */
         function visitGenerator(node) {
             switch (node.kind) {
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return visitFunctionDeclaration(node);
-                case 201 /* FunctionExpression */:
+                case 208 /* FunctionExpression */:
                     return visitFunctionExpression(node);
                 default:
                     return ts.Debug.failBadSyntaxKind(node);
@@ -86854,7 +91923,7 @@ var ts;
         function visitFunctionDeclaration(node) {
             // Currently, we only support generators that were originally async functions.
             if (node.asteriskToken) {
-                node = ts.setOriginalNode(ts.setTextRange(ts.createFunctionDeclaration(
+                node = ts.setOriginalNode(ts.setTextRange(factory.createFunctionDeclaration(
                 /*decorators*/ undefined, node.modifiers, 
                 /*asteriskToken*/ undefined, node.name, 
                 /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
@@ -86892,7 +91961,7 @@ var ts;
         function visitFunctionExpression(node) {
             // Currently, we only support generators that were originally async functions.
             if (node.asteriskToken) {
-                node = ts.setOriginalNode(ts.setTextRange(ts.createFunctionExpression(
+                node = ts.setOriginalNode(ts.setTextRange(factory.createFunctionExpression(
                 /*modifiers*/ undefined, 
                 /*asteriskToken*/ undefined, node.name, 
                 /*typeParameters*/ undefined, ts.visitParameterList(node.parameters, visitor, context), 
@@ -86962,14 +92031,14 @@ var ts;
             operations = undefined;
             operationArguments = undefined;
             operationLocations = undefined;
-            state = ts.createTempVariable(/*recordTempVariable*/ undefined);
+            state = factory.createTempVariable(/*recordTempVariable*/ undefined);
             // Build the generator
             resumeLexicalEnvironment();
-            var statementOffset = ts.addPrologue(statements, body.statements, /*ensureUseStrict*/ false, visitor);
+            var statementOffset = factory.copyPrologue(body.statements, statements, /*ensureUseStrict*/ false, visitor);
             transformAndEmitStatements(body.statements, statementOffset);
             var buildResult = build();
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
-            statements.push(ts.createReturn(buildResult));
+            statements.push(factory.createReturnStatement(buildResult));
             // Restore previous generator state
             inGeneratorFunctionBody = savedInGeneratorFunctionBody;
             inStatementContainingYield = savedInStatementContainingYield;
@@ -86984,7 +92053,7 @@ var ts;
             operationArguments = savedOperationArguments;
             operationLocations = savedOperationLocations;
             state = savedState;
-            return ts.setTextRange(ts.createBlock(statements, body.multiLine), body);
+            return ts.setTextRange(factory.createBlock(statements, body.multiLine), body);
         }
         /**
          * Visits a variable statement.
@@ -87012,7 +92081,7 @@ var ts;
                 if (variables.length === 0) {
                     return undefined;
                 }
-                return ts.setSourceMapRange(ts.createExpressionStatement(ts.inlineExpressions(ts.map(variables, transformInitializedVariable))), node);
+                return ts.setSourceMapRange(factory.createExpressionStatement(factory.inlineExpressions(ts.map(variables, transformInitializedVariable))), node);
             }
         }
         /**
@@ -87044,7 +92113,7 @@ var ts;
             if (containsYield(right)) {
                 var target = void 0;
                 switch (left.kind) {
-                    case 194 /* PropertyAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
                         // [source]
                         //      a.b = yield;
                         //
@@ -87054,9 +92123,9 @@ var ts;
                         //  .yield resumeLabel
                         //  .mark resumeLabel
                         //      _a.b = %sent%;
-                        target = ts.updatePropertyAccess(left, cacheExpression(ts.visitNode(left.expression, visitor, ts.isLeftHandSideExpression)), left.name);
+                        target = factory.updatePropertyAccessExpression(left, cacheExpression(ts.visitNode(left.expression, visitor, ts.isLeftHandSideExpression)), left.name);
                         break;
-                    case 195 /* ElementAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
                         // [source]
                         //      a[b] = yield;
                         //
@@ -87067,7 +92136,7 @@ var ts;
                         //  .yield resumeLabel
                         //  .mark resumeLabel
                         //      _a[_b] = %sent%;
-                        target = ts.updateElementAccess(left, cacheExpression(ts.visitNode(left.expression, visitor, ts.isLeftHandSideExpression)), cacheExpression(ts.visitNode(left.argumentExpression, visitor, ts.isExpression)));
+                        target = factory.updateElementAccessExpression(left, cacheExpression(ts.visitNode(left.expression, visitor, ts.isLeftHandSideExpression)), cacheExpression(ts.visitNode(left.argumentExpression, visitor, ts.isExpression)));
                         break;
                     default:
                         target = ts.visitNode(left, visitor, ts.isExpression);
@@ -87075,10 +92144,10 @@ var ts;
                 }
                 var operator = node.operatorToken.kind;
                 if (ts.isCompoundAssignment(operator)) {
-                    return ts.setTextRange(ts.createAssignment(target, ts.setTextRange(ts.createBinary(cacheExpression(target), ts.getNonAssignmentOperatorForCompoundAssignment(operator), ts.visitNode(right, visitor, ts.isExpression)), node)), node);
+                    return ts.setTextRange(factory.createAssignment(target, ts.setTextRange(factory.createBinaryExpression(cacheExpression(target), ts.getNonAssignmentOperatorForCompoundAssignment(operator), ts.visitNode(right, visitor, ts.isExpression)), node)), node);
                 }
                 else {
-                    return ts.updateBinary(node, target, ts.visitNode(right, visitor, ts.isExpression));
+                    return factory.updateBinaryExpression(node, target, node.operatorToken, ts.visitNode(right, visitor, ts.isExpression));
                 }
             }
             return ts.visitEachChild(node, visitor, context);
@@ -87099,13 +92168,65 @@ var ts;
                 //      _a = a();
                 //  .yield resumeLabel
                 //      _a + %sent% + c()
-                var clone_6 = ts.getMutableClone(node);
-                clone_6.left = cacheExpression(ts.visitNode(node.left, visitor, ts.isExpression));
-                clone_6.right = ts.visitNode(node.right, visitor, ts.isExpression);
-                return clone_6;
+                return factory.updateBinaryExpression(node, cacheExpression(ts.visitNode(node.left, visitor, ts.isExpression)), node.operatorToken, ts.visitNode(node.right, visitor, ts.isExpression));
             }
             return ts.visitEachChild(node, visitor, context);
         }
+        /**
+         * Visits a comma expression containing `yield`.
+         *
+         * @param node The node to visit.
+         */
+        function visitCommaExpression(node) {
+            // [source]
+            //      x = a(), yield, b();
+            //
+            // [intermediate]
+            //      a();
+            //  .yield resumeLabel
+            //  .mark resumeLabel
+            //      x = %sent%, b();
+            var pendingExpressions = [];
+            visit(node.left);
+            visit(node.right);
+            return factory.inlineExpressions(pendingExpressions);
+            function visit(node) {
+                if (ts.isBinaryExpression(node) && node.operatorToken.kind === 27 /* CommaToken */) {
+                    visit(node.left);
+                    visit(node.right);
+                }
+                else {
+                    if (containsYield(node) && pendingExpressions.length > 0) {
+                        emitWorker(1 /* Statement */, [factory.createExpressionStatement(factory.inlineExpressions(pendingExpressions))]);
+                        pendingExpressions = [];
+                    }
+                    pendingExpressions.push(ts.visitNode(node, visitor, ts.isExpression));
+                }
+            }
+        }
+        /**
+         * Visits a comma-list expression.
+         *
+         * @param node The node to visit.
+         */
+        function visitCommaListExpression(node) {
+            // flattened version of `visitCommaExpression`
+            var pendingExpressions = [];
+            for (var _i = 0, _a = node.elements; _i < _a.length; _i++) {
+                var elem = _a[_i];
+                if (ts.isBinaryExpression(elem) && elem.operatorToken.kind === 27 /* CommaToken */) {
+                    pendingExpressions.push(visitCommaExpression(elem));
+                }
+                else {
+                    if (containsYield(elem) && pendingExpressions.length > 0) {
+                        emitWorker(1 /* Statement */, [factory.createExpressionStatement(factory.inlineExpressions(pendingExpressions))]);
+                        pendingExpressions = [];
+                    }
+                    pendingExpressions.push(ts.visitNode(elem, visitor, ts.isExpression));
+                }
+            }
+            return factory.inlineExpressions(pendingExpressions);
+        }
         /**
          * Visits a logical binary expression containing `yield`.
          *
@@ -87155,38 +92276,6 @@ var ts;
             markLabel(resultLabel);
             return resultLocal;
         }
-        /**
-         * Visits a comma expression containing `yield`.
-         *
-         * @param node The node to visit.
-         */
-        function visitCommaExpression(node) {
-            // [source]
-            //      x = a(), yield, b();
-            //
-            // [intermediate]
-            //      a();
-            //  .yield resumeLabel
-            //  .mark resumeLabel
-            //      x = %sent%, b();
-            var pendingExpressions = [];
-            visit(node.left);
-            visit(node.right);
-            return ts.inlineExpressions(pendingExpressions);
-            function visit(node) {
-                if (ts.isBinaryExpression(node) && node.operatorToken.kind === 27 /* CommaToken */) {
-                    visit(node.left);
-                    visit(node.right);
-                }
-                else {
-                    if (containsYield(node) && pendingExpressions.length > 0) {
-                        emitWorker(1 /* Statement */, [ts.createExpressionStatement(ts.inlineExpressions(pendingExpressions))]);
-                        pendingExpressions = [];
-                    }
-                    pendingExpressions.push(ts.visitNode(node, visitor, ts.isExpression));
-                }
-            }
-        }
         /**
          * Visits a conditional expression containing `yield`.
          *
@@ -87240,8 +92329,9 @@ var ts;
             var resumeLabel = defineLabel();
             var expression = ts.visitNode(node.expression, visitor, ts.isExpression);
             if (node.asteriskToken) {
+                // NOTE: `expression` must be defined for `yield*`.
                 var iterator = (ts.getEmitFlags(node.expression) & 8388608 /* Iterator */) === 0
-                    ? ts.createValuesHelper(context, expression, /*location*/ node)
+                    ? ts.setTextRange(emitHelpers().createValuesHelper(expression), node)
                     : expression;
                 emitYieldStar(iterator, /*location*/ node);
             }
@@ -87281,14 +92371,14 @@ var ts;
             if (numInitialElements > 0) {
                 temp = declareLocal();
                 var initialElements = ts.visitNodes(elements, visitor, ts.isExpression, 0, numInitialElements);
-                emitAssignment(temp, ts.createArrayLiteral(leadingElement
+                emitAssignment(temp, factory.createArrayLiteralExpression(leadingElement
                     ? __spreadArrays([leadingElement], initialElements) : initialElements));
                 leadingElement = undefined;
             }
             var expressions = ts.reduceLeft(elements, reduceElement, [], numInitialElements);
             return temp
-                ? ts.createArrayConcat(temp, [ts.createArrayLiteral(expressions, multiLine)])
-                : ts.setTextRange(ts.createArrayLiteral(leadingElement ? __spreadArrays([leadingElement], expressions) : expressions, multiLine), location);
+                ? factory.createArrayConcatCall(temp, [factory.createArrayLiteralExpression(expressions, multiLine)])
+                : ts.setTextRange(factory.createArrayLiteralExpression(leadingElement ? __spreadArrays([leadingElement], expressions) : expressions, multiLine), location);
             function reduceElement(expressions, element) {
                 if (containsYield(element) && expressions.length > 0) {
                     var hasAssignedTemp = temp !== undefined;
@@ -87296,8 +92386,8 @@ var ts;
                         temp = declareLocal();
                     }
                     emitAssignment(temp, hasAssignedTemp
-                        ? ts.createArrayConcat(temp, [ts.createArrayLiteral(expressions, multiLine)])
-                        : ts.createArrayLiteral(leadingElement ? __spreadArrays([leadingElement], expressions) : expressions, multiLine));
+                        ? factory.createArrayConcatCall(temp, [factory.createArrayLiteralExpression(expressions, multiLine)])
+                        : factory.createArrayLiteralExpression(leadingElement ? __spreadArrays([leadingElement], expressions) : expressions, multiLine));
                     leadingElement = undefined;
                     expressions = [];
                 }
@@ -87327,16 +92417,17 @@ var ts;
             var multiLine = node.multiLine;
             var numInitialProperties = countInitialNodesWithoutYield(properties);
             var temp = declareLocal();
-            emitAssignment(temp, ts.createObjectLiteral(ts.visitNodes(properties, visitor, ts.isObjectLiteralElementLike, 0, numInitialProperties), multiLine));
+            emitAssignment(temp, factory.createObjectLiteralExpression(ts.visitNodes(properties, visitor, ts.isObjectLiteralElementLike, 0, numInitialProperties), multiLine));
             var expressions = ts.reduceLeft(properties, reduceProperty, [], numInitialProperties);
-            expressions.push(multiLine ? ts.startOnNewLine(ts.getMutableClone(temp)) : temp);
-            return ts.inlineExpressions(expressions);
+            // TODO(rbuckton): Does this need to be parented?
+            expressions.push(multiLine ? ts.startOnNewLine(ts.setParent(ts.setTextRange(factory.cloneNode(temp), temp), temp.parent)) : temp);
+            return factory.inlineExpressions(expressions);
             function reduceProperty(expressions, property) {
                 if (containsYield(property) && expressions.length > 0) {
-                    emitStatement(ts.createExpressionStatement(ts.inlineExpressions(expressions)));
+                    emitStatement(factory.createExpressionStatement(factory.inlineExpressions(expressions)));
                     expressions = [];
                 }
-                var expression = ts.createExpressionForObjectLiteralElementLike(node, property, temp);
+                var expression = ts.createExpressionForObjectLiteralElementLike(factory, node, property, temp);
                 var visited = ts.visitNode(expression, visitor, ts.isExpression);
                 if (visited) {
                     if (multiLine) {
@@ -87363,10 +92454,7 @@ var ts;
                 //  .yield resumeLabel
                 //  .mark resumeLabel
                 //      a = _a[%sent%]
-                var clone_7 = ts.getMutableClone(node);
-                clone_7.expression = cacheExpression(ts.visitNode(node.expression, visitor, ts.isLeftHandSideExpression));
-                clone_7.argumentExpression = ts.visitNode(node.argumentExpression, visitor, ts.isExpression);
-                return clone_7;
+                return factory.updateElementAccessExpression(node, cacheExpression(ts.visitNode(node.expression, visitor, ts.isLeftHandSideExpression)), ts.visitNode(node.argumentExpression, visitor, ts.isExpression));
             }
             return ts.visitEachChild(node, visitor, context);
         }
@@ -87382,9 +92470,8 @@ var ts;
                 //  .yield resumeLabel
                 //  .mark resumeLabel
                 //      _b.apply(_a, _c.concat([%sent%, 2]));
-                var _a = ts.createCallBinding(node.expression, hoistVariableDeclaration, languageVersion, /*cacheIdentifiers*/ true), target = _a.target, thisArg = _a.thisArg;
-                return ts.setOriginalNode(ts.createFunctionApply(cacheExpression(ts.visitNode(target, visitor, ts.isLeftHandSideExpression)), thisArg, visitElements(node.arguments), 
-                /*location*/ node), node);
+                var _a = factory.createCallBinding(node.expression, hoistVariableDeclaration, languageVersion, /*cacheIdentifiers*/ true), target = _a.target, thisArg = _a.thisArg;
+                return ts.setOriginalNode(ts.setTextRange(factory.createFunctionApplyCall(cacheExpression(ts.visitNode(target, visitor, ts.isLeftHandSideExpression)), thisArg, visitElements(node.arguments)), node), node);
             }
             return ts.visitEachChild(node, visitor, context);
         }
@@ -87400,9 +92487,9 @@ var ts;
                 //  .yield resumeLabel
                 //  .mark resumeLabel
                 //      new (_b.apply(_a, _c.concat([%sent%, 2])));
-                var _a = ts.createCallBinding(ts.createPropertyAccess(node.expression, "bind"), hoistVariableDeclaration), target = _a.target, thisArg = _a.thisArg;
-                return ts.setOriginalNode(ts.setTextRange(ts.createNew(ts.createFunctionApply(cacheExpression(ts.visitNode(target, visitor, ts.isExpression)), thisArg, visitElements(node.arguments, 
-                /*leadingElement*/ ts.createVoidZero())), 
+                var _a = factory.createCallBinding(factory.createPropertyAccessExpression(node.expression, "bind"), hoistVariableDeclaration), target = _a.target, thisArg = _a.thisArg;
+                return ts.setOriginalNode(ts.setTextRange(factory.createNewExpression(factory.createFunctionApplyCall(cacheExpression(ts.visitNode(target, visitor, ts.isExpression)), thisArg, visitElements(node.arguments, 
+                /*leadingElement*/ factory.createVoidZero())), 
                 /*typeArguments*/ undefined, []), node), node);
             }
             return ts.visitEachChild(node, visitor, context);
@@ -87432,35 +92519,35 @@ var ts;
         }
         function transformAndEmitStatementWorker(node) {
             switch (node.kind) {
-                case 223 /* Block */:
+                case 230 /* Block */:
                     return transformAndEmitBlock(node);
-                case 226 /* ExpressionStatement */:
+                case 233 /* ExpressionStatement */:
                     return transformAndEmitExpressionStatement(node);
-                case 227 /* IfStatement */:
+                case 234 /* IfStatement */:
                     return transformAndEmitIfStatement(node);
-                case 228 /* DoStatement */:
+                case 235 /* DoStatement */:
                     return transformAndEmitDoStatement(node);
-                case 229 /* WhileStatement */:
+                case 236 /* WhileStatement */:
                     return transformAndEmitWhileStatement(node);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return transformAndEmitForStatement(node);
-                case 231 /* ForInStatement */:
+                case 238 /* ForInStatement */:
                     return transformAndEmitForInStatement(node);
-                case 233 /* ContinueStatement */:
+                case 240 /* ContinueStatement */:
                     return transformAndEmitContinueStatement(node);
-                case 234 /* BreakStatement */:
+                case 241 /* BreakStatement */:
                     return transformAndEmitBreakStatement(node);
-                case 235 /* ReturnStatement */:
+                case 242 /* ReturnStatement */:
                     return transformAndEmitReturnStatement(node);
-                case 236 /* WithStatement */:
+                case 243 /* WithStatement */:
                     return transformAndEmitWithStatement(node);
-                case 237 /* SwitchStatement */:
+                case 244 /* SwitchStatement */:
                     return transformAndEmitSwitchStatement(node);
-                case 238 /* LabeledStatement */:
+                case 245 /* LabeledStatement */:
                     return transformAndEmitLabeledStatement(node);
-                case 239 /* ThrowStatement */:
+                case 246 /* ThrowStatement */:
                     return transformAndEmitThrowStatement(node);
-                case 240 /* TryStatement */:
+                case 247 /* TryStatement */:
                     return transformAndEmitTryStatement(node);
                 default:
                     return emitStatement(ts.visitNode(node, visitor, ts.isStatement));
@@ -87480,7 +92567,7 @@ var ts;
         function transformAndEmitVariableDeclarationList(node) {
             for (var _i = 0, _a = node.declarations; _i < _a.length; _i++) {
                 var variable = _a[_i];
-                var name = ts.getSynthesizedClone(variable.name);
+                var name = factory.cloneNode(variable.name);
                 ts.setCommentRange(name, variable.name);
                 hoistVariableDeclaration(name);
             }
@@ -87497,7 +92584,7 @@ var ts;
                     pendingExpressions.push(transformInitializedVariable(variable));
                 }
                 if (pendingExpressions.length) {
-                    emitStatement(ts.createExpressionStatement(ts.inlineExpressions(pendingExpressions)));
+                    emitStatement(factory.createExpressionStatement(factory.inlineExpressions(pendingExpressions)));
                     variablesWritten += pendingExpressions.length;
                     pendingExpressions = [];
                 }
@@ -87505,7 +92592,7 @@ var ts;
             return undefined;
         }
         function transformInitializedVariable(node) {
-            return ts.setSourceMapRange(ts.createAssignment(ts.setSourceMapRange(ts.getSynthesizedClone(node.name), node.name), ts.visitNode(node.initializer, visitor, ts.isExpression)), node);
+            return ts.setSourceMapRange(factory.createAssignment(ts.setSourceMapRange(factory.cloneNode(node.name), node.name), ts.visitNode(node.initializer, visitor, ts.isExpression)), node);
         }
         function transformAndEmitIfStatement(node) {
             if (containsYield(node)) {
@@ -87648,7 +92735,7 @@ var ts;
                         transformAndEmitVariableDeclarationList(initializer);
                     }
                     else {
-                        emitStatement(ts.setTextRange(ts.createExpressionStatement(ts.visitNode(initializer, visitor, ts.isExpression)), initializer));
+                        emitStatement(ts.setTextRange(factory.createExpressionStatement(ts.visitNode(initializer, visitor, ts.isExpression)), initializer));
                     }
                 }
                 markLabel(conditionLabel);
@@ -87658,7 +92745,7 @@ var ts;
                 transformAndEmitEmbeddedStatement(node.statement);
                 markLabel(incrementLabel);
                 if (node.incrementor) {
-                    emitStatement(ts.setTextRange(ts.createExpressionStatement(ts.visitNode(node.incrementor, visitor, ts.isExpression)), node.incrementor));
+                    emitStatement(ts.setTextRange(factory.createExpressionStatement(ts.visitNode(node.incrementor, visitor, ts.isExpression)), node.incrementor));
                 }
                 emitBreak(conditionLabel);
                 endLoopBlock();
@@ -87678,9 +92765,9 @@ var ts;
                     hoistVariableDeclaration(variable.name);
                 }
                 var variables = ts.getInitializedVariables(initializer);
-                node = ts.updateFor(node, variables.length > 0
-                    ? ts.inlineExpressions(ts.map(variables, transformInitializedVariable))
-                    : undefined, ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitor, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
+                node = factory.updateForStatement(node, variables.length > 0
+                    ? factory.inlineExpressions(ts.map(variables, transformInitializedVariable))
+                    : undefined, ts.visitNode(node.condition, visitor, ts.isExpression), ts.visitNode(node.incrementor, visitor, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement, factory.liftToBlock));
             }
             else {
                 node = ts.visitEachChild(node, visitor, context);
@@ -87715,34 +92802,34 @@ var ts;
                 //  .mark endLoopLabel
                 var keysArray = declareLocal(); // _a
                 var key = declareLocal(); // _b
-                var keysIndex = ts.createLoopVariable(); // _i
+                var keysIndex = factory.createLoopVariable(); // _i
                 var initializer = node.initializer;
                 hoistVariableDeclaration(keysIndex);
-                emitAssignment(keysArray, ts.createArrayLiteral());
-                emitStatement(ts.createForIn(key, ts.visitNode(node.expression, visitor, ts.isExpression), ts.createExpressionStatement(ts.createCall(ts.createPropertyAccess(keysArray, "push"), 
+                emitAssignment(keysArray, factory.createArrayLiteralExpression());
+                emitStatement(factory.createForInStatement(key, ts.visitNode(node.expression, visitor, ts.isExpression), factory.createExpressionStatement(factory.createCallExpression(factory.createPropertyAccessExpression(keysArray, "push"), 
                 /*typeArguments*/ undefined, [key]))));
-                emitAssignment(keysIndex, ts.createLiteral(0));
+                emitAssignment(keysIndex, factory.createNumericLiteral(0));
                 var conditionLabel = defineLabel();
                 var incrementLabel = defineLabel();
                 var endLabel = beginLoopBlock(incrementLabel);
                 markLabel(conditionLabel);
-                emitBreakWhenFalse(endLabel, ts.createLessThan(keysIndex, ts.createPropertyAccess(keysArray, "length")));
+                emitBreakWhenFalse(endLabel, factory.createLessThan(keysIndex, factory.createPropertyAccessExpression(keysArray, "length")));
                 var variable = void 0;
                 if (ts.isVariableDeclarationList(initializer)) {
                     for (var _i = 0, _a = initializer.declarations; _i < _a.length; _i++) {
                         var variable_1 = _a[_i];
                         hoistVariableDeclaration(variable_1.name);
                     }
-                    variable = ts.getSynthesizedClone(initializer.declarations[0].name);
+                    variable = factory.cloneNode(initializer.declarations[0].name);
                 }
                 else {
                     variable = ts.visitNode(initializer, visitor, ts.isExpression);
                     ts.Debug.assert(ts.isLeftHandSideExpression(variable));
                 }
-                emitAssignment(variable, ts.createElementAccess(keysArray, keysIndex));
+                emitAssignment(variable, factory.createElementAccessExpression(keysArray, keysIndex));
                 transformAndEmitEmbeddedStatement(node.statement);
                 markLabel(incrementLabel);
-                emitStatement(ts.createExpressionStatement(ts.createPostfixIncrement(keysIndex)));
+                emitStatement(factory.createExpressionStatement(factory.createPostfixIncrement(keysIndex)));
                 emitBreak(conditionLabel);
                 endLoopBlock();
             }
@@ -87772,7 +92859,7 @@ var ts;
                     var variable = _a[_i];
                     hoistVariableDeclaration(variable.name);
                 }
-                node = ts.updateForIn(node, initializer.declarations[0].name, ts.visitNode(node.expression, visitor, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement, ts.liftToBlock));
+                node = factory.updateForInStatement(node, initializer.declarations[0].name, ts.visitNode(node.expression, visitor, ts.isExpression), ts.visitNode(node.statement, visitor, ts.isStatement, factory.liftToBlock));
             }
             else {
                 node = ts.visitEachChild(node, visitor, context);
@@ -87890,7 +92977,7 @@ var ts;
                 for (var i = 0; i < numClauses; i++) {
                     var clause = caseBlock.clauses[i];
                     clauseLabels.push(defineLabel());
-                    if (clause.kind === 278 /* DefaultClause */ && defaultClauseIndex === -1) {
+                    if (clause.kind === 285 /* DefaultClause */ && defaultClauseIndex === -1) {
                         defaultClauseIndex = i;
                     }
                 }
@@ -87903,11 +92990,11 @@ var ts;
                     var defaultClausesSkipped = 0;
                     for (var i = clausesWritten; i < numClauses; i++) {
                         var clause = caseBlock.clauses[i];
-                        if (clause.kind === 277 /* CaseClause */) {
+                        if (clause.kind === 284 /* CaseClause */) {
                             if (containsYield(clause.expression) && pendingClauses.length > 0) {
                                 break;
                             }
-                            pendingClauses.push(ts.createCaseClause(ts.visitNode(clause.expression, visitor, ts.isExpression), [
+                            pendingClauses.push(factory.createCaseClause(ts.visitNode(clause.expression, visitor, ts.isExpression), [
                                 createInlineBreak(clauseLabels[i], /*location*/ clause.expression)
                             ]));
                         }
@@ -87916,7 +93003,7 @@ var ts;
                         }
                     }
                     if (pendingClauses.length) {
-                        emitStatement(ts.createSwitch(expression, ts.createCaseBlock(pendingClauses)));
+                        emitStatement(factory.createSwitchStatement(expression, factory.createCaseBlock(pendingClauses)));
                         clausesWritten += pendingClauses.length;
                         pendingClauses = [];
                     }
@@ -87982,7 +93069,9 @@ var ts;
             return node;
         }
         function transformAndEmitThrowStatement(node) {
-            emitThrow(ts.visitNode(node.expression, visitor, ts.isExpression), 
+            var _a;
+            // TODO(rbuckton): `expression` should be required on `throw`.
+            emitThrow(ts.visitNode((_a = node.expression) !== null && _a !== void 0 ? _a : factory.createVoidZero(), visitor, ts.isExpression), 
             /*location*/ node);
         }
         function transformAndEmitTryStatement(node) {
@@ -88065,10 +93154,11 @@ var ts;
                     if (declaration) {
                         var name = renamedCatchVariableDeclarations[ts.getOriginalNodeId(declaration)];
                         if (name) {
-                            var clone_8 = ts.getMutableClone(name);
-                            ts.setSourceMapRange(clone_8, node);
-                            ts.setCommentRange(clone_8, node);
-                            return clone_8;
+                            // TODO(rbuckton): Does this need to be parented?
+                            var clone_5 = ts.setParent(ts.setTextRange(factory.cloneNode(name), name), name.parent);
+                            ts.setSourceMapRange(clone_5, node);
+                            ts.setCommentRange(clone_5, node);
+                            return clone_5;
                         }
                     }
                 }
@@ -88079,14 +93169,14 @@ var ts;
             if (ts.isGeneratedIdentifier(node) || ts.getEmitFlags(node) & 4096 /* HelperName */) {
                 return node;
             }
-            var temp = ts.createTempVariable(hoistVariableDeclaration);
+            var temp = factory.createTempVariable(hoistVariableDeclaration);
             emitAssignment(temp, node, /*location*/ node);
             return temp;
         }
         function declareLocal(name) {
             var temp = name
-                ? ts.createUniqueName(name)
-                : ts.createTempVariable(/*recordTempVariable*/ undefined);
+                ? factory.createUniqueName(name)
+                : factory.createTempVariable(/*recordTempVariable*/ undefined);
             hoistVariableDeclaration(temp);
             return temp;
         }
@@ -88212,9 +93302,9 @@ var ts;
                 var text = ts.idText(variable.name);
                 name = declareLocal(text);
                 if (!renamedCatchVariables) {
-                    renamedCatchVariables = ts.createMap();
+                    renamedCatchVariables = new ts.Map();
                     renamedCatchVariableDeclarations = [];
-                    context.enableSubstitution(75 /* Identifier */);
+                    context.enableSubstitution(78 /* Identifier */);
                 }
                 renamedCatchVariables.set(text, true);
                 renamedCatchVariableDeclarations[ts.getOriginalNodeId(variable)] = name;
@@ -88228,7 +93318,7 @@ var ts;
             exception.state = 1 /* Catch */;
             exception.catchVariable = name;
             exception.catchLabel = catchLabel;
-            emitAssignment(name, ts.createCall(ts.createPropertyAccess(state, "sent"), /*typeArguments*/ undefined, []));
+            emitAssignment(name, factory.createCallExpression(factory.createPropertyAccessExpression(state, "sent"), /*typeArguments*/ undefined, []));
             emitNop();
         }
         /**
@@ -88471,7 +93561,7 @@ var ts;
                 if (labelExpressions === undefined) {
                     labelExpressions = [];
                 }
-                var expression = ts.createLiteral(-1);
+                var expression = factory.createNumericLiteral(-1);
                 if (labelExpressions[label] === undefined) {
                     labelExpressions[label] = [expression];
                 }
@@ -88480,13 +93570,13 @@ var ts;
                 }
                 return expression;
             }
-            return ts.createOmittedExpression();
+            return factory.createOmittedExpression();
         }
         /**
          * Creates a numeric literal for the provided instruction.
          */
         function createInstruction(instruction) {
-            var literal = ts.createLiteral(instruction);
+            var literal = factory.createNumericLiteral(instruction);
             ts.addSyntheticTrailingComment(literal, 3 /* MultiLineCommentTrivia */, getInstructionName(instruction));
             return literal;
         }
@@ -88498,7 +93588,7 @@ var ts;
          */
         function createInlineBreak(label, location) {
             ts.Debug.assertLessThan(0, label, "Invalid label");
-            return ts.setTextRange(ts.createReturn(ts.createArrayLiteral([
+            return ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression([
                 createInstruction(3 /* Break */),
                 createLabel(label)
             ])), location);
@@ -88510,7 +93600,7 @@ var ts;
          * @param location An optional source map location for the statement.
          */
         function createInlineReturn(expression, location) {
-            return ts.setTextRange(ts.createReturn(ts.createArrayLiteral(expression
+            return ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression(expression
                 ? [createInstruction(2 /* Return */), expression]
                 : [createInstruction(2 /* Return */)])), location);
         }
@@ -88518,7 +93608,7 @@ var ts;
          * Creates an expression that can be used to resume from a Yield operation.
          */
         function createGeneratorResume(location) {
-            return ts.setTextRange(ts.createCall(ts.createPropertyAccess(state, "sent"), 
+            return ts.setTextRange(factory.createCallExpression(factory.createPropertyAccessExpression(state, "sent"), 
             /*typeArguments*/ undefined, []), location);
         }
         /**
@@ -88659,12 +93749,12 @@ var ts;
             currentExceptionBlock = undefined;
             withBlockStack = undefined;
             var buildResult = buildStatements();
-            return createGeneratorHelper(context, ts.setEmitFlags(ts.createFunctionExpression(
+            return emitHelpers().createGeneratorHelper(ts.setEmitFlags(factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
-            /*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, state)], 
-            /*type*/ undefined, ts.createBlock(buildResult, 
+            /*typeParameters*/ undefined, [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, state)], 
+            /*type*/ undefined, factory.createBlock(buildResult, 
             /*multiLine*/ buildResult.length > 0)), 524288 /* ReuseTempVariableScope */));
         }
         /**
@@ -88681,8 +93771,8 @@ var ts;
                 flushFinalLabel(0);
             }
             if (clauses) {
-                var labelExpression = ts.createPropertyAccess(state, "label");
-                var switchStatement = ts.createSwitch(labelExpression, ts.createCaseBlock(clauses));
+                var labelExpression = factory.createPropertyAccessExpression(state, "label");
+                var switchStatement = factory.createSwitchStatement(labelExpression, factory.createCaseBlock(clauses));
                 return [ts.startOnNewLine(switchStatement)];
             }
             if (statements) {
@@ -88757,7 +93847,7 @@ var ts;
                     // surround the statements in generated `with` blocks to create the same environment.
                     for (var i = withBlockStack.length - 1; i >= 0; i--) {
                         var withBlock = withBlockStack[i];
-                        statements = [ts.createWith(withBlock.expression, ts.createBlock(statements))];
+                        statements = [factory.createWithStatement(withBlock.expression, factory.createBlock(statements))];
                     }
                 }
                 if (currentExceptionBlock) {
@@ -88765,9 +93855,9 @@ var ts;
                     // indicate entry into a protected region by pushing the label numbers
                     // for each block in the protected region.
                     var startLabel = currentExceptionBlock.startLabel, catchLabel = currentExceptionBlock.catchLabel, finallyLabel = currentExceptionBlock.finallyLabel, endLabel = currentExceptionBlock.endLabel;
-                    statements.unshift(ts.createExpressionStatement(ts.createCall(ts.createPropertyAccess(ts.createPropertyAccess(state, "trys"), "push"), 
+                    statements.unshift(factory.createExpressionStatement(factory.createCallExpression(factory.createPropertyAccessExpression(factory.createPropertyAccessExpression(state, "trys"), "push"), 
                     /*typeArguments*/ undefined, [
-                        ts.createArrayLiteral([
+                        factory.createArrayLiteralExpression([
                             createLabel(startLabel),
                             createLabel(catchLabel),
                             createLabel(finallyLabel),
@@ -88779,10 +93869,10 @@ var ts;
                 if (markLabelEnd) {
                     // The case clause for the last label falls through to this label, so we
                     // add an assignment statement to reflect the change in labels.
-                    statements.push(ts.createExpressionStatement(ts.createAssignment(ts.createPropertyAccess(state, "label"), ts.createLiteral(labelNumber + 1))));
+                    statements.push(factory.createExpressionStatement(factory.createAssignment(factory.createPropertyAccessExpression(state, "label"), factory.createNumericLiteral(labelNumber + 1))));
                 }
             }
-            clauses.push(ts.createCaseClause(ts.createLiteral(labelNumber), statements || []));
+            clauses.push(factory.createCaseClause(factory.createNumericLiteral(labelNumber), statements || []));
             statements = undefined;
         }
         /**
@@ -88937,7 +94027,7 @@ var ts;
          * @param operationLocation The source map location for the operation.
          */
         function writeAssign(left, right, operationLocation) {
-            writeStatement(ts.setTextRange(ts.createExpressionStatement(ts.createAssignment(left, right)), operationLocation));
+            writeStatement(ts.setTextRange(factory.createExpressionStatement(factory.createAssignment(left, right)), operationLocation));
         }
         /**
          * Writes a Throw operation to the current label's statement list.
@@ -88948,7 +94038,7 @@ var ts;
         function writeThrow(expression, operationLocation) {
             lastOperationWasAbrupt = true;
             lastOperationWasCompletion = true;
-            writeStatement(ts.setTextRange(ts.createThrow(expression), operationLocation));
+            writeStatement(ts.setTextRange(factory.createThrowStatement(expression), operationLocation));
         }
         /**
          * Writes a Return operation to the current label's statement list.
@@ -88959,7 +94049,7 @@ var ts;
         function writeReturn(expression, operationLocation) {
             lastOperationWasAbrupt = true;
             lastOperationWasCompletion = true;
-            writeStatement(ts.setEmitFlags(ts.setTextRange(ts.createReturn(ts.createArrayLiteral(expression
+            writeStatement(ts.setEmitFlags(ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression(expression
                 ? [createInstruction(2 /* Return */), expression]
                 : [createInstruction(2 /* Return */)])), operationLocation), 384 /* NoTokenSourceMaps */));
         }
@@ -88971,7 +94061,7 @@ var ts;
          */
         function writeBreak(label, operationLocation) {
             lastOperationWasAbrupt = true;
-            writeStatement(ts.setEmitFlags(ts.setTextRange(ts.createReturn(ts.createArrayLiteral([
+            writeStatement(ts.setEmitFlags(ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression([
                 createInstruction(3 /* Break */),
                 createLabel(label)
             ])), operationLocation), 384 /* NoTokenSourceMaps */));
@@ -88984,7 +94074,7 @@ var ts;
          * @param operationLocation The source map location for the operation.
          */
         function writeBreakWhenTrue(label, condition, operationLocation) {
-            writeStatement(ts.setEmitFlags(ts.createIf(condition, ts.setEmitFlags(ts.setTextRange(ts.createReturn(ts.createArrayLiteral([
+            writeStatement(ts.setEmitFlags(factory.createIfStatement(condition, ts.setEmitFlags(ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression([
                 createInstruction(3 /* Break */),
                 createLabel(label)
             ])), operationLocation), 384 /* NoTokenSourceMaps */)), 1 /* SingleLine */));
@@ -88997,7 +94087,7 @@ var ts;
          * @param operationLocation The source map location for the operation.
          */
         function writeBreakWhenFalse(label, condition, operationLocation) {
-            writeStatement(ts.setEmitFlags(ts.createIf(ts.createLogicalNot(condition), ts.setEmitFlags(ts.setTextRange(ts.createReturn(ts.createArrayLiteral([
+            writeStatement(ts.setEmitFlags(factory.createIfStatement(factory.createLogicalNot(condition), ts.setEmitFlags(ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression([
                 createInstruction(3 /* Break */),
                 createLabel(label)
             ])), operationLocation), 384 /* NoTokenSourceMaps */)), 1 /* SingleLine */));
@@ -89010,7 +94100,7 @@ var ts;
          */
         function writeYield(expression, operationLocation) {
             lastOperationWasAbrupt = true;
-            writeStatement(ts.setEmitFlags(ts.setTextRange(ts.createReturn(ts.createArrayLiteral(expression
+            writeStatement(ts.setEmitFlags(ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression(expression
                 ? [createInstruction(4 /* Yield */), expression]
                 : [createInstruction(4 /* Yield */)])), operationLocation), 384 /* NoTokenSourceMaps */));
         }
@@ -89022,7 +94112,7 @@ var ts;
          */
         function writeYieldStar(expression, operationLocation) {
             lastOperationWasAbrupt = true;
-            writeStatement(ts.setEmitFlags(ts.setTextRange(ts.createReturn(ts.createArrayLiteral([
+            writeStatement(ts.setEmitFlags(ts.setTextRange(factory.createReturnStatement(factory.createArrayLiteralExpression([
                 createInstruction(5 /* YieldStar */),
                 expression
             ])), operationLocation), 384 /* NoTokenSourceMaps */));
@@ -89032,83 +94122,12 @@ var ts;
          */
         function writeEndfinally() {
             lastOperationWasAbrupt = true;
-            writeStatement(ts.createReturn(ts.createArrayLiteral([
+            writeStatement(factory.createReturnStatement(factory.createArrayLiteralExpression([
                 createInstruction(7 /* Endfinally */)
             ])));
         }
     }
     ts.transformGenerators = transformGenerators;
-    function createGeneratorHelper(context, body) {
-        context.requestEmitHelper(ts.generatorHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__generator"), 
-        /*typeArguments*/ undefined, [ts.createThis(), body]);
-    }
-    // The __generator helper is used by down-level transformations to emulate the runtime
-    // semantics of an ES2015 generator function. When called, this helper returns an
-    // object that implements the Iterator protocol, in that it has `next`, `return`, and
-    // `throw` methods that step through the generator when invoked.
-    //
-    // parameters:
-    //  @param thisArg  The value to use as the `this` binding for the transformed generator body.
-    //  @param body     A function that acts as the transformed generator body.
-    //
-    // variables:
-    //  _       Persistent state for the generator that is shared between the helper and the
-    //          generator body. The state object has the following members:
-    //            sent() - A method that returns or throws the current completion value.
-    //            label  - The next point at which to resume evaluation of the generator body.
-    //            trys   - A stack of protected regions (try/catch/finally blocks).
-    //            ops    - A stack of pending instructions when inside of a finally block.
-    //  f       A value indicating whether the generator is executing.
-    //  y       An iterator to delegate for a yield*.
-    //  t       A temporary variable that holds one of the following values (note that these
-    //          cases do not overlap):
-    //          - The completion value when resuming from a `yield` or `yield*`.
-    //          - The error value for a catch block.
-    //          - The current protected region (array of try/catch/finally/end labels).
-    //          - The verb (`next`, `throw`, or `return` method) to delegate to the expression
-    //            of a `yield*`.
-    //          - The result of evaluating the verb delegated to the expression of a `yield*`.
-    //
-    // functions:
-    //  verb(n)     Creates a bound callback to the `step` function for opcode `n`.
-    //  step(op)    Evaluates opcodes in a generator body until execution is suspended or
-    //              completed.
-    //
-    // The __generator helper understands a limited set of instructions:
-    //  0: next(value?)     - Start or resume the generator with the specified value.
-    //  1: throw(error)     - Resume the generator with an exception. If the generator is
-    //                        suspended inside of one or more protected regions, evaluates
-    //                        any intervening finally blocks between the current label and
-    //                        the nearest catch block or function boundary. If uncaught, the
-    //                        exception is thrown to the caller.
-    //  2: return(value?)   - Resume the generator as if with a return. If the generator is
-    //                        suspended inside of one or more protected regions, evaluates any
-    //                        intervening finally blocks.
-    //  3: break(label)     - Jump to the specified label. If the label is outside of the
-    //                        current protected region, evaluates any intervening finally
-    //                        blocks.
-    //  4: yield(value?)    - Yield execution to the caller with an optional value. When
-    //                        resumed, the generator will continue at the next label.
-    //  5: yield*(value)    - Delegates evaluation to the supplied iterator. When
-    //                        delegation completes, the generator will continue at the next
-    //                        label.
-    //  6: catch(error)     - Handles an exception thrown from within the generator body. If
-    //                        the current label is inside of one or more protected regions,
-    //                        evaluates any intervening finally blocks between the current
-    //                        label and the nearest catch block or function boundary. If
-    //                        uncaught, the exception is thrown to the caller.
-    //  7: endfinally       - Ends a finally block, resuming the last instruction prior to
-    //                        entering a finally block.
-    //
-    // For examples of how these are used, see the comments in ./transformers/generators.ts
-    ts.generatorHelper = {
-        name: "typescript:generator",
-        importName: "__generator",
-        scoped: false,
-        priority: 6,
-        text: "\n            var __generator = (this && this.__generator) || function (thisArg, body) {\n                var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\n                return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n                function verb(n) { return function (v) { return step([n, v]); }; }\n                function step(op) {\n                    if (f) throw new TypeError(\"Generator is already executing.\");\n                    while (_) try {\n                        if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n                        if (y = 0, t) op = [op[0] & 2, t.value];\n                        switch (op[0]) {\n                            case 0: case 1: t = op; break;\n                            case 4: _.label++; return { value: op[1], done: false };\n                            case 5: _.label++; y = op[1]; op = [0]; continue;\n                            case 7: op = _.ops.pop(); _.trys.pop(); continue;\n                            default:\n                                if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n                                if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n                                if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n                                if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n                                if (t[2]) _.ops.pop();\n                                _.trys.pop(); continue;\n                        }\n                        op = body.call(thisArg, _);\n                    } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n                    if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n                }\n            };"
-    };
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -89121,7 +94140,7 @@ var ts;
                 default: return transformCommonJSModule;
             }
         }
-        var startLexicalEnvironment = context.startLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory, startLexicalEnvironment = context.startLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var compilerOptions = context.getCompilerOptions();
         var resolver = context.getEmitResolver();
         var host = context.getEmitHost();
@@ -89131,19 +94150,19 @@ var ts;
         var previousOnEmitNode = context.onEmitNode;
         context.onSubstituteNode = onSubstituteNode;
         context.onEmitNode = onEmitNode;
-        context.enableSubstitution(75 /* Identifier */); // Substitutes expression identifiers with imported/exported symbols.
-        context.enableSubstitution(209 /* BinaryExpression */); // Substitutes assignments to exported symbols.
-        context.enableSubstitution(207 /* PrefixUnaryExpression */); // Substitutes updates to exported symbols.
-        context.enableSubstitution(208 /* PostfixUnaryExpression */); // Substitutes updates to exported symbols.
-        context.enableSubstitution(282 /* ShorthandPropertyAssignment */); // Substitutes shorthand property assignments for imported/exported symbols.
-        context.enableEmitNotification(290 /* SourceFile */); // Restore state when substituting nodes in a file.
+        context.enableSubstitution(78 /* Identifier */); // Substitutes expression identifiers with imported/exported symbols.
+        context.enableSubstitution(216 /* BinaryExpression */); // Substitutes assignments to exported symbols.
+        context.enableSubstitution(214 /* PrefixUnaryExpression */); // Substitutes updates to exported symbols.
+        context.enableSubstitution(215 /* PostfixUnaryExpression */); // Substitutes updates to exported symbols.
+        context.enableSubstitution(289 /* ShorthandPropertyAssignment */); // Substitutes shorthand property assignments for imported/exported symbols.
+        context.enableEmitNotification(297 /* SourceFile */); // Restore state when substituting nodes in a file.
         var moduleInfoMap = []; // The ExternalModuleInfo for each file.
         var deferredExports = []; // Exports to defer until an EndOfDeclarationMarker is found.
         var currentSourceFile; // The current file.
         var currentModuleInfo; // The ExternalModuleInfo for the current file.
         var noSubstitution; // Set of nodes for which substitution rules should be ignored.
         var needUMDDynamicImportHelper;
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         /**
          * Transforms the module aspects of a SourceFile.
          *
@@ -89153,11 +94172,11 @@ var ts;
             if (node.isDeclarationFile ||
                 !(ts.isEffectiveExternalModule(node, compilerOptions) ||
                     node.transformFlags & 2097152 /* ContainsDynamicImport */ ||
-                    (ts.isJsonSourceFile(node) && ts.hasJsonModuleEmitEnabled(compilerOptions) && (compilerOptions.out || compilerOptions.outFile)))) {
+                    (ts.isJsonSourceFile(node) && ts.hasJsonModuleEmitEnabled(compilerOptions) && ts.outFile(compilerOptions)))) {
                 return node;
             }
             currentSourceFile = node;
-            currentModuleInfo = ts.collectExternalModuleInfo(node, resolver, compilerOptions);
+            currentModuleInfo = ts.collectExternalModuleInfo(context, node, resolver, compilerOptions);
             moduleInfoMap[ts.getOriginalNodeId(node)] = currentModuleInfo;
             // Perform the transformation.
             var transformModule = getTransformModuleDelegate(moduleKind);
@@ -89165,7 +94184,7 @@ var ts;
             currentSourceFile = undefined;
             currentModuleInfo = undefined;
             needUMDDynamicImportHelper = false;
-            return ts.aggregateTransformFlags(updated);
+            return updated;
         }
         function shouldEmitUnderscoreUnderscoreESModule() {
             if (!currentModuleInfo.exportEquals && ts.isExternalModule(currentSourceFile)) {
@@ -89182,18 +94201,21 @@ var ts;
             startLexicalEnvironment();
             var statements = [];
             var ensureUseStrict = ts.getStrictOptionValue(compilerOptions, "alwaysStrict") || (!compilerOptions.noImplicitUseStrict && ts.isExternalModule(currentSourceFile));
-            var statementOffset = ts.addPrologue(statements, node.statements, ensureUseStrict && !ts.isJsonSourceFile(node), sourceElementVisitor);
+            var statementOffset = factory.copyPrologue(node.statements, statements, ensureUseStrict && !ts.isJsonSourceFile(node), sourceElementVisitor);
             if (shouldEmitUnderscoreUnderscoreESModule()) {
                 ts.append(statements, createUnderscoreUnderscoreESModule());
             }
             if (ts.length(currentModuleInfo.exportedNames)) {
-                ts.append(statements, ts.createExpressionStatement(ts.reduceLeft(currentModuleInfo.exportedNames, function (prev, nextId) { return ts.createAssignment(ts.createPropertyAccess(ts.createIdentifier("exports"), ts.createIdentifier(ts.idText(nextId))), prev); }, ts.createVoidZero())));
+                var chunkSize = 50;
+                for (var i = 0; i < currentModuleInfo.exportedNames.length; i += chunkSize) {
+                    ts.append(statements, factory.createExpressionStatement(ts.reduceLeft(currentModuleInfo.exportedNames.slice(i, i + chunkSize), function (prev, nextId) { return factory.createAssignment(factory.createPropertyAccessExpression(factory.createIdentifier("exports"), factory.createIdentifier(ts.idText(nextId))), prev); }, factory.createVoidZero())));
+                }
             }
             ts.append(statements, ts.visitNode(currentModuleInfo.externalHelpersImportDeclaration, sourceElementVisitor, ts.isStatement));
             ts.addRange(statements, ts.visitNodes(node.statements, sourceElementVisitor, ts.isStatement, statementOffset));
             addExportEqualsIfNeeded(statements, /*emitAsReturn*/ false);
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
-            var updated = ts.updateSourceFileNode(node, ts.setTextRange(ts.createNodeArray(statements), node.statements));
+            var updated = factory.updateSourceFile(node, ts.setTextRange(factory.createNodeArray(statements), node.statements));
             ts.addEmitHelpers(updated, context.readEmitHelpers());
             return updated;
         }
@@ -89203,8 +94225,8 @@ var ts;
          * @param node The SourceFile node.
          */
         function transformAMDModule(node) {
-            var define = ts.createIdentifier("define");
-            var moduleName = ts.tryGetModuleNameFromFile(node, host, compilerOptions);
+            var define = factory.createIdentifier("define");
+            var moduleName = ts.tryGetModuleNameFromFile(factory, node, host, compilerOptions);
             var jsonSourceFile = ts.isJsonSourceFile(node) && node;
             // An AMD define function has the following shape:
             //
@@ -89229,29 +94251,29 @@ var ts;
             var _a = collectAsynchronousDependencies(node, /*includeNonAmdDependencies*/ true), aliasedModuleNames = _a.aliasedModuleNames, unaliasedModuleNames = _a.unaliasedModuleNames, importAliasNames = _a.importAliasNames;
             // Create an updated SourceFile:
             //
-            //     define(moduleName?, ["module1", "module2"], function ...
-            var updated = ts.updateSourceFileNode(node, ts.setTextRange(ts.createNodeArray([
-                ts.createExpressionStatement(ts.createCall(define, 
+            //     define(mofactory.updateSourceFile", "module2"], function ...
+            var updated = factory.updateSourceFile(node, ts.setTextRange(factory.createNodeArray([
+                factory.createExpressionStatement(factory.createCallExpression(define, 
                 /*typeArguments*/ undefined, __spreadArrays((moduleName ? [moduleName] : []), [
                     // Add the dependency array argument:
                     //
                     //     ["require", "exports", module1", "module2", ...]
-                    ts.createArrayLiteral(jsonSourceFile ? ts.emptyArray : __spreadArrays([
-                        ts.createLiteral("require"),
-                        ts.createLiteral("exports")
+                    factory.createArrayLiteralExpression(jsonSourceFile ? ts.emptyArray : __spreadArrays([
+                        factory.createStringLiteral("require"),
+                        factory.createStringLiteral("exports")
                     ], aliasedModuleNames, unaliasedModuleNames)),
                     // Add the module body function argument:
                     //
                     //     function (require, exports, module1, module2) ...
                     jsonSourceFile ?
-                        jsonSourceFile.statements.length ? jsonSourceFile.statements[0].expression : ts.createObjectLiteral() :
-                        ts.createFunctionExpression(
+                        jsonSourceFile.statements.length ? jsonSourceFile.statements[0].expression : factory.createObjectLiteralExpression() :
+                        factory.createFunctionExpression(
                         /*modifiers*/ undefined, 
                         /*asteriskToken*/ undefined, 
                         /*name*/ undefined, 
                         /*typeParameters*/ undefined, __spreadArrays([
-                            ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "require"),
-                            ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "exports")
+                            factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "require"),
+                            factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "exports")
                         ], importAliasNames), 
                         /*type*/ undefined, transformAsynchronousModuleBody(node))
                 ])))
@@ -89267,32 +94289,33 @@ var ts;
          */
         function transformUMDModule(node) {
             var _a = collectAsynchronousDependencies(node, /*includeNonAmdDependencies*/ false), aliasedModuleNames = _a.aliasedModuleNames, unaliasedModuleNames = _a.unaliasedModuleNames, importAliasNames = _a.importAliasNames;
-            var moduleName = ts.tryGetModuleNameFromFile(node, host, compilerOptions);
-            var umdHeader = ts.createFunctionExpression(
+            var moduleName = ts.tryGetModuleNameFromFile(factory, node, host, compilerOptions);
+            var umdHeader = factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
-            /*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "factory")], 
-            /*type*/ undefined, ts.setTextRange(ts.createBlock([
-                ts.createIf(ts.createLogicalAnd(ts.createTypeCheck(ts.createIdentifier("module"), "object"), ts.createTypeCheck(ts.createPropertyAccess(ts.createIdentifier("module"), "exports"), "object")), ts.createBlock([
-                    ts.createVariableStatement(
+            /*typeParameters*/ undefined, [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "factory")], 
+            /*type*/ undefined, ts.setTextRange(factory.createBlock([
+                factory.createIfStatement(factory.createLogicalAnd(factory.createTypeCheck(factory.createIdentifier("module"), "object"), factory.createTypeCheck(factory.createPropertyAccessExpression(factory.createIdentifier("module"), "exports"), "object")), factory.createBlock([
+                    factory.createVariableStatement(
                     /*modifiers*/ undefined, [
-                        ts.createVariableDeclaration("v", 
-                        /*type*/ undefined, ts.createCall(ts.createIdentifier("factory"), 
+                        factory.createVariableDeclaration("v", 
+                        /*exclamationToken*/ undefined, 
+                        /*type*/ undefined, factory.createCallExpression(factory.createIdentifier("factory"), 
                         /*typeArguments*/ undefined, [
-                            ts.createIdentifier("require"),
-                            ts.createIdentifier("exports")
+                            factory.createIdentifier("require"),
+                            factory.createIdentifier("exports")
                         ]))
                     ]),
-                    ts.setEmitFlags(ts.createIf(ts.createStrictInequality(ts.createIdentifier("v"), ts.createIdentifier("undefined")), ts.createExpressionStatement(ts.createAssignment(ts.createPropertyAccess(ts.createIdentifier("module"), "exports"), ts.createIdentifier("v")))), 1 /* SingleLine */)
-                ]), ts.createIf(ts.createLogicalAnd(ts.createTypeCheck(ts.createIdentifier("define"), "function"), ts.createPropertyAccess(ts.createIdentifier("define"), "amd")), ts.createBlock([
-                    ts.createExpressionStatement(ts.createCall(ts.createIdentifier("define"), 
+                    ts.setEmitFlags(factory.createIfStatement(factory.createStrictInequality(factory.createIdentifier("v"), factory.createIdentifier("undefined")), factory.createExpressionStatement(factory.createAssignment(factory.createPropertyAccessExpression(factory.createIdentifier("module"), "exports"), factory.createIdentifier("v")))), 1 /* SingleLine */)
+                ]), factory.createIfStatement(factory.createLogicalAnd(factory.createTypeCheck(factory.createIdentifier("define"), "function"), factory.createPropertyAccessExpression(factory.createIdentifier("define"), "amd")), factory.createBlock([
+                    factory.createExpressionStatement(factory.createCallExpression(factory.createIdentifier("define"), 
                     /*typeArguments*/ undefined, __spreadArrays((moduleName ? [moduleName] : []), [
-                        ts.createArrayLiteral(__spreadArrays([
-                            ts.createLiteral("require"),
-                            ts.createLiteral("exports")
+                        factory.createArrayLiteralExpression(__spreadArrays([
+                            factory.createStringLiteral("require"),
+                            factory.createStringLiteral("exports")
                         ], aliasedModuleNames, unaliasedModuleNames)),
-                        ts.createIdentifier("factory")
+                        factory.createIdentifier("factory")
                     ])))
                 ])))
             ], 
@@ -89309,19 +94332,19 @@ var ts;
             //          define(["require", "exports"], factory);
             //      }
             //  })(function ...)
-            var updated = ts.updateSourceFileNode(node, ts.setTextRange(ts.createNodeArray([
-                ts.createExpressionStatement(ts.createCall(umdHeader, 
+            var updated = factory.updateSourceFile(node, ts.setTextRange(factory.createNodeArray([
+                factory.createExpressionStatement(factory.createCallExpression(umdHeader, 
                 /*typeArguments*/ undefined, [
                     // Add the module body function argument:
                     //
                     //     function (require, exports) ...
-                    ts.createFunctionExpression(
+                    factory.createFunctionExpression(
                     /*modifiers*/ undefined, 
                     /*asteriskToken*/ undefined, 
                     /*name*/ undefined, 
                     /*typeParameters*/ undefined, __spreadArrays([
-                        ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "require"),
-                        ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "exports")
+                        factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "require"),
+                        factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, "exports")
                     ], importAliasNames), 
                     /*type*/ undefined, transformAsynchronousModuleBody(node))
                 ]))
@@ -89349,19 +94372,19 @@ var ts;
             for (var _i = 0, _a = node.amdDependencies; _i < _a.length; _i++) {
                 var amdDependency = _a[_i];
                 if (amdDependency.name) {
-                    aliasedModuleNames.push(ts.createLiteral(amdDependency.path));
-                    importAliasNames.push(ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, amdDependency.name));
+                    aliasedModuleNames.push(factory.createStringLiteral(amdDependency.path));
+                    importAliasNames.push(factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, amdDependency.name));
                 }
                 else {
-                    unaliasedModuleNames.push(ts.createLiteral(amdDependency.path));
+                    unaliasedModuleNames.push(factory.createStringLiteral(amdDependency.path));
                 }
             }
             for (var _b = 0, _c = currentModuleInfo.externalImports; _b < _c.length; _b++) {
                 var importNode = _c[_b];
                 // Find the name of the external module
-                var externalModuleName = ts.getExternalModuleNameLiteral(importNode, currentSourceFile, host, resolver, compilerOptions);
+                var externalModuleName = ts.getExternalModuleNameLiteral(factory, importNode, currentSourceFile, host, resolver, compilerOptions);
                 // Find the name of the module alias, if there is one
-                var importAliasName = ts.getLocalNameForExternalImport(importNode, currentSourceFile);
+                var importAliasName = ts.getLocalNameForExternalImport(factory, importNode, currentSourceFile);
                 // It is possible that externalModuleName is undefined if it is not string literal.
                 // This can happen in the invalid import syntax.
                 // E.g : "import * from alias from 'someLib';"
@@ -89371,7 +94394,7 @@ var ts;
                         // This is so that when printer will not substitute the identifier
                         ts.setEmitFlags(importAliasName, 4 /* NoSubstitution */);
                         aliasedModuleNames.push(externalModuleName);
-                        importAliasNames.push(ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, importAliasName));
+                        importAliasNames.push(factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, importAliasName));
                     }
                     else {
                         unaliasedModuleNames.push(externalModuleName);
@@ -89381,15 +94404,15 @@ var ts;
             return { aliasedModuleNames: aliasedModuleNames, unaliasedModuleNames: unaliasedModuleNames, importAliasNames: importAliasNames };
         }
         function getAMDImportExpressionForImport(node) {
-            if (ts.isImportEqualsDeclaration(node) || ts.isExportDeclaration(node) || !ts.getExternalModuleNameLiteral(node, currentSourceFile, host, resolver, compilerOptions)) {
+            if (ts.isImportEqualsDeclaration(node) || ts.isExportDeclaration(node) || !ts.getExternalModuleNameLiteral(factory, node, currentSourceFile, host, resolver, compilerOptions)) {
                 return undefined;
             }
-            var name = ts.getLocalNameForExternalImport(node, currentSourceFile); // TODO: GH#18217
+            var name = ts.getLocalNameForExternalImport(factory, node, currentSourceFile); // TODO: GH#18217
             var expr = getHelperExpressionForImport(node, name);
             if (expr === name) {
                 return undefined;
             }
-            return ts.createExpressionStatement(ts.createAssignment(name, expr));
+            return factory.createExpressionStatement(factory.createAssignment(name, expr));
         }
         /**
          * Transforms a SourceFile into an AMD or UMD module body.
@@ -89399,12 +94422,12 @@ var ts;
         function transformAsynchronousModuleBody(node) {
             startLexicalEnvironment();
             var statements = [];
-            var statementOffset = ts.addPrologue(statements, node.statements, /*ensureUseStrict*/ !compilerOptions.noImplicitUseStrict, sourceElementVisitor);
+            var statementOffset = factory.copyPrologue(node.statements, statements, /*ensureUseStrict*/ !compilerOptions.noImplicitUseStrict, sourceElementVisitor);
             if (shouldEmitUnderscoreUnderscoreESModule()) {
                 ts.append(statements, createUnderscoreUnderscoreESModule());
             }
             if (ts.length(currentModuleInfo.exportedNames)) {
-                ts.append(statements, ts.createExpressionStatement(ts.reduceLeft(currentModuleInfo.exportedNames, function (prev, nextId) { return ts.createAssignment(ts.createPropertyAccess(ts.createIdentifier("exports"), ts.createIdentifier(ts.idText(nextId))), prev); }, ts.createVoidZero())));
+                ts.append(statements, factory.createExpressionStatement(ts.reduceLeft(currentModuleInfo.exportedNames, function (prev, nextId) { return factory.createAssignment(factory.createPropertyAccessExpression(factory.createIdentifier("exports"), factory.createIdentifier(ts.idText(nextId))), prev); }, factory.createVoidZero())));
             }
             // Visit each statement of the module body.
             ts.append(statements, ts.visitNode(currentModuleInfo.externalHelpersImportDeclaration, sourceElementVisitor, ts.isStatement));
@@ -89417,7 +94440,7 @@ var ts;
             // End the lexical environment for the module body
             // and merge any new lexical declarations.
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
-            var body = ts.createBlock(statements, /*multiLine*/ true);
+            var body = factory.createBlock(statements, /*multiLine*/ true);
             if (needUMDDynamicImportHelper) {
                 ts.addEmitHelper(body, dynamicImportUMDHelper);
             }
@@ -89436,13 +94459,13 @@ var ts;
                 var expressionResult = ts.visitNode(currentModuleInfo.exportEquals.expression, moduleExpressionElementVisitor);
                 if (expressionResult) {
                     if (emitAsReturn) {
-                        var statement = ts.createReturn(expressionResult);
+                        var statement = factory.createReturnStatement(expressionResult);
                         ts.setTextRange(statement, currentModuleInfo.exportEquals);
                         ts.setEmitFlags(statement, 384 /* NoTokenSourceMaps */ | 1536 /* NoComments */);
                         statements.push(statement);
                     }
                     else {
-                        var statement = ts.createExpressionStatement(ts.createAssignment(ts.createPropertyAccess(ts.createIdentifier("module"), "exports"), expressionResult));
+                        var statement = factory.createExpressionStatement(factory.createAssignment(factory.createPropertyAccessExpression(factory.createIdentifier("module"), "exports"), expressionResult));
                         ts.setTextRange(statement, currentModuleInfo.exportEquals);
                         ts.setEmitFlags(statement, 1536 /* NoComments */);
                         statements.push(statement);
@@ -89460,23 +94483,23 @@ var ts;
          */
         function sourceElementVisitor(node) {
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     return visitImportDeclaration(node);
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return visitImportEqualsDeclaration(node);
-                case 260 /* ExportDeclaration */:
+                case 267 /* ExportDeclaration */:
                     return visitExportDeclaration(node);
-                case 259 /* ExportAssignment */:
+                case 266 /* ExportAssignment */:
                     return visitExportAssignment(node);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return visitVariableStatement(node);
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return visitFunctionDeclaration(node);
-                case 245 /* ClassDeclaration */:
+                case 252 /* ClassDeclaration */:
                     return visitClassDeclaration(node);
-                case 328 /* MergeDeclarationMarker */:
+                case 338 /* MergeDeclarationMarker */:
                     return visitMergeDeclarationMarker(node);
-                case 329 /* EndOfDeclarationMarker */:
+                case 339 /* EndOfDeclarationMarker */:
                     return visitEndOfDeclarationMarker(node);
                 default:
                     return ts.visitEachChild(node, moduleExpressionElementVisitor, context);
@@ -89503,24 +94526,24 @@ var ts;
                 for (var _i = 0, _a = node.properties; _i < _a.length; _i++) {
                     var elem = _a[_i];
                     switch (elem.kind) {
-                        case 281 /* PropertyAssignment */:
+                        case 288 /* PropertyAssignment */:
                             if (destructuringNeedsFlattening(elem.initializer)) {
                                 return true;
                             }
                             break;
-                        case 282 /* ShorthandPropertyAssignment */:
+                        case 289 /* ShorthandPropertyAssignment */:
                             if (destructuringNeedsFlattening(elem.name)) {
                                 return true;
                             }
                             break;
-                        case 283 /* SpreadAssignment */:
+                        case 290 /* SpreadAssignment */:
                             if (destructuringNeedsFlattening(elem.expression)) {
                                 return true;
                             }
                             break;
-                        case 161 /* MethodDeclaration */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
+                        case 165 /* MethodDeclaration */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
                             return false;
                         default: ts.Debug.assertNever(elem, "Unhandled object member kind");
                     }
@@ -89557,7 +94580,7 @@ var ts;
                 case ts.ModuleKind.AMD:
                     return createImportCallExpressionAMD(argument, containsLexicalThis);
                 case ts.ModuleKind.UMD:
-                    return createImportCallExpressionUMD(argument, containsLexicalThis);
+                    return createImportCallExpressionUMD(argument !== null && argument !== void 0 ? argument : factory.createVoidZero(), containsLexicalThis);
                 case ts.ModuleKind.CommonJS:
                 default:
                     return createImportCallExpressionCommonJS(argument, containsLexicalThis);
@@ -89579,17 +94602,21 @@ var ts;
             // });
             needUMDDynamicImportHelper = true;
             if (ts.isSimpleCopiableExpression(arg)) {
-                var argClone = ts.isGeneratedIdentifier(arg) ? arg : ts.isStringLiteral(arg) ? ts.createLiteral(arg) : ts.setEmitFlags(ts.setTextRange(ts.getSynthesizedClone(arg), arg), 1536 /* NoComments */);
-                return ts.createConditional(
-                /*condition*/ ts.createIdentifier("__syncRequire"), 
+                var argClone = ts.isGeneratedIdentifier(arg) ? arg : ts.isStringLiteral(arg) ? factory.createStringLiteralFromNode(arg) : ts.setEmitFlags(ts.setTextRange(factory.cloneNode(arg), arg), 1536 /* NoComments */);
+                return factory.createConditionalExpression(
+                /*condition*/ factory.createIdentifier("__syncRequire"), 
+                /*questionToken*/ undefined, 
                 /*whenTrue*/ createImportCallExpressionCommonJS(arg, containsLexicalThis), 
+                /*colonToken*/ undefined, 
                 /*whenFalse*/ createImportCallExpressionAMD(argClone, containsLexicalThis));
             }
             else {
-                var temp = ts.createTempVariable(hoistVariableDeclaration);
-                return ts.createComma(ts.createAssignment(temp, arg), ts.createConditional(
-                /*condition*/ ts.createIdentifier("__syncRequire"), 
+                var temp = factory.createTempVariable(hoistVariableDeclaration);
+                return factory.createComma(factory.createAssignment(temp, arg), factory.createConditionalExpression(
+                /*condition*/ factory.createIdentifier("__syncRequire"), 
+                /*questionToken*/ undefined, 
                 /*whenTrue*/ createImportCallExpressionCommonJS(temp, containsLexicalThis), 
+                /*colonToken*/ undefined, 
                 /*whenFalse*/ createImportCallExpressionAMD(temp, containsLexicalThis)));
             }
         }
@@ -89600,26 +94627,26 @@ var ts;
             //     ...
             //     new Promise(function (_a, _b) { require([x], _a, _b); }); /*Amd Require*/
             // });
-            var resolve = ts.createUniqueName("resolve");
-            var reject = ts.createUniqueName("reject");
+            var resolve = factory.createUniqueName("resolve");
+            var reject = factory.createUniqueName("reject");
             var parameters = [
-                ts.createParameter(/*decorator*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, /*name*/ resolve),
-                ts.createParameter(/*decorator*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, /*name*/ reject)
+                factory.createParameterDeclaration(/*decorator*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, /*name*/ resolve),
+                factory.createParameterDeclaration(/*decorator*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, /*name*/ reject)
             ];
-            var body = ts.createBlock([
-                ts.createExpressionStatement(ts.createCall(ts.createIdentifier("require"), 
-                /*typeArguments*/ undefined, [ts.createArrayLiteral([arg || ts.createOmittedExpression()]), resolve, reject]))
+            var body = factory.createBlock([
+                factory.createExpressionStatement(factory.createCallExpression(factory.createIdentifier("require"), 
+                /*typeArguments*/ undefined, [factory.createArrayLiteralExpression([arg || factory.createOmittedExpression()]), resolve, reject]))
             ]);
             var func;
             if (languageVersion >= 2 /* ES2015 */) {
-                func = ts.createArrowFunction(
+                func = factory.createArrowFunction(
                 /*modifiers*/ undefined, 
                 /*typeParameters*/ undefined, parameters, 
                 /*type*/ undefined, 
                 /*equalsGreaterThanToken*/ undefined, body);
             }
             else {
-                func = ts.createFunctionExpression(
+                func = factory.createFunctionExpression(
                 /*modifiers*/ undefined, 
                 /*asteriskToken*/ undefined, 
                 /*name*/ undefined, 
@@ -89632,10 +94659,9 @@ var ts;
                     ts.setEmitFlags(func, 8 /* CapturesThis */);
                 }
             }
-            var promise = ts.createNew(ts.createIdentifier("Promise"), /*typeArguments*/ undefined, [func]);
+            var promise = factory.createNewExpression(factory.createIdentifier("Promise"), /*typeArguments*/ undefined, [func]);
             if (compilerOptions.esModuleInterop) {
-                context.requestEmitHelper(ts.importStarHelper);
-                return ts.createCall(ts.createPropertyAccess(promise, ts.createIdentifier("then")), /*typeArguments*/ undefined, [ts.getUnscopedHelperName("__importStar")]);
+                return factory.createCallExpression(factory.createPropertyAccessExpression(promise, factory.createIdentifier("then")), /*typeArguments*/ undefined, [emitHelpers().createImportStarCallbackHelper()]);
             }
             return promise;
         }
@@ -89645,15 +94671,14 @@ var ts;
             // Promise.resolve().then(function () { return require(x); }) /*CommonJs Require*/
             // We have to wrap require in then callback so that require is done in asynchronously
             // if we simply do require in resolve callback in Promise constructor. We will execute the loading immediately
-            var promiseResolveCall = ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Promise"), "resolve"), /*typeArguments*/ undefined, /*argumentsArray*/ []);
-            var requireCall = ts.createCall(ts.createIdentifier("require"), /*typeArguments*/ undefined, arg ? [arg] : []);
+            var promiseResolveCall = factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("Promise"), "resolve"), /*typeArguments*/ undefined, /*argumentsArray*/ []);
+            var requireCall = factory.createCallExpression(factory.createIdentifier("require"), /*typeArguments*/ undefined, arg ? [arg] : []);
             if (compilerOptions.esModuleInterop) {
-                context.requestEmitHelper(ts.importStarHelper);
-                requireCall = ts.createCall(ts.getUnscopedHelperName("__importStar"), /*typeArguments*/ undefined, [requireCall]);
+                requireCall = emitHelpers().createImportStarHelper(requireCall);
             }
             var func;
             if (languageVersion >= 2 /* ES2015 */) {
-                func = ts.createArrowFunction(
+                func = factory.createArrowFunction(
                 /*modifiers*/ undefined, 
                 /*typeParameters*/ undefined, 
                 /*parameters*/ [], 
@@ -89661,13 +94686,13 @@ var ts;
                 /*equalsGreaterThanToken*/ undefined, requireCall);
             }
             else {
-                func = ts.createFunctionExpression(
+                func = factory.createFunctionExpression(
                 /*modifiers*/ undefined, 
                 /*asteriskToken*/ undefined, 
                 /*name*/ undefined, 
                 /*typeParameters*/ undefined, 
                 /*parameters*/ [], 
-                /*type*/ undefined, ts.createBlock([ts.createReturn(requireCall)]));
+                /*type*/ undefined, factory.createBlock([factory.createReturnStatement(requireCall)]));
                 // if there is a lexical 'this' in the import call arguments, ensure we indicate
                 // that this new function expression indicates it captures 'this' so that the
                 // es2015 transformer will properly substitute 'this' with '_this'.
@@ -89675,15 +94700,14 @@ var ts;
                     ts.setEmitFlags(func, 8 /* CapturesThis */);
                 }
             }
-            return ts.createCall(ts.createPropertyAccess(promiseResolveCall, "then"), /*typeArguments*/ undefined, [func]);
+            return factory.createCallExpression(factory.createPropertyAccessExpression(promiseResolveCall, "then"), /*typeArguments*/ undefined, [func]);
         }
         function getHelperExpressionForExport(node, innerExpr) {
             if (!compilerOptions.esModuleInterop || ts.getEmitFlags(node) & 67108864 /* NeverApplyImportHelper */) {
                 return innerExpr;
             }
             if (ts.getExportNeedsImportStarHelper(node)) {
-                context.requestEmitHelper(ts.importStarHelper);
-                return ts.createCall(ts.getUnscopedHelperName("__importStar"), /*typeArguments*/ undefined, [innerExpr]);
+                return emitHelpers().createImportStarHelper(innerExpr);
             }
             return innerExpr;
         }
@@ -89692,12 +94716,10 @@ var ts;
                 return innerExpr;
             }
             if (ts.getImportNeedsImportStarHelper(node)) {
-                context.requestEmitHelper(ts.importStarHelper);
-                return ts.createCall(ts.getUnscopedHelperName("__importStar"), /*typeArguments*/ undefined, [innerExpr]);
+                return emitHelpers().createImportStarHelper(innerExpr);
             }
             if (ts.getImportNeedsImportDefaultHelper(node)) {
-                context.requestEmitHelper(ts.importDefaultHelper);
-                return ts.createCall(ts.getUnscopedHelperName("__importDefault"), /*typeArguments*/ undefined, [innerExpr]);
+                return emitHelpers().createImportDefaultHelper(innerExpr);
             }
             return innerExpr;
         }
@@ -89712,13 +94734,14 @@ var ts;
             if (moduleKind !== ts.ModuleKind.AMD) {
                 if (!node.importClause) {
                     // import "mod";
-                    return ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createRequireCall(node)), node), node);
+                    return ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(createRequireCall(node)), node), node);
                 }
                 else {
                     var variables = [];
                     if (namespaceDeclaration && !ts.isDefaultImport(node)) {
                         // import * as n from "mod";
-                        variables.push(ts.createVariableDeclaration(ts.getSynthesizedClone(namespaceDeclaration.name), 
+                        variables.push(factory.createVariableDeclaration(factory.cloneNode(namespaceDeclaration.name), 
+                        /*exclamationToken*/ undefined, 
                         /*type*/ undefined, getHelperExpressionForImport(node, createRequireCall(node))));
                     }
                     else {
@@ -89726,25 +94749,28 @@ var ts;
                         // import { x, y } from "mod";
                         // import d, { x, y } from "mod";
                         // import d, * as n from "mod";
-                        variables.push(ts.createVariableDeclaration(ts.getGeneratedNameForNode(node), 
+                        variables.push(factory.createVariableDeclaration(factory.getGeneratedNameForNode(node), 
+                        /*exclamationToken*/ undefined, 
                         /*type*/ undefined, getHelperExpressionForImport(node, createRequireCall(node))));
                         if (namespaceDeclaration && ts.isDefaultImport(node)) {
-                            variables.push(ts.createVariableDeclaration(ts.getSynthesizedClone(namespaceDeclaration.name), 
-                            /*type*/ undefined, ts.getGeneratedNameForNode(node)));
+                            variables.push(factory.createVariableDeclaration(factory.cloneNode(namespaceDeclaration.name), 
+                            /*exclamationToken*/ undefined, 
+                            /*type*/ undefined, factory.getGeneratedNameForNode(node)));
                         }
                     }
-                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createVariableStatement(
-                    /*modifiers*/ undefined, ts.createVariableDeclarationList(variables, languageVersion >= 2 /* ES2015 */ ? 2 /* Const */ : 0 /* None */)), 
+                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(factory.createVariableStatement(
+                    /*modifiers*/ undefined, factory.createVariableDeclarationList(variables, languageVersion >= 2 /* ES2015 */ ? 2 /* Const */ : 0 /* None */)), 
                     /*location*/ node), 
                     /*original*/ node));
                 }
             }
             else if (namespaceDeclaration && ts.isDefaultImport(node)) {
                 // import d, * as n from "mod";
-                statements = ts.append(statements, ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                    ts.setOriginalNode(ts.setTextRange(ts.createVariableDeclaration(ts.getSynthesizedClone(namespaceDeclaration.name), 
-                    /*type*/ undefined, ts.getGeneratedNameForNode(node)), 
+                statements = ts.append(statements, factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                    ts.setOriginalNode(ts.setTextRange(factory.createVariableDeclaration(factory.cloneNode(namespaceDeclaration.name), 
+                    /*exclamationToken*/ undefined, 
+                    /*type*/ undefined, factory.getGeneratedNameForNode(node)), 
                     /*location*/ node), 
                     /*original*/ node)
                 ], languageVersion >= 2 /* ES2015 */ ? 2 /* Const */ : 0 /* None */)));
@@ -89765,12 +94791,12 @@ var ts;
          * @param importNode The declararation to import.
          */
         function createRequireCall(importNode) {
-            var moduleName = ts.getExternalModuleNameLiteral(importNode, currentSourceFile, host, resolver, compilerOptions);
+            var moduleName = ts.getExternalModuleNameLiteral(factory, importNode, currentSourceFile, host, resolver, compilerOptions);
             var args = [];
             if (moduleName) {
                 args.push(moduleName);
             }
-            return ts.createCall(ts.createIdentifier("require"), /*typeArguments*/ undefined, args);
+            return factory.createCallExpression(factory.createIdentifier("require"), /*typeArguments*/ undefined, args);
         }
         /**
          * Visits an ImportEqualsDeclaration node.
@@ -89781,21 +94807,22 @@ var ts;
             ts.Debug.assert(ts.isExternalModuleImportEqualsDeclaration(node), "import= for internal module references should be handled in an earlier transformer.");
             var statements;
             if (moduleKind !== ts.ModuleKind.AMD) {
-                if (ts.hasModifier(node, 1 /* Export */)) {
-                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createExportExpression(node.name, createRequireCall(node))), node), node));
+                if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(createExportExpression(node.name, createRequireCall(node))), node), node));
                 }
                 else {
-                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createVariableStatement(
-                    /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                        ts.createVariableDeclaration(ts.getSynthesizedClone(node.name), 
+                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(factory.createVariableStatement(
+                    /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                        factory.createVariableDeclaration(factory.cloneNode(node.name), 
+                        /*exclamationToken*/ undefined, 
                         /*type*/ undefined, createRequireCall(node))
                     ], 
                     /*flags*/ languageVersion >= 2 /* ES2015 */ ? 2 /* Const */ : 0 /* None */)), node), node));
                 }
             }
             else {
-                if (ts.hasModifier(node, 1 /* Export */)) {
-                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createExportExpression(ts.getExportName(node), ts.getLocalName(node))), node), node));
+                if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(createExportExpression(factory.getExportName(node), factory.getLocalName(node))), node), node));
                 }
             }
             if (hasAssociatedEndOfDeclarationMarker(node)) {
@@ -89819,14 +94846,15 @@ var ts;
                 // elsewhere.
                 return undefined;
             }
-            var generatedName = ts.getGeneratedNameForNode(node);
+            var generatedName = factory.getGeneratedNameForNode(node);
             if (node.exportClause && ts.isNamedExports(node.exportClause)) {
                 var statements = [];
                 // export { x, y } from "mod";
                 if (moduleKind !== ts.ModuleKind.AMD) {
-                    statements.push(ts.setOriginalNode(ts.setTextRange(ts.createVariableStatement(
-                    /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                        ts.createVariableDeclaration(generatedName, 
+                    statements.push(ts.setOriginalNode(ts.setTextRange(factory.createVariableStatement(
+                    /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                        factory.createVariableDeclaration(generatedName, 
+                        /*exclamationToken*/ undefined, 
                         /*type*/ undefined, createRequireCall(node))
                     ])), 
                     /*location*/ node), 
@@ -89835,11 +94863,14 @@ var ts;
                 for (var _i = 0, _a = node.exportClause.elements; _i < _a.length; _i++) {
                     var specifier = _a[_i];
                     if (languageVersion === 0 /* ES3 */) {
-                        statements.push(ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createCreateBindingHelper(context, generatedName, ts.createLiteral(specifier.propertyName || specifier.name), specifier.propertyName ? ts.createLiteral(specifier.name) : undefined)), specifier), specifier));
+                        statements.push(ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(emitHelpers().createCreateBindingHelper(generatedName, factory.createStringLiteralFromNode(specifier.propertyName || specifier.name), specifier.propertyName ? factory.createStringLiteralFromNode(specifier.name) : undefined)), specifier), specifier));
                     }
                     else {
-                        var exportedValue = ts.createPropertyAccess(generatedName, specifier.propertyName || specifier.name);
-                        statements.push(ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createExportExpression(ts.getExportName(specifier), exportedValue, /* location */ undefined, /* liveBinding */ true)), specifier), specifier));
+                        var exportNeedsImportDefault = !!compilerOptions.esModuleInterop &&
+                            !(ts.getEmitFlags(node) & 67108864 /* NeverApplyImportHelper */) &&
+                            ts.idText(specifier.propertyName || specifier.name) === "default";
+                        var exportedValue = factory.createPropertyAccessExpression(exportNeedsImportDefault ? emitHelpers().createImportDefaultHelper(generatedName) : generatedName, specifier.propertyName || specifier.name);
+                        statements.push(ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(createExportExpression(factory.getExportName(specifier), exportedValue, /* location */ undefined, /* liveBinding */ true)), specifier), specifier));
                     }
                 }
                 return ts.singleOrMany(statements);
@@ -89847,14 +94878,16 @@ var ts;
             else if (node.exportClause) {
                 var statements = [];
                 // export * as ns from "mod";
-                statements.push(ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createExportExpression(ts.getSynthesizedClone(node.exportClause.name), moduleKind !== ts.ModuleKind.AMD ?
-                    getHelperExpressionForExport(node, createRequireCall(node)) :
-                    ts.createIdentifier(ts.idText(node.exportClause.name)))), node), node));
+                // export * as default from "mod";
+                statements.push(ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(createExportExpression(factory.cloneNode(node.exportClause.name), getHelperExpressionForExport(node, moduleKind !== ts.ModuleKind.AMD ?
+                    createRequireCall(node) :
+                    ts.isExportNamespaceAsDefaultDeclaration(node) ? generatedName :
+                        factory.createIdentifier(ts.idText(node.exportClause.name))))), node), node));
                 return ts.singleOrMany(statements);
             }
             else {
                 // export * from "mod";
-                return ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(createExportStarHelper(context, moduleKind !== ts.ModuleKind.AMD ? createRequireCall(node) : generatedName)), node), node);
+                return ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(emitHelpers().createExportStarHelper(moduleKind !== ts.ModuleKind.AMD ? createRequireCall(node) : generatedName)), node), node);
             }
         }
         /**
@@ -89871,10 +94904,10 @@ var ts;
             if (original && hasAssociatedEndOfDeclarationMarker(original)) {
                 // Defer exports until we encounter an EndOfDeclarationMarker node
                 var id = ts.getOriginalNodeId(node);
-                deferredExports[id] = appendExportStatement(deferredExports[id], ts.createIdentifier("default"), ts.visitNode(node.expression, moduleExpressionElementVisitor), /*location*/ node, /*allowComments*/ true);
+                deferredExports[id] = appendExportStatement(deferredExports[id], factory.createIdentifier("default"), ts.visitNode(node.expression, moduleExpressionElementVisitor), /*location*/ node, /*allowComments*/ true);
             }
             else {
-                statements = appendExportStatement(statements, ts.createIdentifier("default"), ts.visitNode(node.expression, moduleExpressionElementVisitor), /*location*/ node, /*allowComments*/ true);
+                statements = appendExportStatement(statements, factory.createIdentifier("default"), ts.visitNode(node.expression, moduleExpressionElementVisitor), /*location*/ node, /*allowComments*/ true);
             }
             return ts.singleOrMany(statements);
         }
@@ -89885,9 +94918,9 @@ var ts;
          */
         function visitFunctionDeclaration(node) {
             var statements;
-            if (ts.hasModifier(node, 1 /* Export */)) {
-                statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createFunctionDeclaration(
-                /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, ts.getDeclarationName(node, /*allowComments*/ true, /*allowSourceMaps*/ true), 
+            if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(factory.createFunctionDeclaration(
+                /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, factory.getDeclarationName(node, /*allowComments*/ true, /*allowSourceMaps*/ true), 
                 /*typeParameters*/ undefined, ts.visitNodes(node.parameters, moduleExpressionElementVisitor), 
                 /*type*/ undefined, ts.visitEachChild(node.body, moduleExpressionElementVisitor, context)), 
                 /*location*/ node), 
@@ -89913,9 +94946,9 @@ var ts;
          */
         function visitClassDeclaration(node) {
             var statements;
-            if (ts.hasModifier(node, 1 /* Export */)) {
-                statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createClassDeclaration(
-                /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), ts.getDeclarationName(node, /*allowComments*/ true, /*allowSourceMaps*/ true), 
+            if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(factory.createClassDeclaration(
+                /*decorators*/ undefined, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), factory.getDeclarationName(node, /*allowComments*/ true, /*allowSourceMaps*/ true), 
                 /*typeParameters*/ undefined, ts.visitNodes(node.heritageClauses, moduleExpressionElementVisitor), ts.visitNodes(node.members, moduleExpressionElementVisitor)), node), node));
             }
             else {
@@ -89940,8 +94973,9 @@ var ts;
             var statements;
             var variables;
             var expressions;
-            if (ts.hasModifier(node, 1 /* Export */)) {
+            if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
                 var modifiers = void 0;
+                var removeCommentsOnExpressions = false;
                 // If we're exporting these variables, then these just become assignments to 'exports.x'.
                 for (var _i = 0, _a = node.declarationList.declarations; _i < _a.length; _i++) {
                     var variable = _a[_i];
@@ -89952,14 +94986,28 @@ var ts;
                         variables = ts.append(variables, variable);
                     }
                     else if (variable.initializer) {
-                        expressions = ts.append(expressions, transformInitializedVariable(variable));
+                        if (!ts.isBindingPattern(variable.name) && (ts.isArrowFunction(variable.initializer) || ts.isFunctionExpression(variable.initializer) || ts.isClassExpression(variable.initializer))) {
+                            var expression = factory.createAssignment(ts.setTextRange(factory.createPropertyAccessExpression(factory.createIdentifier("exports"), variable.name), 
+                            /*location*/ variable.name), factory.createIdentifier(ts.getTextOfIdentifierOrLiteral(variable.name)));
+                            var updatedVariable = factory.createVariableDeclaration(variable.name, variable.exclamationToken, variable.type, ts.visitNode(variable.initializer, moduleExpressionElementVisitor));
+                            variables = ts.append(variables, updatedVariable);
+                            expressions = ts.append(expressions, expression);
+                            removeCommentsOnExpressions = true;
+                        }
+                        else {
+                            expressions = ts.append(expressions, transformInitializedVariable(variable));
+                        }
                     }
                 }
                 if (variables) {
-                    statements = ts.append(statements, ts.updateVariableStatement(node, modifiers, ts.updateVariableDeclarationList(node.declarationList, variables)));
+                    statements = ts.append(statements, factory.updateVariableStatement(node, modifiers, factory.updateVariableDeclarationList(node.declarationList, variables)));
                 }
                 if (expressions) {
-                    statements = ts.append(statements, ts.setOriginalNode(ts.setTextRange(ts.createExpressionStatement(ts.inlineExpressions(expressions)), node), node));
+                    var statement = ts.setOriginalNode(ts.setTextRange(factory.createExpressionStatement(factory.inlineExpressions(expressions)), node), node);
+                    if (removeCommentsOnExpressions) {
+                        ts.removeAllComments(statement);
+                    }
+                    statements = ts.append(statements, statement);
                 }
             }
             else {
@@ -89979,7 +95027,7 @@ var ts;
             var exportedNames = getExports(name);
             if (exportedNames) {
                 // For each additional export of the declaration, apply an export assignment.
-                var expression = ts.isExportName(name) ? value : ts.createAssignment(name, value);
+                var expression = ts.isExportName(name) ? value : factory.createAssignment(name, value);
                 for (var _i = 0, exportedNames_1 = exportedNames; _i < exportedNames_1.length; _i++) {
                     var exportName = exportedNames_1[_i];
                     // Mark the node to prevent triggering substitution.
@@ -89988,7 +95036,7 @@ var ts;
                 }
                 return expression;
             }
-            return ts.createAssignment(name, value);
+            return factory.createAssignment(name, value);
         }
         /**
          * Transforms an exported variable with an initializer into an expression.
@@ -90002,8 +95050,8 @@ var ts;
                 /*needsValue*/ false, createAllExportExpressions);
             }
             else {
-                return ts.createAssignment(ts.setTextRange(ts.createPropertyAccess(ts.createIdentifier("exports"), node.name), 
-                /*location*/ node.name), node.initializer ? ts.visitNode(node.initializer, moduleExpressionElementVisitor) : ts.createVoidZero());
+                return factory.createAssignment(ts.setTextRange(factory.createPropertyAccessExpression(factory.createIdentifier("exports"), node.name), 
+                /*location*/ node.name), node.initializer ? ts.visitNode(node.initializer, moduleExpressionElementVisitor) : factory.createVoidZero());
             }
         }
         /**
@@ -90020,7 +95068,7 @@ var ts;
             //
             // To balance the declaration, add the exports of the elided variable
             // statement.
-            if (hasAssociatedEndOfDeclarationMarker(node) && node.original.kind === 225 /* VariableStatement */) {
+            if (hasAssociatedEndOfDeclarationMarker(node) && node.original.kind === 232 /* VariableStatement */) {
                 var id = ts.getOriginalNodeId(node);
                 deferredExports[id] = appendExportsOfVariableStatement(deferredExports[id], node.original);
             }
@@ -90075,10 +95123,10 @@ var ts;
             var namedBindings = importClause.namedBindings;
             if (namedBindings) {
                 switch (namedBindings.kind) {
-                    case 256 /* NamespaceImport */:
+                    case 263 /* NamespaceImport */:
                         statements = appendExportsOfDeclaration(statements, namedBindings);
                         break;
-                    case 257 /* NamedImports */:
+                    case 264 /* NamedImports */:
                         for (var _i = 0, _a = namedBindings.elements; _i < _a.length; _i++) {
                             var importBinding = _a[_i];
                             statements = appendExportsOfDeclaration(statements, importBinding, /* liveBinding */ true);
@@ -90161,9 +95209,9 @@ var ts;
             if (currentModuleInfo.exportEquals) {
                 return statements;
             }
-            if (ts.hasModifier(decl, 1 /* Export */)) {
-                var exportName = ts.hasModifier(decl, 512 /* Default */) ? ts.createIdentifier("default") : ts.getDeclarationName(decl);
-                statements = appendExportStatement(statements, exportName, ts.getLocalName(decl), /*location*/ decl);
+            if (ts.hasSyntacticModifier(decl, 1 /* Export */)) {
+                var exportName = ts.hasSyntacticModifier(decl, 512 /* Default */) ? factory.createIdentifier("default") : factory.getDeclarationName(decl);
+                statements = appendExportStatement(statements, exportName, factory.getLocalName(decl), /*location*/ decl);
             }
             if (decl.name) {
                 statements = appendExportsOfDeclaration(statements, decl);
@@ -90179,7 +95227,7 @@ var ts;
          * @param decl The declaration to export.
          */
         function appendExportsOfDeclaration(statements, decl, liveBinding) {
-            var name = ts.getDeclarationName(decl);
+            var name = factory.getDeclarationName(decl);
             var exportSpecifiers = currentModuleInfo.exportSpecifiers.get(ts.idText(name));
             if (exportSpecifiers) {
                 for (var _i = 0, exportSpecifiers_1 = exportSpecifiers; _i < exportSpecifiers_1.length; _i++) {
@@ -90208,15 +95256,15 @@ var ts;
         function createUnderscoreUnderscoreESModule() {
             var statement;
             if (languageVersion === 0 /* ES3 */) {
-                statement = ts.createExpressionStatement(createExportExpression(ts.createIdentifier("__esModule"), ts.createLiteral(/*value*/ true)));
+                statement = factory.createExpressionStatement(createExportExpression(factory.createIdentifier("__esModule"), factory.createTrue()));
             }
             else {
-                statement = ts.createExpressionStatement(ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Object"), "defineProperty"), 
+                statement = factory.createExpressionStatement(factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("Object"), "defineProperty"), 
                 /*typeArguments*/ undefined, [
-                    ts.createIdentifier("exports"),
-                    ts.createLiteral("__esModule"),
-                    ts.createObjectLiteral([
-                        ts.createPropertyAssignment("value", ts.createLiteral(/*value*/ true))
+                    factory.createIdentifier("exports"),
+                    factory.createStringLiteral("__esModule"),
+                    factory.createObjectLiteralExpression([
+                        factory.createPropertyAssignment("value", factory.createTrue())
                     ])
                 ]));
             }
@@ -90232,7 +95280,7 @@ var ts;
          * @param allowComments An optional value indicating whether to emit comments for the statement.
          */
         function createExportStatement(name, value, location, allowComments, liveBinding) {
-            var statement = ts.setTextRange(ts.createExpressionStatement(createExportExpression(name, value, /* location */ undefined, liveBinding)), location);
+            var statement = ts.setTextRange(factory.createExpressionStatement(createExportExpression(name, value, /* location */ undefined, liveBinding)), location);
             ts.startOnNewLine(statement);
             if (!allowComments) {
                 ts.setEmitFlags(statement, 1536 /* NoComments */);
@@ -90247,21 +95295,21 @@ var ts;
          * @param location The location to use for source maps and comments for the export.
          */
         function createExportExpression(name, value, location, liveBinding) {
-            return ts.setTextRange(liveBinding && languageVersion !== 0 /* ES3 */ ? ts.createCall(ts.createPropertyAccess(ts.createIdentifier("Object"), "defineProperty"), 
+            return ts.setTextRange(liveBinding && languageVersion !== 0 /* ES3 */ ? factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("Object"), "defineProperty"), 
             /*typeArguments*/ undefined, [
-                ts.createIdentifier("exports"),
-                ts.createLiteral(name),
-                ts.createObjectLiteral([
-                    ts.createPropertyAssignment("enumerable", ts.createLiteral(/*value*/ true)),
-                    ts.createPropertyAssignment("get", ts.createFunctionExpression(
+                factory.createIdentifier("exports"),
+                factory.createStringLiteralFromNode(name),
+                factory.createObjectLiteralExpression([
+                    factory.createPropertyAssignment("enumerable", factory.createTrue()),
+                    factory.createPropertyAssignment("get", factory.createFunctionExpression(
                     /*modifiers*/ undefined, 
                     /*asteriskToken*/ undefined, 
                     /*name*/ undefined, 
                     /*typeParameters*/ undefined, 
                     /*parameters*/ [], 
-                    /*type*/ undefined, ts.createBlock([ts.createReturn(value)])))
+                    /*type*/ undefined, factory.createBlock([factory.createReturnStatement(value)])))
                 ])
-            ]) : ts.createAssignment(ts.createPropertyAccess(ts.createIdentifier("exports"), ts.getSynthesizedClone(name)), value), location);
+            ]) : factory.createAssignment(factory.createPropertyAccessExpression(factory.createIdentifier("exports"), factory.cloneNode(name)), value), location);
         }
         //
         // Modifier Visitors
@@ -90274,8 +95322,8 @@ var ts;
         function modifierVisitor(node) {
             // Elide module-specific modifiers.
             switch (node.kind) {
-                case 89 /* ExportKeyword */:
-                case 84 /* DefaultKeyword */:
+                case 92 /* ExportKeyword */:
+                case 87 /* DefaultKeyword */:
                     return undefined;
             }
             return node;
@@ -90291,7 +95339,7 @@ var ts;
          * @param emit A callback used to emit the node in the printer.
          */
         function onEmitNode(hint, node, emitCallback) {
-            if (node.kind === 290 /* SourceFile */) {
+            if (node.kind === 297 /* SourceFile */) {
                 currentSourceFile = node;
                 currentModuleInfo = moduleInfoMap[ts.getOriginalNodeId(currentSourceFile)];
                 noSubstitution = [];
@@ -90339,10 +95387,10 @@ var ts;
                 // A shorthand property with an assignment initializer is probably part of a
                 // destructuring assignment
                 if (node.objectAssignmentInitializer) {
-                    var initializer = ts.createAssignment(exportedOrImportedName, node.objectAssignmentInitializer);
-                    return ts.setTextRange(ts.createPropertyAssignment(name, initializer), node);
+                    var initializer = factory.createAssignment(exportedOrImportedName, node.objectAssignmentInitializer);
+                    return ts.setTextRange(factory.createPropertyAssignment(name, initializer), node);
                 }
-                return ts.setTextRange(ts.createPropertyAssignment(name, exportedOrImportedName), node);
+                return ts.setTextRange(factory.createPropertyAssignment(name, exportedOrImportedName), node);
             }
             return node;
         }
@@ -90353,12 +95401,12 @@ var ts;
          */
         function substituteExpression(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return substituteExpressionIdentifier(node);
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return substituteBinaryExpression(node);
-                case 208 /* PostfixUnaryExpression */:
-                case 207 /* PrefixUnaryExpression */:
+                case 215 /* PostfixUnaryExpression */:
+                case 214 /* PrefixUnaryExpression */:
                     return substituteUnaryExpression(node);
             }
             return node;
@@ -90373,25 +95421,25 @@ var ts;
             if (ts.getEmitFlags(node) & 4096 /* HelperName */) {
                 var externalHelpersModuleName = ts.getExternalHelpersModuleName(currentSourceFile);
                 if (externalHelpersModuleName) {
-                    return ts.createPropertyAccess(externalHelpersModuleName, node);
+                    return factory.createPropertyAccessExpression(externalHelpersModuleName, node);
                 }
                 return node;
             }
-            if (!ts.isGeneratedIdentifier(node) && !ts.isLocalName(node)) {
+            if (!(ts.isGeneratedIdentifier(node) && !(node.autoGenerateFlags & 64 /* AllowNameSubstitution */)) && !ts.isLocalName(node)) {
                 var exportContainer = resolver.getReferencedExportContainer(node, ts.isExportName(node));
-                if (exportContainer && exportContainer.kind === 290 /* SourceFile */) {
-                    return ts.setTextRange(ts.createPropertyAccess(ts.createIdentifier("exports"), ts.getSynthesizedClone(node)), 
+                if (exportContainer && exportContainer.kind === 297 /* SourceFile */) {
+                    return ts.setTextRange(factory.createPropertyAccessExpression(factory.createIdentifier("exports"), factory.cloneNode(node)), 
                     /*location*/ node);
                 }
                 var importDeclaration = resolver.getReferencedImportDeclaration(node);
                 if (importDeclaration) {
                     if (ts.isImportClause(importDeclaration)) {
-                        return ts.setTextRange(ts.createPropertyAccess(ts.getGeneratedNameForNode(importDeclaration.parent), ts.createIdentifier("default")), 
+                        return ts.setTextRange(factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(importDeclaration.parent), factory.createIdentifier("default")), 
                         /*location*/ node);
                     }
                     else if (ts.isImportSpecifier(importDeclaration)) {
                         var name = importDeclaration.propertyName || importDeclaration.name;
-                        return ts.setTextRange(ts.createPropertyAccess(ts.getGeneratedNameForNode(importDeclaration.parent.parent.parent), ts.getSynthesizedClone(name)), 
+                        return ts.setTextRange(factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(importDeclaration.parent.parent.parent), factory.cloneNode(name)), 
                         /*location*/ node);
                     }
                 }
@@ -90454,8 +95502,8 @@ var ts;
                 && !ts.isDeclarationNameOfEnumOrNamespace(node.operand)) {
                 var exportedNames = getExports(node.operand);
                 if (exportedNames) {
-                    var expression = node.kind === 208 /* PostfixUnaryExpression */
-                        ? ts.setTextRange(ts.createBinary(node.operand, ts.createToken(node.operator === 45 /* PlusPlusToken */ ? 63 /* PlusEqualsToken */ : 64 /* MinusEqualsToken */), ts.createLiteral(1)), 
+                    var expression = node.kind === 215 /* PostfixUnaryExpression */
+                        ? ts.setTextRange(factory.createBinaryExpression(node.operand, factory.createToken(node.operator === 45 /* PlusPlusToken */ ? 63 /* PlusEqualsToken */ : 64 /* MinusEqualsToken */), factory.createNumericLiteral(1)), 
                         /*location*/ node)
                         : node;
                     for (var _i = 0, exportedNames_3 = exportedNames; _i < exportedNames_3.length; _i++) {
@@ -90486,65 +95534,18 @@ var ts;
         }
     }
     ts.transformModule = transformModule;
-    ts.createBindingHelper = {
-        name: "typescript:commonjscreatebinding",
-        importName: "__createBinding",
-        scoped: false,
-        priority: 1,
-        text: "\nvar __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {\n    if (k2 === undefined) k2 = k;\n    Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });\n}) : (function(o, m, k, k2) {\n    if (k2 === undefined) k2 = k;\n    o[k2] = m[k];\n}));"
-    };
-    function createCreateBindingHelper(context, module, inputName, outputName) {
-        context.requestEmitHelper(ts.createBindingHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__createBinding"), /*typeArguments*/ undefined, __spreadArrays([ts.createIdentifier("exports"), module, inputName], (outputName ? [outputName] : [])));
-    }
-    ts.setModuleDefaultHelper = {
-        name: "typescript:commonjscreatevalue",
-        importName: "__setModuleDefault",
-        scoped: false,
-        priority: 1,
-        text: "\nvar __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {\n    Object.defineProperty(o, \"default\", { enumerable: true, value: v });\n}) : function(o, v) {\n    o[\"default\"] = v;\n});"
-    };
-    // emit output for the __export helper function
-    var exportStarHelper = {
-        name: "typescript:export-star",
-        importName: "__exportStar",
-        scoped: false,
-        dependencies: [ts.createBindingHelper],
-        priority: 2,
-        text: "\n            var __exportStar = (this && this.__exportStar) || function(m, exports) {\n                for (var p in m) if (p !== \"default\" && !exports.hasOwnProperty(p)) __createBinding(exports, m, p);\n            };"
-    };
-    function createExportStarHelper(context, module) {
-        context.requestEmitHelper(exportStarHelper);
-        return ts.createCall(ts.getUnscopedHelperName("__exportStar"), /*typeArguments*/ undefined, [module, ts.createIdentifier("exports")]);
-    }
     // emit helper for dynamic import
     var dynamicImportUMDHelper = {
         name: "typescript:dynamicimport-sync-require",
         scoped: true,
         text: "\n            var __syncRequire = typeof module === \"object\" && typeof module.exports === \"object\";"
     };
-    // emit helper for `import * as Name from "foo"`
-    ts.importStarHelper = {
-        name: "typescript:commonjsimportstar",
-        importName: "__importStar",
-        scoped: false,
-        dependencies: [ts.createBindingHelper, ts.setModuleDefaultHelper],
-        priority: 2,
-        text: "\nvar __importStar = (this && this.__importStar) || function (mod) {\n    if (mod && mod.__esModule) return mod;\n    var result = {};\n    if (mod != null) for (var k in mod) if (k !== \"default\" && Object.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\n    __setModuleDefault(result, mod);\n    return result;\n};"
-    };
-    // emit helper for `import Name from "foo"`
-    ts.importDefaultHelper = {
-        name: "typescript:commonjsimportdefault",
-        importName: "__importDefault",
-        scoped: false,
-        text: "\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n    return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};"
-    };
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
 (function (ts) {
     function transformSystemModule(context) {
-        var startLexicalEnvironment = context.startLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
+        var factory = context.factory, startLexicalEnvironment = context.startLexicalEnvironment, endLexicalEnvironment = context.endLexicalEnvironment, hoistVariableDeclaration = context.hoistVariableDeclaration;
         var compilerOptions = context.getCompilerOptions();
         var resolver = context.getEmitResolver();
         var host = context.getEmitHost();
@@ -90552,13 +95553,13 @@ var ts;
         var previousOnEmitNode = context.onEmitNode;
         context.onSubstituteNode = onSubstituteNode;
         context.onEmitNode = onEmitNode;
-        context.enableSubstitution(75 /* Identifier */); // Substitutes expression identifiers for imported symbols.
-        context.enableSubstitution(282 /* ShorthandPropertyAssignment */); // Substitutes expression identifiers for imported symbols
-        context.enableSubstitution(209 /* BinaryExpression */); // Substitutes assignments to exported symbols.
-        context.enableSubstitution(207 /* PrefixUnaryExpression */); // Substitutes updates to exported symbols.
-        context.enableSubstitution(208 /* PostfixUnaryExpression */); // Substitutes updates to exported symbols.
-        context.enableSubstitution(219 /* MetaProperty */); // Substitutes 'import.meta'
-        context.enableEmitNotification(290 /* SourceFile */); // Restore state when substituting nodes in a file.
+        context.enableSubstitution(78 /* Identifier */); // Substitutes expression identifiers for imported symbols.
+        context.enableSubstitution(289 /* ShorthandPropertyAssignment */); // Substitutes expression identifiers for imported symbols
+        context.enableSubstitution(216 /* BinaryExpression */); // Substitutes assignments to exported symbols.
+        context.enableSubstitution(214 /* PrefixUnaryExpression */); // Substitutes updates to exported symbols.
+        context.enableSubstitution(215 /* PostfixUnaryExpression */); // Substitutes updates to exported symbols.
+        context.enableSubstitution(226 /* MetaProperty */); // Substitutes 'import.meta'
+        context.enableEmitNotification(297 /* SourceFile */); // Restore state when substituting nodes in a file.
         var moduleInfoMap = []; // The ExternalModuleInfo for each file.
         var deferredExports = []; // Exports to defer until an EndOfDeclarationMarker is found.
         var exportFunctionsMap = []; // The export function associated with a source file.
@@ -90571,7 +95572,7 @@ var ts;
         var hoistedStatements;
         var enclosingBlockScopedContainer;
         var noSubstitution; // Set of nodes for which substitution rules should be ignored.
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         /**
          * Transforms the module aspects of a SourceFile.
          *
@@ -90597,36 +95598,36 @@ var ts;
             // The only exception in this rule is postfix unary operators,
             // see comment to 'substitutePostfixUnaryExpression' for more details
             // Collect information about the external module and dependency groups.
-            moduleInfo = moduleInfoMap[id] = ts.collectExternalModuleInfo(node, resolver, compilerOptions);
+            moduleInfo = moduleInfoMap[id] = ts.collectExternalModuleInfo(context, node, resolver, compilerOptions);
             // Make sure that the name of the 'exports' function does not conflict with
             // existing identifiers.
-            exportFunction = ts.createUniqueName("exports");
+            exportFunction = factory.createUniqueName("exports");
             exportFunctionsMap[id] = exportFunction;
-            contextObject = contextObjectMap[id] = ts.createUniqueName("context");
+            contextObject = contextObjectMap[id] = factory.createUniqueName("context");
             // Add the body of the module.
             var dependencyGroups = collectDependencyGroups(moduleInfo.externalImports);
             var moduleBodyBlock = createSystemModuleBody(node, dependencyGroups);
-            var moduleBodyFunction = ts.createFunctionExpression(
+            var moduleBodyFunction = factory.createFunctionExpression(
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, 
             /*name*/ undefined, 
             /*typeParameters*/ undefined, [
-                ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, exportFunction),
-                ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, contextObject)
+                factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, exportFunction),
+                factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, contextObject)
             ], 
             /*type*/ undefined, moduleBodyBlock);
             // Write the call to `System.register`
             // Clear the emit-helpers flag for later passes since we'll have already used it in the module body
             // So the helper will be emit at the correct position instead of at the top of the source-file
-            var moduleName = ts.tryGetModuleNameFromFile(node, host, compilerOptions);
-            var dependencies = ts.createArrayLiteral(ts.map(dependencyGroups, function (dependencyGroup) { return dependencyGroup.name; }));
-            var updated = ts.setEmitFlags(ts.updateSourceFileNode(node, ts.setTextRange(ts.createNodeArray([
-                ts.createExpressionStatement(ts.createCall(ts.createPropertyAccess(ts.createIdentifier("System"), "register"), 
+            var moduleName = ts.tryGetModuleNameFromFile(factory, node, host, compilerOptions);
+            var dependencies = factory.createArrayLiteralExpression(ts.map(dependencyGroups, function (dependencyGroup) { return dependencyGroup.name; }));
+            var updated = ts.setEmitFlags(factory.updateSourceFile(node, ts.setTextRange(factory.createNodeArray([
+                factory.createExpressionStatement(factory.createCallExpression(factory.createPropertyAccessExpression(factory.createIdentifier("System"), "register"), 
                 /*typeArguments*/ undefined, moduleName
                     ? [moduleName, dependencies, moduleBodyFunction]
                     : [dependencies, moduleBodyFunction]))
             ]), node.statements)), 1024 /* NoTrailingComments */);
-            if (!(compilerOptions.outFile || compilerOptions.out)) {
+            if (!ts.outFile(compilerOptions)) {
                 ts.moveEmitHelpers(updated, moduleBodyBlock, function (helper) { return !helper.scoped; });
             }
             if (noSubstitution) {
@@ -90639,7 +95640,7 @@ var ts;
             contextObject = undefined;
             hoistedStatements = undefined;
             enclosingBlockScopedContainer = undefined;
-            return ts.aggregateTransformFlags(updated);
+            return updated;
         }
         /**
          * Collects the dependency groups for this files imports.
@@ -90647,11 +95648,11 @@ var ts;
          * @param externalImports The imports for the file.
          */
         function collectDependencyGroups(externalImports) {
-            var groupIndices = ts.createMap();
+            var groupIndices = new ts.Map();
             var dependencyGroups = [];
             for (var _i = 0, externalImports_1 = externalImports; _i < externalImports_1.length; _i++) {
                 var externalImport = externalImports_1[_i];
-                var externalModuleName = ts.getExternalModuleNameLiteral(externalImport, currentSourceFile, host, resolver, compilerOptions);
+                var externalModuleName = ts.getExternalModuleNameLiteral(factory, externalImport, currentSourceFile, host, resolver, compilerOptions);
                 if (externalModuleName) {
                     var text = externalModuleName.text;
                     var groupIndex = groupIndices.get(text);
@@ -90724,12 +95725,13 @@ var ts;
             startLexicalEnvironment();
             // Add any prologue directives.
             var ensureUseStrict = ts.getStrictOptionValue(compilerOptions, "alwaysStrict") || (!compilerOptions.noImplicitUseStrict && ts.isExternalModule(currentSourceFile));
-            var statementOffset = ts.addPrologue(statements, node.statements, ensureUseStrict, sourceElementVisitor);
+            var statementOffset = factory.copyPrologue(node.statements, statements, ensureUseStrict, sourceElementVisitor);
             // var __moduleName = context_1 && context_1.id;
-            statements.push(ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                ts.createVariableDeclaration("__moduleName", 
-                /*type*/ undefined, ts.createLogicalAnd(contextObject, ts.createPropertyAccess(contextObject, "id")))
+            statements.push(factory.createVariableStatement(
+            /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                factory.createVariableDeclaration("__moduleName", 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, factory.createLogicalAnd(contextObject, factory.createPropertyAccessExpression(contextObject, "id")))
             ])));
             // Visit the synthetic external helpers import declaration if present
             ts.visitNode(moduleInfo.externalHelpersImportDeclaration, sourceElementVisitor, ts.isStatement);
@@ -90747,20 +95749,19 @@ var ts;
             ts.insertStatementsAfterStandardPrologue(statements, endLexicalEnvironment());
             var exportStarFunction = addExportStarIfNeeded(statements); // TODO: GH#18217
             var modifiers = node.transformFlags & 524288 /* ContainsAwait */ ?
-                ts.createModifiersFromModifierFlags(256 /* Async */) :
+                factory.createModifiersFromModifierFlags(256 /* Async */) :
                 undefined;
-            var moduleObject = ts.createObjectLiteral([
-                ts.createPropertyAssignment("setters", createSettersArray(exportStarFunction, dependencyGroups)),
-                ts.createPropertyAssignment("execute", ts.createFunctionExpression(modifiers, 
+            var moduleObject = factory.createObjectLiteralExpression([
+                factory.createPropertyAssignment("setters", createSettersArray(exportStarFunction, dependencyGroups)),
+                factory.createPropertyAssignment("execute", factory.createFunctionExpression(modifiers, 
                 /*asteriskToken*/ undefined, 
                 /*name*/ undefined, 
                 /*typeParameters*/ undefined, 
                 /*parameters*/ [], 
-                /*type*/ undefined, ts.createBlock(executeStatements, /*multiLine*/ true)))
-            ]);
-            moduleObject.multiLine = true;
-            statements.push(ts.createReturn(moduleObject));
-            return ts.createBlock(statements, /*multiLine*/ true);
+                /*type*/ undefined, factory.createBlock(executeStatements, /*multiLine*/ true)))
+            ], /*multiLine*/ true);
+            statements.push(factory.createReturnStatement(moduleObject));
+            return factory.createBlock(statements, /*multiLine*/ true);
         }
         /**
          * Adds an exportStar function to a statement list if it is needed for the file.
@@ -90782,7 +95783,7 @@ var ts;
                 var hasExportDeclarationWithExportClause = false;
                 for (var _i = 0, _a = moduleInfo.externalImports; _i < _a.length; _i++) {
                     var externalImport = _a[_i];
-                    if (externalImport.kind === 260 /* ExportDeclaration */ && externalImport.exportClause) {
+                    if (externalImport.kind === 267 /* ExportDeclaration */ && externalImport.exportClause) {
                         hasExportDeclarationWithExportClause = true;
                         break;
                     }
@@ -90802,34 +95803,15 @@ var ts;
                         continue;
                     }
                     // write name of exported declaration, i.e 'export var x...'
-                    exportedNames.push(ts.createPropertyAssignment(ts.createLiteral(exportedLocalName), ts.createTrue()));
+                    exportedNames.push(factory.createPropertyAssignment(factory.createStringLiteralFromNode(exportedLocalName), factory.createTrue()));
                 }
             }
-            for (var _d = 0, _e = moduleInfo.externalImports; _d < _e.length; _d++) {
-                var externalImport = _e[_d];
-                if (externalImport.kind !== 260 /* ExportDeclaration */) {
-                    continue;
-                }
-                if (!externalImport.exportClause) {
-                    // export * from ...
-                    continue;
-                }
-                if (ts.isNamedExports(externalImport.exportClause)) {
-                    for (var _f = 0, _g = externalImport.exportClause.elements; _f < _g.length; _f++) {
-                        var element = _g[_f];
-                        // write name of indirectly exported entry, i.e. 'export {x} from ...'
-                        exportedNames.push(ts.createPropertyAssignment(ts.createLiteral(ts.idText(element.name || element.propertyName)), ts.createTrue()));
-                    }
-                }
-                else {
-                    exportedNames.push(ts.createPropertyAssignment(ts.createLiteral(ts.idText(externalImport.exportClause.name)), ts.createTrue()));
-                }
-            }
-            var exportedNamesStorageRef = ts.createUniqueName("exportedNames");
-            statements.push(ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(exportedNamesStorageRef, 
-                /*type*/ undefined, ts.createObjectLiteral(exportedNames, /*multiline*/ true))
+            var exportedNamesStorageRef = factory.createUniqueName("exportedNames");
+            statements.push(factory.createVariableStatement(
+            /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                factory.createVariableDeclaration(exportedNamesStorageRef, 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, factory.createObjectLiteralExpression(exportedNames, /*multiline*/ true))
             ])));
             var exportStarFunction = createExportStarFunction(exportedNamesStorageRef);
             statements.push(exportStarFunction);
@@ -90843,32 +95825,33 @@ var ts;
          * names.
          */
         function createExportStarFunction(localNames) {
-            var exportStarFunction = ts.createUniqueName("exportStar");
-            var m = ts.createIdentifier("m");
-            var n = ts.createIdentifier("n");
-            var exports = ts.createIdentifier("exports");
-            var condition = ts.createStrictInequality(n, ts.createLiteral("default"));
+            var exportStarFunction = factory.createUniqueName("exportStar");
+            var m = factory.createIdentifier("m");
+            var n = factory.createIdentifier("n");
+            var exports = factory.createIdentifier("exports");
+            var condition = factory.createStrictInequality(n, factory.createStringLiteral("default"));
             if (localNames) {
-                condition = ts.createLogicalAnd(condition, ts.createLogicalNot(ts.createCall(ts.createPropertyAccess(localNames, "hasOwnProperty"), 
+                condition = factory.createLogicalAnd(condition, factory.createLogicalNot(factory.createCallExpression(factory.createPropertyAccessExpression(localNames, "hasOwnProperty"), 
                 /*typeArguments*/ undefined, [n])));
             }
-            return ts.createFunctionDeclaration(
+            return factory.createFunctionDeclaration(
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, 
             /*asteriskToken*/ undefined, exportStarFunction, 
-            /*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, m)], 
-            /*type*/ undefined, ts.createBlock([
-                ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                    ts.createVariableDeclaration(exports, 
-                    /*type*/ undefined, ts.createObjectLiteral([]))
+            /*typeParameters*/ undefined, [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, m)], 
+            /*type*/ undefined, factory.createBlock([
+                factory.createVariableStatement(
+                /*modifiers*/ undefined, factory.createVariableDeclarationList([
+                    factory.createVariableDeclaration(exports, 
+                    /*exclamationToken*/ undefined, 
+                    /*type*/ undefined, factory.createObjectLiteralExpression([]))
                 ])),
-                ts.createForIn(ts.createVariableDeclarationList([
-                    ts.createVariableDeclaration(n, /*type*/ undefined)
-                ]), m, ts.createBlock([
-                    ts.setEmitFlags(ts.createIf(condition, ts.createExpressionStatement(ts.createAssignment(ts.createElementAccess(exports, n), ts.createElementAccess(m, n)))), 1 /* SingleLine */)
+                factory.createForInStatement(factory.createVariableDeclarationList([
+                    factory.createVariableDeclaration(n)
+                ]), m, factory.createBlock([
+                    ts.setEmitFlags(factory.createIfStatement(condition, factory.createExpressionStatement(factory.createAssignment(factory.createElementAccessExpression(exports, n), factory.createElementAccessExpression(m, n)))), 1 /* SingleLine */)
                 ])),
-                ts.createExpressionStatement(ts.createCall(exportFunction, 
+                factory.createExpressionStatement(factory.createCallExpression(exportFunction, 
                 /*typeArguments*/ undefined, [exports]))
             ], /*multiline*/ true));
         }
@@ -90883,26 +95866,26 @@ var ts;
             for (var _i = 0, dependencyGroups_1 = dependencyGroups; _i < dependencyGroups_1.length; _i++) {
                 var group_2 = dependencyGroups_1[_i];
                 // derive a unique name for parameter from the first named entry in the group
-                var localName = ts.forEach(group_2.externalImports, function (i) { return ts.getLocalNameForExternalImport(i, currentSourceFile); });
-                var parameterName = localName ? ts.getGeneratedNameForNode(localName) : ts.createUniqueName("");
+                var localName = ts.forEach(group_2.externalImports, function (i) { return ts.getLocalNameForExternalImport(factory, i, currentSourceFile); });
+                var parameterName = localName ? factory.getGeneratedNameForNode(localName) : factory.createUniqueName("");
                 var statements = [];
                 for (var _a = 0, _b = group_2.externalImports; _a < _b.length; _a++) {
                     var entry = _b[_a];
-                    var importVariableName = ts.getLocalNameForExternalImport(entry, currentSourceFile); // TODO: GH#18217
+                    var importVariableName = ts.getLocalNameForExternalImport(factory, entry, currentSourceFile); // TODO: GH#18217
                     switch (entry.kind) {
-                        case 254 /* ImportDeclaration */:
+                        case 261 /* ImportDeclaration */:
                             if (!entry.importClause) {
                                 // 'import "..."' case
                                 // module is imported only for side-effects, no emit required
                                 break;
                             }
                         // falls through
-                        case 253 /* ImportEqualsDeclaration */:
+                        case 260 /* ImportEqualsDeclaration */:
                             ts.Debug.assert(importVariableName !== undefined);
                             // save import into the local
-                            statements.push(ts.createExpressionStatement(ts.createAssignment(importVariableName, parameterName)));
+                            statements.push(factory.createExpressionStatement(factory.createAssignment(importVariableName, parameterName)));
                             break;
-                        case 260 /* ExportDeclaration */:
+                        case 267 /* ExportDeclaration */:
                             ts.Debug.assert(importVariableName !== undefined);
                             if (entry.exportClause) {
                                 if (ts.isNamedExports(entry.exportClause)) {
@@ -90917,15 +95900,15 @@ var ts;
                                     var properties = [];
                                     for (var _c = 0, _d = entry.exportClause.elements; _c < _d.length; _c++) {
                                         var e = _d[_c];
-                                        properties.push(ts.createPropertyAssignment(ts.createLiteral(ts.idText(e.name)), ts.createElementAccess(parameterName, ts.createLiteral(ts.idText(e.propertyName || e.name)))));
+                                        properties.push(factory.createPropertyAssignment(factory.createStringLiteral(ts.idText(e.name)), factory.createElementAccessExpression(parameterName, factory.createStringLiteral(ts.idText(e.propertyName || e.name)))));
                                     }
-                                    statements.push(ts.createExpressionStatement(ts.createCall(exportFunction, 
-                                    /*typeArguments*/ undefined, [ts.createObjectLiteral(properties, /*multiline*/ true)])));
+                                    statements.push(factory.createExpressionStatement(factory.createCallExpression(exportFunction, 
+                                    /*typeArguments*/ undefined, [factory.createObjectLiteralExpression(properties, /*multiline*/ true)])));
                                 }
                                 else {
-                                    statements.push(ts.createExpressionStatement(ts.createCall(exportFunction, 
+                                    statements.push(factory.createExpressionStatement(factory.createCallExpression(exportFunction, 
                                     /*typeArguments*/ undefined, [
-                                        ts.createLiteral(ts.idText(entry.exportClause.name)),
+                                        factory.createStringLiteral(ts.idText(entry.exportClause.name)),
                                         parameterName
                                     ])));
                                 }
@@ -90936,20 +95919,20 @@ var ts;
                                 // emit as:
                                 //
                                 //  exportStar(foo_1_1);
-                                statements.push(ts.createExpressionStatement(ts.createCall(exportStarFunction, 
+                                statements.push(factory.createExpressionStatement(factory.createCallExpression(exportStarFunction, 
                                 /*typeArguments*/ undefined, [parameterName])));
                             }
                             break;
                     }
                 }
-                setters.push(ts.createFunctionExpression(
+                setters.push(factory.createFunctionExpression(
                 /*modifiers*/ undefined, 
                 /*asteriskToken*/ undefined, 
                 /*name*/ undefined, 
-                /*typeParameters*/ undefined, [ts.createParameter(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, parameterName)], 
-                /*type*/ undefined, ts.createBlock(statements, /*multiLine*/ true)));
+                /*typeParameters*/ undefined, [factory.createParameterDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*dotDotDotToken*/ undefined, parameterName)], 
+                /*type*/ undefined, factory.createBlock(statements, /*multiLine*/ true)));
             }
-            return ts.createArrayLiteral(setters, /*multiLine*/ true);
+            return factory.createArrayLiteralExpression(setters, /*multiLine*/ true);
         }
         //
         // Top-level Source Element Visitors
@@ -90961,13 +95944,13 @@ var ts;
          */
         function sourceElementVisitor(node) {
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     return visitImportDeclaration(node);
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return visitImportEqualsDeclaration(node);
-                case 260 /* ExportDeclaration */:
+                case 267 /* ExportDeclaration */:
                     return visitExportDeclaration(node);
-                case 259 /* ExportAssignment */:
+                case 266 /* ExportAssignment */:
                     return visitExportAssignment(node);
                 default:
                     return nestedElementVisitor(node);
@@ -90981,7 +95964,7 @@ var ts;
         function visitImportDeclaration(node) {
             var statements;
             if (node.importClause) {
-                hoistVariableDeclaration(ts.getLocalNameForExternalImport(node, currentSourceFile)); // TODO: GH#18217
+                hoistVariableDeclaration(ts.getLocalNameForExternalImport(factory, node, currentSourceFile)); // TODO: GH#18217
             }
             if (hasAssociatedEndOfDeclarationMarker(node)) {
                 // Defer exports until we encounter an EndOfDeclarationMarker node
@@ -91005,7 +95988,7 @@ var ts;
         function visitImportEqualsDeclaration(node) {
             ts.Debug.assert(ts.isExternalModuleImportEqualsDeclaration(node), "import= for internal module references should be handled in an earlier transformer.");
             var statements;
-            hoistVariableDeclaration(ts.getLocalNameForExternalImport(node, currentSourceFile)); // TODO: GH#18217
+            hoistVariableDeclaration(ts.getLocalNameForExternalImport(factory, node, currentSourceFile)); // TODO: GH#18217
             if (hasAssociatedEndOfDeclarationMarker(node)) {
                 // Defer exports until we encounter an EndOfDeclarationMarker node
                 var id = ts.getOriginalNodeId(node);
@@ -91031,10 +96014,10 @@ var ts;
             if (original && hasAssociatedEndOfDeclarationMarker(original)) {
                 // Defer exports until we encounter an EndOfDeclarationMarker node
                 var id = ts.getOriginalNodeId(node);
-                deferredExports[id] = appendExportStatement(deferredExports[id], ts.createIdentifier("default"), expression, /*allowComments*/ true);
+                deferredExports[id] = appendExportStatement(deferredExports[id], factory.createIdentifier("default"), expression, /*allowComments*/ true);
             }
             else {
-                return createExportStatement(ts.createIdentifier("default"), expression, /*allowComments*/ true);
+                return createExportStatement(factory.createIdentifier("default"), expression, /*allowComments*/ true);
             }
         }
         /**
@@ -91043,8 +96026,8 @@ var ts;
          * @param node The node to visit.
          */
         function visitFunctionDeclaration(node) {
-            if (ts.hasModifier(node, 1 /* Export */)) {
-                hoistedStatements = ts.append(hoistedStatements, ts.updateFunctionDeclaration(node, node.decorators, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, ts.getDeclarationName(node, /*allowComments*/ true, /*allowSourceMaps*/ true), 
+            if (ts.hasSyntacticModifier(node, 1 /* Export */)) {
+                hoistedStatements = ts.append(hoistedStatements, factory.updateFunctionDeclaration(node, node.decorators, ts.visitNodes(node.modifiers, modifierVisitor, ts.isModifier), node.asteriskToken, factory.getDeclarationName(node, /*allowComments*/ true, /*allowSourceMaps*/ true), 
                 /*typeParameters*/ undefined, ts.visitNodes(node.parameters, destructuringAndImportCallVisitor, ts.isParameterDeclaration), 
                 /*type*/ undefined, ts.visitNode(node.body, destructuringAndImportCallVisitor, ts.isBlock)));
             }
@@ -91069,10 +96052,10 @@ var ts;
         function visitClassDeclaration(node) {
             var statements;
             // Hoist the name of the class declaration to the outer module body function.
-            var name = ts.getLocalName(node);
+            var name = factory.getLocalName(node);
             hoistVariableDeclaration(name);
             // Rewrite the class declaration into an assignment of a class expression.
-            statements = ts.append(statements, ts.setTextRange(ts.createExpressionStatement(ts.createAssignment(name, ts.setTextRange(ts.createClassExpression(
+            statements = ts.append(statements, ts.setTextRange(factory.createExpressionStatement(factory.createAssignment(name, ts.setTextRange(factory.createClassExpression(ts.visitNodes(node.decorators, destructuringAndImportCallVisitor, ts.isDecorator), 
             /*modifiers*/ undefined, node.name, 
             /*typeParameters*/ undefined, ts.visitNodes(node.heritageClauses, destructuringAndImportCallVisitor, ts.isHeritageClause), ts.visitNodes(node.members, destructuringAndImportCallVisitor, ts.isClassElement)), node))), node));
             if (hasAssociatedEndOfDeclarationMarker(node)) {
@@ -91096,7 +96079,7 @@ var ts;
                 return ts.visitNode(node, destructuringAndImportCallVisitor, ts.isStatement);
             }
             var expressions;
-            var isExportedDeclaration = ts.hasModifier(node, 1 /* Export */);
+            var isExportedDeclaration = ts.hasSyntacticModifier(node, 1 /* Export */);
             var isMarkedDeclaration = hasAssociatedEndOfDeclarationMarker(node);
             for (var _i = 0, _a = node.declarationList.declarations; _i < _a.length; _i++) {
                 var variable = _a[_i];
@@ -91109,7 +96092,7 @@ var ts;
             }
             var statements;
             if (expressions) {
-                statements = ts.append(statements, ts.setTextRange(ts.createExpressionStatement(ts.inlineExpressions(expressions)), node));
+                statements = ts.append(statements, ts.setTextRange(factory.createExpressionStatement(factory.inlineExpressions(expressions)), node));
             }
             if (isMarkedDeclaration) {
                 // Defer exports until we encounter an EndOfDeclarationMarker node
@@ -91136,7 +96119,7 @@ var ts;
                 }
             }
             else {
-                hoistVariableDeclaration(ts.getSynthesizedClone(node.name));
+                hoistVariableDeclaration(factory.cloneNode(node.name));
             }
         }
         /**
@@ -91147,7 +96130,7 @@ var ts;
         function shouldHoistVariableDeclarationList(node) {
             // hoist only non-block scoped declarations or block scoped declarations parented by source file
             return (ts.getEmitFlags(node) & 2097152 /* NoHoisting */) === 0
-                && (enclosingBlockScopedContainer.kind === 290 /* SourceFile */
+                && (enclosingBlockScopedContainer.kind === 297 /* SourceFile */
                     || (ts.getOriginalNode(node).flags & 3 /* BlockScoped */) === 0);
         }
         /**
@@ -91192,10 +96175,10 @@ var ts;
          * @param isExportedDeclaration A value indicating whether the variable is exported.
          */
         function createVariableAssignment(name, value, location, isExportedDeclaration) {
-            hoistVariableDeclaration(ts.getSynthesizedClone(name));
+            hoistVariableDeclaration(factory.cloneNode(name));
             return isExportedDeclaration
-                ? createExportExpression(name, preventSubstitution(ts.setTextRange(ts.createAssignment(name, value), location)))
-                : preventSubstitution(ts.setTextRange(ts.createAssignment(name, value), location));
+                ? createExportExpression(name, preventSubstitution(ts.setTextRange(factory.createAssignment(name, value), location)))
+                : preventSubstitution(ts.setTextRange(factory.createAssignment(name, value), location));
         }
         /**
          * Visits a MergeDeclarationMarker used as a placeholder for the beginning of a merged
@@ -91211,9 +96194,9 @@ var ts;
             //
             // To balance the declaration, we defer the exports of the elided variable
             // statement until we visit this declaration's `EndOfDeclarationMarker`.
-            if (hasAssociatedEndOfDeclarationMarker(node) && node.original.kind === 225 /* VariableStatement */) {
+            if (hasAssociatedEndOfDeclarationMarker(node) && node.original.kind === 232 /* VariableStatement */) {
                 var id = ts.getOriginalNodeId(node);
-                var isExportedDeclaration = ts.hasModifier(node.original, 1 /* Export */);
+                var isExportedDeclaration = ts.hasSyntacticModifier(node.original, 1 /* Export */);
                 deferredExports[id] = appendExportsOfVariableStatement(deferredExports[id], node.original, isExportedDeclaration);
             }
             return node;
@@ -91273,10 +96256,10 @@ var ts;
             var namedBindings = importClause.namedBindings;
             if (namedBindings) {
                 switch (namedBindings.kind) {
-                    case 256 /* NamespaceImport */:
+                    case 263 /* NamespaceImport */:
                         statements = appendExportsOfDeclaration(statements, namedBindings);
                         break;
-                    case 257 /* NamedImports */:
+                    case 264 /* NamedImports */:
                         for (var _i = 0, _a = namedBindings.elements; _i < _a.length; _i++) {
                             var importBinding = _a[_i];
                             statements = appendExportsOfDeclaration(statements, importBinding);
@@ -91349,7 +96332,7 @@ var ts;
             else if (!ts.isGeneratedIdentifier(decl.name)) {
                 var excludeName = void 0;
                 if (exportSelf) {
-                    statements = appendExportStatement(statements, decl.name, ts.getLocalName(decl));
+                    statements = appendExportStatement(statements, decl.name, factory.getLocalName(decl));
                     excludeName = ts.idText(decl.name);
                 }
                 statements = appendExportsOfDeclaration(statements, decl, excludeName);
@@ -91370,9 +96353,9 @@ var ts;
                 return statements;
             }
             var excludeName;
-            if (ts.hasModifier(decl, 1 /* Export */)) {
-                var exportName = ts.hasModifier(decl, 512 /* Default */) ? ts.createLiteral("default") : decl.name;
-                statements = appendExportStatement(statements, exportName, ts.getLocalName(decl));
+            if (ts.hasSyntacticModifier(decl, 1 /* Export */)) {
+                var exportName = ts.hasSyntacticModifier(decl, 512 /* Default */) ? factory.createStringLiteral("default") : decl.name;
+                statements = appendExportStatement(statements, exportName, factory.getLocalName(decl));
                 excludeName = ts.getTextOfIdentifierOrLiteral(exportName);
             }
             if (decl.name) {
@@ -91393,7 +96376,7 @@ var ts;
             if (moduleInfo.exportEquals) {
                 return statements;
             }
-            var name = ts.getDeclarationName(decl);
+            var name = factory.getDeclarationName(decl);
             var exportSpecifiers = moduleInfo.exportSpecifiers.get(ts.idText(name));
             if (exportSpecifiers) {
                 for (var _i = 0, exportSpecifiers_2 = exportSpecifiers; _i < exportSpecifiers_2.length; _i++) {
@@ -91428,7 +96411,7 @@ var ts;
          * @param allowComments An optional value indicating whether to emit comments for the statement.
          */
         function createExportStatement(name, value, allowComments) {
-            var statement = ts.createExpressionStatement(createExportExpression(name, value));
+            var statement = factory.createExpressionStatement(createExportExpression(name, value));
             ts.startOnNewLine(statement);
             if (!allowComments) {
                 ts.setEmitFlags(statement, 1536 /* NoComments */);
@@ -91442,9 +96425,9 @@ var ts;
          * @param value The exported value.
          */
         function createExportExpression(name, value) {
-            var exportName = ts.isIdentifier(name) ? ts.createLiteral(name) : name;
+            var exportName = ts.isIdentifier(name) ? factory.createStringLiteralFromNode(name) : name;
             ts.setEmitFlags(value, ts.getEmitFlags(value) | 1536 /* NoComments */);
-            return ts.setCommentRange(ts.createCall(exportFunction, /*typeArguments*/ undefined, [exportName, value]), value);
+            return ts.setCommentRange(factory.createCallExpression(exportFunction, /*typeArguments*/ undefined, [exportName, value]), value);
         }
         //
         // Top-Level or Nested Source Element Visitors
@@ -91456,43 +96439,43 @@ var ts;
          */
         function nestedElementVisitor(node) {
             switch (node.kind) {
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return visitVariableStatement(node);
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return visitFunctionDeclaration(node);
-                case 245 /* ClassDeclaration */:
+                case 252 /* ClassDeclaration */:
                     return visitClassDeclaration(node);
-                case 230 /* ForStatement */:
+                case 237 /* ForStatement */:
                     return visitForStatement(node);
-                case 231 /* ForInStatement */:
+                case 238 /* ForInStatement */:
                     return visitForInStatement(node);
-                case 232 /* ForOfStatement */:
+                case 239 /* ForOfStatement */:
                     return visitForOfStatement(node);
-                case 228 /* DoStatement */:
+                case 235 /* DoStatement */:
                     return visitDoStatement(node);
-                case 229 /* WhileStatement */:
+                case 236 /* WhileStatement */:
                     return visitWhileStatement(node);
-                case 238 /* LabeledStatement */:
+                case 245 /* LabeledStatement */:
                     return visitLabeledStatement(node);
-                case 236 /* WithStatement */:
+                case 243 /* WithStatement */:
                     return visitWithStatement(node);
-                case 237 /* SwitchStatement */:
+                case 244 /* SwitchStatement */:
                     return visitSwitchStatement(node);
-                case 251 /* CaseBlock */:
+                case 258 /* CaseBlock */:
                     return visitCaseBlock(node);
-                case 277 /* CaseClause */:
+                case 284 /* CaseClause */:
                     return visitCaseClause(node);
-                case 278 /* DefaultClause */:
+                case 285 /* DefaultClause */:
                     return visitDefaultClause(node);
-                case 240 /* TryStatement */:
+                case 247 /* TryStatement */:
                     return visitTryStatement(node);
-                case 280 /* CatchClause */:
+                case 287 /* CatchClause */:
                     return visitCatchClause(node);
-                case 223 /* Block */:
+                case 230 /* Block */:
                     return visitBlock(node);
-                case 328 /* MergeDeclarationMarker */:
+                case 338 /* MergeDeclarationMarker */:
                     return visitMergeDeclarationMarker(node);
-                case 329 /* EndOfDeclarationMarker */:
+                case 339 /* EndOfDeclarationMarker */:
                     return visitEndOfDeclarationMarker(node);
                 default:
                     return destructuringAndImportCallVisitor(node);
@@ -91506,7 +96489,7 @@ var ts;
         function visitForStatement(node) {
             var savedEnclosingBlockScopedContainer = enclosingBlockScopedContainer;
             enclosingBlockScopedContainer = node;
-            node = ts.updateFor(node, node.initializer && visitForInitializer(node.initializer), ts.visitNode(node.condition, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.incrementor, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement));
+            node = factory.updateForStatement(node, node.initializer && visitForInitializer(node.initializer), ts.visitNode(node.condition, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.incrementor, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement));
             enclosingBlockScopedContainer = savedEnclosingBlockScopedContainer;
             return node;
         }
@@ -91518,7 +96501,7 @@ var ts;
         function visitForInStatement(node) {
             var savedEnclosingBlockScopedContainer = enclosingBlockScopedContainer;
             enclosingBlockScopedContainer = node;
-            node = ts.updateForIn(node, visitForInitializer(node.initializer), ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, ts.liftToBlock));
+            node = factory.updateForInStatement(node, visitForInitializer(node.initializer), ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, factory.liftToBlock));
             enclosingBlockScopedContainer = savedEnclosingBlockScopedContainer;
             return node;
         }
@@ -91530,7 +96513,7 @@ var ts;
         function visitForOfStatement(node) {
             var savedEnclosingBlockScopedContainer = enclosingBlockScopedContainer;
             enclosingBlockScopedContainer = node;
-            node = ts.updateForOf(node, node.awaitModifier, visitForInitializer(node.initializer), ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, ts.liftToBlock));
+            node = factory.updateForOfStatement(node, node.awaitModifier, visitForInitializer(node.initializer), ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, factory.liftToBlock));
             enclosingBlockScopedContainer = savedEnclosingBlockScopedContainer;
             return node;
         }
@@ -91559,7 +96542,7 @@ var ts;
                         hoistBindingElement(variable);
                     }
                 }
-                return expressions ? ts.inlineExpressions(expressions) : ts.createOmittedExpression();
+                return expressions ? factory.inlineExpressions(expressions) : factory.createOmittedExpression();
             }
             else {
                 return ts.visitEachChild(node, nestedElementVisitor, context);
@@ -91571,7 +96554,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitDoStatement(node) {
-            return ts.updateDo(node, ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, ts.liftToBlock), ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression));
+            return factory.updateDoStatement(node, ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, factory.liftToBlock), ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression));
         }
         /**
          * Visits the body of a WhileStatement to hoist declarations.
@@ -91579,7 +96562,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitWhileStatement(node) {
-            return ts.updateWhile(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, ts.liftToBlock));
+            return factory.updateWhileStatement(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, factory.liftToBlock));
         }
         /**
          * Visits the body of a LabeledStatement to hoist declarations.
@@ -91587,7 +96570,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitLabeledStatement(node) {
-            return ts.updateLabel(node, node.label, ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, ts.liftToBlock));
+            return factory.updateLabeledStatement(node, node.label, ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, factory.liftToBlock));
         }
         /**
          * Visits the body of a WithStatement to hoist declarations.
@@ -91595,7 +96578,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitWithStatement(node) {
-            return ts.updateWith(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, ts.liftToBlock));
+            return factory.updateWithStatement(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.statement, nestedElementVisitor, ts.isStatement, factory.liftToBlock));
         }
         /**
          * Visits the body of a SwitchStatement to hoist declarations.
@@ -91603,7 +96586,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitSwitchStatement(node) {
-            return ts.updateSwitch(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.caseBlock, nestedElementVisitor, ts.isCaseBlock));
+            return factory.updateSwitchStatement(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNode(node.caseBlock, nestedElementVisitor, ts.isCaseBlock));
         }
         /**
          * Visits the body of a CaseBlock to hoist declarations.
@@ -91613,7 +96596,7 @@ var ts;
         function visitCaseBlock(node) {
             var savedEnclosingBlockScopedContainer = enclosingBlockScopedContainer;
             enclosingBlockScopedContainer = node;
-            node = ts.updateCaseBlock(node, ts.visitNodes(node.clauses, nestedElementVisitor, ts.isCaseOrDefaultClause));
+            node = factory.updateCaseBlock(node, ts.visitNodes(node.clauses, nestedElementVisitor, ts.isCaseOrDefaultClause));
             enclosingBlockScopedContainer = savedEnclosingBlockScopedContainer;
             return node;
         }
@@ -91623,7 +96606,7 @@ var ts;
          * @param node The node to visit.
          */
         function visitCaseClause(node) {
-            return ts.updateCaseClause(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNodes(node.statements, nestedElementVisitor, ts.isStatement));
+            return factory.updateCaseClause(node, ts.visitNode(node.expression, destructuringAndImportCallVisitor, ts.isExpression), ts.visitNodes(node.statements, nestedElementVisitor, ts.isStatement));
         }
         /**
          * Visits the body of a DefaultClause to hoist declarations.
@@ -91649,7 +96632,7 @@ var ts;
         function visitCatchClause(node) {
             var savedEnclosingBlockScopedContainer = enclosingBlockScopedContainer;
             enclosingBlockScopedContainer = node;
-            node = ts.updateCatchClause(node, node.variableDeclaration, ts.visitNode(node.block, nestedElementVisitor, ts.isBlock));
+            node = factory.updateCatchClause(node, node.variableDeclaration, ts.visitNode(node.block, nestedElementVisitor, ts.isBlock));
             enclosingBlockScopedContainer = savedEnclosingBlockScopedContainer;
             return node;
         }
@@ -91698,7 +96681,7 @@ var ts;
             //         }
             //     };
             // });
-            return ts.createCall(ts.createPropertyAccess(contextObject, ts.createIdentifier("import")), 
+            return factory.createCallExpression(factory.createPropertyAccessExpression(contextObject, factory.createIdentifier("import")), 
             /*typeArguments*/ undefined, ts.some(node.arguments) ? [ts.visitNode(node.arguments[0], destructuringAndImportCallVisitor)] : []);
         }
         /**
@@ -91739,7 +96722,7 @@ var ts;
             }
             else if (ts.isIdentifier(node)) {
                 var container = resolver.getReferencedExportContainer(node);
-                return container !== undefined && container.kind === 290 /* SourceFile */;
+                return container !== undefined && container.kind === 297 /* SourceFile */;
             }
             else {
                 return false;
@@ -91755,8 +96738,8 @@ var ts;
          */
         function modifierVisitor(node) {
             switch (node.kind) {
-                case 89 /* ExportKeyword */:
-                case 84 /* DefaultKeyword */:
+                case 92 /* ExportKeyword */:
+                case 87 /* DefaultKeyword */:
                     return undefined;
             }
             return node;
@@ -91772,7 +96755,7 @@ var ts;
          * @param emitCallback A callback used to emit the node in the printer.
          */
         function onEmitNode(hint, node, emitCallback) {
-            if (node.kind === 290 /* SourceFile */) {
+            if (node.kind === 297 /* SourceFile */) {
                 var id = ts.getOriginalNodeId(node);
                 currentSourceFile = node;
                 moduleInfo = moduleInfoMap[id];
@@ -91822,7 +96805,7 @@ var ts;
          */
         function substituteUnspecified(node) {
             switch (node.kind) {
-                case 282 /* ShorthandPropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
                     return substituteShorthandPropertyAssignment(node);
             }
             return node;
@@ -91838,11 +96821,11 @@ var ts;
                 var importDeclaration = resolver.getReferencedImportDeclaration(name);
                 if (importDeclaration) {
                     if (ts.isImportClause(importDeclaration)) {
-                        return ts.setTextRange(ts.createPropertyAssignment(ts.getSynthesizedClone(name), ts.createPropertyAccess(ts.getGeneratedNameForNode(importDeclaration.parent), ts.createIdentifier("default"))), 
+                        return ts.setTextRange(factory.createPropertyAssignment(factory.cloneNode(name), factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(importDeclaration.parent), factory.createIdentifier("default"))), 
                         /*location*/ node);
                     }
                     else if (ts.isImportSpecifier(importDeclaration)) {
-                        return ts.setTextRange(ts.createPropertyAssignment(ts.getSynthesizedClone(name), ts.createPropertyAccess(ts.getGeneratedNameForNode(importDeclaration.parent.parent.parent), ts.getSynthesizedClone(importDeclaration.propertyName || importDeclaration.name))), 
+                        return ts.setTextRange(factory.createPropertyAssignment(factory.cloneNode(name), factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(importDeclaration.parent.parent.parent), factory.cloneNode(importDeclaration.propertyName || importDeclaration.name))), 
                         /*location*/ node);
                     }
                 }
@@ -91856,14 +96839,14 @@ var ts;
          */
         function substituteExpression(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return substituteExpressionIdentifier(node);
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return substituteBinaryExpression(node);
-                case 207 /* PrefixUnaryExpression */:
-                case 208 /* PostfixUnaryExpression */:
+                case 214 /* PrefixUnaryExpression */:
+                case 215 /* PostfixUnaryExpression */:
                     return substituteUnaryExpression(node);
-                case 219 /* MetaProperty */:
+                case 226 /* MetaProperty */:
                     return substituteMetaProperty(node);
             }
             return node;
@@ -91877,7 +96860,7 @@ var ts;
             if (ts.getEmitFlags(node) & 4096 /* HelperName */) {
                 var externalHelpersModuleName = ts.getExternalHelpersModuleName(currentSourceFile);
                 if (externalHelpersModuleName) {
-                    return ts.createPropertyAccess(externalHelpersModuleName, node);
+                    return factory.createPropertyAccessExpression(externalHelpersModuleName, node);
                 }
                 return node;
             }
@@ -91891,11 +96874,11 @@ var ts;
                 var importDeclaration = resolver.getReferencedImportDeclaration(node);
                 if (importDeclaration) {
                     if (ts.isImportClause(importDeclaration)) {
-                        return ts.setTextRange(ts.createPropertyAccess(ts.getGeneratedNameForNode(importDeclaration.parent), ts.createIdentifier("default")), 
+                        return ts.setTextRange(factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(importDeclaration.parent), factory.createIdentifier("default")), 
                         /*location*/ node);
                     }
                     else if (ts.isImportSpecifier(importDeclaration)) {
-                        return ts.setTextRange(ts.createPropertyAccess(ts.getGeneratedNameForNode(importDeclaration.parent.parent.parent), ts.getSynthesizedClone(importDeclaration.propertyName || importDeclaration.name)), 
+                        return ts.setTextRange(factory.createPropertyAccessExpression(factory.getGeneratedNameForNode(importDeclaration.parent.parent.parent), factory.cloneNode(importDeclaration.propertyName || importDeclaration.name)), 
                         /*location*/ node);
                     }
                 }
@@ -91956,17 +96939,17 @@ var ts;
                 && !ts.isDeclarationNameOfEnumOrNamespace(node.operand)) {
                 var exportedNames = getExports(node.operand);
                 if (exportedNames) {
-                    var expression = node.kind === 208 /* PostfixUnaryExpression */
-                        ? ts.setTextRange(ts.createPrefix(node.operator, node.operand), node)
+                    var expression = node.kind === 215 /* PostfixUnaryExpression */
+                        ? ts.setTextRange(factory.createPrefixUnaryExpression(node.operator, node.operand), node)
                         : node;
                     for (var _i = 0, exportedNames_5 = exportedNames; _i < exportedNames_5.length; _i++) {
                         var exportName = exportedNames_5[_i];
                         expression = createExportExpression(exportName, preventSubstitution(expression));
                     }
-                    if (node.kind === 208 /* PostfixUnaryExpression */) {
+                    if (node.kind === 215 /* PostfixUnaryExpression */) {
                         expression = node.operator === 45 /* PlusPlusToken */
-                            ? ts.createSubtract(preventSubstitution(expression), ts.createLiteral(1))
-                            : ts.createAdd(preventSubstitution(expression), ts.createLiteral(1));
+                            ? factory.createSubtract(preventSubstitution(expression), factory.createNumericLiteral(1))
+                            : factory.createAdd(preventSubstitution(expression), factory.createNumericLiteral(1));
                     }
                     return expression;
                 }
@@ -91975,7 +96958,7 @@ var ts;
         }
         function substituteMetaProperty(node) {
             if (ts.isImportMeta(node)) {
-                return ts.createPropertyAccess(contextObject, ts.createIdentifier("meta"));
+                return factory.createPropertyAccessExpression(contextObject, factory.createIdentifier("meta"));
             }
             return node;
         }
@@ -91991,8 +96974,8 @@ var ts;
                     || resolver.getReferencedValueDeclaration(name);
                 if (valueDeclaration) {
                     var exportContainer = resolver.getReferencedExportContainer(name, /*prefixLocals*/ false);
-                    if (exportContainer && exportContainer.kind === 290 /* SourceFile */) {
-                        exportedNames = ts.append(exportedNames, ts.getDeclarationName(valueDeclaration));
+                    if (exportContainer && exportContainer.kind === 297 /* SourceFile */) {
+                        exportedNames = ts.append(exportedNames, factory.getDeclarationName(valueDeclaration));
                     }
                     exportedNames = ts.addRange(exportedNames, moduleInfo && moduleInfo.exportedBindings[ts.getOriginalNodeId(valueDeclaration)]);
                 }
@@ -92025,42 +97008,50 @@ var ts;
 var ts;
 (function (ts) {
     function transformECMAScriptModule(context) {
+        var factory = context.factory, emitHelpers = context.getEmitHelperFactory;
         var compilerOptions = context.getCompilerOptions();
         var previousOnEmitNode = context.onEmitNode;
         var previousOnSubstituteNode = context.onSubstituteNode;
         context.onEmitNode = onEmitNode;
         context.onSubstituteNode = onSubstituteNode;
-        context.enableEmitNotification(290 /* SourceFile */);
-        context.enableSubstitution(75 /* Identifier */);
+        context.enableEmitNotification(297 /* SourceFile */);
+        context.enableSubstitution(78 /* Identifier */);
         var helperNameSubstitutions;
-        return ts.chainBundle(transformSourceFile);
+        return ts.chainBundle(context, transformSourceFile);
         function transformSourceFile(node) {
             if (node.isDeclarationFile) {
                 return node;
             }
             if (ts.isExternalModule(node) || compilerOptions.isolatedModules) {
-                var externalHelpersImportDeclaration = ts.createExternalHelpersImportDeclarationIfNeeded(node, compilerOptions);
-                if (externalHelpersImportDeclaration) {
-                    var statements = [];
-                    var statementOffset = ts.addPrologue(statements, node.statements);
-                    ts.append(statements, externalHelpersImportDeclaration);
-                    ts.addRange(statements, ts.visitNodes(node.statements, visitor, ts.isStatement, statementOffset));
-                    return ts.updateSourceFileNode(node, ts.setTextRange(ts.createNodeArray(statements), node.statements));
-                }
-                else {
-                    return ts.visitEachChild(node, visitor, context);
+                var result = updateExternalModule(node);
+                if (!ts.isExternalModule(node) || ts.some(result.statements, ts.isExternalModuleIndicator)) {
+                    return result;
                 }
+                return factory.updateSourceFile(result, ts.setTextRange(factory.createNodeArray(__spreadArrays(result.statements, [ts.createEmptyExports(factory)])), result.statements));
             }
             return node;
         }
+        function updateExternalModule(node) {
+            var externalHelpersImportDeclaration = ts.createExternalHelpersImportDeclarationIfNeeded(factory, emitHelpers(), node, compilerOptions);
+            if (externalHelpersImportDeclaration) {
+                var statements = [];
+                var statementOffset = factory.copyPrologue(node.statements, statements);
+                ts.append(statements, externalHelpersImportDeclaration);
+                ts.addRange(statements, ts.visitNodes(node.statements, visitor, ts.isStatement, statementOffset));
+                return factory.updateSourceFile(node, ts.setTextRange(factory.createNodeArray(statements), node.statements));
+            }
+            else {
+                return ts.visitEachChild(node, visitor, context);
+            }
+        }
         function visitor(node) {
             switch (node.kind) {
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     // Elide `import=` as it is not legal with --module ES6
                     return undefined;
-                case 259 /* ExportAssignment */:
+                case 266 /* ExportAssignment */:
                     return visitExportAssignment(node);
-                case 260 /* ExportDeclaration */:
+                case 267 /* ExportDeclaration */:
                     var exportDecl = node;
                     return visitExportDeclaration(exportDecl);
             }
@@ -92080,14 +97071,17 @@ var ts;
                 return node;
             }
             var oldIdentifier = node.exportClause.name;
-            var synthName = ts.getGeneratedNameForNode(oldIdentifier);
-            var importDecl = ts.createImportDeclaration(
+            var synthName = factory.getGeneratedNameForNode(oldIdentifier);
+            var importDecl = factory.createImportDeclaration(
             /*decorators*/ undefined, 
-            /*modifiers*/ undefined, ts.createImportClause(/*name*/ undefined, ts.createNamespaceImport(synthName)), node.moduleSpecifier);
+            /*modifiers*/ undefined, factory.createImportClause(
+            /*isTypeOnly*/ false, 
+            /*name*/ undefined, factory.createNamespaceImport(synthName)), node.moduleSpecifier);
             ts.setOriginalNode(importDecl, node.exportClause);
-            var exportDecl = ts.createExportDeclaration(
+            var exportDecl = ts.isExportNamespaceAsDefaultDeclaration(node) ? factory.createExportDefault(synthName) : factory.createExportDeclaration(
             /*decorators*/ undefined, 
-            /*modifiers*/ undefined, ts.createNamedExports([ts.createExportSpecifier(synthName, oldIdentifier)]));
+            /*modifiers*/ undefined, 
+            /*isTypeOnly*/ false, factory.createNamedExports([factory.createExportSpecifier(synthName, oldIdentifier)]));
             ts.setOriginalNode(exportDecl, node);
             return [importDecl, exportDecl];
         }
@@ -92104,7 +97098,7 @@ var ts;
         function onEmitNode(hint, node, emitCallback) {
             if (ts.isSourceFile(node)) {
                 if ((ts.isExternalModule(node) || compilerOptions.isolatedModules) && compilerOptions.importHelpers) {
-                    helperNameSubstitutions = ts.createMap();
+                    helperNameSubstitutions = new ts.Map();
                 }
                 previousOnEmitNode(hint, node, emitCallback);
                 helperNameSubstitutions = undefined;
@@ -92133,7 +97127,7 @@ var ts;
             var name = ts.idText(node);
             var substitution = helperNameSubstitutions.get(name);
             if (!substitution) {
-                helperNameSubstitutions.set(name, substitution = ts.createFileLevelUniqueName(name));
+                helperNameSubstitutions.set(name, substitution = factory.createUniqueName(name, 16 /* Optimistic */ | 32 /* FileLevel */));
             }
             return substitution;
         }
@@ -92184,14 +97178,14 @@ var ts;
             } : undefined;
         }
         function getAccessorNameVisibilityDiagnosticMessage(symbolAccessibilityResult) {
-            if (ts.hasModifier(node, 32 /* Static */)) {
+            if (ts.hasSyntacticModifier(node, 32 /* Static */)) {
                 return symbolAccessibilityResult.errorModuleName ?
                     symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                         ts.Diagnostics.Public_static_property_0_of_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
                         ts.Diagnostics.Public_static_property_0_of_exported_class_has_or_is_using_name_1_from_private_module_2 :
                     ts.Diagnostics.Public_static_property_0_of_exported_class_has_or_is_using_private_name_1;
             }
-            else if (node.parent.kind === 245 /* ClassDeclaration */) {
+            else if (node.parent.kind === 252 /* ClassDeclaration */) {
                 return symbolAccessibilityResult.errorModuleName ?
                     symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                         ts.Diagnostics.Public_property_0_of_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92213,14 +97207,14 @@ var ts;
             } : undefined;
         }
         function getMethodNameVisibilityDiagnosticMessage(symbolAccessibilityResult) {
-            if (ts.hasModifier(node, 32 /* Static */)) {
+            if (ts.hasSyntacticModifier(node, 32 /* Static */)) {
                 return symbolAccessibilityResult.errorModuleName ?
                     symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                         ts.Diagnostics.Public_static_method_0_of_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
                         ts.Diagnostics.Public_static_method_0_of_exported_class_has_or_is_using_name_1_from_private_module_2 :
                     ts.Diagnostics.Public_static_method_0_of_exported_class_has_or_is_using_private_name_1;
             }
-            else if (node.parent.kind === 245 /* ClassDeclaration */) {
+            else if (node.parent.kind === 252 /* ClassDeclaration */) {
                 return symbolAccessibilityResult.errorModuleName ?
                     symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                         ts.Diagnostics.Public_method_0_of_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92246,7 +97240,7 @@ var ts;
             return getReturnTypeVisibilityError;
         }
         else if (ts.isParameter(node)) {
-            if (ts.isParameterPropertyDeclaration(node, node.parent) && ts.hasModifier(node.parent, 8 /* Private */)) {
+            if (ts.isParameterPropertyDeclaration(node, node.parent) && ts.hasSyntacticModifier(node.parent, 8 /* Private */)) {
                 return getVariableDeclarationTypeVisibilityError;
             }
             return getParameterDeclarationTypeVisibilityError;
@@ -92267,7 +97261,7 @@ var ts;
             return ts.Debug.assertNever(node, "Attempted to set a declaration diagnostic context for unhandled node kind: " + ts.SyntaxKind[node.kind]);
         }
         function getVariableDeclarationTypeVisibilityDiagnosticMessage(symbolAccessibilityResult) {
-            if (node.kind === 242 /* VariableDeclaration */ || node.kind === 191 /* BindingElement */) {
+            if (node.kind === 249 /* VariableDeclaration */ || node.kind === 198 /* BindingElement */) {
                 return symbolAccessibilityResult.errorModuleName ?
                     symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                         ts.Diagnostics.Exported_variable_0_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92276,17 +97270,17 @@ var ts;
             }
             // This check is to ensure we don't report error on constructor parameter property as that error would be reported during parameter emit
             // The only exception here is if the constructor was marked as private. we are not emitting the constructor parameters at all.
-            else if (node.kind === 159 /* PropertyDeclaration */ || node.kind === 194 /* PropertyAccessExpression */ || node.kind === 158 /* PropertySignature */ ||
-                (node.kind === 156 /* Parameter */ && ts.hasModifier(node.parent, 8 /* Private */))) {
+            else if (node.kind === 163 /* PropertyDeclaration */ || node.kind === 201 /* PropertyAccessExpression */ || node.kind === 162 /* PropertySignature */ ||
+                (node.kind === 160 /* Parameter */ && ts.hasSyntacticModifier(node.parent, 8 /* Private */))) {
                 // TODO(jfreeman): Deal with computed properties in error reporting.
-                if (ts.hasModifier(node, 32 /* Static */)) {
+                if (ts.hasSyntacticModifier(node, 32 /* Static */)) {
                     return symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Public_static_property_0_of_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
                             ts.Diagnostics.Public_static_property_0_of_exported_class_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Public_static_property_0_of_exported_class_has_or_is_using_private_name_1;
                 }
-                else if (node.parent.kind === 245 /* ClassDeclaration */ || node.kind === 156 /* Parameter */) {
+                else if (node.parent.kind === 252 /* ClassDeclaration */ || node.kind === 160 /* Parameter */) {
                     return symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Public_property_0_of_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92311,10 +97305,10 @@ var ts;
         }
         function getAccessorDeclarationTypeVisibilityError(symbolAccessibilityResult) {
             var diagnosticMessage;
-            if (node.kind === 164 /* SetAccessor */) {
+            if (node.kind === 168 /* SetAccessor */) {
                 // Getters can infer the return type from the returned expression, but setters cannot, so the
                 // "_from_external_module_1_but_cannot_be_named" case cannot occur.
-                if (ts.hasModifier(node, 32 /* Static */)) {
+                if (ts.hasSyntacticModifier(node, 32 /* Static */)) {
                     diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Parameter_type_of_public_static_setter_0_from_exported_class_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Parameter_type_of_public_static_setter_0_from_exported_class_has_or_is_using_private_name_1;
@@ -92326,7 +97320,7 @@ var ts;
                 }
             }
             else {
-                if (ts.hasModifier(node, 32 /* Static */)) {
+                if (ts.hasSyntacticModifier(node, 32 /* Static */)) {
                     diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Return_type_of_public_static_getter_0_from_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92350,34 +97344,34 @@ var ts;
         function getReturnTypeVisibilityError(symbolAccessibilityResult) {
             var diagnosticMessage;
             switch (node.kind) {
-                case 166 /* ConstructSignature */:
+                case 170 /* ConstructSignature */:
                     // Interfaces cannot have return types that cannot be named
                     diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Return_type_of_constructor_signature_from_exported_interface_has_or_is_using_name_0_from_private_module_1 :
                         ts.Diagnostics.Return_type_of_constructor_signature_from_exported_interface_has_or_is_using_private_name_0;
                     break;
-                case 165 /* CallSignature */:
+                case 169 /* CallSignature */:
                     // Interfaces cannot have return types that cannot be named
                     diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Return_type_of_call_signature_from_exported_interface_has_or_is_using_name_0_from_private_module_1 :
                         ts.Diagnostics.Return_type_of_call_signature_from_exported_interface_has_or_is_using_private_name_0;
                     break;
-                case 167 /* IndexSignature */:
+                case 171 /* IndexSignature */:
                     // Interfaces cannot have return types that cannot be named
                     diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Return_type_of_index_signature_from_exported_interface_has_or_is_using_name_0_from_private_module_1 :
                         ts.Diagnostics.Return_type_of_index_signature_from_exported_interface_has_or_is_using_private_name_0;
                     break;
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    if (ts.hasModifier(node, 32 /* Static */)) {
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    if (ts.hasSyntacticModifier(node, 32 /* Static */)) {
                         diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                             symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                                 ts.Diagnostics.Return_type_of_public_static_method_from_exported_class_has_or_is_using_name_0_from_external_module_1_but_cannot_be_named :
                                 ts.Diagnostics.Return_type_of_public_static_method_from_exported_class_has_or_is_using_name_0_from_private_module_1 :
                             ts.Diagnostics.Return_type_of_public_static_method_from_exported_class_has_or_is_using_private_name_0;
                     }
-                    else if (node.parent.kind === 245 /* ClassDeclaration */) {
+                    else if (node.parent.kind === 252 /* ClassDeclaration */) {
                         diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                             symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                                 ts.Diagnostics.Return_type_of_public_method_from_exported_class_has_or_is_using_name_0_from_external_module_1_but_cannot_be_named :
@@ -92391,7 +97385,7 @@ var ts;
                             ts.Diagnostics.Return_type_of_method_from_exported_interface_has_or_is_using_private_name_0;
                     }
                     break;
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     diagnosticMessage = symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Return_type_of_exported_function_has_or_is_using_name_0_from_external_module_1_but_cannot_be_named :
@@ -92416,38 +97410,38 @@ var ts;
         }
         function getParameterDeclarationTypeVisibilityDiagnosticMessage(symbolAccessibilityResult) {
             switch (node.parent.kind) {
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     return symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Parameter_0_of_constructor_from_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
                             ts.Diagnostics.Parameter_0_of_constructor_from_exported_class_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Parameter_0_of_constructor_from_exported_class_has_or_is_using_private_name_1;
-                case 166 /* ConstructSignature */:
-                case 171 /* ConstructorType */:
+                case 170 /* ConstructSignature */:
+                case 175 /* ConstructorType */:
                     // Interfaces cannot have parameter types that cannot be named
                     return symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Parameter_0_of_constructor_signature_from_exported_interface_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Parameter_0_of_constructor_signature_from_exported_interface_has_or_is_using_private_name_1;
-                case 165 /* CallSignature */:
+                case 169 /* CallSignature */:
                     // Interfaces cannot have parameter types that cannot be named
                     return symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Parameter_0_of_call_signature_from_exported_interface_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Parameter_0_of_call_signature_from_exported_interface_has_or_is_using_private_name_1;
-                case 167 /* IndexSignature */:
+                case 171 /* IndexSignature */:
                     // Interfaces cannot have parameter types that cannot be named
                     return symbolAccessibilityResult.errorModuleName ?
                         ts.Diagnostics.Parameter_0_of_index_signature_from_exported_interface_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Parameter_0_of_index_signature_from_exported_interface_has_or_is_using_private_name_1;
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    if (ts.hasModifier(node.parent, 32 /* Static */)) {
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    if (ts.hasSyntacticModifier(node.parent, 32 /* Static */)) {
                         return symbolAccessibilityResult.errorModuleName ?
                             symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                                 ts.Diagnostics.Parameter_0_of_public_static_method_from_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
                                 ts.Diagnostics.Parameter_0_of_public_static_method_from_exported_class_has_or_is_using_name_1_from_private_module_2 :
                             ts.Diagnostics.Parameter_0_of_public_static_method_from_exported_class_has_or_is_using_private_name_1;
                     }
-                    else if (node.parent.parent.kind === 245 /* ClassDeclaration */) {
+                    else if (node.parent.parent.kind === 252 /* ClassDeclaration */) {
                         return symbolAccessibilityResult.errorModuleName ?
                             symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                                 ts.Diagnostics.Parameter_0_of_public_method_from_exported_class_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92460,15 +97454,15 @@ var ts;
                             ts.Diagnostics.Parameter_0_of_method_from_exported_interface_has_or_is_using_name_1_from_private_module_2 :
                             ts.Diagnostics.Parameter_0_of_method_from_exported_interface_has_or_is_using_private_name_1;
                     }
-                case 244 /* FunctionDeclaration */:
-                case 170 /* FunctionType */:
+                case 251 /* FunctionDeclaration */:
+                case 174 /* FunctionType */:
                     return symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Parameter_0_of_exported_function_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
                             ts.Diagnostics.Parameter_0_of_exported_function_has_or_is_using_name_1_from_private_module_2 :
                         ts.Diagnostics.Parameter_0_of_exported_function_has_or_is_using_private_name_1;
-                case 164 /* SetAccessor */:
-                case 163 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 167 /* GetAccessor */:
                     return symbolAccessibilityResult.errorModuleName ?
                         symbolAccessibilityResult.accessibility === 2 /* CannotBeNamed */ ?
                             ts.Diagnostics.Parameter_0_of_accessor_has_or_is_using_name_1_from_external_module_2_but_cannot_be_named :
@@ -92482,39 +97476,39 @@ var ts;
             // Type parameter constraints are named by user so we should always be able to name it
             var diagnosticMessage;
             switch (node.parent.kind) {
-                case 245 /* ClassDeclaration */:
+                case 252 /* ClassDeclaration */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_exported_class_has_or_is_using_private_name_1;
                     break;
-                case 246 /* InterfaceDeclaration */:
+                case 253 /* InterfaceDeclaration */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_exported_interface_has_or_is_using_private_name_1;
                     break;
-                case 186 /* MappedType */:
+                case 190 /* MappedType */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_exported_mapped_object_type_is_using_private_name_1;
                     break;
-                case 171 /* ConstructorType */:
-                case 166 /* ConstructSignature */:
+                case 175 /* ConstructorType */:
+                case 170 /* ConstructSignature */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_constructor_signature_from_exported_interface_has_or_is_using_private_name_1;
                     break;
-                case 165 /* CallSignature */:
+                case 169 /* CallSignature */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_call_signature_from_exported_interface_has_or_is_using_private_name_1;
                     break;
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                    if (ts.hasModifier(node.parent, 32 /* Static */)) {
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                    if (ts.hasSyntacticModifier(node.parent, 32 /* Static */)) {
                         diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_public_static_method_from_exported_class_has_or_is_using_private_name_1;
                     }
-                    else if (node.parent.parent.kind === 245 /* ClassDeclaration */) {
+                    else if (node.parent.parent.kind === 252 /* ClassDeclaration */) {
                         diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_public_method_from_exported_class_has_or_is_using_private_name_1;
                     }
                     else {
                         diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_method_from_exported_interface_has_or_is_using_private_name_1;
                     }
                     break;
-                case 170 /* FunctionType */:
-                case 244 /* FunctionDeclaration */:
+                case 174 /* FunctionType */:
+                case 251 /* FunctionDeclaration */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_exported_function_has_or_is_using_private_name_1;
                     break;
-                case 247 /* TypeAliasDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
                     diagnosticMessage = ts.Diagnostics.Type_parameter_0_of_exported_type_alias_has_or_is_using_private_name_1;
                     break;
                 default:
@@ -92529,11 +97523,12 @@ var ts;
         function getHeritageClauseVisibilityError() {
             var diagnosticMessage;
             // Heritage clause is written by user so it can always be named
-            if (node.parent.parent.kind === 245 /* ClassDeclaration */) {
+            if (ts.isClassDeclaration(node.parent.parent)) {
                 // Class or Interface implemented/extended is inaccessible
-                diagnosticMessage = ts.isHeritageClause(node.parent) && node.parent.token === 113 /* ImplementsKeyword */ ?
+                diagnosticMessage = ts.isHeritageClause(node.parent) && node.parent.token === 116 /* ImplementsKeyword */ ?
                     ts.Diagnostics.Implements_clause_of_exported_class_0_has_or_is_using_private_name_1 :
-                    ts.Diagnostics.extends_clause_of_exported_class_0_has_or_is_using_private_name_1;
+                    node.parent.parent.name ? ts.Diagnostics.extends_clause_of_exported_class_0_has_or_is_using_private_name_1 :
+                        ts.Diagnostics.extends_clause_of_exported_class_has_or_is_using_private_name_0;
             }
             else {
                 // interface is inaccessible
@@ -92566,11 +97561,8 @@ var ts;
 var ts;
 (function (ts) {
     function getDeclarationDiagnostics(host, resolver, file) {
-        if (file && ts.isJsonSourceFile(file)) {
-            return []; // No declaration diagnostics for json for now
-        }
         var compilerOptions = host.getCompilerOptions();
-        var result = ts.transformNodes(resolver, host, compilerOptions, file ? [file] : ts.filter(host.getSourceFiles(), ts.isSourceFileNotJson), [transformDeclarations], /*allowDtsFiles*/ false);
+        var result = ts.transformNodes(resolver, host, ts.factory, compilerOptions, file ? [file] : ts.filter(host.getSourceFiles(), ts.isSourceFileNotJson), [transformDeclarations], /*allowDtsFiles*/ false);
         return result.diagnostics;
     }
     ts.getDeclarationDiagnostics = getDeclarationDiagnostics;
@@ -92580,7 +97572,7 @@ var ts;
     }
     function isInternalDeclaration(node, currentSourceFile) {
         var parseTreeNode = ts.getParseTreeNode(node);
-        if (parseTreeNode && parseTreeNode.kind === 156 /* Parameter */) {
+        if (parseTreeNode && parseTreeNode.kind === 160 /* Parameter */) {
             var paramIdx = parseTreeNode.parent.parameters.indexOf(parseTreeNode);
             var previousSibling = paramIdx > 0 ? parseTreeNode.parent.parameters[paramIdx - 1] : undefined;
             var text = currentSourceFile.text;
@@ -92626,13 +97618,16 @@ var ts;
         var lateStatementReplacementMap;
         var suppressNewDiagnosticContexts;
         var exportedModulesFromDeclarationEmit;
+        var factory = context.factory;
         var host = context.getEmitHost();
         var symbolTracker = {
             trackSymbol: trackSymbol,
             reportInaccessibleThisError: reportInaccessibleThisError,
             reportInaccessibleUniqueSymbolError: reportInaccessibleUniqueSymbolError,
+            reportCyclicStructureError: reportCyclicStructureError,
             reportPrivateInBaseOfClassExpression: reportPrivateInBaseOfClassExpression,
             reportLikelyUnsafeImportRequiredError: reportLikelyUnsafeImportRequiredError,
+            reportTruncationError: reportTruncationError,
             moduleResolverHost: host,
             trackReferencedAmbientModule: trackReferencedAmbientModule,
             trackExternalModuleSymbolOfImportTypeNode: trackExternalModuleSymbolOfImportTypeNode,
@@ -92651,10 +97646,10 @@ var ts;
             if (!typeReferenceDirectives) {
                 return;
             }
-            necessaryTypeReferences = necessaryTypeReferences || ts.createMap();
+            necessaryTypeReferences = necessaryTypeReferences || new ts.Set();
             for (var _i = 0, typeReferenceDirectives_2 = typeReferenceDirectives; _i < typeReferenceDirectives_2.length; _i++) {
                 var ref = typeReferenceDirectives_2[_i];
-                necessaryTypeReferences.set(ref, true);
+                necessaryTypeReferences.add(ref);
             }
         }
         function trackReferencedAmbientModule(node, symbol) {
@@ -92665,7 +97660,7 @@ var ts;
             }
             // Otherwise we should emit a path-based reference
             var container = ts.getSourceFileOfNode(node);
-            refs.set("" + ts.getOriginalNodeId(container), container);
+            refs.set(ts.getOriginalNodeId(container), container);
         }
         function handleSymbolAccessibilityError(symbolAccessibilityResult) {
             if (symbolAccessibilityResult.accessibility === 0 /* Accessible */) {
@@ -92717,6 +97712,11 @@ var ts;
                 context.addDiagnostic(ts.createDiagnosticForNode(errorNameNode, ts.Diagnostics.The_inferred_type_of_0_references_an_inaccessible_1_type_A_type_annotation_is_necessary, ts.declarationNameToString(errorNameNode), "unique symbol"));
             }
         }
+        function reportCyclicStructureError() {
+            if (errorNameNode) {
+                context.addDiagnostic(ts.createDiagnosticForNode(errorNameNode, ts.Diagnostics.The_inferred_type_of_0_references_a_type_with_a_cyclic_structure_which_cannot_be_trivially_serialized_A_type_annotation_is_necessary, ts.declarationNameToString(errorNameNode)));
+            }
+        }
         function reportInaccessibleThisError() {
             if (errorNameNode) {
                 context.addDiagnostic(ts.createDiagnosticForNode(errorNameNode, ts.Diagnostics.The_inferred_type_of_0_references_an_inaccessible_1_type_A_type_annotation_is_necessary, ts.declarationNameToString(errorNameNode), "this"));
@@ -92727,6 +97727,11 @@ var ts;
                 context.addDiagnostic(ts.createDiagnosticForNode(errorNameNode, ts.Diagnostics.The_inferred_type_of_0_cannot_be_named_without_a_reference_to_1_This_is_likely_not_portable_A_type_annotation_is_necessary, ts.declarationNameToString(errorNameNode), specifier));
             }
         }
+        function reportTruncationError() {
+            if (errorNameNode) {
+                context.addDiagnostic(ts.createDiagnosticForNode(errorNameNode, ts.Diagnostics.The_inferred_type_of_this_node_exceeds_the_maximum_length_the_compiler_will_serialize_An_explicit_type_annotation_is_needed));
+            }
+        }
         function reportNonlocalAugmentation(containingFile, parentSymbol, symbol) {
             var primaryDeclaration = ts.find(parentSymbol.declarations, function (d) { return ts.getSourceFileOfNode(d) === containingFile; });
             var augmentingDeclarations = ts.filter(symbol.declarations, function (d) { return ts.getSourceFileOfNode(d) !== containingFile; });
@@ -92748,15 +97753,15 @@ var ts;
             return result;
         }
         function transformRoot(node) {
-            if (node.kind === 290 /* SourceFile */ && node.isDeclarationFile) {
+            if (node.kind === 297 /* SourceFile */ && node.isDeclarationFile) {
                 return node;
             }
-            if (node.kind === 291 /* Bundle */) {
+            if (node.kind === 298 /* Bundle */) {
                 isBundledEmit = true;
-                refs = ts.createMap();
-                libs = ts.createMap();
+                refs = new ts.Map();
+                libs = new ts.Map();
                 var hasNoDefaultLib_1 = false;
-                var bundle = ts.createBundle(ts.map(node.sourceFiles, function (sourceFile) {
+                var bundle = factory.createBundle(ts.map(node.sourceFiles, function (sourceFile) {
                     if (sourceFile.isDeclarationFile)
                         return undefined; // Omit declaration files from bundle results, too // TODO: GH#18217
                     hasNoDefaultLib_1 = hasNoDefaultLib_1 || sourceFile.hasNoDefaultLib;
@@ -92764,7 +97769,7 @@ var ts;
                     enclosingDeclaration = sourceFile;
                     lateMarkedStatements = undefined;
                     suppressNewDiagnosticContexts = false;
-                    lateStatementReplacementMap = ts.createMap();
+                    lateStatementReplacementMap = new ts.Map();
                     getSymbolAccessibilityDiagnostic = throwDiagnostic;
                     needsScopeFixMarker = false;
                     resultHasScopeMarker = false;
@@ -92773,15 +97778,15 @@ var ts;
                     if (ts.isExternalOrCommonJsModule(sourceFile) || ts.isJsonSourceFile(sourceFile)) {
                         resultHasExternalModuleIndicator = false; // unused in external module bundle emit (all external modules are within module blocks, therefore are known to be modules)
                         needsDeclare = false;
-                        var statements = ts.isSourceFileJS(sourceFile) ? ts.createNodeArray(transformDeclarationsForJS(sourceFile, /*bundled*/ true)) : ts.visitNodes(sourceFile.statements, visitDeclarationStatements);
-                        var newFile = ts.updateSourceFileNode(sourceFile, [ts.createModuleDeclaration([], [ts.createModifier(130 /* DeclareKeyword */)], ts.createLiteral(ts.getResolvedExternalModuleName(context.getEmitHost(), sourceFile)), ts.createModuleBlock(ts.setTextRange(ts.createNodeArray(transformAndReplaceLatePaintedStatements(statements)), sourceFile.statements)))], /*isDeclarationFile*/ true, /*referencedFiles*/ [], /*typeReferences*/ [], /*hasNoDefaultLib*/ false, /*libReferences*/ []);
+                        var statements = ts.isSourceFileJS(sourceFile) ? factory.createNodeArray(transformDeclarationsForJS(sourceFile, /*bundled*/ true)) : ts.visitNodes(sourceFile.statements, visitDeclarationStatements);
+                        var newFile = factory.updateSourceFile(sourceFile, [factory.createModuleDeclaration([], [factory.createModifier(133 /* DeclareKeyword */)], factory.createStringLiteral(ts.getResolvedExternalModuleName(context.getEmitHost(), sourceFile)), factory.createModuleBlock(ts.setTextRange(factory.createNodeArray(transformAndReplaceLatePaintedStatements(statements)), sourceFile.statements)))], /*isDeclarationFile*/ true, /*referencedFiles*/ [], /*typeReferences*/ [], /*hasNoDefaultLib*/ false, /*libReferences*/ []);
                         return newFile;
                     }
                     needsDeclare = true;
-                    var updated = ts.isSourceFileJS(sourceFile) ? ts.createNodeArray(transformDeclarationsForJS(sourceFile)) : ts.visitNodes(sourceFile.statements, visitDeclarationStatements);
-                    return ts.updateSourceFileNode(sourceFile, transformAndReplaceLatePaintedStatements(updated), /*isDeclarationFile*/ true, /*referencedFiles*/ [], /*typeReferences*/ [], /*hasNoDefaultLib*/ false, /*libReferences*/ []);
+                    var updated = ts.isSourceFileJS(sourceFile) ? factory.createNodeArray(transformDeclarationsForJS(sourceFile)) : ts.visitNodes(sourceFile.statements, visitDeclarationStatements);
+                    return factory.updateSourceFile(sourceFile, transformAndReplaceLatePaintedStatements(updated), /*isDeclarationFile*/ true, /*referencedFiles*/ [], /*typeReferences*/ [], /*hasNoDefaultLib*/ false, /*libReferences*/ []);
                 }), ts.mapDefined(node.prepends, function (prepend) {
-                    if (prepend.kind === 293 /* InputFiles */) {
+                    if (prepend.kind === 300 /* InputFiles */) {
                         var sourceFile = ts.createUnparsedSourceFile(prepend, "dts", stripInternal);
                         hasNoDefaultLib_1 = hasNoDefaultLib_1 || !!sourceFile.hasNoDefaultLib;
                         collectReferences(sourceFile, refs);
@@ -92811,29 +97816,29 @@ var ts;
             resultHasExternalModuleIndicator = false;
             suppressNewDiagnosticContexts = false;
             lateMarkedStatements = undefined;
-            lateStatementReplacementMap = ts.createMap();
+            lateStatementReplacementMap = new ts.Map();
             necessaryTypeReferences = undefined;
-            refs = collectReferences(currentSourceFile, ts.createMap());
-            libs = collectLibs(currentSourceFile, ts.createMap());
+            refs = collectReferences(currentSourceFile, new ts.Map());
+            libs = collectLibs(currentSourceFile, new ts.Map());
             var references = [];
             var outputFilePath = ts.getDirectoryPath(ts.normalizeSlashes(ts.getOutputPathsFor(node, host, /*forceDtsPaths*/ true).declarationFilePath));
             var referenceVisitor = mapReferencesIntoArray(references, outputFilePath);
             var combinedStatements;
             if (ts.isSourceFileJS(currentSourceFile)) {
-                combinedStatements = ts.createNodeArray(transformDeclarationsForJS(node));
+                combinedStatements = factory.createNodeArray(transformDeclarationsForJS(node));
                 refs.forEach(referenceVisitor);
                 emittedImports = ts.filter(combinedStatements, ts.isAnyImportSyntax);
             }
             else {
                 var statements = ts.visitNodes(node.statements, visitDeclarationStatements);
-                combinedStatements = ts.setTextRange(ts.createNodeArray(transformAndReplaceLatePaintedStatements(statements)), node.statements);
+                combinedStatements = ts.setTextRange(factory.createNodeArray(transformAndReplaceLatePaintedStatements(statements)), node.statements);
                 refs.forEach(referenceVisitor);
                 emittedImports = ts.filter(combinedStatements, ts.isAnyImportSyntax);
                 if (ts.isExternalModule(node) && (!resultHasExternalModuleIndicator || (needsScopeFixMarker && !resultHasScopeMarker))) {
-                    combinedStatements = ts.setTextRange(ts.createNodeArray(__spreadArrays(combinedStatements, [ts.createEmptyExports()])), combinedStatements);
+                    combinedStatements = ts.setTextRange(factory.createNodeArray(__spreadArrays(combinedStatements, [ts.createEmptyExports(factory)])), combinedStatements);
                 }
             }
-            var updated = ts.updateSourceFileNode(node, combinedStatements, /*isDeclarationFile*/ true, references, getFileReferencesForUsedTypeReferences(), node.hasNoDefaultLib, getLibReferences());
+            var updated = factory.updateSourceFile(node, combinedStatements, /*isDeclarationFile*/ true, references, getFileReferencesForUsedTypeReferences(), node.hasNoDefaultLib, getLibReferences());
             updated.exportedModulesFromDeclarationEmit = exportedModulesFromDeclarationEmit;
             return updated;
             function getLibReferences() {
@@ -92903,7 +97908,7 @@ var ts;
             ts.forEach(sourceFile.referencedFiles, function (f) {
                 var elem = host.getSourceFileFromReference(sourceFile, f);
                 if (elem) {
-                    ret.set("" + ts.getOriginalNodeId(elem), elem);
+                    ret.set(ts.getOriginalNodeId(elem), elem);
                 }
             });
             return ret;
@@ -92918,22 +97923,22 @@ var ts;
             return ret;
         }
         function filterBindingPatternInitializers(name) {
-            if (name.kind === 75 /* Identifier */) {
+            if (name.kind === 78 /* Identifier */) {
                 return name;
             }
             else {
-                if (name.kind === 190 /* ArrayBindingPattern */) {
-                    return ts.updateArrayBindingPattern(name, ts.visitNodes(name.elements, visitBindingElement));
+                if (name.kind === 197 /* ArrayBindingPattern */) {
+                    return factory.updateArrayBindingPattern(name, ts.visitNodes(name.elements, visitBindingElement));
                 }
                 else {
-                    return ts.updateObjectBindingPattern(name, ts.visitNodes(name.elements, visitBindingElement));
+                    return factory.updateObjectBindingPattern(name, ts.visitNodes(name.elements, visitBindingElement));
                 }
             }
             function visitBindingElement(elem) {
-                if (elem.kind === 215 /* OmittedExpression */) {
+                if (elem.kind === 222 /* OmittedExpression */) {
                     return elem;
                 }
-                return ts.updateBindingElement(elem, elem.dotDotDotToken, elem.propertyName, filterBindingPatternInitializers(elem.name), shouldPrintWithInitializer(elem) ? elem.initializer : undefined);
+                return factory.updateBindingElement(elem, elem.dotDotDotToken, elem.propertyName, filterBindingPatternInitializers(elem.name), shouldPrintWithInitializer(elem) ? elem.initializer : undefined);
             }
         }
         function ensureParameter(p, modifierMask, type) {
@@ -92942,8 +97947,8 @@ var ts;
                 oldDiag = getSymbolAccessibilityDiagnostic;
                 getSymbolAccessibilityDiagnostic = ts.createGetSymbolAccessibilityDiagnosticForNode(p);
             }
-            var newParam = ts.updateParameter(p, 
-            /*decorators*/ undefined, maskModifiers(p, modifierMask), p.dotDotDotToken, filterBindingPatternInitializers(p.name), resolver.isOptionalParameter(p) ? (p.questionToken || ts.createToken(57 /* QuestionToken */)) : undefined, ensureType(p, type || p.type, /*ignorePrivate*/ true), // Ignore private param props, since this type is going straight back into a param
+            var newParam = factory.updateParameterDeclaration(p, 
+            /*decorators*/ undefined, maskModifiers(p, modifierMask), p.dotDotDotToken, filterBindingPatternInitializers(p.name), resolver.isOptionalParameter(p) ? (p.questionToken || factory.createToken(57 /* QuestionToken */)) : undefined, ensureType(p, type || p.type, /*ignorePrivate*/ true), // Ignore private param props, since this type is going straight back into a param
             ensureNoInitializer(p));
             if (!suppressNewDiagnosticContexts) {
                 getSymbolAccessibilityDiagnostic = oldDiag;
@@ -92960,7 +97965,7 @@ var ts;
             return undefined;
         }
         function ensureType(node, type, ignorePrivate) {
-            if (!ignorePrivate && ts.hasModifier(node, 8 /* Private */)) {
+            if (!ignorePrivate && ts.hasEffectiveModifier(node, 8 /* Private */)) {
                 // Private nodes emit no types (except private parameter properties, whose parameter types are actually visible)
                 return;
             }
@@ -92968,19 +97973,19 @@ var ts;
                 // Literal const declarations will have an initializer ensured rather than a type
                 return;
             }
-            var shouldUseResolverType = node.kind === 156 /* Parameter */ &&
+            var shouldUseResolverType = node.kind === 160 /* Parameter */ &&
                 (resolver.isRequiredInitializedParameter(node) ||
                     resolver.isOptionalUninitializedParameterProperty(node));
             if (type && !shouldUseResolverType) {
                 return ts.visitNode(type, visitDeclarationSubtree);
             }
             if (!ts.getParseTreeNode(node)) {
-                return type ? ts.visitNode(type, visitDeclarationSubtree) : ts.createKeywordTypeNode(125 /* AnyKeyword */);
+                return type ? ts.visitNode(type, visitDeclarationSubtree) : factory.createKeywordTypeNode(128 /* AnyKeyword */);
             }
-            if (node.kind === 164 /* SetAccessor */) {
+            if (node.kind === 168 /* SetAccessor */) {
                 // Set accessors with no associated type node (from it's param or get accessor return) are `any` since they are never contextually typed right now
                 // (The inferred type here will be void, but the old declaration emitter printed `any`, so this replicates that)
-                return ts.createKeywordTypeNode(125 /* AnyKeyword */);
+                return factory.createKeywordTypeNode(128 /* AnyKeyword */);
             }
             errorNameNode = node.name;
             var oldDiag;
@@ -92988,12 +97993,12 @@ var ts;
                 oldDiag = getSymbolAccessibilityDiagnostic;
                 getSymbolAccessibilityDiagnostic = ts.createGetSymbolAccessibilityDiagnosticForNode(node);
             }
-            if (node.kind === 242 /* VariableDeclaration */ || node.kind === 191 /* BindingElement */) {
+            if (node.kind === 249 /* VariableDeclaration */ || node.kind === 198 /* BindingElement */) {
                 return cleanup(resolver.createTypeOfDeclaration(node, enclosingDeclaration, declarationEmitNodeBuilderFlags, symbolTracker));
             }
-            if (node.kind === 156 /* Parameter */
-                || node.kind === 159 /* PropertyDeclaration */
-                || node.kind === 158 /* PropertySignature */) {
+            if (node.kind === 160 /* Parameter */
+                || node.kind === 163 /* PropertyDeclaration */
+                || node.kind === 162 /* PropertySignature */) {
                 if (!node.initializer)
                     return cleanup(resolver.createTypeOfDeclaration(node, enclosingDeclaration, declarationEmitNodeBuilderFlags, symbolTracker, shouldUseResolverType));
                 return cleanup(resolver.createTypeOfDeclaration(node, enclosingDeclaration, declarationEmitNodeBuilderFlags, symbolTracker, shouldUseResolverType) || resolver.createTypeOfExpression(node.initializer, enclosingDeclaration, declarationEmitNodeBuilderFlags, symbolTracker));
@@ -93004,26 +98009,26 @@ var ts;
                 if (!suppressNewDiagnosticContexts) {
                     getSymbolAccessibilityDiagnostic = oldDiag;
                 }
-                return returnValue || ts.createKeywordTypeNode(125 /* AnyKeyword */);
+                return returnValue || factory.createKeywordTypeNode(128 /* AnyKeyword */);
             }
         }
         function isDeclarationAndNotVisible(node) {
             node = ts.getParseTreeNode(node);
             switch (node.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 249 /* ModuleDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 248 /* EnumDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 255 /* EnumDeclaration */:
                     return !resolver.isDeclarationVisible(node);
                 // The following should be doing their own visibility checks based on filtering their members
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return !getBindingNameVisible(node);
-                case 253 /* ImportEqualsDeclaration */:
-                case 254 /* ImportDeclaration */:
-                case 260 /* ExportDeclaration */:
-                case 259 /* ExportAssignment */:
+                case 260 /* ImportEqualsDeclaration */:
+                case 261 /* ImportDeclaration */:
+                case 267 /* ExportDeclaration */:
+                case 266 /* ExportAssignment */:
                     return false;
             }
             return false;
@@ -93041,14 +98046,14 @@ var ts;
             }
         }
         function updateParamsList(node, params, modifierMask) {
-            if (ts.hasModifier(node, 8 /* Private */)) {
+            if (ts.hasEffectiveModifier(node, 8 /* Private */)) {
                 return undefined; // TODO: GH#18217
             }
             var newParams = ts.map(params, function (p) { return ensureParameter(p, modifierMask); });
             if (!newParams) {
                 return undefined; // TODO: GH#18217
             }
-            return ts.createNodeArray(newParams, params.hasTrailingComma);
+            return factory.createNodeArray(newParams, params.hasTrailingComma);
         }
         function updateAccessorParamsList(input, isPrivate) {
             var newParams;
@@ -93068,17 +98073,17 @@ var ts;
                     }
                 }
                 if (!newValueParameter) {
-                    newValueParameter = ts.createParameter(
+                    newValueParameter = factory.createParameterDeclaration(
                     /*decorators*/ undefined, 
                     /*modifiers*/ undefined, 
                     /*dotDotDotToken*/ undefined, "value");
                 }
                 newParams = ts.append(newParams, newValueParameter);
             }
-            return ts.createNodeArray(newParams || ts.emptyArray);
+            return factory.createNodeArray(newParams || ts.emptyArray);
         }
         function ensureTypeParams(node, params) {
-            return ts.hasModifier(node, 8 /* Private */) ? undefined : ts.visitNodes(params, visitDeclarationSubtree);
+            return ts.hasEffectiveModifier(node, 8 /* Private */) ? undefined : ts.visitNodes(params, visitDeclarationSubtree);
         }
         function isEnclosingDeclaration(node) {
             return ts.isSourceFile(node)
@@ -93104,12 +98109,12 @@ var ts;
         function rewriteModuleSpecifier(parent, input) {
             if (!input)
                 return undefined; // TODO: GH#18217
-            resultHasExternalModuleIndicator = resultHasExternalModuleIndicator || (parent.kind !== 249 /* ModuleDeclaration */ && parent.kind !== 188 /* ImportType */);
+            resultHasExternalModuleIndicator = resultHasExternalModuleIndicator || (parent.kind !== 256 /* ModuleDeclaration */ && parent.kind !== 195 /* ImportType */);
             if (ts.isStringLiteralLike(input)) {
                 if (isBundledEmit) {
                     var newName = ts.getExternalModuleNameFromDeclaration(context.getEmitHost(), resolver, parent);
                     if (newName) {
-                        return ts.createLiteral(newName);
+                        return factory.createStringLiteral(newName);
                     }
                 }
                 else {
@@ -93124,11 +98129,11 @@ var ts;
         function transformImportEqualsDeclaration(decl) {
             if (!resolver.isDeclarationVisible(decl))
                 return;
-            if (decl.moduleReference.kind === 265 /* ExternalModuleReference */) {
+            if (decl.moduleReference.kind === 272 /* ExternalModuleReference */) {
                 // Rewrite external module names if necessary
                 var specifier = ts.getExternalModuleImportEqualsDeclarationExpression(decl);
-                return ts.updateImportEqualsDeclaration(decl, 
-                /*decorators*/ undefined, decl.modifiers, decl.name, ts.updateExternalModuleReference(decl.moduleReference, rewriteModuleSpecifier(decl, specifier)));
+                return factory.updateImportEqualsDeclaration(decl, 
+                /*decorators*/ undefined, decl.modifiers, decl.name, factory.updateExternalModuleReference(decl.moduleReference, rewriteModuleSpecifier(decl, specifier)));
             }
             else {
                 var oldDiag = getSymbolAccessibilityDiagnostic;
@@ -93141,30 +98146,30 @@ var ts;
         function transformImportDeclaration(decl) {
             if (!decl.importClause) {
                 // import "mod" - possibly needed for side effects? (global interface patches, module augmentations, etc)
-                return ts.updateImportDeclaration(decl, 
+                return factory.updateImportDeclaration(decl, 
                 /*decorators*/ undefined, decl.modifiers, decl.importClause, rewriteModuleSpecifier(decl, decl.moduleSpecifier));
             }
             // The `importClause` visibility corresponds to the default's visibility.
             var visibleDefaultBinding = decl.importClause && decl.importClause.name && resolver.isDeclarationVisible(decl.importClause) ? decl.importClause.name : undefined;
             if (!decl.importClause.namedBindings) {
                 // No named bindings (either namespace or list), meaning the import is just default or should be elided
-                return visibleDefaultBinding && ts.updateImportDeclaration(decl, /*decorators*/ undefined, decl.modifiers, ts.updateImportClause(decl.importClause, visibleDefaultBinding, 
-                /*namedBindings*/ undefined, decl.importClause.isTypeOnly), rewriteModuleSpecifier(decl, decl.moduleSpecifier));
+                return visibleDefaultBinding && factory.updateImportDeclaration(decl, /*decorators*/ undefined, decl.modifiers, factory.updateImportClause(decl.importClause, decl.importClause.isTypeOnly, visibleDefaultBinding, 
+                /*namedBindings*/ undefined), rewriteModuleSpecifier(decl, decl.moduleSpecifier));
             }
-            if (decl.importClause.namedBindings.kind === 256 /* NamespaceImport */) {
+            if (decl.importClause.namedBindings.kind === 263 /* NamespaceImport */) {
                 // Namespace import (optionally with visible default)
                 var namedBindings = resolver.isDeclarationVisible(decl.importClause.namedBindings) ? decl.importClause.namedBindings : /*namedBindings*/ undefined;
-                return visibleDefaultBinding || namedBindings ? ts.updateImportDeclaration(decl, /*decorators*/ undefined, decl.modifiers, ts.updateImportClause(decl.importClause, visibleDefaultBinding, namedBindings, decl.importClause.isTypeOnly), rewriteModuleSpecifier(decl, decl.moduleSpecifier)) : undefined;
+                return visibleDefaultBinding || namedBindings ? factory.updateImportDeclaration(decl, /*decorators*/ undefined, decl.modifiers, factory.updateImportClause(decl.importClause, decl.importClause.isTypeOnly, visibleDefaultBinding, namedBindings), rewriteModuleSpecifier(decl, decl.moduleSpecifier)) : undefined;
             }
             // Named imports (optionally with visible default)
             var bindingList = ts.mapDefined(decl.importClause.namedBindings.elements, function (b) { return resolver.isDeclarationVisible(b) ? b : undefined; });
             if ((bindingList && bindingList.length) || visibleDefaultBinding) {
-                return ts.updateImportDeclaration(decl, 
-                /*decorators*/ undefined, decl.modifiers, ts.updateImportClause(decl.importClause, visibleDefaultBinding, bindingList && bindingList.length ? ts.updateNamedImports(decl.importClause.namedBindings, bindingList) : undefined, decl.importClause.isTypeOnly), rewriteModuleSpecifier(decl, decl.moduleSpecifier));
+                return factory.updateImportDeclaration(decl, 
+                /*decorators*/ undefined, decl.modifiers, factory.updateImportClause(decl.importClause, decl.importClause.isTypeOnly, visibleDefaultBinding, bindingList && bindingList.length ? factory.updateNamedImports(decl.importClause.namedBindings, bindingList) : undefined), rewriteModuleSpecifier(decl, decl.moduleSpecifier));
             }
             // Augmentation of export depends on import
             if (resolver.isImportRequiredByAugmentation(decl)) {
-                return ts.updateImportDeclaration(decl, 
+                return factory.updateImportDeclaration(decl, 
                 /*decorators*/ undefined, decl.modifiers, 
                 /*importClause*/ undefined, rewriteModuleSpecifier(decl, decl.moduleSpecifier));
             }
@@ -93194,14 +98199,14 @@ var ts;
                 needsDeclare = i.parent && ts.isSourceFile(i.parent) && !(ts.isExternalModule(i.parent) && isBundledEmit);
                 var result = transformTopLevelDeclaration(i);
                 needsDeclare = priorNeedsDeclare;
-                lateStatementReplacementMap.set("" + ts.getOriginalNodeId(i), result);
+                lateStatementReplacementMap.set(ts.getOriginalNodeId(i), result);
             }
             // And lastly, we need to get the final form of all those indetermine import declarations from before and add them to the output list
             // (and remove them from the set to examine for outter declarations)
             return ts.visitNodes(statements, visitLateVisibilityMarkedStatements);
             function visitLateVisibilityMarkedStatements(statement) {
                 if (ts.isLateVisibilityPaintedStatement(statement)) {
-                    var key = "" + ts.getOriginalNodeId(statement);
+                    var key = ts.getOriginalNodeId(statement);
                     if (lateStatementReplacementMap.has(key)) {
                         var result = lateStatementReplacementMap.get(key);
                         lateStatementReplacementMap.delete(key);
@@ -93246,13 +98251,13 @@ var ts;
             // We'd see a TDZ violation at runtime
             var canProduceDiagnostic = ts.canProduceDiagnostics(input);
             var oldWithinObjectLiteralType = suppressNewDiagnosticContexts;
-            var shouldEnterSuppressNewDiagnosticsContextContext = ((input.kind === 173 /* TypeLiteral */ || input.kind === 186 /* MappedType */) && input.parent.kind !== 247 /* TypeAliasDeclaration */);
+            var shouldEnterSuppressNewDiagnosticsContextContext = ((input.kind === 177 /* TypeLiteral */ || input.kind === 190 /* MappedType */) && input.parent.kind !== 254 /* TypeAliasDeclaration */);
             // Emit methods which are private as properties with no type information
             if (ts.isMethodDeclaration(input) || ts.isMethodSignature(input)) {
-                if (ts.hasModifier(input, 8 /* Private */)) {
+                if (ts.hasEffectiveModifier(input, 8 /* Private */)) {
                     if (input.symbol && input.symbol.declarations && input.symbol.declarations[0] !== input)
                         return; // Elide all but the first overload
-                    return cleanup(ts.createProperty(/*decorators*/ undefined, ensureModifiers(input), input.name, /*questionToken*/ undefined, /*type*/ undefined, /*initializer*/ undefined));
+                    return cleanup(factory.createPropertyDeclaration(/*decorators*/ undefined, ensureModifiers(input), input.name, /*questionToken*/ undefined, /*type*/ undefined, /*initializer*/ undefined));
                 }
             }
             if (canProduceDiagnostic && !suppressNewDiagnosticContexts) {
@@ -93267,93 +98272,94 @@ var ts;
             }
             if (isProcessedComponent(input)) {
                 switch (input.kind) {
-                    case 216 /* ExpressionWithTypeArguments */: {
+                    case 223 /* ExpressionWithTypeArguments */: {
                         if ((ts.isEntityName(input.expression) || ts.isEntityNameExpression(input.expression))) {
                             checkEntityNameVisibility(input.expression, enclosingDeclaration);
                         }
                         var node = ts.visitEachChild(input, visitDeclarationSubtree, context);
-                        return cleanup(ts.updateExpressionWithTypeArguments(node, ts.parenthesizeTypeParameters(node.typeArguments), node.expression));
+                        return cleanup(factory.updateExpressionWithTypeArguments(node, node.expression, node.typeArguments));
                     }
-                    case 169 /* TypeReference */: {
+                    case 173 /* TypeReference */: {
                         checkEntityNameVisibility(input.typeName, enclosingDeclaration);
                         var node = ts.visitEachChild(input, visitDeclarationSubtree, context);
-                        return cleanup(ts.updateTypeReferenceNode(node, node.typeName, ts.parenthesizeTypeParameters(node.typeArguments)));
+                        return cleanup(factory.updateTypeReferenceNode(node, node.typeName, node.typeArguments));
                     }
-                    case 166 /* ConstructSignature */:
-                        return cleanup(ts.updateConstructSignature(input, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type)));
-                    case 162 /* Constructor */: {
+                    case 170 /* ConstructSignature */:
+                        return cleanup(factory.updateConstructSignature(input, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type)));
+                    case 166 /* Constructor */: {
                         // A constructor declaration may not have a type annotation
-                        var ctor = ts.createSignatureDeclaration(162 /* Constructor */, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters, 0 /* None */), 
-                        /*type*/ undefined);
-                        ctor.modifiers = ts.createNodeArray(ensureModifiers(input));
+                        var ctor = factory.createConstructorDeclaration(
+                        /*decorators*/ undefined, 
+                        /*modifiers*/ ensureModifiers(input), updateParamsList(input, input.parameters, 0 /* None */), 
+                        /*body*/ undefined);
                         return cleanup(ctor);
                     }
-                    case 161 /* MethodDeclaration */: {
+                    case 165 /* MethodDeclaration */: {
                         if (ts.isPrivateIdentifier(input.name)) {
                             return cleanup(/*returnValue*/ undefined);
                         }
-                        var sig = ts.createSignatureDeclaration(160 /* MethodSignature */, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type));
-                        sig.name = input.name;
-                        sig.modifiers = ts.createNodeArray(ensureModifiers(input));
-                        sig.questionToken = input.questionToken;
+                        var sig = factory.createMethodDeclaration(
+                        /*decorators*/ undefined, ensureModifiers(input), 
+                        /*asteriskToken*/ undefined, input.name, input.questionToken, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type), 
+                        /*body*/ undefined);
                         return cleanup(sig);
                     }
-                    case 163 /* GetAccessor */: {
+                    case 167 /* GetAccessor */: {
                         if (ts.isPrivateIdentifier(input.name)) {
                             return cleanup(/*returnValue*/ undefined);
                         }
                         var accessorType = getTypeAnnotationFromAllAccessorDeclarations(input, resolver.getAllAccessorDeclarations(input));
-                        return cleanup(ts.updateGetAccessor(input, 
-                        /*decorators*/ undefined, ensureModifiers(input), input.name, updateAccessorParamsList(input, ts.hasModifier(input, 8 /* Private */)), ensureType(input, accessorType), 
+                        return cleanup(factory.updateGetAccessorDeclaration(input, 
+                        /*decorators*/ undefined, ensureModifiers(input), input.name, updateAccessorParamsList(input, ts.hasEffectiveModifier(input, 8 /* Private */)), ensureType(input, accessorType), 
                         /*body*/ undefined));
                     }
-                    case 164 /* SetAccessor */: {
+                    case 168 /* SetAccessor */: {
                         if (ts.isPrivateIdentifier(input.name)) {
                             return cleanup(/*returnValue*/ undefined);
                         }
-                        return cleanup(ts.updateSetAccessor(input, 
-                        /*decorators*/ undefined, ensureModifiers(input), input.name, updateAccessorParamsList(input, ts.hasModifier(input, 8 /* Private */)), 
+                        return cleanup(factory.updateSetAccessorDeclaration(input, 
+                        /*decorators*/ undefined, ensureModifiers(input), input.name, updateAccessorParamsList(input, ts.hasEffectiveModifier(input, 8 /* Private */)), 
                         /*body*/ undefined));
                     }
-                    case 159 /* PropertyDeclaration */:
+                    case 163 /* PropertyDeclaration */:
                         if (ts.isPrivateIdentifier(input.name)) {
                             return cleanup(/*returnValue*/ undefined);
                         }
-                        return cleanup(ts.updateProperty(input, 
+                        return cleanup(factory.updatePropertyDeclaration(input, 
                         /*decorators*/ undefined, ensureModifiers(input), input.name, input.questionToken, ensureType(input, input.type), ensureNoInitializer(input)));
-                    case 158 /* PropertySignature */:
+                    case 162 /* PropertySignature */:
                         if (ts.isPrivateIdentifier(input.name)) {
                             return cleanup(/*returnValue*/ undefined);
                         }
-                        return cleanup(ts.updatePropertySignature(input, ensureModifiers(input), input.name, input.questionToken, ensureType(input, input.type), ensureNoInitializer(input)));
-                    case 160 /* MethodSignature */: {
+                        return cleanup(factory.updatePropertySignature(input, ensureModifiers(input), input.name, input.questionToken, ensureType(input, input.type)));
+                    case 164 /* MethodSignature */: {
                         if (ts.isPrivateIdentifier(input.name)) {
                             return cleanup(/*returnValue*/ undefined);
                         }
-                        return cleanup(ts.updateMethodSignature(input, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type), input.name, input.questionToken));
+                        return cleanup(factory.updateMethodSignature(input, ensureModifiers(input), input.name, input.questionToken, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type)));
                     }
-                    case 165 /* CallSignature */: {
-                        return cleanup(ts.updateCallSignature(input, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type)));
+                    case 169 /* CallSignature */: {
+                        return cleanup(factory.updateCallSignature(input, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type)));
                     }
-                    case 167 /* IndexSignature */: {
-                        return cleanup(ts.updateIndexSignature(input, 
-                        /*decorators*/ undefined, ensureModifiers(input), updateParamsList(input, input.parameters), ts.visitNode(input.type, visitDeclarationSubtree) || ts.createKeywordTypeNode(125 /* AnyKeyword */)));
+                    case 171 /* IndexSignature */: {
+                        return cleanup(factory.updateIndexSignature(input, 
+                        /*decorators*/ undefined, ensureModifiers(input), updateParamsList(input, input.parameters), ts.visitNode(input.type, visitDeclarationSubtree) || factory.createKeywordTypeNode(128 /* AnyKeyword */)));
                     }
-                    case 242 /* VariableDeclaration */: {
+                    case 249 /* VariableDeclaration */: {
                         if (ts.isBindingPattern(input.name)) {
                             return recreateBindingPattern(input.name);
                         }
                         shouldEnterSuppressNewDiagnosticsContextContext = true;
                         suppressNewDiagnosticContexts = true; // Variable declaration types also suppress new diagnostic contexts, provided the contexts wouldn't be made for binding pattern types
-                        return cleanup(ts.updateTypeScriptVariableDeclaration(input, input.name, /*exclaimationToken*/ undefined, ensureType(input, input.type), ensureNoInitializer(input)));
+                        return cleanup(factory.updateVariableDeclaration(input, input.name, /*exclamationToken*/ undefined, ensureType(input, input.type), ensureNoInitializer(input)));
                     }
-                    case 155 /* TypeParameter */: {
+                    case 159 /* TypeParameter */: {
                         if (isPrivateMethodTypeParameter(input) && (input.default || input.constraint)) {
-                            return cleanup(ts.updateTypeParameterDeclaration(input, input.name, /*constraint*/ undefined, /*defaultType*/ undefined));
+                            return cleanup(factory.updateTypeParameterDeclaration(input, input.name, /*constraint*/ undefined, /*defaultType*/ undefined));
                         }
                         return cleanup(ts.visitEachChild(input, visitDeclarationSubtree, context));
                     }
-                    case 180 /* ConditionalType */: {
+                    case 184 /* ConditionalType */: {
                         // We have to process conditional types in a special way because for visibility purposes we need to push a new enclosingDeclaration
                         // just for the `infer` types in the true branch. It's an implicit declaration scope that only applies to _part_ of the type.
                         var checkType = ts.visitNode(input.checkType, visitDeclarationSubtree);
@@ -93363,22 +98369,25 @@ var ts;
                         var trueType = ts.visitNode(input.trueType, visitDeclarationSubtree);
                         enclosingDeclaration = oldEnclosingDecl;
                         var falseType = ts.visitNode(input.falseType, visitDeclarationSubtree);
-                        return cleanup(ts.updateConditionalTypeNode(input, checkType, extendsType, trueType, falseType));
+                        return cleanup(factory.updateConditionalTypeNode(input, checkType, extendsType, trueType, falseType));
                     }
-                    case 170 /* FunctionType */: {
-                        return cleanup(ts.updateFunctionTypeNode(input, ts.visitNodes(input.typeParameters, visitDeclarationSubtree), updateParamsList(input, input.parameters), ts.visitNode(input.type, visitDeclarationSubtree)));
+                    case 174 /* FunctionType */: {
+                        return cleanup(factory.updateFunctionTypeNode(input, ts.visitNodes(input.typeParameters, visitDeclarationSubtree), updateParamsList(input, input.parameters), ts.visitNode(input.type, visitDeclarationSubtree)));
                     }
-                    case 171 /* ConstructorType */: {
-                        return cleanup(ts.updateConstructorTypeNode(input, ts.visitNodes(input.typeParameters, visitDeclarationSubtree), updateParamsList(input, input.parameters), ts.visitNode(input.type, visitDeclarationSubtree)));
+                    case 175 /* ConstructorType */: {
+                        return cleanup(factory.updateConstructorTypeNode(input, ts.visitNodes(input.typeParameters, visitDeclarationSubtree), updateParamsList(input, input.parameters), ts.visitNode(input.type, visitDeclarationSubtree)));
                     }
-                    case 188 /* ImportType */: {
+                    case 195 /* ImportType */: {
                         if (!ts.isLiteralImportTypeNode(input))
                             return cleanup(input);
-                        return cleanup(ts.updateImportTypeNode(input, ts.updateLiteralTypeNode(input.argument, rewriteModuleSpecifier(input, input.argument.literal)), input.qualifier, ts.visitNodes(input.typeArguments, visitDeclarationSubtree, ts.isTypeNode), input.isTypeOf));
+                        return cleanup(factory.updateImportTypeNode(input, factory.updateLiteralTypeNode(input.argument, rewriteModuleSpecifier(input, input.argument.literal)), input.qualifier, ts.visitNodes(input.typeArguments, visitDeclarationSubtree, ts.isTypeNode), input.isTypeOf));
                     }
                     default: ts.Debug.assertNever(input, "Attempted to process unhandled node kind: " + ts.SyntaxKind[input.kind]);
                 }
             }
+            if (ts.isTupleTypeNode(input) && (ts.getLineAndCharacterOfPosition(currentSourceFile, input.pos).line === ts.getLineAndCharacterOfPosition(currentSourceFile, input.end).line)) {
+                ts.setEmitFlags(input, 1 /* SingleLine */);
+            }
             return cleanup(ts.visitEachChild(input, visitDeclarationSubtree, context));
             function cleanup(returnValue) {
                 if (returnValue && canProduceDiagnostic && ts.hasDynamicName(input)) {
@@ -93400,7 +98409,7 @@ var ts;
             }
         }
         function isPrivateMethodTypeParameter(node) {
-            return node.parent.kind === 161 /* MethodDeclaration */ && ts.hasModifier(node.parent, 8 /* Private */);
+            return node.parent.kind === 165 /* MethodDeclaration */ && ts.hasEffectiveModifier(node.parent, 8 /* Private */);
         }
         function visitDeclarationStatements(input) {
             if (!isPreservedDeclarationStatement(input)) {
@@ -93410,61 +98419,59 @@ var ts;
             if (shouldStripInternal(input))
                 return;
             switch (input.kind) {
-                case 260 /* ExportDeclaration */: {
+                case 267 /* ExportDeclaration */: {
                     if (ts.isSourceFile(input.parent)) {
                         resultHasExternalModuleIndicator = true;
                     }
                     resultHasScopeMarker = true;
                     // Always visible if the parent node isn't dropped for being not visible
                     // Rewrite external module names if necessary
-                    return ts.updateExportDeclaration(input, 
-                    /*decorators*/ undefined, input.modifiers, input.exportClause, rewriteModuleSpecifier(input, input.moduleSpecifier), input.isTypeOnly);
+                    return factory.updateExportDeclaration(input, 
+                    /*decorators*/ undefined, input.modifiers, input.isTypeOnly, input.exportClause, rewriteModuleSpecifier(input, input.moduleSpecifier));
                 }
-                case 259 /* ExportAssignment */: {
+                case 266 /* ExportAssignment */: {
                     // Always visible if the parent node isn't dropped for being not visible
                     if (ts.isSourceFile(input.parent)) {
                         resultHasExternalModuleIndicator = true;
                     }
                     resultHasScopeMarker = true;
-                    if (input.expression.kind === 75 /* Identifier */) {
+                    if (input.expression.kind === 78 /* Identifier */) {
                         return input;
                     }
                     else {
-                        var newId = ts.createOptimisticUniqueName("_default");
+                        var newId = factory.createUniqueName("_default", 16 /* Optimistic */);
                         getSymbolAccessibilityDiagnostic = function () { return ({
                             diagnosticMessage: ts.Diagnostics.Default_export_of_the_module_has_or_is_using_private_name_0,
                             errorNode: input
                         }); };
-                        var varDecl = ts.createVariableDeclaration(newId, resolver.createTypeOfExpression(input.expression, input, declarationEmitNodeBuilderFlags, symbolTracker), /*initializer*/ undefined);
-                        var statement = ts.createVariableStatement(needsDeclare ? [ts.createModifier(130 /* DeclareKeyword */)] : [], ts.createVariableDeclarationList([varDecl], 2 /* Const */));
-                        return [statement, ts.updateExportAssignment(input, input.decorators, input.modifiers, newId)];
+                        var varDecl = factory.createVariableDeclaration(newId, /*exclamationToken*/ undefined, resolver.createTypeOfExpression(input.expression, input, declarationEmitNodeBuilderFlags, symbolTracker), /*initializer*/ undefined);
+                        var statement = factory.createVariableStatement(needsDeclare ? [factory.createModifier(133 /* DeclareKeyword */)] : [], factory.createVariableDeclarationList([varDecl], 2 /* Const */));
+                        return [statement, factory.updateExportAssignment(input, input.decorators, input.modifiers, newId)];
                     }
                 }
             }
             var result = transformTopLevelDeclaration(input);
             // Don't actually transform yet; just leave as original node - will be elided/swapped by late pass
-            lateStatementReplacementMap.set("" + ts.getOriginalNodeId(input), result);
+            lateStatementReplacementMap.set(ts.getOriginalNodeId(input), result);
             return input;
         }
         function stripExportModifiers(statement) {
-            if (ts.isImportEqualsDeclaration(statement) || ts.hasModifier(statement, 512 /* Default */)) {
+            if (ts.isImportEqualsDeclaration(statement) || ts.hasEffectiveModifier(statement, 512 /* Default */) || !ts.canHaveModifiers(statement)) {
                 // `export import` statements should remain as-is, as imports are _not_ implicitly exported in an ambient namespace
                 // Likewise, `export default` classes and the like and just be `default`, so we preserve their `export` modifiers, too
                 return statement;
             }
-            var clone = ts.getMutableClone(statement);
-            var modifiers = ts.createModifiersFromModifierFlags(ts.getModifierFlags(statement) & (3071 /* All */ ^ 1 /* Export */));
-            clone.modifiers = modifiers.length ? ts.createNodeArray(modifiers) : undefined;
-            return clone;
+            var modifiers = factory.createModifiersFromModifierFlags(ts.getEffectiveModifierFlags(statement) & (11263 /* All */ ^ 1 /* Export */));
+            return factory.updateModifiers(statement, modifiers);
         }
         function transformTopLevelDeclaration(input) {
             if (shouldStripInternal(input))
                 return;
             switch (input.kind) {
-                case 253 /* ImportEqualsDeclaration */: {
+                case 260 /* ImportEqualsDeclaration */: {
                     return transformImportEqualsDeclaration(input);
                 }
-                case 254 /* ImportDeclaration */: {
+                case 261 /* ImportDeclaration */: {
                     return transformImportDeclaration(input);
                 }
             }
@@ -93485,26 +98492,27 @@ var ts;
             }
             var previousNeedsDeclare = needsDeclare;
             switch (input.kind) {
-                case 247 /* TypeAliasDeclaration */: // Type aliases get `declare`d if need be (for legacy support), but that's all
-                    return cleanup(ts.updateTypeAliasDeclaration(input, 
+                case 254 /* TypeAliasDeclaration */: // Type aliases get `declare`d if need be (for legacy support), but that's all
+                    return cleanup(factory.updateTypeAliasDeclaration(input, 
                     /*decorators*/ undefined, ensureModifiers(input), input.name, ts.visitNodes(input.typeParameters, visitDeclarationSubtree, ts.isTypeParameterDeclaration), ts.visitNode(input.type, visitDeclarationSubtree, ts.isTypeNode)));
-                case 246 /* InterfaceDeclaration */: {
-                    return cleanup(ts.updateInterfaceDeclaration(input, 
+                case 253 /* InterfaceDeclaration */: {
+                    return cleanup(factory.updateInterfaceDeclaration(input, 
                     /*decorators*/ undefined, ensureModifiers(input), input.name, ensureTypeParams(input, input.typeParameters), transformHeritageClauses(input.heritageClauses), ts.visitNodes(input.members, visitDeclarationSubtree)));
                 }
-                case 244 /* FunctionDeclaration */: {
+                case 251 /* FunctionDeclaration */: {
                     // Generators lose their generator-ness, excepting their return type
-                    var clean = cleanup(ts.updateFunctionDeclaration(input, 
+                    var clean = cleanup(factory.updateFunctionDeclaration(input, 
                     /*decorators*/ undefined, ensureModifiers(input), 
                     /*asteriskToken*/ undefined, input.name, ensureTypeParams(input, input.typeParameters), updateParamsList(input, input.parameters), ensureType(input, input.type), 
                     /*body*/ undefined));
                     if (clean && resolver.isExpandoFunctionDeclaration(input)) {
                         var props = resolver.getPropertiesOfContainerFunction(input);
-                        var fakespace_1 = ts.createModuleDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, clean.name || ts.createIdentifier("_default"), ts.createModuleBlock([]), 16 /* Namespace */);
-                        fakespace_1.flags ^= 8 /* Synthesized */; // unset synthesized so it is usable as an enclosing declaration
-                        fakespace_1.parent = enclosingDeclaration;
+                        // Use parseNodeFactory so it is usable as an enclosing declaration
+                        var fakespace_1 = ts.parseNodeFactory.createModuleDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, clean.name || factory.createIdentifier("_default"), factory.createModuleBlock([]), 16 /* Namespace */);
+                        ts.setParent(fakespace_1, enclosingDeclaration);
                         fakespace_1.locals = ts.createSymbolTable(props);
                         fakespace_1.symbol = props[0].parent;
+                        var exportMappings_1 = [];
                         var declarations = ts.mapDefined(props, function (p) {
                             if (!ts.isPropertyAccessExpression(p.valueDeclaration)) {
                                 return undefined; // TODO GH#33569: Handle element access expressions that created late bound names (rather than silently omitting them)
@@ -93512,21 +98520,39 @@ var ts;
                             getSymbolAccessibilityDiagnostic = ts.createGetSymbolAccessibilityDiagnosticForNode(p.valueDeclaration);
                             var type = resolver.createTypeOfDeclaration(p.valueDeclaration, fakespace_1, declarationEmitNodeBuilderFlags, symbolTracker);
                             getSymbolAccessibilityDiagnostic = oldDiag;
-                            var varDecl = ts.createVariableDeclaration(ts.unescapeLeadingUnderscores(p.escapedName), type, /*initializer*/ undefined);
-                            return ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([varDecl]));
+                            var nameStr = ts.unescapeLeadingUnderscores(p.escapedName);
+                            var isNonContextualKeywordName = ts.isStringANonContextualKeyword(nameStr);
+                            var name = isNonContextualKeywordName ? factory.getGeneratedNameForNode(p.valueDeclaration) : factory.createIdentifier(nameStr);
+                            if (isNonContextualKeywordName) {
+                                exportMappings_1.push([name, nameStr]);
+                            }
+                            var varDecl = factory.createVariableDeclaration(name, /*exclamationToken*/ undefined, type, /*initializer*/ undefined);
+                            return factory.createVariableStatement(isNonContextualKeywordName ? undefined : [factory.createToken(92 /* ExportKeyword */)], factory.createVariableDeclarationList([varDecl]));
                         });
-                        var namespaceDecl = ts.createModuleDeclaration(/*decorators*/ undefined, ensureModifiers(input), input.name, ts.createModuleBlock(declarations), 16 /* Namespace */);
-                        if (!ts.hasModifier(clean, 512 /* Default */)) {
+                        if (!exportMappings_1.length) {
+                            declarations = ts.mapDefined(declarations, function (declaration) { return factory.updateModifiers(declaration, 0 /* None */); });
+                        }
+                        else {
+                            declarations.push(factory.createExportDeclaration(
+                            /*decorators*/ undefined, 
+                            /*modifiers*/ undefined, 
+                            /*isTypeOnly*/ false, factory.createNamedExports(ts.map(exportMappings_1, function (_a) {
+                                var gen = _a[0], exp = _a[1];
+                                return factory.createExportSpecifier(gen, exp);
+                            }))));
+                        }
+                        var namespaceDecl = factory.createModuleDeclaration(/*decorators*/ undefined, ensureModifiers(input), input.name, factory.createModuleBlock(declarations), 16 /* Namespace */);
+                        if (!ts.hasEffectiveModifier(clean, 512 /* Default */)) {
                             return [clean, namespaceDecl];
                         }
-                        var modifiers = ts.createModifiersFromModifierFlags((ts.getModifierFlags(clean) & ~513 /* ExportDefault */) | 2 /* Ambient */);
-                        var cleanDeclaration = ts.updateFunctionDeclaration(clean, 
+                        var modifiers = factory.createModifiersFromModifierFlags((ts.getEffectiveModifierFlags(clean) & ~513 /* ExportDefault */) | 2 /* Ambient */);
+                        var cleanDeclaration = factory.updateFunctionDeclaration(clean, 
                         /*decorators*/ undefined, modifiers, 
                         /*asteriskToken*/ undefined, clean.name, clean.typeParameters, clean.parameters, clean.type, 
                         /*body*/ undefined);
-                        var namespaceDeclaration = ts.updateModuleDeclaration(namespaceDecl, 
+                        var namespaceDeclaration = factory.updateModuleDeclaration(namespaceDecl, 
                         /*decorators*/ undefined, modifiers, namespaceDecl.name, namespaceDecl.body);
-                        var exportDefaultDeclaration = ts.createExportAssignment(
+                        var exportDefaultDeclaration = factory.createExportAssignment(
                         /*decorators*/ undefined, 
                         /*modifiers*/ undefined, 
                         /*isExportEquals*/ false, namespaceDecl.name);
@@ -93540,10 +98566,10 @@ var ts;
                         return clean;
                     }
                 }
-                case 249 /* ModuleDeclaration */: {
+                case 256 /* ModuleDeclaration */: {
                     needsDeclare = false;
                     var inner = input.body;
-                    if (inner && inner.kind === 250 /* ModuleBlock */) {
+                    if (inner && inner.kind === 257 /* ModuleBlock */) {
                         var oldNeedsScopeFix = needsScopeFixMarker;
                         var oldHasScopeFix = resultHasScopeMarker;
                         resultHasScopeMarker = false;
@@ -93559,18 +98585,18 @@ var ts;
                         // 3. Some things are exported, some are not, and there's no marker - add an empty marker
                         if (!ts.isGlobalScopeAugmentation(input) && !hasScopeMarker(lateStatements) && !resultHasScopeMarker) {
                             if (needsScopeFixMarker) {
-                                lateStatements = ts.createNodeArray(__spreadArrays(lateStatements, [ts.createEmptyExports()]));
+                                lateStatements = factory.createNodeArray(__spreadArrays(lateStatements, [ts.createEmptyExports(factory)]));
                             }
                             else {
                                 lateStatements = ts.visitNodes(lateStatements, stripExportModifiers);
                             }
                         }
-                        var body = ts.updateModuleBlock(inner, lateStatements);
+                        var body = factory.updateModuleBlock(inner, lateStatements);
                         needsDeclare = previousNeedsDeclare;
                         needsScopeFixMarker = oldNeedsScopeFix;
                         resultHasScopeMarker = oldHasScopeFix;
                         var mods = ensureModifiers(input);
-                        return cleanup(ts.updateModuleDeclaration(input, 
+                        return cleanup(factory.updateModuleDeclaration(input, 
                         /*decorators*/ undefined, mods, ts.isExternalModuleAugmentation(input) ? rewriteModuleSpecifier(input, input.name) : input.name, body));
                     }
                     else {
@@ -93579,26 +98605,26 @@ var ts;
                         needsDeclare = false;
                         ts.visitNode(inner, visitDeclarationStatements);
                         // eagerly transform nested namespaces (the nesting doesn't need any elision or painting done)
-                        var id = "" + ts.getOriginalNodeId(inner); // TODO: GH#18217
+                        var id = ts.getOriginalNodeId(inner); // TODO: GH#18217
                         var body = lateStatementReplacementMap.get(id);
                         lateStatementReplacementMap.delete(id);
-                        return cleanup(ts.updateModuleDeclaration(input, 
+                        return cleanup(factory.updateModuleDeclaration(input, 
                         /*decorators*/ undefined, mods, input.name, body));
                     }
                 }
-                case 245 /* ClassDeclaration */: {
-                    var modifiers = ts.createNodeArray(ensureModifiers(input));
+                case 252 /* ClassDeclaration */: {
+                    var modifiers = factory.createNodeArray(ensureModifiers(input));
                     var typeParameters = ensureTypeParams(input, input.typeParameters);
                     var ctor = ts.getFirstConstructorWithBody(input);
                     var parameterProperties = void 0;
                     if (ctor) {
                         var oldDiag_1 = getSymbolAccessibilityDiagnostic;
                         parameterProperties = ts.compact(ts.flatMap(ctor.parameters, function (param) {
-                            if (!ts.hasModifier(param, 92 /* ParameterPropertyModifier */) || shouldStripInternal(param))
+                            if (!ts.hasSyntacticModifier(param, 92 /* ParameterPropertyModifier */) || shouldStripInternal(param))
                                 return;
                             getSymbolAccessibilityDiagnostic = ts.createGetSymbolAccessibilityDiagnosticForNode(param);
-                            if (param.name.kind === 75 /* Identifier */) {
-                                return preserveJsDoc(ts.createProperty(
+                            if (param.name.kind === 78 /* Identifier */) {
+                                return preserveJsDoc(factory.createPropertyDeclaration(
                                 /*decorators*/ undefined, ensureModifiers(param), param.name, param.questionToken, ensureType(param, param.type), ensureNoInitializer(param)), param);
                             }
                             else {
@@ -93615,7 +98641,7 @@ var ts;
                                         elems = ts.concatenate(elems, walkBindingPattern(elem.name));
                                     }
                                     elems = elems || [];
-                                    elems.push(ts.createProperty(
+                                    elems.push(factory.createPropertyDeclaration(
                                     /*decorators*/ undefined, ensureModifiers(param), elem.name, 
                                     /*questionToken*/ undefined, ensureType(elem, /*type*/ undefined), 
                                     /*initializer*/ undefined));
@@ -93627,56 +98653,56 @@ var ts;
                     }
                     var hasPrivateIdentifier = ts.some(input.members, function (member) { return !!member.name && ts.isPrivateIdentifier(member.name); });
                     var privateIdentifier = hasPrivateIdentifier ? [
-                        ts.createProperty(
+                        factory.createPropertyDeclaration(
                         /*decorators*/ undefined, 
-                        /*modifiers*/ undefined, ts.createPrivateIdentifier("#private"), 
+                        /*modifiers*/ undefined, factory.createPrivateIdentifier("#private"), 
                         /*questionToken*/ undefined, 
                         /*type*/ undefined, 
                         /*initializer*/ undefined)
                     ] : undefined;
                     var memberNodes = ts.concatenate(ts.concatenate(privateIdentifier, parameterProperties), ts.visitNodes(input.members, visitDeclarationSubtree));
-                    var members = ts.createNodeArray(memberNodes);
+                    var members = factory.createNodeArray(memberNodes);
                     var extendsClause_1 = ts.getEffectiveBaseTypeNode(input);
-                    if (extendsClause_1 && !ts.isEntityNameExpression(extendsClause_1.expression) && extendsClause_1.expression.kind !== 100 /* NullKeyword */) {
+                    if (extendsClause_1 && !ts.isEntityNameExpression(extendsClause_1.expression) && extendsClause_1.expression.kind !== 103 /* NullKeyword */) {
                         // We must add a temporary declaration for the extends clause expression
                         var oldId = input.name ? ts.unescapeLeadingUnderscores(input.name.escapedText) : "default";
-                        var newId_1 = ts.createOptimisticUniqueName(oldId + "_base");
+                        var newId_1 = factory.createUniqueName(oldId + "_base", 16 /* Optimistic */);
                         getSymbolAccessibilityDiagnostic = function () { return ({
                             diagnosticMessage: ts.Diagnostics.extends_clause_of_exported_class_0_has_or_is_using_private_name_1,
                             errorNode: extendsClause_1,
                             typeName: input.name
                         }); };
-                        var varDecl = ts.createVariableDeclaration(newId_1, resolver.createTypeOfExpression(extendsClause_1.expression, input, declarationEmitNodeBuilderFlags, symbolTracker), /*initializer*/ undefined);
-                        var statement = ts.createVariableStatement(needsDeclare ? [ts.createModifier(130 /* DeclareKeyword */)] : [], ts.createVariableDeclarationList([varDecl], 2 /* Const */));
-                        var heritageClauses = ts.createNodeArray(ts.map(input.heritageClauses, function (clause) {
-                            if (clause.token === 90 /* ExtendsKeyword */) {
+                        var varDecl = factory.createVariableDeclaration(newId_1, /*exclamationToken*/ undefined, resolver.createTypeOfExpression(extendsClause_1.expression, input, declarationEmitNodeBuilderFlags, symbolTracker), /*initializer*/ undefined);
+                        var statement = factory.createVariableStatement(needsDeclare ? [factory.createModifier(133 /* DeclareKeyword */)] : [], factory.createVariableDeclarationList([varDecl], 2 /* Const */));
+                        var heritageClauses = factory.createNodeArray(ts.map(input.heritageClauses, function (clause) {
+                            if (clause.token === 93 /* ExtendsKeyword */) {
                                 var oldDiag_2 = getSymbolAccessibilityDiagnostic;
                                 getSymbolAccessibilityDiagnostic = ts.createGetSymbolAccessibilityDiagnosticForNode(clause.types[0]);
-                                var newClause = ts.updateHeritageClause(clause, ts.map(clause.types, function (t) { return ts.updateExpressionWithTypeArguments(t, ts.visitNodes(t.typeArguments, visitDeclarationSubtree), newId_1); }));
+                                var newClause = factory.updateHeritageClause(clause, ts.map(clause.types, function (t) { return factory.updateExpressionWithTypeArguments(t, newId_1, ts.visitNodes(t.typeArguments, visitDeclarationSubtree)); }));
                                 getSymbolAccessibilityDiagnostic = oldDiag_2;
                                 return newClause;
                             }
-                            return ts.updateHeritageClause(clause, ts.visitNodes(ts.createNodeArray(ts.filter(clause.types, function (t) { return ts.isEntityNameExpression(t.expression) || t.expression.kind === 100 /* NullKeyword */; })), visitDeclarationSubtree));
+                            return factory.updateHeritageClause(clause, ts.visitNodes(factory.createNodeArray(ts.filter(clause.types, function (t) { return ts.isEntityNameExpression(t.expression) || t.expression.kind === 103 /* NullKeyword */; })), visitDeclarationSubtree));
                         }));
-                        return [statement, cleanup(ts.updateClassDeclaration(input, 
+                        return [statement, cleanup(factory.updateClassDeclaration(input, 
                             /*decorators*/ undefined, modifiers, input.name, typeParameters, heritageClauses, members))]; // TODO: GH#18217
                     }
                     else {
                         var heritageClauses = transformHeritageClauses(input.heritageClauses);
-                        return cleanup(ts.updateClassDeclaration(input, 
+                        return cleanup(factory.updateClassDeclaration(input, 
                         /*decorators*/ undefined, modifiers, input.name, typeParameters, heritageClauses, members));
                     }
                 }
-                case 225 /* VariableStatement */: {
+                case 232 /* VariableStatement */: {
                     return cleanup(transformVariableStatement(input));
                 }
-                case 248 /* EnumDeclaration */: {
-                    return cleanup(ts.updateEnumDeclaration(input, /*decorators*/ undefined, ts.createNodeArray(ensureModifiers(input)), input.name, ts.createNodeArray(ts.mapDefined(input.members, function (m) {
+                case 255 /* EnumDeclaration */: {
+                    return cleanup(factory.updateEnumDeclaration(input, /*decorators*/ undefined, factory.createNodeArray(ensureModifiers(input)), input.name, factory.createNodeArray(ts.mapDefined(input.members, function (m) {
                         if (shouldStripInternal(m))
                             return;
                         // Rewrite enum values to their constants, if available
                         var constValue = resolver.getConstantValue(m);
-                        return preserveJsDoc(ts.updateEnumMember(m, m.name, constValue !== undefined ? ts.createLiteral(constValue) : undefined), m);
+                        return preserveJsDoc(factory.updateEnumMember(m, m.name, constValue !== undefined ? typeof constValue === "string" ? factory.createStringLiteral(constValue) : factory.createNumericLiteral(constValue) : undefined), m);
                     }))));
                 }
             }
@@ -93689,7 +98715,7 @@ var ts;
                 if (canProdiceDiagnostic) {
                     getSymbolAccessibilityDiagnostic = oldDiag;
                 }
-                if (input.kind === 249 /* ModuleDeclaration */) {
+                if (input.kind === 256 /* ModuleDeclaration */) {
                     needsDeclare = previousNeedsDeclare;
                 }
                 if (node === input) {
@@ -93704,13 +98730,13 @@ var ts;
             var nodes = ts.visitNodes(input.declarationList.declarations, visitDeclarationSubtree);
             if (!ts.length(nodes))
                 return;
-            return ts.updateVariableStatement(input, ts.createNodeArray(ensureModifiers(input)), ts.updateVariableDeclarationList(input.declarationList, nodes));
+            return factory.updateVariableStatement(input, factory.createNodeArray(ensureModifiers(input)), factory.updateVariableDeclarationList(input.declarationList, nodes));
         }
         function recreateBindingPattern(d) {
             return ts.flatten(ts.mapDefined(d.elements, function (e) { return recreateBindingElement(e); }));
         }
         function recreateBindingElement(e) {
-            if (e.kind === 215 /* OmittedExpression */) {
+            if (e.kind === 222 /* OmittedExpression */) {
                 return;
             }
             if (e.name) {
@@ -93720,7 +98746,7 @@ var ts;
                     return recreateBindingPattern(e.name);
                 }
                 else {
-                    return ts.createVariableDeclaration(e.name, ensureType(e, /*type*/ undefined), /*initializer*/ undefined);
+                    return factory.createVariableDeclaration(e.name, /*exclamationToken*/ undefined, ensureType(e, /*type*/ undefined), /*initializer*/ undefined);
                 }
             }
         }
@@ -93750,17 +98776,17 @@ var ts;
             return ts.some(statements, isScopeMarker);
         }
         function ensureModifiers(node) {
-            var currentFlags = ts.getModifierFlags(node);
+            var currentFlags = ts.getEffectiveModifierFlags(node);
             var newFlags = ensureModifierFlags(node);
             if (currentFlags === newFlags) {
                 return node.modifiers;
             }
-            return ts.createModifiersFromModifierFlags(newFlags);
+            return factory.createModifiersFromModifierFlags(newFlags);
         }
         function ensureModifierFlags(node) {
-            var mask = 3071 /* All */ ^ (4 /* Public */ | 256 /* Async */); // No async modifiers in declaration files
+            var mask = 11263 /* All */ ^ (4 /* Public */ | 256 /* Async */); // No async modifiers in declaration files
             var additions = (needsDeclare && !isAlwaysType(node)) ? 2 /* Ambient */ : 0 /* None */;
-            var parentIsFile = node.parent.kind === 290 /* SourceFile */;
+            var parentIsFile = node.parent.kind === 297 /* SourceFile */;
             if (!parentIsFile || (isBundledEmit && parentIsFile && ts.isExternalModule(node.parent))) {
                 mask ^= 2 /* Ambient */;
                 additions = 0 /* None */;
@@ -93782,26 +98808,26 @@ var ts;
             return accessorType;
         }
         function transformHeritageClauses(nodes) {
-            return ts.createNodeArray(ts.filter(ts.map(nodes, function (clause) { return ts.updateHeritageClause(clause, ts.visitNodes(ts.createNodeArray(ts.filter(clause.types, function (t) {
-                return ts.isEntityNameExpression(t.expression) || (clause.token === 90 /* ExtendsKeyword */ && t.expression.kind === 100 /* NullKeyword */);
+            return factory.createNodeArray(ts.filter(ts.map(nodes, function (clause) { return factory.updateHeritageClause(clause, ts.visitNodes(factory.createNodeArray(ts.filter(clause.types, function (t) {
+                return ts.isEntityNameExpression(t.expression) || (clause.token === 93 /* ExtendsKeyword */ && t.expression.kind === 103 /* NullKeyword */);
             })), visitDeclarationSubtree)); }), function (clause) { return clause.types && !!clause.types.length; }));
         }
     }
     ts.transformDeclarations = transformDeclarations;
     function isAlwaysType(node) {
-        if (node.kind === 246 /* InterfaceDeclaration */) {
+        if (node.kind === 253 /* InterfaceDeclaration */) {
             return true;
         }
         return false;
     }
     // Elide "public" modifier, as it is the default
     function maskModifiers(node, modifierMask, modifierAdditions) {
-        return ts.createModifiersFromModifierFlags(maskModifierFlags(node, modifierMask, modifierAdditions));
+        return ts.factory.createModifiersFromModifierFlags(maskModifierFlags(node, modifierMask, modifierAdditions));
     }
     function maskModifierFlags(node, modifierMask, modifierAdditions) {
-        if (modifierMask === void 0) { modifierMask = 3071 /* All */ ^ 4 /* Public */; }
+        if (modifierMask === void 0) { modifierMask = 11263 /* All */ ^ 4 /* Public */; }
         if (modifierAdditions === void 0) { modifierAdditions = 0 /* None */; }
-        var flags = (ts.getModifierFlags(node) & modifierMask) | modifierAdditions;
+        var flags = (ts.getEffectiveModifierFlags(node) & modifierMask) | modifierAdditions;
         if (flags & 512 /* Default */ && !(flags & 1 /* Export */)) {
             // A non-exported default is a nonsequitor - we usually try to remove all export modifiers
             // from statements in ambient declarations; but a default export must retain its export modifier to be syntactically valid
@@ -93814,7 +98840,7 @@ var ts;
     }
     function getTypeAnnotationFromAccessor(accessor) {
         if (accessor) {
-            return accessor.kind === 163 /* GetAccessor */
+            return accessor.kind === 167 /* GetAccessor */
                 ? accessor.type // Getter - return type
                 : accessor.parameters.length > 0
                     ? accessor.parameters[0].type // Setter parameter type
@@ -93823,52 +98849,52 @@ var ts;
     }
     function canHaveLiteralInitializer(node) {
         switch (node.kind) {
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-                return !ts.hasModifier(node, 8 /* Private */);
-            case 156 /* Parameter */:
-            case 242 /* VariableDeclaration */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+                return !ts.hasEffectiveModifier(node, 8 /* Private */);
+            case 160 /* Parameter */:
+            case 249 /* VariableDeclaration */:
                 return true;
         }
         return false;
     }
     function isPreservedDeclarationStatement(node) {
         switch (node.kind) {
-            case 244 /* FunctionDeclaration */:
-            case 249 /* ModuleDeclaration */:
-            case 253 /* ImportEqualsDeclaration */:
-            case 246 /* InterfaceDeclaration */:
-            case 245 /* ClassDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-            case 248 /* EnumDeclaration */:
-            case 225 /* VariableStatement */:
-            case 254 /* ImportDeclaration */:
-            case 260 /* ExportDeclaration */:
-            case 259 /* ExportAssignment */:
+            case 251 /* FunctionDeclaration */:
+            case 256 /* ModuleDeclaration */:
+            case 260 /* ImportEqualsDeclaration */:
+            case 253 /* InterfaceDeclaration */:
+            case 252 /* ClassDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+            case 255 /* EnumDeclaration */:
+            case 232 /* VariableStatement */:
+            case 261 /* ImportDeclaration */:
+            case 267 /* ExportDeclaration */:
+            case 266 /* ExportAssignment */:
                 return true;
         }
         return false;
     }
     function isProcessedComponent(node) {
         switch (node.kind) {
-            case 166 /* ConstructSignature */:
-            case 162 /* Constructor */:
-            case 161 /* MethodDeclaration */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-            case 160 /* MethodSignature */:
-            case 165 /* CallSignature */:
-            case 167 /* IndexSignature */:
-            case 242 /* VariableDeclaration */:
-            case 155 /* TypeParameter */:
-            case 216 /* ExpressionWithTypeArguments */:
-            case 169 /* TypeReference */:
-            case 180 /* ConditionalType */:
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
-            case 188 /* ImportType */:
+            case 170 /* ConstructSignature */:
+            case 166 /* Constructor */:
+            case 165 /* MethodDeclaration */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+            case 164 /* MethodSignature */:
+            case 169 /* CallSignature */:
+            case 171 /* IndexSignature */:
+            case 249 /* VariableDeclaration */:
+            case 159 /* TypeParameter */:
+            case 223 /* ExpressionWithTypeArguments */:
+            case 173 /* TypeReference */:
+            case 184 /* ConditionalType */:
+            case 174 /* FunctionType */:
+            case 175 /* ConstructorType */:
+            case 195 /* ImportType */:
                 return true;
         }
         return false;
@@ -93912,14 +98938,13 @@ var ts;
     function getScriptTransformers(compilerOptions, customTransformers, emitOnlyDtsFiles) {
         if (emitOnlyDtsFiles)
             return ts.emptyArray;
-        var jsx = compilerOptions.jsx;
         var languageVersion = ts.getEmitScriptTarget(compilerOptions);
         var moduleKind = ts.getEmitModuleKind(compilerOptions);
         var transformers = [];
         ts.addRange(transformers, customTransformers && ts.map(customTransformers.before, wrapScriptTransformerFactory));
         transformers.push(ts.transformTypeScript);
         transformers.push(ts.transformClassFields);
-        if (jsx === 2 /* React */) {
+        if (ts.getJSXTransformEnabled(compilerOptions)) {
             transformers.push(ts.transformJsx);
         }
         if (languageVersion < 99 /* ESNext */) {
@@ -93972,7 +98997,7 @@ var ts;
         return function (context) {
             var customTransformer = transformer(context);
             return typeof customTransformer === "function"
-                ? handleDefault(customTransformer)
+                ? handleDefault(context, customTransformer)
                 : wrapCustomTransformer(customTransformer);
         };
     }
@@ -93980,7 +99005,7 @@ var ts;
         return wrapCustomTransformerFactory(transformer, ts.chainBundle);
     }
     function wrapDeclarationTransformerFactory(transformer) {
-        return wrapCustomTransformerFactory(transformer, ts.identity);
+        return wrapCustomTransformerFactory(transformer, function (_, node) { return node; });
     }
     function noEmitSubstitution(_hint, node) {
         return node;
@@ -94000,8 +99025,8 @@ var ts;
      * @param transforms An array of `TransformerFactory` callbacks.
      * @param allowDtsFiles A value indicating whether to allow the transformation of .d.ts files.
      */
-    function transformNodes(resolver, host, options, nodes, transformers, allowDtsFiles) {
-        var enabledSyntaxKindFeatures = new Array(331 /* Count */);
+    function transformNodes(resolver, host, factory, options, nodes, transformers, allowDtsFiles) {
+        var enabledSyntaxKindFeatures = new Array(341 /* Count */);
         var lexicalEnvironmentVariableDeclarations;
         var lexicalEnvironmentFunctionDeclarations;
         var lexicalEnvironmentStatements;
@@ -94020,9 +99045,11 @@ var ts;
         // The transformation context is provided to each transformer as part of transformer
         // initialization.
         var context = {
+            factory: factory,
             getCompilerOptions: function () { return options; },
             getEmitResolver: function () { return resolver; },
             getEmitHost: function () { return host; },
+            getEmitHelperFactory: ts.memoize(function () { return ts.createEmitHelperFactory(context); }),
             startLexicalEnvironment: startLexicalEnvironment,
             suspendLexicalEnvironment: suspendLexicalEnvironment,
             resumeLexicalEnvironment: resumeLexicalEnvironment,
@@ -94055,8 +99082,8 @@ var ts;
             }
         };
         // Ensure the parse tree is clean before applying transformations
-        for (var _i = 0, nodes_4 = nodes; _i < nodes_4.length; _i++) {
-            var node = nodes_4[_i];
+        for (var _i = 0, nodes_2 = nodes; _i < nodes_2.length; _i++) {
+            var node = nodes_2[_i];
             ts.disposeEmitNodes(ts.getSourceFileOfNode(ts.getParseTreeNode(node)));
         }
         ts.performance.mark("beforeTransform");
@@ -94072,7 +99099,13 @@ var ts;
         // prevent modification of transformation hooks.
         state = 1 /* Initialized */;
         // Transform each node.
-        var transformed = ts.map(nodes, allowDtsFiles ? transformation : transformRoot);
+        var transformed = [];
+        for (var _a = 0, nodes_3 = nodes; _a < nodes_3.length; _a++) {
+            var node = nodes_3[_a];
+            ts.tracing.push("emit" /* Emit */, "transformNodes", node.kind === 297 /* SourceFile */ ? { path: node.path } : { kind: node.kind, pos: node.pos, end: node.end });
+            transformed.push((allowDtsFiles ? transformation : transformRoot)(node));
+            ts.tracing.pop();
+        }
         // prevent modification of the lexical environment.
         state = 2 /* Completed */;
         ts.performance.mark("afterTransform");
@@ -94154,7 +99187,7 @@ var ts;
         function hoistVariableDeclaration(name) {
             ts.Debug.assert(state > 0 /* Uninitialized */, "Cannot modify the lexical environment during initialization.");
             ts.Debug.assert(state < 2 /* Completed */, "Cannot modify the lexical environment after transformation has completed.");
-            var decl = ts.setEmitFlags(ts.createVariableDeclaration(name), 64 /* NoNestedSourceMaps */);
+            var decl = ts.setEmitFlags(factory.createVariableDeclaration(name), 64 /* NoNestedSourceMaps */);
             if (!lexicalEnvironmentVariableDeclarations) {
                 lexicalEnvironmentVariableDeclarations = [decl];
             }
@@ -94245,8 +99278,8 @@ var ts;
                     statements = __spreadArrays(lexicalEnvironmentFunctionDeclarations);
                 }
                 if (lexicalEnvironmentVariableDeclarations) {
-                    var statement = ts.createVariableStatement(
-                    /*modifiers*/ undefined, ts.createVariableDeclarationList(lexicalEnvironmentVariableDeclarations));
+                    var statement = factory.createVariableStatement(
+                    /*modifiers*/ undefined, factory.createVariableDeclarationList(lexicalEnvironmentVariableDeclarations));
                     ts.setEmitFlags(statement, 1048576 /* CustomPrologue */);
                     if (!statements) {
                         statements = [statement];
@@ -94308,8 +99341,8 @@ var ts;
         function dispose() {
             if (state < 3 /* Disposed */) {
                 // Clean up emit nodes on parse tree
-                for (var _i = 0, nodes_5 = nodes; _i < nodes_5.length; _i++) {
-                    var node = nodes_5[_i];
+                for (var _i = 0, nodes_4 = nodes; _i < nodes_4.length; _i++) {
+                    var node = nodes_4[_i];
                     ts.disposeEmitNodes(ts.getSourceFileOfNode(ts.getParseTreeNode(node)));
                 }
                 // Release references to external entries for GC purposes.
@@ -94326,6 +99359,31 @@ var ts;
         }
     }
     ts.transformNodes = transformNodes;
+    ts.nullTransformationContext = {
+        get factory() { return ts.factory; },
+        enableEmitNotification: ts.noop,
+        enableSubstitution: ts.noop,
+        endLexicalEnvironment: ts.returnUndefined,
+        getCompilerOptions: function () { return ({}); },
+        getEmitHost: ts.notImplemented,
+        getEmitResolver: ts.notImplemented,
+        getEmitHelperFactory: ts.notImplemented,
+        setLexicalEnvironmentFlags: ts.noop,
+        getLexicalEnvironmentFlags: function () { return 0; },
+        hoistFunctionDeclaration: ts.noop,
+        hoistVariableDeclaration: ts.noop,
+        addInitializationStatement: ts.noop,
+        isEmitNotificationEnabled: ts.notImplemented,
+        isSubstitutionEnabled: ts.notImplemented,
+        onEmitNode: ts.noop,
+        onSubstituteNode: ts.notImplemented,
+        readEmitHelpers: ts.notImplemented,
+        requestEmitHelper: ts.noop,
+        resumeLexicalEnvironment: ts.noop,
+        startLexicalEnvironment: ts.noop,
+        suspendLexicalEnvironment: ts.noop,
+        addDiagnostic: ts.noop,
+    };
 })(ts || (ts = {}));
 var ts;
 (function (ts) {
@@ -94350,10 +99408,10 @@ var ts;
         if (forceDtsEmit === void 0) { forceDtsEmit = false; }
         var sourceFiles = ts.isArray(sourceFilesOrTargetSourceFile) ? sourceFilesOrTargetSourceFile : ts.getSourceFilesToEmit(host, sourceFilesOrTargetSourceFile, forceDtsEmit);
         var options = host.getCompilerOptions();
-        if (options.outFile || options.out) {
+        if (ts.outFile(options)) {
             var prepends = host.getPrependNodes();
             if (sourceFiles.length || prepends.length) {
-                var bundle = ts.createBundle(sourceFiles, prepends);
+                var bundle = ts.factory.createBundle(sourceFiles, prepends);
                 var result = action(getOutputPathsFor(bundle, host, forceDtsEmit), bundle);
                 if (result) {
                     return result;
@@ -94371,7 +99429,7 @@ var ts;
                 }
             }
             if (includeBuildInfo) {
-                var buildInfoPath = getTsBuildInfoEmitOutputFilePath(host.getCompilerOptions());
+                var buildInfoPath = getTsBuildInfoEmitOutputFilePath(options);
                 if (buildInfoPath)
                     return action({ buildInfoPath: buildInfoPath }, /*sourceFileOrBundle*/ undefined);
             }
@@ -94384,7 +99442,7 @@ var ts;
             return undefined;
         if (options.tsBuildInfoFile)
             return options.tsBuildInfoFile;
-        var outPath = options.outFile || options.out;
+        var outPath = ts.outFile(options);
         var buildInfoExtensionLess;
         if (outPath) {
             buildInfoExtensionLess = ts.removeFileExtension(outPath);
@@ -94404,7 +99462,7 @@ var ts;
     ts.getTsBuildInfoEmitOutputFilePath = getTsBuildInfoEmitOutputFilePath;
     /*@internal*/
     function getOutputPathsForBundle(options, forceDtsPaths) {
-        var outPath = options.outFile || options.out;
+        var outPath = ts.outFile(options);
         var jsFilePath = options.emitDeclarationOnly ? undefined : outPath;
         var sourceMapFilePath = jsFilePath && getSourceMapFilePath(jsFilePath, options);
         var declarationFilePath = (forceDtsPaths || ts.getEmitDeclarations(options)) ? ts.removeFileExtension(outPath) + ".d.ts" /* Dts */ : undefined;
@@ -94416,7 +99474,7 @@ var ts;
     /*@internal*/
     function getOutputPathsFor(sourceFile, host, forceDtsPaths) {
         var options = host.getCompilerOptions();
-        if (sourceFile.kind === 291 /* Bundle */) {
+        if (sourceFile.kind === 298 /* Bundle */) {
             return getOutputPathsForBundle(options, forceDtsPaths);
         }
         else {
@@ -94526,7 +99584,7 @@ var ts;
     /*@internal*/
     function getAllProjectOutputs(configFile, ignoreCase) {
         var _a = createAddOutput(), addOutput = _a.addOutput, getOutputs = _a.getOutputs;
-        if (configFile.options.outFile || configFile.options.out) {
+        if (ts.outFile(configFile.options)) {
             getSingleOutputFileNames(configFile, addOutput);
         }
         else {
@@ -94543,7 +99601,7 @@ var ts;
         inputFileName = ts.normalizePath(inputFileName);
         ts.Debug.assert(ts.contains(commandLine.fileNames, inputFileName), "Expected fileName to be present in command line");
         var _a = createAddOutput(), addOutput = _a.addOutput, getOutputs = _a.getOutputs;
-        if (commandLine.options.outFile || commandLine.options.out) {
+        if (ts.outFile(commandLine.options)) {
             getSingleOutputFileNames(commandLine, addOutput);
         }
         else {
@@ -94554,7 +99612,7 @@ var ts;
     ts.getOutputFileNames = getOutputFileNames;
     /*@internal*/
     function getFirstProjectOutput(configFile, ignoreCase) {
-        if (configFile.options.outFile || configFile.options.out) {
+        if (ts.outFile(configFile.options)) {
             var jsFilePath = getOutputPathsForBundle(configFile.options, /*forceDtsPaths*/ false).jsFilePath;
             return ts.Debug.checkDefined(jsFilePath, "project " + configFile.options.configFilePath + " expected to have at least one output");
         }
@@ -94612,9 +99670,15 @@ var ts;
                     sourceFiles: sourceFileOrBundle.sourceFiles.map(function (file) { return relativeToBuildInfo(ts.getNormalizedAbsolutePath(file.fileName, host.getCurrentDirectory())); })
                 };
             }
+            ts.tracing.push("emit" /* Emit */, "emitJsFileOrBundle", { jsFilePath: jsFilePath });
             emitJsFileOrBundle(sourceFileOrBundle, jsFilePath, sourceMapFilePath, relativeToBuildInfo);
+            ts.tracing.pop();
+            ts.tracing.push("emit" /* Emit */, "emitDeclarationFileOrBundle", { declarationFilePath: declarationFilePath });
             emitDeclarationFileOrBundle(sourceFileOrBundle, declarationFilePath, declarationMapPath, relativeToBuildInfo);
+            ts.tracing.pop();
+            ts.tracing.push("emit" /* Emit */, "emitBuildInfo", { buildInfoPath: buildInfoPath });
             emitBuildInfo(bundleBuildInfo, buildInfoPath);
+            ts.tracing.pop();
             if (!emitSkipped && emittedFilesList) {
                 if (!emitOnlyDtsFiles) {
                     if (jsFilePath) {
@@ -94643,7 +99707,7 @@ var ts;
             if (!buildInfoPath || targetSourceFile || emitSkipped)
                 return;
             var program = host.getProgramBuildInfo();
-            if (host.isEmitBlocked(buildInfoPath) || compilerOptions.noEmit) {
+            if (host.isEmitBlocked(buildInfoPath)) {
                 emitSkipped = true;
                 return;
             }
@@ -94660,7 +99724,7 @@ var ts;
                 return;
             }
             // Transform the source files
-            var transform = ts.transformNodes(resolver, host, compilerOptions, [sourceFileOrBundle], scriptTransformers, /*allowDtsFiles*/ false);
+            var transform = ts.transformNodes(resolver, host, ts.factory, compilerOptions, [sourceFileOrBundle], scriptTransformers, /*allowDtsFiles*/ false);
             var printerOptions = {
                 removeComments: compilerOptions.removeComments,
                 newLine: compilerOptions.newLine,
@@ -94701,13 +99765,13 @@ var ts;
             var sourceFiles = ts.isSourceFile(sourceFileOrBundle) ? [sourceFileOrBundle] : sourceFileOrBundle.sourceFiles;
             var filesForEmit = forceDtsEmit ? sourceFiles : ts.filter(sourceFiles, ts.isSourceFileNotJson);
             // Setup and perform the transformation to retrieve declarations from the input files
-            var inputListOrBundle = (compilerOptions.outFile || compilerOptions.out) ? [ts.createBundle(filesForEmit, !ts.isSourceFile(sourceFileOrBundle) ? sourceFileOrBundle.prepends : undefined)] : filesForEmit;
+            var inputListOrBundle = ts.outFile(compilerOptions) ? [ts.factory.createBundle(filesForEmit, !ts.isSourceFile(sourceFileOrBundle) ? sourceFileOrBundle.prepends : undefined)] : filesForEmit;
             if (emitOnlyDtsFiles && !ts.getEmitDeclarations(compilerOptions)) {
                 // Checker wont collect the linked aliases since thats only done when declaration is enabled.
                 // Do that here when emitting only dts files
                 filesForEmit.forEach(collectLinkedAliases);
             }
-            var declarationTransform = ts.transformNodes(resolver, host, compilerOptions, inputListOrBundle, declarationTransformers, /*allowDtsFiles*/ false);
+            var declarationTransform = ts.transformNodes(resolver, host, ts.factory, compilerOptions, inputListOrBundle, declarationTransformers, /*allowDtsFiles*/ false);
             if (ts.length(declarationTransform.diagnostics)) {
                 for (var _a = 0, _b = declarationTransform.diagnostics; _a < _b.length; _a++) {
                     var diagnostic = _b[_a];
@@ -94746,7 +99810,7 @@ var ts;
                     mapRoot: compilerOptions.mapRoot,
                     extendedDiagnostics: compilerOptions.extendedDiagnostics,
                 });
-                if (forceDtsEmit && declarationTransform.transformed[0].kind === 290 /* SourceFile */) {
+                if (forceDtsEmit && declarationTransform.transformed[0].kind === 297 /* SourceFile */) {
                     var sourceFile = declarationTransform.transformed[0];
                     exportedModulesFromDeclarationEmit = sourceFile.exportedModulesFromDeclarationEmit;
                 }
@@ -94757,7 +99821,7 @@ var ts;
         }
         function collectLinkedAliases(node) {
             if (ts.isExportAssignment(node)) {
-                if (node.expression.kind === 75 /* Identifier */) {
+                if (node.expression.kind === 78 /* Identifier */) {
                     resolver.collectLinkedAliases(node.expression, /*setVisibility*/ true);
                 }
                 return;
@@ -94769,8 +99833,8 @@ var ts;
             ts.forEachChild(node, collectLinkedAliases);
         }
         function printSourceFileOrBundle(jsFilePath, sourceMapFilePath, sourceFileOrBundle, printer, mapOptions) {
-            var bundle = sourceFileOrBundle.kind === 291 /* Bundle */ ? sourceFileOrBundle : undefined;
-            var sourceFile = sourceFileOrBundle.kind === 290 /* SourceFile */ ? sourceFileOrBundle : undefined;
+            var bundle = sourceFileOrBundle.kind === 298 /* Bundle */ ? sourceFileOrBundle : undefined;
+            var sourceFile = sourceFileOrBundle.kind === 297 /* SourceFile */ ? sourceFileOrBundle : undefined;
             var sourceFiles = bundle ? bundle.sourceFiles : [sourceFile];
             var sourceMapGenerator;
             if (shouldEmitSourceMaps(mapOptions, sourceFileOrBundle)) {
@@ -94811,7 +99875,7 @@ var ts;
         }
         function shouldEmitSourceMaps(mapOptions, sourceFileOrBundle) {
             return (mapOptions.sourceMap || mapOptions.inlineSourceMap)
-                && (sourceFileOrBundle.kind !== 290 /* SourceFile */ || !ts.fileExtensionIs(sourceFileOrBundle.fileName, ".json" /* Json */));
+                && (sourceFileOrBundle.kind !== 297 /* SourceFile */ || !ts.fileExtensionIs(sourceFileOrBundle.fileName, ".json" /* Json */));
         }
         function getSourceRoot(mapOptions) {
             // Normalize source root and make sure it has trailing "/" so that it can be used to combine paths with the
@@ -94855,16 +99919,16 @@ var ts;
                 if (ts.getRootLength(sourceMapDir) === 0) {
                     // The relative paths are relative to the common directory
                     sourceMapDir = ts.combinePaths(host.getCommonSourceDirectory(), sourceMapDir);
-                    return ts.getRelativePathToDirectoryOrUrl(ts.getDirectoryPath(ts.normalizePath(filePath)), // get the relative sourceMapDir path based on jsFilePath
+                    return encodeURI(ts.getRelativePathToDirectoryOrUrl(ts.getDirectoryPath(ts.normalizePath(filePath)), // get the relative sourceMapDir path based on jsFilePath
                     ts.combinePaths(sourceMapDir, sourceMapFile), // this is where user expects to see sourceMap
                     host.getCurrentDirectory(), host.getCanonicalFileName, 
-                    /*isAbsolutePathAnUrl*/ true);
+                    /*isAbsolutePathAnUrl*/ true));
                 }
                 else {
-                    return ts.combinePaths(sourceMapDir, sourceMapFile);
+                    return encodeURI(ts.combinePaths(sourceMapDir, sourceMapFile));
                 }
             }
-            return sourceMapFile;
+            return encodeURI(sourceMapFile);
         }
     }
     ts.emitFiles = emitFiles;
@@ -94915,6 +99979,7 @@ var ts;
         getTypeReferenceDirectivesForSymbol: ts.notImplemented,
         isLiteralConstDeclaration: ts.notImplemented,
         getJsxFactoryEntity: ts.notImplemented,
+        getJsxFragmentFactoryEntity: ts.notImplemented,
         getAllAccessorDeclarations: ts.notImplemented,
         getSymbolOfExternalModuleSpecifier: ts.notImplemented,
         isBindingCapturedByNode: ts.notImplemented,
@@ -94922,26 +99987,28 @@ var ts;
         isImportRequiredByAugmentation: ts.notImplemented,
     };
     function createSourceFilesFromBundleBuildInfo(bundle, buildInfoDirectory, host) {
-        var sourceFiles = bundle.sourceFiles.map(function (fileName) {
-            var sourceFile = ts.createNode(290 /* SourceFile */, 0, 0);
-            sourceFile.fileName = ts.getRelativePathFromDirectory(host.getCurrentDirectory(), ts.getNormalizedAbsolutePath(fileName, buildInfoDirectory), !host.useCaseSensitiveFileNames());
-            sourceFile.text = "";
-            sourceFile.statements = ts.createNodeArray();
-            return sourceFile;
-        });
+        var _a;
         var jsBundle = ts.Debug.checkDefined(bundle.js);
-        ts.forEach(jsBundle.sources && jsBundle.sources.prologues, function (prologueInfo) {
-            var sourceFile = sourceFiles[prologueInfo.file];
-            sourceFile.text = prologueInfo.text;
-            sourceFile.end = prologueInfo.text.length;
-            sourceFile.statements = ts.createNodeArray(prologueInfo.directives.map(function (directive) {
-                var statement = ts.createNode(226 /* ExpressionStatement */, directive.pos, directive.end);
-                statement.expression = ts.createNode(10 /* StringLiteral */, directive.expression.pos, directive.expression.end);
-                statement.expression.text = directive.expression.text;
+        var prologueMap = ((_a = jsBundle.sources) === null || _a === void 0 ? void 0 : _a.prologues) && ts.arrayToMap(jsBundle.sources.prologues, function (prologueInfo) { return prologueInfo.file; });
+        return bundle.sourceFiles.map(function (fileName, index) {
+            var _a, _b;
+            var prologueInfo = prologueMap === null || prologueMap === void 0 ? void 0 : prologueMap.get(index);
+            var statements = prologueInfo === null || prologueInfo === void 0 ? void 0 : prologueInfo.directives.map(function (directive) {
+                var literal = ts.setTextRange(ts.factory.createStringLiteral(directive.expression.text), directive.expression);
+                var statement = ts.setTextRange(ts.factory.createExpressionStatement(literal), directive);
+                ts.setParent(literal, statement);
                 return statement;
-            }));
+            });
+            var eofToken = ts.factory.createToken(1 /* EndOfFileToken */);
+            var sourceFile = ts.factory.createSourceFile(statements !== null && statements !== void 0 ? statements : [], eofToken, 0 /* None */);
+            sourceFile.fileName = ts.getRelativePathFromDirectory(host.getCurrentDirectory(), ts.getNormalizedAbsolutePath(fileName, buildInfoDirectory), !host.useCaseSensitiveFileNames());
+            sourceFile.text = (_a = prologueInfo === null || prologueInfo === void 0 ? void 0 : prologueInfo.text) !== null && _a !== void 0 ? _a : "";
+            ts.setTextRangePosWidth(sourceFile, 0, (_b = prologueInfo === null || prologueInfo === void 0 ? void 0 : prologueInfo.text.length) !== null && _b !== void 0 ? _b : 0);
+            ts.setEachParent(sourceFile.statements, sourceFile);
+            ts.setTextRangePosWidth(eofToken, sourceFile.end, 0);
+            ts.setParent(eofToken, sourceFile);
+            return sourceFile;
         });
-        return sourceFiles;
     }
     /*@internal*/
     function emitUsingBuildInfo(config, host, getCommandLine, customTransformers) {
@@ -95051,7 +100118,7 @@ var ts;
         var extendedDiagnostics = !!printerOptions.extendedDiagnostics;
         var newLine = ts.getNewLineCharacter(printerOptions);
         var moduleKind = ts.getEmitModuleKind(printerOptions);
-        var bundledHelpers = ts.createMap();
+        var bundledHelpers = new ts.Map();
         var currentSourceFile;
         var nodeIdToGeneratedName; // Map of generated names for specific nodes.
         var autoGeneratedIdToGeneratedName; // Map of generated names for temp and loop variables.
@@ -95061,6 +100128,7 @@ var ts;
         var reservedNamesStack; // Stack of TempFlags reserved in enclosing name generation scopes.
         var reservedNames; // TempFlags to reserve in nested name generation scopes.
         var preserveSourceNewlines = printerOptions.preserveSourceNewlines; // Can be overridden inside nodes with the `IgnoreSourceNewlines` emit flag.
+        var nextListElementPos; // See comment in `getLeadingLineTerminatorCount`.
         var writer;
         var ownWriter; // Reusable `EmitTextWriter` for basic printing.
         var write = writeBase;
@@ -95075,6 +100143,8 @@ var ts;
         var sourceMapGenerator;
         var sourceMapSource;
         var sourceMapSourceIndex = -1;
+        var mostRecentlyAddedSourceMapSource;
+        var mostRecentlyAddedSourceMapSourceIndex = -1;
         // Comments
         var containerPos = -1;
         var containerEnd = -1;
@@ -95113,9 +100183,9 @@ var ts;
                     break;
             }
             switch (node.kind) {
-                case 290 /* SourceFile */: return printFile(node);
-                case 291 /* Bundle */: return printBundle(node);
-                case 292 /* UnparsedSource */: return printUnparsedSource(node);
+                case 297 /* SourceFile */: return printFile(node);
+                case 298 /* Bundle */: return printBundle(node);
+                case 299 /* UnparsedSource */: return printUnparsedSource(node);
             }
             writeNode(hint, node, sourceFile, beginPrint());
             return endPrint();
@@ -95305,7 +100375,7 @@ var ts;
         function reset() {
             nodeIdToGeneratedName = [];
             autoGeneratedIdToGeneratedName = [];
-            generatedNames = ts.createMap();
+            generatedNames = new ts.Set();
             tempFlagsStack = [];
             tempFlags = 0 /* Auto */;
             reservedNamesStack = [];
@@ -95371,12 +100441,12 @@ var ts;
                     }
                 // falls through
                 case 2 /* Comments */:
-                    if (!commentsDisabled && node.kind !== 290 /* SourceFile */) {
+                    if (!commentsDisabled && node.kind !== 297 /* SourceFile */) {
                         return pipelineEmitWithComments;
                     }
                 // falls through
                 case 3 /* SourceMaps */:
-                    if (!sourceMapsDisabled && node.kind !== 290 /* SourceFile */ && !ts.isInJsonFile(node)) {
+                    if (!sourceMapsDisabled && node.kind !== 297 /* SourceFile */ && !ts.isInJsonFile(node)) {
                         return pipelineEmitWithSourceMap;
                     }
                 // falls through
@@ -95418,278 +100488,288 @@ var ts;
                     case 16 /* TemplateMiddle */:
                     case 17 /* TemplateTail */:
                         return emitLiteral(node, /*jsxAttributeEscape*/ false);
-                    case 292 /* UnparsedSource */:
-                    case 286 /* UnparsedPrepend */:
+                    case 299 /* UnparsedSource */:
+                    case 293 /* UnparsedPrepend */:
                         return emitUnparsedSourceOrPrepend(node);
-                    case 285 /* UnparsedPrologue */:
+                    case 292 /* UnparsedPrologue */:
                         return writeUnparsedNode(node);
-                    case 287 /* UnparsedText */:
-                    case 288 /* UnparsedInternalText */:
+                    case 294 /* UnparsedText */:
+                    case 295 /* UnparsedInternalText */:
                         return emitUnparsedTextLike(node);
-                    case 289 /* UnparsedSyntheticReference */:
+                    case 296 /* UnparsedSyntheticReference */:
                         return emitUnparsedSyntheticReference(node);
                     // Identifiers
-                    case 75 /* Identifier */:
+                    case 78 /* Identifier */:
                         return emitIdentifier(node);
                     // PrivateIdentifiers
-                    case 76 /* PrivateIdentifier */:
+                    case 79 /* PrivateIdentifier */:
                         return emitPrivateIdentifier(node);
                     // Parse tree nodes
                     // Names
-                    case 153 /* QualifiedName */:
+                    case 157 /* QualifiedName */:
                         return emitQualifiedName(node);
-                    case 154 /* ComputedPropertyName */:
+                    case 158 /* ComputedPropertyName */:
                         return emitComputedPropertyName(node);
                     // Signature elements
-                    case 155 /* TypeParameter */:
+                    case 159 /* TypeParameter */:
                         return emitTypeParameter(node);
-                    case 156 /* Parameter */:
+                    case 160 /* Parameter */:
                         return emitParameter(node);
-                    case 157 /* Decorator */:
+                    case 161 /* Decorator */:
                         return emitDecorator(node);
                     // Type members
-                    case 158 /* PropertySignature */:
+                    case 162 /* PropertySignature */:
                         return emitPropertySignature(node);
-                    case 159 /* PropertyDeclaration */:
+                    case 163 /* PropertyDeclaration */:
                         return emitPropertyDeclaration(node);
-                    case 160 /* MethodSignature */:
+                    case 164 /* MethodSignature */:
                         return emitMethodSignature(node);
-                    case 161 /* MethodDeclaration */:
+                    case 165 /* MethodDeclaration */:
                         return emitMethodDeclaration(node);
-                    case 162 /* Constructor */:
+                    case 166 /* Constructor */:
                         return emitConstructor(node);
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
                         return emitAccessorDeclaration(node);
-                    case 165 /* CallSignature */:
+                    case 169 /* CallSignature */:
                         return emitCallSignature(node);
-                    case 166 /* ConstructSignature */:
+                    case 170 /* ConstructSignature */:
                         return emitConstructSignature(node);
-                    case 167 /* IndexSignature */:
+                    case 171 /* IndexSignature */:
                         return emitIndexSignature(node);
+                    case 194 /* TemplateLiteralTypeSpan */:
+                        return emitTemplateTypeSpan(node);
                     // Types
-                    case 168 /* TypePredicate */:
+                    case 172 /* TypePredicate */:
                         return emitTypePredicate(node);
-                    case 169 /* TypeReference */:
+                    case 173 /* TypeReference */:
                         return emitTypeReference(node);
-                    case 170 /* FunctionType */:
+                    case 174 /* FunctionType */:
                         return emitFunctionType(node);
-                    case 300 /* JSDocFunctionType */:
+                    case 308 /* JSDocFunctionType */:
                         return emitJSDocFunctionType(node);
-                    case 171 /* ConstructorType */:
+                    case 175 /* ConstructorType */:
                         return emitConstructorType(node);
-                    case 172 /* TypeQuery */:
+                    case 176 /* TypeQuery */:
                         return emitTypeQuery(node);
-                    case 173 /* TypeLiteral */:
+                    case 177 /* TypeLiteral */:
                         return emitTypeLiteral(node);
-                    case 174 /* ArrayType */:
+                    case 178 /* ArrayType */:
                         return emitArrayType(node);
-                    case 175 /* TupleType */:
+                    case 179 /* TupleType */:
                         return emitTupleType(node);
-                    case 176 /* OptionalType */:
+                    case 180 /* OptionalType */:
                         return emitOptionalType(node);
-                    case 178 /* UnionType */:
+                    case 182 /* UnionType */:
                         return emitUnionType(node);
-                    case 179 /* IntersectionType */:
+                    case 183 /* IntersectionType */:
                         return emitIntersectionType(node);
-                    case 180 /* ConditionalType */:
+                    case 184 /* ConditionalType */:
                         return emitConditionalType(node);
-                    case 181 /* InferType */:
+                    case 185 /* InferType */:
                         return emitInferType(node);
-                    case 182 /* ParenthesizedType */:
+                    case 186 /* ParenthesizedType */:
                         return emitParenthesizedType(node);
-                    case 216 /* ExpressionWithTypeArguments */:
+                    case 223 /* ExpressionWithTypeArguments */:
                         return emitExpressionWithTypeArguments(node);
-                    case 183 /* ThisType */:
+                    case 187 /* ThisType */:
                         return emitThisType();
-                    case 184 /* TypeOperator */:
+                    case 188 /* TypeOperator */:
                         return emitTypeOperator(node);
-                    case 185 /* IndexedAccessType */:
+                    case 189 /* IndexedAccessType */:
                         return emitIndexedAccessType(node);
-                    case 186 /* MappedType */:
+                    case 190 /* MappedType */:
                         return emitMappedType(node);
-                    case 187 /* LiteralType */:
+                    case 191 /* LiteralType */:
                         return emitLiteralType(node);
-                    case 188 /* ImportType */:
+                    case 193 /* TemplateLiteralType */:
+                        return emitTemplateType(node);
+                    case 195 /* ImportType */:
                         return emitImportTypeNode(node);
-                    case 295 /* JSDocAllType */:
+                    case 303 /* JSDocAllType */:
                         writePunctuation("*");
                         return;
-                    case 296 /* JSDocUnknownType */:
+                    case 304 /* JSDocUnknownType */:
                         writePunctuation("?");
                         return;
-                    case 297 /* JSDocNullableType */:
+                    case 305 /* JSDocNullableType */:
                         return emitJSDocNullableType(node);
-                    case 298 /* JSDocNonNullableType */:
+                    case 306 /* JSDocNonNullableType */:
                         return emitJSDocNonNullableType(node);
-                    case 299 /* JSDocOptionalType */:
+                    case 307 /* JSDocOptionalType */:
                         return emitJSDocOptionalType(node);
-                    case 177 /* RestType */:
-                    case 301 /* JSDocVariadicType */:
+                    case 181 /* RestType */:
+                    case 309 /* JSDocVariadicType */:
                         return emitRestOrJSDocVariadicType(node);
+                    case 192 /* NamedTupleMember */:
+                        return emitNamedTupleMember(node);
                     // Binding patterns
-                    case 189 /* ObjectBindingPattern */:
+                    case 196 /* ObjectBindingPattern */:
                         return emitObjectBindingPattern(node);
-                    case 190 /* ArrayBindingPattern */:
+                    case 197 /* ArrayBindingPattern */:
                         return emitArrayBindingPattern(node);
-                    case 191 /* BindingElement */:
+                    case 198 /* BindingElement */:
                         return emitBindingElement(node);
                     // Misc
-                    case 221 /* TemplateSpan */:
+                    case 228 /* TemplateSpan */:
                         return emitTemplateSpan(node);
-                    case 222 /* SemicolonClassElement */:
+                    case 229 /* SemicolonClassElement */:
                         return emitSemicolonClassElement();
                     // Statements
-                    case 223 /* Block */:
+                    case 230 /* Block */:
                         return emitBlock(node);
-                    case 225 /* VariableStatement */:
+                    case 232 /* VariableStatement */:
                         return emitVariableStatement(node);
-                    case 224 /* EmptyStatement */:
+                    case 231 /* EmptyStatement */:
                         return emitEmptyStatement(/*isEmbeddedStatement*/ false);
-                    case 226 /* ExpressionStatement */:
+                    case 233 /* ExpressionStatement */:
                         return emitExpressionStatement(node);
-                    case 227 /* IfStatement */:
+                    case 234 /* IfStatement */:
                         return emitIfStatement(node);
-                    case 228 /* DoStatement */:
+                    case 235 /* DoStatement */:
                         return emitDoStatement(node);
-                    case 229 /* WhileStatement */:
+                    case 236 /* WhileStatement */:
                         return emitWhileStatement(node);
-                    case 230 /* ForStatement */:
+                    case 237 /* ForStatement */:
                         return emitForStatement(node);
-                    case 231 /* ForInStatement */:
+                    case 238 /* ForInStatement */:
                         return emitForInStatement(node);
-                    case 232 /* ForOfStatement */:
+                    case 239 /* ForOfStatement */:
                         return emitForOfStatement(node);
-                    case 233 /* ContinueStatement */:
+                    case 240 /* ContinueStatement */:
                         return emitContinueStatement(node);
-                    case 234 /* BreakStatement */:
+                    case 241 /* BreakStatement */:
                         return emitBreakStatement(node);
-                    case 235 /* ReturnStatement */:
+                    case 242 /* ReturnStatement */:
                         return emitReturnStatement(node);
-                    case 236 /* WithStatement */:
+                    case 243 /* WithStatement */:
                         return emitWithStatement(node);
-                    case 237 /* SwitchStatement */:
+                    case 244 /* SwitchStatement */:
                         return emitSwitchStatement(node);
-                    case 238 /* LabeledStatement */:
+                    case 245 /* LabeledStatement */:
                         return emitLabeledStatement(node);
-                    case 239 /* ThrowStatement */:
+                    case 246 /* ThrowStatement */:
                         return emitThrowStatement(node);
-                    case 240 /* TryStatement */:
+                    case 247 /* TryStatement */:
                         return emitTryStatement(node);
-                    case 241 /* DebuggerStatement */:
+                    case 248 /* DebuggerStatement */:
                         return emitDebuggerStatement(node);
                     // Declarations
-                    case 242 /* VariableDeclaration */:
+                    case 249 /* VariableDeclaration */:
                         return emitVariableDeclaration(node);
-                    case 243 /* VariableDeclarationList */:
+                    case 250 /* VariableDeclarationList */:
                         return emitVariableDeclarationList(node);
-                    case 244 /* FunctionDeclaration */:
+                    case 251 /* FunctionDeclaration */:
                         return emitFunctionDeclaration(node);
-                    case 245 /* ClassDeclaration */:
+                    case 252 /* ClassDeclaration */:
                         return emitClassDeclaration(node);
-                    case 246 /* InterfaceDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
                         return emitInterfaceDeclaration(node);
-                    case 247 /* TypeAliasDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
                         return emitTypeAliasDeclaration(node);
-                    case 248 /* EnumDeclaration */:
+                    case 255 /* EnumDeclaration */:
                         return emitEnumDeclaration(node);
-                    case 249 /* ModuleDeclaration */:
+                    case 256 /* ModuleDeclaration */:
                         return emitModuleDeclaration(node);
-                    case 250 /* ModuleBlock */:
+                    case 257 /* ModuleBlock */:
                         return emitModuleBlock(node);
-                    case 251 /* CaseBlock */:
+                    case 258 /* CaseBlock */:
                         return emitCaseBlock(node);
-                    case 252 /* NamespaceExportDeclaration */:
+                    case 259 /* NamespaceExportDeclaration */:
                         return emitNamespaceExportDeclaration(node);
-                    case 253 /* ImportEqualsDeclaration */:
+                    case 260 /* ImportEqualsDeclaration */:
                         return emitImportEqualsDeclaration(node);
-                    case 254 /* ImportDeclaration */:
+                    case 261 /* ImportDeclaration */:
                         return emitImportDeclaration(node);
-                    case 255 /* ImportClause */:
+                    case 262 /* ImportClause */:
                         return emitImportClause(node);
-                    case 256 /* NamespaceImport */:
+                    case 263 /* NamespaceImport */:
                         return emitNamespaceImport(node);
-                    case 262 /* NamespaceExport */:
+                    case 269 /* NamespaceExport */:
                         return emitNamespaceExport(node);
-                    case 257 /* NamedImports */:
+                    case 264 /* NamedImports */:
                         return emitNamedImports(node);
-                    case 258 /* ImportSpecifier */:
+                    case 265 /* ImportSpecifier */:
                         return emitImportSpecifier(node);
-                    case 259 /* ExportAssignment */:
+                    case 266 /* ExportAssignment */:
                         return emitExportAssignment(node);
-                    case 260 /* ExportDeclaration */:
+                    case 267 /* ExportDeclaration */:
                         return emitExportDeclaration(node);
-                    case 261 /* NamedExports */:
+                    case 268 /* NamedExports */:
                         return emitNamedExports(node);
-                    case 263 /* ExportSpecifier */:
+                    case 270 /* ExportSpecifier */:
                         return emitExportSpecifier(node);
-                    case 264 /* MissingDeclaration */:
+                    case 271 /* MissingDeclaration */:
                         return;
                     // Module references
-                    case 265 /* ExternalModuleReference */:
+                    case 272 /* ExternalModuleReference */:
                         return emitExternalModuleReference(node);
                     // JSX (non-expression)
                     case 11 /* JsxText */:
                         return emitJsxText(node);
-                    case 268 /* JsxOpeningElement */:
-                    case 271 /* JsxOpeningFragment */:
+                    case 275 /* JsxOpeningElement */:
+                    case 278 /* JsxOpeningFragment */:
                         return emitJsxOpeningElementOrFragment(node);
-                    case 269 /* JsxClosingElement */:
-                    case 272 /* JsxClosingFragment */:
+                    case 276 /* JsxClosingElement */:
+                    case 279 /* JsxClosingFragment */:
                         return emitJsxClosingElementOrFragment(node);
-                    case 273 /* JsxAttribute */:
+                    case 280 /* JsxAttribute */:
                         return emitJsxAttribute(node);
-                    case 274 /* JsxAttributes */:
+                    case 281 /* JsxAttributes */:
                         return emitJsxAttributes(node);
-                    case 275 /* JsxSpreadAttribute */:
+                    case 282 /* JsxSpreadAttribute */:
                         return emitJsxSpreadAttribute(node);
-                    case 276 /* JsxExpression */:
+                    case 283 /* JsxExpression */:
                         return emitJsxExpression(node);
                     // Clauses
-                    case 277 /* CaseClause */:
+                    case 284 /* CaseClause */:
                         return emitCaseClause(node);
-                    case 278 /* DefaultClause */:
+                    case 285 /* DefaultClause */:
                         return emitDefaultClause(node);
-                    case 279 /* HeritageClause */:
+                    case 286 /* HeritageClause */:
                         return emitHeritageClause(node);
-                    case 280 /* CatchClause */:
+                    case 287 /* CatchClause */:
                         return emitCatchClause(node);
                     // Property assignments
-                    case 281 /* PropertyAssignment */:
+                    case 288 /* PropertyAssignment */:
                         return emitPropertyAssignment(node);
-                    case 282 /* ShorthandPropertyAssignment */:
+                    case 289 /* ShorthandPropertyAssignment */:
                         return emitShorthandPropertyAssignment(node);
-                    case 283 /* SpreadAssignment */:
+                    case 290 /* SpreadAssignment */:
                         return emitSpreadAssignment(node);
                     // Enum
-                    case 284 /* EnumMember */:
+                    case 291 /* EnumMember */:
                         return emitEnumMember(node);
                     // JSDoc nodes (only used in codefixes currently)
-                    case 317 /* JSDocParameterTag */:
-                    case 323 /* JSDocPropertyTag */:
+                    case 326 /* JSDocParameterTag */:
+                    case 333 /* JSDocPropertyTag */:
                         return emitJSDocPropertyLikeTag(node);
-                    case 318 /* JSDocReturnTag */:
-                    case 320 /* JSDocTypeTag */:
-                    case 319 /* JSDocThisTag */:
-                    case 316 /* JSDocEnumTag */:
+                    case 327 /* JSDocReturnTag */:
+                    case 329 /* JSDocTypeTag */:
+                    case 328 /* JSDocThisTag */:
+                    case 325 /* JSDocEnumTag */:
                         return emitJSDocSimpleTypedTag(node);
-                    case 308 /* JSDocImplementsTag */:
-                    case 307 /* JSDocAugmentsTag */:
+                    case 316 /* JSDocImplementsTag */:
+                    case 315 /* JSDocAugmentsTag */:
                         return emitJSDocHeritageTag(node);
-                    case 321 /* JSDocTemplateTag */:
+                    case 330 /* JSDocTemplateTag */:
                         return emitJSDocTemplateTag(node);
-                    case 322 /* JSDocTypedefTag */:
+                    case 331 /* JSDocTypedefTag */:
                         return emitJSDocTypedefTag(node);
-                    case 315 /* JSDocCallbackTag */:
+                    case 324 /* JSDocCallbackTag */:
                         return emitJSDocCallbackTag(node);
-                    case 305 /* JSDocSignature */:
+                    case 313 /* JSDocSignature */:
                         return emitJSDocSignature(node);
-                    case 304 /* JSDocTypeLiteral */:
+                    case 312 /* JSDocTypeLiteral */:
                         return emitJSDocTypeLiteral(node);
-                    case 310 /* JSDocClassTag */:
-                    case 306 /* JSDocTag */:
+                    case 319 /* JSDocClassTag */:
+                    case 314 /* JSDocTag */:
                         return emitJSDocSimpleTag(node);
-                    case 303 /* JSDocComment */:
+                    case 332 /* JSDocSeeTag */:
+                        return emitJSDocSeeTag(node);
+                    case 302 /* JSDocNameReference */:
+                        return emitJSDocNameReference(node);
+                    case 311 /* JSDocComment */:
                         return emitJSDoc(node);
                     // Transformation nodes (ignored)
                 }
@@ -95714,83 +100794,83 @@ var ts;
                     case 14 /* NoSubstitutionTemplateLiteral */:
                         return emitLiteral(node, /*jsxAttributeEscape*/ false);
                     // Identifiers
-                    case 75 /* Identifier */:
+                    case 78 /* Identifier */:
                         return emitIdentifier(node);
                     // Reserved words
-                    case 91 /* FalseKeyword */:
-                    case 100 /* NullKeyword */:
-                    case 102 /* SuperKeyword */:
-                    case 106 /* TrueKeyword */:
-                    case 104 /* ThisKeyword */:
-                    case 96 /* ImportKeyword */:
+                    case 94 /* FalseKeyword */:
+                    case 103 /* NullKeyword */:
+                    case 105 /* SuperKeyword */:
+                    case 109 /* TrueKeyword */:
+                    case 107 /* ThisKeyword */:
+                    case 99 /* ImportKeyword */:
                         writeTokenNode(node, writeKeyword);
                         return;
                     // Expressions
-                    case 192 /* ArrayLiteralExpression */:
+                    case 199 /* ArrayLiteralExpression */:
                         return emitArrayLiteralExpression(node);
-                    case 193 /* ObjectLiteralExpression */:
+                    case 200 /* ObjectLiteralExpression */:
                         return emitObjectLiteralExpression(node);
-                    case 194 /* PropertyAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
                         return emitPropertyAccessExpression(node);
-                    case 195 /* ElementAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
                         return emitElementAccessExpression(node);
-                    case 196 /* CallExpression */:
+                    case 203 /* CallExpression */:
                         return emitCallExpression(node);
-                    case 197 /* NewExpression */:
+                    case 204 /* NewExpression */:
                         return emitNewExpression(node);
-                    case 198 /* TaggedTemplateExpression */:
+                    case 205 /* TaggedTemplateExpression */:
                         return emitTaggedTemplateExpression(node);
-                    case 199 /* TypeAssertionExpression */:
+                    case 206 /* TypeAssertionExpression */:
                         return emitTypeAssertionExpression(node);
-                    case 200 /* ParenthesizedExpression */:
+                    case 207 /* ParenthesizedExpression */:
                         return emitParenthesizedExpression(node);
-                    case 201 /* FunctionExpression */:
+                    case 208 /* FunctionExpression */:
                         return emitFunctionExpression(node);
-                    case 202 /* ArrowFunction */:
+                    case 209 /* ArrowFunction */:
                         return emitArrowFunction(node);
-                    case 203 /* DeleteExpression */:
+                    case 210 /* DeleteExpression */:
                         return emitDeleteExpression(node);
-                    case 204 /* TypeOfExpression */:
+                    case 211 /* TypeOfExpression */:
                         return emitTypeOfExpression(node);
-                    case 205 /* VoidExpression */:
+                    case 212 /* VoidExpression */:
                         return emitVoidExpression(node);
-                    case 206 /* AwaitExpression */:
+                    case 213 /* AwaitExpression */:
                         return emitAwaitExpression(node);
-                    case 207 /* PrefixUnaryExpression */:
+                    case 214 /* PrefixUnaryExpression */:
                         return emitPrefixUnaryExpression(node);
-                    case 208 /* PostfixUnaryExpression */:
+                    case 215 /* PostfixUnaryExpression */:
                         return emitPostfixUnaryExpression(node);
-                    case 209 /* BinaryExpression */:
+                    case 216 /* BinaryExpression */:
                         return emitBinaryExpression(node);
-                    case 210 /* ConditionalExpression */:
+                    case 217 /* ConditionalExpression */:
                         return emitConditionalExpression(node);
-                    case 211 /* TemplateExpression */:
+                    case 218 /* TemplateExpression */:
                         return emitTemplateExpression(node);
-                    case 212 /* YieldExpression */:
+                    case 219 /* YieldExpression */:
                         return emitYieldExpression(node);
-                    case 213 /* SpreadElement */:
+                    case 220 /* SpreadElement */:
                         return emitSpreadExpression(node);
-                    case 214 /* ClassExpression */:
+                    case 221 /* ClassExpression */:
                         return emitClassExpression(node);
-                    case 215 /* OmittedExpression */:
+                    case 222 /* OmittedExpression */:
                         return;
-                    case 217 /* AsExpression */:
+                    case 224 /* AsExpression */:
                         return emitAsExpression(node);
-                    case 218 /* NonNullExpression */:
+                    case 225 /* NonNullExpression */:
                         return emitNonNullExpression(node);
-                    case 219 /* MetaProperty */:
+                    case 226 /* MetaProperty */:
                         return emitMetaProperty(node);
                     // JSX
-                    case 266 /* JsxElement */:
+                    case 273 /* JsxElement */:
                         return emitJsxElement(node);
-                    case 267 /* JsxSelfClosingElement */:
+                    case 274 /* JsxSelfClosingElement */:
                         return emitJsxSelfClosingElement(node);
-                    case 270 /* JsxFragment */:
+                    case 277 /* JsxFragment */:
                         return emitJsxFragment(node);
                     // Transformation nodes
-                    case 326 /* PartiallyEmittedExpression */:
+                    case 336 /* PartiallyEmittedExpression */:
                         return emitPartiallyEmittedExpression(node);
-                    case 327 /* CommaListExpression */:
+                    case 337 /* CommaListExpression */:
                         return emitCommaList(node);
                 }
             }
@@ -95813,15 +100893,15 @@ var ts;
             if (moduleKind === ts.ModuleKind.None || printerOptions.noEmitHelpers) {
                 return undefined;
             }
-            var bundledHelpers = ts.createMap();
+            var bundledHelpers = new ts.Map();
             for (var _a = 0, _b = bundle.sourceFiles; _a < _b.length; _a++) {
                 var sourceFile = _b[_a];
                 var shouldSkip = ts.getExternalHelpersModuleName(sourceFile) !== undefined;
                 var helpers = getSortedEmitHelpers(sourceFile);
                 if (!helpers)
                     continue;
-                for (var _c = 0, helpers_4 = helpers; _c < helpers_4.length; _c++) {
-                    var helper = helpers_4[_c];
+                for (var _c = 0, helpers_5 = helpers; _c < helpers_5.length; _c++) {
+                    var helper = helpers_5[_c];
                     if (!helper.scoped && !shouldSkip && !bundledHelpers.get(helper.name)) {
                         bundledHelpers.set(helper.name, true);
                         (result || (result = [])).push(helper.name);
@@ -95832,7 +100912,7 @@ var ts;
         }
         function emitHelpers(node) {
             var helpersEmitted = false;
-            var bundle = node.kind === 291 /* Bundle */ ? node : undefined;
+            var bundle = node.kind === 298 /* Bundle */ ? node : undefined;
             if (bundle && moduleKind === ts.ModuleKind.None) {
                 return;
             }
@@ -95845,8 +100925,8 @@ var ts;
                 var shouldBundle = (ts.isSourceFile(currentNode) || ts.isUnparsedSource(currentNode)) && !isOwnFileEmit;
                 var helpers = ts.isUnparsedSource(currentNode) ? currentNode.helpers : getSortedEmitHelpers(currentNode);
                 if (helpers) {
-                    for (var _a = 0, helpers_5 = helpers; _a < helpers_5.length; _a++) {
-                        var helper = helpers_5[_a];
+                    for (var _a = 0, helpers_6 = helpers; _a < helpers_6.length; _a++) {
+                        var helper = helpers_6[_a];
                         if (!helper.scoped) {
                             // Skip the helper if it can be skipped and the noEmitHelpers compiler
                             // option is set, or if it can be imported and the importHelpers compiler
@@ -95932,7 +101012,7 @@ var ts;
             var pos = getTextPosWithWriteLine();
             writeUnparsedNode(unparsed);
             if (bundleFileInfo) {
-                updateOrPushBundleFileTextLike(pos, writer.getTextPos(), unparsed.kind === 287 /* UnparsedText */ ?
+                updateOrPushBundleFileTextLike(pos, writer.getTextPos(), unparsed.kind === 294 /* UnparsedText */ ?
                     "text" /* Text */ :
                     "internal" /* Internal */);
             }
@@ -95969,7 +101049,7 @@ var ts;
             emit(node.right);
         }
         function emitEntityName(node) {
-            if (node.kind === 75 /* Identifier */) {
+            if (node.kind === 78 /* Identifier */) {
                 emitExpression(node);
             }
             else {
@@ -96005,7 +101085,7 @@ var ts;
             emit(node.dotDotDotToken);
             emitNodeWithWriter(node.name, writeParameter);
             emit(node.questionToken);
-            if (node.parent && node.parent.kind === 300 /* JSDocFunctionType */ && !node.name) {
+            if (node.parent && node.parent.kind === 308 /* JSDocFunctionType */ && !node.name) {
                 emit(node.type);
             }
             else {
@@ -96067,7 +101147,7 @@ var ts;
         function emitAccessorDeclaration(node) {
             emitDecorators(node, node.decorators);
             emitModifiers(node, node.modifiers);
-            writeKeyword(node.kind === 163 /* GetAccessor */ ? "get" : "set");
+            writeKeyword(node.kind === 167 /* GetAccessor */ ? "get" : "set");
             writeSpace();
             emit(node.name);
             emitSignatureAndBody(node, emitSignatureHead);
@@ -96101,6 +101181,10 @@ var ts;
             emitTypeAnnotation(node.type);
             writeTrailingSemicolon();
         }
+        function emitTemplateTypeSpan(node) {
+            emit(node.type);
+            emit(node.literal);
+        }
         function emitSemicolonClassElement() {
             writeTrailingSemicolon();
         }
@@ -96185,9 +101269,18 @@ var ts;
             emit(node.type);
         }
         function emitTupleType(node) {
-            writePunctuation("[");
-            emitList(node, node.elementTypes, 528 /* TupleTypeElements */);
-            writePunctuation("]");
+            emitTokenWithComment(22 /* OpenBracketToken */, node.pos, writePunctuation, node);
+            var flags = ts.getEmitFlags(node) & 1 /* SingleLine */ ? 528 /* SingleLineTupleTypeElements */ : 657 /* MultiLineTupleTypeElements */;
+            emitList(node, node.elements, flags | 524288 /* NoSpaceIfEmpty */);
+            emitTokenWithComment(23 /* CloseBracketToken */, node.elements.end, writePunctuation, node);
+        }
+        function emitNamedTupleMember(node) {
+            emit(node.dotDotDotToken);
+            emit(node.name);
+            emit(node.questionToken);
+            emitTokenWithComment(58 /* ColonToken */, node.name.end, writePunctuation, node);
+            writeSpace();
+            emit(node.type);
         }
         function emitOptionalType(node) {
             emit(node.type);
@@ -96250,13 +101343,19 @@ var ts;
             }
             if (node.readonlyToken) {
                 emit(node.readonlyToken);
-                if (node.readonlyToken.kind !== 138 /* ReadonlyKeyword */) {
+                if (node.readonlyToken.kind !== 142 /* ReadonlyKeyword */) {
                     writeKeyword("readonly");
                 }
                 writeSpace();
             }
             writePunctuation("[");
             pipelineEmit(3 /* MappedTypeParameter */, node.typeParameter);
+            if (node.nameType) {
+                writeSpace();
+                writeKeyword("as");
+                writeSpace();
+                emit(node.nameType);
+            }
             writePunctuation("]");
             if (node.questionToken) {
                 emit(node.questionToken);
@@ -96280,6 +101379,10 @@ var ts;
         function emitLiteralType(node) {
             emitExpression(node.literal);
         }
+        function emitTemplateType(node) {
+            emit(node.head);
+            emitList(node, node.templateSpans, 262144 /* TemplateExpressionSpans */);
+        }
         function emitImportTypeNode(node) {
             if (node.isTypeOf) {
                 writeKeyword("typeof");
@@ -96341,7 +101444,7 @@ var ts;
         }
         function emitPropertyAccessExpression(node) {
             var expression = ts.cast(emitExpression(node.expression), ts.isExpression);
-            var token = node.questionDotToken || ts.createNode(24 /* DotToken */, node.expression.end, node.name.pos);
+            var token = node.questionDotToken || ts.setTextRangePosEnd(ts.factory.createToken(24 /* DotToken */), node.expression.end, node.name.pos);
             var linesBeforeDot = getLinesBetweenNodes(node, node.expression, token);
             var linesAfterDot = getLinesBetweenNodes(node, token, node.name);
             writeLinesAndIndent(linesBeforeDot, /*writeSpaceIfNotIndenting*/ false);
@@ -96395,7 +101498,7 @@ var ts;
             emitExpressionList(node, node.arguments, 2576 /* CallExpressionArguments */);
         }
         function emitNewExpression(node) {
-            emitTokenWithComment(99 /* NewKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(102 /* NewKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
             emitTypeArguments(node, node.typeArguments);
@@ -96438,22 +101541,22 @@ var ts;
             emit(node.equalsGreaterThanToken);
         }
         function emitDeleteExpression(node) {
-            emitTokenWithComment(85 /* DeleteKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(88 /* DeleteKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
         }
         function emitTypeOfExpression(node) {
-            emitTokenWithComment(108 /* TypeOfKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(111 /* TypeOfKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
         }
         function emitVoidExpression(node) {
-            emitTokenWithComment(110 /* VoidKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(113 /* VoidKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
         }
         function emitAwaitExpression(node) {
-            emitTokenWithComment(127 /* AwaitKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(130 /* AwaitKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
         }
@@ -96478,7 +101581,7 @@ var ts;
             // expression a prefix increment whose operand is a plus expression - (++(+x))
             // The same is true of minus of course.
             var operand = node.operand;
-            return operand.kind === 207 /* PrefixUnaryExpression */
+            return operand.kind === 214 /* PrefixUnaryExpression */
                 && ((node.operator === 39 /* PlusToken */ && (operand.operator === 39 /* PlusToken */ || operand.operator === 45 /* PlusPlusToken */))
                     || (node.operator === 40 /* MinusToken */ && (operand.operator === 40 /* MinusToken */ || operand.operator === 46 /* MinusMinusToken */)));
         }
@@ -96514,7 +101617,7 @@ var ts;
                         var linesAfterOperator = getLinesBetweenNodes(node, node.operatorToken, node.right);
                         writeLinesAndIndent(linesBeforeOperator, isCommaOperator);
                         emitLeadingCommentsOfPosition(node.operatorToken.pos);
-                        writeTokenNode(node.operatorToken, node.operatorToken.kind === 97 /* InKeyword */ ? writeKeyword : writeOperator);
+                        writeTokenNode(node.operatorToken, node.operatorToken.kind === 100 /* InKeyword */ ? writeKeyword : writeOperator);
                         emitTrailingCommentsOfPosition(node.operatorToken.end, /*prefixSpace*/ true); // Binary operators should have a space before the comment starts
                         writeLinesAndIndent(linesAfterOperator, /*writeSpaceIfNotIndenting*/ true);
                         maybePipelineEmitExpression(node.right);
@@ -96580,7 +101683,7 @@ var ts;
             emitList(node, node.templateSpans, 262144 /* TemplateExpressionSpans */);
         }
         function emitYieldExpression(node) {
-            emitTokenWithComment(121 /* YieldKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(124 /* YieldKeyword */, node.pos, writeKeyword, node);
             emit(node.asteriskToken);
             emitExpressionWithLeadingSpace(node.expression);
         }
@@ -96657,16 +101760,16 @@ var ts;
             }
         }
         function emitIfStatement(node) {
-            var openParenPos = emitTokenWithComment(95 /* IfKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(98 /* IfKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
             emitExpression(node.expression);
             emitTokenWithComment(21 /* CloseParenToken */, node.expression.end, writePunctuation, node);
             emitEmbeddedStatement(node, node.thenStatement);
             if (node.elseStatement) {
-                writeLineOrSpace(node);
-                emitTokenWithComment(87 /* ElseKeyword */, node.thenStatement.end, writeKeyword, node);
-                if (node.elseStatement.kind === 227 /* IfStatement */) {
+                writeLineOrSpace(node, node.thenStatement, node.elseStatement);
+                emitTokenWithComment(90 /* ElseKeyword */, node.thenStatement.end, writeKeyword, node);
+                if (node.elseStatement.kind === 234 /* IfStatement */) {
                     writeSpace();
                     emit(node.elseStatement);
                 }
@@ -96676,20 +101779,20 @@ var ts;
             }
         }
         function emitWhileClause(node, startPos) {
-            var openParenPos = emitTokenWithComment(111 /* WhileKeyword */, startPos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(114 /* WhileKeyword */, startPos, writeKeyword, node);
             writeSpace();
             emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
             emitExpression(node.expression);
             emitTokenWithComment(21 /* CloseParenToken */, node.expression.end, writePunctuation, node);
         }
         function emitDoStatement(node) {
-            emitTokenWithComment(86 /* DoKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(89 /* DoKeyword */, node.pos, writeKeyword, node);
             emitEmbeddedStatement(node, node.statement);
-            if (ts.isBlock(node.statement)) {
+            if (ts.isBlock(node.statement) && !preserveSourceNewlines) {
                 writeSpace();
             }
             else {
-                writeLineOrSpace(node);
+                writeLineOrSpace(node, node.statement, node.expression);
             }
             emitWhileClause(node, node.statement.end);
             writeTrailingSemicolon();
@@ -96699,7 +101802,7 @@ var ts;
             emitEmbeddedStatement(node, node.statement);
         }
         function emitForStatement(node) {
-            var openParenPos = emitTokenWithComment(93 /* ForKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(96 /* ForKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             var pos = emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, /*contextNode*/ node);
             emitForBinding(node.initializer);
@@ -96711,25 +101814,25 @@ var ts;
             emitEmbeddedStatement(node, node.statement);
         }
         function emitForInStatement(node) {
-            var openParenPos = emitTokenWithComment(93 /* ForKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(96 /* ForKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
             emitForBinding(node.initializer);
             writeSpace();
-            emitTokenWithComment(97 /* InKeyword */, node.initializer.end, writeKeyword, node);
+            emitTokenWithComment(100 /* InKeyword */, node.initializer.end, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
             emitTokenWithComment(21 /* CloseParenToken */, node.expression.end, writePunctuation, node);
             emitEmbeddedStatement(node, node.statement);
         }
         function emitForOfStatement(node) {
-            var openParenPos = emitTokenWithComment(93 /* ForKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(96 /* ForKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitWithTrailingSpace(node.awaitModifier);
             emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
             emitForBinding(node.initializer);
             writeSpace();
-            emitTokenWithComment(152 /* OfKeyword */, node.initializer.end, writeKeyword, node);
+            emitTokenWithComment(156 /* OfKeyword */, node.initializer.end, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
             emitTokenWithComment(21 /* CloseParenToken */, node.expression.end, writePunctuation, node);
@@ -96737,7 +101840,7 @@ var ts;
         }
         function emitForBinding(node) {
             if (node !== undefined) {
-                if (node.kind === 243 /* VariableDeclarationList */) {
+                if (node.kind === 250 /* VariableDeclarationList */) {
                     emit(node);
                 }
                 else {
@@ -96746,12 +101849,12 @@ var ts;
             }
         }
         function emitContinueStatement(node) {
-            emitTokenWithComment(82 /* ContinueKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(85 /* ContinueKeyword */, node.pos, writeKeyword, node);
             emitWithLeadingSpace(node.label);
             writeTrailingSemicolon();
         }
         function emitBreakStatement(node) {
-            emitTokenWithComment(77 /* BreakKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(80 /* BreakKeyword */, node.pos, writeKeyword, node);
             emitWithLeadingSpace(node.label);
             writeTrailingSemicolon();
         }
@@ -96779,12 +101882,12 @@ var ts;
             return pos;
         }
         function emitReturnStatement(node) {
-            emitTokenWithComment(101 /* ReturnKeyword */, node.pos, writeKeyword, /*contextNode*/ node);
+            emitTokenWithComment(104 /* ReturnKeyword */, node.pos, writeKeyword, /*contextNode*/ node);
             emitExpressionWithLeadingSpace(node.expression);
             writeTrailingSemicolon();
         }
         function emitWithStatement(node) {
-            var openParenPos = emitTokenWithComment(112 /* WithKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(115 /* WithKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
             emitExpression(node.expression);
@@ -96792,7 +101895,7 @@ var ts;
             emitEmbeddedStatement(node, node.statement);
         }
         function emitSwitchStatement(node) {
-            var openParenPos = emitTokenWithComment(103 /* SwitchKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(106 /* SwitchKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
             emitExpression(node.expression);
@@ -96807,27 +101910,27 @@ var ts;
             emit(node.statement);
         }
         function emitThrowStatement(node) {
-            emitTokenWithComment(105 /* ThrowKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(108 /* ThrowKeyword */, node.pos, writeKeyword, node);
             emitExpressionWithLeadingSpace(node.expression);
             writeTrailingSemicolon();
         }
         function emitTryStatement(node) {
-            emitTokenWithComment(107 /* TryKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(110 /* TryKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emit(node.tryBlock);
             if (node.catchClause) {
-                writeLineOrSpace(node);
+                writeLineOrSpace(node, node.tryBlock, node.catchClause);
                 emit(node.catchClause);
             }
             if (node.finallyBlock) {
-                writeLineOrSpace(node);
-                emitTokenWithComment(92 /* FinallyKeyword */, (node.catchClause || node.tryBlock).end, writeKeyword, node);
+                writeLineOrSpace(node, node.catchClause || node.tryBlock, node.finallyBlock);
+                emitTokenWithComment(95 /* FinallyKeyword */, (node.catchClause || node.tryBlock).end, writeKeyword, node);
                 writeSpace();
                 emit(node.finallyBlock);
             }
         }
         function emitDebuggerStatement(node) {
-            writeToken(83 /* DebuggerKeyword */, node.pos, writeKeyword);
+            writeToken(86 /* DebuggerKeyword */, node.pos, writeKeyword);
             writeTrailingSemicolon();
         }
         //
@@ -97033,7 +102136,7 @@ var ts;
             var body = node.body;
             if (!body)
                 return writeTrailingSemicolon();
-            while (body.kind === 249 /* ModuleDeclaration */) {
+            while (body.kind === 256 /* ModuleDeclaration */) {
                 writePunctuation(".");
                 emit(body.name);
                 body = body.body;
@@ -97054,7 +102157,7 @@ var ts;
         }
         function emitImportEqualsDeclaration(node) {
             emitModifiers(node, node.modifiers);
-            emitTokenWithComment(96 /* ImportKeyword */, node.modifiers ? node.modifiers.end : node.pos, writeKeyword, node);
+            emitTokenWithComment(99 /* ImportKeyword */, node.modifiers ? node.modifiers.end : node.pos, writeKeyword, node);
             writeSpace();
             emit(node.name);
             writeSpace();
@@ -97064,7 +102167,7 @@ var ts;
             writeTrailingSemicolon();
         }
         function emitModuleReference(node) {
-            if (node.kind === 75 /* Identifier */) {
+            if (node.kind === 78 /* Identifier */) {
                 emitExpression(node);
             }
             else {
@@ -97073,12 +102176,12 @@ var ts;
         }
         function emitImportDeclaration(node) {
             emitModifiers(node, node.modifiers);
-            emitTokenWithComment(96 /* ImportKeyword */, node.modifiers ? node.modifiers.end : node.pos, writeKeyword, node);
+            emitTokenWithComment(99 /* ImportKeyword */, node.modifiers ? node.modifiers.end : node.pos, writeKeyword, node);
             writeSpace();
             if (node.importClause) {
                 emit(node.importClause);
                 writeSpace();
-                emitTokenWithComment(149 /* FromKeyword */, node.importClause.end, writeKeyword, node);
+                emitTokenWithComment(153 /* FromKeyword */, node.importClause.end, writeKeyword, node);
                 writeSpace();
             }
             emitExpression(node.moduleSpecifier);
@@ -97086,7 +102189,7 @@ var ts;
         }
         function emitImportClause(node) {
             if (node.isTypeOnly) {
-                emitTokenWithComment(145 /* TypeKeyword */, node.pos, writeKeyword, node);
+                emitTokenWithComment(149 /* TypeKeyword */, node.pos, writeKeyword, node);
                 writeSpace();
             }
             emit(node.name);
@@ -97099,7 +102202,7 @@ var ts;
         function emitNamespaceImport(node) {
             var asPos = emitTokenWithComment(41 /* AsteriskToken */, node.pos, writePunctuation, node);
             writeSpace();
-            emitTokenWithComment(123 /* AsKeyword */, asPos, writeKeyword, node);
+            emitTokenWithComment(126 /* AsKeyword */, asPos, writeKeyword, node);
             writeSpace();
             emit(node.name);
         }
@@ -97110,23 +102213,23 @@ var ts;
             emitImportOrExportSpecifier(node);
         }
         function emitExportAssignment(node) {
-            var nextPos = emitTokenWithComment(89 /* ExportKeyword */, node.pos, writeKeyword, node);
+            var nextPos = emitTokenWithComment(92 /* ExportKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             if (node.isExportEquals) {
                 emitTokenWithComment(62 /* EqualsToken */, nextPos, writeOperator, node);
             }
             else {
-                emitTokenWithComment(84 /* DefaultKeyword */, nextPos, writeKeyword, node);
+                emitTokenWithComment(87 /* DefaultKeyword */, nextPos, writeKeyword, node);
             }
             writeSpace();
             emitExpression(node.expression);
             writeTrailingSemicolon();
         }
         function emitExportDeclaration(node) {
-            var nextPos = emitTokenWithComment(89 /* ExportKeyword */, node.pos, writeKeyword, node);
+            var nextPos = emitTokenWithComment(92 /* ExportKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             if (node.isTypeOnly) {
-                nextPos = emitTokenWithComment(145 /* TypeKeyword */, nextPos, writeKeyword, node);
+                nextPos = emitTokenWithComment(149 /* TypeKeyword */, nextPos, writeKeyword, node);
                 writeSpace();
             }
             if (node.exportClause) {
@@ -97138,18 +102241,18 @@ var ts;
             if (node.moduleSpecifier) {
                 writeSpace();
                 var fromPos = node.exportClause ? node.exportClause.end : nextPos;
-                emitTokenWithComment(149 /* FromKeyword */, fromPos, writeKeyword, node);
+                emitTokenWithComment(153 /* FromKeyword */, fromPos, writeKeyword, node);
                 writeSpace();
                 emitExpression(node.moduleSpecifier);
             }
             writeTrailingSemicolon();
         }
         function emitNamespaceExportDeclaration(node) {
-            var nextPos = emitTokenWithComment(89 /* ExportKeyword */, node.pos, writeKeyword, node);
+            var nextPos = emitTokenWithComment(92 /* ExportKeyword */, node.pos, writeKeyword, node);
             writeSpace();
-            nextPos = emitTokenWithComment(123 /* AsKeyword */, nextPos, writeKeyword, node);
+            nextPos = emitTokenWithComment(126 /* AsKeyword */, nextPos, writeKeyword, node);
             writeSpace();
-            nextPos = emitTokenWithComment(136 /* NamespaceKeyword */, nextPos, writeKeyword, node);
+            nextPos = emitTokenWithComment(140 /* NamespaceKeyword */, nextPos, writeKeyword, node);
             writeSpace();
             emit(node.name);
             writeTrailingSemicolon();
@@ -97157,7 +102260,7 @@ var ts;
         function emitNamespaceExport(node) {
             var asPos = emitTokenWithComment(41 /* AsteriskToken */, node.pos, writePunctuation, node);
             writeSpace();
-            emitTokenWithComment(123 /* AsKeyword */, asPos, writeKeyword, node);
+            emitTokenWithComment(126 /* AsKeyword */, asPos, writeKeyword, node);
             writeSpace();
             emit(node.name);
         }
@@ -97176,7 +102279,7 @@ var ts;
             if (node.propertyName) {
                 emit(node.propertyName);
                 writeSpace();
-                emitTokenWithComment(123 /* AsKeyword */, node.propertyName.end, writeKeyword, node);
+                emitTokenWithComment(126 /* AsKeyword */, node.propertyName.end, writeKeyword, node);
                 writeSpace();
             }
             emit(node.name);
@@ -97257,7 +102360,7 @@ var ts;
             }
         }
         function emitJsxTagName(node) {
-            if (node.kind === 75 /* Identifier */) {
+            if (node.kind === 78 /* Identifier */) {
                 emitExpression(node);
             }
             else {
@@ -97268,13 +102371,13 @@ var ts;
         // Clauses
         //
         function emitCaseClause(node) {
-            emitTokenWithComment(78 /* CaseKeyword */, node.pos, writeKeyword, node);
+            emitTokenWithComment(81 /* CaseKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             emitExpression(node.expression);
             emitCaseOrDefaultClauseRest(node, node.statements, node.expression.end);
         }
         function emitDefaultClause(node) {
-            var pos = emitTokenWithComment(84 /* DefaultKeyword */, node.pos, writeKeyword, node);
+            var pos = emitTokenWithComment(87 /* DefaultKeyword */, node.pos, writeKeyword, node);
             emitCaseOrDefaultClauseRest(node, node.statements, pos);
         }
         function emitCaseOrDefaultClauseRest(parentNode, statements, colonPos) {
@@ -97302,7 +102405,7 @@ var ts;
             emitList(node, node.types, 528 /* HeritageClauseTypes */);
         }
         function emitCatchClause(node) {
-            var openParenPos = emitTokenWithComment(79 /* CatchKeyword */, node.pos, writeKeyword, node);
+            var openParenPos = emitTokenWithComment(82 /* CatchKeyword */, node.pos, writeKeyword, node);
             writeSpace();
             if (node.variableDeclaration) {
                 emitTokenWithComment(20 /* OpenParenToken */, openParenPos, writePunctuation, node);
@@ -97372,7 +102475,7 @@ var ts;
                 }
             }
             if (node.tags) {
-                if (node.tags.length === 1 && node.tags[0].kind === 320 /* JSDocTypeTag */ && !node.comment) {
+                if (node.tags.length === 1 && node.tags[0].kind === 329 /* JSDocTypeTag */ && !node.comment) {
                     writeSpace();
                     emit(node.tags[0]);
                 }
@@ -97388,6 +102491,17 @@ var ts;
             emitJSDocTypeExpression(tag.typeExpression);
             emitJSDocComment(tag.comment);
         }
+        function emitJSDocSeeTag(tag) {
+            emitJSDocTagName(tag.tagName);
+            emit(tag.name);
+            emitJSDocComment(tag.comment);
+        }
+        function emitJSDocNameReference(node) {
+            writeSpace();
+            writePunctuation("{");
+            emit(node.name);
+            writePunctuation("}");
+        }
         function emitJSDocHeritageTag(tag) {
             emitJSDocTagName(tag.tagName);
             writeSpace();
@@ -97406,7 +102520,7 @@ var ts;
         function emitJSDocTypedefTag(tag) {
             emitJSDocTagName(tag.tagName);
             if (tag.typeExpression) {
-                if (tag.typeExpression.kind === 294 /* JSDocTypeExpression */) {
+                if (tag.typeExpression.kind === 301 /* JSDocTypeExpression */) {
                     emitJSDocTypeExpression(tag.typeExpression);
                 }
                 else {
@@ -97425,7 +102539,7 @@ var ts;
                 emit(tag.fullName);
             }
             emitJSDocComment(tag.comment);
-            if (tag.typeExpression && tag.typeExpression.kind === 304 /* JSDocTypeLiteral */) {
+            if (tag.typeExpression && tag.typeExpression.kind === 312 /* JSDocTypeLiteral */) {
                 emitJSDocTypeLiteral(tag.typeExpression);
             }
         }
@@ -97443,14 +102557,14 @@ var ts;
             emitJSDocComment(tag.comment);
         }
         function emitJSDocTypeLiteral(lit) {
-            emitList(lit, ts.createNodeArray(lit.jsDocPropertyTags), 33 /* JSDocComment */);
+            emitList(lit, ts.factory.createNodeArray(lit.jsDocPropertyTags), 33 /* JSDocComment */);
         }
         function emitJSDocSignature(sig) {
             if (sig.typeParameters) {
-                emitList(sig, ts.createNodeArray(sig.typeParameters), 33 /* JSDocComment */);
+                emitList(sig, ts.factory.createNodeArray(sig.typeParameters), 33 /* JSDocComment */);
             }
             if (sig.parameters) {
-                emitList(sig, ts.createNodeArray(sig.parameters), 33 /* JSDocComment */);
+                emitList(sig, ts.factory.createNodeArray(sig.parameters), 33 /* JSDocComment */);
             }
             if (sig.type) {
                 writeLine();
@@ -97559,8 +102673,8 @@ var ts;
                     bundleFileInfo.sections.push({ pos: pos, end: writer.getTextPos(), kind: "reference" /* Reference */, data: directive.fileName });
                 writeLine();
             }
-            for (var _d = 0, types_22 = types; _d < types_22.length; _d++) {
-                var directive = types_22[_d];
+            for (var _d = 0, types_23 = types; _d < types_23.length; _d++) {
+                var directive = types_23[_d];
                 var pos = writer.getTextPos();
                 writeComment("/// <reference types=\"" + directive.fileName + "\" />");
                 if (bundleFileInfo)
@@ -97614,7 +102728,7 @@ var ts;
                         if (recordBundleFileSection && bundleFileInfo)
                             bundleFileInfo.sections.push({ pos: pos, end: writer.getTextPos(), kind: "prologue" /* Prologue */, data: statement.expression.text });
                         if (seenPrologueDirectives) {
-                            seenPrologueDirectives.set(statement.expression.text, true);
+                            seenPrologueDirectives.add(statement.expression.text);
                         }
                     }
                 }
@@ -97635,7 +102749,7 @@ var ts;
                     if (bundleFileInfo)
                         bundleFileInfo.sections.push({ pos: pos, end: writer.getTextPos(), kind: "prologue" /* Prologue */, data: prologue.data });
                     if (seenPrologueDirectives) {
-                        seenPrologueDirectives.set(prologue.data, true);
+                        seenPrologueDirectives.add(prologue.data);
                     }
                 }
             }
@@ -97645,7 +102759,7 @@ var ts;
                 emitPrologueDirectives(sourceFileOrBundle.statements, sourceFileOrBundle);
             }
             else {
-                var seenPrologueDirectives = ts.createMap();
+                var seenPrologueDirectives = new ts.Set();
                 for (var _a = 0, _b = sourceFileOrBundle.prepends; _a < _b.length; _a++) {
                     var prepend = _b[_a];
                     emitUnparsedPrologues(prepend.prologues, seenPrologueDirectives);
@@ -97658,7 +102772,7 @@ var ts;
             }
         }
         function getPrologueDirectivesFromBundledSourceFiles(bundle) {
-            var seenPrologueDirectives = ts.createMap();
+            var seenPrologueDirectives = new ts.Set();
             var prologues;
             for (var index = 0; index < bundle.sourceFiles.length; index++) {
                 var sourceFile = bundle.sourceFiles[index];
@@ -97670,7 +102784,7 @@ var ts;
                         break;
                     if (seenPrologueDirectives.has(statement.expression.text))
                         continue;
-                    seenPrologueDirectives.set(statement.expression.text, true);
+                    seenPrologueDirectives.add(statement.expression.text);
                     (directives || (directives = [])).push({
                         pos: statement.pos,
                         end: statement.end,
@@ -97960,6 +103074,7 @@ var ts;
                     else {
                         shouldEmitInterveningComments = mayEmitInterveningComments;
                     }
+                    nextListElementPos = child.pos;
                     emit(child);
                     if (shouldDecreaseIndentAfterEmit) {
                         decreaseIndent();
@@ -98074,10 +103189,19 @@ var ts;
             writer(tokenString);
             return pos < 0 ? pos : pos + tokenString.length;
         }
-        function writeLineOrSpace(node) {
-            if (ts.getEmitFlags(node) & 1 /* SingleLine */) {
+        function writeLineOrSpace(parentNode, prevChildNode, nextChildNode) {
+            if (ts.getEmitFlags(parentNode) & 1 /* SingleLine */) {
                 writeSpace();
             }
+            else if (preserveSourceNewlines) {
+                var lines = getLinesBetweenNodes(parentNode, prevChildNode, nextChildNode);
+                if (lines) {
+                    writeLine(lines);
+                }
+                else {
+                    writeSpace();
+                }
+            }
             else {
                 writeLine();
             }
@@ -98124,11 +103248,31 @@ var ts;
                 if (firstChild_1 === undefined) {
                     return ts.rangeIsOnSingleLine(parentNode, currentSourceFile) ? 0 : 1;
                 }
+                if (firstChild_1.pos === nextListElementPos) {
+                    // If this child starts at the beginning of a list item in a parent list, its leading
+                    // line terminators have already been written as the separating line terminators of the
+                    // parent list. Example:
+                    //
+                    // class Foo {
+                    //   constructor() {}
+                    //   public foo() {}
+                    // }
+                    //
+                    // The outer list is the list of class members, with one line terminator between the
+                    // constructor and the method. The constructor is written, the separating line terminator
+                    // is written, and then we start emitting the method. Its modifiers ([public]) constitute an inner
+                    // list, so we look for its leading line terminators. If we didn't know that we had already
+                    // written a newline as part of the parent list, it would appear that we need to write a
+                    // leading newline to start the modifiers.
+                    return 0;
+                }
                 if (firstChild_1.kind === 11 /* JsxText */) {
                     // JsxText will be written with its leading whitespace, so don't add more manually.
                     return 0;
                 }
-                if (!ts.positionIsSynthesized(parentNode.pos) && !ts.nodeIsSynthesized(firstChild_1) && (!firstChild_1.parent || firstChild_1.parent === parentNode)) {
+                if (!ts.positionIsSynthesized(parentNode.pos) &&
+                    !ts.nodeIsSynthesized(firstChild_1) &&
+                    (!firstChild_1.parent || ts.getOriginalNode(firstChild_1.parent) === ts.getOriginalNode(parentNode))) {
                     if (preserveSourceNewlines) {
                         return getEffectiveLines(function (includeComments) { return ts.getLinesBetweenPositionAndPrecedingNonWhitespaceCharacter(firstChild_1.pos, parentNode.pos, currentSourceFile, includeComments); });
                     }
@@ -98169,17 +103313,18 @@ var ts;
                 if (format & 65536 /* PreferNewLine */) {
                     return 1;
                 }
-                var lastChild_1 = ts.lastOrUndefined(children);
-                if (lastChild_1 === undefined) {
+                var lastChild = ts.lastOrUndefined(children);
+                if (lastChild === undefined) {
                     return ts.rangeIsOnSingleLine(parentNode, currentSourceFile) ? 0 : 1;
                 }
-                if (!ts.positionIsSynthesized(parentNode.pos) && !ts.nodeIsSynthesized(lastChild_1) && (!lastChild_1.parent || lastChild_1.parent === parentNode)) {
+                if (!ts.positionIsSynthesized(parentNode.pos) && !ts.nodeIsSynthesized(lastChild) && (!lastChild.parent || lastChild.parent === parentNode)) {
                     if (preserveSourceNewlines) {
-                        return getEffectiveLines(function (includeComments) { return ts.getLinesBetweenPositionAndNextNonWhitespaceCharacter(lastChild_1.end, parentNode.end, currentSourceFile, includeComments); });
+                        var end_2 = ts.isNodeArray(children) && !ts.positionIsSynthesized(children.end) ? children.end : lastChild.end;
+                        return getEffectiveLines(function (includeComments) { return ts.getLinesBetweenPositionAndNextNonWhitespaceCharacter(end_2, parentNode.end, currentSourceFile, includeComments); });
                     }
-                    return ts.rangeEndPositionsAreOnSameLine(parentNode, lastChild_1, currentSourceFile) ? 0 : 1;
+                    return ts.rangeEndPositionsAreOnSameLine(parentNode, lastChild, currentSourceFile) ? 0 : 1;
                 }
-                if (synthesizedNodeStartsOnNewLine(lastChild_1, format)) {
+                if (synthesizedNodeStartsOnNewLine(lastChild, format)) {
                     return 1;
                 }
             }
@@ -98258,7 +103403,7 @@ var ts;
                 && ts.rangeEndIsOnSameLineAsRangeStart(block, block, currentSourceFile);
         }
         function skipSynthesizedParentheses(node) {
-            while (node.kind === 200 /* ParenthesizedExpression */ && ts.nodeIsSynthesized(node)) {
+            while (node.kind === 207 /* ParenthesizedExpression */ && ts.nodeIsSynthesized(node)) {
                 node = node.expression;
             }
             return node;
@@ -98281,16 +103426,20 @@ var ts;
         function getLiteralTextOfNode(node, neverAsciiEscape, jsxAttributeEscape) {
             if (node.kind === 10 /* StringLiteral */ && node.textSourceNode) {
                 var textSourceNode = node.textSourceNode;
-                if (ts.isIdentifier(textSourceNode)) {
-                    return jsxAttributeEscape ? "\"" + ts.escapeJsxAttributeString(getTextOfNode(textSourceNode)) + "\"" :
-                        neverAsciiEscape || (ts.getEmitFlags(node) & 16777216 /* NoAsciiEscaping */) ? "\"" + ts.escapeString(getTextOfNode(textSourceNode)) + "\"" :
-                            "\"" + ts.escapeNonAsciiString(getTextOfNode(textSourceNode)) + "\"";
+                if (ts.isIdentifier(textSourceNode) || ts.isNumericLiteral(textSourceNode)) {
+                    var text = ts.isNumericLiteral(textSourceNode) ? textSourceNode.text : getTextOfNode(textSourceNode);
+                    return jsxAttributeEscape ? "\"" + ts.escapeJsxAttributeString(text) + "\"" :
+                        neverAsciiEscape || (ts.getEmitFlags(node) & 16777216 /* NoAsciiEscaping */) ? "\"" + ts.escapeString(text) + "\"" :
+                            "\"" + ts.escapeNonAsciiString(text) + "\"";
                 }
                 else {
                     return getLiteralTextOfNode(textSourceNode, neverAsciiEscape, jsxAttributeEscape);
                 }
             }
-            return ts.getLiteralText(node, currentSourceFile, neverAsciiEscape, jsxAttributeEscape);
+            var flags = (neverAsciiEscape ? 1 /* NeverAsciiEscape */ : 0)
+                | (jsxAttributeEscape ? 2 /* JsxAttributeEscape */ : 0)
+                | (printerOptions.terminateUnterminatedLiterals ? 4 /* TerminateUnterminatedLiterals */ : 0);
+            return ts.getLiteralText(node, currentSourceFile, flags);
         }
         /**
          * Push a new name generation scope.
@@ -98315,92 +103464,92 @@ var ts;
         }
         function reserveNameInNestedScopes(name) {
             if (!reservedNames || reservedNames === ts.lastOrUndefined(reservedNamesStack)) {
-                reservedNames = ts.createMap();
+                reservedNames = new ts.Set();
             }
-            reservedNames.set(name, true);
+            reservedNames.add(name);
         }
         function generateNames(node) {
             if (!node)
                 return;
             switch (node.kind) {
-                case 223 /* Block */:
+                case 230 /* Block */:
                     ts.forEach(node.statements, generateNames);
                     break;
-                case 238 /* LabeledStatement */:
-                case 236 /* WithStatement */:
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
+                case 245 /* LabeledStatement */:
+                case 243 /* WithStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
                     generateNames(node.statement);
                     break;
-                case 227 /* IfStatement */:
+                case 234 /* IfStatement */:
                     generateNames(node.thenStatement);
                     generateNames(node.elseStatement);
                     break;
-                case 230 /* ForStatement */:
-                case 232 /* ForOfStatement */:
-                case 231 /* ForInStatement */:
+                case 237 /* ForStatement */:
+                case 239 /* ForOfStatement */:
+                case 238 /* ForInStatement */:
                     generateNames(node.initializer);
                     generateNames(node.statement);
                     break;
-                case 237 /* SwitchStatement */:
+                case 244 /* SwitchStatement */:
                     generateNames(node.caseBlock);
                     break;
-                case 251 /* CaseBlock */:
+                case 258 /* CaseBlock */:
                     ts.forEach(node.clauses, generateNames);
                     break;
-                case 277 /* CaseClause */:
-                case 278 /* DefaultClause */:
+                case 284 /* CaseClause */:
+                case 285 /* DefaultClause */:
                     ts.forEach(node.statements, generateNames);
                     break;
-                case 240 /* TryStatement */:
+                case 247 /* TryStatement */:
                     generateNames(node.tryBlock);
                     generateNames(node.catchClause);
                     generateNames(node.finallyBlock);
                     break;
-                case 280 /* CatchClause */:
+                case 287 /* CatchClause */:
                     generateNames(node.variableDeclaration);
                     generateNames(node.block);
                     break;
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     generateNames(node.declarationList);
                     break;
-                case 243 /* VariableDeclarationList */:
+                case 250 /* VariableDeclarationList */:
                     ts.forEach(node.declarations, generateNames);
                     break;
-                case 242 /* VariableDeclaration */:
-                case 156 /* Parameter */:
-                case 191 /* BindingElement */:
-                case 245 /* ClassDeclaration */:
+                case 249 /* VariableDeclaration */:
+                case 160 /* Parameter */:
+                case 198 /* BindingElement */:
+                case 252 /* ClassDeclaration */:
                     generateNameIfNeeded(node.name);
                     break;
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     generateNameIfNeeded(node.name);
                     if (ts.getEmitFlags(node) & 524288 /* ReuseTempVariableScope */) {
                         ts.forEach(node.parameters, generateNames);
                         generateNames(node.body);
                     }
                     break;
-                case 189 /* ObjectBindingPattern */:
-                case 190 /* ArrayBindingPattern */:
+                case 196 /* ObjectBindingPattern */:
+                case 197 /* ArrayBindingPattern */:
                     ts.forEach(node.elements, generateNames);
                     break;
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     generateNames(node.importClause);
                     break;
-                case 255 /* ImportClause */:
+                case 262 /* ImportClause */:
                     generateNameIfNeeded(node.name);
                     generateNames(node.namedBindings);
                     break;
-                case 256 /* NamespaceImport */:
+                case 263 /* NamespaceImport */:
                     generateNameIfNeeded(node.name);
                     break;
-                case 262 /* NamespaceExport */:
+                case 269 /* NamespaceExport */:
                     generateNameIfNeeded(node.name);
                     break;
-                case 257 /* NamedImports */:
+                case 264 /* NamedImports */:
                     ts.forEach(node.elements, generateNames);
                     break;
-                case 258 /* ImportSpecifier */:
+                case 265 /* ImportSpecifier */:
                     generateNameIfNeeded(node.propertyName || node.name);
                     break;
             }
@@ -98409,12 +103558,12 @@ var ts;
             if (!node)
                 return;
             switch (node.kind) {
-                case 281 /* PropertyAssignment */:
-                case 282 /* ShorthandPropertyAssignment */:
-                case 159 /* PropertyDeclaration */:
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 288 /* PropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
+                case 163 /* PropertyDeclaration */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     generateNameIfNeeded(node.name);
                     break;
             }
@@ -98527,7 +103676,7 @@ var ts;
                         reserveNameInNestedScopes(baseName);
                     }
                     else {
-                        generatedNames.set(baseName, true);
+                        generatedNames.add(baseName);
                     }
                     return baseName;
                 }
@@ -98544,7 +103693,7 @@ var ts;
                         reserveNameInNestedScopes(generatedName);
                     }
                     else {
-                        generatedNames.set(generatedName, true);
+                        generatedNames.add(generatedName);
                     }
                     return generatedName;
                 }
@@ -98594,25 +103743,25 @@ var ts;
          */
         function generateNameForNode(node, flags) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return makeUniqueName(getTextOfNode(node), isUniqueName, !!(flags & 16 /* Optimistic */), !!(flags & 8 /* ReservedInNestedScopes */));
-                case 249 /* ModuleDeclaration */:
-                case 248 /* EnumDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 255 /* EnumDeclaration */:
                     return generateNameForModuleOrEnum(node);
-                case 254 /* ImportDeclaration */:
-                case 260 /* ExportDeclaration */:
+                case 261 /* ImportDeclaration */:
+                case 267 /* ExportDeclaration */:
                     return generateNameForImportOrExportDeclaration(node);
-                case 244 /* FunctionDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 259 /* ExportAssignment */:
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 266 /* ExportAssignment */:
                     return generateNameForExportDefault();
-                case 214 /* ClassExpression */:
+                case 221 /* ClassExpression */:
                     return generateNameForClassExpression();
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return generateNameForMethodOrAccessor(node);
-                case 154 /* ComputedPropertyName */:
+                case 158 /* ComputedPropertyName */:
                     return makeTempVariableName(0 /* Auto */, /*reserveInNestedScopes*/ true);
                 default:
                     return makeTempVariableName(0 /* Auto */);
@@ -98660,7 +103809,7 @@ var ts;
             hasWrittenComment = false;
             var emitFlags = ts.getEmitFlags(node);
             var _a = ts.getCommentRange(node), pos = _a.pos, end = _a.end;
-            var isEmittedNode = node.kind !== 325 /* NotEmittedStatement */;
+            var isEmittedNode = node.kind !== 335 /* NotEmittedStatement */;
             // We have to explicitly check that the node is JsxText because if the compilerOptions.jsx is "preserve" we will not do any transformation.
             // It is expensive to walk entire tree just to set one kind of node to have no comments.
             var skipLeadingComments = pos < 0 || (emitFlags & 512 /* NoLeadingComments */) !== 0 || node.kind === 11 /* JsxText */;
@@ -98684,7 +103833,7 @@ var ts;
                     containerEnd = end;
                     // To avoid invalid comment emit in a down-level binding pattern, we
                     // keep track of the last declaration list container's end
-                    if (node.kind === 243 /* VariableDeclarationList */) {
+                    if (node.kind === 250 /* VariableDeclarationList */) {
                         declarationListContainerEnd = end;
                     }
                 }
@@ -98717,7 +103866,7 @@ var ts;
             ts.Debug.assert(lastNode === node || lastSubstitution === node);
         }
         function emitLeadingSynthesizedComment(comment) {
-            if (comment.kind === 2 /* SingleLineCommentTrivia */) {
+            if (comment.hasLeadingNewline || comment.kind === 2 /* SingleLineCommentTrivia */) {
                 writer.writeLine();
             }
             writeSynthesizedComment(comment);
@@ -98777,7 +103926,12 @@ var ts;
         function emitLeadingComments(pos, isEmittedNode) {
             hasWrittenComment = false;
             if (isEmittedNode) {
-                forEachLeadingCommentToEmit(pos, emitLeadingComment);
+                if (pos === 0 && (currentSourceFile === null || currentSourceFile === void 0 ? void 0 : currentSourceFile.isDeclarationFile)) {
+                    forEachLeadingCommentToEmit(pos, emitNonTripleSlashLeadingComment);
+                }
+                else {
+                    forEachLeadingCommentToEmit(pos, emitLeadingComment);
+                }
             }
             else if (pos === 0) {
                 // If the node will not be emitted in JS, remove all the comments(normal, pinned and ///) associated with the node,
@@ -98796,6 +103950,11 @@ var ts;
                 emitLeadingComment(commentPos, commentEnd, kind, hasTrailingNewLine, rangePos);
             }
         }
+        function emitNonTripleSlashLeadingComment(commentPos, commentEnd, kind, hasTrailingNewLine, rangePos) {
+            if (!isTripleSlashComment(commentPos, commentEnd)) {
+                emitLeadingComment(commentPos, commentEnd, kind, hasTrailingNewLine, rangePos);
+            }
+        }
         function shouldWriteComment(text, pos) {
             if (printerOptions.onlyPrintJsDocStyle) {
                 return (ts.isJSDocLikeText(text, pos) || ts.isPinnedComment(text, pos));
@@ -98943,7 +104102,7 @@ var ts;
             else {
                 var _a = ts.getSourceMapRange(node), pos = _a.pos, end = _a.end, _b = _a.source, source = _b === void 0 ? sourceMapSource : _b;
                 var emitFlags = ts.getEmitFlags(node);
-                if (node.kind !== 325 /* NotEmittedStatement */
+                if (node.kind !== 335 /* NotEmittedStatement */
                     && (emitFlags & 16 /* NoLeadingSourceMap */) === 0
                     && pos >= 0) {
                     emitSourcePos(source, skipSourceTrivia(source, pos));
@@ -98956,7 +104115,7 @@ var ts;
                 else {
                     pipelinePhase(hint, node);
                 }
-                if (node.kind !== 325 /* NotEmittedStatement */
+                if (node.kind !== 335 /* NotEmittedStatement */
                     && (emitFlags & 32 /* NoTrailingSourceMap */) === 0
                     && end >= 0) {
                     emitSourcePos(source, end);
@@ -98989,9 +104148,10 @@ var ts;
         function emitSourcePos(source, pos) {
             if (source !== sourceMapSource) {
                 var savedSourceMapSource = sourceMapSource;
+                var savedSourceMapSourceIndex = sourceMapSourceIndex;
                 setSourceMapSource(source);
                 emitPos(pos);
-                setSourceMapSource(savedSourceMapSource);
+                resetSourceMapSource(savedSourceMapSource, savedSourceMapSourceIndex);
             }
             else {
                 emitPos(pos);
@@ -99030,6 +104190,12 @@ var ts;
                 return;
             }
             sourceMapSource = source;
+            if (source === mostRecentlyAddedSourceMapSource) {
+                // Fast path for when the new source map is the most recently added, in which case
+                // we use its captured index without going through the source map generator.
+                sourceMapSourceIndex = mostRecentlyAddedSourceMapSourceIndex;
+                return;
+            }
             if (isJsonSourceMapSource(source)) {
                 return;
             }
@@ -99037,6 +104203,12 @@ var ts;
             if (printerOptions.inlineSources) {
                 sourceMapGenerator.setSourceContent(sourceMapSourceIndex, source.text);
             }
+            mostRecentlyAddedSourceMapSource = source;
+            mostRecentlyAddedSourceMapSourceIndex = sourceMapSourceIndex;
+        }
+        function resetSourceMapSource(source, sourceIndex) {
+            sourceMapSource = source;
+            sourceMapSourceIndex = sourceIndex;
         }
         function isJsonSourceMapSource(sourceFile) {
             return ts.fileExtensionIs(sourceFile.fileName, ".json" /* Json */);
@@ -99072,7 +104244,7 @@ var ts;
         if (!host.getDirectories || !host.readDirectory) {
             return undefined;
         }
-        var cachedReadDirectoryResult = ts.createMap();
+        var cachedReadDirectoryResult = new ts.Map();
         var getCanonicalFileName = ts.createGetCanonicalFileName(useCaseSensitiveFileNames);
         return {
             useCaseSensitiveFileNames: useCaseSensitiveFileNames,
@@ -99261,7 +104433,8 @@ var ts;
      */
     function updateMissingFilePathsWatch(program, missingFileWatches, createMissingFileWatch) {
         var missingFilePaths = program.getMissingFilePaths();
-        var newMissingFilePathMap = ts.arrayToSet(missingFilePaths);
+        // TODO(rbuckton): Should be a `Set` but that requires changing the below code that uses `mutateMap`
+        var newMissingFilePathMap = ts.arrayToMap(missingFilePaths, ts.identity, ts.returnTrue);
         // Update the missing file paths watcher
         ts.mutateMap(missingFileWatches, newMissingFilePathMap, {
             // Watch the missing files
@@ -99304,6 +104477,61 @@ var ts;
         }
     }
     ts.updateWatchingWildcardDirectories = updateWatchingWildcardDirectories;
+    /* @internal */
+    function isIgnoredFileFromWildCardWatching(_a) {
+        var watchedDirPath = _a.watchedDirPath, fileOrDirectory = _a.fileOrDirectory, fileOrDirectoryPath = _a.fileOrDirectoryPath, configFileName = _a.configFileName, options = _a.options, configFileSpecs = _a.configFileSpecs, program = _a.program, extraFileExtensions = _a.extraFileExtensions, currentDirectory = _a.currentDirectory, useCaseSensitiveFileNames = _a.useCaseSensitiveFileNames, writeLog = _a.writeLog;
+        var newPath = ts.removeIgnoredPath(fileOrDirectoryPath);
+        if (!newPath) {
+            writeLog("Project: " + configFileName + " Detected ignored path: " + fileOrDirectory);
+            return true;
+        }
+        fileOrDirectoryPath = newPath;
+        if (fileOrDirectoryPath === watchedDirPath)
+            return false;
+        // If the the added or created file or directory is not supported file name, ignore the file
+        // But when watched directory is added/removed, we need to reload the file list
+        if (ts.hasExtension(fileOrDirectoryPath) && !ts.isSupportedSourceFileName(fileOrDirectory, options, extraFileExtensions)) {
+            writeLog("Project: " + configFileName + " Detected file add/remove of non supported extension: " + fileOrDirectory);
+            return true;
+        }
+        if (ts.isExcludedFile(fileOrDirectory, configFileSpecs, ts.getNormalizedAbsolutePath(ts.getDirectoryPath(configFileName), currentDirectory), useCaseSensitiveFileNames, currentDirectory)) {
+            writeLog("Project: " + configFileName + " Detected excluded file: " + fileOrDirectory);
+            return true;
+        }
+        if (!program)
+            return false;
+        // We want to ignore emit file check if file is not going to be emitted next to source file
+        // In that case we follow config file inclusion rules
+        if (options.outFile || options.outDir)
+            return false;
+        // File if emitted next to input needs to be ignored
+        if (ts.fileExtensionIs(fileOrDirectoryPath, ".d.ts" /* Dts */)) {
+            // If its declaration directory: its not ignored if not excluded by config
+            if (options.declarationDir)
+                return false;
+        }
+        else if (!ts.fileExtensionIsOneOf(fileOrDirectoryPath, ts.supportedJSExtensions)) {
+            return false;
+        }
+        // just check if sourceFile with the name exists
+        var filePathWithoutExtension = ts.removeFileExtension(fileOrDirectoryPath);
+        var realProgram = isBuilderProgram(program) ? program.getProgramOrUndefined() : program;
+        if (hasSourceFile((filePathWithoutExtension + ".ts" /* Ts */)) ||
+            hasSourceFile((filePathWithoutExtension + ".tsx" /* Tsx */))) {
+            writeLog("Project: " + configFileName + " Detected output file: " + fileOrDirectory);
+            return true;
+        }
+        return false;
+        function hasSourceFile(file) {
+            return realProgram ?
+                !!realProgram.getSourceFileByPath(file) :
+                program.getState().fileInfos.has(file);
+        }
+    }
+    ts.isIgnoredFileFromWildCardWatching = isIgnoredFileFromWildCardWatching;
+    function isBuilderProgram(program) {
+        return !!program.getState;
+    }
     function isEmittedFileOfProgram(program, file) {
         if (!program) {
             return false;
@@ -99477,8 +104705,9 @@ var ts;
     // TODO(shkamat): update this after reworking ts build API
     function createCompilerHostWorker(options, setParentNodes, system) {
         if (system === void 0) { system = ts.sys; }
-        var existingDirectories = ts.createMap();
+        var existingDirectories = new ts.Map();
         var getCanonicalFileName = ts.createGetCanonicalFileName(system.useCaseSensitiveFileNames);
+        var computeHash = ts.maybeBind(system, system.createHash) || ts.generateDjb2Hash;
         function getSourceFile(fileName, languageVersion, onError) {
             var text;
             try {
@@ -99523,14 +104752,14 @@ var ts;
         }
         var outputFingerprints;
         function writeFileWorker(fileName, data, writeByteOrderMark) {
-            if (!ts.isWatchSet(options) || !system.createHash || !system.getModifiedTime) {
+            if (!ts.isWatchSet(options) || !system.getModifiedTime) {
                 system.writeFile(fileName, data, writeByteOrderMark);
                 return;
             }
             if (!outputFingerprints) {
-                outputFingerprints = ts.createMap();
+                outputFingerprints = new ts.Map();
             }
-            var hash = system.createHash(data);
+            var hash = computeHash(data);
             var mtimeBefore = system.getModifiedTime(fileName);
             if (mtimeBefore) {
                 var fingerprint = outputFingerprints.get(fileName);
@@ -99585,10 +104814,10 @@ var ts;
         var originalDirectoryExists = host.directoryExists;
         var originalCreateDirectory = host.createDirectory;
         var originalWriteFile = host.writeFile;
-        var readFileCache = ts.createMap();
-        var fileExistsCache = ts.createMap();
-        var directoryExistsCache = ts.createMap();
-        var sourceFileCache = ts.createMap();
+        var readFileCache = new ts.Map();
+        var fileExistsCache = new ts.Map();
+        var directoryExistsCache = new ts.Map();
+        var sourceFileCache = new ts.Map();
         var readFileWithCache = function (fileName) {
             var key = toPath(fileName);
             var value = readFileCache.get(key);
@@ -99694,8 +104923,8 @@ var ts;
     ts.getPreEmitDiagnostics = getPreEmitDiagnostics;
     function formatDiagnostics(diagnostics, host) {
         var output = "";
-        for (var _i = 0, diagnostics_2 = diagnostics; _i < diagnostics_2.length; _i++) {
-            var diagnostic = diagnostics_2[_i];
+        for (var _i = 0, diagnostics_3 = diagnostics; _i < diagnostics_3.length; _i++) {
+            var diagnostic = diagnostics_3[_i];
             output += formatDiagnostic(diagnostic, host);
         }
         return output;
@@ -99803,8 +105032,8 @@ var ts;
     ts.formatLocation = formatLocation;
     function formatDiagnosticsWithColorAndContext(diagnostics, host) {
         var output = "";
-        for (var _i = 0, diagnostics_3 = diagnostics; _i < diagnostics_3.length; _i++) {
-            var diagnostic = diagnostics_3[_i];
+        for (var _i = 0, diagnostics_4 = diagnostics; _i < diagnostics_4.length; _i++) {
+            var diagnostic = diagnostics_4[_i];
             if (diagnostic.file) {
                 var file = diagnostic.file, start = diagnostic.start;
                 output += formatLocation(file, start, host); // TODO: GH#18217
@@ -99867,7 +105096,7 @@ var ts;
             return [];
         }
         var resolutions = [];
-        var cache = ts.createMap();
+        var cache = new ts.Map();
         for (var _i = 0, names_2 = names; _i < names_2.length; _i++) {
             var name = names_2[_i];
             var result = void 0;
@@ -99883,6 +105112,35 @@ var ts;
     }
     ts.loadWithLocalCache = loadWithLocalCache;
     /* @internal */
+    function forEachResolvedProjectReference(resolvedProjectReferences, cb) {
+        return forEachProjectReference(/*projectReferences*/ undefined, resolvedProjectReferences, function (resolvedRef, parent) { return resolvedRef && cb(resolvedRef, parent); });
+    }
+    ts.forEachResolvedProjectReference = forEachResolvedProjectReference;
+    function forEachProjectReference(projectReferences, resolvedProjectReferences, cbResolvedRef, cbRef) {
+        var seenResolvedRefs;
+        return worker(projectReferences, resolvedProjectReferences, /*parent*/ undefined);
+        function worker(projectReferences, resolvedProjectReferences, parent) {
+            // Visit project references first
+            if (cbRef) {
+                var result = cbRef(projectReferences, parent);
+                if (result) {
+                    return result;
+                }
+            }
+            return ts.forEach(resolvedProjectReferences, function (resolvedRef, index) {
+                if (resolvedRef && (seenResolvedRefs === null || seenResolvedRefs === void 0 ? void 0 : seenResolvedRefs.has(resolvedRef.sourceFile.path))) {
+                    // ignore recursives
+                    return undefined;
+                }
+                var result = cbResolvedRef(resolvedRef, parent, index);
+                if (result || !resolvedRef)
+                    return result;
+                (seenResolvedRefs || (seenResolvedRefs = new ts.Set())).add(resolvedRef.sourceFile.path);
+                return worker(resolvedRef.commandLine.projectReferences, resolvedRef.references, resolvedRef);
+            });
+        }
+    }
+    /* @internal */
     ts.inferredTypesContainingFile = "__inferred type names__.ts";
     /**
      * Determines if program structure is upto date or needs to be recreated
@@ -99890,7 +105148,7 @@ var ts;
     /* @internal */
     function isProgramUptoDate(program, rootFileNames, newOptions, getSourceVersion, fileExists, hasInvalidatedResolution, hasChangedAutomaticTypeDirectiveNames, projectReferences) {
         // If we haven't created a program yet or have changed automatic type directives, then it is not up-to-date
-        if (!program || hasChangedAutomaticTypeDirectiveNames) {
+        if (!program || (hasChangedAutomaticTypeDirectiveNames === null || hasChangedAutomaticTypeDirectiveNames === void 0 ? void 0 : hasChangedAutomaticTypeDirectiveNames())) {
             return false;
         }
         // If root file names don't match
@@ -99985,7 +105243,7 @@ var ts;
         };
     }
     function createProgram(rootNamesOrOptions, _options, _host, _oldProgram, _configFileParsingDiagnostics) {
-        var _a;
+        var _a, _b;
         var createProgramOptions = ts.isArray(rootNamesOrOptions) ? createCreateProgramOptions(rootNamesOrOptions, _options, _host, _oldProgram, _configFileParsingDiagnostics) : rootNamesOrOptions; // TODO: GH#18217
         var rootNames = createProgramOptions.rootNames, options = createProgramOptions.options, configFileParsingDiagnostics = createProgramOptions.configFileParsingDiagnostics, projectReferences = createProgramOptions.projectReferences;
         var oldProgram = createProgramOptions.oldProgram;
@@ -99997,12 +105255,12 @@ var ts;
         var diagnosticsProducingTypeChecker;
         var noDiagnosticsTypeChecker;
         var classifiableNames;
-        var ambientModuleNameToUnmodifiedFileName = ts.createMap();
+        var ambientModuleNameToUnmodifiedFileName = new ts.Map();
         // Todo:: Use this to report why file was included in --extendedDiagnostics
         var refFileMap;
         var cachedBindAndCheckDiagnosticsForFile = {};
         var cachedDeclarationDiagnosticsForFile = {};
-        var resolvedTypeReferenceDirectives = ts.createMap();
+        var resolvedTypeReferenceDirectives = new ts.Map();
         var fileProcessingDiagnostics = ts.createDiagnosticCollection();
         // The below settings are to track if a .js file should be add to the program if loaded via searching under node_modules.
         // This works as imported modules are discovered recursively in a depth first manner, specifically:
@@ -100015,9 +105273,11 @@ var ts;
         var currentNodeModulesDepth = 0;
         // If a module has some of its imports skipped due to being at the depth limit under node_modules, then track
         // this, as it may be imported at a shallower depth later, and then it will need its skipped imports processed.
-        var modulesWithElidedImports = ts.createMap();
+        var modulesWithElidedImports = new ts.Map();
         // Track source files that are source files found by searching under node_modules, as these shouldn't be compiled.
-        var sourceFilesFoundSearchingNodeModules = ts.createMap();
+        var sourceFilesFoundSearchingNodeModules = new ts.Map();
+        var tracingData = ["program" /* Program */, "createProgram"];
+        ts.tracing.begin.apply(ts.tracing, tracingData);
         ts.performance.mark("beforeProgram");
         var host = createProgramOptions.host || createCompilerHost(options);
         var configParsingHost = parseConfigHostFromCompilerHostLike(host);
@@ -100029,7 +105289,7 @@ var ts;
         var supportedExtensions = ts.getSupportedExtensions(options);
         var supportedExtensionsWithJsonIfResolveJsonModule = ts.getSuppoertedExtensionsWithJsonIfResolveJsonModule(options, supportedExtensions);
         // Map storing if there is emit blocking diagnostics for given input
-        var hasEmitBlockingDiagnostics = ts.createMap();
+        var hasEmitBlockingDiagnostics = new ts.Map();
         var _compilerOptionsObjectLiteralSyntax;
         var moduleResolutionCache;
         var actualResolveModuleNamesWorker;
@@ -100061,9 +105321,9 @@ var ts;
         // Map from a stringified PackageId to the source file with that id.
         // Only one source file may have a given packageId. Others become redirects (see createRedirectSourceFile).
         // `packageIdToSourceFile` is only used while building the program, while `sourceFileToPackageName` and `isSourceFileTargetOfRedirect` are kept around.
-        var packageIdToSourceFile = ts.createMap();
+        var packageIdToSourceFile = new ts.Map();
         // Maps from a SourceFile's `.path` to the name of the package it was imported with.
-        var sourceFileToPackageName = ts.createMap();
+        var sourceFileToPackageName = new ts.Map();
         // Key is a file name. Value is the (non-empty, or undefined) list of files that redirect to it.
         var redirectTargetsMap = ts.createMultiMap();
         /**
@@ -100072,11 +105332,11 @@ var ts;
          * - false if sourceFile missing for source of project reference redirect
          * - undefined otherwise
          */
-        var filesByName = ts.createMap();
+        var filesByName = new ts.Map();
         var missingFilePaths;
         // stores 'filename -> file association' ignoring case
         // used to track cases when two file names differ only in casing
-        var filesByNameIgnoreCase = host.useCaseSensitiveFileNames() ? ts.createMap() : undefined;
+        var filesByNameIgnoreCase = host.useCaseSensitiveFileNames() ? new ts.Map() : undefined;
         // A parallel array to projectReferences storing the results of reading in the referenced tsconfig files
         var resolvedProjectReferences;
         var projectReferenceRedirects;
@@ -100084,20 +105344,25 @@ var ts;
         var mapFromToProjectReferenceRedirectSource;
         var useSourceOfProjectReferenceRedirect = !!((_a = host.useSourceOfProjectReferenceRedirect) === null || _a === void 0 ? void 0 : _a.call(host)) &&
             !options.disableSourceOfProjectReferenceRedirect;
-        var _b = updateHostForUseSourceOfProjectReferenceRedirect({
+        var _c = updateHostForUseSourceOfProjectReferenceRedirect({
             compilerHost: host,
+            getSymlinkCache: getSymlinkCache,
             useSourceOfProjectReferenceRedirect: useSourceOfProjectReferenceRedirect,
             toPath: toPath,
             getResolvedProjectReferences: getResolvedProjectReferences,
             getSourceOfProjectReferenceRedirect: getSourceOfProjectReferenceRedirect,
             forEachResolvedProjectReference: forEachResolvedProjectReference
-        }), onProgramCreateComplete = _b.onProgramCreateComplete, fileExists = _b.fileExists;
+        }), onProgramCreateComplete = _c.onProgramCreateComplete, fileExists = _c.fileExists, directoryExists = _c.directoryExists;
+        ts.tracing.push("program" /* Program */, "shouldProgramCreateNewSourceFiles", { hasOldProgram: !!oldProgram });
         var shouldCreateNewSourceFile = shouldProgramCreateNewSourceFiles(oldProgram, options);
+        ts.tracing.pop();
         // We set `structuralIsReused` to `undefined` because `tryReuseStructureFromOldProgram` calls `tryReuseStructureFromOldProgram` which checks
         // `structuralIsReused`, which would be a TDZ violation if it was not set in advance to `undefined`.
-        var structuralIsReused;
-        structuralIsReused = tryReuseStructureFromOldProgram(); // eslint-disable-line prefer-const
-        if (structuralIsReused !== 2 /* Completely */) {
+        var structureIsReused;
+        ts.tracing.push("program" /* Program */, "tryReuseStructureFromOldProgram", {});
+        structureIsReused = tryReuseStructureFromOldProgram(); // eslint-disable-line prefer-const
+        ts.tracing.pop();
+        if (structureIsReused !== 2 /* Completely */) {
             processingDefaultLibFiles = [];
             processingOtherFiles = [];
             if (projectReferences) {
@@ -100109,11 +105374,11 @@ var ts;
                         var parsedRef = resolvedProjectReferences_1[_i];
                         if (!parsedRef)
                             continue;
-                        var out = parsedRef.commandLine.options.outFile || parsedRef.commandLine.options.out;
+                        var out = ts.outFile(parsedRef.commandLine.options);
                         if (useSourceOfProjectReferenceRedirect) {
                             if (out || ts.getEmitModuleKind(parsedRef.commandLine.options) === ts.ModuleKind.None) {
-                                for (var _c = 0, _d = parsedRef.commandLine.fileNames; _c < _d.length; _c++) {
-                                    var fileName = _d[_c];
+                                for (var _d = 0, _e = parsedRef.commandLine.fileNames; _d < _e.length; _d++) {
+                                    var fileName = _e[_d];
                                     processSourceFile(fileName, /*isDefaultLib*/ false, /*ignoreNoDefaultLib*/ false, /*packageId*/ undefined);
                                 }
                             }
@@ -100123,8 +105388,8 @@ var ts;
                                 processSourceFile(ts.changeExtension(out, ".d.ts"), /*isDefaultLib*/ false, /*ignoreNoDefaultLib*/ false, /*packageId*/ undefined);
                             }
                             else if (ts.getEmitModuleKind(parsedRef.commandLine.options) === ts.ModuleKind.None) {
-                                for (var _e = 0, _f = parsedRef.commandLine.fileNames; _e < _f.length; _e++) {
-                                    var fileName = _f[_e];
+                                for (var _f = 0, _g = parsedRef.commandLine.fileNames; _f < _g.length; _f++) {
+                                    var fileName = _g[_f];
                                     if (!ts.fileExtensionIs(fileName, ".d.ts" /* Dts */) && !ts.fileExtensionIs(fileName, ".json" /* Json */)) {
                                         processSourceFile(ts.getOutputDeclarationFileName(fileName, parsedRef.commandLine, !host.useCaseSensitiveFileNames()), /*isDefaultLib*/ false, /*ignoreNoDefaultLib*/ false, /*packageId*/ undefined);
                                     }
@@ -100134,10 +105399,13 @@ var ts;
                     }
                 }
             }
+            ts.tracing.push("program" /* Program */, "processRootFiles", { count: rootNames.length });
             ts.forEach(rootNames, function (name) { return processRootFile(name, /*isDefaultLib*/ false, /*ignoreNoDefaultLib*/ false); });
+            ts.tracing.pop();
             // load type declarations specified via 'types' argument or implicitly from types/ and node_modules/@types folders
             var typeReferences = rootNames.length ? ts.getAutomaticTypeDirectiveNames(options, host) : ts.emptyArray;
             if (typeReferences.length) {
+                ts.tracing.push("program" /* Program */, "processTypeReferences", { count: typeReferences.length });
                 // This containingFilename needs to match with the one used in managed-side
                 var containingDirectory = options.configFilePath ? ts.getDirectoryPath(options.configFilePath) : host.getCurrentDirectory();
                 var containingFilename = ts.combinePaths(containingDirectory, ts.inferredTypesContainingFile);
@@ -100145,6 +105413,7 @@ var ts;
                 for (var i = 0; i < typeReferences.length; i++) {
                     processTypeReferenceDirective(typeReferences[i], resolutions[i]);
                 }
+                ts.tracing.pop();
             }
             // Do not process the default library if:
             //  - The '--noLib' flag is used.
@@ -100176,8 +105445,8 @@ var ts;
         // not part of the new program.
         if (oldProgram && host.onReleaseOldSourceFile) {
             var oldSourceFiles = oldProgram.getSourceFiles();
-            for (var _g = 0, oldSourceFiles_1 = oldSourceFiles; _g < oldSourceFiles_1.length; _g++) {
-                var oldSourceFile = oldSourceFiles_1[_g];
+            for (var _h = 0, oldSourceFiles_1 = oldSourceFiles; _h < oldSourceFiles_1.length; _h++) {
+                var oldSourceFile = oldSourceFiles_1[_h];
                 var newFile = getSourceFileByPath(oldSourceFile.resolvedPath);
                 if (shouldCreateNewSourceFile || !newFile ||
                     // old file wasnt redirect but new file is
@@ -100185,8 +105454,8 @@ var ts;
                     host.onReleaseOldSourceFile(oldSourceFile, oldProgram.getCompilerOptions(), !!getSourceFileByPath(oldSourceFile.path));
                 }
             }
-            oldProgram.forEachResolvedProjectReference(function (resolvedProjectReference, resolvedProjectReferencePath) {
-                if (resolvedProjectReference && !getResolvedProjectReferenceByPath(resolvedProjectReferencePath)) {
+            oldProgram.forEachResolvedProjectReference(function (resolvedProjectReference) {
+                if (!getResolvedProjectReferenceByPath(resolvedProjectReference.sourceFile.path)) {
                     host.onReleaseOldSourceFile(resolvedProjectReference.sourceFile, oldProgram.getCompilerOptions(), /*hasSourceFileByPath*/ false);
                 }
             });
@@ -100206,6 +105475,7 @@ var ts;
             getOptionsDiagnostics: getOptionsDiagnostics,
             getGlobalDiagnostics: getGlobalDiagnostics,
             getSemanticDiagnostics: getSemanticDiagnostics,
+            getCachedSemanticDiagnostics: getCachedSemanticDiagnostics,
             getSuggestionDiagnostics: getSuggestionDiagnostics,
             getDeclarationDiagnostics: getDeclarationDiagnostics,
             getBindAndCheckDiagnostics: getBindAndCheckDiagnostics,
@@ -100219,6 +105489,7 @@ var ts;
             getNodeCount: function () { return getDiagnosticsProducingTypeChecker().getNodeCount(); },
             getIdentifierCount: function () { return getDiagnosticsProducingTypeChecker().getIdentifierCount(); },
             getSymbolCount: function () { return getDiagnosticsProducingTypeChecker().getSymbolCount(); },
+            getTypeCatalog: function () { return getDiagnosticsProducingTypeChecker().getTypeCatalog(); },
             getTypeCount: function () { return getDiagnosticsProducingTypeChecker().getTypeCount(); },
             getInstantiationCount: function () { return getDiagnosticsProducingTypeChecker().getInstantiationCount(); },
             getRelationCacheSizes: function () { return getDiagnosticsProducingTypeChecker().getRelationCacheSizes(); },
@@ -100243,28 +105514,77 @@ var ts;
             isSourceOfProjectReferenceRedirect: isSourceOfProjectReferenceRedirect,
             emitBuildInfo: emitBuildInfo,
             fileExists: fileExists,
-            getProbableSymlinks: getProbableSymlinks,
+            directoryExists: directoryExists,
+            getSymlinkCache: getSymlinkCache,
+            realpath: (_b = host.realpath) === null || _b === void 0 ? void 0 : _b.bind(host),
             useCaseSensitiveFileNames: function () { return host.useCaseSensitiveFileNames(); },
+            structureIsReused: structureIsReused,
         };
         onProgramCreateComplete();
         verifyCompilerOptions();
         ts.performance.mark("afterProgram");
         ts.performance.measure("Program", "beforeProgram", "afterProgram");
+        ts.tracing.end.apply(ts.tracing, tracingData);
         return program;
-        function resolveModuleNamesWorker(moduleNames, containingFile, reusedNames, redirectedReference) {
+        function resolveModuleNamesWorker(moduleNames, containingFile, reusedNames) {
+            if (!moduleNames.length)
+                return ts.emptyArray;
+            var containingFileName = ts.getNormalizedAbsolutePath(containingFile.originalFileName, currentDirectory);
+            var redirectedReference = getRedirectReferenceForResolution(containingFile);
+            ts.tracing.push("program" /* Program */, "resolveModuleNamesWorker", { containingFileName: containingFileName });
             ts.performance.mark("beforeResolveModule");
-            var result = actualResolveModuleNamesWorker(moduleNames, containingFile, reusedNames, redirectedReference);
+            var result = actualResolveModuleNamesWorker(moduleNames, containingFileName, reusedNames, redirectedReference);
             ts.performance.mark("afterResolveModule");
             ts.performance.measure("ResolveModule", "beforeResolveModule", "afterResolveModule");
+            ts.tracing.pop();
             return result;
         }
-        function resolveTypeReferenceDirectiveNamesWorker(typeDirectiveNames, containingFile, redirectedReference) {
+        function resolveTypeReferenceDirectiveNamesWorker(typeDirectiveNames, containingFile) {
+            if (!typeDirectiveNames.length)
+                return [];
+            var containingFileName = !ts.isString(containingFile) ? ts.getNormalizedAbsolutePath(containingFile.originalFileName, currentDirectory) : containingFile;
+            var redirectedReference = !ts.isString(containingFile) ? getRedirectReferenceForResolution(containingFile) : undefined;
+            ts.tracing.push("program" /* Program */, "resolveTypeReferenceDirectiveNamesWorker", { containingFileName: containingFileName });
             ts.performance.mark("beforeResolveTypeReference");
-            var result = actualResolveTypeReferenceDirectiveNamesWorker(typeDirectiveNames, containingFile, redirectedReference);
+            var result = actualResolveTypeReferenceDirectiveNamesWorker(typeDirectiveNames, containingFileName, redirectedReference);
             ts.performance.mark("afterResolveTypeReference");
             ts.performance.measure("ResolveTypeReference", "beforeResolveTypeReference", "afterResolveTypeReference");
+            ts.tracing.pop();
             return result;
         }
+        function getRedirectReferenceForResolution(file) {
+            var redirect = getResolvedProjectReferenceToRedirect(file.originalFileName);
+            if (redirect || !ts.fileExtensionIs(file.originalFileName, ".d.ts" /* Dts */))
+                return redirect;
+            // The originalFileName could not be actual source file name if file found was d.ts from referecned project
+            // So in this case try to look up if this is output from referenced project, if it is use the redirected project in that case
+            var resultFromDts = getRedirectReferenceForResolutionFromSourceOfProject(file.originalFileName, file.path);
+            if (resultFromDts)
+                return resultFromDts;
+            // If preserveSymlinks is true, module resolution wont jump the symlink
+            // but the resolved real path may be the .d.ts from project reference
+            // Note:: Currently we try the real path only if the
+            // file is from node_modules to avoid having to run real path on all file paths
+            if (!host.realpath || !options.preserveSymlinks || !ts.stringContains(file.originalFileName, ts.nodeModulesPathPart))
+                return undefined;
+            var realDeclarationFileName = host.realpath(file.originalFileName);
+            var realDeclarationPath = toPath(realDeclarationFileName);
+            return realDeclarationPath === file.path ? undefined : getRedirectReferenceForResolutionFromSourceOfProject(realDeclarationFileName, realDeclarationPath);
+        }
+        function getRedirectReferenceForResolutionFromSourceOfProject(fileName, filePath) {
+            var source = getSourceOfProjectReferenceRedirect(fileName);
+            if (ts.isString(source))
+                return getResolvedProjectReferenceToRedirect(source);
+            if (!source)
+                return undefined;
+            // Output of .d.ts file so return resolved ref that matches the out file name
+            return forEachResolvedProjectReference(function (resolvedRef) {
+                var out = ts.outFile(resolvedRef.commandLine.options);
+                if (!out)
+                    return undefined;
+                return toPath(out) === filePath ? resolvedRef : undefined;
+            });
+        }
         function compareDefaultLibFiles(a, b) {
             return ts.compareValues(getDefaultLibFilePriority(a), getDefaultLibFilePriority(b));
         }
@@ -100311,24 +105631,25 @@ var ts;
             return commonSourceDirectory;
         }
         function getClassifiableNames() {
+            var _a;
             if (!classifiableNames) {
                 // Initialize a checker so that all our files are bound.
                 getTypeChecker();
-                classifiableNames = ts.createUnderscoreEscapedMap();
+                classifiableNames = new ts.Set();
                 for (var _i = 0, files_2 = files; _i < files_2.length; _i++) {
                     var sourceFile = files_2[_i];
-                    ts.copyEntries(sourceFile.classifiableNames, classifiableNames);
+                    (_a = sourceFile.classifiableNames) === null || _a === void 0 ? void 0 : _a.forEach(function (value) { return classifiableNames.add(value); });
                 }
             }
             return classifiableNames;
         }
-        function resolveModuleNamesReusingOldState(moduleNames, containingFile, file) {
-            if (structuralIsReused === 0 /* Not */ && !file.ambientModuleNames.length) {
+        function resolveModuleNamesReusingOldState(moduleNames, file) {
+            if (structureIsReused === 0 /* Not */ && !file.ambientModuleNames.length) {
                 // If the old program state does not permit reusing resolutions and `file` does not contain locally defined ambient modules,
                 // the best we can do is fallback to the default logic.
-                return resolveModuleNamesWorker(moduleNames, containingFile, /*reusedNames*/ undefined, getResolvedProjectReferenceToRedirect(file.originalFileName));
+                return resolveModuleNamesWorker(moduleNames, file, /*reusedNames*/ undefined);
             }
-            var oldSourceFile = oldProgram && oldProgram.getSourceFile(containingFile);
+            var oldSourceFile = oldProgram && oldProgram.getSourceFile(file.fileName);
             if (oldSourceFile !== file && file.resolvedModules) {
                 // `file` was created for the new program.
                 //
@@ -100338,13 +105659,13 @@ var ts;
                 // which per above occurred during the current program creation.
                 // Since we assume the filesystem does not change during program creation,
                 // it is safe to reuse resolutions from the earlier call.
-                var result_11 = [];
+                var result_14 = [];
                 for (var _i = 0, moduleNames_1 = moduleNames; _i < moduleNames_1.length; _i++) {
                     var moduleName = moduleNames_1[_i];
                     var resolvedModule = file.resolvedModules.get(moduleName);
-                    result_11.push(resolvedModule);
+                    result_14.push(resolvedModule);
                 }
-                return result_11;
+                return result_14;
             }
             // At this point, we know at least one of the following hold:
             // - file has local declarations for ambient modules
@@ -100369,10 +105690,10 @@ var ts;
                 var moduleName = moduleNames[i];
                 // If the source file is unchanged and doesnt have invalidated resolution, reuse the module resolutions
                 if (file === oldSourceFile && !hasInvalidatedResolution(oldSourceFile.path)) {
-                    var oldResolvedModule = oldSourceFile && oldSourceFile.resolvedModules.get(moduleName);
+                    var oldResolvedModule = ts.getResolvedModule(oldSourceFile, moduleName);
                     if (oldResolvedModule) {
                         if (ts.isTraceEnabled(options, host)) {
-                            ts.trace(host, ts.Diagnostics.Reusing_resolution_of_module_0_to_file_1_from_old_program, moduleName, containingFile);
+                            ts.trace(host, ts.Diagnostics.Reusing_resolution_of_module_0_to_file_1_from_old_program, moduleName, ts.getNormalizedAbsolutePath(file.originalFileName, currentDirectory));
                         }
                         (result || (result = new Array(moduleNames.length)))[i] = oldResolvedModule;
                         (reusedNames || (reusedNames = [])).push(moduleName);
@@ -100387,7 +105708,7 @@ var ts;
                 if (ts.contains(file.ambientModuleNames, moduleName)) {
                     resolvesToAmbientModuleInNonModifiedFile = true;
                     if (ts.isTraceEnabled(options, host)) {
-                        ts.trace(host, ts.Diagnostics.Module_0_was_resolved_as_locally_declared_ambient_module_in_file_1, moduleName, containingFile);
+                        ts.trace(host, ts.Diagnostics.Module_0_was_resolved_as_locally_declared_ambient_module_in_file_1, moduleName, ts.getNormalizedAbsolutePath(file.originalFileName, currentDirectory));
                     }
                 }
                 else {
@@ -100402,7 +105723,7 @@ var ts;
                 }
             }
             var resolutions = unknownModuleNames && unknownModuleNames.length
-                ? resolveModuleNamesWorker(unknownModuleNames, containingFile, reusedNames, getResolvedProjectReferenceToRedirect(file.originalFileName))
+                ? resolveModuleNamesWorker(unknownModuleNames, file, reusedNames)
                 : ts.emptyArray;
             // Combine results of resolutions and predicted results
             if (!result) {
@@ -100450,7 +105771,7 @@ var ts;
             }
         }
         function canReuseProjectReferences() {
-            return !forEachProjectReference(oldProgram.getProjectReferences(), oldProgram.getResolvedProjectReferences(), function (oldResolvedRef, index, parent) {
+            return !forEachProjectReference(oldProgram.getProjectReferences(), oldProgram.getResolvedProjectReferences(), function (oldResolvedRef, parent, index) {
                 var newRef = (parent ? parent.commandLine.projectReferences : projectReferences)[index];
                 var newResolvedRef = parseProjectReferenceConfigFile(newRef);
                 if (oldResolvedRef) {
@@ -100468,6 +105789,7 @@ var ts;
             });
         }
         function tryReuseStructureFromOldProgram() {
+            var _a;
             if (!oldProgram) {
                 return 0 /* Not */;
             }
@@ -100475,20 +105797,19 @@ var ts;
             // if any of these properties has changed - structure cannot be reused
             var oldOptions = oldProgram.getCompilerOptions();
             if (ts.changesAffectModuleResolution(oldOptions, options)) {
-                return oldProgram.structureIsReused = 0 /* Not */;
+                return 0 /* Not */;
             }
-            ts.Debug.assert(!(oldProgram.structureIsReused & (2 /* Completely */ | 1 /* SafeModules */)));
             // there is an old program, check if we can reuse its structure
             var oldRootNames = oldProgram.getRootFileNames();
             if (!ts.arrayIsEqualTo(oldRootNames, rootNames)) {
-                return oldProgram.structureIsReused = 0 /* Not */;
+                return 0 /* Not */;
             }
             if (!ts.arrayIsEqualTo(options.types, oldOptions.types)) {
-                return oldProgram.structureIsReused = 0 /* Not */;
+                return 0 /* Not */;
             }
             // Check if any referenced project tsconfig files are different
             if (!canReuseProjectReferences()) {
-                return oldProgram.structureIsReused = 0 /* Not */;
+                return 0 /* Not */;
             }
             if (projectReferences) {
                 resolvedProjectReferences = projectReferences.map(parseProjectReferenceConfigFile);
@@ -100496,12 +105817,12 @@ var ts;
             // check if program source files has changed in the way that can affect structure of the program
             var newSourceFiles = [];
             var modifiedSourceFiles = [];
-            oldProgram.structureIsReused = 2 /* Completely */;
+            structureIsReused = 2 /* Completely */;
             // If the missing file paths are now present, it can change the progam structure,
             // and hence cant reuse the structure.
             // This is same as how we dont reuse the structure if one of the file from old program is now missing
             if (oldProgram.getMissingFilePaths().some(function (missingFilePath) { return host.fileExists(missingFilePath); })) {
-                return oldProgram.structureIsReused = 0 /* Not */;
+                return 0 /* Not */;
             }
             var oldSourceFiles = oldProgram.getSourceFiles();
             var SeenPackageName;
@@ -100509,14 +105830,14 @@ var ts;
                 SeenPackageName[SeenPackageName["Exists"] = 0] = "Exists";
                 SeenPackageName[SeenPackageName["Modified"] = 1] = "Modified";
             })(SeenPackageName || (SeenPackageName = {}));
-            var seenPackageNames = ts.createMap();
+            var seenPackageNames = new ts.Map();
             for (var _i = 0, oldSourceFiles_2 = oldSourceFiles; _i < oldSourceFiles_2.length; _i++) {
                 var oldSourceFile = oldSourceFiles_2[_i];
                 var newSourceFile = host.getSourceFileByPath
                     ? host.getSourceFileByPath(oldSourceFile.fileName, oldSourceFile.resolvedPath, options.target, /*onError*/ undefined, shouldCreateNewSourceFile)
                     : host.getSourceFile(oldSourceFile.fileName, options.target, /*onError*/ undefined, shouldCreateNewSourceFile); // TODO: GH#18217
                 if (!newSourceFile) {
-                    return oldProgram.structureIsReused = 0 /* Not */;
+                    return 0 /* Not */;
                 }
                 ts.Debug.assert(!newSourceFile.redirectInfo, "Host should not return a redirect source file from `getSourceFile`");
                 var fileChanged = void 0;
@@ -100525,7 +105846,7 @@ var ts;
                     // This lets us know if the unredirected file has changed. If it has we should break the redirect.
                     if (newSourceFile !== oldSourceFile.redirectInfo.unredirected) {
                         // Underlying file has changed. Might not redirect anymore. Must rebuild program.
-                        return oldProgram.structureIsReused = 0 /* Not */;
+                        return 0 /* Not */;
                     }
                     fileChanged = false;
                     newSourceFile = oldSourceFile; // Use the redirect.
@@ -100533,7 +105854,7 @@ var ts;
                 else if (oldProgram.redirectTargetsMap.has(oldSourceFile.path)) {
                     // If a redirected-to source file changes, the redirect may be broken.
                     if (newSourceFile !== oldSourceFile) {
-                        return oldProgram.structureIsReused = 0 /* Not */;
+                        return 0 /* Not */;
                     }
                     fileChanged = false;
                 }
@@ -100552,7 +105873,7 @@ var ts;
                     var prevKind = seenPackageNames.get(packageName);
                     var newKind = fileChanged ? 1 /* Modified */ : 0 /* Exists */;
                     if ((prevKind !== undefined && newKind === 1 /* Modified */) || prevKind === 1 /* Modified */) {
-                        return oldProgram.structureIsReused = 0 /* Not */;
+                        return 0 /* Not */;
                     }
                     seenPackageNames.set(packageName, newKind);
                 }
@@ -100560,103 +105881,100 @@ var ts;
                     // The `newSourceFile` object was created for the new program.
                     if (!ts.arrayIsEqualTo(oldSourceFile.libReferenceDirectives, newSourceFile.libReferenceDirectives, fileReferenceIsEqualTo)) {
                         // 'lib' references has changed. Matches behavior in changesAffectModuleResolution
-                        return oldProgram.structureIsReused = 0 /* Not */;
+                        return 0 /* Not */;
                     }
                     if (oldSourceFile.hasNoDefaultLib !== newSourceFile.hasNoDefaultLib) {
                         // value of no-default-lib has changed
                         // this will affect if default library is injected into the list of files
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
+                        structureIsReused = 1 /* SafeModules */;
                     }
                     // check tripleslash references
                     if (!ts.arrayIsEqualTo(oldSourceFile.referencedFiles, newSourceFile.referencedFiles, fileReferenceIsEqualTo)) {
                         // tripleslash references has changed
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
+                        structureIsReused = 1 /* SafeModules */;
                     }
                     // check imports and module augmentations
                     collectExternalModuleReferences(newSourceFile);
                     if (!ts.arrayIsEqualTo(oldSourceFile.imports, newSourceFile.imports, moduleNameIsEqualTo)) {
                         // imports has changed
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
+                        structureIsReused = 1 /* SafeModules */;
                     }
                     if (!ts.arrayIsEqualTo(oldSourceFile.moduleAugmentations, newSourceFile.moduleAugmentations, moduleNameIsEqualTo)) {
                         // moduleAugmentations has changed
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
+                        structureIsReused = 1 /* SafeModules */;
                     }
                     if ((oldSourceFile.flags & 3145728 /* PermanentlySetIncrementalFlags */) !== (newSourceFile.flags & 3145728 /* PermanentlySetIncrementalFlags */)) {
                         // dynamicImport has changed
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
+                        structureIsReused = 1 /* SafeModules */;
                     }
                     if (!ts.arrayIsEqualTo(oldSourceFile.typeReferenceDirectives, newSourceFile.typeReferenceDirectives, fileReferenceIsEqualTo)) {
                         // 'types' references has changed
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
+                        structureIsReused = 1 /* SafeModules */;
                     }
                     // tentatively approve the file
                     modifiedSourceFiles.push({ oldFile: oldSourceFile, newFile: newSourceFile });
                 }
                 else if (hasInvalidatedResolution(oldSourceFile.path)) {
                     // 'module/types' references could have changed
-                    oldProgram.structureIsReused = 1 /* SafeModules */;
+                    structureIsReused = 1 /* SafeModules */;
                     // add file to the modified list so that we will resolve it later
                     modifiedSourceFiles.push({ oldFile: oldSourceFile, newFile: newSourceFile });
                 }
                 // if file has passed all checks it should be safe to reuse it
                 newSourceFiles.push(newSourceFile);
             }
-            if (oldProgram.structureIsReused !== 2 /* Completely */) {
-                return oldProgram.structureIsReused;
+            if (structureIsReused !== 2 /* Completely */) {
+                return structureIsReused;
             }
             var modifiedFiles = modifiedSourceFiles.map(function (f) { return f.oldFile; });
-            for (var _a = 0, oldSourceFiles_3 = oldSourceFiles; _a < oldSourceFiles_3.length; _a++) {
-                var oldFile = oldSourceFiles_3[_a];
+            for (var _b = 0, oldSourceFiles_3 = oldSourceFiles; _b < oldSourceFiles_3.length; _b++) {
+                var oldFile = oldSourceFiles_3[_b];
                 if (!ts.contains(modifiedFiles, oldFile)) {
-                    for (var _b = 0, _c = oldFile.ambientModuleNames; _b < _c.length; _b++) {
-                        var moduleName = _c[_b];
+                    for (var _c = 0, _d = oldFile.ambientModuleNames; _c < _d.length; _c++) {
+                        var moduleName = _d[_c];
                         ambientModuleNameToUnmodifiedFileName.set(moduleName, oldFile.fileName);
                     }
                 }
             }
             // try to verify results of module resolution
-            for (var _d = 0, modifiedSourceFiles_1 = modifiedSourceFiles; _d < modifiedSourceFiles_1.length; _d++) {
-                var _e = modifiedSourceFiles_1[_d], oldSourceFile = _e.oldFile, newSourceFile = _e.newFile;
-                var newSourceFilePath = ts.getNormalizedAbsolutePath(newSourceFile.originalFileName, currentDirectory);
+            for (var _e = 0, modifiedSourceFiles_1 = modifiedSourceFiles; _e < modifiedSourceFiles_1.length; _e++) {
+                var _f = modifiedSourceFiles_1[_e], oldSourceFile = _f.oldFile, newSourceFile = _f.newFile;
                 var moduleNames = getModuleNames(newSourceFile);
-                var resolutions = resolveModuleNamesReusingOldState(moduleNames, newSourceFilePath, newSourceFile);
+                var resolutions = resolveModuleNamesReusingOldState(moduleNames, newSourceFile);
                 // ensure that module resolution results are still correct
                 var resolutionsChanged = ts.hasChangesInResolutions(moduleNames, resolutions, oldSourceFile.resolvedModules, ts.moduleResolutionIsEqualTo);
                 if (resolutionsChanged) {
-                    oldProgram.structureIsReused = 1 /* SafeModules */;
+                    structureIsReused = 1 /* SafeModules */;
                     newSourceFile.resolvedModules = ts.zipToMap(moduleNames, resolutions);
                 }
                 else {
                     newSourceFile.resolvedModules = oldSourceFile.resolvedModules;
                 }
-                if (resolveTypeReferenceDirectiveNamesWorker) {
-                    // We lower-case all type references because npm automatically lowercases all packages. See GH#9824.
-                    var typesReferenceDirectives = ts.map(newSourceFile.typeReferenceDirectives, function (ref) { return ts.toFileNameLowerCase(ref.fileName); });
-                    var resolutions_1 = resolveTypeReferenceDirectiveNamesWorker(typesReferenceDirectives, newSourceFilePath, getResolvedProjectReferenceToRedirect(newSourceFile.originalFileName));
-                    // ensure that types resolutions are still correct
-                    var resolutionsChanged_1 = ts.hasChangesInResolutions(typesReferenceDirectives, resolutions_1, oldSourceFile.resolvedTypeReferenceDirectiveNames, ts.typeDirectiveIsEqualTo);
-                    if (resolutionsChanged_1) {
-                        oldProgram.structureIsReused = 1 /* SafeModules */;
-                        newSourceFile.resolvedTypeReferenceDirectiveNames = ts.zipToMap(typesReferenceDirectives, resolutions_1);
-                    }
-                    else {
-                        newSourceFile.resolvedTypeReferenceDirectiveNames = oldSourceFile.resolvedTypeReferenceDirectiveNames;
-                    }
+                // We lower-case all type references because npm automatically lowercases all packages. See GH#9824.
+                var typesReferenceDirectives = ts.map(newSourceFile.typeReferenceDirectives, function (ref) { return ts.toFileNameLowerCase(ref.fileName); });
+                var typeReferenceResolutions = resolveTypeReferenceDirectiveNamesWorker(typesReferenceDirectives, newSourceFile);
+                // ensure that types resolutions are still correct
+                var typeReferenceEesolutionsChanged = ts.hasChangesInResolutions(typesReferenceDirectives, typeReferenceResolutions, oldSourceFile.resolvedTypeReferenceDirectiveNames, ts.typeDirectiveIsEqualTo);
+                if (typeReferenceEesolutionsChanged) {
+                    structureIsReused = 1 /* SafeModules */;
+                    newSourceFile.resolvedTypeReferenceDirectiveNames = ts.zipToMap(typesReferenceDirectives, typeReferenceResolutions);
+                }
+                else {
+                    newSourceFile.resolvedTypeReferenceDirectiveNames = oldSourceFile.resolvedTypeReferenceDirectiveNames;
                 }
             }
-            if (oldProgram.structureIsReused !== 2 /* Completely */) {
-                return oldProgram.structureIsReused;
+            if (structureIsReused !== 2 /* Completely */) {
+                return structureIsReused;
             }
-            if (host.hasChangedAutomaticTypeDirectiveNames) {
-                return oldProgram.structureIsReused = 1 /* SafeModules */;
+            if ((_a = host.hasChangedAutomaticTypeDirectiveNames) === null || _a === void 0 ? void 0 : _a.call(host)) {
+                return 1 /* SafeModules */;
             }
             missingFilePaths = oldProgram.getMissingFilePaths();
             refFileMap = oldProgram.getRefFileMap();
             // update fileName -> file mapping
             ts.Debug.assert(newSourceFiles.length === oldProgram.getSourceFiles().length);
-            for (var _f = 0, newSourceFiles_1 = newSourceFiles; _f < newSourceFiles_1.length; _f++) {
-                var newSourceFile = newSourceFiles_1[_f];
+            for (var _g = 0, newSourceFiles_1 = newSourceFiles; _g < newSourceFiles_1.length; _g++) {
+                var newSourceFile = newSourceFiles_1[_g];
                 filesByName.set(newSourceFile.path, newSourceFile);
             }
             var oldFilesByNameMap = oldProgram.getFilesByNameMap();
@@ -100676,14 +105994,14 @@ var ts;
             });
             files = newSourceFiles;
             fileProcessingDiagnostics = oldProgram.getFileProcessingDiagnostics();
-            for (var _g = 0, modifiedSourceFiles_2 = modifiedSourceFiles; _g < modifiedSourceFiles_2.length; _g++) {
-                var modifiedFile = modifiedSourceFiles_2[_g];
+            for (var _h = 0, modifiedSourceFiles_2 = modifiedSourceFiles; _h < modifiedSourceFiles_2.length; _h++) {
+                var modifiedFile = modifiedSourceFiles_2[_h];
                 fileProcessingDiagnostics.reattachFileDiagnostics(modifiedFile.newFile);
             }
             resolvedTypeReferenceDirectives = oldProgram.getResolvedTypeReferenceDirectives();
             sourceFileToPackageName = oldProgram.sourceFileToPackageName;
             redirectTargetsMap = oldProgram.redirectTargetsMap;
-            return oldProgram.structureIsReused = 2 /* Completely */;
+            return 2 /* Completely */;
         }
         function getEmitHost(writeFileCallback) {
             return {
@@ -100701,7 +106019,7 @@ var ts;
                 getResolvedProjectReferenceToRedirect: getResolvedProjectReferenceToRedirect,
                 getProjectReferenceRedirect: getProjectReferenceRedirect,
                 isSourceOfProjectReferenceRedirect: isSourceOfProjectReferenceRedirect,
-                getProbableSymlinks: getProbableSymlinks,
+                getSymlinkCache: getSymlinkCache,
                 writeFile: writeFileCallback || (function (fileName, data, writeByteOrderMark, onError, sourceFiles) { return host.writeFile(fileName, data, writeByteOrderMark, onError, sourceFiles); }),
                 isEmitBlocked: isEmitBlocked,
                 readFile: function (f) { return host.readFile(f); },
@@ -100722,7 +106040,9 @@ var ts;
             };
         }
         function emitBuildInfo(writeFileCallback) {
-            ts.Debug.assert(!options.out && !options.outFile);
+            ts.Debug.assert(!ts.outFile(options));
+            var tracingData = ["emit" /* Emit */, "emitBuildInfo"];
+            ts.tracing.begin.apply(ts.tracing, tracingData);
             ts.performance.mark("beforeEmit");
             var emitResult = ts.emitFiles(ts.notImplementedResolver, getEmitHost(writeFileCallback), 
             /*targetSourceFile*/ undefined, 
@@ -100731,6 +106051,7 @@ var ts;
             /*onlyBuildInfo*/ true);
             ts.performance.mark("afterEmit");
             ts.performance.measure("Emit", "beforeEmit", "afterEmit");
+            ts.tracing.end.apply(ts.tracing, tracingData);
             return emitResult;
         }
         function getResolvedProjectReferences() {
@@ -100740,7 +106061,7 @@ var ts;
             return projectReferences;
         }
         function getPrependNodes() {
-            return createPrependNodes(projectReferences, function (_ref, index) { return resolvedProjectReferences[index].commandLine; }, function (fileName) {
+            return createPrependNodes(projectReferences, function (_ref, index) { var _a; return (_a = resolvedProjectReferences[index]) === null || _a === void 0 ? void 0 : _a.commandLine; }, function (fileName) {
                 var path = toPath(fileName);
                 var sourceFile = getSourceFileByPath(path);
                 return sourceFile ? sourceFile.text : filesByName.has(path) ? undefined : host.readFile(path);
@@ -100776,14 +106097,18 @@ var ts;
             return noDiagnosticsTypeChecker || (noDiagnosticsTypeChecker = ts.createTypeChecker(program, /*produceDiagnostics:*/ false));
         }
         function emit(sourceFile, writeFileCallback, cancellationToken, emitOnlyDtsFiles, transformers, forceDtsEmit) {
-            return runWithCancellationToken(function () { return emitWorker(program, sourceFile, writeFileCallback, cancellationToken, emitOnlyDtsFiles, transformers, forceDtsEmit); });
+            var tracingData = ["emit" /* Emit */, "emit", { path: sourceFile === null || sourceFile === void 0 ? void 0 : sourceFile.path }];
+            ts.tracing.begin.apply(ts.tracing, tracingData);
+            var result = runWithCancellationToken(function () { return emitWorker(program, sourceFile, writeFileCallback, cancellationToken, emitOnlyDtsFiles, transformers, forceDtsEmit); });
+            ts.tracing.end.apply(ts.tracing, tracingData);
+            return result;
         }
         function isEmitBlocked(emitFileName) {
             return hasEmitBlockingDiagnostics.has(toPath(emitFileName));
         }
         function emitWorker(program, sourceFile, writeFileCallback, cancellationToken, emitOnlyDtsFiles, customTransformers, forceDtsEmit) {
             if (!forceDtsEmit) {
-                var result = handleNoEmitOptions(program, sourceFile, cancellationToken);
+                var result = handleNoEmitOptions(program, sourceFile, writeFileCallback, cancellationToken);
                 if (result)
                     return result;
             }
@@ -100795,7 +106120,7 @@ var ts;
             // This is because in the -out scenario all files need to be emitted, and therefore all
             // files need to be type checked. And the way to specify that all files need to be type
             // checked is to not pass the file to getEmitResolver.
-            var emitResolver = getDiagnosticsProducingTypeChecker().getEmitResolver((options.outFile || options.out) ? undefined : sourceFile, cancellationToken);
+            var emitResolver = getDiagnosticsProducingTypeChecker().getEmitResolver(ts.outFile(options) ? undefined : sourceFile, cancellationToken);
             ts.performance.mark("beforeEmit");
             var emitResult = ts.emitFiles(emitResolver, getEmitHost(writeFileCallback), sourceFile, ts.getTransformers(options, customTransformers, emitOnlyDtsFiles), emitOnlyDtsFiles, 
             /*onlyBuildInfo*/ false, forceDtsEmit);
@@ -100826,6 +106151,11 @@ var ts;
         function getSemanticDiagnostics(sourceFile, cancellationToken) {
             return getDiagnosticsHelper(sourceFile, getSemanticDiagnosticsForFile, cancellationToken);
         }
+        function getCachedSemanticDiagnostics(sourceFile) {
+            var _a;
+            return sourceFile
+                ? (_a = cachedBindAndCheckDiagnosticsForFile.perFile) === null || _a === void 0 ? void 0 : _a.get(sourceFile.path) : cachedBindAndCheckDiagnosticsForFile.allDiagnostics;
+        }
         function getBindAndCheckDiagnostics(sourceFile, cancellationToken) {
             return getBindAndCheckDiagnosticsForFile(sourceFile, cancellationToken);
         }
@@ -100852,7 +106182,7 @@ var ts;
         function getDeclarationDiagnostics(sourceFile, cancellationToken) {
             var options = program.getCompilerOptions();
             // collect diagnostics from the program only once if either no source file was specified or out/outFile is set (bundled emit)
-            if (!sourceFile || options.out || options.outFile) {
+            if (!sourceFile || ts.outFile(options)) {
                 return getDeclarationDiagnosticsWorker(sourceFile, cancellationToken);
             }
             else {
@@ -100892,7 +106222,7 @@ var ts;
             }
         }
         function getSemanticDiagnosticsForFile(sourceFile, cancellationToken) {
-            return ts.concatenate(getBindAndCheckDiagnosticsForFile(sourceFile, cancellationToken), getProgramDiagnostics(sourceFile));
+            return ts.concatenate(filterSemanticDiagnotics(getBindAndCheckDiagnosticsForFile(sourceFile, cancellationToken), options), getProgramDiagnostics(sourceFile));
         }
         function getBindAndCheckDiagnosticsForFile(sourceFile, cancellationToken) {
             return getAndCacheDiagnostics(sourceFile, cancellationToken, cachedBindAndCheckDiagnosticsForFile, getBindAndCheckDiagnosticsForFileNoCache);
@@ -100911,17 +106241,17 @@ var ts;
                     sourceFile.scriptKind === 5 /* External */ || isCheckJs || sourceFile.scriptKind === 7 /* Deferred */);
                 var bindDiagnostics = includeBindAndCheckDiagnostics ? sourceFile.bindDiagnostics : ts.emptyArray;
                 var checkDiagnostics = includeBindAndCheckDiagnostics ? typeChecker.getDiagnostics(sourceFile, cancellationToken) : ts.emptyArray;
-                return getMergedBindAndCheckDiagnostics(sourceFile, bindDiagnostics, checkDiagnostics, isCheckJs ? sourceFile.jsDocDiagnostics : undefined);
+                return getMergedBindAndCheckDiagnostics(sourceFile, includeBindAndCheckDiagnostics, bindDiagnostics, checkDiagnostics, isCheckJs ? sourceFile.jsDocDiagnostics : undefined);
             });
         }
-        function getMergedBindAndCheckDiagnostics(sourceFile) {
+        function getMergedBindAndCheckDiagnostics(sourceFile, includeBindAndCheckDiagnostics) {
             var _a;
             var allDiagnostics = [];
-            for (var _i = 1; _i < arguments.length; _i++) {
-                allDiagnostics[_i - 1] = arguments[_i];
+            for (var _i = 2; _i < arguments.length; _i++) {
+                allDiagnostics[_i - 2] = arguments[_i];
             }
             var flatDiagnostics = ts.flatten(allDiagnostics);
-            if (!((_a = sourceFile.commentDirectives) === null || _a === void 0 ? void 0 : _a.length)) {
+            if (!includeBindAndCheckDiagnostics || !((_a = sourceFile.commentDirectives) === null || _a === void 0 ? void 0 : _a.length)) {
                 return flatDiagnostics;
             }
             var _b = getDiagnosticsWithPrecedingDirectives(sourceFile, sourceFile.commentDirectives, flatDiagnostics), diagnostics = _b.diagnostics, directives = _b.directives;
@@ -100982,22 +106312,22 @@ var ts;
                     // Return directly from the case if the given node doesnt want to visit each child
                     // Otherwise break to visit each child
                     switch (parent.kind) {
-                        case 156 /* Parameter */:
-                        case 159 /* PropertyDeclaration */:
-                        case 161 /* MethodDeclaration */:
+                        case 160 /* Parameter */:
+                        case 163 /* PropertyDeclaration */:
+                        case 165 /* MethodDeclaration */:
                             if (parent.questionToken === node) {
                                 diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.The_0_modifier_can_only_be_used_in_TypeScript_files, "?"));
                                 return "skip";
                             }
                         // falls through
-                        case 160 /* MethodSignature */:
-                        case 162 /* Constructor */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
-                        case 201 /* FunctionExpression */:
-                        case 244 /* FunctionDeclaration */:
-                        case 202 /* ArrowFunction */:
-                        case 242 /* VariableDeclaration */:
+                        case 164 /* MethodSignature */:
+                        case 166 /* Constructor */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
+                        case 208 /* FunctionExpression */:
+                        case 251 /* FunctionDeclaration */:
+                        case 209 /* ArrowFunction */:
+                        case 249 /* VariableDeclaration */:
                             // type annotation
                             if (parent.type === node) {
                                 diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Type_annotations_can_only_be_used_in_TypeScript_files));
@@ -101005,58 +106335,58 @@ var ts;
                             }
                     }
                     switch (node.kind) {
-                        case 255 /* ImportClause */:
+                        case 262 /* ImportClause */:
                             if (node.isTypeOnly) {
-                                diagnostics.push(createDiagnosticForNode(node.parent, ts.Diagnostics._0_declarations_can_only_be_used_in_TypeScript_files, "import type"));
+                                diagnostics.push(createDiagnosticForNode(parent, ts.Diagnostics._0_declarations_can_only_be_used_in_TypeScript_files, "import type"));
                                 return "skip";
                             }
                             break;
-                        case 260 /* ExportDeclaration */:
+                        case 267 /* ExportDeclaration */:
                             if (node.isTypeOnly) {
                                 diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics._0_declarations_can_only_be_used_in_TypeScript_files, "export type"));
                                 return "skip";
                             }
                             break;
-                        case 253 /* ImportEqualsDeclaration */:
+                        case 260 /* ImportEqualsDeclaration */:
                             diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.import_can_only_be_used_in_TypeScript_files));
                             return "skip";
-                        case 259 /* ExportAssignment */:
+                        case 266 /* ExportAssignment */:
                             if (node.isExportEquals) {
                                 diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.export_can_only_be_used_in_TypeScript_files));
                                 return "skip";
                             }
                             break;
-                        case 279 /* HeritageClause */:
+                        case 286 /* HeritageClause */:
                             var heritageClause = node;
-                            if (heritageClause.token === 113 /* ImplementsKeyword */) {
+                            if (heritageClause.token === 116 /* ImplementsKeyword */) {
                                 diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.implements_clauses_can_only_be_used_in_TypeScript_files));
                                 return "skip";
                             }
                             break;
-                        case 246 /* InterfaceDeclaration */:
-                            var interfaceKeyword = ts.tokenToString(114 /* InterfaceKeyword */);
+                        case 253 /* InterfaceDeclaration */:
+                            var interfaceKeyword = ts.tokenToString(117 /* InterfaceKeyword */);
                             ts.Debug.assertIsDefined(interfaceKeyword);
                             diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics._0_declarations_can_only_be_used_in_TypeScript_files, interfaceKeyword));
                             return "skip";
-                        case 249 /* ModuleDeclaration */:
-                            var moduleKeyword = node.flags & 16 /* Namespace */ ? ts.tokenToString(136 /* NamespaceKeyword */) : ts.tokenToString(135 /* ModuleKeyword */);
+                        case 256 /* ModuleDeclaration */:
+                            var moduleKeyword = node.flags & 16 /* Namespace */ ? ts.tokenToString(140 /* NamespaceKeyword */) : ts.tokenToString(139 /* ModuleKeyword */);
                             ts.Debug.assertIsDefined(moduleKeyword);
                             diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics._0_declarations_can_only_be_used_in_TypeScript_files, moduleKeyword));
                             return "skip";
-                        case 247 /* TypeAliasDeclaration */:
+                        case 254 /* TypeAliasDeclaration */:
                             diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Type_aliases_can_only_be_used_in_TypeScript_files));
                             return "skip";
-                        case 248 /* EnumDeclaration */:
-                            var enumKeyword = ts.Debug.checkDefined(ts.tokenToString(88 /* EnumKeyword */));
+                        case 255 /* EnumDeclaration */:
+                            var enumKeyword = ts.Debug.checkDefined(ts.tokenToString(91 /* EnumKeyword */));
                             diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics._0_declarations_can_only_be_used_in_TypeScript_files, enumKeyword));
                             return "skip";
-                        case 218 /* NonNullExpression */:
+                        case 225 /* NonNullExpression */:
                             diagnostics.push(createDiagnosticForNode(node, ts.Diagnostics.Non_null_assertions_can_only_be_used_in_TypeScript_files));
                             return "skip";
-                        case 217 /* AsExpression */:
+                        case 224 /* AsExpression */:
                             diagnostics.push(createDiagnosticForNode(node.type, ts.Diagnostics.Type_assertion_expressions_can_only_be_used_in_TypeScript_files));
                             return "skip";
-                        case 199 /* TypeAssertionExpression */:
+                        case 206 /* TypeAssertionExpression */:
                             ts.Debug.fail(); // Won't parse these in a JS file anyway, as they are interpreted as JSX.
                     }
                 }
@@ -101065,53 +106395,53 @@ var ts;
                         diagnostics.push(createDiagnosticForNode(parent, ts.Diagnostics.Experimental_support_for_decorators_is_a_feature_that_is_subject_to_change_in_a_future_release_Set_the_experimentalDecorators_option_in_your_tsconfig_or_jsconfig_to_remove_this_warning));
                     }
                     switch (parent.kind) {
-                        case 245 /* ClassDeclaration */:
-                        case 214 /* ClassExpression */:
-                        case 161 /* MethodDeclaration */:
-                        case 162 /* Constructor */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
-                        case 201 /* FunctionExpression */:
-                        case 244 /* FunctionDeclaration */:
-                        case 202 /* ArrowFunction */:
+                        case 252 /* ClassDeclaration */:
+                        case 221 /* ClassExpression */:
+                        case 165 /* MethodDeclaration */:
+                        case 166 /* Constructor */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
+                        case 208 /* FunctionExpression */:
+                        case 251 /* FunctionDeclaration */:
+                        case 209 /* ArrowFunction */:
                             // Check type parameters
                             if (nodes === parent.typeParameters) {
                                 diagnostics.push(createDiagnosticForNodeArray(nodes, ts.Diagnostics.Type_parameter_declarations_can_only_be_used_in_TypeScript_files));
                                 return "skip";
                             }
                         // falls through
-                        case 225 /* VariableStatement */:
+                        case 232 /* VariableStatement */:
                             // Check modifiers
                             if (nodes === parent.modifiers) {
-                                checkModifiers(parent.modifiers, parent.kind === 225 /* VariableStatement */);
+                                checkModifiers(parent.modifiers, parent.kind === 232 /* VariableStatement */);
                                 return "skip";
                             }
                             break;
-                        case 159 /* PropertyDeclaration */:
+                        case 163 /* PropertyDeclaration */:
                             // Check modifiers of property declaration
                             if (nodes === parent.modifiers) {
                                 for (var _i = 0, _a = nodes; _i < _a.length; _i++) {
                                     var modifier = _a[_i];
-                                    if (modifier.kind !== 120 /* StaticKeyword */) {
+                                    if (modifier.kind !== 123 /* StaticKeyword */) {
                                         diagnostics.push(createDiagnosticForNode(modifier, ts.Diagnostics.The_0_modifier_can_only_be_used_in_TypeScript_files, ts.tokenToString(modifier.kind)));
                                     }
                                 }
                                 return "skip";
                             }
                             break;
-                        case 156 /* Parameter */:
+                        case 160 /* Parameter */:
                             // Check modifiers of parameter declaration
                             if (nodes === parent.modifiers) {
                                 diagnostics.push(createDiagnosticForNodeArray(nodes, ts.Diagnostics.Parameter_modifiers_can_only_be_used_in_TypeScript_files));
                                 return "skip";
                             }
                             break;
-                        case 196 /* CallExpression */:
-                        case 197 /* NewExpression */:
-                        case 216 /* ExpressionWithTypeArguments */:
-                        case 267 /* JsxSelfClosingElement */:
-                        case 268 /* JsxOpeningElement */:
-                        case 198 /* TaggedTemplateExpression */:
+                        case 203 /* CallExpression */:
+                        case 204 /* NewExpression */:
+                        case 223 /* ExpressionWithTypeArguments */:
+                        case 274 /* JsxSelfClosingElement */:
+                        case 275 /* JsxOpeningElement */:
+                        case 205 /* TaggedTemplateExpression */:
                             // Check type arguments
                             if (nodes === parent.typeArguments) {
                                 diagnostics.push(createDiagnosticForNodeArray(nodes, ts.Diagnostics.Type_arguments_can_only_be_used_in_TypeScript_files));
@@ -101121,27 +106451,27 @@ var ts;
                     }
                 }
                 function checkModifiers(modifiers, isConstValid) {
-                    for (var _i = 0, modifiers_1 = modifiers; _i < modifiers_1.length; _i++) {
-                        var modifier = modifiers_1[_i];
+                    for (var _i = 0, modifiers_2 = modifiers; _i < modifiers_2.length; _i++) {
+                        var modifier = modifiers_2[_i];
                         switch (modifier.kind) {
-                            case 81 /* ConstKeyword */:
+                            case 84 /* ConstKeyword */:
                                 if (isConstValid) {
                                     continue;
                                 }
                             // to report error,
                             // falls through
-                            case 119 /* PublicKeyword */:
-                            case 117 /* PrivateKeyword */:
-                            case 118 /* ProtectedKeyword */:
-                            case 138 /* ReadonlyKeyword */:
-                            case 130 /* DeclareKeyword */:
-                            case 122 /* AbstractKeyword */:
+                            case 122 /* PublicKeyword */:
+                            case 120 /* PrivateKeyword */:
+                            case 121 /* ProtectedKeyword */:
+                            case 142 /* ReadonlyKeyword */:
+                            case 133 /* DeclareKeyword */:
+                            case 125 /* AbstractKeyword */:
                                 diagnostics.push(createDiagnosticForNode(modifier, ts.Diagnostics.The_0_modifier_can_only_be_used_in_TypeScript_files, ts.tokenToString(modifier.kind)));
                                 break;
                             // These are all legal modifiers.
-                            case 120 /* StaticKeyword */:
-                            case 89 /* ExportKeyword */:
-                            case 84 /* DefaultKeyword */:
+                            case 123 /* StaticKeyword */:
+                            case 92 /* ExportKeyword */:
+                            case 87 /* DefaultKeyword */:
                         }
                     }
                 }
@@ -101167,18 +106497,15 @@ var ts;
             });
         }
         function getAndCacheDiagnostics(sourceFile, cancellationToken, cache, getDiagnostics) {
+            var _a;
             var cachedResult = sourceFile
-                ? cache.perFile && cache.perFile.get(sourceFile.path)
-                : cache.allDiagnostics;
+                ? (_a = cache.perFile) === null || _a === void 0 ? void 0 : _a.get(sourceFile.path) : cache.allDiagnostics;
             if (cachedResult) {
                 return cachedResult;
             }
             var result = getDiagnostics(sourceFile, cancellationToken);
             if (sourceFile) {
-                if (!cache.perFile) {
-                    cache.perFile = ts.createMap();
-                }
-                cache.perFile.set(sourceFile.path, result);
+                (cache.perFile || (cache.perFile = new ts.Map())).set(sourceFile.path, result);
             }
             else {
                 cache.allDiagnostics = result;
@@ -101197,9 +106524,7 @@ var ts;
             }
             var diagnostics = programDiagnostics.getDiagnostics(options.configFile.fileName);
             forEachResolvedProjectReference(function (resolvedRef) {
-                if (resolvedRef) {
-                    diagnostics = ts.concatenate(diagnostics, programDiagnostics.getDiagnostics(resolvedRef.sourceFile.fileName));
-                }
+                diagnostics = ts.concatenate(diagnostics, programDiagnostics.getDiagnostics(resolvedRef.sourceFile.fileName));
             });
             return diagnostics;
         }
@@ -101216,10 +106541,22 @@ var ts;
             return a.fileName === b.fileName;
         }
         function moduleNameIsEqualTo(a, b) {
-            return a.kind === 75 /* Identifier */
-                ? b.kind === 75 /* Identifier */ && a.escapedText === b.escapedText
+            return a.kind === 78 /* Identifier */
+                ? b.kind === 78 /* Identifier */ && a.escapedText === b.escapedText
                 : b.kind === 10 /* StringLiteral */ && a.text === b.text;
         }
+        function createSyntheticImport(text, file) {
+            var externalHelpersModuleReference = ts.factory.createStringLiteral(text);
+            var importDecl = ts.factory.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*importClause*/ undefined, externalHelpersModuleReference);
+            ts.addEmitFlags(importDecl, 67108864 /* NeverApplyImportHelper */);
+            ts.setParent(externalHelpersModuleReference, importDecl);
+            ts.setParent(importDecl, file);
+            // explicitly unset the synthesized flag on these declarations so the checker API will answer questions about them
+            // (which is required to build the dependency graph for incremental emit)
+            externalHelpersModuleReference.flags &= ~8 /* Synthesized */;
+            importDecl.flags &= ~8 /* Synthesized */;
+            return externalHelpersModuleReference;
+        }
         function collectExternalModuleReferences(file) {
             if (file.imports) {
                 return;
@@ -101232,16 +106569,17 @@ var ts;
             var ambientModules;
             // If we are importing helpers, we need to add a synthetic reference to resolve the
             // helpers library.
-            if (options.importHelpers
-                && (options.isolatedModules || isExternalModuleFile)
+            if ((options.isolatedModules || isExternalModuleFile)
                 && !file.isDeclarationFile) {
-                // synthesize 'import "tslib"' declaration
-                var externalHelpersModuleReference = ts.createLiteral(ts.externalHelpersModuleNameText);
-                var importDecl = ts.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, /*importClause*/ undefined, externalHelpersModuleReference);
-                ts.addEmitFlags(importDecl, 67108864 /* NeverApplyImportHelper */);
-                externalHelpersModuleReference.parent = importDecl;
-                importDecl.parent = file;
-                imports = [externalHelpersModuleReference];
+                if (options.importHelpers) {
+                    // synthesize 'import "tslib"' declaration
+                    imports = [createSyntheticImport(ts.externalHelpersModuleNameText, file)];
+                }
+                var jsxImport = ts.getJSXRuntimeImport(ts.getJSXImplicitImportBase(options, file), options);
+                if (jsxImport) {
+                    // synthesize `import "base/jsx-runtime"` declaration
+                    (imports || (imports = [])).push(createSyntheticImport(jsxImport, file));
+                }
             }
             for (var _i = 0, _a = file.statements; _i < _a.length; _i++) {
                 var node = _a[_i];
@@ -101265,7 +106603,7 @@ var ts;
                     }
                 }
                 else if (ts.isModuleDeclaration(node)) {
-                    if (ts.isAmbientModule(node) && (inAmbientModule || ts.hasModifier(node, 2 /* Ambient */) || file.isDeclarationFile)) {
+                    if (ts.isAmbientModule(node) && (inAmbientModule || ts.hasSyntacticModifier(node, 2 /* Ambient */) || file.isDeclarationFile)) {
                         var nameText = ts.getTextOfIdentifierOrLiteral(node.name);
                         // Ambient module declarations can be interpreted as augmentations for some existing external modules.
                         // This will happen in two cases:
@@ -101300,7 +106638,7 @@ var ts;
                 var r = /import|require/g;
                 while (r.exec(file.text) !== null) { // eslint-disable-line no-null/no-null
                     var node = getNodeAtPosition(file, r.lastIndex);
-                    if (ts.isRequireCall(node, /*checkArgumentIsStringLiteralLike*/ true)) {
+                    if (isJavaScriptFile && ts.isRequireCall(node, /*checkArgumentIsStringLiteralLike*/ true)) {
                         imports = ts.append(imports, node.arguments[0]);
                     }
                     // we have to check the argument list has length of 1. We will still have to process these even though we have parsing error.
@@ -101425,6 +106763,16 @@ var ts;
         }
         // Get source file from normalized fileName
         function findSourceFile(fileName, path, isDefaultLib, ignoreNoDefaultLib, refFile, packageId) {
+            ts.tracing.push("program" /* Program */, "findSourceFile", {
+                fileName: fileName,
+                isDefaultLib: isDefaultLib || undefined,
+                refKind: refFile ? ts.RefFileKind[refFile.kind] : undefined,
+            });
+            var result = findSourceFileWorker(fileName, path, isDefaultLib, ignoreNoDefaultLib, refFile, packageId);
+            ts.tracing.pop();
+            return result;
+        }
+        function findSourceFileWorker(fileName, path, isDefaultLib, ignoreNoDefaultLib, refFile, packageId) {
             if (useSourceOfProjectReferenceRedirect) {
                 var source = getSourceOfProjectReferenceRedirect(fileName);
                 // If preserveSymlinks is true, module resolution wont jump the symlink
@@ -101495,7 +106843,7 @@ var ts;
             if (refFile && !useSourceOfProjectReferenceRedirect) {
                 var redirectProject = getProjectReferenceRedirectProject(fileName);
                 if (redirectProject) {
-                    if (redirectProject.commandLine.options.outFile || redirectProject.commandLine.options.out) {
+                    if (ts.outFile(redirectProject.commandLine.options)) {
                         // Shouldnt create many to 1 mapping file in --out scenario
                         return undefined;
                     }
@@ -101601,7 +106949,7 @@ var ts;
             return getResolvedProjectReferenceToRedirect(fileName);
         }
         function getProjectReferenceOutputName(referencedProject, fileName) {
-            var out = referencedProject.commandLine.options.outFile || referencedProject.commandLine.options.out;
+            var out = ts.outFile(referencedProject.commandLine.options);
             return out ?
                 ts.changeExtension(out, ".d.ts" /* Dts */) :
                 ts.getOutputDeclarationFileName(fileName, referencedProject.commandLine, !host.useCaseSensitiveFileNames());
@@ -101611,13 +106959,12 @@ var ts;
          */
         function getResolvedProjectReferenceToRedirect(fileName) {
             if (mapFromFileToProjectReferenceRedirects === undefined) {
-                mapFromFileToProjectReferenceRedirects = ts.createMap();
-                forEachResolvedProjectReference(function (referencedProject, referenceProjectPath) {
+                mapFromFileToProjectReferenceRedirects = new ts.Map();
+                forEachResolvedProjectReference(function (referencedProject) {
                     // not input file from the referenced project, ignore
-                    if (referencedProject &&
-                        toPath(options.configFilePath) !== referenceProjectPath) {
+                    if (toPath(options.configFilePath) !== referencedProject.sourceFile.path) {
                         referencedProject.commandLine.fileNames.forEach(function (f) {
-                            return mapFromFileToProjectReferenceRedirects.set(toPath(f), referenceProjectPath);
+                            return mapFromFileToProjectReferenceRedirects.set(toPath(f), referencedProject.sourceFile.path);
                         });
                     }
                 });
@@ -101626,33 +106973,27 @@ var ts;
             return referencedProjectPath && getResolvedProjectReferenceByPath(referencedProjectPath);
         }
         function forEachResolvedProjectReference(cb) {
-            return forEachProjectReference(projectReferences, resolvedProjectReferences, function (resolvedRef, index, parent) {
-                var ref = (parent ? parent.commandLine.projectReferences : projectReferences)[index];
-                var resolvedRefPath = toPath(resolveProjectReferencePath(ref));
-                return cb(resolvedRef, resolvedRefPath);
-            });
+            return ts.forEachResolvedProjectReference(resolvedProjectReferences, cb);
         }
         function getSourceOfProjectReferenceRedirect(file) {
             if (!ts.isDeclarationFileName(file))
                 return undefined;
             if (mapFromToProjectReferenceRedirectSource === undefined) {
-                mapFromToProjectReferenceRedirectSource = ts.createMap();
+                mapFromToProjectReferenceRedirectSource = new ts.Map();
                 forEachResolvedProjectReference(function (resolvedRef) {
-                    if (resolvedRef) {
-                        var out = resolvedRef.commandLine.options.outFile || resolvedRef.commandLine.options.out;
-                        if (out) {
-                            // Dont know which source file it means so return true?
-                            var outputDts = ts.changeExtension(out, ".d.ts" /* Dts */);
-                            mapFromToProjectReferenceRedirectSource.set(toPath(outputDts), true);
-                        }
-                        else {
-                            ts.forEach(resolvedRef.commandLine.fileNames, function (fileName) {
-                                if (!ts.fileExtensionIs(fileName, ".d.ts" /* Dts */) && !ts.fileExtensionIs(fileName, ".json" /* Json */)) {
-                                    var outputDts = ts.getOutputDeclarationFileName(fileName, resolvedRef.commandLine, host.useCaseSensitiveFileNames());
-                                    mapFromToProjectReferenceRedirectSource.set(toPath(outputDts), fileName);
-                                }
-                            });
-                        }
+                    var out = ts.outFile(resolvedRef.commandLine.options);
+                    if (out) {
+                        // Dont know which source file it means so return true?
+                        var outputDts = ts.changeExtension(out, ".d.ts" /* Dts */);
+                        mapFromToProjectReferenceRedirectSource.set(toPath(outputDts), true);
+                    }
+                    else {
+                        ts.forEach(resolvedRef.commandLine.fileNames, function (fileName) {
+                            if (!ts.fileExtensionIs(fileName, ".d.ts" /* Dts */) && !ts.fileExtensionIs(fileName, ".json" /* Json */)) {
+                                var outputDts = ts.getOutputDeclarationFileName(fileName, resolvedRef.commandLine, host.useCaseSensitiveFileNames());
+                                mapFromToProjectReferenceRedirectSource.set(toPath(outputDts), fileName);
+                            }
+                        });
                     }
                 });
             }
@@ -101661,33 +107002,6 @@ var ts;
         function isSourceOfProjectReferenceRedirect(fileName) {
             return useSourceOfProjectReferenceRedirect && !!getResolvedProjectReferenceToRedirect(fileName);
         }
-        function forEachProjectReference(projectReferences, resolvedProjectReferences, cbResolvedRef, cbRef) {
-            var seenResolvedRefs;
-            return worker(projectReferences, resolvedProjectReferences, /*parent*/ undefined, cbResolvedRef, cbRef);
-            function worker(projectReferences, resolvedProjectReferences, parent, cbResolvedRef, cbRef) {
-                // Visit project references first
-                if (cbRef) {
-                    var result = cbRef(projectReferences, parent);
-                    if (result) {
-                        return result;
-                    }
-                }
-                return ts.forEach(resolvedProjectReferences, function (resolvedRef, index) {
-                    if (ts.contains(seenResolvedRefs, resolvedRef)) {
-                        // ignore recursives
-                        return undefined;
-                    }
-                    var result = cbResolvedRef(resolvedRef, index, parent);
-                    if (result) {
-                        return result;
-                    }
-                    if (!resolvedRef)
-                        return undefined;
-                    (seenResolvedRefs || (seenResolvedRefs = [])).push(resolvedRef);
-                    return worker(resolvedRef.commandLine.projectReferences, resolvedRef.references, resolvedRef, cbResolvedRef, cbRef);
-                });
-            }
-        }
         function getResolvedProjectReferenceByPath(projectReferencePath) {
             if (!projectReferenceRedirects) {
                 return undefined;
@@ -101696,7 +107010,7 @@ var ts;
         }
         function processReferencedFiles(file, isDefaultLib) {
             ts.forEach(file.referencedFiles, function (ref, index) {
-                var referencedFileName = resolveTripleslashReference(ref.fileName, file.originalFileName);
+                var referencedFileName = resolveTripleslashReference(ref.fileName, file.fileName);
                 processSourceFile(referencedFileName, isDefaultLib, 
                 /*ignoreNoDefaultLib*/ false, 
                 /*packageId*/ undefined, {
@@ -101714,7 +107028,7 @@ var ts;
             if (!typeDirectives) {
                 return;
             }
-            var resolutions = resolveTypeReferenceDirectiveNamesWorker(typeDirectives, file.originalFileName, getResolvedProjectReferenceToRedirect(file.originalFileName));
+            var resolutions = resolveTypeReferenceDirectiveNamesWorker(typeDirectives, file);
             for (var i = 0; i < typeDirectives.length; i++) {
                 var ref = file.typeReferenceDirectives[i];
                 var resolvedTypeReferenceDirective = resolutions[i];
@@ -101731,6 +107045,11 @@ var ts;
             }
         }
         function processTypeReferenceDirective(typeReferenceDirective, resolvedTypeReferenceDirective, refFile) {
+            ts.tracing.push("program" /* Program */, "processTypeReferenceDirective", { directive: typeReferenceDirective, hasResolved: !!resolveModuleNamesReusingOldState, refKind: refFile === null || refFile === void 0 ? void 0 : refFile.kind, refPath: refFile === null || refFile === void 0 ? void 0 : refFile.file.path });
+            processTypeReferenceDirectiveWorker(typeReferenceDirective, resolvedTypeReferenceDirective, refFile);
+            ts.tracing.pop();
+        }
+        function processTypeReferenceDirectiveWorker(typeReferenceDirective, resolvedTypeReferenceDirective, refFile) {
             // If we already found this library as a primary reference - nothing to do
             var previousResolution = resolvedTypeReferenceDirectives.get(typeReferenceDirective);
             if (previousResolution && previousResolution.primary) {
@@ -101815,7 +107134,7 @@ var ts;
             if (file.imports.length || file.moduleAugmentations.length) {
                 // Because global augmentation doesn't have string literal name, we can check for global augmentation as such.
                 var moduleNames = getModuleNames(file);
-                var resolutions = resolveModuleNamesReusingOldState(moduleNames, ts.getNormalizedAbsolutePath(file.originalFileName, currentDirectory), file);
+                var resolutions = resolveModuleNamesReusingOldState(moduleNames, file);
                 ts.Debug.assert(resolutions.length === moduleNames.length);
                 for (var i = 0; i < moduleNames.length; i++) {
                     var resolution = resolutions[i];
@@ -101843,7 +107162,7 @@ var ts;
                         && !options.noResolve
                         && i < file.imports.length
                         && !elideImport
-                        && !(isJsFile && !options.allowJs)
+                        && !(isJsFile && !ts.getAllowJSCompilerOption(options))
                         && (ts.isInJSFile(file.imports[i]) || !(file.imports[i].flags & 4194304 /* JSDoc */));
                     if (elideImport) {
                         modulesWithElidedImports.set(file.path, true);
@@ -101885,7 +107204,7 @@ var ts;
                     var absoluteSourceFilePath = host.getCanonicalFileName(ts.getNormalizedAbsolutePath(sourceFile.fileName, currentDirectory));
                     if (absoluteSourceFilePath.indexOf(absoluteRootDirectoryPath) !== 0) {
                         if (!rootPaths)
-                            rootPaths = ts.arrayToSet(rootNames, toPath);
+                            rootPaths = new ts.Set(rootNames.map(toPath));
                         addProgramDiagnosticAtRefPath(sourceFile, rootPaths, ts.Diagnostics.File_0_is_not_under_rootDir_1_rootDir_is_expected_to_contain_all_source_files, sourceFile.fileName, rootDirectory);
                         allFilesBelongToPath = false;
                     }
@@ -101895,7 +107214,7 @@ var ts;
         }
         function parseProjectReferenceConfigFile(ref) {
             if (!projectReferenceRedirects) {
-                projectReferenceRedirects = ts.createMap();
+                projectReferenceRedirects = new ts.Map();
             }
             // The actual filename (i.e. add "/tsconfig.json" if necessary)
             var refPath = resolveProjectReferencePath(ref);
@@ -101959,9 +107278,6 @@ var ts;
                     createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_with_option_1, "mapRoot", "inlineSourceMap");
                 }
             }
-            if (options.paths && options.baseUrl === undefined) {
-                createDiagnosticForOptionName(ts.Diagnostics.Option_paths_cannot_be_used_without_specifying_baseUrl_option, "paths");
-            }
             if (options.composite) {
                 if (options.declaration === false) {
                     createDiagnosticForOptionName(ts.Diagnostics.Composite_projects_may_not_disable_declaration_emit, "declaration");
@@ -101970,21 +107286,19 @@ var ts;
                     createDiagnosticForOptionName(ts.Diagnostics.Composite_projects_may_not_disable_incremental_compilation, "declaration");
                 }
             }
+            var outputFile = ts.outFile(options);
             if (options.tsBuildInfoFile) {
                 if (!ts.isIncrementalCompilation(options)) {
                     createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "tsBuildInfoFile", "incremental", "composite");
                 }
             }
-            else if (options.incremental && !options.outFile && !options.out && !options.configFilePath) {
+            else if (options.incremental && !outputFile && !options.configFilePath) {
                 programDiagnostics.add(ts.createCompilerDiagnostic(ts.Diagnostics.Option_incremental_can_only_be_specified_using_tsconfig_emitting_to_single_file_or_when_option_tsBuildInfoFile_is_specified));
             }
-            if (!options.listFilesOnly && options.noEmit && ts.isIncrementalCompilation(options)) {
-                createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_with_option_1, "noEmit", options.incremental ? "incremental" : "composite");
-            }
             verifyProjectReferences();
             // List of collected files is complete; validate exhautiveness if this is a project with a file list
             if (options.composite) {
-                var rootPaths = ts.arrayToSet(rootNames, toPath);
+                var rootPaths = new ts.Set(rootNames.map(toPath));
                 for (var _i = 0, files_3 = files; _i < files_3.length; _i++) {
                     var file = files_3[_i];
                     // Ignore file that is not emitted
@@ -102013,6 +107327,9 @@ var ts;
                                 if (!ts.hasZeroOrOneAsteriskCharacter(subst)) {
                                     createDiagnosticForOptionPathKeyValue(key, i, ts.Diagnostics.Substitution_0_in_pattern_1_can_have_at_most_one_Asterisk_character, subst, key);
                                 }
+                                if (!options.baseUrl && !ts.pathIsRelative(subst) && !ts.pathIsAbsolute(subst)) {
+                                    createDiagnosticForOptionPathKeyValue(key, i, ts.Diagnostics.Non_relative_paths_are_not_allowed_when_baseUrl_is_not_set_Did_you_forget_a_leading_Slash);
+                                }
                             }
                             else {
                                 createDiagnosticForOptionPathKeyValue(key, i, ts.Diagnostics.Substitution_0_for_pattern_1_has_incorrect_type_expected_string_got_2, subst, key, typeOfSubst);
@@ -102043,7 +107360,7 @@ var ts;
                 if (!ts.getEmitDeclarations(options)) {
                     createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "declarationDir", "declaration", "composite");
                 }
-                if (options.out || options.outFile) {
+                if (outputFile) {
                     createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_with_option_1, "declarationDir", options.out ? "out" : "outFile");
                 }
             }
@@ -102057,7 +107374,6 @@ var ts;
                 createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_with_option_1, "noImplicitUseStrict", "alwaysStrict");
             }
             var languageVersion = options.target || 0 /* ES3 */;
-            var outFile = options.outFile || options.out;
             var firstNonAmbientExternalModuleSourceFile = ts.find(files, function (f) { return ts.isExternalModule(f) && !f.isDeclarationFile; });
             if (options.isolatedModules) {
                 if (options.module === ts.ModuleKind.None && languageVersion < 2 /* ES2015 */) {
@@ -102066,7 +107382,7 @@ var ts;
                 var firstNonExternalModuleSourceFile = ts.find(files, function (f) { return !ts.isExternalModule(f) && !ts.isSourceFileJS(f) && !f.isDeclarationFile && f.scriptKind !== 6 /* JSON */; });
                 if (firstNonExternalModuleSourceFile) {
                     var span = ts.getErrorSpanForNode(firstNonExternalModuleSourceFile, firstNonExternalModuleSourceFile);
-                    programDiagnostics.add(ts.createFileDiagnostic(firstNonExternalModuleSourceFile, span.start, span.length, ts.Diagnostics.All_files_must_be_modules_when_the_isolatedModules_flag_is_provided));
+                    programDiagnostics.add(ts.createFileDiagnostic(firstNonExternalModuleSourceFile, span.start, span.length, ts.Diagnostics._0_cannot_be_compiled_under_isolatedModules_because_it_is_considered_a_global_script_file_Add_an_import_export_or_an_empty_export_statement_to_make_it_a_module, ts.getBaseFileName(firstNonExternalModuleSourceFile.fileName)));
                 }
             }
             else if (firstNonAmbientExternalModuleSourceFile && languageVersion < 2 /* ES2015 */ && options.module === ts.ModuleKind.None) {
@@ -102075,7 +107391,7 @@ var ts;
                 programDiagnostics.add(ts.createFileDiagnostic(firstNonAmbientExternalModuleSourceFile, span.start, span.length, ts.Diagnostics.Cannot_use_imports_exports_or_module_augmentations_when_module_is_none));
             }
             // Cannot specify module gen that isn't amd or system with --out
-            if (outFile && !options.emitDeclarationOnly) {
+            if (outputFile && !options.emitDeclarationOnly) {
                 if (options.module && !(options.module === ts.ModuleKind.AMD || options.module === ts.ModuleKind.System)) {
                     createDiagnosticForOptionName(ts.Diagnostics.Only_amd_and_system_modules_are_supported_alongside_0, options.out ? "out" : "outFile", "module");
                 }
@@ -102108,7 +107424,7 @@ var ts;
             if (options.useDefineForClassFields && languageVersion === 0 /* ES3 */) {
                 createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_when_option_target_is_ES3, "useDefineForClassFields");
             }
-            if (options.checkJs && !options.allowJs) {
+            if (options.checkJs && !ts.getAllowJSCompilerOption(options)) {
                 programDiagnostics.add(ts.createCompilerDiagnostic(ts.Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "checkJs", "allowJs"));
             }
             if (options.emitDeclarationOnly) {
@@ -102127,6 +107443,9 @@ var ts;
                 if (options.reactNamespace) {
                     createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_with_option_1, "reactNamespace", "jsxFactory");
                 }
+                if (options.jsx === 4 /* ReactJSX */ || options.jsx === 5 /* ReactJSXDev */) {
+                    createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "jsxFactory", ts.inverseJsxOptionMap.get("" + options.jsx));
+                }
                 if (!ts.parseIsolatedEntityName(options.jsxFactory, languageVersion)) {
                     createOptionValueDiagnostic("jsxFactory", ts.Diagnostics.Invalid_value_for_jsxFactory_0_is_not_a_valid_identifier_or_qualified_name, options.jsxFactory);
                 }
@@ -102134,10 +107453,31 @@ var ts;
             else if (options.reactNamespace && !ts.isIdentifierText(options.reactNamespace, languageVersion)) {
                 createOptionValueDiagnostic("reactNamespace", ts.Diagnostics.Invalid_value_for_reactNamespace_0_is_not_a_valid_identifier, options.reactNamespace);
             }
+            if (options.jsxFragmentFactory) {
+                if (!options.jsxFactory) {
+                    createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "jsxFragmentFactory", "jsxFactory");
+                }
+                if (options.jsx === 4 /* ReactJSX */ || options.jsx === 5 /* ReactJSXDev */) {
+                    createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "jsxFragmentFactory", ts.inverseJsxOptionMap.get("" + options.jsx));
+                }
+                if (!ts.parseIsolatedEntityName(options.jsxFragmentFactory, languageVersion)) {
+                    createOptionValueDiagnostic("jsxFragmentFactory", ts.Diagnostics.Invalid_value_for_jsxFragmentFactory_0_is_not_a_valid_identifier_or_qualified_name, options.jsxFragmentFactory);
+                }
+            }
+            if (options.reactNamespace) {
+                if (options.jsx === 4 /* ReactJSX */ || options.jsx === 5 /* ReactJSXDev */) {
+                    createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "reactNamespace", ts.inverseJsxOptionMap.get("" + options.jsx));
+                }
+            }
+            if (options.jsxImportSource) {
+                if (options.jsx === 2 /* React */) {
+                    createDiagnosticForOptionName(ts.Diagnostics.Option_0_cannot_be_specified_when_option_jsx_is_1, "jsxImportSource", ts.inverseJsxOptionMap.get("" + options.jsx));
+                }
+            }
             // If the emit is enabled make sure that every output file is unique and not overwriting any of the input files
             if (!options.noEmit && !options.suppressOutputPathCheck) {
                 var emitHost = getEmitHost();
-                var emitFilesSeen_1 = ts.createMap();
+                var emitFilesSeen_1 = new ts.Set();
                 ts.forEachEmittedFile(emitHost, function (emitFileNames) {
                     if (!options.emitDeclarationOnly) {
                         verifyEmitFilePath(emitFileNames.jsFilePath, emitFilesSeen_1);
@@ -102166,7 +107506,7 @@ var ts;
                         blockEmittingOfFile(emitFileName, ts.createCompilerDiagnostic(ts.Diagnostics.Cannot_write_file_0_because_it_would_be_overwritten_by_multiple_input_files, emitFileName));
                     }
                     else {
-                        emitFilesSeen.set(emitFileKey, true);
+                        emitFilesSeen.add(emitFileKey);
                     }
                 }
             }
@@ -102201,14 +107541,14 @@ var ts;
             for (var _i = 3; _i < arguments.length; _i++) {
                 args[_i - 3] = arguments[_i];
             }
-            var refPaths = refFileMap && refFileMap.get(file.path);
+            var refPaths = refFileMap === null || refFileMap === void 0 ? void 0 : refFileMap.get(file.path);
             var refPathToReportErrorOn = ts.forEach(refPaths, function (refPath) { return rootPaths.has(refPath.file) ? refPath : undefined; }) ||
                 ts.elementAt(refPaths, 0);
             programDiagnostics.add(refPathToReportErrorOn ? createFileDiagnosticAtReference.apply(void 0, __spreadArrays([refPathToReportErrorOn, message], args)) : ts.createCompilerDiagnostic.apply(void 0, __spreadArrays([message], args)));
         }
         function verifyProjectReferences() {
-            var buildInfoPath = !options.noEmit && !options.suppressOutputPathCheck ? ts.getTsBuildInfoEmitOutputFilePath(options) : undefined;
-            forEachProjectReference(projectReferences, resolvedProjectReferences, function (resolvedRef, index, parent) {
+            var buildInfoPath = !options.suppressOutputPathCheck ? ts.getTsBuildInfoEmitOutputFilePath(options) : undefined;
+            forEachProjectReference(projectReferences, resolvedProjectReferences, function (resolvedRef, parent, index) {
                 var ref = (parent ? parent.commandLine.projectReferences : projectReferences)[index];
                 var parentFile = parent && parent.sourceFile;
                 if (!resolvedRef) {
@@ -102216,15 +107556,18 @@ var ts;
                     return;
                 }
                 var options = resolvedRef.commandLine.options;
-                if (!options.composite) {
+                if (!options.composite || options.noEmit) {
                     // ok to not have composite if the current program is container only
                     var inputs = parent ? parent.commandLine.fileNames : rootNames;
                     if (inputs.length) {
-                        createDiagnosticForReference(parentFile, index, ts.Diagnostics.Referenced_project_0_must_have_setting_composite_Colon_true, ref.path);
+                        if (!options.composite)
+                            createDiagnosticForReference(parentFile, index, ts.Diagnostics.Referenced_project_0_must_have_setting_composite_Colon_true, ref.path);
+                        if (options.noEmit)
+                            createDiagnosticForReference(parentFile, index, ts.Diagnostics.Referenced_project_0_may_not_disable_emit, ref.path);
                     }
                 }
                 if (ref.prepend) {
-                    var out = options.outFile || options.out;
+                    var out = ts.outFile(options);
                     if (out) {
                         if (!host.fileExists(out)) {
                             createDiagnosticForReference(parentFile, index, ts.Diagnostics.Output_file_0_from_project_1_does_not_exist, out, ref.path);
@@ -102345,7 +107688,7 @@ var ts;
                 return false;
             }
             // If options have --outFile or --out just check that
-            var out = options.outFile || options.out;
+            var out = ts.outFile(options);
             if (out) {
                 return isSameFile(filePath, out) || isSameFile(filePath, ts.removeFileExtension(out) + ".d.ts" /* Dts */);
             }
@@ -102368,18 +107711,16 @@ var ts;
         function isSameFile(file1, file2) {
             return ts.comparePaths(file1, file2, currentDirectory, !host.useCaseSensitiveFileNames()) === 0 /* EqualTo */;
         }
-        function getProbableSymlinks() {
-            if (host.getSymlinks) {
-                return host.getSymlinks();
+        function getSymlinkCache() {
+            if (host.getSymlinkCache) {
+                return host.getSymlinkCache();
             }
             return symlinks || (symlinks = ts.discoverProbableSymlinks(files, getCanonicalFileName, host.getCurrentDirectory()));
         }
     }
     ts.createProgram = createProgram;
     function updateHostForUseSourceOfProjectReferenceRedirect(host) {
-        var mapOfDeclarationDirectories;
-        var symlinkedDirectories;
-        var symlinkedFiles;
+        var setOfDeclarationDirectories;
         var originalFileExists = host.compilerHost.fileExists;
         var originalDirectoryExists = host.compilerHost.directoryExists;
         var originalGetDirectories = host.compilerHost.getDirectories;
@@ -102387,31 +107728,30 @@ var ts;
         if (!host.useSourceOfProjectReferenceRedirect)
             return { onProgramCreateComplete: ts.noop, fileExists: fileExists };
         host.compilerHost.fileExists = fileExists;
+        var directoryExists;
         if (originalDirectoryExists) {
             // This implementation of directoryExists checks if the directory being requested is
             // directory of .d.ts file for the referenced Project.
             // If it is it returns true irrespective of whether that directory exists on host
-            host.compilerHost.directoryExists = function (path) {
+            directoryExists = host.compilerHost.directoryExists = function (path) {
                 if (originalDirectoryExists.call(host.compilerHost, path)) {
                     handleDirectoryCouldBeSymlink(path);
                     return true;
                 }
                 if (!host.getResolvedProjectReferences())
                     return false;
-                if (!mapOfDeclarationDirectories) {
-                    mapOfDeclarationDirectories = ts.createMap();
+                if (!setOfDeclarationDirectories) {
+                    setOfDeclarationDirectories = new ts.Set();
                     host.forEachResolvedProjectReference(function (ref) {
-                        if (!ref)
-                            return;
-                        var out = ref.commandLine.options.outFile || ref.commandLine.options.out;
+                        var out = ts.outFile(ref.commandLine.options);
                         if (out) {
-                            mapOfDeclarationDirectories.set(ts.getDirectoryPath(host.toPath(out)), true);
+                            setOfDeclarationDirectories.add(ts.getDirectoryPath(host.toPath(out)));
                         }
                         else {
                             // Set declaration's in different locations only, if they are next to source the directory present doesnt change
                             var declarationDir = ref.commandLine.options.declarationDir || ref.commandLine.options.outDir;
                             if (declarationDir) {
-                                mapOfDeclarationDirectories.set(host.toPath(declarationDir), true);
+                                setOfDeclarationDirectories.add(host.toPath(declarationDir));
                             }
                         }
                     });
@@ -102431,11 +107771,12 @@ var ts;
         // This is something we keep for life time of the host
         if (originalRealpath) {
             host.compilerHost.realpath = function (s) {
-                return (symlinkedFiles === null || symlinkedFiles === void 0 ? void 0 : symlinkedFiles.get(host.toPath(s))) ||
+                var _a;
+                return ((_a = host.getSymlinkCache().getSymlinkedFiles()) === null || _a === void 0 ? void 0 : _a.get(host.toPath(s))) ||
                     originalRealpath.call(host.compilerHost, s);
             };
         }
-        return { onProgramCreateComplete: onProgramCreateComplete, fileExists: fileExists };
+        return { onProgramCreateComplete: onProgramCreateComplete, fileExists: fileExists, directoryExists: directoryExists };
         function onProgramCreateComplete() {
             host.compilerHost.fileExists = originalFileExists;
             host.compilerHost.directoryExists = originalDirectoryExists;
@@ -102464,37 +107805,38 @@ var ts;
         function directoryExistsIfProjectReferenceDeclDir(dir) {
             var dirPath = host.toPath(dir);
             var dirPathWithTrailingDirectorySeparator = "" + dirPath + ts.directorySeparator;
-            return ts.forEachKey(mapOfDeclarationDirectories, function (declDirPath) { return dirPath === declDirPath ||
+            return ts.forEachKey(setOfDeclarationDirectories, function (declDirPath) { return dirPath === declDirPath ||
                 // Any parent directory of declaration dir
                 ts.startsWith(declDirPath, dirPathWithTrailingDirectorySeparator) ||
                 // Any directory inside declaration dir
                 ts.startsWith(dirPath, declDirPath + "/"); });
         }
         function handleDirectoryCouldBeSymlink(directory) {
+            var _a;
             if (!host.getResolvedProjectReferences())
                 return;
             // Because we already watch node_modules, handle symlinks in there
             if (!originalRealpath || !ts.stringContains(directory, ts.nodeModulesPathPart))
                 return;
-            if (!symlinkedDirectories)
-                symlinkedDirectories = ts.createMap();
+            var symlinkCache = host.getSymlinkCache();
             var directoryPath = ts.ensureTrailingDirectorySeparator(host.toPath(directory));
-            if (symlinkedDirectories.has(directoryPath))
+            if ((_a = symlinkCache.getSymlinkedDirectories()) === null || _a === void 0 ? void 0 : _a.has(directoryPath))
                 return;
             var real = ts.normalizePath(originalRealpath.call(host.compilerHost, directory));
             var realPath;
             if (real === directory ||
                 (realPath = ts.ensureTrailingDirectorySeparator(host.toPath(real))) === directoryPath) {
                 // not symlinked
-                symlinkedDirectories.set(directoryPath, false);
+                symlinkCache.setSymlinkedDirectory(directoryPath, false);
                 return;
             }
-            symlinkedDirectories.set(directoryPath, {
+            symlinkCache.setSymlinkedDirectory(directoryPath, {
                 real: ts.ensureTrailingDirectorySeparator(real),
                 realPath: realPath
             });
         }
         function fileOrDirectoryExistsUsingSource(fileOrDirectory, isFile) {
+            var _a;
             var fileOrDirectoryExistsUsingSource = isFile ?
                 function (file) { return fileExistsIfProjectReferenceDts(file); } :
                 function (dir) { return directoryExistsIfProjectReferenceDeclDir(dir); };
@@ -102502,12 +107844,14 @@ var ts;
             var result = fileOrDirectoryExistsUsingSource(fileOrDirectory);
             if (result !== undefined)
                 return result;
+            var symlinkCache = host.getSymlinkCache();
+            var symlinkedDirectories = symlinkCache.getSymlinkedDirectories();
             if (!symlinkedDirectories)
                 return false;
             var fileOrDirectoryPath = host.toPath(fileOrDirectory);
             if (!ts.stringContains(fileOrDirectoryPath, ts.nodeModulesPathPart))
                 return false;
-            if (isFile && symlinkedFiles && symlinkedFiles.has(fileOrDirectoryPath))
+            if (isFile && ((_a = symlinkCache.getSymlinkedFiles()) === null || _a === void 0 ? void 0 : _a.has(fileOrDirectoryPath)))
                 return true;
             // If it contains node_modules check if its one of the symlinked path we know of
             return ts.firstDefinedIterator(symlinkedDirectories.entries(), function (_a) {
@@ -102516,21 +107860,25 @@ var ts;
                     return undefined;
                 var result = fileOrDirectoryExistsUsingSource(fileOrDirectoryPath.replace(directoryPath, symlinkedDirectory.realPath));
                 if (isFile && result) {
-                    if (!symlinkedFiles)
-                        symlinkedFiles = ts.createMap();
                     // Store the real path for the file'
                     var absolutePath = ts.getNormalizedAbsolutePath(fileOrDirectory, host.compilerHost.getCurrentDirectory());
-                    symlinkedFiles.set(fileOrDirectoryPath, "" + symlinkedDirectory.real + absolutePath.replace(new RegExp(directoryPath, "i"), ""));
+                    symlinkCache.setSymlinkedFile(fileOrDirectoryPath, "" + symlinkedDirectory.real + absolutePath.replace(new RegExp(directoryPath, "i"), ""));
                 }
                 return result;
             }) || false;
         }
     }
     /*@internal*/
-    function handleNoEmitOptions(program, sourceFile, cancellationToken) {
+    ts.emitSkippedWithNoDiagnostics = { diagnostics: ts.emptyArray, sourceMaps: undefined, emittedFiles: undefined, emitSkipped: true };
+    /*@internal*/
+    function handleNoEmitOptions(program, sourceFile, writeFile, cancellationToken) {
         var options = program.getCompilerOptions();
         if (options.noEmit) {
-            return { diagnostics: ts.emptyArray, sourceMaps: undefined, emittedFiles: undefined, emitSkipped: true };
+            // Cache the semantic diagnostics
+            program.getSemanticDiagnostics(sourceFile, cancellationToken);
+            return sourceFile || ts.outFile(options) ?
+                ts.emitSkippedWithNoDiagnostics :
+                program.emitBuildInfo(writeFile, cancellationToken);
         }
         // If the noEmitOnError flag is set, then check if we have any errors so far.  If so,
         // immediately bail out.  Note that we pass 'undefined' for 'sourceFile' so that we
@@ -102541,11 +107889,23 @@ var ts;
         if (diagnostics.length === 0 && ts.getEmitDeclarations(program.getCompilerOptions())) {
             diagnostics = program.getDeclarationDiagnostics(/*sourceFile*/ undefined, cancellationToken);
         }
-        return diagnostics.length > 0 ?
-            { diagnostics: diagnostics, sourceMaps: undefined, emittedFiles: undefined, emitSkipped: true } :
-            undefined;
+        if (!diagnostics.length)
+            return undefined;
+        var emittedFiles;
+        if (!sourceFile && !ts.outFile(options)) {
+            var emitResult = program.emitBuildInfo(writeFile, cancellationToken);
+            if (emitResult.diagnostics)
+                diagnostics = __spreadArrays(diagnostics, emitResult.diagnostics);
+            emittedFiles = emitResult.emittedFiles;
+        }
+        return { diagnostics: diagnostics, sourceMaps: undefined, emittedFiles: emittedFiles, emitSkipped: true };
     }
     ts.handleNoEmitOptions = handleNoEmitOptions;
+    /*@internal*/
+    function filterSemanticDiagnotics(diagnostic, option) {
+        return ts.filter(diagnostic, function (d) { return !d.skippedOn || !option[d.skippedOn]; });
+    }
+    ts.filterSemanticDiagnotics = filterSemanticDiagnotics;
     /* @internal */
     function parseConfigHostFromCompilerHostLike(host, directoryStructureHost) {
         if (directoryStructureHost === void 0) { directoryStructureHost = host; }
@@ -102572,7 +107932,7 @@ var ts;
             var ref = projectReferences[i];
             var resolvedRefOpts = getCommandLine(ref, i);
             if (ref.prepend && resolvedRefOpts && resolvedRefOpts.options) {
-                var out = resolvedRefOpts.options.outFile || resolvedRefOpts.options.out;
+                var out = ts.outFile(resolvedRefOpts.options);
                 // Upstream project didn't have outFile set -- skip (error will have been issued earlier)
                 if (!out)
                     continue;
@@ -102615,7 +107975,7 @@ var ts;
             return options.jsx ? undefined : ts.Diagnostics.Module_0_was_resolved_to_1_but_jsx_is_not_set;
         }
         function needAllowJs() {
-            return options.allowJs || !ts.getStrictOptionValue(options, "noImplicitAny") ? undefined : ts.Diagnostics.Could_not_find_a_declaration_file_for_module_0_1_implicitly_has_an_any_type;
+            return ts.getAllowJSCompilerOption(options) || !ts.getStrictOptionValue(options, "noImplicitAny") ? undefined : ts.Diagnostics.Could_not_find_a_declaration_file_for_module_0_1_implicitly_has_an_any_type;
         }
         function needResolveJsonModule() {
             return options.resolveJsonModule ? undefined : ts.Diagnostics.Module_0_was_resolved_to_1_but_resolveJsonModule_is_not_used;
@@ -102745,10 +108105,7 @@ var ts;
                 }
             }
             function addReferencedFile(referencedPath) {
-                if (!referencedFiles) {
-                    referencedFiles = ts.createMap();
-                }
-                referencedFiles.set(referencedPath, true);
+                (referencedFiles || (referencedFiles = new ts.Set())).add(referencedPath);
             }
         }
         /**
@@ -102762,15 +108119,17 @@ var ts;
          * Creates the state of file references and signature for the new program from oldState if it is safe
          */
         function create(newProgram, getCanonicalFileName, oldState) {
-            var fileInfos = ts.createMap();
-            var referencedMap = newProgram.getCompilerOptions().module !== ts.ModuleKind.None ? ts.createMap() : undefined;
-            var exportedModulesMap = referencedMap ? ts.createMap() : undefined;
-            var hasCalledUpdateShapeSignature = ts.createMap();
+            var fileInfos = new ts.Map();
+            var referencedMap = newProgram.getCompilerOptions().module !== ts.ModuleKind.None ? new ts.Map() : undefined;
+            var exportedModulesMap = referencedMap ? new ts.Map() : undefined;
+            var hasCalledUpdateShapeSignature = new ts.Set();
             var useOldState = canReuseOldState(referencedMap, oldState);
+            // Ensure source files have parent pointers set
+            newProgram.getTypeChecker();
             // Create the reference map, and set the file infos
             for (var _i = 0, _a = newProgram.getSourceFiles(); _i < _a.length; _i++) {
                 var sourceFile = _a[_i];
-                var version_1 = ts.Debug.checkDefined(sourceFile.version, "Program intended to be used with Builder should have source files with versions set");
+                var version_2 = ts.Debug.checkDefined(sourceFile.version, "Program intended to be used with Builder should have source files with versions set");
                 var oldInfo = useOldState ? oldState.fileInfos.get(sourceFile.resolvedPath) : undefined;
                 if (referencedMap) {
                     var newReferences = getReferencedFiles(newProgram, sourceFile, getCanonicalFileName);
@@ -102785,7 +108144,7 @@ var ts;
                         }
                     }
                 }
-                fileInfos.set(sourceFile.resolvedPath, { version: version_1, signature: oldInfo && oldInfo.signature, affectsGlobalScope: isFileAffectingGlobalScope(sourceFile) });
+                fileInfos.set(sourceFile.resolvedPath, { version: version_2, signature: oldInfo && oldInfo.signature, affectsGlobalScope: isFileAffectingGlobalScope(sourceFile) });
             }
             return {
                 fileInfos: fileInfos,
@@ -102807,16 +108166,12 @@ var ts;
          * Creates a clone of the state
          */
         function clone(state) {
-            var fileInfos = ts.createMap();
-            state.fileInfos.forEach(function (value, key) {
-                fileInfos.set(key, __assign({}, value));
-            });
             // Dont need to backup allFiles info since its cache anyway
             return {
-                fileInfos: fileInfos,
-                referencedMap: cloneMapOrUndefined(state.referencedMap),
-                exportedModulesMap: cloneMapOrUndefined(state.exportedModulesMap),
-                hasCalledUpdateShapeSignature: ts.cloneMap(state.hasCalledUpdateShapeSignature),
+                fileInfos: new ts.Map(state.fileInfos),
+                referencedMap: state.referencedMap && new ts.Map(state.referencedMap),
+                exportedModulesMap: state.exportedModulesMap && new ts.Map(state.exportedModulesMap),
+                hasCalledUpdateShapeSignature: new ts.Set(state.hasCalledUpdateShapeSignature),
             };
         }
         BuilderState.clone = clone;
@@ -102828,7 +108183,7 @@ var ts;
             // They will be committed once it is safe to use them
             // eg when calling this api from tsserver, if there is no cancellation of the operation
             // In the other cases the affected files signatures are committed only after the iteration through the result is complete
-            var signatureCache = cacheToUpdateSignature || ts.createMap();
+            var signatureCache = cacheToUpdateSignature || new ts.Map();
             var sourceFile = programOfThisState.getSourceFileByPath(path);
             if (!sourceFile) {
                 return ts.emptyArray;
@@ -102854,7 +108209,7 @@ var ts;
         BuilderState.updateSignaturesFromCache = updateSignaturesFromCache;
         function updateSignatureOfFile(state, signature, path) {
             state.fileInfos.get(path).signature = signature;
-            state.hasCalledUpdateShapeSignature.set(path, true);
+            state.hasCalledUpdateShapeSignature.add(path);
         }
         BuilderState.updateSignatureOfFile = updateSignatureOfFile;
         /**
@@ -102891,7 +108246,7 @@ var ts;
                     emitOutput_1.outputFiles.length > 0 ? emitOutput_1.outputFiles[0] : undefined;
                 if (firstDts_1) {
                     ts.Debug.assert(ts.fileExtensionIs(firstDts_1.name, ".d.ts" /* Dts */), "File extension for signature expected to be dts", function () { return "Found: " + ts.getAnyExtensionFromPath(firstDts_1.name) + " for " + firstDts_1.name + ":: All output files: " + JSON.stringify(emitOutput_1.outputFiles.map(function (f) { return f.name; })); });
-                    latestSignature = computeHash(firstDts_1.text);
+                    latestSignature = (computeHash || ts.generateDjb2Hash)(firstDts_1.text);
                     if (exportedModulesMapCache && latestSignature !== prevSignature) {
                         updateExportedModules(sourceFile, emitOutput_1.exportedModulesFromDeclarationEmit, exportedModulesMapCache);
                     }
@@ -102918,9 +108273,9 @@ var ts;
             function addExportedModule(exportedModulePath) {
                 if (exportedModulePath) {
                     if (!exportedModules) {
-                        exportedModules = ts.createMap();
+                        exportedModules = new ts.Set();
                     }
-                    exportedModules.set(exportedModulePath, true);
+                    exportedModules.add(exportedModulePath);
                 }
             }
         }
@@ -102948,7 +108303,7 @@ var ts;
         function getAllDependencies(state, programOfThisState, sourceFile) {
             var compilerOptions = programOfThisState.getCompilerOptions();
             // With --out or --outFile all outputs go into single file, all files depend on each other
-            if (compilerOptions.outFile || compilerOptions.out) {
+            if (ts.outFile(compilerOptions)) {
                 return getAllFileNames(state, programOfThisState);
             }
             // If this is non module emit, or its a global file, it depends on all the source files
@@ -102956,12 +108311,12 @@ var ts;
                 return getAllFileNames(state, programOfThisState);
             }
             // Get the references, traversing deep from the referenceMap
-            var seenMap = ts.createMap();
+            var seenMap = new ts.Set();
             var queue = [sourceFile.resolvedPath];
             while (queue.length) {
                 var path = queue.pop();
                 if (!seenMap.has(path)) {
-                    seenMap.set(path, true);
+                    seenMap.add(path);
                     var references = state.referencedMap.get(path);
                     if (references) {
                         var iterator = references.keys();
@@ -102971,10 +108326,7 @@ var ts;
                     }
                 }
             }
-            return ts.arrayFrom(ts.mapDefinedIterator(seenMap.keys(), function (path) {
-                var file = programOfThisState.getSourceFileByPath(path);
-                return file ? file.fileName : path;
-            }));
+            return ts.arrayFrom(ts.mapDefinedIterator(seenMap.keys(), function (path) { var _a, _b; return (_b = (_a = programOfThisState.getSourceFileByPath(path)) === null || _a === void 0 ? void 0 : _a.fileName) !== null && _b !== void 0 ? _b : path; }));
         }
         BuilderState.getAllDependencies = getAllDependencies;
         /**
@@ -103059,7 +108411,7 @@ var ts;
             var compilerOptions = programOfThisState.getCompilerOptions();
             // If `--out` or `--outFile` is specified, any new emit will result in re-emitting the entire project,
             // so returning the file itself is good enough.
-            if (compilerOptions && (compilerOptions.out || compilerOptions.outFile)) {
+            if (compilerOptions && ts.outFile(compilerOptions)) {
                 return [sourceFileWithUpdatedShape];
             }
             return getAllFilesExcludingDefaultLibraryFile(state, programOfThisState, sourceFileWithUpdatedShape);
@@ -103072,13 +108424,13 @@ var ts;
                 return getAllFilesExcludingDefaultLibraryFile(state, programOfThisState, sourceFileWithUpdatedShape);
             }
             var compilerOptions = programOfThisState.getCompilerOptions();
-            if (compilerOptions && (compilerOptions.isolatedModules || compilerOptions.out || compilerOptions.outFile)) {
+            if (compilerOptions && (compilerOptions.isolatedModules || ts.outFile(compilerOptions))) {
                 return [sourceFileWithUpdatedShape];
             }
             // Now we need to if each file in the referencedBy list has a shape change as well.
             // Because if so, its own referencedBy files need to be saved as well to make the
             // emitting result consistent with files on disk.
-            var seenFileNamesMap = ts.createMap();
+            var seenFileNamesMap = new ts.Map();
             // Start with the paths this file was referenced by
             seenFileNamesMap.set(sourceFileWithUpdatedShape.resolvedPath, sourceFileWithUpdatedShape);
             var queue = getReferencedByPaths(state, sourceFileWithUpdatedShape.resolvedPath);
@@ -103087,21 +108439,15 @@ var ts;
                 if (!seenFileNamesMap.has(currentPath)) {
                     var currentSourceFile = programOfThisState.getSourceFileByPath(currentPath);
                     seenFileNamesMap.set(currentPath, currentSourceFile);
-                    if (currentSourceFile && updateShapeSignature(state, programOfThisState, currentSourceFile, cacheToUpdateSignature, cancellationToken, computeHash, exportedModulesMapCache)) { // TODO: GH#18217
-                        queue.push.apply(// TODO: GH#18217
-                        queue, getReferencedByPaths(state, currentSourceFile.resolvedPath));
+                    if (currentSourceFile && updateShapeSignature(state, programOfThisState, currentSourceFile, cacheToUpdateSignature, cancellationToken, computeHash, exportedModulesMapCache)) {
+                        queue.push.apply(queue, getReferencedByPaths(state, currentSourceFile.resolvedPath));
                     }
                 }
             }
             // Return array of values that needs emit
-            // Return array of values that needs emit
             return ts.arrayFrom(ts.mapDefinedIterator(seenFileNamesMap.values(), function (value) { return value; }));
         }
     })(BuilderState = ts.BuilderState || (ts.BuilderState = {}));
-    function cloneMapOrUndefined(map) {
-        return map ? ts.cloneMap(map) : undefined;
-    }
-    ts.cloneMapOrUndefined = cloneMapOrUndefined;
 })(ts || (ts = {}));
 /*@internal*/
 var ts;
@@ -103124,10 +108470,10 @@ var ts;
         var compilerOptions = newProgram.getCompilerOptions();
         state.compilerOptions = compilerOptions;
         // With --out or --outFile, any change affects all semantic diagnostics so no need to cache them
-        if (!compilerOptions.outFile && !compilerOptions.out) {
-            state.semanticDiagnosticsPerFile = ts.createMap();
+        if (!ts.outFile(compilerOptions)) {
+            state.semanticDiagnosticsPerFile = new ts.Map();
         }
-        state.changedFilesSet = ts.createMap();
+        state.changedFilesSet = new ts.Set();
         var useOldState = ts.BuilderState.canReuseOldState(state.referencedMap, oldState);
         var oldCompilerOptions = useOldState ? oldState.compilerOptions : undefined;
         var canCopySemanticDiagnostics = useOldState && oldState.semanticDiagnosticsPerFile && !!state.semanticDiagnosticsPerFile &&
@@ -103143,14 +108489,12 @@ var ts;
                 ts.Debug.assert(!changedFilesSet || !ts.forEachKey(changedFilesSet, function (path) { return oldState.semanticDiagnosticsPerFile.has(path); }), "Semantic diagnostics shouldnt be available for changed files");
             }
             // Copy old state's changed files set
-            if (changedFilesSet) {
-                ts.copyEntries(changedFilesSet, state.changedFilesSet);
-            }
-            if (!compilerOptions.outFile && !compilerOptions.out && oldState.affectedFilesPendingEmit) {
+            changedFilesSet === null || changedFilesSet === void 0 ? void 0 : changedFilesSet.forEach(function (value) { return state.changedFilesSet.add(value); });
+            if (!ts.outFile(compilerOptions) && oldState.affectedFilesPendingEmit) {
                 state.affectedFilesPendingEmit = oldState.affectedFilesPendingEmit.slice();
-                state.affectedFilesPendingEmitKind = ts.cloneMapOrUndefined(oldState.affectedFilesPendingEmitKind);
+                state.affectedFilesPendingEmitKind = oldState.affectedFilesPendingEmitKind && new ts.Map(oldState.affectedFilesPendingEmitKind);
                 state.affectedFilesPendingEmitIndex = oldState.affectedFilesPendingEmitIndex;
-                state.seenAffectedFiles = ts.createMap();
+                state.seenAffectedFiles = new ts.Set();
             }
         }
         // Update changed files and copy semantic diagnostics if we can
@@ -103172,7 +108516,7 @@ var ts;
                 // Referenced file was deleted in the new program
                 newReferences && ts.forEachKey(newReferences, function (path) { return !state.fileInfos.has(path) && oldState.fileInfos.has(path); })) {
                 // Register file as changed file and do not copy semantic diagnostics, since all changed files need to be re-evaluated
-                state.changedFilesSet.set(sourceFilePath, true);
+                state.changedFilesSet.add(sourceFilePath);
             }
             else if (canCopySemanticDiagnostics) {
                 var sourceFile = newProgram.getSourceFileByPath(sourceFilePath);
@@ -103187,24 +108531,24 @@ var ts;
                 if (diagnostics) {
                     state.semanticDiagnosticsPerFile.set(sourceFilePath, oldState.hasReusableDiagnostic ? convertToDiagnostics(diagnostics, newProgram, getCanonicalFileName) : diagnostics);
                     if (!state.semanticDiagnosticsFromOldState) {
-                        state.semanticDiagnosticsFromOldState = ts.createMap();
+                        state.semanticDiagnosticsFromOldState = new ts.Set();
                     }
-                    state.semanticDiagnosticsFromOldState.set(sourceFilePath, true);
+                    state.semanticDiagnosticsFromOldState.add(sourceFilePath);
                 }
             }
         });
         // If the global file is removed, add all files as changed
         if (useOldState && ts.forEachEntry(oldState.fileInfos, function (info, sourceFilePath) { return info.affectsGlobalScope && !state.fileInfos.has(sourceFilePath); })) {
             ts.BuilderState.getAllFilesExcludingDefaultLibraryFile(state, newProgram, /*firstSourceFile*/ undefined)
-                .forEach(function (file) { return state.changedFilesSet.set(file.resolvedPath, true); });
+                .forEach(function (file) { return state.changedFilesSet.add(file.resolvedPath); });
         }
-        else if (oldCompilerOptions && ts.compilerOptionsAffectEmit(compilerOptions, oldCompilerOptions)) {
+        else if (oldCompilerOptions && !ts.outFile(compilerOptions) && ts.compilerOptionsAffectEmit(compilerOptions, oldCompilerOptions)) {
             // Add all files to affectedFilesPendingEmit since emit changed
             newProgram.getSourceFiles().forEach(function (f) { return addToAffectedFilesPendingEmit(state, f.resolvedPath, 1 /* Full */); });
             ts.Debug.assert(!state.seenAffectedFiles || !state.seenAffectedFiles.size);
-            state.seenAffectedFiles = state.seenAffectedFiles || ts.createMap();
+            state.seenAffectedFiles = state.seenAffectedFiles || new ts.Set();
         }
-        state.emittedBuildInfo = !state.changedFilesSet.size && !state.affectedFilesPendingEmit;
+        state.buildInfoEmitPending = !!state.changedFilesSet.size;
         return state;
     }
     function convertToDiagnostics(diagnostics, newProgram, getCanonicalFileName) {
@@ -103214,12 +108558,14 @@ var ts;
         return diagnostics.map(function (diagnostic) {
             var result = convertToDiagnosticRelatedInformation(diagnostic, newProgram, toPath);
             result.reportsUnnecessary = diagnostic.reportsUnnecessary;
+            result.reportsDeprecated = diagnostic.reportDeprecated;
             result.source = diagnostic.source;
+            result.skippedOn = diagnostic.skippedOn;
             var relatedInformation = diagnostic.relatedInformation;
             result.relatedInformation = relatedInformation ?
                 relatedInformation.length ?
                     relatedInformation.map(function (r) { return convertToDiagnosticRelatedInformation(r, newProgram, toPath); }) :
-                    ts.emptyArray :
+                    [] :
                 undefined;
             return result;
         });
@@ -103243,22 +108589,22 @@ var ts;
      */
     function cloneBuilderProgramState(state) {
         var newState = ts.BuilderState.clone(state);
-        newState.semanticDiagnosticsPerFile = ts.cloneMapOrUndefined(state.semanticDiagnosticsPerFile);
-        newState.changedFilesSet = ts.cloneMap(state.changedFilesSet);
+        newState.semanticDiagnosticsPerFile = state.semanticDiagnosticsPerFile && new ts.Map(state.semanticDiagnosticsPerFile);
+        newState.changedFilesSet = new ts.Set(state.changedFilesSet);
         newState.affectedFiles = state.affectedFiles;
         newState.affectedFilesIndex = state.affectedFilesIndex;
         newState.currentChangedFilePath = state.currentChangedFilePath;
-        newState.currentAffectedFilesSignatures = ts.cloneMapOrUndefined(state.currentAffectedFilesSignatures);
-        newState.currentAffectedFilesExportedModulesMap = ts.cloneMapOrUndefined(state.currentAffectedFilesExportedModulesMap);
-        newState.seenAffectedFiles = ts.cloneMapOrUndefined(state.seenAffectedFiles);
+        newState.currentAffectedFilesSignatures = state.currentAffectedFilesSignatures && new ts.Map(state.currentAffectedFilesSignatures);
+        newState.currentAffectedFilesExportedModulesMap = state.currentAffectedFilesExportedModulesMap && new ts.Map(state.currentAffectedFilesExportedModulesMap);
+        newState.seenAffectedFiles = state.seenAffectedFiles && new ts.Set(state.seenAffectedFiles);
         newState.cleanedDiagnosticsOfLibFiles = state.cleanedDiagnosticsOfLibFiles;
-        newState.semanticDiagnosticsFromOldState = ts.cloneMapOrUndefined(state.semanticDiagnosticsFromOldState);
+        newState.semanticDiagnosticsFromOldState = state.semanticDiagnosticsFromOldState && new ts.Set(state.semanticDiagnosticsFromOldState);
         newState.program = state.program;
         newState.compilerOptions = state.compilerOptions;
         newState.affectedFilesPendingEmit = state.affectedFilesPendingEmit && state.affectedFilesPendingEmit.slice();
-        newState.affectedFilesPendingEmitKind = ts.cloneMapOrUndefined(state.affectedFilesPendingEmitKind);
+        newState.affectedFilesPendingEmitKind = state.affectedFilesPendingEmitKind && new ts.Map(state.affectedFilesPendingEmitKind);
         newState.affectedFilesPendingEmitIndex = state.affectedFilesPendingEmitIndex;
-        newState.seenEmittedFiles = ts.cloneMapOrUndefined(state.seenEmittedFiles);
+        newState.seenEmittedFiles = state.seenEmittedFiles && new ts.Map(state.seenEmittedFiles);
         newState.programEmitComplete = state.programEmitComplete;
         return newState;
     }
@@ -103309,19 +108655,22 @@ var ts;
             // so operations are performed directly on program, return program
             var program = ts.Debug.checkDefined(state.program);
             var compilerOptions = program.getCompilerOptions();
-            if (compilerOptions.outFile || compilerOptions.out) {
+            if (ts.outFile(compilerOptions)) {
                 ts.Debug.assert(!state.semanticDiagnosticsPerFile);
                 return program;
             }
             // Get next batch of affected files
-            state.currentAffectedFilesSignatures = state.currentAffectedFilesSignatures || ts.createMap();
+            if (!state.currentAffectedFilesSignatures)
+                state.currentAffectedFilesSignatures = new ts.Map();
             if (state.exportedModulesMap) {
-                state.currentAffectedFilesExportedModulesMap = state.currentAffectedFilesExportedModulesMap || ts.createMap();
+                if (!state.currentAffectedFilesExportedModulesMap)
+                    state.currentAffectedFilesExportedModulesMap = new ts.Map();
             }
             state.affectedFiles = ts.BuilderState.getFilesAffectedBy(state, program, nextKey.value, cancellationToken, computeHash, state.currentAffectedFilesSignatures, state.currentAffectedFilesExportedModulesMap);
             state.currentChangedFilePath = nextKey.value;
             state.affectedFilesIndex = 0;
-            state.seenAffectedFiles = state.seenAffectedFiles || ts.createMap();
+            if (!state.seenAffectedFiles)
+                state.seenAffectedFiles = new ts.Set();
         }
     }
     /**
@@ -103330,7 +108679,7 @@ var ts;
     function getNextAffectedFilePendingEmit(state) {
         var affectedFilesPendingEmit = state.affectedFilesPendingEmit;
         if (affectedFilesPendingEmit) {
-            var seenEmittedFiles = state.seenEmittedFiles || (state.seenEmittedFiles = ts.createMap());
+            var seenEmittedFiles = (state.seenEmittedFiles || (state.seenEmittedFiles = new ts.Map()));
             for (var i = state.affectedFilesPendingEmitIndex; i < affectedFilesPendingEmit.length; i++) {
                 var affectedFile = ts.Debug.checkDefined(state.program).getSourceFileByPath(affectedFilesPendingEmit[i]);
                 if (affectedFile) {
@@ -103408,10 +108757,10 @@ var ts;
         state.semanticDiagnosticsPerFile.delete(path);
         return !state.semanticDiagnosticsFromOldState.size;
     }
-    function isChangedSignagure(state, path) {
+    function isChangedSignature(state, path) {
         var newSignature = ts.Debug.checkDefined(state.currentAffectedFilesSignatures).get(path);
-        var oldSignagure = ts.Debug.checkDefined(state.fileInfos.get(path)).signature;
-        return newSignature !== oldSignagure;
+        var oldSignature = ts.Debug.checkDefined(state.fileInfos.get(path)).signature;
+        return newSignature !== oldSignature;
     }
     /**
      * Iterate on referencing modules that export entities from affected file
@@ -103422,12 +108771,12 @@ var ts;
         if (!state.exportedModulesMap || !state.changedFilesSet.has(affectedFile.resolvedPath)) {
             return;
         }
-        if (!isChangedSignagure(state, affectedFile.resolvedPath))
+        if (!isChangedSignature(state, affectedFile.resolvedPath))
             return;
         // Since isolated modules dont change js files, files affected by change in signature is itself
         // But we need to cleanup semantic diagnostics and queue dts emit for affected files
         if (state.compilerOptions.isolatedModules) {
-            var seenFileNamesMap = ts.createMap();
+            var seenFileNamesMap = new ts.Map();
             seenFileNamesMap.set(affectedFile.resolvedPath, true);
             var queue = ts.BuilderState.getReferencedByPaths(state, affectedFile.resolvedPath);
             while (queue.length > 0) {
@@ -103435,7 +108784,7 @@ var ts;
                 if (!seenFileNamesMap.has(currentPath)) {
                     seenFileNamesMap.set(currentPath, true);
                     var result = fn(state, currentPath);
-                    if (result && isChangedSignagure(state, currentPath)) {
+                    if (result && isChangedSignature(state, currentPath)) {
                         var currentSourceFile = ts.Debug.checkDefined(state.program).getSourceFileByPath(currentPath);
                         queue.push.apply(queue, ts.BuilderState.getReferencedByPaths(state, currentSourceFile.resolvedPath));
                     }
@@ -103443,7 +108792,7 @@ var ts;
             }
         }
         ts.Debug.assert(!!state.currentAffectedFilesExportedModulesMap);
-        var seenFileAndExportsOfFile = ts.createMap();
+        var seenFileAndExportsOfFile = new ts.Set();
         // Go through exported modules from cache first
         // If exported modules has path, all files referencing file exported from are affected
         if (ts.forEachEntry(state.currentAffectedFilesExportedModulesMap, function (exportedModules, exportedFromPath) {
@@ -103472,7 +108821,7 @@ var ts;
      * fn on file and iterate on anything that exports this file
      */
     function forEachFileAndExportsOfFile(state, filePath, seenFileAndExportsOfFile, fn) {
-        if (!ts.addToSeen(seenFileAndExportsOfFile, filePath)) {
+        if (!ts.tryAddToSet(seenFileAndExportsOfFile, filePath)) {
             return false;
         }
         if (fn(state, filePath)) {
@@ -103511,19 +108860,20 @@ var ts;
      */
     function doneWithAffectedFile(state, affected, emitKind, isPendingEmit, isBuildInfoEmit) {
         if (isBuildInfoEmit) {
-            state.emittedBuildInfo = true;
+            state.buildInfoEmitPending = false;
         }
         else if (affected === state.program) {
             state.changedFilesSet.clear();
             state.programEmitComplete = true;
         }
         else {
-            state.seenAffectedFiles.set(affected.resolvedPath, true);
+            state.seenAffectedFiles.add(affected.resolvedPath);
             if (emitKind !== undefined) {
-                (state.seenEmittedFiles || (state.seenEmittedFiles = ts.createMap())).set(affected.resolvedPath, emitKind);
+                (state.seenEmittedFiles || (state.seenEmittedFiles = new ts.Map())).set(affected.resolvedPath, emitKind);
             }
             if (isPendingEmit) {
                 state.affectedFilesPendingEmitIndex++;
+                state.buildInfoEmitPending = true;
             }
             else {
                 state.affectedFilesIndex++;
@@ -103561,7 +108911,7 @@ var ts;
             var cachedDiagnostics = state.semanticDiagnosticsPerFile.get(path);
             // Report the bind and check diagnostics from the cache if we already have those diagnostics present
             if (cachedDiagnostics) {
-                return cachedDiagnostics;
+                return ts.filterSemanticDiagnotics(cachedDiagnostics, state.compilerOptions);
             }
         }
         // Diagnostics werent cached, get them from program, and cache the result
@@ -103569,13 +108919,13 @@ var ts;
         if (state.semanticDiagnosticsPerFile) {
             state.semanticDiagnosticsPerFile.set(path, diagnostics);
         }
-        return diagnostics;
+        return ts.filterSemanticDiagnotics(diagnostics, state.compilerOptions);
     }
     /**
      * Gets the program information to be emitted in buildInfo so that we can use it to create new program
      */
     function getProgramBuildInfo(state, getCanonicalFileName) {
-        if (state.compilerOptions.outFile || state.compilerOptions.out)
+        if (ts.outFile(state.compilerOptions))
             return undefined;
         var currentDirectory = ts.Debug.checkDefined(state.program).getCurrentDirectory();
         var buildInfoDirectory = ts.getDirectoryPath(ts.getNormalizedAbsolutePath(ts.getTsBuildInfoEmitOutputFilePath(state.compilerOptions), currentDirectory));
@@ -103626,6 +108976,17 @@ var ts;
             }
             result.semanticDiagnosticsPerFile = semanticDiagnosticsPerFile;
         }
+        if (state.affectedFilesPendingEmit) {
+            var affectedFilesPendingEmit = [];
+            var seenFiles = new ts.Set();
+            for (var _f = 0, _g = state.affectedFilesPendingEmit.slice(state.affectedFilesPendingEmitIndex).sort(ts.compareStringsCaseSensitive); _f < _g.length; _f++) {
+                var path = _g[_f];
+                if (ts.tryAddToSet(seenFiles, path)) {
+                    affectedFilesPendingEmit.push([relativeToBuildInfo(path), state.affectedFilesPendingEmitKind.get(path)]);
+                }
+            }
+            result.affectedFilesPendingEmit = affectedFilesPendingEmit;
+        }
         return result;
         function relativeToBuildInfoEnsuringAbsolutePath(path) {
             return relativeToBuildInfo(ts.getNormalizedAbsolutePath(path, currentDirectory));
@@ -103666,12 +109027,14 @@ var ts;
         return diagnostics.map(function (diagnostic) {
             var result = convertToReusableDiagnosticRelatedInformation(diagnostic, relativeToBuildInfo);
             result.reportsUnnecessary = diagnostic.reportsUnnecessary;
+            result.reportDeprecated = diagnostic.reportsDeprecated;
             result.source = diagnostic.source;
+            result.skippedOn = diagnostic.skippedOn;
             var relatedInformation = diagnostic.relatedInformation;
             result.relatedInformation = relatedInformation ?
                 relatedInformation.length ?
                     relatedInformation.map(function (r) { return convertToReusableDiagnosticRelatedInformation(r, relativeToBuildInfo); }) :
-                    ts.emptyArray :
+                    [] :
                 undefined;
             return result;
         });
@@ -103733,7 +109096,7 @@ var ts;
         /**
          * Computing hash to for signature verification
          */
-        var computeHash = host.createHash || ts.generateDjb2Hash;
+        var computeHash = ts.maybeBind(host, host.createHash);
         var state = createBuilderProgramState(newProgram, getCanonicalFileName, oldState);
         var backupState;
         newProgram.getProgramBuildInfo = function () { return getProgramBuildInfo(state, getCanonicalFileName); };
@@ -103764,11 +109127,20 @@ var ts;
         else if (kind === BuilderProgramKind.EmitAndSemanticDiagnosticsBuilderProgram) {
             builderProgram.getSemanticDiagnosticsOfNextAffectedFile = getSemanticDiagnosticsOfNextAffectedFile;
             builderProgram.emitNextAffectedFile = emitNextAffectedFile;
+            builderProgram.emitBuildInfo = emitBuildInfo;
         }
         else {
             ts.notImplemented();
         }
         return builderProgram;
+        function emitBuildInfo(writeFile, cancellationToken) {
+            if (state.buildInfoEmitPending) {
+                var result = ts.Debug.checkDefined(state.program).emitBuildInfo(writeFile || ts.maybeBind(host, host.writeFile), cancellationToken);
+                state.buildInfoEmitPending = false;
+                return result;
+            }
+            return ts.emitSkippedWithNoDiagnostics;
+        }
         /**
          * Emits the next affected file's emit result (EmitResult and sourceFiles emitted) or returns undefined if iteration is complete
          * The first of writeFile if provided, writeFile of BuilderProgramHost if provided, writeFile of compiler host
@@ -103779,10 +109151,10 @@ var ts;
             var emitKind = 1 /* Full */;
             var isPendingEmitFile = false;
             if (!affected) {
-                if (!state.compilerOptions.out && !state.compilerOptions.outFile) {
+                if (!ts.outFile(state.compilerOptions)) {
                     var pendingAffectedFile = getNextAffectedFilePendingEmit(state);
                     if (!pendingAffectedFile) {
-                        if (state.emittedBuildInfo) {
+                        if (!state.buildInfoEmitPending) {
                             return undefined;
                         }
                         var affected_1 = ts.Debug.checkDefined(state.program);
@@ -103820,31 +109192,54 @@ var ts;
          * in that order would be used to write the files
          */
         function emit(targetSourceFile, writeFile, cancellationToken, emitOnlyDtsFiles, customTransformers) {
+            var restorePendingEmitOnHandlingNoEmitSuccess = false;
+            var savedAffectedFilesPendingEmit;
+            var savedAffectedFilesPendingEmitKind;
+            var savedAffectedFilesPendingEmitIndex;
+            // Backup and restore affected pendings emit state for non emit Builder if noEmitOnError is enabled and emitBuildInfo could be written in case there are errors
+            // This ensures pending files to emit is updated in tsbuildinfo
+            // Note that when there are no errors, emit proceeds as if everything is emitted as it is callers reponsibility to write the files to disk if at all (because its builder that doesnt track files to emit)
+            if (kind !== BuilderProgramKind.EmitAndSemanticDiagnosticsBuilderProgram &&
+                !targetSourceFile &&
+                !ts.outFile(state.compilerOptions) &&
+                !state.compilerOptions.noEmit &&
+                state.compilerOptions.noEmitOnError) {
+                restorePendingEmitOnHandlingNoEmitSuccess = true;
+                savedAffectedFilesPendingEmit = state.affectedFilesPendingEmit && state.affectedFilesPendingEmit.slice();
+                savedAffectedFilesPendingEmitKind = state.affectedFilesPendingEmitKind && new ts.Map(state.affectedFilesPendingEmitKind);
+                savedAffectedFilesPendingEmitIndex = state.affectedFilesPendingEmitIndex;
+            }
             if (kind === BuilderProgramKind.EmitAndSemanticDiagnosticsBuilderProgram) {
                 assertSourceFileOkWithoutNextAffectedCall(state, targetSourceFile);
-                var result = ts.handleNoEmitOptions(builderProgram, targetSourceFile, cancellationToken);
-                if (result)
-                    return result;
-                if (!targetSourceFile) {
-                    // Emit and report any errors we ran into.
-                    var sourceMaps = [];
-                    var emitSkipped = false;
-                    var diagnostics = void 0;
-                    var emittedFiles = [];
-                    var affectedEmitResult = void 0;
-                    while (affectedEmitResult = emitNextAffectedFile(writeFile, cancellationToken, emitOnlyDtsFiles, customTransformers)) {
-                        emitSkipped = emitSkipped || affectedEmitResult.result.emitSkipped;
-                        diagnostics = ts.addRange(diagnostics, affectedEmitResult.result.diagnostics);
-                        emittedFiles = ts.addRange(emittedFiles, affectedEmitResult.result.emittedFiles);
-                        sourceMaps = ts.addRange(sourceMaps, affectedEmitResult.result.sourceMaps);
-                    }
-                    return {
-                        emitSkipped: emitSkipped,
-                        diagnostics: diagnostics || ts.emptyArray,
-                        emittedFiles: emittedFiles,
-                        sourceMaps: sourceMaps
-                    };
+            }
+            var result = ts.handleNoEmitOptions(builderProgram, targetSourceFile, writeFile, cancellationToken);
+            if (result)
+                return result;
+            if (restorePendingEmitOnHandlingNoEmitSuccess) {
+                state.affectedFilesPendingEmit = savedAffectedFilesPendingEmit;
+                state.affectedFilesPendingEmitKind = savedAffectedFilesPendingEmitKind;
+                state.affectedFilesPendingEmitIndex = savedAffectedFilesPendingEmitIndex;
+            }
+            // Emit only affected files if using builder for emit
+            if (!targetSourceFile && kind === BuilderProgramKind.EmitAndSemanticDiagnosticsBuilderProgram) {
+                // Emit and report any errors we ran into.
+                var sourceMaps = [];
+                var emitSkipped = false;
+                var diagnostics = void 0;
+                var emittedFiles = [];
+                var affectedEmitResult = void 0;
+                while (affectedEmitResult = emitNextAffectedFile(writeFile, cancellationToken, emitOnlyDtsFiles, customTransformers)) {
+                    emitSkipped = emitSkipped || affectedEmitResult.result.emitSkipped;
+                    diagnostics = ts.addRange(diagnostics, affectedEmitResult.result.diagnostics);
+                    emittedFiles = ts.addRange(emittedFiles, affectedEmitResult.result.emittedFiles);
+                    sourceMaps = ts.addRange(sourceMaps, affectedEmitResult.result.sourceMaps);
                 }
+                return {
+                    emitSkipped: emitSkipped,
+                    diagnostics: diagnostics || ts.emptyArray,
+                    emittedFiles: emittedFiles,
+                    sourceMaps: sourceMaps
+                };
             }
             return ts.Debug.checkDefined(state.program).emit(targetSourceFile, writeFile || ts.maybeBind(host, host.writeFile), cancellationToken, emitOnlyDtsFiles, customTransformers);
         }
@@ -103864,7 +109259,8 @@ var ts;
                     return toAffectedFileResult(state, state.program.getSemanticDiagnostics(/*targetSourceFile*/ undefined, cancellationToken), affected);
                 }
                 // Add file to affected file pending emit to handle for later emit time
-                if (kind === BuilderProgramKind.EmitAndSemanticDiagnosticsBuilderProgram) {
+                // Apart for emit builder do this for tsbuildinfo, do this for non emit builder when noEmit is set as tsbuildinfo is written and reused between emitters
+                if (kind === BuilderProgramKind.EmitAndSemanticDiagnosticsBuilderProgram || state.compilerOptions.noEmit || state.compilerOptions.noEmitOnError) {
                     addToAffectedFilesPendingEmit(state, affected.resolvedPath, 1 /* Full */);
                 }
                 // Get diagnostics for the affected file if its not ignored
@@ -103887,7 +109283,7 @@ var ts;
         function getSemanticDiagnostics(sourceFile, cancellationToken) {
             assertSourceFileOkWithoutNextAffectedCall(state, sourceFile);
             var compilerOptions = ts.Debug.checkDefined(state.program).getCompilerOptions();
-            if (compilerOptions.outFile || compilerOptions.out) {
+            if (ts.outFile(compilerOptions)) {
                 ts.Debug.assert(!state.semanticDiagnosticsPerFile);
                 // We dont need to cache the diagnostics just return them from program
                 return ts.Debug.checkDefined(state.program).getSemanticDiagnostics(sourceFile, cancellationToken);
@@ -103913,7 +109309,7 @@ var ts;
         if (!state.affectedFilesPendingEmit)
             state.affectedFilesPendingEmit = [];
         if (!state.affectedFilesPendingEmitKind)
-            state.affectedFilesPendingEmitKind = ts.createMap();
+            state.affectedFilesPendingEmitKind = new ts.Map();
         var existingKind = state.affectedFilesPendingEmitKind.get(affectedFilePendingEmit);
         state.affectedFilesPendingEmit.push(affectedFilePendingEmit);
         state.affectedFilesPendingEmitKind.set(affectedFilePendingEmit, existingKind || kind);
@@ -103928,12 +109324,12 @@ var ts;
     function getMapOfReferencedSet(mapLike, toPath) {
         if (!mapLike)
             return undefined;
-        var map = ts.createMap();
+        var map = new ts.Map();
         // Copies keys/values from template. Note that for..in will not throw if
         // template is undefined, and instead will just exit the loop.
         for (var key in mapLike) {
             if (ts.hasProperty(mapLike, key)) {
-                map.set(toPath(key), ts.arrayToSet(mapLike[key], toPath));
+                map.set(toPath(key), new ts.Set(mapLike[key].map(toPath)));
             }
         }
         return map;
@@ -103941,7 +109337,7 @@ var ts;
     function createBuildProgramUsingProgramBuildInfo(program, buildInfoPath, host) {
         var buildInfoDirectory = ts.getDirectoryPath(ts.getNormalizedAbsolutePath(buildInfoPath, host.getCurrentDirectory()));
         var getCanonicalFileName = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames());
-        var fileInfos = ts.createMap();
+        var fileInfos = new ts.Map();
         for (var key in program.fileInfos) {
             if (ts.hasProperty(program.fileInfos, key)) {
                 fileInfos.set(toPath(key), program.fileInfos[key]);
@@ -103953,7 +109349,10 @@ var ts;
             referencedMap: getMapOfReferencedSet(program.referencedMap, toPath),
             exportedModulesMap: getMapOfReferencedSet(program.exportedModulesMap, toPath),
             semanticDiagnosticsPerFile: program.semanticDiagnosticsPerFile && ts.arrayToMap(program.semanticDiagnosticsPerFile, function (value) { return toPath(ts.isString(value) ? value : value[0]); }, function (value) { return ts.isString(value) ? ts.emptyArray : value[1]; }),
-            hasReusableDiagnostic: true
+            hasReusableDiagnostic: true,
+            affectedFilesPendingEmit: ts.map(program.affectedFilesPendingEmit, function (value) { return toPath(value[0]); }),
+            affectedFilesPendingEmitKind: program.affectedFilesPendingEmit && ts.arrayToMap(program.affectedFilesPendingEmit, function (value) { return toPath(value[0]); }, function (value) { return value[1]; }),
+            affectedFilesPendingEmitIndex: program.affectedFilesPendingEmit && 0,
         };
         return {
             getState: function () { return state; },
@@ -103976,6 +109375,7 @@ var ts;
             getCurrentDirectory: ts.notImplemented,
             emitNextAffectedFile: ts.notImplemented,
             getSemanticDiagnosticsOfNextAffectedFile: ts.notImplemented,
+            emitBuildInfo: ts.notImplemented,
             close: ts.noop,
         };
         function toPath(path) {
@@ -104004,6 +109404,7 @@ var ts;
             getDeclarationDiagnostics: function (sourceFile, cancellationToken) { return getProgram().getDeclarationDiagnostics(sourceFile, cancellationToken); },
             getSemanticDiagnostics: function (sourceFile, cancellationToken) { return getProgram().getSemanticDiagnostics(sourceFile, cancellationToken); },
             emit: function (sourceFile, writeFile, cancellationToken, emitOnlyDts, customTransformers) { return getProgram().emit(sourceFile, writeFile, cancellationToken, emitOnlyDts, customTransformers); },
+            emitBuildInfo: function (writeFile, cancellationToken) { return getProgram().emitBuildInfo(writeFile, cancellationToken); },
             getAllDependencies: ts.notImplemented,
             getCurrentDirectory: function () { return getProgram().getCurrentDirectory(); },
             close: ts.noop,
@@ -104094,16 +109495,20 @@ var ts;
         var nonRelativeExternalModuleResolutions = ts.createMultiMap();
         var resolutionsWithFailedLookups = [];
         var resolvedFileToResolution = ts.createMultiMap();
+        var hasChangedAutomaticTypeDirectiveNames = false;
+        var failedLookupChecks = [];
+        var startsWithPathChecks = [];
+        var isInDirectoryChecks = [];
         var getCurrentDirectory = ts.memoize(function () { return resolutionHost.getCurrentDirectory(); }); // TODO: GH#18217
         var cachedDirectoryStructureHost = resolutionHost.getCachedDirectoryStructureHost();
         // The resolvedModuleNames and resolvedTypeReferenceDirectives are the cache of resolutions per file.
         // The key in the map is source file's path.
         // The values are Map of resolutions with key being name lookedup.
-        var resolvedModuleNames = ts.createMap();
+        var resolvedModuleNames = new ts.Map();
         var perDirectoryResolvedModuleNames = ts.createCacheWithRedirects();
         var nonRelativeModuleNameCache = ts.createCacheWithRedirects();
         var moduleResolutionCache = ts.createModuleResolutionCacheWithMaps(perDirectoryResolvedModuleNames, nonRelativeModuleNameCache, getCurrentDirectory(), resolutionHost.getCanonicalFileName);
-        var resolvedTypeReferenceDirectives = ts.createMap();
+        var resolvedTypeReferenceDirectives = new ts.Map();
         var perDirectoryResolvedTypeReferenceDirectives = ts.createCacheWithRedirects();
         /**
          * These are the extensions that failed lookup files will have by default,
@@ -104112,13 +109517,13 @@ var ts;
          * Note that .d.ts file also has .d.ts extension hence will be part of default extensions
          */
         var failedLookupDefaultExtensions = [".ts" /* Ts */, ".tsx" /* Tsx */, ".js" /* Js */, ".jsx" /* Jsx */, ".json" /* Json */];
-        var customFailedLookupPaths = ts.createMap();
-        var directoryWatchesOfFailedLookups = ts.createMap();
+        var customFailedLookupPaths = new ts.Map();
+        var directoryWatchesOfFailedLookups = new ts.Map();
         var rootDir = rootDirForResolution && ts.removeTrailingDirectorySeparator(ts.getNormalizedAbsolutePath(rootDirForResolution, getCurrentDirectory()));
         var rootPath = (rootDir && resolutionHost.toPath(rootDir)); // TODO: GH#18217
         var rootSplitLength = rootPath !== undefined ? rootPath.split(ts.directorySeparator).length : 0;
         // TypeRoot watches for the types that get added as part of getAutomaticTypeDirectiveNames
-        var typeRootsWatches = ts.createMap();
+        var typeRootsWatches = new ts.Map();
         return {
             startRecordingFilesWithChangedResolutions: startRecordingFilesWithChangedResolutions,
             finishRecordingFilesWithChangedResolutions: finishRecordingFilesWithChangedResolutions,
@@ -104131,7 +109536,9 @@ var ts;
             resolveTypeReferenceDirectives: resolveTypeReferenceDirectives,
             removeResolutionsFromProjectReferenceRedirects: removeResolutionsFromProjectReferenceRedirects,
             removeResolutionsOfFile: removeResolutionsOfFile,
+            hasChangedAutomaticTypeDirectiveNames: function () { return hasChangedAutomaticTypeDirectiveNames; },
             invalidateResolutionOfFile: invalidateResolutionOfFile,
+            invalidateResolutionsOfFailedLookupLocations: invalidateResolutionsOfFailedLookupLocations,
             setFilesWithInvalidatedNonRelativeUnresolvedImports: setFilesWithInvalidatedNonRelativeUnresolvedImports,
             createHasInvalidatedResolution: createHasInvalidatedResolution,
             updateTypeRootsWatch: updateTypeRootsWatch,
@@ -104159,9 +109566,13 @@ var ts;
             resolvedTypeReferenceDirectives.clear();
             resolvedFileToResolution.clear();
             resolutionsWithFailedLookups.length = 0;
+            failedLookupChecks.length = 0;
+            startsWithPathChecks.length = 0;
+            isInDirectoryChecks.length = 0;
             // perDirectoryResolvedModuleNames and perDirectoryResolvedTypeReferenceDirectives could be non empty if there was exception during program update
             // (between startCachingPerDirectoryResolution and finishCachingPerDirectoryResolution)
             clearPerDirectoryResolutions();
+            hasChangedAutomaticTypeDirectiveNames = false;
         }
         function startRecordingFilesWithChangedResolutions() {
             filesWithChangedSetOfUnresolvedImports = [];
@@ -104180,6 +109591,8 @@ var ts;
             return !!value && !!value.length;
         }
         function createHasInvalidatedResolution(forceAllFilesAsInvalidated) {
+            // Ensure pending resolutions are applied
+            invalidateResolutionsOfFailedLookupLocations();
             if (forceAllFilesAsInvalidated) {
                 // Any file asked would have invalidated resolution
                 filesWithInvalidatedResolutions = undefined;
@@ -104206,6 +109619,7 @@ var ts;
                     watcher.watcher.close();
                 }
             });
+            hasChangedAutomaticTypeDirectiveNames = false;
         }
         function resolveModuleName(moduleName, containingFile, compilerOptions, host, redirectedReference) {
             var _a;
@@ -104234,12 +109648,12 @@ var ts;
             var _b;
             var names = _a.names, containingFile = _a.containingFile, redirectedReference = _a.redirectedReference, cache = _a.cache, perDirectoryCacheWithRedirects = _a.perDirectoryCacheWithRedirects, loader = _a.loader, getResolutionWithResolvedFileName = _a.getResolutionWithResolvedFileName, shouldRetryResolution = _a.shouldRetryResolution, reusedNames = _a.reusedNames, logChanges = _a.logChanges;
             var path = resolutionHost.toPath(containingFile);
-            var resolutionsInFile = cache.get(path) || cache.set(path, ts.createMap()).get(path);
+            var resolutionsInFile = cache.get(path) || cache.set(path, new ts.Map()).get(path);
             var dirPath = ts.getDirectoryPath(path);
             var perDirectoryCache = perDirectoryCacheWithRedirects.getOrCreateMapOfCacheRedirects(redirectedReference);
             var perDirectoryResolution = perDirectoryCache.get(dirPath);
             if (!perDirectoryResolution) {
-                perDirectoryResolution = ts.createMap();
+                perDirectoryResolution = new ts.Map();
                 perDirectoryCache.set(dirPath, perDirectoryResolution);
             }
             var resolvedModules = [];
@@ -104251,7 +109665,7 @@ var ts;
             var unmatchedRedirects = oldRedirect ?
                 !redirectedReference || redirectedReference.sourceFile.path !== oldRedirect.sourceFile.path :
                 !!redirectedReference;
-            var seenNamesInFile = ts.createMap();
+            var seenNamesInFile = new ts.Map();
             for (var _i = 0, names_3 = names; _i < names_3.length; _i++) {
                 var name = names_3[_i];
                 var resolution = resolutionsInFile.get(name);
@@ -104333,7 +109747,7 @@ var ts;
                 getResolutionWithResolvedFileName: getResolvedModule,
                 shouldRetryResolution: function (resolution) { return !resolution.resolvedModule || !ts.resolutionExtensionIsTSOrJson(resolution.resolvedModule.extension); },
                 reusedNames: reusedNames,
-                logChanges: logChangesWhenResolvingModule
+                logChanges: logChangesWhenResolvingModule,
             });
         }
         function getResolvedModuleWithFailedLookupLocationsFromCache(moduleName, containingFile) {
@@ -104406,7 +109820,7 @@ var ts;
             }
             else {
                 resolution.refCount = 1;
-                ts.Debug.assert(resolution.files === undefined);
+                ts.Debug.assert(ts.length(resolution.files) === 0); // This resolution shouldnt be referenced by any file yet
                 if (ts.isExternalModuleNameRelative(name)) {
                     watchFailedLookupLocationOfResolution(resolution);
                 }
@@ -104525,9 +109939,7 @@ var ts;
                     // Since the file existence changed, update the sourceFiles cache
                     cachedDirectoryStructureHost.addOrDeleteFileOrDirectory(fileOrDirectory, fileOrDirectoryPath);
                 }
-                if (invalidateResolutionOfFailedLookupLocation(fileOrDirectoryPath, dirPath === fileOrDirectoryPath)) {
-                    resolutionHost.onInvalidatedResolution();
-                }
+                scheduleInvalidateResolutionOfFailedLookupLocation(fileOrDirectoryPath, dirPath === fileOrDirectoryPath);
             }, nonRecursive ? 0 /* None */ : 1 /* Recursive */);
         }
         function removeResolutionsOfFileFromCache(cache, filePath, getResolutionWithResolvedFileName) {
@@ -104558,34 +109970,43 @@ var ts;
             removeResolutionsOfFileFromCache(resolvedModuleNames, filePath, getResolvedModule);
             removeResolutionsOfFileFromCache(resolvedTypeReferenceDirectives, filePath, getResolvedTypeReferenceDirective);
         }
-        function invalidateResolution(resolution) {
-            resolution.isInvalidated = true;
-            var changedInAutoTypeReferenced = false;
-            for (var _i = 0, _a = ts.Debug.assertDefined(resolution.files); _i < _a.length; _i++) {
-                var containingFilePath = _a[_i];
-                (filesWithInvalidatedResolutions || (filesWithInvalidatedResolutions = ts.createMap())).set(containingFilePath, true);
-                // When its a file with inferred types resolution, invalidate type reference directive resolution
-                changedInAutoTypeReferenced = changedInAutoTypeReferenced || containingFilePath.endsWith(ts.inferredTypesContainingFile);
-            }
-            if (changedInAutoTypeReferenced) {
-                resolutionHost.onChangedAutomaticTypeDirectiveNames();
+        function invalidateResolutions(resolutions, canInvalidate) {
+            if (!resolutions)
+                return false;
+            var invalidated = false;
+            for (var _i = 0, resolutions_1 = resolutions; _i < resolutions_1.length; _i++) {
+                var resolution = resolutions_1[_i];
+                if (resolution.isInvalidated || !canInvalidate(resolution))
+                    continue;
+                resolution.isInvalidated = invalidated = true;
+                for (var _a = 0, _b = ts.Debug.assertDefined(resolution.files); _a < _b.length; _a++) {
+                    var containingFilePath = _b[_a];
+                    (filesWithInvalidatedResolutions || (filesWithInvalidatedResolutions = new ts.Set())).add(containingFilePath);
+                    // When its a file with inferred types resolution, invalidate type reference directive resolution
+                    hasChangedAutomaticTypeDirectiveNames = hasChangedAutomaticTypeDirectiveNames || ts.endsWith(containingFilePath, ts.inferredTypesContainingFile);
+                }
             }
+            return invalidated;
         }
         function invalidateResolutionOfFile(filePath) {
             removeResolutionsOfFile(filePath);
             // Resolution is invalidated if the resulting file name is same as the deleted file path
-            ts.forEach(resolvedFileToResolution.get(filePath), invalidateResolution);
+            var prevHasChangedAutomaticTypeDirectiveNames = hasChangedAutomaticTypeDirectiveNames;
+            if (invalidateResolutions(resolvedFileToResolution.get(filePath), ts.returnTrue) &&
+                hasChangedAutomaticTypeDirectiveNames &&
+                !prevHasChangedAutomaticTypeDirectiveNames) {
+                resolutionHost.onChangedAutomaticTypeDirectiveNames();
+            }
         }
         function setFilesWithInvalidatedNonRelativeUnresolvedImports(filesMap) {
             ts.Debug.assert(filesWithInvalidatedNonRelativeUnresolvedImports === filesMap || filesWithInvalidatedNonRelativeUnresolvedImports === undefined);
             filesWithInvalidatedNonRelativeUnresolvedImports = filesMap;
         }
-        function invalidateResolutionOfFailedLookupLocation(fileOrDirectoryPath, isCreatingWatchedDirectory) {
-            var isChangedFailedLookupLocation;
+        function scheduleInvalidateResolutionOfFailedLookupLocation(fileOrDirectoryPath, isCreatingWatchedDirectory) {
             if (isCreatingWatchedDirectory) {
                 // Watching directory is created
                 // Invalidate any resolution has failed lookup in this directory
-                isChangedFailedLookupLocation = function (location) { return isInDirectoryPath(fileOrDirectoryPath, resolutionHost.toPath(location)); };
+                isInDirectoryChecks.push(fileOrDirectoryPath);
             }
             else {
                 // If something to do with folder/file starting with "." in node_modules folder, skip it
@@ -104603,10 +110024,8 @@ var ts;
                 if (isNodeModulesAtTypesDirectory(fileOrDirectoryPath) || ts.isNodeModulesDirectory(fileOrDirectoryPath) ||
                     isNodeModulesAtTypesDirectory(dirOfFileOrDirectory) || ts.isNodeModulesDirectory(dirOfFileOrDirectory)) {
                     // Invalidate any resolution from this directory
-                    isChangedFailedLookupLocation = function (location) {
-                        var locationPath = resolutionHost.toPath(location);
-                        return locationPath === fileOrDirectoryPath || ts.startsWith(resolutionHost.toPath(location), fileOrDirectoryPath);
-                    };
+                    failedLookupChecks.push(fileOrDirectoryPath);
+                    startsWithPathChecks.push(fileOrDirectoryPath);
                 }
                 else {
                     if (!isPathWithDefaultFailedLookupExtension(fileOrDirectoryPath) && !customFailedLookupPaths.has(fileOrDirectoryPath)) {
@@ -104617,20 +110036,29 @@ var ts;
                         return false;
                     }
                     // Resolution need to be invalidated if failed lookup location is same as the file or directory getting created
-                    isChangedFailedLookupLocation = function (location) { return resolutionHost.toPath(location) === fileOrDirectoryPath; };
+                    failedLookupChecks.push(fileOrDirectoryPath);
                 }
             }
-            var invalidated = false;
-            // Resolution is invalidated if the resulting file name is same as the deleted file path
-            for (var _i = 0, resolutionsWithFailedLookups_1 = resolutionsWithFailedLookups; _i < resolutionsWithFailedLookups_1.length; _i++) {
-                var resolution = resolutionsWithFailedLookups_1[_i];
-                if (resolution.failedLookupLocations.some(isChangedFailedLookupLocation)) {
-                    invalidateResolution(resolution);
-                    invalidated = true;
-                }
+            resolutionHost.scheduleInvalidateResolutionsOfFailedLookupLocations();
+        }
+        function invalidateResolutionsOfFailedLookupLocations() {
+            if (!failedLookupChecks.length && !startsWithPathChecks.length && !isInDirectoryChecks.length) {
+                return false;
             }
+            var invalidated = invalidateResolutions(resolutionsWithFailedLookups, canInvalidateFailedLookupResolution);
+            failedLookupChecks.length = 0;
+            startsWithPathChecks.length = 0;
+            isInDirectoryChecks.length = 0;
             return invalidated;
         }
+        function canInvalidateFailedLookupResolution(resolution) {
+            return resolution.failedLookupLocations.some(function (location) {
+                var locationPath = resolutionHost.toPath(location);
+                return ts.contains(failedLookupChecks, locationPath) ||
+                    startsWithPathChecks.some(function (fileOrDirectoryPath) { return ts.startsWith(locationPath, fileOrDirectoryPath); }) ||
+                    isInDirectoryChecks.some(function (fileOrDirectoryPath) { return isInDirectoryPath(fileOrDirectoryPath, locationPath); });
+            });
+        }
         function closeTypeRootsWatch() {
             ts.clearMap(typeRootsWatches, ts.closeFileWatcher);
         }
@@ -104652,12 +110080,13 @@ var ts;
                 // For now just recompile
                 // We could potentially store more data here about whether it was/would be really be used or not
                 // and with that determine to trigger compilation but for now this is enough
+                hasChangedAutomaticTypeDirectiveNames = true;
                 resolutionHost.onChangedAutomaticTypeDirectiveNames();
                 // Since directory watchers invoked are flaky, the failed lookup location events might not be triggered
                 // So handle to failed lookup locations here as well to ensure we are invalidating resolutions
                 var dirPath = getDirectoryToWatchFailedLookupLocationFromTypeRoot(typeRoot, typeRootPath);
-                if (dirPath && invalidateResolutionOfFailedLookupLocation(fileOrDirectoryPath, dirPath === fileOrDirectoryPath)) {
-                    resolutionHost.onInvalidatedResolution();
+                if (dirPath) {
+                    scheduleInvalidateResolutionOfFailedLookupLocation(fileOrDirectoryPath, dirPath === fileOrDirectoryPath);
                 }
             }, 1 /* Recursive */);
         }
@@ -104758,14 +110187,14 @@ var ts;
         function getNodeModulesPackageName(compilerOptions, importingSourceFileName, nodeModulesFileName, host) {
             var info = getInfo(importingSourceFileName, host);
             var modulePaths = getAllModulePaths(importingSourceFileName, nodeModulesFileName, host);
-            return ts.firstDefined(modulePaths, function (moduleFileName) { return tryGetModuleNameAsNodeModule(moduleFileName, info, host, compilerOptions, /*packageNameOnly*/ true); });
+            return ts.firstDefined(modulePaths, function (modulePath) { return tryGetModuleNameAsNodeModule(modulePath, info, host, compilerOptions, /*packageNameOnly*/ true); });
         }
         moduleSpecifiers.getNodeModulesPackageName = getNodeModulesPackageName;
         function getModuleSpecifierWorker(compilerOptions, importingSourceFileName, toFileName, host, preferences) {
             var info = getInfo(importingSourceFileName, host);
             var modulePaths = getAllModulePaths(importingSourceFileName, toFileName, host);
-            return ts.firstDefined(modulePaths, function (moduleFileName) { return tryGetModuleNameAsNodeModule(moduleFileName, info, host, compilerOptions); }) ||
-                getLocalModuleSpecifier(toFileName, info, compilerOptions, preferences);
+            return ts.firstDefined(modulePaths, function (modulePath) { return tryGetModuleNameAsNodeModule(modulePath, info, host, compilerOptions); }) ||
+                getLocalModuleSpecifier(toFileName, info, compilerOptions, host, preferences);
         }
         /** Returns an import for each symlink and for the realpath. */
         function getModuleSpecifiers(moduleSymbol, compilerOptions, importingSourceFile, host, userPreferences) {
@@ -104776,8 +110205,46 @@ var ts;
             var moduleSourceFile = ts.getSourceFileOfNode(moduleSymbol.valueDeclaration || ts.getNonAugmentationDeclaration(moduleSymbol));
             var modulePaths = getAllModulePaths(importingSourceFile.path, moduleSourceFile.originalFileName, host);
             var preferences = getPreferences(userPreferences, compilerOptions, importingSourceFile);
-            var global = ts.mapDefined(modulePaths, function (moduleFileName) { return tryGetModuleNameAsNodeModule(moduleFileName, info, host, compilerOptions); });
-            return global.length ? global : modulePaths.map(function (moduleFileName) { return getLocalModuleSpecifier(moduleFileName, info, compilerOptions, preferences); });
+            var importedFileIsInNodeModules = ts.some(modulePaths, function (p) { return p.isInNodeModules; });
+            // Module specifier priority:
+            //   1. "Bare package specifiers" (e.g. "@foo/bar") resulting from a path through node_modules to a package.json's "types" entry
+            //   2. Specifiers generated using "paths" from tsconfig
+            //   3. Non-relative specfiers resulting from a path through node_modules (e.g. "@foo/bar/path/to/file")
+            //   4. Relative paths
+            var nodeModulesSpecifiers;
+            var pathsSpecifiers;
+            var relativeSpecifiers;
+            for (var _i = 0, modulePaths_1 = modulePaths; _i < modulePaths_1.length; _i++) {
+                var modulePath = modulePaths_1[_i];
+                var specifier = tryGetModuleNameAsNodeModule(modulePath, info, host, compilerOptions);
+                nodeModulesSpecifiers = ts.append(nodeModulesSpecifiers, specifier);
+                if (specifier && modulePath.isRedirect) {
+                    // If we got a specifier for a redirect, it was a bare package specifier (e.g. "@foo/bar",
+                    // not "@foo/bar/path/to/file"). No other specifier will be this good, so stop looking.
+                    return nodeModulesSpecifiers;
+                }
+                if (!specifier && !modulePath.isRedirect) {
+                    var local = getLocalModuleSpecifier(modulePath.path, info, compilerOptions, host, preferences);
+                    if (ts.pathIsBareSpecifier(local)) {
+                        pathsSpecifiers = ts.append(pathsSpecifiers, local);
+                    }
+                    else if (!importedFileIsInNodeModules || modulePath.isInNodeModules) {
+                        // Why this extra conditional, not just an `else`? If some path to the file contained
+                        // 'node_modules', but we can't create a non-relative specifier (e.g. "@foo/bar/path/to/file"),
+                        // that means we had to go through a *sibling's* node_modules, not one we can access directly.
+                        // If some path to the file was in node_modules but another was not, this likely indicates that
+                        // we have a monorepo structure with symlinks. In this case, the non-node_modules path is
+                        // probably the realpath, e.g. "../bar/path/to/file", but a relative path to another package
+                        // in a monorepo is probably not portable. So, the module specifier we actually go with will be
+                        // the relative path through node_modules, so that the declaration emitter can produce a
+                        // portability error. (See declarationEmitReexportedSymlinkReference3)
+                        relativeSpecifiers = ts.append(relativeSpecifiers, local);
+                    }
+                }
+            }
+            return (pathsSpecifiers === null || pathsSpecifiers === void 0 ? void 0 : pathsSpecifiers.length) ? pathsSpecifiers :
+                (nodeModulesSpecifiers === null || nodeModulesSpecifiers === void 0 ? void 0 : nodeModulesSpecifiers.length) ? nodeModulesSpecifiers :
+                    ts.Debug.checkDefined(relativeSpecifiers);
         }
         moduleSpecifiers.getModuleSpecifiers = getModuleSpecifiers;
         // importingSourceFileName is separate because getEditsForFileRename may need to specify an updated path
@@ -104786,22 +110253,26 @@ var ts;
             var sourceDirectory = ts.getDirectoryPath(importingSourceFileName);
             return { getCanonicalFileName: getCanonicalFileName, sourceDirectory: sourceDirectory };
         }
-        function getLocalModuleSpecifier(moduleFileName, _a, compilerOptions, _b) {
+        function getLocalModuleSpecifier(moduleFileName, _a, compilerOptions, host, _b) {
             var getCanonicalFileName = _a.getCanonicalFileName, sourceDirectory = _a.sourceDirectory;
             var ending = _b.ending, relativePreference = _b.relativePreference;
             var baseUrl = compilerOptions.baseUrl, paths = compilerOptions.paths, rootDirs = compilerOptions.rootDirs;
             var relativePath = rootDirs && tryGetModuleNameFromRootDirs(rootDirs, moduleFileName, sourceDirectory, getCanonicalFileName, ending, compilerOptions) ||
                 removeExtensionAndIndexPostFix(ts.ensurePathIsNonModuleName(ts.getRelativePathFromDirectory(sourceDirectory, moduleFileName, getCanonicalFileName)), ending, compilerOptions);
-            if (!baseUrl || relativePreference === 0 /* Relative */) {
+            if (!baseUrl && !paths || relativePreference === 0 /* Relative */) {
                 return relativePath;
             }
-            var relativeToBaseUrl = getRelativePathIfInDirectory(moduleFileName, baseUrl, getCanonicalFileName);
+            var baseDirectory = ts.getPathsBasePath(compilerOptions, host) || baseUrl;
+            var relativeToBaseUrl = getRelativePathIfInDirectory(moduleFileName, baseDirectory, getCanonicalFileName);
             if (!relativeToBaseUrl) {
                 return relativePath;
             }
             var importRelativeToBaseUrl = removeExtensionAndIndexPostFix(relativeToBaseUrl, ending, compilerOptions);
             var fromPaths = paths && tryGetModuleNameFromPaths(ts.removeFileExtension(relativeToBaseUrl), importRelativeToBaseUrl, paths);
-            var nonRelative = fromPaths === undefined ? importRelativeToBaseUrl : fromPaths;
+            var nonRelative = fromPaths === undefined && baseUrl !== undefined ? importRelativeToBaseUrl : fromPaths;
+            if (!nonRelative) {
+                return relativePath;
+            }
             if (relativePreference === 1 /* NonRelative */) {
                 return nonRelative;
             }
@@ -104830,8 +110301,8 @@ var ts;
             var match = str.match(/\//g);
             return match ? match.length : 0;
         }
-        function comparePathsByNumberOfDirectorySeparators(a, b) {
-            return ts.compareValues(numberOfDirectorySeparators(a), numberOfDirectorySeparators(b));
+        function comparePathsByRedirectAndNumberOfDirectorySeparators(a, b) {
+            return ts.compareBooleans(b.isRedirect, a.isRedirect) || ts.compareValues(numberOfDirectorySeparators(a.path), numberOfDirectorySeparators(b.path));
         }
         function forEachFileNameOfModule(importingFileName, importedFileName, host, preferSymlinks, cb) {
             var getCanonicalFileName = ts.hostGetCanonicalFileName(host);
@@ -104841,31 +110312,36 @@ var ts;
             var importedFileNames = __spreadArrays((referenceRedirect ? [referenceRedirect] : ts.emptyArray), [importedFileName], redirects);
             var targets = importedFileNames.map(function (f) { return ts.getNormalizedAbsolutePath(f, cwd); });
             if (!preferSymlinks) {
-                var result_12 = ts.forEach(targets, cb);
-                if (result_12)
-                    return result_12;
+                var result_15 = ts.forEach(targets, function (p) { return cb(p, referenceRedirect === p); });
+                if (result_15)
+                    return result_15;
             }
-            var links = host.getProbableSymlinks
-                ? host.getProbableSymlinks(host.getSourceFiles())
+            var links = host.getSymlinkCache
+                ? host.getSymlinkCache()
                 : ts.discoverProbableSymlinks(host.getSourceFiles(), getCanonicalFileName, cwd);
-            var compareStrings = (!host.useCaseSensitiveFileNames || host.useCaseSensitiveFileNames()) ? ts.compareStringsCaseSensitive : ts.compareStringsCaseInsensitive;
-            var result = ts.forEachEntry(links, function (resolved, path) {
-                if (ts.startsWithDirectory(importingFileName, resolved, getCanonicalFileName)) {
-                    return undefined; // Don't want to a package to globally import from itself
-                }
-                var target = ts.find(targets, function (t) { return compareStrings(t.slice(0, resolved.length + 1), resolved + "/") === 0 /* EqualTo */; });
-                if (target === undefined)
+            var symlinkedDirectories = links.getSymlinkedDirectories();
+            var useCaseSensitiveFileNames = !host.useCaseSensitiveFileNames || host.useCaseSensitiveFileNames();
+            var result = symlinkedDirectories && ts.forEachEntry(symlinkedDirectories, function (resolved, path) {
+                if (resolved === false)
                     return undefined;
-                var relative = ts.getRelativePathFromDirectory(resolved, target, getCanonicalFileName);
-                var option = ts.resolvePath(path, relative);
-                if (!host.fileExists || host.fileExists(option)) {
-                    var result_13 = cb(option);
-                    if (result_13)
-                        return result_13;
+                if (ts.startsWithDirectory(importingFileName, resolved.realPath, getCanonicalFileName)) {
+                    return undefined; // Don't want to a package to globally import from itself
                 }
+                return ts.forEach(targets, function (target) {
+                    if (!ts.containsPath(resolved.real, target, !useCaseSensitiveFileNames)) {
+                        return;
+                    }
+                    var relative = ts.getRelativePathFromDirectory(resolved.real, target, getCanonicalFileName);
+                    var option = ts.resolvePath(path, relative);
+                    if (!host.fileExists || host.fileExists(option)) {
+                        var result_16 = cb(option, target === referenceRedirect);
+                        if (result_16)
+                            return result_16;
+                    }
+                });
             });
             return result ||
-                (preferSymlinks ? ts.forEach(targets, cb) : undefined);
+                (preferSymlinks ? ts.forEach(targets, function (p) { return cb(p, p === referenceRedirect); }) : undefined);
         }
         moduleSpecifiers.forEachFileNameOfModule = forEachFileNameOfModule;
         /**
@@ -104875,31 +110351,30 @@ var ts;
         function getAllModulePaths(importingFileName, importedFileName, host) {
             var cwd = host.getCurrentDirectory();
             var getCanonicalFileName = ts.hostGetCanonicalFileName(host);
-            var allFileNames = ts.createMap();
+            var allFileNames = new ts.Map();
             var importedFileFromNodeModules = false;
             forEachFileNameOfModule(importingFileName, importedFileName, host, 
-            /*preferSymlinks*/ true, function (path) {
-                // dont return value, so we collect everything
-                allFileNames.set(path, getCanonicalFileName(path));
-                importedFileFromNodeModules = importedFileFromNodeModules || ts.pathContainsNodeModules(path);
+            /*preferSymlinks*/ true, function (path, isRedirect) {
+                var isInNodeModules = ts.pathContainsNodeModules(path);
+                allFileNames.set(path, { path: getCanonicalFileName(path), isRedirect: isRedirect, isInNodeModules: isInNodeModules });
+                importedFileFromNodeModules = importedFileFromNodeModules || isInNodeModules;
+                // don't return value, so we collect everything
             });
             // Sort by paths closest to importing file Name directory
             var sortedPaths = [];
-            var _loop_20 = function (directory) {
+            var _loop_24 = function (directory) {
                 var directoryStart = ts.ensureTrailingDirectorySeparator(directory);
                 var pathsInDirectory;
-                allFileNames.forEach(function (canonicalFileName, fileName) {
-                    if (ts.startsWith(canonicalFileName, directoryStart)) {
-                        // If the importedFile is from node modules, use only paths in node_modules folder as option
-                        if (!importedFileFromNodeModules || ts.pathContainsNodeModules(fileName)) {
-                            (pathsInDirectory || (pathsInDirectory = [])).push(fileName);
-                        }
+                allFileNames.forEach(function (_a, fileName) {
+                    var path = _a.path, isRedirect = _a.isRedirect, isInNodeModules = _a.isInNodeModules;
+                    if (ts.startsWith(path, directoryStart)) {
+                        (pathsInDirectory || (pathsInDirectory = [])).push({ path: fileName, isRedirect: isRedirect, isInNodeModules: isInNodeModules });
                         allFileNames.delete(fileName);
                     }
                 });
                 if (pathsInDirectory) {
                     if (pathsInDirectory.length > 1) {
-                        pathsInDirectory.sort(comparePathsByNumberOfDirectorySeparators);
+                        pathsInDirectory.sort(comparePathsByRedirectAndNumberOfDirectorySeparators);
                     }
                     sortedPaths.push.apply(sortedPaths, pathsInDirectory);
                 }
@@ -104911,7 +110386,7 @@ var ts;
             };
             var out_directory_1;
             for (var directory = ts.getDirectoryPath(ts.toPath(importingFileName, cwd, getCanonicalFileName)); allFileNames.size !== 0;) {
-                var state_8 = _loop_20(directory);
+                var state_8 = _loop_24(directory);
                 directory = out_directory_1;
                 if (state_8 === "break")
                     break;
@@ -104919,7 +110394,7 @@ var ts;
             if (allFileNames.size) {
                 var remainingPaths = ts.arrayFrom(allFileNames.values());
                 if (remainingPaths.length > 1)
-                    remainingPaths.sort(comparePathsByNumberOfDirectorySeparators);
+                    remainingPaths.sort(comparePathsByRedirectAndNumberOfDirectorySeparators);
                 sortedPaths.push.apply(sortedPaths, remainingPaths);
             }
             return sortedPaths;
@@ -104964,37 +110439,43 @@ var ts;
                 ? removeExtensionAndIndexPostFix(relativePath, ending, compilerOptions)
                 : ts.removeFileExtension(relativePath);
         }
-        function tryGetModuleNameAsNodeModule(moduleFileName, _a, host, options, packageNameOnly) {
-            var getCanonicalFileName = _a.getCanonicalFileName, sourceDirectory = _a.sourceDirectory;
+        function tryGetModuleNameAsNodeModule(_a, _b, host, options, packageNameOnly) {
+            var path = _a.path, isRedirect = _a.isRedirect;
+            var getCanonicalFileName = _b.getCanonicalFileName, sourceDirectory = _b.sourceDirectory;
             if (!host.fileExists || !host.readFile) {
                 return undefined;
             }
-            var parts = getNodeModulePathParts(moduleFileName);
+            var parts = getNodeModulePathParts(path);
             if (!parts) {
                 return undefined;
             }
             // Simplify the full file path to something that can be resolved by Node.
-            var moduleSpecifier = moduleFileName;
+            var moduleSpecifier = path;
+            var isPackageRootPath = false;
             if (!packageNameOnly) {
                 var packageRootIndex = parts.packageRootIndex;
                 var moduleFileNameForExtensionless = void 0;
                 while (true) {
                     // If the module could be imported by a directory name, use that directory's name
-                    var _b = tryDirectoryWithPackageJson(packageRootIndex), moduleFileToTry = _b.moduleFileToTry, packageRootPath = _b.packageRootPath;
+                    var _c = tryDirectoryWithPackageJson(packageRootIndex), moduleFileToTry = _c.moduleFileToTry, packageRootPath = _c.packageRootPath;
                     if (packageRootPath) {
                         moduleSpecifier = packageRootPath;
+                        isPackageRootPath = true;
                         break;
                     }
                     if (!moduleFileNameForExtensionless)
                         moduleFileNameForExtensionless = moduleFileToTry;
                     // try with next level of directory
-                    packageRootIndex = moduleFileName.indexOf(ts.directorySeparator, packageRootIndex + 1);
+                    packageRootIndex = path.indexOf(ts.directorySeparator, packageRootIndex + 1);
                     if (packageRootIndex === -1) {
                         moduleSpecifier = getExtensionlessFileName(moduleFileNameForExtensionless);
                         break;
                     }
                 }
             }
+            if (isRedirect && !isPackageRootPath) {
+                return undefined;
+            }
             var globalTypingsCacheLocation = host.getGlobalTypingsCacheLocation && host.getGlobalTypingsCacheLocation();
             // Get a path that's relative to node_modules or the importing file's path
             // if node_modules folder is in this folder or any of its parent folders, no need to keep it.
@@ -105008,16 +110489,16 @@ var ts;
             // For classic resolution, only allow importing from node_modules/@types, not other node_modules
             return ts.getEmitModuleResolutionKind(options) !== ts.ModuleResolutionKind.NodeJs && packageName === nodeModulesDirectoryName ? undefined : packageName;
             function tryDirectoryWithPackageJson(packageRootIndex) {
-                var packageRootPath = moduleFileName.substring(0, packageRootIndex);
+                var packageRootPath = path.substring(0, packageRootIndex);
                 var packageJsonPath = ts.combinePaths(packageRootPath, "package.json");
-                var moduleFileToTry = moduleFileName;
+                var moduleFileToTry = path;
                 if (host.fileExists(packageJsonPath)) {
                     var packageJsonContent = JSON.parse(host.readFile(packageJsonPath));
                     var versionPaths = packageJsonContent.typesVersions
                         ? ts.getPackageJsonTypesVersionsPaths(packageJsonContent.typesVersions)
                         : undefined;
                     if (versionPaths) {
-                        var subModuleName = moduleFileName.slice(packageRootPath.length + 1);
+                        var subModuleName = path.slice(packageRootPath.length + 1);
                         var fromPaths = tryGetModuleNameFromPaths(ts.removeFileExtension(subModuleName), removeExtensionAndIndexPostFix(subModuleName, 0 /* Minimal */, options), versionPaths.paths);
                         if (fromPaths !== undefined) {
                             moduleFileToTry = ts.combinePaths(packageRootPath, fromPaths);
@@ -105334,13 +110815,14 @@ var ts;
     }
     ts.emitFilesAndReportErrorsAndGetExitStatus = emitFilesAndReportErrorsAndGetExitStatus;
     ts.noopFileWatcher = { close: ts.noop };
+    ts.returnNoopFileWatcher = function () { return ts.noopFileWatcher; };
     function createWatchHost(system, reportWatchStatus) {
         if (system === void 0) { system = ts.sys; }
         var onWatchStatusChange = reportWatchStatus || createWatchStatusReporter(system);
         return {
             onWatchStatusChange: onWatchStatusChange,
-            watchFile: ts.maybeBind(system, system.watchFile) || (function () { return ts.noopFileWatcher; }),
-            watchDirectory: ts.maybeBind(system, system.watchDirectory) || (function () { return ts.noopFileWatcher; }),
+            watchFile: ts.maybeBind(system, system.watchFile) || ts.returnNoopFileWatcher,
+            watchDirectory: ts.maybeBind(system, system.watchDirectory) || ts.returnNoopFileWatcher,
             setTimeout: ts.maybeBind(system, system.setTimeout) || ts.noop,
             clearTimeout: ts.maybeBind(system, system.clearTimeout) || ts.noop
         };
@@ -105420,7 +110902,7 @@ var ts;
     ts.createCompilerHostFromProgramHost = createCompilerHostFromProgramHost;
     function setGetSourceFileAsHashVersioned(compilerHost, host) {
         var originalGetSourceFile = compilerHost.getSourceFile;
-        var computeHash = host.createHash || ts.generateDjb2Hash;
+        var computeHash = ts.maybeBind(host, host.createHash) || ts.generateDjb2Hash;
         compilerHost.getSourceFile = function () {
             var args = [];
             for (var _i = 0; _i < arguments.length; _i++) {
@@ -105428,7 +110910,7 @@ var ts;
             }
             var result = originalGetSourceFile.call.apply(originalGetSourceFile, __spreadArrays([compilerHost], args));
             if (result) {
-                result.version = computeHash.call(host, result.text);
+                result.version = computeHash(result.text);
             }
             return result;
         };
@@ -105524,7 +111006,7 @@ var ts;
 var ts;
 (function (ts) {
     function readBuilderProgram(compilerOptions, host) {
-        if (compilerOptions.out || compilerOptions.outFile)
+        if (ts.outFile(compilerOptions))
             return undefined;
         var buildInfoPath = ts.getTsBuildInfoEmitOutputFilePath(compilerOptions);
         if (!buildInfoPath)
@@ -105590,10 +111072,10 @@ var ts;
         var missingFilesMap; // Map of file watchers for the missing files
         var watchedWildcardDirectories; // map of watchers for the wild card directories in the config file
         var timerToUpdateProgram; // timer callback to recompile the program
-        var sourceFilesCache = ts.createMap(); // Cache that stores the source file and version info
+        var timerToInvalidateFailedLookupResolutions; // timer callback to invalidate resolutions for changes in failed lookup locations
+        var sourceFilesCache = new ts.Map(); // Cache that stores the source file and version info
         var missingFilePathsRequestedForRelease; // These paths are held temparirly so that we can remove the entry from source file cache if the file is not tracked by missing files
         var hasChangedCompilerOptions = false; // True if the compiler options have changed between compilations
-        var hasChangedAutomaticTypeDirectiveNames = false; // True if the automatic type directives have changed
         var useCaseSensitiveFileNames = host.useCaseSensitiveFileNames();
         var currentDirectory = host.getCurrentDirectory();
         var configFileName = host.configFileName, _a = host.optionsToExtend, optionsToExtendForConfigFile = _a === void 0 ? {} : _a, watchOptionsToExtend = host.watchOptionsToExtend, extraFileExtensions = host.extraFileExtensions, createProgram = host.createProgram;
@@ -105647,11 +111129,9 @@ var ts;
         compilerHost.watchDirectoryOfFailedLookupLocation = function (dir, cb, flags) { return watchDirectory(host, dir, cb, flags, watchOptions, ts.WatchType.FailedLookupLocations); };
         compilerHost.watchTypeRootsDirectory = function (dir, cb, flags) { return watchDirectory(host, dir, cb, flags, watchOptions, ts.WatchType.TypeRoots); };
         compilerHost.getCachedDirectoryStructureHost = function () { return cachedDirectoryStructureHost; };
+        compilerHost.scheduleInvalidateResolutionsOfFailedLookupLocations = scheduleInvalidateResolutionsOfFailedLookupLocations;
         compilerHost.onInvalidatedResolution = scheduleProgramUpdate;
-        compilerHost.onChangedAutomaticTypeDirectiveNames = function () {
-            hasChangedAutomaticTypeDirectiveNames = true;
-            scheduleProgramUpdate();
-        };
+        compilerHost.onChangedAutomaticTypeDirectiveNames = scheduleProgramUpdate;
         compilerHost.fileIsOpen = ts.returnFalse;
         compilerHost.getCurrentProgram = getCurrentProgram;
         compilerHost.writeLog = writeLog;
@@ -105688,6 +111168,7 @@ var ts;
             { getCurrentProgram: getCurrentBuilderProgram, getProgram: updateProgram, close: close } :
             { getCurrentProgram: getCurrentBuilderProgram, getProgram: updateProgram, updateRootFileNames: updateRootFileNames, close: close };
         function close() {
+            clearInvalidateResolutionsOfFailedLookupLocations();
             resolutionCache.clear();
             ts.clearMap(sourceFilesCache, function (value) {
                 if (value && value.fileWatcher) {
@@ -105716,6 +111197,7 @@ var ts;
         }
         function synchronizeProgram() {
             writeLog("Synchronizing program");
+            clearInvalidateResolutionsOfFailedLookupLocations();
             var program = getCurrentBuilderProgram();
             if (hasChangedCompilerOptions) {
                 newLine = updateNewLine();
@@ -105750,11 +111232,10 @@ var ts;
             resolutionCache.startCachingPerDirectoryResolution();
             compilerHost.hasInvalidatedResolution = hasInvalidatedResolution;
             compilerHost.hasChangedAutomaticTypeDirectiveNames = hasChangedAutomaticTypeDirectiveNames;
-            hasChangedAutomaticTypeDirectiveNames = false;
             builderProgram = createProgram(rootFileNames, compilerOptions, compilerHost, builderProgram, configFileParsingDiagnostics, projectReferences);
             resolutionCache.finishCachingPerDirectoryResolution();
             // Update watches
-            ts.updateMissingFilePathsWatch(builderProgram.getProgram(), missingFilesMap || (missingFilesMap = ts.createMap()), watchMissingFilePath);
+            ts.updateMissingFilePathsWatch(builderProgram.getProgram(), missingFilesMap || (missingFilesMap = new ts.Map()), watchMissingFilePath);
             if (needsUpdateInTypeRootWatch) {
                 resolutionCache.updateTypeRootsWatch();
             }
@@ -105881,6 +111362,30 @@ var ts;
                 host.onWatchStatusChange(ts.createCompilerDiagnostic(message), newLine, compilerOptions || optionsToExtendForConfigFile);
             }
         }
+        function hasChangedAutomaticTypeDirectiveNames() {
+            return resolutionCache.hasChangedAutomaticTypeDirectiveNames();
+        }
+        function clearInvalidateResolutionsOfFailedLookupLocations() {
+            if (!timerToInvalidateFailedLookupResolutions)
+                return false;
+            host.clearTimeout(timerToInvalidateFailedLookupResolutions);
+            timerToInvalidateFailedLookupResolutions = undefined;
+            return true;
+        }
+        function scheduleInvalidateResolutionsOfFailedLookupLocations() {
+            if (!host.setTimeout || !host.clearTimeout) {
+                return resolutionCache.invalidateResolutionsOfFailedLookupLocations();
+            }
+            var pending = clearInvalidateResolutionsOfFailedLookupLocations();
+            writeLog("Scheduling invalidateFailedLookup" + (pending ? ", Cancelled earlier one" : ""));
+            timerToInvalidateFailedLookupResolutions = host.setTimeout(invalidateResolutionsOfFailedLookup, 250);
+        }
+        function invalidateResolutionsOfFailedLookup() {
+            timerToInvalidateFailedLookupResolutions = undefined;
+            if (resolutionCache.invalidateResolutionsOfFailedLookupLocations()) {
+                scheduleProgramUpdate();
+            }
+        }
         // Upon detecting a file change, wait for 250ms and then perform a recompilation. This gives batch
         // operations (such as saving all modified files in an editor) a chance to complete before we kick
         // off a new compilation.
@@ -105924,7 +111429,7 @@ var ts;
         }
         function reloadFileNamesFromConfigFile() {
             writeLog("Reloading new file names and options");
-            var result = ts.getFileNamesFromConfigSpecs(configFileSpecs, ts.getNormalizedAbsolutePath(ts.getDirectoryPath(configFileName), currentDirectory), compilerOptions, parseConfigFileHost);
+            var result = ts.getFileNamesFromConfigSpecs(configFileSpecs, ts.getNormalizedAbsolutePath(ts.getDirectoryPath(configFileName), currentDirectory), compilerOptions, parseConfigFileHost, extraFileExtensions);
             if (ts.updateErrorForNoInputFiles(result, ts.getNormalizedAbsolutePath(configFileName, currentDirectory), configFileSpecs, configFileParsingDiagnostics, canConfigFileJsonReportNoInputFiles)) {
                 hasChangedConfigFileParsingErrors = true;
             }
@@ -105954,7 +111459,7 @@ var ts;
             configFileSpecs = configFileParseResult.configFileSpecs; // TODO: GH#18217
             projectReferences = configFileParseResult.projectReferences;
             configFileParsingDiagnostics = ts.getConfigFileParsingDiagnostics(configFileParseResult).slice();
-            canConfigFileJsonReportNoInputFiles = ts.canJsonReportNoInutFiles(configFileParseResult.raw);
+            canConfigFileJsonReportNoInputFiles = ts.canJsonReportNoInputFiles(configFileParseResult.raw);
             hasChangedConfigFileParsingErrors = true;
         }
         function onSourceFileChange(fileName, eventKind, path) {
@@ -105989,7 +111494,7 @@ var ts;
         }
         function watchConfigFileWildCardDirectories() {
             if (configFileSpecs) {
-                ts.updateWatchingWildcardDirectories(watchedWildcardDirectories || (watchedWildcardDirectories = ts.createMap()), ts.createMapFromTemplate(configFileSpecs.wildcardDirectories), watchWildcardDirectory);
+                ts.updateWatchingWildcardDirectories(watchedWildcardDirectories || (watchedWildcardDirectories = new ts.Map()), new ts.Map(ts.getEntries(configFileSpecs.wildcardDirectories)), watchWildcardDirectory);
             }
             else if (watchedWildcardDirectories) {
                 ts.clearMap(watchedWildcardDirectories, ts.closeFileWatcherOf);
@@ -106004,15 +111509,20 @@ var ts;
                     cachedDirectoryStructureHost.addOrDeleteFileOrDirectory(fileOrDirectory, fileOrDirectoryPath);
                 }
                 nextSourceFileVersion(fileOrDirectoryPath);
-                fileOrDirectoryPath = ts.removeIgnoredPath(fileOrDirectoryPath);
-                if (!fileOrDirectoryPath)
-                    return;
-                // If the the added or created file or directory is not supported file name, ignore the file
-                // But when watched directory is added/removed, we need to reload the file list
-                if (fileOrDirectoryPath !== directory && ts.hasExtension(fileOrDirectoryPath) && !ts.isSupportedSourceFileName(fileOrDirectory, compilerOptions)) {
-                    writeLog("Project: " + configFileName + " Detected file add/remove of non supported extension: " + fileOrDirectory);
+                if (ts.isIgnoredFileFromWildCardWatching({
+                    watchedDirPath: toPath(directory),
+                    fileOrDirectory: fileOrDirectory,
+                    fileOrDirectoryPath: fileOrDirectoryPath,
+                    configFileName: configFileName,
+                    configFileSpecs: configFileSpecs,
+                    extraFileExtensions: extraFileExtensions,
+                    options: compilerOptions,
+                    program: getCurrentBuilderProgram(),
+                    currentDirectory: currentDirectory,
+                    useCaseSensitiveFileNames: useCaseSensitiveFileNames,
+                    writeLog: writeLog
+                }))
                     return;
-                }
                 // Reload is pending, do the reload
                 if (reloadLevel !== ts.ConfigFileProgramReloadLevel.Full) {
                     reloadLevel = ts.ConfigFileProgramReloadLevel.Partial;
@@ -106086,9 +111596,6 @@ var ts;
         BuildResultFlags[BuildResultFlags["EmitErrors"] = 64] = "EmitErrors";
         BuildResultFlags[BuildResultFlags["AnyErrors"] = 124] = "AnyErrors";
     })(BuildResultFlags || (BuildResultFlags = {}));
-    function createConfigFileMap() {
-        return ts.createMap();
-    }
     function getOrCreateValueFromConfigFileMap(configFileMap, resolved, createT) {
         var existingValue = configFileMap.get(resolved);
         var newValue;
@@ -106099,7 +111606,7 @@ var ts;
         return existingValue || newValue;
     }
     function getOrCreateValueMapFromConfigFileMap(configFileMap, resolved) {
-        return getOrCreateValueFromConfigFileMap(configFileMap, resolved, ts.createMap);
+        return getOrCreateValueFromConfigFileMap(configFileMap, resolved, function () { return new ts.Map(); });
     }
     function newer(date1, date2) {
         return date2 > date1 ? date2 : date1;
@@ -106201,15 +111708,15 @@ var ts;
             baseCompilerOptions: baseCompilerOptions,
             rootNames: rootNames,
             baseWatchOptions: baseWatchOptions,
-            resolvedConfigFilePaths: ts.createMap(),
-            configFileCache: createConfigFileMap(),
-            projectStatus: createConfigFileMap(),
-            buildInfoChecked: createConfigFileMap(),
-            extendedConfigCache: ts.createMap(),
-            builderPrograms: createConfigFileMap(),
-            diagnostics: createConfigFileMap(),
-            projectPendingBuild: createConfigFileMap(),
-            projectErrorsReported: createConfigFileMap(),
+            resolvedConfigFilePaths: new ts.Map(),
+            configFileCache: new ts.Map(),
+            projectStatus: new ts.Map(),
+            buildInfoChecked: new ts.Map(),
+            extendedConfigCache: new ts.Map(),
+            builderPrograms: new ts.Map(),
+            diagnostics: new ts.Map(),
+            projectPendingBuild: new ts.Map(),
+            projectErrorsReported: new ts.Map(),
             compilerHost: compilerHost,
             moduleResolutionCache: moduleResolutionCache,
             // Mutable state
@@ -106223,9 +111730,9 @@ var ts;
             currentInvalidatedProject: undefined,
             // Watch state
             watch: watch,
-            allWatchedWildcardDirectories: createConfigFileMap(),
-            allWatchedInputFiles: createConfigFileMap(),
-            allWatchedConfigFiles: createConfigFileMap(),
+            allWatchedWildcardDirectories: new ts.Map(),
+            allWatchedInputFiles: new ts.Map(),
+            allWatchedConfigFiles: new ts.Map(),
             timerToBuildInvalidatedProject: undefined,
             reportFileChangeDetected: false,
             watchFile: watchFile,
@@ -106276,8 +111783,8 @@ var ts;
         return ts.resolveConfigFileProjectName(ts.resolvePath(state.currentDirectory, name));
     }
     function createBuildOrder(state, roots) {
-        var temporaryMarks = ts.createMap();
-        var permanentMarks = ts.createMap();
+        var temporaryMarks = new ts.Map();
+        var permanentMarks = new ts.Map();
         var circularityReportStack = [];
         var buildOrder;
         var circularDiagnostics;
@@ -106322,7 +111829,8 @@ var ts;
         var buildOrder = createBuildOrder(state, state.rootNames.map(function (f) { return resolveProjectName(state, f); }));
         // Clear all to ResolvedConfigFilePaths cache to start fresh
         state.resolvedConfigFilePaths.clear();
-        var currentProjects = ts.arrayToSet(getBuildOrderFromAnyBuildOrder(buildOrder), function (resolved) { return toResolvedConfigFilePath(state, resolved); });
+        // TODO(rbuckton): Should be a `Set`, but that requires changing the code below that uses `mutateMapSkippingNewValues`
+        var currentProjects = new ts.Map(getBuildOrderFromAnyBuildOrder(buildOrder).map(function (resolved) { return [toResolvedConfigFilePath(state, resolved), true]; }));
         var noopOnDelete = { onDeleteValue: ts.noop };
         // Config file cache
         ts.mutateMapSkippingNewValues(state.configFileCache, currentProjects, noopOnDelete);
@@ -106466,19 +111974,20 @@ var ts;
             }
         };
     }
+    var BuildStep;
+    (function (BuildStep) {
+        BuildStep[BuildStep["CreateProgram"] = 0] = "CreateProgram";
+        BuildStep[BuildStep["SyntaxDiagnostics"] = 1] = "SyntaxDiagnostics";
+        BuildStep[BuildStep["SemanticDiagnostics"] = 2] = "SemanticDiagnostics";
+        BuildStep[BuildStep["Emit"] = 3] = "Emit";
+        BuildStep[BuildStep["EmitBundle"] = 4] = "EmitBundle";
+        BuildStep[BuildStep["EmitBuildInfo"] = 5] = "EmitBuildInfo";
+        BuildStep[BuildStep["BuildInvalidatedProjectOfBundle"] = 6] = "BuildInvalidatedProjectOfBundle";
+        BuildStep[BuildStep["QueueReferencingProjects"] = 7] = "QueueReferencingProjects";
+        BuildStep[BuildStep["Done"] = 8] = "Done";
+    })(BuildStep || (BuildStep = {}));
     function createBuildOrUpdateInvalidedProject(kind, state, project, projectPath, projectIndex, config, buildOrder) {
-        var Step;
-        (function (Step) {
-            Step[Step["CreateProgram"] = 0] = "CreateProgram";
-            Step[Step["SyntaxDiagnostics"] = 1] = "SyntaxDiagnostics";
-            Step[Step["SemanticDiagnostics"] = 2] = "SemanticDiagnostics";
-            Step[Step["Emit"] = 3] = "Emit";
-            Step[Step["EmitBundle"] = 4] = "EmitBundle";
-            Step[Step["BuildInvalidatedProjectOfBundle"] = 5] = "BuildInvalidatedProjectOfBundle";
-            Step[Step["QueueReferencingProjects"] = 6] = "QueueReferencingProjects";
-            Step[Step["Done"] = 7] = "Done";
-        })(Step || (Step = {}));
-        var step = kind === InvalidatedProjectKind.Build ? Step.CreateProgram : Step.EmitBundle;
+        var step = kind === InvalidatedProjectKind.Build ? BuildStep.CreateProgram : BuildStep.EmitBundle;
         var program;
         var buildResult;
         var invalidatedProjectOfBundle;
@@ -106528,8 +112037,11 @@ var ts;
                     if (targetSourceFile || emitOnlyDtsFiles) {
                         return withProgramOrUndefined(function (program) { return program.emit(targetSourceFile, writeFile, cancellationToken, emitOnlyDtsFiles, customTransformers); });
                     }
-                    executeSteps(Step.SemanticDiagnostics, cancellationToken);
-                    if (step !== Step.Emit)
+                    executeSteps(BuildStep.SemanticDiagnostics, cancellationToken);
+                    if (step === BuildStep.EmitBuildInfo) {
+                        return emitBuildInfo(writeFile, cancellationToken);
+                    }
+                    if (step !== BuildStep.Emit)
                         return undefined;
                     return emit(writeFile, cancellationToken, customTransformers);
                 },
@@ -106543,18 +112055,18 @@ var ts;
                 getCompilerOptions: function () { return config.options; },
                 getCurrentDirectory: function () { return state.currentDirectory; },
                 emit: function (writeFile, customTransformers) {
-                    if (step !== Step.EmitBundle)
+                    if (step !== BuildStep.EmitBundle)
                         return invalidatedProjectOfBundle;
                     return emitBundle(writeFile, customTransformers);
                 },
                 done: done,
             };
         function done(cancellationToken, writeFile, customTransformers) {
-            executeSteps(Step.Done, cancellationToken, writeFile, customTransformers);
+            executeSteps(BuildStep.Done, cancellationToken, writeFile, customTransformers);
             return doneInvalidatedProject(state, projectPath);
         }
         function withProgramOrUndefined(action) {
-            executeSteps(Step.CreateProgram);
+            executeSteps(BuildStep.CreateProgram);
             return program && action(program);
         }
         function withProgramOrEmptyArray(action) {
@@ -106565,7 +112077,7 @@ var ts;
             if (state.options.dry) {
                 reportStatus(state, ts.Diagnostics.A_non_dry_build_would_build_project_0, project);
                 buildResult = BuildResultFlags.Success;
-                step = Step.QueueReferencingProjects;
+                step = BuildStep.QueueReferencingProjects;
                 return;
             }
             if (state.options.verbose)
@@ -106574,7 +112086,7 @@ var ts;
                 reportAndStoreErrors(state, projectPath, ts.getConfigFileParsingDiagnostics(config));
                 // Nothing to build - must be a solution file, basically
                 buildResult = BuildResultFlags.None;
-                step = Step.QueueReferencingProjects;
+                step = BuildStep.QueueReferencingProjects;
                 return;
             }
             var host = state.host, compilerHost = state.compilerHost;
@@ -106583,12 +112095,15 @@ var ts;
             updateModuleResolutionCache(state, project, config);
             // Create program
             program = host.createProgram(config.fileNames, config.options, compilerHost, getOldProgram(state, projectPath, config), ts.getConfigFileParsingDiagnostics(config), config.projectReferences);
+            if (state.watch) {
+                state.builderPrograms.set(projectPath, program);
+            }
             step++;
         }
         function handleDiagnostics(diagnostics, errorFlags, errorType) {
+            var _a;
             if (diagnostics.length) {
-                buildResult = buildErrors(state, projectPath, program, config, diagnostics, errorFlags, errorType);
-                step = Step.QueueReferencingProjects;
+                (_a = buildErrors(state, projectPath, program, config, diagnostics, errorFlags, errorType), buildResult = _a.buildResult, step = _a.step);
             }
             else {
                 step++;
@@ -106602,8 +112117,9 @@ var ts;
             handleDiagnostics(ts.Debug.checkDefined(program).getSemanticDiagnostics(/*sourceFile*/ undefined, cancellationToken), BuildResultFlags.TypeErrors, "Semantic");
         }
         function emit(writeFileCallback, cancellationToken, customTransformers) {
+            var _a;
             ts.Debug.assertIsDefined(program);
-            ts.Debug.assert(step === Step.Emit);
+            ts.Debug.assert(step === BuildStep.Emit);
             // Before emitting lets backup state, so we can revert it back if there are declaration errors to handle emit and declaration errors correctly
             program.backupState();
             var declDiagnostics;
@@ -106616,8 +112132,7 @@ var ts;
             // Don't emit .d.ts if there are decl file errors
             if (declDiagnostics) {
                 program.restoreState();
-                buildResult = buildErrors(state, projectPath, program, config, declDiagnostics, BuildResultFlags.DeclarationEmitErrors, "Declaration file");
-                step = Step.QueueReferencingProjects;
+                (_a = buildErrors(state, projectPath, program, config, declDiagnostics, BuildResultFlags.DeclarationEmitErrors, "Declaration file"), buildResult = _a.buildResult, step = _a.step);
                 return {
                     emitSkipped: true,
                     diagnostics: emitResult.diagnostics
@@ -106629,7 +112144,7 @@ var ts;
             var newestDeclarationFileContentChangedTime = minimumDate;
             var anyDtsChanged = false;
             var emitterDiagnostics = ts.createDiagnosticCollection();
-            var emittedOutputs = ts.createMap();
+            var emittedOutputs = new ts.Map();
             outputFiles.forEach(function (_a) {
                 var name = _a.name, text = _a.text, writeByteOrderMark = _a.writeByteOrderMark;
                 var priorChangeTime;
@@ -106653,17 +112168,31 @@ var ts;
             /*newestDeclarationFileContentChangedTimeIsMaximumDate*/ anyDtsChanged, outputFiles.length ? outputFiles[0].name : ts.getFirstProjectOutput(config, !host.useCaseSensitiveFileNames()), resultFlags);
             return emitResult;
         }
+        function emitBuildInfo(writeFileCallback, cancellationToken) {
+            ts.Debug.assertIsDefined(program);
+            ts.Debug.assert(step === BuildStep.EmitBuildInfo);
+            var emitResult = program.emitBuildInfo(writeFileCallback, cancellationToken);
+            if (emitResult.diagnostics.length) {
+                reportErrors(state, emitResult.diagnostics);
+                state.diagnostics.set(projectPath, __spreadArrays(state.diagnostics.get(projectPath), emitResult.diagnostics));
+                buildResult = BuildResultFlags.EmitErrors & buildResult;
+            }
+            if (emitResult.emittedFiles && state.writeFileName) {
+                emitResult.emittedFiles.forEach(function (name) { return listEmittedFile(state, config, name); });
+            }
+            afterProgramDone(state, program, config);
+            step = BuildStep.QueueReferencingProjects;
+            return emitResult;
+        }
         function finishEmit(emitterDiagnostics, emittedOutputs, priorNewestUpdateTime, newestDeclarationFileContentChangedTimeIsMaximumDate, oldestOutputFileName, resultFlags) {
+            var _a;
             var emitDiagnostics = emitterDiagnostics.getDiagnostics();
             if (emitDiagnostics.length) {
-                buildResult = buildErrors(state, projectPath, program, config, emitDiagnostics, BuildResultFlags.EmitErrors, "Emit");
-                step = Step.QueueReferencingProjects;
+                (_a = buildErrors(state, projectPath, program, config, emitDiagnostics, BuildResultFlags.EmitErrors, "Emit"), buildResult = _a.buildResult, step = _a.step);
                 return emitDiagnostics;
             }
             if (state.writeFileName) {
                 emittedOutputs.forEach(function (name) { return listEmittedFile(state, config, name); });
-                if (program)
-                    ts.listFiles(program, state.writeFileName);
             }
             // Update time stamps for rest of the outputs
             var newestDeclarationFileContentChangedTime = updateOutputTimestampsWorker(state, config, priorNewestUpdateTime, ts.Diagnostics.Updating_unchanged_output_timestamps_of_project_0, emittedOutputs);
@@ -106675,9 +112204,8 @@ var ts;
                     newestDeclarationFileContentChangedTime,
                 oldestOutputFileName: oldestOutputFileName
             });
-            afterProgramDone(state, projectPath, program, config);
-            state.projectCompilerOptions = state.baseCompilerOptions;
-            step = Step.QueueReferencingProjects;
+            afterProgramDone(state, program, config);
+            step = BuildStep.QueueReferencingProjects;
             buildResult = resultFlags;
             return emitDiagnostics;
         }
@@ -106686,7 +112214,7 @@ var ts;
             if (state.options.dry) {
                 reportStatus(state, ts.Diagnostics.A_non_dry_build_would_update_output_of_project_0, project);
                 buildResult = BuildResultFlags.Success;
-                step = Step.QueueReferencingProjects;
+                step = BuildStep.QueueReferencingProjects;
                 return undefined;
             }
             if (state.options.verbose)
@@ -106700,13 +112228,13 @@ var ts;
             }, customTransformers);
             if (ts.isString(outputFiles)) {
                 reportStatus(state, ts.Diagnostics.Cannot_update_output_of_project_0_because_there_was_error_reading_file_1, project, relName(state, outputFiles));
-                step = Step.BuildInvalidatedProjectOfBundle;
+                step = BuildStep.BuildInvalidatedProjectOfBundle;
                 return invalidatedProjectOfBundle = createBuildOrUpdateInvalidedProject(InvalidatedProjectKind.Build, state, project, projectPath, projectIndex, config, buildOrder);
             }
             // Actual Emit
             ts.Debug.assert(!!outputFiles.length);
             var emitterDiagnostics = ts.createDiagnosticCollection();
-            var emittedOutputs = ts.createMap();
+            var emittedOutputs = new ts.Map();
             outputFiles.forEach(function (_a) {
                 var name = _a.name, text = _a.text, writeByteOrderMark = _a.writeByteOrderMark;
                 emittedOutputs.set(toPath(state, name), name);
@@ -106717,34 +112245,37 @@ var ts;
             return { emitSkipped: false, diagnostics: emitDiagnostics };
         }
         function executeSteps(till, cancellationToken, writeFile, customTransformers) {
-            while (step <= till && step < Step.Done) {
+            while (step <= till && step < BuildStep.Done) {
                 var currentStep = step;
                 switch (step) {
-                    case Step.CreateProgram:
+                    case BuildStep.CreateProgram:
                         createProgram();
                         break;
-                    case Step.SyntaxDiagnostics:
+                    case BuildStep.SyntaxDiagnostics:
                         getSyntaxDiagnostics(cancellationToken);
                         break;
-                    case Step.SemanticDiagnostics:
+                    case BuildStep.SemanticDiagnostics:
                         getSemanticDiagnostics(cancellationToken);
                         break;
-                    case Step.Emit:
+                    case BuildStep.Emit:
                         emit(writeFile, cancellationToken, customTransformers);
                         break;
-                    case Step.EmitBundle:
+                    case BuildStep.EmitBuildInfo:
+                        emitBuildInfo(writeFile, cancellationToken);
+                        break;
+                    case BuildStep.EmitBundle:
                         emitBundle(writeFile, customTransformers);
                         break;
-                    case Step.BuildInvalidatedProjectOfBundle:
+                    case BuildStep.BuildInvalidatedProjectOfBundle:
                         ts.Debug.checkDefined(invalidatedProjectOfBundle).done(cancellationToken);
-                        step = Step.Done;
+                        step = BuildStep.Done;
                         break;
-                    case Step.QueueReferencingProjects:
+                    case BuildStep.QueueReferencingProjects:
                         queueReferencingProjects(state, project, projectPath, projectIndex, config, buildOrder, ts.Debug.checkDefined(buildResult));
                         step++;
                         break;
                     // Should never be done
-                    case Step.Done:
+                    case BuildStep.Done:
                     default:
                         ts.assertType(step);
                 }
@@ -106796,7 +112327,7 @@ var ts;
             else if (reloadLevel === ts.ConfigFileProgramReloadLevel.Partial) {
                 // Update file names
                 var result = ts.getFileNamesFromConfigSpecs(config.configFileSpecs, ts.getDirectoryPath(project), config.options, state.parseConfigFileHost);
-                ts.updateErrorForNoInputFiles(result, project, config.configFileSpecs, config.errors, ts.canJsonReportNoInutFiles(config.raw));
+                ts.updateErrorForNoInputFiles(result, project, config.configFileSpecs, config.errors, ts.canJsonReportNoInputFiles(config.raw));
                 config.fileNames = result.fileNames;
                 watchInputFiles(state, project, projectPath, config);
             }
@@ -106855,30 +112386,29 @@ var ts;
             return value;
         return ts.readBuilderProgram(parsed.options, compilerHost);
     }
-    function afterProgramDone(_a, proj, program, config) {
-        var host = _a.host, watch = _a.watch, builderPrograms = _a.builderPrograms;
+    function afterProgramDone(state, program, config) {
         if (program) {
-            if (host.afterProgramEmitAndDiagnostics) {
-                host.afterProgramEmitAndDiagnostics(program);
-            }
-            if (watch) {
-                program.releaseProgram();
-                builderPrograms.set(proj, program);
+            if (program && state.writeFileName)
+                ts.listFiles(program, state.writeFileName);
+            if (state.host.afterProgramEmitAndDiagnostics) {
+                state.host.afterProgramEmitAndDiagnostics(program);
             }
+            program.releaseProgram();
         }
-        else if (host.afterEmitBundle) {
-            host.afterEmitBundle(config);
+        else if (state.host.afterEmitBundle) {
+            state.host.afterEmitBundle(config);
         }
+        state.projectCompilerOptions = state.baseCompilerOptions;
     }
-    function buildErrors(state, resolvedPath, program, config, diagnostics, errorFlags, errorType) {
+    function buildErrors(state, resolvedPath, program, config, diagnostics, buildResult, errorType) {
+        var canEmitBuildInfo = !(buildResult & BuildResultFlags.SyntaxErrors) && program && !ts.outFile(program.getCompilerOptions());
         reportAndStoreErrors(state, resolvedPath, diagnostics);
         // List files if any other build error using program (emit errors already report files)
-        if (program && state.writeFileName)
-            ts.listFiles(program, state.writeFileName);
         state.projectStatus.set(resolvedPath, { type: ts.UpToDateStatusType.Unbuildable, reason: errorType + " errors" });
-        afterProgramDone(state, resolvedPath, program, config);
-        state.projectCompilerOptions = state.baseCompilerOptions;
-        return errorFlags;
+        if (canEmitBuildInfo)
+            return { buildResult: buildResult, step: BuildStep.EmitBuildInfo };
+        afterProgramDone(state, program, config);
+        return { buildResult: buildResult, step: BuildStep.QueueReferencingProjects };
     }
     function updateModuleResolutionCache(state, proj, config) {
         if (!state.moduleResolutionCache)
@@ -106936,7 +112466,7 @@ var ts;
             }
         }
         // Container if no files are specified in the project
-        if (!project.fileNames.length && !ts.canJsonReportNoInutFiles(project.raw)) {
+        if (!project.fileNames.length && !ts.canJsonReportNoInputFiles(project.raw)) {
             return {
                 type: ts.UpToDateStatusType.ContainerOnly
             };
@@ -107327,45 +112857,23 @@ var ts;
             invalidateProjectAndScheduleBuilds(state, resolvedPath, ts.ConfigFileProgramReloadLevel.Full);
         }, ts.PollingInterval.High, parsed === null || parsed === void 0 ? void 0 : parsed.watchOptions, ts.WatchType.ConfigFile, resolved));
     }
-    function isSameFile(state, file1, file2) {
-        return ts.comparePaths(file1, file2, state.currentDirectory, !state.host.useCaseSensitiveFileNames()) === 0 /* EqualTo */;
-    }
-    function isOutputFile(state, fileName, configFile) {
-        if (configFile.options.noEmit)
-            return false;
-        // ts or tsx files are not output
-        if (!ts.fileExtensionIs(fileName, ".d.ts" /* Dts */) &&
-            (ts.fileExtensionIs(fileName, ".ts" /* Ts */) || ts.fileExtensionIs(fileName, ".tsx" /* Tsx */))) {
-            return false;
-        }
-        // If options have --outFile or --out, check if its that
-        var out = configFile.options.outFile || configFile.options.out;
-        if (out && (isSameFile(state, fileName, out) || isSameFile(state, fileName, ts.removeFileExtension(out) + ".d.ts" /* Dts */))) {
-            return true;
-        }
-        // If declarationDir is specified, return if its a file in that directory
-        if (configFile.options.declarationDir && ts.containsPath(configFile.options.declarationDir, fileName, state.currentDirectory, !state.host.useCaseSensitiveFileNames())) {
-            return true;
-        }
-        // If --outDir, check if file is in that directory
-        if (configFile.options.outDir && ts.containsPath(configFile.options.outDir, fileName, state.currentDirectory, !state.host.useCaseSensitiveFileNames())) {
-            return true;
-        }
-        return !ts.forEach(configFile.fileNames, function (inputFile) { return isSameFile(state, fileName, inputFile); });
-    }
     function watchWildCardDirectories(state, resolved, resolvedPath, parsed) {
         if (!state.watch)
             return;
-        ts.updateWatchingWildcardDirectories(getOrCreateValueMapFromConfigFileMap(state.allWatchedWildcardDirectories, resolvedPath), ts.createMapFromTemplate(parsed.configFileSpecs.wildcardDirectories), function (dir, flags) { return state.watchDirectory(state.hostWithWatch, dir, function (fileOrDirectory) {
-            var fileOrDirectoryPath = toPath(state, fileOrDirectory);
-            if (fileOrDirectoryPath !== toPath(state, dir) && ts.hasExtension(fileOrDirectoryPath) && !ts.isSupportedSourceFileName(fileOrDirectory, parsed.options)) {
-                state.writeLog("Project: " + resolved + " Detected file add/remove of non supported extension: " + fileOrDirectory);
-                return;
-            }
-            if (isOutputFile(state, fileOrDirectory, parsed)) {
-                state.writeLog(fileOrDirectory + " is output file");
+        ts.updateWatchingWildcardDirectories(getOrCreateValueMapFromConfigFileMap(state.allWatchedWildcardDirectories, resolvedPath), new ts.Map(ts.getEntries(parsed.configFileSpecs.wildcardDirectories)), function (dir, flags) { return state.watchDirectory(state.hostWithWatch, dir, function (fileOrDirectory) {
+            if (ts.isIgnoredFileFromWildCardWatching({
+                watchedDirPath: toPath(state, dir),
+                fileOrDirectory: fileOrDirectory,
+                fileOrDirectoryPath: toPath(state, fileOrDirectory),
+                configFileName: resolved,
+                configFileSpecs: parsed.configFileSpecs,
+                currentDirectory: state.currentDirectory,
+                options: parsed.options,
+                program: state.builderPrograms.get(resolvedPath),
+                useCaseSensitiveFileNames: state.parseConfigFileHost.useCaseSensitiveFileNames,
+                writeLog: function (s) { return state.writeLog(s); }
+            }))
                 return;
-            }
             invalidateProjectAndScheduleBuilds(state, resolvedPath, ts.ConfigFileProgramReloadLevel.Partial);
         }, flags, parsed === null || parsed === void 0 ? void 0 : parsed.watchOptions, ts.WatchType.WildcardDirectory, resolved); });
     }
@@ -107596,7 +113104,7 @@ var ts;
         function nowString() {
             // E.g. "12:34:56.789"
             var d = new Date();
-            return d.getHours() + ":" + d.getMinutes() + ":" + d.getSeconds() + "." + d.getMilliseconds();
+            return ts.padLeft(d.getHours().toString(), 2, "0") + ":" + ts.padLeft(d.getMinutes().toString(), 2, "0") + ":" + ts.padLeft(d.getSeconds().toString(), 2, "0") + "." + ts.padLeft(d.getMilliseconds().toString(), 3, "0");
         }
         server.nowString = nowString;
     })(server = ts.server || (ts.server = {}));
@@ -107649,20 +113157,20 @@ var ts;
             "vm",
             "zlib"
         ];
-        JsTyping.nodeCoreModules = ts.arrayToSet(JsTyping.nodeCoreModuleList);
+        JsTyping.nodeCoreModules = new ts.Set(JsTyping.nodeCoreModuleList);
         function nonRelativeModuleNameForTypingCache(moduleName) {
             return JsTyping.nodeCoreModules.has(moduleName) ? "node" : moduleName;
         }
         JsTyping.nonRelativeModuleNameForTypingCache = nonRelativeModuleNameForTypingCache;
         function loadSafeList(host, safeListPath) {
             var result = ts.readConfigFile(safeListPath, function (path) { return host.readFile(path); });
-            return ts.createMapFromTemplate(result.config);
+            return new ts.Map(ts.getEntries(result.config));
         }
         JsTyping.loadSafeList = loadSafeList;
         function loadTypesMap(host, typesMapPath) {
             var result = ts.readConfigFile(typesMapPath, function (path) { return host.readFile(path); });
             if (result.config) {
-                return ts.createMapFromTemplate(result.config.simpleMap);
+                return new ts.Map(ts.getEntries(result.config.simpleMap));
             }
             return undefined;
         }
@@ -107681,7 +113189,7 @@ var ts;
                 return { cachedTypingPaths: [], newTypingNames: [], filesToWatch: [] };
             }
             // A typing name to typing file path mapping
-            var inferredTypings = ts.createMap();
+            var inferredTypings = new ts.Map();
             // Only infer typings for .js and .jsx files
             fileNames = ts.mapDefined(fileNames, function (fileName) {
                 var path = ts.normalizePath(fileName);
@@ -107694,9 +113202,9 @@ var ts;
                 addInferredTypings(typeAcquisition.include, "Explicitly included types");
             var exclude = typeAcquisition.exclude || [];
             // Directories to search for package.json, bower.json and other typing information
-            var possibleSearchDirs = ts.arrayToSet(fileNames, ts.getDirectoryPath);
-            possibleSearchDirs.set(projectRootPath, true);
-            possibleSearchDirs.forEach(function (_true, searchDir) {
+            var possibleSearchDirs = new ts.Set(fileNames.map(ts.getDirectoryPath));
+            possibleSearchDirs.add(projectRootPath);
+            possibleSearchDirs.forEach(function (searchDir) {
                 var packageJsonPath = ts.combinePaths(searchDir, "package.json");
                 getTypingNamesFromJson(packageJsonPath, filesToWatch);
                 var bowerJsonPath = ts.combinePaths(searchDir, "bower.json");
@@ -107706,7 +113214,9 @@ var ts;
                 var nodeModulesPath = ts.combinePaths(searchDir, "node_modules");
                 getTypingNamesFromPackagesFolder(nodeModulesPath, filesToWatch);
             });
-            getTypingNamesFromSourceFileNames(fileNames);
+            if (!typeAcquisition.disableFilenameBasedTypeAcquisition) {
+                getTypingNamesFromSourceFileNames(fileNames);
+            }
             // add typings for unresolved imports
             if (unresolvedImports) {
                 var module_1 = ts.deduplicate(unresolvedImports.map(nonRelativeModuleNameForTypingCache), ts.equateStringsCaseSensitive, ts.compareStringsCaseSensitive);
@@ -107953,7 +113463,25 @@ var ts;
         PackageJsonDependencyGroup[PackageJsonDependencyGroup["All"] = 15] = "All";
     })(PackageJsonDependencyGroup = ts.PackageJsonDependencyGroup || (ts.PackageJsonDependencyGroup = {}));
     /* @internal */
+    var PackageJsonAutoImportPreference;
+    (function (PackageJsonAutoImportPreference) {
+        PackageJsonAutoImportPreference[PackageJsonAutoImportPreference["Off"] = 0] = "Off";
+        PackageJsonAutoImportPreference[PackageJsonAutoImportPreference["On"] = 1] = "On";
+        PackageJsonAutoImportPreference[PackageJsonAutoImportPreference["Auto"] = 2] = "Auto";
+    })(PackageJsonAutoImportPreference = ts.PackageJsonAutoImportPreference || (ts.PackageJsonAutoImportPreference = {}));
+    var LanguageServiceMode;
+    (function (LanguageServiceMode) {
+        LanguageServiceMode[LanguageServiceMode["Semantic"] = 0] = "Semantic";
+        LanguageServiceMode[LanguageServiceMode["PartialSemantic"] = 1] = "PartialSemantic";
+        LanguageServiceMode[LanguageServiceMode["Syntactic"] = 2] = "Syntactic";
+    })(LanguageServiceMode = ts.LanguageServiceMode || (ts.LanguageServiceMode = {}));
+    /* @internal */
     ts.emptyOptions = {};
+    var SemanticClassificationFormat;
+    (function (SemanticClassificationFormat) {
+        SemanticClassificationFormat["Original"] = "original";
+        SemanticClassificationFormat["TwentyTwenty"] = "2020";
+    })(SemanticClassificationFormat = ts.SemanticClassificationFormat || (ts.SemanticClassificationFormat = {}));
     var HighlightSpanKind;
     (function (HighlightSpanKind) {
         HighlightSpanKind["none"] = "none";
@@ -108147,6 +113675,7 @@ var ts;
         ScriptElementKindModifier["staticModifier"] = "static";
         ScriptElementKindModifier["abstractModifier"] = "abstract";
         ScriptElementKindModifier["optionalModifier"] = "optional";
+        ScriptElementKindModifier["deprecatedModifier"] = "deprecated";
         ScriptElementKindModifier["dtsModifier"] = ".d.ts";
         ScriptElementKindModifier["tsModifier"] = ".ts";
         ScriptElementKindModifier["tsxModifier"] = ".tsx";
@@ -108226,37 +113755,37 @@ var ts;
     })(SemanticMeaning = ts.SemanticMeaning || (ts.SemanticMeaning = {}));
     function getMeaningFromDeclaration(node) {
         switch (node.kind) {
-            case 242 /* VariableDeclaration */:
+            case 249 /* VariableDeclaration */:
                 return ts.isInJSFile(node) && ts.getJSDocEnumTag(node) ? 7 /* All */ : 1 /* Value */;
-            case 156 /* Parameter */:
-            case 191 /* BindingElement */:
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-            case 281 /* PropertyAssignment */:
-            case 282 /* ShorthandPropertyAssignment */:
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
-            case 162 /* Constructor */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
-            case 280 /* CatchClause */:
-            case 273 /* JsxAttribute */:
+            case 160 /* Parameter */:
+            case 198 /* BindingElement */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+            case 288 /* PropertyAssignment */:
+            case 289 /* ShorthandPropertyAssignment */:
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
+            case 166 /* Constructor */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
+            case 287 /* CatchClause */:
+            case 280 /* JsxAttribute */:
                 return 1 /* Value */;
-            case 155 /* TypeParameter */:
-            case 246 /* InterfaceDeclaration */:
-            case 247 /* TypeAliasDeclaration */:
-            case 173 /* TypeLiteral */:
+            case 159 /* TypeParameter */:
+            case 253 /* InterfaceDeclaration */:
+            case 254 /* TypeAliasDeclaration */:
+            case 177 /* TypeLiteral */:
                 return 2 /* Type */;
-            case 322 /* JSDocTypedefTag */:
+            case 331 /* JSDocTypedefTag */:
                 // If it has no name node, it shares the name with the value declaration below it.
                 return node.name === undefined ? 1 /* Value */ | 2 /* Type */ : 2 /* Type */;
-            case 284 /* EnumMember */:
-            case 245 /* ClassDeclaration */:
+            case 291 /* EnumMember */:
+            case 252 /* ClassDeclaration */:
                 return 1 /* Value */ | 2 /* Type */;
-            case 249 /* ModuleDeclaration */:
+            case 256 /* ModuleDeclaration */:
                 if (ts.isAmbientModule(node)) {
                     return 4 /* Namespace */ | 1 /* Value */;
                 }
@@ -108266,16 +113795,16 @@ var ts;
                 else {
                     return 4 /* Namespace */;
                 }
-            case 248 /* EnumDeclaration */:
-            case 257 /* NamedImports */:
-            case 258 /* ImportSpecifier */:
-            case 253 /* ImportEqualsDeclaration */:
-            case 254 /* ImportDeclaration */:
-            case 259 /* ExportAssignment */:
-            case 260 /* ExportDeclaration */:
+            case 255 /* EnumDeclaration */:
+            case 264 /* NamedImports */:
+            case 265 /* ImportSpecifier */:
+            case 260 /* ImportEqualsDeclaration */:
+            case 261 /* ImportDeclaration */:
+            case 266 /* ExportAssignment */:
+            case 267 /* ExportDeclaration */:
                 return 7 /* All */;
             // An external module can be a Value
-            case 290 /* SourceFile */:
+            case 297 /* SourceFile */:
                 return 4 /* Namespace */ | 1 /* Value */;
         }
         return 7 /* All */;
@@ -108283,13 +113812,13 @@ var ts;
     ts.getMeaningFromDeclaration = getMeaningFromDeclaration;
     function getMeaningFromLocation(node) {
         node = getAdjustedReferenceLocation(node);
-        if (node.kind === 290 /* SourceFile */) {
+        if (node.kind === 297 /* SourceFile */) {
             return 1 /* Value */;
         }
-        else if (node.parent.kind === 259 /* ExportAssignment */
-            || node.parent.kind === 265 /* ExternalModuleReference */
-            || node.parent.kind === 258 /* ImportSpecifier */
-            || node.parent.kind === 255 /* ImportClause */
+        else if (node.parent.kind === 266 /* ExportAssignment */
+            || node.parent.kind === 272 /* ExternalModuleReference */
+            || node.parent.kind === 265 /* ImportSpecifier */
+            || node.parent.kind === 262 /* ImportClause */
             || ts.isImportEqualsDeclaration(node.parent) && node === node.parent.name) {
             return 7 /* All */;
         }
@@ -108299,6 +113828,9 @@ var ts;
         else if (ts.isDeclarationName(node)) {
             return getMeaningFromDeclaration(node.parent);
         }
+        else if (ts.isEntityName(node) && ts.isJSDocNameReference(node.parent)) {
+            return 7 /* All */;
+        }
         else if (isTypeReference(node)) {
             return 2 /* Type */;
         }
@@ -108322,11 +113854,11 @@ var ts;
         //     import a = |b|; // Namespace
         //     import a = |b.c|; // Value, type, namespace
         //     import a = |b.c|.d; // Namespace
-        var name = node.kind === 153 /* QualifiedName */ ? node : ts.isQualifiedName(node.parent) && node.parent.right === node ? node.parent : undefined;
-        return name && name.parent.kind === 253 /* ImportEqualsDeclaration */ ? 7 /* All */ : 4 /* Namespace */;
+        var name = node.kind === 157 /* QualifiedName */ ? node : ts.isQualifiedName(node.parent) && node.parent.right === node ? node.parent : undefined;
+        return name && name.parent.kind === 260 /* ImportEqualsDeclaration */ ? 7 /* All */ : 4 /* Namespace */;
     }
     function isInRightSideOfInternalImportEqualsDeclaration(node) {
-        while (node.parent.kind === 153 /* QualifiedName */) {
+        while (node.parent.kind === 157 /* QualifiedName */) {
             node = node.parent;
         }
         return ts.isInternalModuleImportEqualsDeclaration(node.parent) && node.parent.moduleReference === node;
@@ -108338,27 +113870,27 @@ var ts;
     function isQualifiedNameNamespaceReference(node) {
         var root = node;
         var isLastClause = true;
-        if (root.parent.kind === 153 /* QualifiedName */) {
-            while (root.parent && root.parent.kind === 153 /* QualifiedName */) {
+        if (root.parent.kind === 157 /* QualifiedName */) {
+            while (root.parent && root.parent.kind === 157 /* QualifiedName */) {
                 root = root.parent;
             }
             isLastClause = root.right === node;
         }
-        return root.parent.kind === 169 /* TypeReference */ && !isLastClause;
+        return root.parent.kind === 173 /* TypeReference */ && !isLastClause;
     }
     function isPropertyAccessNamespaceReference(node) {
         var root = node;
         var isLastClause = true;
-        if (root.parent.kind === 194 /* PropertyAccessExpression */) {
-            while (root.parent && root.parent.kind === 194 /* PropertyAccessExpression */) {
+        if (root.parent.kind === 201 /* PropertyAccessExpression */) {
+            while (root.parent && root.parent.kind === 201 /* PropertyAccessExpression */) {
                 root = root.parent;
             }
             isLastClause = root.name === node;
         }
-        if (!isLastClause && root.parent.kind === 216 /* ExpressionWithTypeArguments */ && root.parent.parent.kind === 279 /* HeritageClause */) {
+        if (!isLastClause && root.parent.kind === 223 /* ExpressionWithTypeArguments */ && root.parent.parent.kind === 286 /* HeritageClause */) {
             var decl = root.parent.parent.parent;
-            return (decl.kind === 245 /* ClassDeclaration */ && root.parent.parent.token === 113 /* ImplementsKeyword */) ||
-                (decl.kind === 246 /* InterfaceDeclaration */ && root.parent.parent.token === 90 /* ExtendsKeyword */);
+            return (decl.kind === 252 /* ClassDeclaration */ && root.parent.parent.token === 116 /* ImplementsKeyword */) ||
+                (decl.kind === 253 /* InterfaceDeclaration */ && root.parent.parent.token === 93 /* ExtendsKeyword */);
         }
         return false;
     }
@@ -108367,17 +113899,17 @@ var ts;
             node = node.parent;
         }
         switch (node.kind) {
-            case 104 /* ThisKeyword */:
+            case 107 /* ThisKeyword */:
                 return !ts.isExpressionNode(node);
-            case 183 /* ThisType */:
+            case 187 /* ThisType */:
                 return true;
         }
         switch (node.parent.kind) {
-            case 169 /* TypeReference */:
+            case 173 /* TypeReference */:
                 return true;
-            case 188 /* ImportType */:
+            case 195 /* ImportType */:
                 return !node.parent.isTypeOf;
-            case 216 /* ExpressionWithTypeArguments */:
+            case 223 /* ExpressionWithTypeArguments */:
                 return !ts.isExpressionWithTypeArgumentsInClassExtendsClause(node.parent);
         }
         return false;
@@ -108444,7 +113976,7 @@ var ts;
     ts.climbPastPropertyOrElementAccess = climbPastPropertyOrElementAccess;
     function getTargetLabel(referenceNode, labelName) {
         while (referenceNode) {
-            if (referenceNode.kind === 238 /* LabeledStatement */ && referenceNode.label.escapedText === labelName) {
+            if (referenceNode.kind === 245 /* LabeledStatement */ && referenceNode.label.escapedText === labelName) {
                 return referenceNode.label;
             }
             referenceNode = referenceNode.parent;
@@ -108505,22 +114037,22 @@ var ts;
     ts.isNameOfFunctionDeclaration = isNameOfFunctionDeclaration;
     function isLiteralNameOfPropertyDeclarationOrIndexAccess(node) {
         switch (node.parent.kind) {
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-            case 281 /* PropertyAssignment */:
-            case 284 /* EnumMember */:
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 249 /* ModuleDeclaration */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+            case 288 /* PropertyAssignment */:
+            case 291 /* EnumMember */:
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 256 /* ModuleDeclaration */:
                 return ts.getNameOfDeclaration(node.parent) === node;
-            case 195 /* ElementAccessExpression */:
+            case 202 /* ElementAccessExpression */:
                 return node.parent.argumentExpression === node;
-            case 154 /* ComputedPropertyName */:
+            case 158 /* ComputedPropertyName */:
                 return true;
-            case 187 /* LiteralType */:
-                return node.parent.parent.kind === 185 /* IndexedAccessType */;
+            case 191 /* LiteralType */:
+                return node.parent.parent.kind === 189 /* IndexedAccessType */;
             default:
                 return false;
         }
@@ -108544,17 +114076,17 @@ var ts;
                 return undefined;
             }
             switch (node.kind) {
-                case 290 /* SourceFile */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 245 /* ClassDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 249 /* ModuleDeclaration */:
+                case 297 /* SourceFile */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 252 /* ClassDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     return node;
             }
         }
@@ -108562,54 +114094,54 @@ var ts;
     ts.getContainerNode = getContainerNode;
     function getNodeKind(node) {
         switch (node.kind) {
-            case 290 /* SourceFile */:
+            case 297 /* SourceFile */:
                 return ts.isExternalModule(node) ? "module" /* moduleElement */ : "script" /* scriptElement */;
-            case 249 /* ModuleDeclaration */:
+            case 256 /* ModuleDeclaration */:
                 return "module" /* moduleElement */;
-            case 245 /* ClassDeclaration */:
-            case 214 /* ClassExpression */:
+            case 252 /* ClassDeclaration */:
+            case 221 /* ClassExpression */:
                 return "class" /* classElement */;
-            case 246 /* InterfaceDeclaration */: return "interface" /* interfaceElement */;
-            case 247 /* TypeAliasDeclaration */:
-            case 315 /* JSDocCallbackTag */:
-            case 322 /* JSDocTypedefTag */:
+            case 253 /* InterfaceDeclaration */: return "interface" /* interfaceElement */;
+            case 254 /* TypeAliasDeclaration */:
+            case 324 /* JSDocCallbackTag */:
+            case 331 /* JSDocTypedefTag */:
                 return "type" /* typeElement */;
-            case 248 /* EnumDeclaration */: return "enum" /* enumElement */;
-            case 242 /* VariableDeclaration */:
+            case 255 /* EnumDeclaration */: return "enum" /* enumElement */;
+            case 249 /* VariableDeclaration */:
                 return getKindOfVariableDeclaration(node);
-            case 191 /* BindingElement */:
+            case 198 /* BindingElement */:
                 return getKindOfVariableDeclaration(ts.getRootDeclaration(node));
-            case 202 /* ArrowFunction */:
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
                 return "function" /* functionElement */;
-            case 163 /* GetAccessor */: return "getter" /* memberGetAccessorElement */;
-            case 164 /* SetAccessor */: return "setter" /* memberSetAccessorElement */;
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
+            case 167 /* GetAccessor */: return "getter" /* memberGetAccessorElement */;
+            case 168 /* SetAccessor */: return "setter" /* memberSetAccessorElement */;
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
                 return "method" /* memberFunctionElement */;
-            case 281 /* PropertyAssignment */:
+            case 288 /* PropertyAssignment */:
                 var initializer = node.initializer;
                 return ts.isFunctionLike(initializer) ? "method" /* memberFunctionElement */ : "property" /* memberVariableElement */;
-            case 159 /* PropertyDeclaration */:
-            case 158 /* PropertySignature */:
-            case 282 /* ShorthandPropertyAssignment */:
-            case 283 /* SpreadAssignment */:
+            case 163 /* PropertyDeclaration */:
+            case 162 /* PropertySignature */:
+            case 289 /* ShorthandPropertyAssignment */:
+            case 290 /* SpreadAssignment */:
                 return "property" /* memberVariableElement */;
-            case 167 /* IndexSignature */: return "index" /* indexSignatureElement */;
-            case 166 /* ConstructSignature */: return "construct" /* constructSignatureElement */;
-            case 165 /* CallSignature */: return "call" /* callSignatureElement */;
-            case 162 /* Constructor */: return "constructor" /* constructorImplementationElement */;
-            case 155 /* TypeParameter */: return "type parameter" /* typeParameterElement */;
-            case 284 /* EnumMember */: return "enum member" /* enumMemberElement */;
-            case 156 /* Parameter */: return ts.hasModifier(node, 92 /* ParameterPropertyModifier */) ? "property" /* memberVariableElement */ : "parameter" /* parameterElement */;
-            case 253 /* ImportEqualsDeclaration */:
-            case 258 /* ImportSpecifier */:
-            case 263 /* ExportSpecifier */:
-            case 256 /* NamespaceImport */:
-            case 262 /* NamespaceExport */:
+            case 171 /* IndexSignature */: return "index" /* indexSignatureElement */;
+            case 170 /* ConstructSignature */: return "construct" /* constructSignatureElement */;
+            case 169 /* CallSignature */: return "call" /* callSignatureElement */;
+            case 166 /* Constructor */: return "constructor" /* constructorImplementationElement */;
+            case 159 /* TypeParameter */: return "type parameter" /* typeParameterElement */;
+            case 291 /* EnumMember */: return "enum member" /* enumMemberElement */;
+            case 160 /* Parameter */: return ts.hasSyntacticModifier(node, 92 /* ParameterPropertyModifier */) ? "property" /* memberVariableElement */ : "parameter" /* parameterElement */;
+            case 260 /* ImportEqualsDeclaration */:
+            case 265 /* ImportSpecifier */:
+            case 270 /* ExportSpecifier */:
+            case 263 /* NamespaceImport */:
+            case 269 /* NamespaceExport */:
                 return "alias" /* alias */;
-            case 209 /* BinaryExpression */:
+            case 216 /* BinaryExpression */:
                 var kind = ts.getAssignmentDeclarationKind(node);
                 var right = node.right;
                 switch (kind) {
@@ -108636,9 +114168,9 @@ var ts;
                         return "" /* unknown */;
                     }
                 }
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 return ts.isImportClause(node.parent) ? "alias" /* alias */ : "" /* unknown */;
-            case 259 /* ExportAssignment */:
+            case 266 /* ExportAssignment */:
                 var scriptKind = getNodeKind(node.expression);
                 // If the expression didn't come back with something (like it does for an identifiers)
                 return scriptKind === "" /* unknown */ ? "const" /* constElement */ : scriptKind;
@@ -108656,12 +114188,12 @@ var ts;
     ts.getNodeKind = getNodeKind;
     function isThis(node) {
         switch (node.kind) {
-            case 104 /* ThisKeyword */:
+            case 107 /* ThisKeyword */:
                 // case SyntaxKind.ThisType: TODO: GH#9267
                 return true;
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 // 'this' as a parameter
-                return ts.identifierIsThisKeyword(node) && node.parent.kind === 156 /* Parameter */;
+                return ts.identifierIsThisKeyword(node) && node.parent.kind === 160 /* Parameter */;
             default:
                 return false;
         }
@@ -108726,42 +114258,42 @@ var ts;
             return false;
         }
         switch (n.kind) {
-            case 245 /* ClassDeclaration */:
-            case 246 /* InterfaceDeclaration */:
-            case 248 /* EnumDeclaration */:
-            case 193 /* ObjectLiteralExpression */:
-            case 189 /* ObjectBindingPattern */:
-            case 173 /* TypeLiteral */:
-            case 223 /* Block */:
-            case 250 /* ModuleBlock */:
-            case 251 /* CaseBlock */:
-            case 257 /* NamedImports */:
-            case 261 /* NamedExports */:
+            case 252 /* ClassDeclaration */:
+            case 253 /* InterfaceDeclaration */:
+            case 255 /* EnumDeclaration */:
+            case 200 /* ObjectLiteralExpression */:
+            case 196 /* ObjectBindingPattern */:
+            case 177 /* TypeLiteral */:
+            case 230 /* Block */:
+            case 257 /* ModuleBlock */:
+            case 258 /* CaseBlock */:
+            case 264 /* NamedImports */:
+            case 268 /* NamedExports */:
                 return nodeEndsWith(n, 19 /* CloseBraceToken */, sourceFile);
-            case 280 /* CatchClause */:
+            case 287 /* CatchClause */:
                 return isCompletedNode(n.block, sourceFile);
-            case 197 /* NewExpression */:
+            case 204 /* NewExpression */:
                 if (!n.arguments) {
                     return true;
                 }
             // falls through
-            case 196 /* CallExpression */:
-            case 200 /* ParenthesizedExpression */:
-            case 182 /* ParenthesizedType */:
+            case 203 /* CallExpression */:
+            case 207 /* ParenthesizedExpression */:
+            case 186 /* ParenthesizedType */:
                 return nodeEndsWith(n, 21 /* CloseParenToken */, sourceFile);
-            case 170 /* FunctionType */:
-            case 171 /* ConstructorType */:
+            case 174 /* FunctionType */:
+            case 175 /* ConstructorType */:
                 return isCompletedNode(n.type, sourceFile);
-            case 162 /* Constructor */:
-            case 163 /* GetAccessor */:
-            case 164 /* SetAccessor */:
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 161 /* MethodDeclaration */:
-            case 160 /* MethodSignature */:
-            case 166 /* ConstructSignature */:
-            case 165 /* CallSignature */:
-            case 202 /* ArrowFunction */:
+            case 166 /* Constructor */:
+            case 167 /* GetAccessor */:
+            case 168 /* SetAccessor */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 165 /* MethodDeclaration */:
+            case 164 /* MethodSignature */:
+            case 170 /* ConstructSignature */:
+            case 169 /* CallSignature */:
+            case 209 /* ArrowFunction */:
                 if (n.body) {
                     return isCompletedNode(n.body, sourceFile);
                 }
@@ -108771,65 +114303,65 @@ var ts;
                 // Even though type parameters can be unclosed, we can get away with
                 // having at least a closing paren.
                 return hasChildOfKind(n, 21 /* CloseParenToken */, sourceFile);
-            case 249 /* ModuleDeclaration */:
+            case 256 /* ModuleDeclaration */:
                 return !!n.body && isCompletedNode(n.body, sourceFile);
-            case 227 /* IfStatement */:
+            case 234 /* IfStatement */:
                 if (n.elseStatement) {
                     return isCompletedNode(n.elseStatement, sourceFile);
                 }
                 return isCompletedNode(n.thenStatement, sourceFile);
-            case 226 /* ExpressionStatement */:
+            case 233 /* ExpressionStatement */:
                 return isCompletedNode(n.expression, sourceFile) ||
                     hasChildOfKind(n, 26 /* SemicolonToken */, sourceFile);
-            case 192 /* ArrayLiteralExpression */:
-            case 190 /* ArrayBindingPattern */:
-            case 195 /* ElementAccessExpression */:
-            case 154 /* ComputedPropertyName */:
-            case 175 /* TupleType */:
+            case 199 /* ArrayLiteralExpression */:
+            case 197 /* ArrayBindingPattern */:
+            case 202 /* ElementAccessExpression */:
+            case 158 /* ComputedPropertyName */:
+            case 179 /* TupleType */:
                 return nodeEndsWith(n, 23 /* CloseBracketToken */, sourceFile);
-            case 167 /* IndexSignature */:
+            case 171 /* IndexSignature */:
                 if (n.type) {
                     return isCompletedNode(n.type, sourceFile);
                 }
                 return hasChildOfKind(n, 23 /* CloseBracketToken */, sourceFile);
-            case 277 /* CaseClause */:
-            case 278 /* DefaultClause */:
+            case 284 /* CaseClause */:
+            case 285 /* DefaultClause */:
                 // there is no such thing as terminator token for CaseClause/DefaultClause so for simplicity always consider them non-completed
                 return false;
-            case 230 /* ForStatement */:
-            case 231 /* ForInStatement */:
-            case 232 /* ForOfStatement */:
-            case 229 /* WhileStatement */:
+            case 237 /* ForStatement */:
+            case 238 /* ForInStatement */:
+            case 239 /* ForOfStatement */:
+            case 236 /* WhileStatement */:
                 return isCompletedNode(n.statement, sourceFile);
-            case 228 /* DoStatement */:
+            case 235 /* DoStatement */:
                 // rough approximation: if DoStatement has While keyword - then if node is completed is checking the presence of ')';
-                return hasChildOfKind(n, 111 /* WhileKeyword */, sourceFile)
+                return hasChildOfKind(n, 114 /* WhileKeyword */, sourceFile)
                     ? nodeEndsWith(n, 21 /* CloseParenToken */, sourceFile)
                     : isCompletedNode(n.statement, sourceFile);
-            case 172 /* TypeQuery */:
+            case 176 /* TypeQuery */:
                 return isCompletedNode(n.exprName, sourceFile);
-            case 204 /* TypeOfExpression */:
-            case 203 /* DeleteExpression */:
-            case 205 /* VoidExpression */:
-            case 212 /* YieldExpression */:
-            case 213 /* SpreadElement */:
+            case 211 /* TypeOfExpression */:
+            case 210 /* DeleteExpression */:
+            case 212 /* VoidExpression */:
+            case 219 /* YieldExpression */:
+            case 220 /* SpreadElement */:
                 var unaryWordExpression = n;
                 return isCompletedNode(unaryWordExpression.expression, sourceFile);
-            case 198 /* TaggedTemplateExpression */:
+            case 205 /* TaggedTemplateExpression */:
                 return isCompletedNode(n.template, sourceFile);
-            case 211 /* TemplateExpression */:
+            case 218 /* TemplateExpression */:
                 var lastSpan = ts.lastOrUndefined(n.templateSpans);
                 return isCompletedNode(lastSpan, sourceFile);
-            case 221 /* TemplateSpan */:
+            case 228 /* TemplateSpan */:
                 return ts.nodeIsPresent(n.literal);
-            case 260 /* ExportDeclaration */:
-            case 254 /* ImportDeclaration */:
+            case 267 /* ExportDeclaration */:
+            case 261 /* ImportDeclaration */:
                 return ts.nodeIsPresent(n.moduleSpecifier);
-            case 207 /* PrefixUnaryExpression */:
+            case 214 /* PrefixUnaryExpression */:
                 return isCompletedNode(n.operand, sourceFile);
-            case 209 /* BinaryExpression */:
+            case 216 /* BinaryExpression */:
                 return isCompletedNode(n.right, sourceFile);
-            case 210 /* ConditionalExpression */:
+            case 217 /* ConditionalExpression */:
                 return isCompletedNode(n.whenFalse, sourceFile);
             default:
                 return true;
@@ -108889,13 +114421,13 @@ var ts;
     }
     ts.findContainingList = findContainingList;
     function isDefaultModifier(node) {
-        return node.kind === 84 /* DefaultKeyword */;
+        return node.kind === 87 /* DefaultKeyword */;
     }
     function isClassKeyword(node) {
-        return node.kind === 80 /* ClassKeyword */;
+        return node.kind === 83 /* ClassKeyword */;
     }
     function isFunctionKeyword(node) {
-        return node.kind === 94 /* FunctionKeyword */;
+        return node.kind === 97 /* FunctionKeyword */;
     }
     function getAdjustedLocationForClass(node) {
         if (ts.isNamedDeclaration(node)) {
@@ -108936,11 +114468,11 @@ var ts;
     function getAdjustedLocationForDeclaration(node, forRename) {
         if (!forRename) {
             switch (node.kind) {
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
                     return getAdjustedLocationForClass(node);
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
                     return getAdjustedLocationForFunction(node);
             }
         }
@@ -109030,30 +114562,30 @@ var ts;
         //
         // NOTE: If the node is a modifier, we don't adjust its location if it is the `default` modifier as that is handled
         // specially by `getSymbolAtLocation`.
-        if (ts.isModifier(node) && (forRename || node.kind !== 84 /* DefaultKeyword */) ? ts.contains(parent.modifiers, node) :
-            node.kind === 80 /* ClassKeyword */ ? ts.isClassDeclaration(parent) || ts.isClassExpression(node) :
-                node.kind === 94 /* FunctionKeyword */ ? ts.isFunctionDeclaration(parent) || ts.isFunctionExpression(node) :
-                    node.kind === 114 /* InterfaceKeyword */ ? ts.isInterfaceDeclaration(parent) :
-                        node.kind === 88 /* EnumKeyword */ ? ts.isEnumDeclaration(parent) :
-                            node.kind === 145 /* TypeKeyword */ ? ts.isTypeAliasDeclaration(parent) :
-                                node.kind === 136 /* NamespaceKeyword */ || node.kind === 135 /* ModuleKeyword */ ? ts.isModuleDeclaration(parent) :
-                                    node.kind === 96 /* ImportKeyword */ ? ts.isImportEqualsDeclaration(parent) :
-                                        node.kind === 131 /* GetKeyword */ ? ts.isGetAccessorDeclaration(parent) :
-                                            node.kind === 142 /* SetKeyword */ && ts.isSetAccessorDeclaration(parent)) {
+        if (ts.isModifier(node) && (forRename || node.kind !== 87 /* DefaultKeyword */) ? ts.contains(parent.modifiers, node) :
+            node.kind === 83 /* ClassKeyword */ ? ts.isClassDeclaration(parent) || ts.isClassExpression(node) :
+                node.kind === 97 /* FunctionKeyword */ ? ts.isFunctionDeclaration(parent) || ts.isFunctionExpression(node) :
+                    node.kind === 117 /* InterfaceKeyword */ ? ts.isInterfaceDeclaration(parent) :
+                        node.kind === 91 /* EnumKeyword */ ? ts.isEnumDeclaration(parent) :
+                            node.kind === 149 /* TypeKeyword */ ? ts.isTypeAliasDeclaration(parent) :
+                                node.kind === 140 /* NamespaceKeyword */ || node.kind === 139 /* ModuleKeyword */ ? ts.isModuleDeclaration(parent) :
+                                    node.kind === 99 /* ImportKeyword */ ? ts.isImportEqualsDeclaration(parent) :
+                                        node.kind === 134 /* GetKeyword */ ? ts.isGetAccessorDeclaration(parent) :
+                                            node.kind === 146 /* SetKeyword */ && ts.isSetAccessorDeclaration(parent)) {
             var location = getAdjustedLocationForDeclaration(parent, forRename);
             if (location) {
                 return location;
             }
         }
         // /**/<var|let|const> [|name|] ...
-        if ((node.kind === 109 /* VarKeyword */ || node.kind === 81 /* ConstKeyword */ || node.kind === 115 /* LetKeyword */) &&
+        if ((node.kind === 112 /* VarKeyword */ || node.kind === 84 /* ConstKeyword */ || node.kind === 118 /* LetKeyword */) &&
             ts.isVariableDeclarationList(parent) && parent.declarations.length === 1) {
             var decl = parent.declarations[0];
             if (ts.isIdentifier(decl.name)) {
                 return decl.name;
             }
         }
-        if (node.kind === 145 /* TypeKeyword */) {
+        if (node.kind === 149 /* TypeKeyword */) {
             // import /**/type [|name|] from ...;
             // import /**/type { [|name|] } from ...;
             // import /**/type { propertyName as [|name|] } from ...;
@@ -109079,7 +114611,7 @@ var ts;
         // import * /**/as [|name|] ...
         // export { propertyName /**/as [|name|] } ...
         // export * /**/as [|name|] ...
-        if (node.kind === 123 /* AsKeyword */) {
+        if (node.kind === 126 /* AsKeyword */) {
             if (ts.isImportSpecifier(parent) && parent.propertyName ||
                 ts.isExportSpecifier(parent) && parent.propertyName ||
                 ts.isNamespaceImport(parent) ||
@@ -109095,13 +114627,13 @@ var ts;
         // /**/import { propertyName as [|name|] } from ...;
         // /**/import ... from "[|module|]";
         // /**/import "[|module|]";
-        if (node.kind === 96 /* ImportKeyword */ && ts.isImportDeclaration(parent)) {
+        if (node.kind === 99 /* ImportKeyword */ && ts.isImportDeclaration(parent)) {
             var location = getAdjustedLocationForImportDeclaration(parent, forRename);
             if (location) {
                 return location;
             }
         }
-        if (node.kind === 89 /* ExportKeyword */) {
+        if (node.kind === 92 /* ExportKeyword */) {
             // /**/export { [|name|] } ...;
             // /**/export { propertyName as [|name|] } ...;
             // /**/export * from "[|module|]";
@@ -109120,12 +114652,12 @@ var ts;
             }
         }
         // import name = /**/require("[|module|]");
-        if (node.kind === 139 /* RequireKeyword */ && ts.isExternalModuleReference(parent)) {
+        if (node.kind === 143 /* RequireKeyword */ && ts.isExternalModuleReference(parent)) {
             return parent.expression;
         }
         // import ... /**/from "[|module|]";
         // export ... /**/from "[|module|]";
-        if (node.kind === 149 /* FromKeyword */ && (ts.isImportDeclaration(parent) || ts.isExportDeclaration(parent)) && parent.moduleSpecifier) {
+        if (node.kind === 153 /* FromKeyword */ && (ts.isImportDeclaration(parent) || ts.isExportDeclaration(parent)) && parent.moduleSpecifier) {
             return parent.moduleSpecifier;
         }
         // class ... /**/extends [|name|] ...
@@ -109133,13 +114665,13 @@ var ts;
         // class ... /**/implements name1, name2 ...
         // interface ... /**/extends [|name|] ...
         // interface ... /**/extends name1, name2 ...
-        if ((node.kind === 90 /* ExtendsKeyword */ || node.kind === 113 /* ImplementsKeyword */) && ts.isHeritageClause(parent) && parent.token === node.kind) {
+        if ((node.kind === 93 /* ExtendsKeyword */ || node.kind === 116 /* ImplementsKeyword */) && ts.isHeritageClause(parent) && parent.token === node.kind) {
             var location = getAdjustedLocationForHeritageClause(parent);
             if (location) {
                 return location;
             }
         }
-        if (node.kind === 90 /* ExtendsKeyword */) {
+        if (node.kind === 93 /* ExtendsKeyword */) {
             // ... <T /**/extends [|U|]> ...
             if (ts.isTypeParameterDeclaration(parent) && parent.constraint && ts.isTypeReferenceNode(parent.constraint)) {
                 return parent.constraint.typeName;
@@ -109150,20 +114682,20 @@ var ts;
             }
         }
         // ... T extends /**/infer [|U|] ? ...
-        if (node.kind === 132 /* InferKeyword */ && ts.isInferTypeNode(parent)) {
+        if (node.kind === 135 /* InferKeyword */ && ts.isInferTypeNode(parent)) {
             return parent.typeParameter.name;
         }
         // { [ [|K|] /**/in keyof T]: ... }
-        if (node.kind === 97 /* InKeyword */ && ts.isTypeParameterDeclaration(parent) && ts.isMappedTypeNode(parent.parent)) {
+        if (node.kind === 100 /* InKeyword */ && ts.isTypeParameterDeclaration(parent) && ts.isMappedTypeNode(parent.parent)) {
             return parent.name;
         }
         // /**/keyof [|T|]
-        if (node.kind === 134 /* KeyOfKeyword */ && ts.isTypeOperatorNode(parent) && parent.operator === 134 /* KeyOfKeyword */ &&
+        if (node.kind === 138 /* KeyOfKeyword */ && ts.isTypeOperatorNode(parent) && parent.operator === 138 /* KeyOfKeyword */ &&
             ts.isTypeReferenceNode(parent.type)) {
             return parent.type.typeName;
         }
         // /**/readonly [|name|][]
-        if (node.kind === 138 /* ReadonlyKeyword */ && ts.isTypeOperatorNode(parent) && parent.operator === 138 /* ReadonlyKeyword */ &&
+        if (node.kind === 142 /* ReadonlyKeyword */ && ts.isTypeOperatorNode(parent) && parent.operator === 142 /* ReadonlyKeyword */ &&
             ts.isArrayTypeNode(parent.type) && ts.isTypeReferenceNode(parent.type.elementType)) {
             return parent.type.elementType.typeName;
         }
@@ -109178,29 +114710,29 @@ var ts;
             // /**/yield [|name|]
             // /**/yield obj.[|name|]
             // /**/delete obj.[|name|]
-            if (node.kind === 99 /* NewKeyword */ && ts.isNewExpression(parent) ||
-                node.kind === 110 /* VoidKeyword */ && ts.isVoidExpression(parent) ||
-                node.kind === 108 /* TypeOfKeyword */ && ts.isTypeOfExpression(parent) ||
-                node.kind === 127 /* AwaitKeyword */ && ts.isAwaitExpression(parent) ||
-                node.kind === 121 /* YieldKeyword */ && ts.isYieldExpression(parent) ||
-                node.kind === 85 /* DeleteKeyword */ && ts.isDeleteExpression(parent)) {
+            if (node.kind === 102 /* NewKeyword */ && ts.isNewExpression(parent) ||
+                node.kind === 113 /* VoidKeyword */ && ts.isVoidExpression(parent) ||
+                node.kind === 111 /* TypeOfKeyword */ && ts.isTypeOfExpression(parent) ||
+                node.kind === 130 /* AwaitKeyword */ && ts.isAwaitExpression(parent) ||
+                node.kind === 124 /* YieldKeyword */ && ts.isYieldExpression(parent) ||
+                node.kind === 88 /* DeleteKeyword */ && ts.isDeleteExpression(parent)) {
                 if (parent.expression) {
                     return ts.skipOuterExpressions(parent.expression);
                 }
             }
             // left /**/in [|name|]
             // left /**/instanceof [|name|]
-            if ((node.kind === 97 /* InKeyword */ || node.kind === 98 /* InstanceOfKeyword */) && ts.isBinaryExpression(parent) && parent.operatorToken === node) {
+            if ((node.kind === 100 /* InKeyword */ || node.kind === 101 /* InstanceOfKeyword */) && ts.isBinaryExpression(parent) && parent.operatorToken === node) {
                 return ts.skipOuterExpressions(parent.right);
             }
             // left /**/as [|name|]
-            if (node.kind === 123 /* AsKeyword */ && ts.isAsExpression(parent) && ts.isTypeReferenceNode(parent.type)) {
+            if (node.kind === 126 /* AsKeyword */ && ts.isAsExpression(parent) && ts.isTypeReferenceNode(parent.type)) {
                 return parent.type.typeName;
             }
             // for (... /**/in [|name|])
             // for (... /**/of [|name|])
-            if (node.kind === 97 /* InKeyword */ && ts.isForInStatement(parent) ||
-                node.kind === 152 /* OfKeyword */ && ts.isForOfStatement(parent)) {
+            if (node.kind === 100 /* InKeyword */ && ts.isForInStatement(parent) ||
+                node.kind === 156 /* OfKeyword */ && ts.isForOfStatement(parent)) {
                 return ts.skipOuterExpressions(parent.expression);
             }
         }
@@ -109318,7 +114850,21 @@ var ts;
                 return n;
             }
             var children = n.getChildren(sourceFile);
-            for (var i = 0; i < children.length; i++) {
+            var i = ts.binarySearchKey(children, position, function (_, i) { return i; }, function (middle, _) {
+                // This last callback is more of a selector than a comparator -
+                // `EqualTo` causes the `middle` result to be returned
+                // `GreaterThan` causes recursion on the left of the middle
+                // `LessThan` causes recursion on the right of the middle
+                if (position < children[middle].end) {
+                    // first element whose end position is greater than the input position
+                    if (!children[middle - 1] || position >= children[middle - 1].end) {
+                        return 0 /* EqualTo */;
+                    }
+                    return 1 /* GreaterThan */;
+                }
+                return -1 /* LessThan */;
+            });
+            if (i >= 0 && children[i]) {
                 var child = children[i];
                 // Note that the span of a node's tokens is [node.getStart(...), node.end).
                 // Given that `position < child.end` and child has constituent tokens, we distinguish these cases:
@@ -109341,7 +114887,7 @@ var ts;
                     }
                 }
             }
-            ts.Debug.assert(startNode !== undefined || n.kind === 290 /* SourceFile */ || n.kind === 1 /* EndOfFileToken */ || ts.isJSDocCommentContainingNode(n));
+            ts.Debug.assert(startNode !== undefined || n.kind === 297 /* SourceFile */ || n.kind === 1 /* EndOfFileToken */ || ts.isJSDocCommentContainingNode(n));
             // Here we know that none of child token nodes embrace the position,
             // the only known case is when position is at the end of the file.
             // Try to find the rightmost token in the file without filtering.
@@ -109359,6 +114905,9 @@ var ts;
             return n;
         }
         var children = n.getChildren(sourceFile);
+        if (children.length === 0) {
+            return n;
+        }
         var candidate = findRightmostChildNodeWithTokens(children, /*exclusiveStartPosition*/ children.length, sourceFile);
         return candidate && findRightmostToken(candidate, sourceFile);
     }
@@ -109411,17 +114960,17 @@ var ts;
             return true;
         }
         // <div> { | </div> or <div a={| </div>
-        if (token.kind === 29 /* LessThanToken */ && token.parent.kind === 276 /* JsxExpression */) {
+        if (token.kind === 29 /* LessThanToken */ && token.parent.kind === 283 /* JsxExpression */) {
             return true;
         }
         // <div> {
         // |
         // } < /div>
-        if (token && token.kind === 19 /* CloseBraceToken */ && token.parent.kind === 276 /* JsxExpression */) {
+        if (token && token.kind === 19 /* CloseBraceToken */ && token.parent.kind === 283 /* JsxExpression */) {
             return true;
         }
         // <div>|</div>
-        if (token.kind === 29 /* LessThanToken */ && token.parent.kind === 269 /* JsxClosingElement */) {
+        if (token.kind === 29 /* LessThanToken */ && token.parent.kind === 276 /* JsxClosingElement */) {
             return true;
         }
         return false;
@@ -109449,6 +114998,33 @@ var ts;
         return false;
     }
     ts.isInJSXText = isInJSXText;
+    function isInsideJsxElement(sourceFile, position) {
+        function isInsideJsxElementTraversal(node) {
+            while (node) {
+                if (node.kind >= 274 /* JsxSelfClosingElement */ && node.kind <= 283 /* JsxExpression */
+                    || node.kind === 11 /* JsxText */
+                    || node.kind === 29 /* LessThanToken */
+                    || node.kind === 31 /* GreaterThanToken */
+                    || node.kind === 78 /* Identifier */
+                    || node.kind === 19 /* CloseBraceToken */
+                    || node.kind === 18 /* OpenBraceToken */
+                    || node.kind === 43 /* SlashToken */) {
+                    node = node.parent;
+                }
+                else if (node.kind === 273 /* JsxElement */) {
+                    if (position > node.getStart(sourceFile))
+                        return true;
+                    node = node.parent;
+                }
+                else {
+                    return false;
+                }
+            }
+            return false;
+        }
+        return isInsideJsxElementTraversal(getTokenAtPosition(sourceFile, position));
+    }
+    ts.isInsideJsxElement = isInsideJsxElement;
     function findPrecedingMatchingToken(token, matchingTokenKind, sourceFile) {
         var tokenKind = token.kind;
         var remainingMatchingTokens = 0;
@@ -109553,16 +115129,16 @@ var ts;
                     break;
                 case 38 /* EqualsGreaterThanToken */:
                 // falls through
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                 case 10 /* StringLiteral */:
                 case 8 /* NumericLiteral */:
                 case 9 /* BigIntLiteral */:
-                case 106 /* TrueKeyword */:
-                case 91 /* FalseKeyword */:
+                case 109 /* TrueKeyword */:
+                case 94 /* FalseKeyword */:
                 // falls through
-                case 108 /* TypeOfKeyword */:
-                case 90 /* ExtendsKeyword */:
-                case 134 /* KeyOfKeyword */:
+                case 111 /* TypeOfKeyword */:
+                case 93 /* ExtendsKeyword */:
+                case 138 /* KeyOfKeyword */:
                 case 24 /* DotToken */:
                 case 51 /* BarToken */:
                 case 57 /* QuestionToken */:
@@ -109601,7 +115177,7 @@ var ts;
         return n.kind === 1 /* EndOfFileToken */ ? !!n.jsDoc : n.getWidth(sourceFile) !== 0;
     }
     function getNodeModifiers(node) {
-        var flags = ts.isDeclaration(node) ? ts.getCombinedModifierFlags(node) : 0 /* None */;
+        var flags = ts.isDeclaration(node) ? ts.getCombinedNodeFlagsAlwaysIncludeJSDoc(node) : 0 /* None */;
         var result = [];
         if (flags & 8 /* Private */)
             result.push("private" /* privateMemberModifier */);
@@ -109615,18 +115191,20 @@ var ts;
             result.push("abstract" /* abstractModifier */);
         if (flags & 1 /* Export */)
             result.push("export" /* exportedModifier */);
+        if (flags & 8192 /* Deprecated */)
+            result.push("deprecated" /* deprecatedModifier */);
         if (node.flags & 8388608 /* Ambient */)
             result.push("declare" /* ambientModifier */);
-        if (node.kind === 259 /* ExportAssignment */)
+        if (node.kind === 266 /* ExportAssignment */)
             result.push("export" /* exportedModifier */);
         return result.length > 0 ? result.join(",") : "" /* none */;
     }
     ts.getNodeModifiers = getNodeModifiers;
     function getTypeArgumentOrTypeParameterList(node) {
-        if (node.kind === 169 /* TypeReference */ || node.kind === 196 /* CallExpression */) {
+        if (node.kind === 173 /* TypeReference */ || node.kind === 203 /* CallExpression */) {
             return node.typeArguments;
         }
-        if (ts.isFunctionLike(node) || node.kind === 245 /* ClassDeclaration */ || node.kind === 246 /* InterfaceDeclaration */) {
+        if (ts.isFunctionLike(node) || node.kind === 252 /* ClassDeclaration */ || node.kind === 253 /* InterfaceDeclaration */) {
             return node.typeParameters;
         }
         return undefined;
@@ -109646,7 +115224,7 @@ var ts;
     }
     ts.isStringOrRegularExpressionOrTemplateLiteral = isStringOrRegularExpressionOrTemplateLiteral;
     function isPunctuation(kind) {
-        return 18 /* FirstPunctuation */ <= kind && kind <= 74 /* LastPunctuation */;
+        return 18 /* FirstPunctuation */ <= kind && kind <= 77 /* LastPunctuation */;
     }
     ts.isPunctuation = isPunctuation;
     function isInsideTemplateLiteral(node, position, sourceFile) {
@@ -109656,9 +115234,9 @@ var ts;
     ts.isInsideTemplateLiteral = isInsideTemplateLiteral;
     function isAccessibilityModifier(kind) {
         switch (kind) {
-            case 119 /* PublicKeyword */:
-            case 117 /* PrivateKeyword */:
-            case 118 /* ProtectedKeyword */:
+            case 122 /* PublicKeyword */:
+            case 120 /* PrivateKeyword */:
+            case 121 /* ProtectedKeyword */:
                 return true;
         }
         return false;
@@ -109671,18 +115249,18 @@ var ts;
     }
     ts.cloneCompilerOptions = cloneCompilerOptions;
     function isArrayLiteralOrObjectLiteralDestructuringPattern(node) {
-        if (node.kind === 192 /* ArrayLiteralExpression */ ||
-            node.kind === 193 /* ObjectLiteralExpression */) {
+        if (node.kind === 199 /* ArrayLiteralExpression */ ||
+            node.kind === 200 /* ObjectLiteralExpression */) {
             // [a,b,c] from:
             // [a, b, c] = someExpression;
-            if (node.parent.kind === 209 /* BinaryExpression */ &&
+            if (node.parent.kind === 216 /* BinaryExpression */ &&
                 node.parent.left === node &&
                 node.parent.operatorToken.kind === 62 /* EqualsToken */) {
                 return true;
             }
             // [a, b, c] from:
             // for([a, b, c] of expression)
-            if (node.parent.kind === 232 /* ForOfStatement */ &&
+            if (node.parent.kind === 239 /* ForOfStatement */ &&
                 node.parent.initializer === node) {
                 return true;
             }
@@ -109690,7 +115268,7 @@ var ts;
             // [x, [a, b, c] ] = someExpression
             // or
             // {x, a: {a, b, c} } = someExpression
-            if (isArrayLiteralOrObjectLiteralDestructuringPattern(node.parent.kind === 281 /* PropertyAssignment */ ? node.parent.parent : node.parent)) {
+            if (isArrayLiteralOrObjectLiteralDestructuringPattern(node.parent.kind === 288 /* PropertyAssignment */ ? node.parent.parent : node.parent)) {
                 return true;
             }
         }
@@ -109752,31 +115330,31 @@ var ts;
     }
     ts.createTextChange = createTextChange;
     ts.typeKeywords = [
-        125 /* AnyKeyword */,
-        124 /* AssertsKeyword */,
-        151 /* BigIntKeyword */,
-        128 /* BooleanKeyword */,
-        91 /* FalseKeyword */,
-        134 /* KeyOfKeyword */,
-        137 /* NeverKeyword */,
-        100 /* NullKeyword */,
-        140 /* NumberKeyword */,
-        141 /* ObjectKeyword */,
-        138 /* ReadonlyKeyword */,
-        143 /* StringKeyword */,
-        144 /* SymbolKeyword */,
-        106 /* TrueKeyword */,
-        110 /* VoidKeyword */,
-        146 /* UndefinedKeyword */,
-        147 /* UniqueKeyword */,
-        148 /* UnknownKeyword */,
+        128 /* AnyKeyword */,
+        127 /* AssertsKeyword */,
+        155 /* BigIntKeyword */,
+        131 /* BooleanKeyword */,
+        94 /* FalseKeyword */,
+        138 /* KeyOfKeyword */,
+        141 /* NeverKeyword */,
+        103 /* NullKeyword */,
+        144 /* NumberKeyword */,
+        145 /* ObjectKeyword */,
+        142 /* ReadonlyKeyword */,
+        147 /* StringKeyword */,
+        148 /* SymbolKeyword */,
+        109 /* TrueKeyword */,
+        113 /* VoidKeyword */,
+        150 /* UndefinedKeyword */,
+        151 /* UniqueKeyword */,
+        152 /* UnknownKeyword */,
     ];
     function isTypeKeyword(kind) {
         return ts.contains(ts.typeKeywords, kind);
     }
     ts.isTypeKeyword = isTypeKeyword;
     function isTypeKeywordToken(node) {
-        return node.kind === 145 /* TypeKeyword */;
+        return node.kind === 149 /* TypeKeyword */;
     }
     ts.isTypeKeywordToken = isTypeKeywordToken;
     /** True if the symbol is for an external module, as opposed to a namespace. */
@@ -109809,7 +115387,7 @@ var ts;
     }
     ts.skipConstraint = skipConstraint;
     function getNameFromPropertyName(name) {
-        return name.kind === 154 /* ComputedPropertyName */
+        return name.kind === 158 /* ComputedPropertyName */
             // treat computed property names where expression is string/numeric literal as just string/numeric literal
             ? ts.isStringOrNumericLiteralLike(name.expression) ? name.expression.text : undefined
             : ts.isPrivateIdentifier(name) ? ts.idText(name) : ts.getTextOfIdentifierOrLiteral(name);
@@ -109835,12 +115413,12 @@ var ts;
             getCurrentDirectory: function () { return host.getCurrentDirectory(); },
             readFile: ts.maybeBind(host, host.readFile),
             useCaseSensitiveFileNames: ts.maybeBind(host, host.useCaseSensitiveFileNames),
-            getProbableSymlinks: ts.maybeBind(host, host.getProbableSymlinks) || (function () { return program.getProbableSymlinks(); }),
+            getSymlinkCache: ts.maybeBind(host, host.getSymlinkCache) || program.getSymlinkCache,
             getGlobalTypingsCacheLocation: ts.maybeBind(host, host.getGlobalTypingsCacheLocation),
             getSourceFiles: function () { return program.getSourceFiles(); },
             redirectTargetsMap: program.redirectTargetsMap,
             getProjectReferenceRedirect: function (fileName) { return program.getProjectReferenceRedirect(fileName); },
-            isSourceOfProjectReferenceRedirect: function (fileName) { return program.isSourceOfProjectReferenceRedirect(fileName); },
+            isSourceOfProjectReferenceRedirect: function (fileName) { return program.isSourceOfProjectReferenceRedirect(fileName); }
         };
     }
     ts.createModuleSpecifierResolutionHost = createModuleSpecifierResolutionHost;
@@ -109853,15 +115431,15 @@ var ts;
     }
     ts.makeImportIfNecessary = makeImportIfNecessary;
     function makeImport(defaultImport, namedImports, moduleSpecifier, quotePreference, isTypeOnly) {
-        return ts.createImportDeclaration(
+        return ts.factory.createImportDeclaration(
         /*decorators*/ undefined, 
         /*modifiers*/ undefined, defaultImport || namedImports
-            ? ts.createImportClause(defaultImport, namedImports && namedImports.length ? ts.createNamedImports(namedImports) : undefined, isTypeOnly)
+            ? ts.factory.createImportClause(!!isTypeOnly, defaultImport, namedImports && namedImports.length ? ts.factory.createNamedImports(namedImports) : undefined)
             : undefined, typeof moduleSpecifier === "string" ? makeStringLiteral(moduleSpecifier, quotePreference) : moduleSpecifier);
     }
     ts.makeImport = makeImport;
     function makeStringLiteral(text, quotePreference) {
-        return ts.createLiteral(text, quotePreference === 0 /* Single */);
+        return ts.factory.createStringLiteral(text, quotePreference === 0 /* Single */);
     }
     ts.makeStringLiteral = makeStringLiteral;
     var QuotePreference;
@@ -109878,7 +115456,9 @@ var ts;
             return preferences.quotePreference === "single" ? 0 /* Single */ : 1 /* Double */;
         }
         else {
-            var firstModuleSpecifier = sourceFile.imports && ts.find(sourceFile.imports, ts.isStringLiteral);
+            // ignore synthetic import added when importHelpers: true
+            var firstModuleSpecifier = sourceFile.imports &&
+                ts.find(sourceFile.imports, function (n) { return ts.isStringLiteral(n) && !ts.nodeIsSynthesized(n.parent); });
             return firstModuleSpecifier ? quotePreferenceFromString(firstModuleSpecifier, sourceFile) : 1 /* Double */;
         }
     }
@@ -109902,7 +115482,7 @@ var ts;
         }
         return ts.firstDefined(symbol.declarations, function (decl) {
             var name = ts.getNameOfDeclaration(decl);
-            return name && name.kind === 75 /* Identifier */ ? name.escapedText : undefined;
+            return name && name.kind === 78 /* Identifier */ ? name.escapedText : undefined;
         });
     }
     ts.symbolEscapedNameNoDefault = symbolEscapedNameNoDefault;
@@ -109927,7 +115507,7 @@ var ts;
      *                                The value of previousIterationSymbol is undefined when the function is first called.
      */
     function getPropertySymbolsFromBaseTypes(symbol, propertyName, checker, cb) {
-        var seen = ts.createMap();
+        var seen = new ts.Map();
         return recur(symbol);
         function recur(symbol) {
             // Use `addToSeen` to ensure we don't infinitely recurse in this situation:
@@ -109970,21 +115550,36 @@ var ts;
     ts.findModifier = findModifier;
     function insertImports(changes, sourceFile, imports, blankLineBetween) {
         var decl = ts.isArray(imports) ? imports[0] : imports;
-        var importKindPredicate = decl.kind === 225 /* VariableStatement */ ? ts.isRequireVariableDeclarationStatement : ts.isAnyImportSyntax;
-        var lastImportDeclaration = ts.findLast(sourceFile.statements, function (statement) { return importKindPredicate(statement); });
-        if (lastImportDeclaration) {
-            if (ts.isArray(imports)) {
-                changes.insertNodesAfter(sourceFile, lastImportDeclaration, imports);
-            }
-            else {
-                changes.insertNodeAfter(sourceFile, lastImportDeclaration, imports);
+        var importKindPredicate = decl.kind === 232 /* VariableStatement */ ? ts.isRequireVariableStatement : ts.isAnyImportSyntax;
+        var existingImportStatements = ts.filter(sourceFile.statements, importKindPredicate);
+        var sortedNewImports = ts.isArray(imports) ? ts.stableSort(imports, ts.OrganizeImports.compareImportsOrRequireStatements) : [imports];
+        if (!existingImportStatements.length) {
+            changes.insertNodesAtTopOfFile(sourceFile, sortedNewImports, blankLineBetween);
+        }
+        else if (existingImportStatements && ts.OrganizeImports.importsAreSorted(existingImportStatements)) {
+            for (var _i = 0, sortedNewImports_1 = sortedNewImports; _i < sortedNewImports_1.length; _i++) {
+                var newImport = sortedNewImports_1[_i];
+                var insertionIndex = ts.OrganizeImports.getImportDeclarationInsertionIndex(existingImportStatements, newImport);
+                if (insertionIndex === 0) {
+                    // If the first import is top-of-file, insert after the leading comment which is likely the header.
+                    var options = existingImportStatements[0] === sourceFile.statements[0] ?
+                        { leadingTriviaOption: ts.textChanges.LeadingTriviaOption.Exclude } : {};
+                    changes.insertNodeBefore(sourceFile, existingImportStatements[0], newImport, /*blankLineBetween*/ false, options);
+                }
+                else {
+                    var prevImport = existingImportStatements[insertionIndex - 1];
+                    changes.insertNodeAfter(sourceFile, prevImport, newImport);
+                }
             }
         }
-        else if (ts.isArray(imports)) {
-            changes.insertNodesAtTopOfFile(sourceFile, imports, blankLineBetween);
-        }
         else {
-            changes.insertNodeAtTopOfFile(sourceFile, imports, blankLineBetween);
+            var lastExistingImport = ts.lastOrUndefined(existingImportStatements);
+            if (lastExistingImport) {
+                changes.insertNodesAfter(sourceFile, lastExistingImport, sortedNewImports);
+            }
+            else {
+                changes.insertNodesAtTopOfFile(sourceFile, sortedNewImports, blankLineBetween);
+            }
         }
     }
     ts.insertImports = insertImports;
@@ -110020,11 +115615,20 @@ var ts;
         return undefined;
     }
     ts.forEachUnique = forEachUnique;
+    function isTextWhiteSpaceLike(text, startPos, endPos) {
+        for (var i = startPos; i < endPos; i++) {
+            if (!ts.isWhiteSpaceLike(text.charCodeAt(i))) {
+                return false;
+            }
+        }
+        return true;
+    }
+    ts.isTextWhiteSpaceLike = isTextWhiteSpaceLike;
     // #endregion
     // Display-part writer helpers
     // #region
     function isFirstDeclarationOfSymbolParameter(symbol) {
-        return symbol.declarations && symbol.declarations.length > 0 && symbol.declarations[0].kind === 156 /* Parameter */;
+        return symbol.declarations && symbol.declarations.length > 0 && symbol.declarations[0].kind === 160 /* Parameter */;
     }
     ts.isFirstDeclarationOfSymbolParameter = isFirstDeclarationOfSymbolParameter;
     var displayPartWriter = getDisplayPartWriter();
@@ -110248,19 +115852,10 @@ var ts;
         return !!location.parent && ts.isImportOrExportSpecifier(location.parent) && location.parent.propertyName === location;
     }
     ts.isImportOrExportSpecifierName = isImportOrExportSpecifierName;
-    function scriptKindIs(fileName, host) {
-        var scriptKinds = [];
-        for (var _i = 2; _i < arguments.length; _i++) {
-            scriptKinds[_i - 2] = arguments[_i];
-        }
-        var scriptKind = getScriptKind(fileName, host);
-        return ts.some(scriptKinds, function (k) { return k === scriptKind; });
-    }
-    ts.scriptKindIs = scriptKindIs;
     function getScriptKind(fileName, host) {
         // First check to see if the script kind was specified by the host. Chances are the host
         // may override the default script kind for the file extension.
-        return ts.ensureScriptKind(fileName, host && host.getScriptKind && host.getScriptKind(fileName));
+        return ts.ensureScriptKind(fileName, host.getScriptKind && host.getScriptKind(fileName));
     }
     ts.getScriptKind = getScriptKind;
     function getSymbolTarget(symbol, checker) {
@@ -110314,46 +115909,28 @@ var ts;
         return clone;
     }
     ts.getSynthesizedDeepClone = getSynthesizedDeepClone;
-    function getSynthesizedDeepCloneWithRenames(node, includeTrivia, renameMap, checker, callback) {
-        if (includeTrivia === void 0) { includeTrivia = true; }
-        var clone;
-        if (renameMap && checker && ts.isBindingElement(node) && ts.isIdentifier(node.name) && ts.isObjectBindingPattern(node.parent)) {
-            var symbol = checker.getSymbolAtLocation(node.name);
-            var renameInfo = symbol && renameMap.get(String(ts.getSymbolId(symbol)));
-            if (renameInfo && renameInfo.text !== (node.name || node.propertyName).getText()) {
-                clone = ts.setOriginalNode(ts.createBindingElement(node.dotDotDotToken, node.propertyName || node.name, renameInfo, node.initializer), node);
-            }
-        }
-        else if (renameMap && checker && ts.isIdentifier(node)) {
-            var symbol = checker.getSymbolAtLocation(node);
-            var renameInfo = symbol && renameMap.get(String(ts.getSymbolId(symbol)));
-            if (renameInfo) {
-                clone = ts.setOriginalNode(ts.createIdentifier(renameInfo.text), node);
-            }
+    function getSynthesizedDeepCloneWithReplacements(node, includeTrivia, replaceNode) {
+        var clone = replaceNode(node);
+        if (clone) {
+            ts.setOriginalNode(clone, node);
         }
-        if (!clone) {
-            clone = getSynthesizedDeepCloneWorker(node, renameMap, checker, callback);
+        else {
+            clone = getSynthesizedDeepCloneWorker(node, replaceNode);
         }
         if (clone && !includeTrivia)
             suppressLeadingAndTrailingTrivia(clone);
-        if (callback && clone)
-            callback(node, clone);
         return clone;
     }
-    ts.getSynthesizedDeepCloneWithRenames = getSynthesizedDeepCloneWithRenames;
-    function getSynthesizedDeepCloneWorker(node, renameMap, checker, callback) {
-        var visited = (renameMap || checker || callback) ?
-            ts.visitEachChild(node, wrapper, ts.nullTransformationContext) :
+    ts.getSynthesizedDeepCloneWithReplacements = getSynthesizedDeepCloneWithReplacements;
+    function getSynthesizedDeepCloneWorker(node, replaceNode) {
+        var visited = replaceNode ?
+            ts.visitEachChild(node, function (n) { return getSynthesizedDeepCloneWithReplacements(n, /*includeTrivia*/ true, replaceNode); }, ts.nullTransformationContext) :
             ts.visitEachChild(node, getSynthesizedDeepClone, ts.nullTransformationContext);
         if (visited === node) {
             // This only happens for leaf nodes - internal nodes always see their children change.
-            var clone_1 = ts.getSynthesizedClone(node);
-            if (ts.isStringLiteral(clone_1)) {
-                clone_1.textSourceNode = node;
-            }
-            else if (ts.isNumericLiteral(clone_1)) {
-                clone_1.numericLiteralFlags = node.numericLiteralFlags;
-            }
+            var clone_1 = ts.isStringLiteral(node) ? ts.setOriginalNode(ts.factory.createStringLiteralFromNode(node), node) :
+                ts.isNumericLiteral(node) ? ts.setOriginalNode(ts.factory.createNumericLiteral(node.text, node.numericLiteralFlags), node) :
+                    ts.factory.cloneNode(node);
             return ts.setTextRange(clone_1, node);
         }
         // PERF: As an optimization, rather than calling getSynthesizedClone, we'll update
@@ -110361,15 +115938,16 @@ var ts;
         // would have made.
         visited.parent = undefined;
         return visited;
-        function wrapper(node) {
-            return getSynthesizedDeepCloneWithRenames(node, /*includeTrivia*/ true, renameMap, checker, callback);
-        }
     }
     function getSynthesizedDeepClones(nodes, includeTrivia) {
         if (includeTrivia === void 0) { includeTrivia = true; }
-        return nodes && ts.createNodeArray(nodes.map(function (n) { return getSynthesizedDeepClone(n, includeTrivia); }), nodes.hasTrailingComma);
+        return nodes && ts.factory.createNodeArray(nodes.map(function (n) { return getSynthesizedDeepClone(n, includeTrivia); }), nodes.hasTrailingComma);
     }
     ts.getSynthesizedDeepClones = getSynthesizedDeepClones;
+    function getSynthesizedDeepClonesWithReplacements(nodes, includeTrivia, replaceNode) {
+        return ts.factory.createNodeArray(nodes.map(function (n) { return getSynthesizedDeepCloneWithReplacements(n, includeTrivia, replaceNode); }), nodes.hasTrailingComma);
+    }
+    ts.getSynthesizedDeepClonesWithReplacements = getSynthesizedDeepClonesWithReplacements;
     /**
      * Sets EmitFlags to suppress leading and trailing trivia on the node.
      */
@@ -110514,35 +116092,26 @@ var ts;
     function getContextualTypeFromParent(node, checker) {
         var parent = node.parent;
         switch (parent.kind) {
-            case 197 /* NewExpression */:
+            case 204 /* NewExpression */:
                 return checker.getContextualType(parent);
-            case 209 /* BinaryExpression */: {
+            case 216 /* BinaryExpression */: {
                 var _a = parent, left = _a.left, operatorToken = _a.operatorToken, right = _a.right;
                 return isEqualityOperatorKind(operatorToken.kind)
                     ? checker.getTypeAtLocation(node === right ? left : right)
                     : checker.getContextualType(node);
             }
-            case 277 /* CaseClause */:
+            case 284 /* CaseClause */:
                 return parent.expression === node ? getSwitchedType(parent, checker) : undefined;
             default:
                 return checker.getContextualType(node);
         }
     }
     ts.getContextualTypeFromParent = getContextualTypeFromParent;
-    function quote(text, preferences) {
+    function quote(sourceFile, preferences, text) {
         // Editors can pass in undefined or empty string - we want to infer the preference in those cases.
-        var quotePreference = preferences.quotePreference || "auto";
+        var quotePreference = getQuotePreference(sourceFile, preferences);
         var quoted = JSON.stringify(text);
-        switch (quotePreference) {
-            // TODO use getQuotePreference to infer the actual quote style.
-            case "auto":
-            case "double":
-                return quoted;
-            case "single":
-                return "'" + ts.stripQuotes(quoted).replace("'", "\\'").replace('\\"', '"') + "'";
-            default:
-                return ts.Debug.assertNever(quotePreference);
-        }
+        return quotePreference === 0 /* Single */ ? "'" + ts.stripQuotes(quoted).replace("'", "\\'").replace('\\"', '"') + "'" : quoted;
     }
     ts.quote = quote;
     function isEqualityOperatorKind(kind) {
@@ -110561,8 +116130,8 @@ var ts;
         switch (node.kind) {
             case 10 /* StringLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
-            case 211 /* TemplateExpression */:
-            case 198 /* TaggedTemplateExpression */:
+            case 218 /* TemplateExpression */:
+            case 205 /* TaggedTemplateExpression */:
                 return true;
             default:
                 return false;
@@ -110582,7 +116151,7 @@ var ts;
         var checker = program.getTypeChecker();
         var typeIsAccessible = true;
         var notAccessible = function () { typeIsAccessible = false; };
-        var res = checker.typeToTypeNode(type, enclosingScope, /*flags*/ undefined, {
+        var res = checker.typeToTypeNode(type, enclosingScope, 1 /* NoTruncation */, {
             trackSymbol: function (symbol, declaration, meaning) {
                 typeIsAccessible = typeIsAccessible && checker.isSymbolAccessible(symbol, declaration, meaning, /*shouldComputeAliasToMarkVisible*/ false).accessibility === 0 /* Accessible */;
             },
@@ -110595,41 +116164,41 @@ var ts;
     }
     ts.getTypeNodeIfAccessible = getTypeNodeIfAccessible;
     function syntaxRequiresTrailingCommaOrSemicolonOrASI(kind) {
-        return kind === 165 /* CallSignature */
-            || kind === 166 /* ConstructSignature */
-            || kind === 167 /* IndexSignature */
-            || kind === 158 /* PropertySignature */
-            || kind === 160 /* MethodSignature */;
+        return kind === 169 /* CallSignature */
+            || kind === 170 /* ConstructSignature */
+            || kind === 171 /* IndexSignature */
+            || kind === 162 /* PropertySignature */
+            || kind === 164 /* MethodSignature */;
     }
     ts.syntaxRequiresTrailingCommaOrSemicolonOrASI = syntaxRequiresTrailingCommaOrSemicolonOrASI;
     function syntaxRequiresTrailingFunctionBlockOrSemicolonOrASI(kind) {
-        return kind === 244 /* FunctionDeclaration */
-            || kind === 162 /* Constructor */
-            || kind === 161 /* MethodDeclaration */
-            || kind === 163 /* GetAccessor */
-            || kind === 164 /* SetAccessor */;
+        return kind === 251 /* FunctionDeclaration */
+            || kind === 166 /* Constructor */
+            || kind === 165 /* MethodDeclaration */
+            || kind === 167 /* GetAccessor */
+            || kind === 168 /* SetAccessor */;
     }
     ts.syntaxRequiresTrailingFunctionBlockOrSemicolonOrASI = syntaxRequiresTrailingFunctionBlockOrSemicolonOrASI;
     function syntaxRequiresTrailingModuleBlockOrSemicolonOrASI(kind) {
-        return kind === 249 /* ModuleDeclaration */;
+        return kind === 256 /* ModuleDeclaration */;
     }
     ts.syntaxRequiresTrailingModuleBlockOrSemicolonOrASI = syntaxRequiresTrailingModuleBlockOrSemicolonOrASI;
     function syntaxRequiresTrailingSemicolonOrASI(kind) {
-        return kind === 225 /* VariableStatement */
-            || kind === 226 /* ExpressionStatement */
-            || kind === 228 /* DoStatement */
-            || kind === 233 /* ContinueStatement */
-            || kind === 234 /* BreakStatement */
-            || kind === 235 /* ReturnStatement */
-            || kind === 239 /* ThrowStatement */
-            || kind === 241 /* DebuggerStatement */
-            || kind === 159 /* PropertyDeclaration */
-            || kind === 247 /* TypeAliasDeclaration */
-            || kind === 254 /* ImportDeclaration */
-            || kind === 253 /* ImportEqualsDeclaration */
-            || kind === 260 /* ExportDeclaration */
-            || kind === 252 /* NamespaceExportDeclaration */
-            || kind === 259 /* ExportAssignment */;
+        return kind === 232 /* VariableStatement */
+            || kind === 233 /* ExpressionStatement */
+            || kind === 235 /* DoStatement */
+            || kind === 240 /* ContinueStatement */
+            || kind === 241 /* BreakStatement */
+            || kind === 242 /* ReturnStatement */
+            || kind === 246 /* ThrowStatement */
+            || kind === 248 /* DebuggerStatement */
+            || kind === 163 /* PropertyDeclaration */
+            || kind === 254 /* TypeAliasDeclaration */
+            || kind === 261 /* ImportDeclaration */
+            || kind === 260 /* ImportEqualsDeclaration */
+            || kind === 267 /* ExportDeclaration */
+            || kind === 259 /* NamespaceExportDeclaration */
+            || kind === 266 /* ExportAssignment */;
     }
     ts.syntaxRequiresTrailingSemicolonOrASI = syntaxRequiresTrailingSemicolonOrASI;
     ts.syntaxMayBeASICandidate = ts.or(syntaxRequiresTrailingCommaOrSemicolonOrASI, syntaxRequiresTrailingFunctionBlockOrSemicolonOrASI, syntaxRequiresTrailingModuleBlockOrSemicolonOrASI, syntaxRequiresTrailingSemicolonOrASI);
@@ -110659,7 +116228,7 @@ var ts;
             return false;
         }
         // See comment in parser’s `parseDoStatement`
-        if (node.kind === 228 /* DoStatement */) {
+        if (node.kind === 235 /* DoStatement */) {
             return true;
         }
         var topNode = ts.findAncestor(node, function (ancestor) { return !ancestor.parent; });
@@ -110791,24 +116360,22 @@ var ts;
             return undefined;
         }
         var dependencyKeys = ["dependencies", "devDependencies", "optionalDependencies", "peerDependencies"];
-        var stringContent = host.readFile(fileName);
-        if (!stringContent)
-            return undefined;
+        var stringContent = host.readFile(fileName) || "";
         var content = tryParseJson(stringContent);
-        if (!content)
-            return false;
         var info = {};
-        for (var _i = 0, dependencyKeys_1 = dependencyKeys; _i < dependencyKeys_1.length; _i++) {
-            var key = dependencyKeys_1[_i];
-            var dependencies = content[key];
-            if (!dependencies) {
-                continue;
-            }
-            var dependencyMap = ts.createMap();
-            for (var packageName in dependencies) {
-                dependencyMap.set(packageName, dependencies[packageName]);
+        if (content) {
+            for (var _i = 0, dependencyKeys_1 = dependencyKeys; _i < dependencyKeys_1.length; _i++) {
+                var key = dependencyKeys_1[_i];
+                var dependencies = content[key];
+                if (!dependencies) {
+                    continue;
+                }
+                var dependencyMap = new ts.Map();
+                for (var packageName in dependencies) {
+                    dependencyMap.set(packageName, dependencies[packageName]);
+                }
+                info[key] = dependencyMap;
             }
-            info[key] = dependencyMap;
         }
         var dependencyGroups = [
             [1 /* Dependencies */, info.dependencies],
@@ -110816,7 +116383,7 @@ var ts;
             [8 /* OptionalDependencies */, info.optionalDependencies],
             [4 /* PeerDependencies */, info.peerDependencies],
         ];
-        return __assign(__assign({}, info), { fileName: fileName,
+        return __assign(__assign({}, info), { parseable: !!content, fileName: fileName,
             get: get,
             has: function (dependencyName, inGroups) {
                 return !!get(dependencyName, inGroups);
@@ -110914,11 +116481,21 @@ var ts;
         if (symbol.escapedName === "export=" /* ExportEquals */ || symbol.escapedName === "default" /* Default */) {
             // Name of "export default foo;" is "foo". Name of "export default 0" is the filename converted to camelCase.
             return ts.firstDefined(symbol.declarations, function (d) { return ts.isExportAssignment(d) && ts.isIdentifier(d.expression) ? d.expression.text : undefined; })
-                || ts.codefix.moduleSymbolToValidIdentifier(ts.Debug.checkDefined(symbol.parent), scriptTarget);
+                || ts.codefix.moduleSymbolToValidIdentifier(getSymbolParentOrFail(symbol), scriptTarget);
         }
         return symbol.name;
     }
     ts.getNameForExportedSymbol = getNameForExportedSymbol;
+    function getSymbolParentOrFail(symbol) {
+        var _a;
+        return ts.Debug.checkDefined(symbol.parent, "Symbol parent was undefined. Flags: " + ts.Debug.formatSymbolFlags(symbol.flags) + ". " +
+            ("Declarations: " + ((_a = symbol.declarations) === null || _a === void 0 ? void 0 : _a.map(function (d) {
+                var kind = ts.Debug.formatSyntaxKind(d.kind);
+                var inJS = ts.isInJSFile(d);
+                var expression = d.expression;
+                return (inJS ? "[JS]" : "") + kind + (expression ? " (expression: " + ts.Debug.formatSyntaxKind(expression.kind) + ")" : "");
+            }).join(", ")) + "."));
+    }
     /**
      * Useful to check whether a string contains another string at a specific index
      * without allocating another string or traversing the entire contents of the outer string.
@@ -111056,7 +116633,7 @@ var ts;
                         }
                         break;
                     case 29 /* LessThanToken */:
-                        if (lastNonTriviaToken === 75 /* Identifier */) {
+                        if (lastNonTriviaToken === 78 /* Identifier */) {
                             // Could be the start of something generic.  Keep track of that by bumping
                             // up the current count of generic contexts we may be in.
                             angleBracketStack++;
@@ -111069,16 +116646,16 @@ var ts;
                             angleBracketStack--;
                         }
                         break;
-                    case 125 /* AnyKeyword */:
-                    case 143 /* StringKeyword */:
-                    case 140 /* NumberKeyword */:
-                    case 128 /* BooleanKeyword */:
-                    case 144 /* SymbolKeyword */:
+                    case 128 /* AnyKeyword */:
+                    case 147 /* StringKeyword */:
+                    case 144 /* NumberKeyword */:
+                    case 131 /* BooleanKeyword */:
+                    case 148 /* SymbolKeyword */:
                         if (angleBracketStack > 0 && !syntacticClassifierAbsent) {
                             // If it looks like we're could be in something generic, don't classify this
                             // as a keyword.  We may just get overwritten by the syntactic classifier,
                             // causing a noisy experience for the user.
-                            token = 75 /* Identifier */;
+                            token = 78 /* Identifier */;
                         }
                         break;
                     case 15 /* TemplateHead */:
@@ -111117,14 +116694,14 @@ var ts;
                             break;
                         }
                         if (lastNonTriviaToken === 24 /* DotToken */) {
-                            token = 75 /* Identifier */;
+                            token = 78 /* Identifier */;
                         }
                         else if (ts.isKeyword(lastNonTriviaToken) && ts.isKeyword(token) && !canFollow(lastNonTriviaToken, token)) {
                             // We have two keywords in a row.  Only treat the second as a keyword if
                             // it's a sequence that could legally occur in the language.  Otherwise
                             // treat it as an identifier.  This way, if someone writes "private var"
                             // we recognize that 'var' is actually an identifier here.
-                            token = 75 /* Identifier */;
+                            token = 78 /* Identifier */;
                         }
                 }
             }
@@ -111138,19 +116715,19 @@ var ts;
     /// we have a series of divide operator. this list allows us to be more accurate by ruling out
     /// locations where a regexp cannot exist.
     var noRegexTable = ts.arrayToNumericMap([
-        75 /* Identifier */,
+        78 /* Identifier */,
         10 /* StringLiteral */,
         8 /* NumericLiteral */,
         9 /* BigIntLiteral */,
         13 /* RegularExpressionLiteral */,
-        104 /* ThisKeyword */,
+        107 /* ThisKeyword */,
         45 /* PlusPlusToken */,
         46 /* MinusMinusToken */,
         21 /* CloseParenToken */,
         23 /* CloseBracketToken */,
         19 /* CloseBraceToken */,
-        106 /* TrueKeyword */,
-        91 /* FalseKeyword */,
+        109 /* TrueKeyword */,
+        94 /* FalseKeyword */,
     ], function (token) { return token; }, function () { return true; });
     function getNewEndOfLineState(scanner, token, lastOnTemplateStack) {
         switch (token) {
@@ -111262,10 +116839,10 @@ var ts;
             return true;
         }
         switch (keyword2) {
-            case 131 /* GetKeyword */:
-            case 142 /* SetKeyword */:
-            case 129 /* ConstructorKeyword */:
-            case 120 /* StaticKeyword */:
+            case 134 /* GetKeyword */:
+            case 146 /* SetKeyword */:
+            case 132 /* ConstructorKeyword */:
+            case 123 /* StaticKeyword */:
                 return true; // Allow things like "public get", "public constructor" and "public static".
             default:
                 return false; // Any other keyword following "public" is actually an identifier, not a real keyword.
@@ -111310,9 +116887,9 @@ var ts;
             case 31 /* GreaterThanToken */:
             case 32 /* LessThanEqualsToken */:
             case 33 /* GreaterThanEqualsToken */:
-            case 98 /* InstanceOfKeyword */:
-            case 97 /* InKeyword */:
-            case 123 /* AsKeyword */:
+            case 101 /* InstanceOfKeyword */:
+            case 100 /* InKeyword */:
+            case 126 /* AsKeyword */:
             case 34 /* EqualsEqualsToken */:
             case 35 /* ExclamationEqualsToken */:
             case 36 /* EqualsEqualsEqualsToken */:
@@ -111324,7 +116901,7 @@ var ts;
             case 56 /* BarBarToken */:
             case 73 /* BarEqualsToken */:
             case 72 /* AmpersandEqualsToken */:
-            case 74 /* CaretEqualsToken */:
+            case 77 /* CaretEqualsToken */:
             case 69 /* LessThanLessThanEqualsToken */:
             case 70 /* GreaterThanGreaterThanEqualsToken */:
             case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
@@ -111336,6 +116913,9 @@ var ts;
             case 62 /* EqualsToken */:
             case 27 /* CommaToken */:
             case 60 /* QuestionQuestionToken */:
+            case 74 /* BarBarEqualsToken */:
+            case 75 /* AmpersandAmpersandEqualsToken */:
+            case 76 /* QuestionQuestionEqualsToken */:
                 return true;
             default:
                 return false;
@@ -111361,7 +116941,7 @@ var ts;
         else if (isBinaryExpressionOperatorToken(token) || isPrefixUnaryExpressionOperatorToken(token)) {
             return 5 /* operator */;
         }
-        else if (token >= 18 /* FirstPunctuation */ && token <= 74 /* LastPunctuation */) {
+        else if (token >= 18 /* FirstPunctuation */ && token <= 77 /* LastPunctuation */) {
             return 10 /* punctuation */;
         }
         switch (token) {
@@ -111380,7 +116960,7 @@ var ts;
             case 5 /* WhitespaceTrivia */:
             case 4 /* NewLineTrivia */:
                 return 8 /* whiteSpace */;
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
             default:
                 if (ts.isTemplateLiteralKind(token)) {
                     return 6 /* stringLiteral */;
@@ -111405,10 +116985,13 @@ var ts;
         // That means we're calling back into the host around every 1.2k of the file we process.
         // Lib.d.ts has similar numbers.
         switch (kind) {
-            case 249 /* ModuleDeclaration */:
-            case 245 /* ClassDeclaration */:
-            case 246 /* InterfaceDeclaration */:
-            case 244 /* FunctionDeclaration */:
+            case 256 /* ModuleDeclaration */:
+            case 252 /* ClassDeclaration */:
+            case 253 /* InterfaceDeclaration */:
+            case 251 /* FunctionDeclaration */:
+            case 221 /* ClassExpression */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
                 cancellationToken.throwIfCancellationRequested();
         }
     }
@@ -111598,7 +117181,7 @@ var ts;
                 var docCommentAndDiagnostics = ts.parseIsolatedJSDocComment(sourceFile.text, start, width);
                 if (docCommentAndDiagnostics && docCommentAndDiagnostics.jsDoc) {
                     // TODO: This should be predicated on `token["kind"]` being compatible with `HasJSDoc["kind"]`
-                    docCommentAndDiagnostics.jsDoc.parent = token;
+                    ts.setParent(docCommentAndDiagnostics.jsDoc, token);
                     classifyJSDocComment(docCommentAndDiagnostics.jsDoc);
                     return;
                 }
@@ -111628,18 +117211,18 @@ var ts;
                     pushClassification(tag.tagName.pos, tag.tagName.end - tag.tagName.pos, 18 /* docCommentTagName */); // e.g. "param"
                     pos = tag.tagName.end;
                     switch (tag.kind) {
-                        case 317 /* JSDocParameterTag */:
+                        case 326 /* JSDocParameterTag */:
                             processJSDocParameterTag(tag);
                             break;
-                        case 321 /* JSDocTemplateTag */:
+                        case 330 /* JSDocTemplateTag */:
                             processJSDocTemplateTag(tag);
                             pos = tag.end;
                             break;
-                        case 320 /* JSDocTypeTag */:
+                        case 329 /* JSDocTypeTag */:
                             processElement(tag.typeExpression);
                             pos = tag.end;
                             break;
-                        case 318 /* JSDocReturnTag */:
+                        case 327 /* JSDocReturnTag */:
                             processElement(tag.typeExpression);
                             pos = tag.end;
                             break;
@@ -111790,22 +117373,22 @@ var ts;
         }
         function tryClassifyJsxElementName(token) {
             switch (token.parent && token.parent.kind) {
-                case 268 /* JsxOpeningElement */:
+                case 275 /* JsxOpeningElement */:
                     if (token.parent.tagName === token) {
                         return 19 /* jsxOpenTagName */;
                     }
                     break;
-                case 269 /* JsxClosingElement */:
+                case 276 /* JsxClosingElement */:
                     if (token.parent.tagName === token) {
                         return 20 /* jsxCloseTagName */;
                     }
                     break;
-                case 267 /* JsxSelfClosingElement */:
+                case 274 /* JsxSelfClosingElement */:
                     if (token.parent.tagName === token) {
                         return 21 /* jsxSelfClosingTagName */;
                     }
                     break;
-                case 273 /* JsxAttribute */:
+                case 280 /* JsxAttribute */:
                     if (token.parent.name === token) {
                         return 22 /* jsxAttribute */;
                     }
@@ -111834,17 +117417,17 @@ var ts;
                     var parent = token.parent;
                     if (tokenKind === 62 /* EqualsToken */) {
                         // the '=' in a variable declaration is special cased here.
-                        if (parent.kind === 242 /* VariableDeclaration */ ||
-                            parent.kind === 159 /* PropertyDeclaration */ ||
-                            parent.kind === 156 /* Parameter */ ||
-                            parent.kind === 273 /* JsxAttribute */) {
+                        if (parent.kind === 249 /* VariableDeclaration */ ||
+                            parent.kind === 163 /* PropertyDeclaration */ ||
+                            parent.kind === 160 /* Parameter */ ||
+                            parent.kind === 280 /* JsxAttribute */) {
                             return 5 /* operator */;
                         }
                     }
-                    if (parent.kind === 209 /* BinaryExpression */ ||
-                        parent.kind === 207 /* PrefixUnaryExpression */ ||
-                        parent.kind === 208 /* PostfixUnaryExpression */ ||
-                        parent.kind === 210 /* ConditionalExpression */) {
+                    if (parent.kind === 216 /* BinaryExpression */ ||
+                        parent.kind === 214 /* PrefixUnaryExpression */ ||
+                        parent.kind === 215 /* PostfixUnaryExpression */ ||
+                        parent.kind === 217 /* ConditionalExpression */) {
                         return 5 /* operator */;
                     }
                 }
@@ -111857,7 +117440,7 @@ var ts;
                 return 25 /* bigintLiteral */;
             }
             else if (tokenKind === 10 /* StringLiteral */) {
-                return token && token.parent.kind === 273 /* JsxAttribute */ ? 24 /* jsxAttributeStringLiteralValue */ : 6 /* stringLiteral */;
+                return token && token.parent.kind === 280 /* JsxAttribute */ ? 24 /* jsxAttributeStringLiteralValue */ : 6 /* stringLiteral */;
             }
             else if (tokenKind === 13 /* RegularExpressionLiteral */) {
                 // TODO: we should get another classification type for these literals.
@@ -111870,35 +117453,35 @@ var ts;
             else if (tokenKind === 11 /* JsxText */) {
                 return 23 /* jsxText */;
             }
-            else if (tokenKind === 75 /* Identifier */) {
+            else if (tokenKind === 78 /* Identifier */) {
                 if (token) {
                     switch (token.parent.kind) {
-                        case 245 /* ClassDeclaration */:
+                        case 252 /* ClassDeclaration */:
                             if (token.parent.name === token) {
                                 return 11 /* className */;
                             }
                             return;
-                        case 155 /* TypeParameter */:
+                        case 159 /* TypeParameter */:
                             if (token.parent.name === token) {
                                 return 15 /* typeParameterName */;
                             }
                             return;
-                        case 246 /* InterfaceDeclaration */:
+                        case 253 /* InterfaceDeclaration */:
                             if (token.parent.name === token) {
                                 return 13 /* interfaceName */;
                             }
                             return;
-                        case 248 /* EnumDeclaration */:
+                        case 255 /* EnumDeclaration */:
                             if (token.parent.name === token) {
                                 return 12 /* enumName */;
                             }
                             return;
-                        case 249 /* ModuleDeclaration */:
+                        case 256 /* ModuleDeclaration */:
                             if (token.parent.name === token) {
                                 return 14 /* moduleName */;
                             }
                             return;
-                        case 156 /* Parameter */:
+                        case 160 /* Parameter */:
                             if (token.parent.name === token) {
                                 return ts.isThisIdentifier(token) ? 3 /* keyword */ : 17 /* parameterName */;
                             }
@@ -111927,6 +117510,254 @@ var ts;
     }
     ts.getEncodedSyntacticClassifications = getEncodedSyntacticClassifications;
 })(ts || (ts = {}));
+/** @internal */
+var ts;
+(function (ts) {
+    var classifier;
+    (function (classifier) {
+        var v2020;
+        (function (v2020) {
+            var TokenEncodingConsts;
+            (function (TokenEncodingConsts) {
+                TokenEncodingConsts[TokenEncodingConsts["typeOffset"] = 8] = "typeOffset";
+                TokenEncodingConsts[TokenEncodingConsts["modifierMask"] = 255] = "modifierMask";
+            })(TokenEncodingConsts = v2020.TokenEncodingConsts || (v2020.TokenEncodingConsts = {}));
+            var TokenType;
+            (function (TokenType) {
+                TokenType[TokenType["class"] = 0] = "class";
+                TokenType[TokenType["enum"] = 1] = "enum";
+                TokenType[TokenType["interface"] = 2] = "interface";
+                TokenType[TokenType["namespace"] = 3] = "namespace";
+                TokenType[TokenType["typeParameter"] = 4] = "typeParameter";
+                TokenType[TokenType["type"] = 5] = "type";
+                TokenType[TokenType["parameter"] = 6] = "parameter";
+                TokenType[TokenType["variable"] = 7] = "variable";
+                TokenType[TokenType["enumMember"] = 8] = "enumMember";
+                TokenType[TokenType["property"] = 9] = "property";
+                TokenType[TokenType["function"] = 10] = "function";
+                TokenType[TokenType["member"] = 11] = "member";
+            })(TokenType = v2020.TokenType || (v2020.TokenType = {}));
+            var TokenModifier;
+            (function (TokenModifier) {
+                TokenModifier[TokenModifier["declaration"] = 0] = "declaration";
+                TokenModifier[TokenModifier["static"] = 1] = "static";
+                TokenModifier[TokenModifier["async"] = 2] = "async";
+                TokenModifier[TokenModifier["readonly"] = 3] = "readonly";
+                TokenModifier[TokenModifier["defaultLibrary"] = 4] = "defaultLibrary";
+                TokenModifier[TokenModifier["local"] = 5] = "local";
+            })(TokenModifier = v2020.TokenModifier || (v2020.TokenModifier = {}));
+            /** This is mainly used internally for testing */
+            function getSemanticClassifications(program, cancellationToken, sourceFile, span) {
+                var classifications = getEncodedSemanticClassifications(program, cancellationToken, sourceFile, span);
+                ts.Debug.assert(classifications.spans.length % 3 === 0);
+                var dense = classifications.spans;
+                var result = [];
+                for (var i = 0; i < dense.length; i += 3) {
+                    result.push({
+                        textSpan: ts.createTextSpan(dense[i], dense[i + 1]),
+                        classificationType: dense[i + 2]
+                    });
+                }
+                return result;
+            }
+            v2020.getSemanticClassifications = getSemanticClassifications;
+            function getEncodedSemanticClassifications(program, cancellationToken, sourceFile, span) {
+                return {
+                    spans: getSemanticTokens(program, sourceFile, span, cancellationToken),
+                    endOfLineState: 0 /* None */
+                };
+            }
+            v2020.getEncodedSemanticClassifications = getEncodedSemanticClassifications;
+            function getSemanticTokens(program, sourceFile, span, cancellationToken) {
+                var resultTokens = [];
+                var collector = function (node, typeIdx, modifierSet) {
+                    resultTokens.push(node.getStart(sourceFile), node.getWidth(sourceFile), ((typeIdx + 1) << 8 /* typeOffset */) + modifierSet);
+                };
+                if (program && sourceFile) {
+                    collectTokens(program, sourceFile, span, collector, cancellationToken);
+                }
+                return resultTokens;
+            }
+            function collectTokens(program, sourceFile, span, collector, cancellationToken) {
+                var typeChecker = program.getTypeChecker();
+                var inJSXElement = false;
+                function visit(node) {
+                    switch (node.kind) {
+                        case 256 /* ModuleDeclaration */:
+                        case 252 /* ClassDeclaration */:
+                        case 253 /* InterfaceDeclaration */:
+                        case 251 /* FunctionDeclaration */:
+                        case 221 /* ClassExpression */:
+                        case 208 /* FunctionExpression */:
+                        case 209 /* ArrowFunction */:
+                            cancellationToken.throwIfCancellationRequested();
+                    }
+                    if (!node || !ts.textSpanIntersectsWith(span, node.pos, node.getFullWidth()) || node.getFullWidth() === 0) {
+                        return;
+                    }
+                    var prevInJSXElement = inJSXElement;
+                    if (ts.isJsxElement(node) || ts.isJsxSelfClosingElement(node)) {
+                        inJSXElement = true;
+                    }
+                    if (ts.isJsxExpression(node)) {
+                        inJSXElement = false;
+                    }
+                    if (ts.isIdentifier(node) && !inJSXElement && !inImportClause(node)) {
+                        var symbol = typeChecker.getSymbolAtLocation(node);
+                        if (symbol) {
+                            if (symbol.flags & 2097152 /* Alias */) {
+                                symbol = typeChecker.getAliasedSymbol(symbol);
+                            }
+                            var typeIdx = classifySymbol(symbol, ts.getMeaningFromLocation(node));
+                            if (typeIdx !== undefined) {
+                                var modifierSet = 0;
+                                if (node.parent) {
+                                    var parentIsDeclaration = (ts.isBindingElement(node.parent) || tokenFromDeclarationMapping.get(node.parent.kind) === typeIdx);
+                                    if (parentIsDeclaration && node.parent.name === node) {
+                                        modifierSet = 1 << 0 /* declaration */;
+                                    }
+                                }
+                                // property declaration in constructor
+                                if (typeIdx === 6 /* parameter */ && isRightSideOfQualifiedNameOrPropertyAccess(node)) {
+                                    typeIdx = 9 /* property */;
+                                }
+                                typeIdx = reclassifyByType(typeChecker, node, typeIdx);
+                                var decl = symbol.valueDeclaration;
+                                if (decl) {
+                                    var modifiers = ts.getCombinedModifierFlags(decl);
+                                    var nodeFlags = ts.getCombinedNodeFlags(decl);
+                                    if (modifiers & 32 /* Static */) {
+                                        modifierSet |= 1 << 1 /* static */;
+                                    }
+                                    if (modifiers & 256 /* Async */) {
+                                        modifierSet |= 1 << 2 /* async */;
+                                    }
+                                    if (typeIdx !== 0 /* class */ && typeIdx !== 2 /* interface */) {
+                                        if ((modifiers & 64 /* Readonly */) || (nodeFlags & 2 /* Const */) || (symbol.getFlags() & 8 /* EnumMember */)) {
+                                            modifierSet |= 1 << 3 /* readonly */;
+                                        }
+                                    }
+                                    if ((typeIdx === 7 /* variable */ || typeIdx === 10 /* function */) && isLocalDeclaration(decl, sourceFile)) {
+                                        modifierSet |= 1 << 5 /* local */;
+                                    }
+                                    if (program.isSourceFileDefaultLibrary(decl.getSourceFile())) {
+                                        modifierSet |= 1 << 4 /* defaultLibrary */;
+                                    }
+                                }
+                                else if (symbol.declarations && symbol.declarations.some(function (d) { return program.isSourceFileDefaultLibrary(d.getSourceFile()); })) {
+                                    modifierSet |= 1 << 4 /* defaultLibrary */;
+                                }
+                                collector(node, typeIdx, modifierSet);
+                            }
+                        }
+                    }
+                    ts.forEachChild(node, visit);
+                    inJSXElement = prevInJSXElement;
+                }
+                visit(sourceFile);
+            }
+            function classifySymbol(symbol, meaning) {
+                var flags = symbol.getFlags();
+                if (flags & 32 /* Class */) {
+                    return 0 /* class */;
+                }
+                else if (flags & 384 /* Enum */) {
+                    return 1 /* enum */;
+                }
+                else if (flags & 524288 /* TypeAlias */) {
+                    return 5 /* type */;
+                }
+                else if (flags & 64 /* Interface */) {
+                    if (meaning & 2 /* Type */) {
+                        return 2 /* interface */;
+                    }
+                }
+                else if (flags & 262144 /* TypeParameter */) {
+                    return 4 /* typeParameter */;
+                }
+                var decl = symbol.valueDeclaration || symbol.declarations && symbol.declarations[0];
+                if (decl && ts.isBindingElement(decl)) {
+                    decl = getDeclarationForBindingElement(decl);
+                }
+                return decl && tokenFromDeclarationMapping.get(decl.kind);
+            }
+            function reclassifyByType(typeChecker, node, typeIdx) {
+                // type based classifications
+                if (typeIdx === 7 /* variable */ || typeIdx === 9 /* property */ || typeIdx === 6 /* parameter */) {
+                    var type_1 = typeChecker.getTypeAtLocation(node);
+                    if (type_1) {
+                        var test = function (condition) {
+                            return condition(type_1) || type_1.isUnion() && type_1.types.some(condition);
+                        };
+                        if (typeIdx !== 6 /* parameter */ && test(function (t) { return t.getConstructSignatures().length > 0; })) {
+                            return 0 /* class */;
+                        }
+                        if (test(function (t) { return t.getCallSignatures().length > 0; }) && !test(function (t) { return t.getProperties().length > 0; }) || isExpressionInCallExpression(node)) {
+                            return typeIdx === 9 /* property */ ? 11 /* member */ : 10 /* function */;
+                        }
+                    }
+                }
+                return typeIdx;
+            }
+            function isLocalDeclaration(decl, sourceFile) {
+                if (ts.isBindingElement(decl)) {
+                    decl = getDeclarationForBindingElement(decl);
+                }
+                if (ts.isVariableDeclaration(decl)) {
+                    return (!ts.isSourceFile(decl.parent.parent.parent) || ts.isCatchClause(decl.parent)) && decl.getSourceFile() === sourceFile;
+                }
+                else if (ts.isFunctionDeclaration(decl)) {
+                    return !ts.isSourceFile(decl.parent) && decl.getSourceFile() === sourceFile;
+                }
+                return false;
+            }
+            function getDeclarationForBindingElement(element) {
+                while (true) {
+                    if (ts.isBindingElement(element.parent.parent)) {
+                        element = element.parent.parent;
+                    }
+                    else {
+                        return element.parent.parent;
+                    }
+                }
+            }
+            function inImportClause(node) {
+                var parent = node.parent;
+                return parent && (ts.isImportClause(parent) || ts.isImportSpecifier(parent) || ts.isNamespaceImport(parent));
+            }
+            function isExpressionInCallExpression(node) {
+                while (isRightSideOfQualifiedNameOrPropertyAccess(node)) {
+                    node = node.parent;
+                }
+                return ts.isCallExpression(node.parent) && node.parent.expression === node;
+            }
+            function isRightSideOfQualifiedNameOrPropertyAccess(node) {
+                return (ts.isQualifiedName(node.parent) && node.parent.right === node) || (ts.isPropertyAccessExpression(node.parent) && node.parent.name === node);
+            }
+            var tokenFromDeclarationMapping = new ts.Map([
+                [249 /* VariableDeclaration */, 7 /* variable */],
+                [160 /* Parameter */, 6 /* parameter */],
+                [163 /* PropertyDeclaration */, 9 /* property */],
+                [256 /* ModuleDeclaration */, 3 /* namespace */],
+                [255 /* EnumDeclaration */, 1 /* enum */],
+                [291 /* EnumMember */, 8 /* enumMember */],
+                [252 /* ClassDeclaration */, 0 /* class */],
+                [165 /* MethodDeclaration */, 11 /* member */],
+                [251 /* FunctionDeclaration */, 10 /* function */],
+                [208 /* FunctionExpression */, 10 /* function */],
+                [164 /* MethodSignature */, 11 /* member */],
+                [167 /* GetAccessor */, 9 /* property */],
+                [168 /* SetAccessor */, 9 /* property */],
+                [162 /* PropertySignature */, 9 /* property */],
+                [253 /* InterfaceDeclaration */, 2 /* interface */],
+                [254 /* TypeAliasDeclaration */, 5 /* type */],
+                [159 /* TypeParameter */, 4 /* typeParameter */],
+                [288 /* PropertyAssignment */, 9 /* property */],
+                [289 /* ShorthandPropertyAssignment */, 9 /* property */]
+            ]);
+        })(v2020 = classifier.v2020 || (classifier.v2020 = {}));
+    })(classifier = ts.classifier || (ts.classifier = {}));
+})(ts || (ts = {}));
 /* @internal */
 var ts;
 (function (ts) {
@@ -111951,23 +117782,24 @@ var ts;
                 if (completion === undefined) {
                     return undefined;
                 }
+                var optionalReplacementSpan = ts.createTextSpanFromStringLiteralLikeContent(contextToken);
                 switch (completion.kind) {
                     case 0 /* Paths */:
                         return convertPathCompletions(completion.paths);
                     case 1 /* Properties */: {
                         var entries = [];
                         Completions.getCompletionEntriesFromSymbols(completion.symbols, entries, contextToken, sourceFile, sourceFile, checker, 99 /* ESNext */, log, 4 /* String */, preferences); // Target will not be used, so arbitrary
-                        return { isGlobalCompletion: false, isMemberCompletion: true, isNewIdentifierLocation: completion.hasIndexSignature, entries: entries };
+                        return { isGlobalCompletion: false, isMemberCompletion: true, isNewIdentifierLocation: completion.hasIndexSignature, optionalReplacementSpan: optionalReplacementSpan, entries: entries };
                     }
                     case 2 /* Types */: {
                         var entries = completion.types.map(function (type) { return ({
                             name: type.value,
                             kindModifiers: "" /* none */,
                             kind: "string" /* string */,
-                            sortText: "0",
+                            sortText: Completions.SortText.LocationPriority,
                             replacementSpan: ts.getReplacementSpanForContextToken(contextToken)
                         }); });
-                        return { isGlobalCompletion: false, isMemberCompletion: false, isNewIdentifierLocation: completion.isNewIdentifier, entries: entries };
+                        return { isGlobalCompletion: false, isMemberCompletion: false, isNewIdentifierLocation: completion.isNewIdentifier, optionalReplacementSpan: optionalReplacementSpan, entries: entries };
                     }
                     default:
                         return ts.Debug.assertNever(completion);
@@ -112026,33 +117858,46 @@ var ts;
                 StringLiteralCompletionKind[StringLiteralCompletionKind["Types"] = 2] = "Types";
             })(StringLiteralCompletionKind || (StringLiteralCompletionKind = {}));
             function getStringLiteralCompletionEntries(sourceFile, node, position, typeChecker, compilerOptions, host) {
-                var parent = node.parent;
+                var parent = walkUpParentheses(node.parent);
                 switch (parent.kind) {
-                    case 187 /* LiteralType */:
-                        switch (parent.parent.kind) {
-                            case 169 /* TypeReference */:
-                                return { kind: 2 /* Types */, types: getStringLiteralTypes(typeChecker.getTypeArgumentConstraint(parent)), isNewIdentifier: false };
-                            case 185 /* IndexedAccessType */:
+                    case 191 /* LiteralType */: {
+                        var grandParent = walkUpParentheses(parent.parent);
+                        switch (grandParent.kind) {
+                            case 173 /* TypeReference */: {
+                                var typeReference_1 = grandParent;
+                                var typeArgument = ts.findAncestor(parent, function (n) { return n.parent === typeReference_1; });
+                                if (typeArgument) {
+                                    return { kind: 2 /* Types */, types: getStringLiteralTypes(typeChecker.getTypeArgumentConstraint(typeArgument)), isNewIdentifier: false };
+                                }
+                                return undefined;
+                            }
+                            case 189 /* IndexedAccessType */:
                                 // Get all apparent property names
                                 // i.e. interface Foo {
                                 //          foo: string;
                                 //          bar: string;
                                 //      }
                                 //      let x: Foo["/*completion position*/"]
-                                return stringLiteralCompletionsFromProperties(typeChecker.getTypeFromTypeNode(parent.parent.objectType));
-                            case 188 /* ImportType */:
+                                var _a = grandParent, indexType = _a.indexType, objectType = _a.objectType;
+                                if (!ts.rangeContainsPosition(indexType, position)) {
+                                    return undefined;
+                                }
+                                return stringLiteralCompletionsFromProperties(typeChecker.getTypeFromTypeNode(objectType));
+                            case 195 /* ImportType */:
                                 return { kind: 0 /* Paths */, paths: getStringLiteralCompletionsFromModuleNames(sourceFile, node, compilerOptions, host, typeChecker) };
-                            case 178 /* UnionType */: {
-                                if (!ts.isTypeReferenceNode(parent.parent.parent))
+                            case 182 /* UnionType */: {
+                                if (!ts.isTypeReferenceNode(grandParent.parent)) {
                                     return undefined;
-                                var alreadyUsedTypes_1 = getAlreadyUsedTypesInStringLiteralUnion(parent.parent, parent);
-                                var types = getStringLiteralTypes(typeChecker.getTypeArgumentConstraint(parent.parent)).filter(function (t) { return !ts.contains(alreadyUsedTypes_1, t.value); });
+                                }
+                                var alreadyUsedTypes_1 = getAlreadyUsedTypesInStringLiteralUnion(grandParent, parent);
+                                var types = getStringLiteralTypes(typeChecker.getTypeArgumentConstraint(grandParent)).filter(function (t) { return !ts.contains(alreadyUsedTypes_1, t.value); });
                                 return { kind: 2 /* Types */, types: types, isNewIdentifier: false };
                             }
                             default:
                                 return undefined;
                         }
-                    case 281 /* PropertyAssignment */:
+                    }
+                    case 288 /* PropertyAssignment */:
                         if (ts.isObjectLiteralExpression(parent.parent) && parent.name === node) {
                             // Get quoted name of properties of the object literal expression
                             // i.e. interface ConfigFiles {
@@ -112069,9 +117914,9 @@ var ts;
                             return stringLiteralCompletionsFromProperties(typeChecker.getContextualType(parent.parent));
                         }
                         return fromContextualType();
-                    case 195 /* ElementAccessExpression */: {
-                        var _a = parent, expression = _a.expression, argumentExpression = _a.argumentExpression;
-                        if (node === argumentExpression) {
+                    case 202 /* ElementAccessExpression */: {
+                        var _b = parent, expression = _b.expression, argumentExpression = _b.argumentExpression;
+                        if (node === ts.skipParentheses(argumentExpression)) {
                             // Get all names of properties on the expression
                             // i.e. interface A {
                             //      'prop1': string
@@ -112082,8 +117927,8 @@ var ts;
                         }
                         return undefined;
                     }
-                    case 196 /* CallExpression */:
-                    case 197 /* NewExpression */:
+                    case 203 /* CallExpression */:
+                    case 204 /* NewExpression */:
                         if (!ts.isRequireCall(parent, /*checkArgumentIsStringLiteralLike*/ false) && !ts.isImportCall(parent)) {
                             var argumentInfo = ts.SignatureHelp.getArgumentInfoForCompletions(node, position, sourceFile);
                             // Get string literal completions from specialized signatures of the target
@@ -112092,9 +117937,9 @@ var ts;
                             return argumentInfo ? getStringLiteralCompletionsFromSignature(argumentInfo, typeChecker) : fromContextualType();
                         }
                     // falls through (is `require("")` or `import("")`)
-                    case 254 /* ImportDeclaration */:
-                    case 260 /* ExportDeclaration */:
-                    case 265 /* ExternalModuleReference */:
+                    case 261 /* ImportDeclaration */:
+                    case 267 /* ExportDeclaration */:
+                    case 272 /* ExternalModuleReference */:
                         // Get all known external module names or complete a path to a module
                         // i.e. import * as ns from "/*completion position*/";
                         //      var y = import("/*completion position*/");
@@ -112111,6 +117956,16 @@ var ts;
                     return { kind: 2 /* Types */, types: getStringLiteralTypes(ts.getContextualTypeFromParent(node, typeChecker)), isNewIdentifier: false };
                 }
             }
+            function walkUpParentheses(node) {
+                switch (node.kind) {
+                    case 186 /* ParenthesizedType */:
+                        return ts.walkUpParenthesizedTypes(node);
+                    case 207 /* ParenthesizedExpression */:
+                        return ts.walkUpParenthesizedExpressions(node);
+                    default:
+                        return node;
+                }
+            }
             function getAlreadyUsedTypesInStringLiteralUnion(union, current) {
                 return ts.mapDefined(union.types, function (type) {
                     return type !== current && ts.isLiteralTypeNode(type) && ts.isStringLiteral(type.literal) ? type.literal.text : undefined;
@@ -112118,7 +117973,7 @@ var ts;
             }
             function getStringLiteralCompletionsFromSignature(argumentInfo, checker) {
                 var isNewIdentifier = false;
-                var uniques = ts.createMap();
+                var uniques = new ts.Map();
                 var candidates = [];
                 checker.getResolvedSignature(argumentInfo.invocation, candidates, argumentInfo.argumentCount);
                 var types = ts.flatMap(candidates, function (candidate) {
@@ -112133,14 +117988,12 @@ var ts;
             function stringLiteralCompletionsFromProperties(type) {
                 return type && {
                     kind: 1 /* Properties */,
-                    symbols: type.getApparentProperties().filter(function (prop) {
-                        return !ts.isPrivateIdentifierPropertyDeclaration(ts.isTransientSymbol(prop) && prop.syntheticOrigin ? prop.syntheticOrigin.valueDeclaration : prop.valueDeclaration);
-                    }),
+                    symbols: ts.filter(type.getApparentProperties(), function (prop) { return !(prop.valueDeclaration && ts.isPrivateIdentifierPropertyDeclaration(prop.valueDeclaration)); }),
                     hasIndexSignature: ts.hasIndexSignature(type)
                 };
             }
             function getStringLiteralTypes(type, uniques) {
-                if (uniques === void 0) { uniques = ts.createMap(); }
+                if (uniques === void 0) { uniques = new ts.Map(); }
                 if (!type)
                     return ts.emptyArray;
                 type = ts.skipConstraint(type);
@@ -112246,7 +118099,7 @@ var ts;
                      *
                      * both foo.ts and foo.tsx become foo
                      */
-                    var foundFiles = ts.createMap(); // maps file to its extension
+                    var foundFiles = new ts.Map(); // maps file to its extension
                     for (var _i = 0, files_1 = files; _i < files_1.length; _i++) {
                         var filePath = files_1[_i];
                         filePath = ts.normalizePath(filePath);
@@ -112456,7 +118309,7 @@ var ts;
             function getCompletionEntriesFromTypings(host, options, scriptPath, fragmentDirectory, extensionOptions, result) {
                 if (result === void 0) { result = []; }
                 // Check for typings specified in compiler options
-                var seen = ts.createMap();
+                var seen = new ts.Map();
                 var typeRoots = ts.tryAndIgnoreErrors(function () { return ts.getEffectiveTypeRoots(options, host); }) || ts.emptyArray;
                 for (var _i = 0, typeRoots_1 = typeRoots; _i < typeRoots_1.length; _i++) {
                     var root = typeRoots_1[_i];
@@ -112559,13 +118412,14 @@ var ts;
     (function (Completions) {
         var SortText;
         (function (SortText) {
-            SortText["LocationPriority"] = "0";
-            SortText["OptionalMember"] = "1";
-            SortText["MemberDeclaredBySpreadAssignment"] = "2";
-            SortText["SuggestedClassMembers"] = "3";
-            SortText["GlobalsOrKeywords"] = "4";
-            SortText["AutoImportSuggestions"] = "5";
-            SortText["JavascriptIdentifiers"] = "6";
+            SortText["LocalDeclarationPriority"] = "0";
+            SortText["LocationPriority"] = "1";
+            SortText["OptionalMember"] = "2";
+            SortText["MemberDeclaredBySpreadAssignment"] = "3";
+            SortText["SuggestedClassMembers"] = "4";
+            SortText["GlobalsOrKeywords"] = "5";
+            SortText["AutoImportSuggestions"] = "6";
+            SortText["JavascriptIdentifiers"] = "7";
         })(SortText = Completions.SortText || (Completions.SortText = {}));
         /**
          * Special values for `CompletionInfo['source']` used to disambiguate
@@ -112602,6 +118456,9 @@ var ts;
         function originIsExport(origin) {
             return !!(origin && origin.kind & 4 /* Export */);
         }
+        function originIsPackageJsonImport(origin) {
+            return originIsExport(origin) && !!origin.isFromPackageJson;
+        }
         function originIsPromise(origin) {
             return !!(origin.kind & 8 /* Promise */);
         }
@@ -112683,7 +118540,7 @@ var ts;
                 return stringCompletions;
             }
             if (contextToken && ts.isBreakOrContinueStatement(contextToken.parent)
-                && (contextToken.kind === 77 /* BreakKeyword */ || contextToken.kind === 82 /* ContinueKeyword */ || contextToken.kind === 75 /* Identifier */)) {
+                && (contextToken.kind === 80 /* BreakKeyword */ || contextToken.kind === 85 /* ContinueKeyword */ || contextToken.kind === 78 /* Identifier */)) {
                 return getLabelCompletionAtPosition(contextToken.parent);
             }
             var completionData = getCompletionData(program, log, sourceFile, isUncheckedFile(sourceFile, compilerOptions), position, preferences, /*detailsEntryId*/ undefined, host);
@@ -112709,6 +118566,10 @@ var ts;
         function jsdocCompletionInfo(entries) {
             return { isGlobalCompletion: false, isMemberCompletion: false, isNewIdentifierLocation: false, entries: entries };
         }
+        function getOptionalReplacementSpan(location) {
+            // StringLiteralLike locations are handled separately in stringCompletions.ts
+            return (location === null || location === void 0 ? void 0 : location.kind) === 78 /* Identifier */ ? ts.createTextSpanFromNode(location) : undefined;
+        }
         function completionInfoFromData(sourceFile, typeChecker, compilerOptions, log, completionData, preferences) {
             var symbols = completionData.symbols, completionKind = completionData.completionKind, isInSnippetScope = completionData.isInSnippetScope, isNewIdentifierLocation = completionData.isNewIdentifierLocation, location = completionData.location, propertyAccessToConvert = completionData.propertyAccessToConvert, keywordFilters = completionData.keywordFilters, literals = completionData.literals, symbolToOriginInfoMap = completionData.symbolToOriginInfoMap, recommendedCompletion = completionData.recommendedCompletion, isJsxInitializer = completionData.isJsxInitializer, insideJsDocTagTypeExpression = completionData.insideJsDocTagTypeExpression, symbolToSortTextMap = completionData.symbolToSortTextMap;
             if (location && location.parent && ts.isJsxClosingElement(location.parent)) {
@@ -112726,7 +118587,7 @@ var ts;
                     kindModifiers: undefined,
                     sortText: SortText.LocationPriority,
                 };
-                return { isGlobalCompletion: false, isMemberCompletion: true, isNewIdentifierLocation: false, entries: [entry] };
+                return { isGlobalCompletion: false, isMemberCompletion: true, isNewIdentifierLocation: false, optionalReplacementSpan: getOptionalReplacementSpan(location), entries: [entry] };
             }
             var entries = [];
             if (isUncheckedFile(sourceFile, compilerOptions)) {
@@ -112742,7 +118603,7 @@ var ts;
                 /* contextToken */ undefined, location, sourceFile, typeChecker, compilerOptions.target, log, completionKind, preferences, propertyAccessToConvert, completionData.isJsxIdentifierExpected, isJsxInitializer, recommendedCompletion, symbolToOriginInfoMap, symbolToSortTextMap);
             }
             if (keywordFilters !== 0 /* None */) {
-                var entryNames = ts.arrayToSet(entries, function (e) { return e.name; });
+                var entryNames = new ts.Set(entries.map(function (e) { return e.name; }));
                 for (var _i = 0, _a = getKeywordCompletions(keywordFilters, !insideJsDocTagTypeExpression && ts.isSourceFileJS(sourceFile)); _i < _a.length; _i++) {
                     var keywordEntry = _a[_i];
                     if (!entryNames.has(keywordEntry.name)) {
@@ -112752,9 +118613,15 @@ var ts;
             }
             for (var _b = 0, literals_1 = literals; _b < literals_1.length; _b++) {
                 var literal = literals_1[_b];
-                entries.push(createCompletionEntryForLiteral(literal, preferences));
+                entries.push(createCompletionEntryForLiteral(sourceFile, preferences, literal));
             }
-            return { isGlobalCompletion: isInSnippetScope, isMemberCompletion: isMemberCompletionKind(completionKind), isNewIdentifierLocation: isNewIdentifierLocation, entries: entries };
+            return {
+                isGlobalCompletion: isInSnippetScope,
+                isMemberCompletion: isMemberCompletionKind(completionKind),
+                isNewIdentifierLocation: isNewIdentifierLocation,
+                optionalReplacementSpan: getOptionalReplacementSpan(location),
+                entries: entries
+            };
         }
         function isUncheckedFile(sourceFile, compilerOptions) {
             return ts.isSourceFileJS(sourceFile) && !ts.isCheckJsEnabledForFile(sourceFile, compilerOptions);
@@ -112788,12 +118655,12 @@ var ts;
                 }
             });
         }
-        function completionNameForLiteral(literal, preferences) {
+        function completionNameForLiteral(sourceFile, preferences, literal) {
             return typeof literal === "object" ? ts.pseudoBigIntToString(literal) + "n" :
-                ts.isString(literal) ? ts.quote(literal, preferences) : JSON.stringify(literal);
+                ts.isString(literal) ? ts.quote(sourceFile, preferences, literal) : JSON.stringify(literal);
         }
-        function createCompletionEntryForLiteral(literal, preferences) {
-            return { name: completionNameForLiteral(literal, preferences), kind: "string" /* string */, kindModifiers: "" /* none */, sortText: SortText.LocationPriority };
+        function createCompletionEntryForLiteral(sourceFile, preferences, literal) {
+            return { name: completionNameForLiteral(sourceFile, preferences, literal), kind: "string" /* string */, kindModifiers: "" /* none */, sortText: SortText.LocationPriority };
         }
         function createCompletionEntry(symbol, sortText, contextToken, location, sourceFile, typeChecker, name, needsConvertPropertyAccess, origin, recommendedCompletion, propertyAccessToConvert, isJsxInitializer, preferences) {
             var insertText;
@@ -112802,13 +118669,13 @@ var ts;
             var useBraces = origin && originIsSymbolMember(origin) || needsConvertPropertyAccess;
             if (origin && originIsThisType(origin)) {
                 insertText = needsConvertPropertyAccess
-                    ? "this" + (insertQuestionDot ? "?." : "") + "[" + quotePropertyName(name, preferences) + "]"
+                    ? "this" + (insertQuestionDot ? "?." : "") + "[" + quotePropertyName(sourceFile, preferences, name) + "]"
                     : "this" + (insertQuestionDot ? "?." : ".") + name;
             }
             // We should only have needsConvertPropertyAccess if there's a property access to convert. But see #21790.
             // Somehow there was a global with a non-identifier name. Hopefully someone will complain about getting a "foo bar" global completion and provide a repro.
             else if ((useBraces || insertQuestionDot) && propertyAccessToConvert) {
-                insertText = useBraces ? needsConvertPropertyAccess ? "[" + quotePropertyName(name, preferences) + "]" : "[" + name + "]" : name;
+                insertText = useBraces ? needsConvertPropertyAccess ? "[" + quotePropertyName(sourceFile, preferences, name) + "]" : "[" + name + "]" : name;
                 if (insertQuestionDot || propertyAccessToConvert.questionDotToken) {
                     insertText = "?." + insertText;
                 }
@@ -112861,13 +118728,14 @@ var ts;
                 isRecommended: isRecommendedCompletionMatch(symbol, recommendedCompletion, typeChecker) || undefined,
                 insertText: insertText,
                 replacementSpan: replacementSpan,
+                isPackageJsonImport: originIsPackageJsonImport(origin) || undefined,
             };
         }
-        function quotePropertyName(name, preferences) {
+        function quotePropertyName(sourceFile, preferences, name) {
             if (/^\d+$/.test(name)) {
                 return name;
             }
-            return ts.quote(name, preferences);
+            return ts.quote(sourceFile, preferences, name);
         }
         function isRecommendedCompletionMatch(localSymbol, recommendedCompletion, checker) {
             return localSymbol === recommendedCompletion ||
@@ -112887,7 +118755,7 @@ var ts;
             // Value is set to false for global variables or completions from external module exports, because we can have multiple of those;
             // true otherwise. Based on the order we add things we will always see locals first, then globals, then module exports.
             // So adding a completion for a local will prevent us from adding completions for external module exports sharing the same name.
-            var uniques = ts.createMap();
+            var uniques = new ts.Map();
             for (var _i = 0, symbols_1 = symbols; _i < symbols_1.length; _i++) {
                 var symbol = symbols_1[_i];
                 var origin = symbolToOriginInfoMap ? symbolToOriginInfoMap[ts.getSymbolId(symbol)] : undefined;
@@ -112926,7 +118794,7 @@ var ts;
         }
         function getLabelStatementCompletions(node) {
             var entries = [];
-            var uniques = ts.createMap();
+            var uniques = new ts.Map();
             var current = node;
             while (current) {
                 if (ts.isFunctionLike(current)) {
@@ -112958,7 +118826,7 @@ var ts;
                 return { type: "request", request: completionData };
             }
             var symbols = completionData.symbols, literals = completionData.literals, location = completionData.location, completionKind = completionData.completionKind, symbolToOriginInfoMap = completionData.symbolToOriginInfoMap, previousToken = completionData.previousToken, isJsxInitializer = completionData.isJsxInitializer, isTypeOnlyLocation = completionData.isTypeOnlyLocation;
-            var literal = ts.find(literals, function (l) { return completionNameForLiteral(l, preferences) === entryId.name; });
+            var literal = ts.find(literals, function (l) { return completionNameForLiteral(sourceFile, preferences, l) === entryId.name; });
             if (literal !== undefined)
                 return { type: "literal", literal: literal };
             // Find the symbol with the matching entry name.
@@ -113004,7 +118872,7 @@ var ts;
                 }
                 case "literal": {
                     var literal = symbolCompletion.literal;
-                    return createSimpleDetails(completionNameForLiteral(literal, preferences), "string" /* string */, typeof literal === "string" ? ts.SymbolDisplayPartKind.stringLiteral : ts.SymbolDisplayPartKind.numericLiteral);
+                    return createSimpleDetails(completionNameForLiteral(sourceFile, preferences, literal), "string" /* string */, typeof literal === "string" ? ts.SymbolDisplayPartKind.stringLiteral : ts.SymbolDisplayPartKind.numericLiteral);
                 }
                 case "none":
                     // Didn't find a symbol with this name.  See if we can find a keyword instead.
@@ -113072,25 +118940,25 @@ var ts;
         function getContextualType(previousToken, position, sourceFile, checker) {
             var parent = previousToken.parent;
             switch (previousToken.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return ts.getContextualTypeFromParent(previousToken, checker);
                 case 62 /* EqualsToken */:
                     switch (parent.kind) {
-                        case 242 /* VariableDeclaration */:
+                        case 249 /* VariableDeclaration */:
                             return checker.getContextualType(parent.initializer); // TODO: GH#18217
-                        case 209 /* BinaryExpression */:
+                        case 216 /* BinaryExpression */:
                             return checker.getTypeAtLocation(parent.left);
-                        case 273 /* JsxAttribute */:
+                        case 280 /* JsxAttribute */:
                             return checker.getContextualTypeForJsxAttribute(parent);
                         default:
                             return undefined;
                     }
-                case 99 /* NewKeyword */:
+                case 102 /* NewKeyword */:
                     return checker.getContextualType(parent);
-                case 78 /* CaseKeyword */:
+                case 81 /* CaseKeyword */:
                     return ts.getSwitchedType(ts.cast(parent, ts.isCaseClause), checker);
                 case 18 /* OpenBraceToken */:
-                    return ts.isJsxExpression(parent) && parent.parent.kind !== 266 /* JsxElement */ ? checker.getContextualTypeForJsxAttribute(parent.parent) : undefined;
+                    return ts.isJsxExpression(parent) && parent.parent.kind !== 273 /* JsxElement */ ? checker.getContextualTypeForJsxAttribute(parent.parent) : undefined;
                 default:
                     var argInfo = ts.SignatureHelp.getArgumentInfoForCompletions(previousToken, position, sourceFile);
                     return argInfo ?
@@ -113109,7 +118977,7 @@ var ts;
             return symbol.parent && (isModuleSymbol(symbol.parent) ? symbol : getFirstSymbolInChain(symbol.parent, enclosingDeclaration, checker));
         }
         function isModuleSymbol(symbol) {
-            return symbol.declarations.some(function (d) { return d.kind === 290 /* SourceFile */; });
+            return symbol.declarations.some(function (d) { return d.kind === 297 /* SourceFile */; });
         }
         function getCompletionData(program, log, sourceFile, isUncheckedFile, position, preferences, detailsEntryId, host) {
             var typeChecker = program.getTypeChecker();
@@ -113147,7 +119015,7 @@ var ts;
                         //    *         |c|
                         //    */
                         var lineStart = ts.getLineStartPositionForPosition(position, sourceFile);
-                        if (!(sourceFile.text.substring(lineStart, position).match(/[^\*|\s|(/\*\*)]/))) {
+                        if (!/[^\*|\s(/)]/.test(sourceFile.text.substring(lineStart, position))) {
                             return { kind: 2 /* JsDocTag */ };
                         }
                     }
@@ -113160,11 +119028,11 @@ var ts;
                     if (tag.tagName.pos <= position && position <= tag.tagName.end) {
                         return { kind: 1 /* JsDocTagName */ };
                     }
-                    if (isTagWithTypeExpression(tag) && tag.typeExpression && tag.typeExpression.kind === 294 /* JSDocTypeExpression */) {
+                    if (isTagWithTypeExpression(tag) && tag.typeExpression && tag.typeExpression.kind === 301 /* JSDocTypeExpression */) {
                         currentToken = ts.getTokenAtPosition(sourceFile, position);
                         if (!currentToken ||
                             (!ts.isDeclarationName(currentToken) &&
-                                (currentToken.parent.kind !== 323 /* JSDocPropertyTag */ ||
+                                (currentToken.parent.kind !== 333 /* JSDocPropertyTag */ ||
                                     currentToken.parent.name !== currentToken))) {
                             // Use as type location if inside tag's type expression
                             insideJsDocTagTypeExpression = isCurrentlyEditingNode(tag.typeExpression);
@@ -113217,26 +119085,27 @@ var ts;
                     isRightOfDot = contextToken.kind === 24 /* DotToken */;
                     isRightOfQuestionDot = contextToken.kind === 28 /* QuestionDotToken */;
                     switch (parent.kind) {
-                        case 194 /* PropertyAccessExpression */:
+                        case 201 /* PropertyAccessExpression */:
                             propertyAccessToConvert = parent;
                             node = propertyAccessToConvert.expression;
-                            if (node.end === contextToken.pos &&
-                                ts.isCallExpression(node) &&
+                            if ((ts.isCallExpression(node) || ts.isFunctionLike(node)) &&
+                                node.end === contextToken.pos &&
                                 node.getChildCount(sourceFile) &&
                                 ts.last(node.getChildren(sourceFile)).kind !== 21 /* CloseParenToken */) {
-                                // This is likely dot from incorrectly parsed call expression and user is starting to write spread
+                                // This is likely dot from incorrectly parsed expression and user is starting to write spread
                                 // eg: Math.min(./**/)
+                                // const x = function (./**/) {}
                                 return undefined;
                             }
                             break;
-                        case 153 /* QualifiedName */:
+                        case 157 /* QualifiedName */:
                             node = parent.left;
                             break;
-                        case 249 /* ModuleDeclaration */:
+                        case 256 /* ModuleDeclaration */:
                             node = parent.name;
                             break;
-                        case 188 /* ImportType */:
-                        case 219 /* MetaProperty */:
+                        case 195 /* ImportType */:
+                        case 226 /* MetaProperty */:
                             node = parent;
                             break;
                         default:
@@ -113249,7 +119118,7 @@ var ts;
                     // <UI.Test /* completion position */ />
                     // If the tagname is a property access expression, we will then walk up to the top most of property access expression.
                     // Then, try to get a JSX container and its associated attributes type.
-                    if (parent && parent.kind === 194 /* PropertyAccessExpression */) {
+                    if (parent && parent.kind === 201 /* PropertyAccessExpression */) {
                         contextToken = parent;
                         parent = parent.parent;
                     }
@@ -113257,44 +119126,56 @@ var ts;
                     if (currentToken.parent === location) {
                         switch (currentToken.kind) {
                             case 31 /* GreaterThanToken */:
-                                if (currentToken.parent.kind === 266 /* JsxElement */ || currentToken.parent.kind === 268 /* JsxOpeningElement */) {
+                                if (currentToken.parent.kind === 273 /* JsxElement */ || currentToken.parent.kind === 275 /* JsxOpeningElement */) {
                                     location = currentToken;
                                 }
                                 break;
                             case 43 /* SlashToken */:
-                                if (currentToken.parent.kind === 267 /* JsxSelfClosingElement */) {
+                                if (currentToken.parent.kind === 274 /* JsxSelfClosingElement */) {
                                     location = currentToken;
                                 }
                                 break;
                         }
                     }
                     switch (parent.kind) {
-                        case 269 /* JsxClosingElement */:
+                        case 276 /* JsxClosingElement */:
                             if (contextToken.kind === 43 /* SlashToken */) {
                                 isStartingCloseTag = true;
                                 location = contextToken;
                             }
                             break;
-                        case 209 /* BinaryExpression */:
+                        case 216 /* BinaryExpression */:
                             if (!binaryExpressionMayBeOpenTag(parent)) {
                                 break;
                             }
                         // falls through
-                        case 267 /* JsxSelfClosingElement */:
-                        case 266 /* JsxElement */:
-                        case 268 /* JsxOpeningElement */:
+                        case 274 /* JsxSelfClosingElement */:
+                        case 273 /* JsxElement */:
+                        case 275 /* JsxOpeningElement */:
                             isJsxIdentifierExpected = true;
                             if (contextToken.kind === 29 /* LessThanToken */) {
                                 isRightOfOpenTag = true;
                                 location = contextToken;
                             }
                             break;
-                        case 273 /* JsxAttribute */:
+                        case 283 /* JsxExpression */:
+                            // For `<div foo={true} [||] ></div>`, `parent` will be `{true}` and `previousToken` will be `}`
+                            if (previousToken.kind === 19 /* CloseBraceToken */ && currentToken.kind === 31 /* GreaterThanToken */) {
+                                isJsxIdentifierExpected = true;
+                            }
+                            break;
+                        case 280 /* JsxAttribute */:
+                            // For `<div className="x" [||] ></div>`, `parent` will be JsxAttribute and `previousToken` will be its initializer
+                            if (parent.initializer === previousToken &&
+                                previousToken.end < position) {
+                                isJsxIdentifierExpected = true;
+                                break;
+                            }
                             switch (previousToken.kind) {
                                 case 62 /* EqualsToken */:
                                     isJsxInitializer = true;
                                     break;
-                                case 75 /* Identifier */:
+                                case 78 /* Identifier */:
                                     isJsxIdentifierExpected = true;
                                     // For `<div x=[|f/**/|]`, `parent` will be `x` and `previousToken.parent` will be `f` (which is its own JsxAttribute)
                                     // Note for `<div someBool f>` we don't want to treat this as a jsx inializer, instead it's the attribute name.
@@ -113371,11 +119252,11 @@ var ts;
             };
             function isTagWithTypeExpression(tag) {
                 switch (tag.kind) {
-                    case 317 /* JSDocParameterTag */:
-                    case 323 /* JSDocPropertyTag */:
-                    case 318 /* JSDocReturnTag */:
-                    case 320 /* JSDocTypeTag */:
-                    case 322 /* JSDocTypedefTag */:
+                    case 326 /* JSDocParameterTag */:
+                    case 333 /* JSDocPropertyTag */:
+                    case 327 /* JSDocReturnTag */:
+                    case 329 /* JSDocTypeTag */:
+                    case 331 /* JSDocTypedefTag */:
                         return true;
                     default:
                         return false;
@@ -113391,7 +119272,7 @@ var ts;
                     || ts.isPartOfTypeNode(node.parent)
                     || ts.isPossiblyTypeArgumentPosition(contextToken, sourceFile, typeChecker);
                 var isRhsOfImportDeclaration = ts.isInRightSideOfInternalImportEqualsDeclaration(node);
-                if (ts.isEntityName(node) || isImportType) {
+                if (ts.isEntityName(node) || isImportType || ts.isPropertyAccessExpression(node)) {
                     var isNamespaceName = ts.isModuleDeclaration(node.parent);
                     if (isNamespaceName)
                         isNewIdentifierLocation = true;
@@ -113420,7 +119301,7 @@ var ts;
                             // If the module is merged with a value, we must get the type of the class and add its propertes (for inherited static methods).
                             if (!isTypeLocation &&
                                 symbol.declarations &&
-                                symbol.declarations.some(function (d) { return d.kind !== 290 /* SourceFile */ && d.kind !== 249 /* ModuleDeclaration */ && d.kind !== 248 /* EnumDeclaration */; })) {
+                                symbol.declarations.some(function (d) { return d.kind !== 297 /* SourceFile */ && d.kind !== 256 /* ModuleDeclaration */ && d.kind !== 255 /* EnumDeclaration */; })) {
                                 var type = typeChecker.getTypeOfSymbolAtLocation(symbol, node).getNonOptionalType();
                                 var insertQuestionDot = false;
                                 if (type.isNullableType()) {
@@ -113440,8 +119321,8 @@ var ts;
                         }
                     }
                 }
-                if (ts.isMetaProperty(node) && (node.keywordToken === 99 /* NewKeyword */ || node.keywordToken === 96 /* ImportKeyword */) && contextToken === node.getChildAt(1)) {
-                    var completion = (node.keywordToken === 99 /* NewKeyword */) ? "target" : "meta";
+                if (ts.isMetaProperty(node) && (node.keywordToken === 102 /* NewKeyword */ || node.keywordToken === 99 /* ImportKeyword */) && contextToken === node.getChildAt(1)) {
+                    var completion = (node.keywordToken === 102 /* NewKeyword */) ? "target" : "meta";
                     symbols.push(typeChecker.createSymbol(4 /* Property */, ts.escapeLeadingUnderscores(completion)));
                     return;
                 }
@@ -113467,20 +119348,20 @@ var ts;
                 if (isRightOfQuestionDot && ts.some(type.getCallSignatures())) {
                     isNewIdentifierLocation = true;
                 }
-                var propertyAccess = node.kind === 188 /* ImportType */ ? node : node.parent;
+                var propertyAccess = node.kind === 195 /* ImportType */ ? node : node.parent;
                 if (isUncheckedFile) {
                     // In javascript files, for union types, we don't just get the members that
                     // the individual types have in common, we also include all the members that
                     // each individual type has. This is because we're going to add all identifiers
                     // anyways. So we might as well elevate the members that were at least part
                     // of the individual types to a higher status since we know what they are.
-                    symbols.push.apply(symbols, getPropertiesForCompletion(type, typeChecker));
+                    symbols.push.apply(symbols, ts.filter(getPropertiesForCompletion(type, typeChecker), function (s) { return typeChecker.isValidPropertyAccessForCompletions(propertyAccess, type, s); }));
                 }
                 else {
                     for (var _i = 0, _a = type.getApparentProperties(); _i < _a.length; _i++) {
                         var symbol = _a[_i];
                         if (typeChecker.isValidPropertyAccessForCompletions(propertyAccess, type, symbol)) {
-                            addPropertySymbol(symbol, /*insertAwait*/ false, insertQuestionDot);
+                            addPropertySymbol(symbol, /* insertAwait */ false, insertQuestionDot);
                         }
                     }
                 }
@@ -113516,13 +119397,20 @@ var ts;
                     }
                     else if (preferences.includeCompletionsWithInsertText) {
                         addSymbolOriginInfo(symbol);
+                        addSymbolSortInfo(symbol);
                         symbols.push(symbol);
                     }
                 }
                 else {
                     addSymbolOriginInfo(symbol);
+                    addSymbolSortInfo(symbol);
                     symbols.push(symbol);
                 }
+                function addSymbolSortInfo(symbol) {
+                    if (isStaticProperty(symbol)) {
+                        symbolToSortTextMap[ts.getSymbolId(symbol)] = SortText.LocalDeclarationPriority;
+                    }
+                }
                 function addSymbolOriginInfo(symbol) {
                     if (preferences.includeCompletionsWithInsertText) {
                         if (insertAwait && !symbolToOriginInfoMap[ts.getSymbolId(symbol)]) {
@@ -113624,7 +119512,7 @@ var ts;
                     }
                 }
                 // Need to insert 'this.' before properties of `this` type, so only do that if `includeInsertTextCompletions`
-                if (preferences.includeCompletionsWithInsertText && scopeNode.kind !== 290 /* SourceFile */) {
+                if (preferences.includeCompletionsWithInsertText && scopeNode.kind !== 297 /* SourceFile */) {
                     var thisType = typeChecker.tryGetThisTypeAt(scopeNode, /*includeGlobalThis*/ false);
                     if (thisType && !isProbablyGlobalType(thisType, sourceFile, typeChecker)) {
                         for (var _a = 0, _b = getPropertiesForCompletion(thisType, typeChecker); _a < _b.length; _a++) {
@@ -113674,10 +119562,10 @@ var ts;
             }
             function isSnippetScope(scopeNode) {
                 switch (scopeNode.kind) {
-                    case 290 /* SourceFile */:
-                    case 211 /* TemplateExpression */:
-                    case 276 /* JsxExpression */:
-                    case 223 /* Block */:
+                    case 297 /* SourceFile */:
+                    case 218 /* TemplateExpression */:
+                    case 283 /* JsxExpression */:
+                    case 230 /* Block */:
                         return true;
                     default:
                         return ts.isStatement(scopeNode);
@@ -113686,7 +119574,7 @@ var ts;
             function filterGlobalCompletion(symbols) {
                 var isTypeOnly = isTypeOnlyCompletion();
                 if (isTypeOnly) {
-                    keywordFilters = isTypeAssertion()
+                    keywordFilters = contextToken && ts.isAssertionExpression(contextToken.parent)
                         ? 6 /* TypeAssertionKeywords */
                         : 7 /* TypeKeywords */;
                 }
@@ -113710,9 +119598,6 @@ var ts;
                     return !!(ts.getCombinedLocalAndExportSymbolFlags(symbol) & 111551 /* Value */);
                 });
             }
-            function isTypeAssertion() {
-                return ts.isAssertionExpression(contextToken.parent);
-            }
             function isTypeOnlyCompletion() {
                 return insideJsDocTagTypeExpression
                     || !isContextTokenValueLocation(contextToken) &&
@@ -113722,35 +119607,35 @@ var ts;
             }
             function isContextTokenValueLocation(contextToken) {
                 return contextToken &&
-                    contextToken.kind === 108 /* TypeOfKeyword */ &&
-                    (contextToken.parent.kind === 172 /* TypeQuery */ || ts.isTypeOfExpression(contextToken.parent));
+                    contextToken.kind === 111 /* TypeOfKeyword */ &&
+                    (contextToken.parent.kind === 176 /* TypeQuery */ || ts.isTypeOfExpression(contextToken.parent));
             }
             function isContextTokenTypeLocation(contextToken) {
                 if (contextToken) {
                     var parentKind = contextToken.parent.kind;
                     switch (contextToken.kind) {
                         case 58 /* ColonToken */:
-                            return parentKind === 159 /* PropertyDeclaration */ ||
-                                parentKind === 158 /* PropertySignature */ ||
-                                parentKind === 156 /* Parameter */ ||
-                                parentKind === 242 /* VariableDeclaration */ ||
+                            return parentKind === 163 /* PropertyDeclaration */ ||
+                                parentKind === 162 /* PropertySignature */ ||
+                                parentKind === 160 /* Parameter */ ||
+                                parentKind === 249 /* VariableDeclaration */ ||
                                 ts.isFunctionLikeKind(parentKind);
                         case 62 /* EqualsToken */:
-                            return parentKind === 247 /* TypeAliasDeclaration */;
-                        case 123 /* AsKeyword */:
-                            return parentKind === 217 /* AsExpression */;
+                            return parentKind === 254 /* TypeAliasDeclaration */;
+                        case 126 /* AsKeyword */:
+                            return parentKind === 224 /* AsExpression */;
                         case 29 /* LessThanToken */:
-                            return parentKind === 169 /* TypeReference */ ||
-                                parentKind === 199 /* TypeAssertionExpression */;
-                        case 90 /* ExtendsKeyword */:
-                            return parentKind === 155 /* TypeParameter */;
+                            return parentKind === 173 /* TypeReference */ ||
+                                parentKind === 206 /* TypeAssertionExpression */;
+                        case 93 /* ExtendsKeyword */:
+                            return parentKind === 159 /* TypeParameter */;
                     }
                 }
                 return false;
             }
             /** True if symbol is a type or a module containing at least one type. */
             function symbolCanBeReferencedAtTypeLocation(symbol, seenModules) {
-                if (seenModules === void 0) { seenModules = ts.createMap(); }
+                if (seenModules === void 0) { seenModules = new ts.Map(); }
                 var sym = ts.skipAlias(symbol.exportSymbol || symbol, typeChecker);
                 return !!(sym.flags & 788968 /* Type */) ||
                     !!(sym.flags & 1536 /* Module */) &&
@@ -113812,21 +119697,22 @@ var ts;
                 }
                 var startTime = ts.timestamp();
                 log("getSymbolsFromOtherSourceFileExports: Recomputing list" + (detailsEntryId ? " for details entry" : ""));
-                var seenResolvedModules = ts.createMap();
-                var seenExports = ts.createMap();
+                var seenResolvedModules = new ts.Map();
+                var seenExports = new ts.Map();
                 /** Bucket B */
-                var aliasesToAlreadyIncludedSymbols = ts.createMap();
+                var aliasesToAlreadyIncludedSymbols = new ts.Map();
                 /** Bucket C */
-                var aliasesToReturnIfOriginalsAreMissing = ts.createMap();
+                var aliasesToReturnIfOriginalsAreMissing = new ts.Map();
                 /** Bucket A */
                 var results = [];
                 /** Ids present in `results` for faster lookup */
-                var resultSymbolIds = ts.createMap();
-                ts.codefix.forEachExternalModuleToImportFrom(program, host, sourceFile, !detailsEntryId, function (moduleSymbol) {
+                var resultSymbolIds = new ts.Map();
+                ts.codefix.forEachExternalModuleToImportFrom(program, host, sourceFile, !detailsEntryId, /*useAutoImportProvider*/ true, function (moduleSymbol, _, program, isFromPackageJson) {
                     // Perf -- ignore other modules if this is a request for details
                     if (detailsEntryId && detailsEntryId.source && ts.stripQuotes(moduleSymbol.name) !== detailsEntryId.source) {
                         return;
                     }
+                    var typeChecker = program.getTypeChecker();
                     var resolvedModuleSymbol = typeChecker.resolveExternalModuleSymbol(moduleSymbol);
                     // resolvedModuleSymbol may be a namespace. A namespace may be `export =` by multiple module declarations, but only keep the first one.
                     if (!ts.addToSeen(seenResolvedModules, ts.getSymbolId(resolvedModuleSymbol))) {
@@ -113835,7 +119721,7 @@ var ts;
                     // Don't add another completion for `export =` of a symbol that's already global.
                     // So in `declare namespace foo {} declare module "foo" { export = foo; }`, there will just be the global completion for `foo`.
                     if (resolvedModuleSymbol !== moduleSymbol && ts.every(resolvedModuleSymbol.declarations, ts.isNonGlobalDeclaration)) {
-                        pushSymbol(resolvedModuleSymbol, moduleSymbol, /*skipFilter*/ true);
+                        pushSymbol(resolvedModuleSymbol, moduleSymbol, isFromPackageJson, /*skipFilter*/ true);
                     }
                     for (var _i = 0, _a = typeChecker.getExportsAndPropertiesOfModule(moduleSymbol); _i < _a.length; _i++) {
                         var symbol = _a[_i];
@@ -113860,7 +119746,7 @@ var ts;
                             var nearestExportSymbolId = ts.getSymbolId(nearestExportSymbol).toString();
                             var symbolHasBeenSeen = resultSymbolIds.has(nearestExportSymbolId) || aliasesToAlreadyIncludedSymbols.has(nearestExportSymbolId);
                             if (!symbolHasBeenSeen) {
-                                aliasesToReturnIfOriginalsAreMissing.set(nearestExportSymbolId, { alias: symbol, moduleSymbol: moduleSymbol });
+                                aliasesToReturnIfOriginalsAreMissing.set(nearestExportSymbolId, { alias: symbol, moduleSymbol: moduleSymbol, isFromPackageJson: isFromPackageJson });
                                 aliasesToAlreadyIncludedSymbols.set(symbolId, true);
                             }
                             else {
@@ -113872,20 +119758,19 @@ var ts;
                         else {
                             // This is not a re-export, so see if we have any aliases pending and remove them (step 3 in diagrammed example)
                             aliasesToReturnIfOriginalsAreMissing.delete(symbolId);
-                            pushSymbol(symbol, moduleSymbol);
+                            pushSymbol(symbol, moduleSymbol, isFromPackageJson, /*skipFilter*/ false);
                         }
                     }
                 });
                 // By this point, any potential duplicates that were actually duplicates have been
                 // removed, so the rest need to be added. (Step 4 in diagrammed example)
                 aliasesToReturnIfOriginalsAreMissing.forEach(function (_a) {
-                    var alias = _a.alias, moduleSymbol = _a.moduleSymbol;
-                    return pushSymbol(alias, moduleSymbol);
+                    var alias = _a.alias, moduleSymbol = _a.moduleSymbol, isFromPackageJson = _a.isFromPackageJson;
+                    return pushSymbol(alias, moduleSymbol, isFromPackageJson, /*skipFilter*/ false);
                 });
                 log("getSymbolsFromOtherSourceFileExports: " + (ts.timestamp() - startTime));
                 return results;
-                function pushSymbol(symbol, moduleSymbol, skipFilter) {
-                    if (skipFilter === void 0) { skipFilter = false; }
+                function pushSymbol(symbol, moduleSymbol, isFromPackageJson, skipFilter) {
                     var isDefaultExport = symbol.escapedName === "default" /* Default */;
                     if (isDefaultExport) {
                         symbol = ts.getLocalSymbolForExportDefault(symbol) || symbol;
@@ -113894,7 +119779,7 @@ var ts;
                         return;
                     }
                     ts.addToSeen(resultSymbolIds, ts.getSymbolId(symbol));
-                    var origin = { kind: 4 /* Export */, moduleSymbol: moduleSymbol, isDefaultExport: isDefaultExport };
+                    var origin = { kind: 4 /* Export */, moduleSymbol: moduleSymbol, isDefaultExport: isDefaultExport, isFromPackageJson: isFromPackageJson };
                     results.push({
                         symbol: symbol,
                         symbolName: ts.getNameForExportedSymbol(symbol, target),
@@ -113954,7 +119839,7 @@ var ts;
                     return true;
                 }
                 if (contextToken.kind === 31 /* GreaterThanToken */ && contextToken.parent) {
-                    if (contextToken.parent.kind === 268 /* JsxOpeningElement */) {
+                    if (contextToken.parent.kind === 275 /* JsxOpeningElement */) {
                         // Two possibilities:
                         //   1. <div>/**/
                         //      - contextToken: GreaterThanToken (before cursor)
@@ -113964,10 +119849,10 @@ var ts;
                         //      - contextToken: GreaterThanToken (before cursor)
                         //      - location: GreaterThanToken (after cursor)
                         //      - same parent (JSXOpeningElement)
-                        return location.parent.kind !== 268 /* JsxOpeningElement */;
+                        return location.parent.kind !== 275 /* JsxOpeningElement */;
                     }
-                    if (contextToken.parent.kind === 269 /* JsxClosingElement */ || contextToken.parent.kind === 267 /* JsxSelfClosingElement */) {
-                        return !!contextToken.parent.parent && contextToken.parent.parent.kind === 266 /* JsxElement */;
+                    if (contextToken.parent.kind === 276 /* JsxClosingElement */ || contextToken.parent.kind === 274 /* JsxSelfClosingElement */) {
+                        return !!contextToken.parent.parent && contextToken.parent.parent.kind === 273 /* JsxElement */;
                     }
                 }
                 return false;
@@ -113978,40 +119863,40 @@ var ts;
                     // Previous token may have been a keyword that was converted to an identifier.
                     switch (keywordForNode(previousToken)) {
                         case 27 /* CommaToken */:
-                            return containingNodeKind === 196 /* CallExpression */ // func( a, |
-                                || containingNodeKind === 162 /* Constructor */ // constructor( a, |   /* public, protected, private keywords are allowed here, so show completion */
-                                || containingNodeKind === 197 /* NewExpression */ // new C(a, |
-                                || containingNodeKind === 192 /* ArrayLiteralExpression */ // [a, |
-                                || containingNodeKind === 209 /* BinaryExpression */ // const x = (a, |
-                                || containingNodeKind === 170 /* FunctionType */; // var x: (s: string, list|
+                            return containingNodeKind === 203 /* CallExpression */ // func( a, |
+                                || containingNodeKind === 166 /* Constructor */ // constructor( a, |   /* public, protected, private keywords are allowed here, so show completion */
+                                || containingNodeKind === 204 /* NewExpression */ // new C(a, |
+                                || containingNodeKind === 199 /* ArrayLiteralExpression */ // [a, |
+                                || containingNodeKind === 216 /* BinaryExpression */ // const x = (a, |
+                                || containingNodeKind === 174 /* FunctionType */; // var x: (s: string, list|
                         case 20 /* OpenParenToken */:
-                            return containingNodeKind === 196 /* CallExpression */ // func( |
-                                || containingNodeKind === 162 /* Constructor */ // constructor( |
-                                || containingNodeKind === 197 /* NewExpression */ // new C(a|
-                                || containingNodeKind === 200 /* ParenthesizedExpression */ // const x = (a|
-                                || containingNodeKind === 182 /* ParenthesizedType */; // function F(pred: (a| /* this can become an arrow function, where 'a' is the argument */
+                            return containingNodeKind === 203 /* CallExpression */ // func( |
+                                || containingNodeKind === 166 /* Constructor */ // constructor( |
+                                || containingNodeKind === 204 /* NewExpression */ // new C(a|
+                                || containingNodeKind === 207 /* ParenthesizedExpression */ // const x = (a|
+                                || containingNodeKind === 186 /* ParenthesizedType */; // function F(pred: (a| /* this can become an arrow function, where 'a' is the argument */
                         case 22 /* OpenBracketToken */:
-                            return containingNodeKind === 192 /* ArrayLiteralExpression */ // [ |
-                                || containingNodeKind === 167 /* IndexSignature */ // [ | : string ]
-                                || containingNodeKind === 154 /* ComputedPropertyName */; // [ |    /* this can become an index signature */
-                        case 135 /* ModuleKeyword */: // module |
-                        case 136 /* NamespaceKeyword */: // namespace |
+                            return containingNodeKind === 199 /* ArrayLiteralExpression */ // [ |
+                                || containingNodeKind === 171 /* IndexSignature */ // [ | : string ]
+                                || containingNodeKind === 158 /* ComputedPropertyName */; // [ |    /* this can become an index signature */
+                        case 139 /* ModuleKeyword */: // module |
+                        case 140 /* NamespaceKeyword */: // namespace |
                             return true;
                         case 24 /* DotToken */:
-                            return containingNodeKind === 249 /* ModuleDeclaration */; // module A.|
+                            return containingNodeKind === 256 /* ModuleDeclaration */; // module A.|
                         case 18 /* OpenBraceToken */:
-                            return containingNodeKind === 245 /* ClassDeclaration */; // class A{ |
+                            return containingNodeKind === 252 /* ClassDeclaration */; // class A{ |
                         case 62 /* EqualsToken */:
-                            return containingNodeKind === 242 /* VariableDeclaration */ // const x = a|
-                                || containingNodeKind === 209 /* BinaryExpression */; // x = a|
+                            return containingNodeKind === 249 /* VariableDeclaration */ // const x = a|
+                                || containingNodeKind === 216 /* BinaryExpression */; // x = a|
                         case 15 /* TemplateHead */:
-                            return containingNodeKind === 211 /* TemplateExpression */; // `aa ${|
+                            return containingNodeKind === 218 /* TemplateExpression */; // `aa ${|
                         case 16 /* TemplateMiddle */:
-                            return containingNodeKind === 221 /* TemplateSpan */; // `aa ${10} dd ${|
-                        case 119 /* PublicKeyword */:
-                        case 117 /* PrivateKeyword */:
-                        case 118 /* ProtectedKeyword */:
-                            return containingNodeKind === 159 /* PropertyDeclaration */; // class A{ public |
+                            return containingNodeKind === 228 /* TemplateSpan */; // `aa ${10} dd ${|
+                        case 122 /* PublicKeyword */:
+                        case 120 /* PrivateKeyword */:
+                        case 121 /* ProtectedKeyword */:
+                            return containingNodeKind === 163 /* PropertyDeclaration */; // class A{ public |
                     }
                 }
                 return false;
@@ -114038,17 +119923,18 @@ var ts;
                 completionKind = 0 /* ObjectPropertyDeclaration */;
                 var typeMembers;
                 var existingMembers;
-                if (objectLikeContainer.kind === 193 /* ObjectLiteralExpression */) {
-                    var instantiatedType = typeChecker.getContextualType(objectLikeContainer);
-                    var completionsType = instantiatedType && typeChecker.getContextualType(objectLikeContainer, 4 /* Completions */);
-                    if (!instantiatedType || !completionsType)
+                if (objectLikeContainer.kind === 200 /* ObjectLiteralExpression */) {
+                    var instantiatedType = tryGetObjectLiteralContextualType(objectLikeContainer, typeChecker);
+                    if (instantiatedType === undefined) {
                         return 2 /* Fail */;
-                    isNewIdentifierLocation = ts.hasIndexSignature(instantiatedType || completionsType);
+                    }
+                    var completionsType = typeChecker.getContextualType(objectLikeContainer, 4 /* Completions */);
+                    isNewIdentifierLocation = ts.hasIndexSignature(completionsType || instantiatedType);
                     typeMembers = getPropertiesForObjectExpression(instantiatedType, completionsType, objectLikeContainer, typeChecker);
                     existingMembers = objectLikeContainer.properties;
                 }
                 else {
-                    ts.Debug.assert(objectLikeContainer.kind === 189 /* ObjectBindingPattern */);
+                    ts.Debug.assert(objectLikeContainer.kind === 196 /* ObjectBindingPattern */);
                     // We are *only* completing on properties from the type being destructured.
                     isNewIdentifierLocation = false;
                     var rootDeclaration = ts.getRootDeclaration(objectLikeContainer.parent);
@@ -114059,12 +119945,12 @@ var ts;
                     // through type declaration or inference.
                     // Also proceed if rootDeclaration is a parameter and if its containing function expression/arrow function is contextually typed -
                     // type of parameter will flow in from the contextual type of the function
-                    var canGetType = ts.hasInitializer(rootDeclaration) || ts.hasType(rootDeclaration) || rootDeclaration.parent.parent.kind === 232 /* ForOfStatement */;
-                    if (!canGetType && rootDeclaration.kind === 156 /* Parameter */) {
+                    var canGetType = ts.hasInitializer(rootDeclaration) || ts.hasType(rootDeclaration) || rootDeclaration.parent.parent.kind === 239 /* ForOfStatement */;
+                    if (!canGetType && rootDeclaration.kind === 160 /* Parameter */) {
                         if (ts.isExpression(rootDeclaration.parent)) {
                             canGetType = !!typeChecker.getContextualType(rootDeclaration.parent);
                         }
-                        else if (rootDeclaration.parent.kind === 161 /* MethodDeclaration */ || rootDeclaration.parent.kind === 164 /* SetAccessor */) {
+                        else if (rootDeclaration.parent.kind === 165 /* MethodDeclaration */ || rootDeclaration.parent.kind === 168 /* SetAccessor */) {
                             canGetType = ts.isExpression(rootDeclaration.parent.parent) && !!typeChecker.getContextualType(rootDeclaration.parent.parent);
                         }
                     }
@@ -114111,17 +119997,17 @@ var ts;
                 if (!namedImportsOrExports)
                     return 0 /* Continue */;
                 // try to show exported member for imported/re-exported module
-                var moduleSpecifier = (namedImportsOrExports.kind === 257 /* NamedImports */ ? namedImportsOrExports.parent.parent : namedImportsOrExports.parent).moduleSpecifier;
+                var moduleSpecifier = (namedImportsOrExports.kind === 264 /* NamedImports */ ? namedImportsOrExports.parent.parent : namedImportsOrExports.parent).moduleSpecifier;
                 if (!moduleSpecifier)
-                    return namedImportsOrExports.kind === 257 /* NamedImports */ ? 2 /* Fail */ : 0 /* Continue */;
+                    return namedImportsOrExports.kind === 264 /* NamedImports */ ? 2 /* Fail */ : 0 /* Continue */;
                 var moduleSpecifierSymbol = typeChecker.getSymbolAtLocation(moduleSpecifier); // TODO: GH#18217
                 if (!moduleSpecifierSymbol)
                     return 2 /* Fail */;
                 completionKind = 3 /* MemberLike */;
                 isNewIdentifierLocation = false;
                 var exports = typeChecker.getExportsAndPropertiesOfModule(moduleSpecifierSymbol);
-                var existing = ts.arrayToSet(namedImportsOrExports.elements, function (n) { return isCurrentlyEditingNode(n) ? undefined : (n.propertyName || n.name).escapedText; });
-                symbols = exports.filter(function (e) { return e.escapedName !== "default" /* Default */ && !existing.get(e.escapedName); });
+                var existing = new ts.Set(namedImportsOrExports.elements.filter(function (n) { return !isCurrentlyEditingNode(n); }).map(function (n) { return (n.propertyName || n.name).escapedText; }));
+                symbols = exports.filter(function (e) { return e.escapedName !== "default" /* Default */ && !existing.has(e.escapedName); });
                 return 1 /* Success */;
             }
             /**
@@ -114171,9 +120057,9 @@ var ts;
                 if (!ts.isClassLike(decl))
                     return 1 /* Success */;
                 var classElement = contextToken.kind === 26 /* SemicolonToken */ ? contextToken.parent.parent : contextToken.parent;
-                var classElementModifierFlags = ts.isClassElement(classElement) ? ts.getModifierFlags(classElement) : 0 /* None */;
+                var classElementModifierFlags = ts.isClassElement(classElement) ? ts.getEffectiveModifierFlags(classElement) : 0 /* None */;
                 // If this is context token is not something we are editing now, consider if this would lead to be modifier
-                if (contextToken.kind === 75 /* Identifier */ && !isCurrentlyEditingNode(contextToken)) {
+                if (contextToken.kind === 78 /* Identifier */ && !isCurrentlyEditingNode(contextToken)) {
                     switch (contextToken.getText()) {
                         case "private":
                             classElementModifierFlags = classElementModifierFlags | 8 /* Private */;
@@ -114188,7 +120074,9 @@ var ts;
                     // List of property symbols of base type that are not private and already implemented
                     var baseSymbols = ts.flatMap(ts.getAllSuperTypeNodes(decl), function (baseTypeNode) {
                         var type = typeChecker.getTypeAtLocation(baseTypeNode);
-                        return type && typeChecker.getPropertiesOfType(classElementModifierFlags & 32 /* Static */ ? typeChecker.getTypeOfSymbolAtLocation(type.symbol, decl) : type);
+                        return classElementModifierFlags & 32 /* Static */ ?
+                            (type === null || type === void 0 ? void 0 : type.symbol) && typeChecker.getPropertiesOfType(typeChecker.getTypeOfSymbolAtLocation(type.symbol, decl)) :
+                            type && typeChecker.getPropertiesOfType(type);
                     });
                     symbols = filterClassMembersList(baseSymbols, decl.members, classElementModifierFlags);
                 }
@@ -114210,7 +120098,7 @@ var ts;
                             break;
                         case 41 /* AsteriskToken */:
                             return ts.isMethodDeclaration(parent) ? ts.tryCast(parent.parent, ts.isObjectLiteralExpression) : undefined;
-                        case 75 /* Identifier */:
+                        case 78 /* Identifier */:
                             return contextToken.text === "async" && ts.isShorthandPropertyAssignment(contextToken.parent)
                                 ? contextToken.parent.parent : undefined;
                     }
@@ -114263,12 +120151,12 @@ var ts;
                         case 31 /* GreaterThanToken */: // End of a type argument list
                         case 30 /* LessThanSlashToken */:
                         case 43 /* SlashToken */:
-                        case 75 /* Identifier */:
-                        case 194 /* PropertyAccessExpression */:
-                        case 274 /* JsxAttributes */:
-                        case 273 /* JsxAttribute */:
-                        case 275 /* JsxSpreadAttribute */:
-                            if (parent && (parent.kind === 267 /* JsxSelfClosingElement */ || parent.kind === 268 /* JsxOpeningElement */)) {
+                        case 78 /* Identifier */:
+                        case 201 /* PropertyAccessExpression */:
+                        case 281 /* JsxAttributes */:
+                        case 280 /* JsxAttribute */:
+                        case 282 /* JsxSpreadAttribute */:
+                            if (parent && (parent.kind === 274 /* JsxSelfClosingElement */ || parent.kind === 275 /* JsxOpeningElement */)) {
                                 if (contextToken.kind === 31 /* GreaterThanToken */) {
                                     var precedingToken = ts.findPrecedingToken(contextToken.pos, sourceFile, /*startNode*/ undefined);
                                     if (!parent.typeArguments || (precedingToken && precedingToken.kind === 43 /* SlashToken */))
@@ -114276,7 +120164,7 @@ var ts;
                                 }
                                 return parent;
                             }
-                            else if (parent.kind === 273 /* JsxAttribute */) {
+                            else if (parent.kind === 280 /* JsxAttribute */) {
                                 // Currently we parse JsxOpeningLikeElement as:
                                 //      JsxOpeningLikeElement
                                 //          attributes: JsxAttributes
@@ -114288,7 +120176,7 @@ var ts;
                         // its parent is a JsxExpression, whose parent is a JsxAttribute,
                         // whose parent is a JsxOpeningLikeElement
                         case 10 /* StringLiteral */:
-                            if (parent && ((parent.kind === 273 /* JsxAttribute */) || (parent.kind === 275 /* JsxSpreadAttribute */))) {
+                            if (parent && ((parent.kind === 280 /* JsxAttribute */) || (parent.kind === 282 /* JsxSpreadAttribute */))) {
                                 // Currently we parse JsxOpeningLikeElement as:
                                 //      JsxOpeningLikeElement
                                 //          attributes: JsxAttributes
@@ -114298,8 +120186,8 @@ var ts;
                             break;
                         case 19 /* CloseBraceToken */:
                             if (parent &&
-                                parent.kind === 276 /* JsxExpression */ &&
-                                parent.parent && parent.parent.kind === 273 /* JsxAttribute */) {
+                                parent.kind === 283 /* JsxExpression */ &&
+                                parent.parent && parent.parent.kind === 280 /* JsxAttribute */) {
                                 // Currently we parse JsxOpeningLikeElement as:
                                 //      JsxOpeningLikeElement
                                 //          attributes: JsxAttributes
@@ -114307,7 +120195,7 @@ var ts;
                                 //                  each JsxAttribute can have initializer as JsxExpression
                                 return parent.parent.parent.parent;
                             }
-                            if (parent && parent.kind === 275 /* JsxSpreadAttribute */) {
+                            if (parent && parent.kind === 282 /* JsxSpreadAttribute */) {
                                 // Currently we parse JsxOpeningLikeElement as:
                                 //      JsxOpeningLikeElement
                                 //          attributes: JsxAttributes
@@ -114327,61 +120215,61 @@ var ts;
                 var containingNodeKind = parent.kind;
                 switch (contextToken.kind) {
                     case 27 /* CommaToken */:
-                        return containingNodeKind === 242 /* VariableDeclaration */ ||
+                        return containingNodeKind === 249 /* VariableDeclaration */ ||
                             isVariableDeclarationListButNotTypeArgument(contextToken) ||
-                            containingNodeKind === 225 /* VariableStatement */ ||
-                            containingNodeKind === 248 /* EnumDeclaration */ || // enum a { foo, |
+                            containingNodeKind === 232 /* VariableStatement */ ||
+                            containingNodeKind === 255 /* EnumDeclaration */ || // enum a { foo, |
                             isFunctionLikeButNotConstructor(containingNodeKind) ||
-                            containingNodeKind === 246 /* InterfaceDeclaration */ || // interface A<T, |
-                            containingNodeKind === 190 /* ArrayBindingPattern */ || // var [x, y|
-                            containingNodeKind === 247 /* TypeAliasDeclaration */ || // type Map, K, |
+                            containingNodeKind === 253 /* InterfaceDeclaration */ || // interface A<T, |
+                            containingNodeKind === 197 /* ArrayBindingPattern */ || // var [x, y|
+                            containingNodeKind === 254 /* TypeAliasDeclaration */ || // type Map, K, |
                             // class A<T, |
                             // var C = class D<T, |
                             (ts.isClassLike(parent) &&
                                 !!parent.typeParameters &&
                                 parent.typeParameters.end >= contextToken.pos);
                     case 24 /* DotToken */:
-                        return containingNodeKind === 190 /* ArrayBindingPattern */; // var [.|
+                        return containingNodeKind === 197 /* ArrayBindingPattern */; // var [.|
                     case 58 /* ColonToken */:
-                        return containingNodeKind === 191 /* BindingElement */; // var {x :html|
+                        return containingNodeKind === 198 /* BindingElement */; // var {x :html|
                     case 22 /* OpenBracketToken */:
-                        return containingNodeKind === 190 /* ArrayBindingPattern */; // var [x|
+                        return containingNodeKind === 197 /* ArrayBindingPattern */; // var [x|
                     case 20 /* OpenParenToken */:
-                        return containingNodeKind === 280 /* CatchClause */ ||
+                        return containingNodeKind === 287 /* CatchClause */ ||
                             isFunctionLikeButNotConstructor(containingNodeKind);
                     case 18 /* OpenBraceToken */:
-                        return containingNodeKind === 248 /* EnumDeclaration */; // enum a { |
+                        return containingNodeKind === 255 /* EnumDeclaration */; // enum a { |
                     case 29 /* LessThanToken */:
-                        return containingNodeKind === 245 /* ClassDeclaration */ || // class A< |
-                            containingNodeKind === 214 /* ClassExpression */ || // var C = class D< |
-                            containingNodeKind === 246 /* InterfaceDeclaration */ || // interface A< |
-                            containingNodeKind === 247 /* TypeAliasDeclaration */ || // type List< |
+                        return containingNodeKind === 252 /* ClassDeclaration */ || // class A< |
+                            containingNodeKind === 221 /* ClassExpression */ || // var C = class D< |
+                            containingNodeKind === 253 /* InterfaceDeclaration */ || // interface A< |
+                            containingNodeKind === 254 /* TypeAliasDeclaration */ || // type List< |
                             ts.isFunctionLikeKind(containingNodeKind);
-                    case 120 /* StaticKeyword */:
-                        return containingNodeKind === 159 /* PropertyDeclaration */ && !ts.isClassLike(parent.parent);
+                    case 123 /* StaticKeyword */:
+                        return containingNodeKind === 163 /* PropertyDeclaration */ && !ts.isClassLike(parent.parent);
                     case 25 /* DotDotDotToken */:
-                        return containingNodeKind === 156 /* Parameter */ ||
-                            (!!parent.parent && parent.parent.kind === 190 /* ArrayBindingPattern */); // var [...z|
-                    case 119 /* PublicKeyword */:
-                    case 117 /* PrivateKeyword */:
-                    case 118 /* ProtectedKeyword */:
-                        return containingNodeKind === 156 /* Parameter */ && !ts.isConstructorDeclaration(parent.parent);
-                    case 123 /* AsKeyword */:
-                        return containingNodeKind === 258 /* ImportSpecifier */ ||
-                            containingNodeKind === 263 /* ExportSpecifier */ ||
-                            containingNodeKind === 256 /* NamespaceImport */;
-                    case 131 /* GetKeyword */:
-                    case 142 /* SetKeyword */:
+                        return containingNodeKind === 160 /* Parameter */ ||
+                            (!!parent.parent && parent.parent.kind === 197 /* ArrayBindingPattern */); // var [...z|
+                    case 122 /* PublicKeyword */:
+                    case 120 /* PrivateKeyword */:
+                    case 121 /* ProtectedKeyword */:
+                        return containingNodeKind === 160 /* Parameter */ && !ts.isConstructorDeclaration(parent.parent);
+                    case 126 /* AsKeyword */:
+                        return containingNodeKind === 265 /* ImportSpecifier */ ||
+                            containingNodeKind === 270 /* ExportSpecifier */ ||
+                            containingNodeKind === 263 /* NamespaceImport */;
+                    case 134 /* GetKeyword */:
+                    case 146 /* SetKeyword */:
                         return !isFromObjectTypeDeclaration(contextToken);
-                    case 80 /* ClassKeyword */:
-                    case 88 /* EnumKeyword */:
-                    case 114 /* InterfaceKeyword */:
-                    case 94 /* FunctionKeyword */:
-                    case 109 /* VarKeyword */:
-                    case 96 /* ImportKeyword */:
-                    case 115 /* LetKeyword */:
-                    case 81 /* ConstKeyword */:
-                    case 145 /* TypeKeyword */: // type htm|
+                    case 83 /* ClassKeyword */:
+                    case 91 /* EnumKeyword */:
+                    case 117 /* InterfaceKeyword */:
+                    case 97 /* FunctionKeyword */:
+                    case 112 /* VarKeyword */:
+                    case 99 /* ImportKeyword */:
+                    case 118 /* LetKeyword */:
+                    case 84 /* ConstKeyword */:
+                    case 149 /* TypeKeyword */: // type htm|
                         return true;
                     case 41 /* AsteriskToken */:
                         return ts.isFunctionLike(contextToken.parent) && !ts.isMethodDeclaration(contextToken.parent);
@@ -114404,21 +120292,21 @@ var ts;
                 }
                 // Previous token may have been a keyword that was converted to an identifier.
                 switch (keywordForNode(contextToken)) {
-                    case 122 /* AbstractKeyword */:
-                    case 80 /* ClassKeyword */:
-                    case 81 /* ConstKeyword */:
-                    case 130 /* DeclareKeyword */:
-                    case 88 /* EnumKeyword */:
-                    case 94 /* FunctionKeyword */:
-                    case 114 /* InterfaceKeyword */:
-                    case 115 /* LetKeyword */:
-                    case 117 /* PrivateKeyword */:
-                    case 118 /* ProtectedKeyword */:
-                    case 119 /* PublicKeyword */:
-                    case 120 /* StaticKeyword */:
-                    case 109 /* VarKeyword */:
+                    case 125 /* AbstractKeyword */:
+                    case 83 /* ClassKeyword */:
+                    case 84 /* ConstKeyword */:
+                    case 133 /* DeclareKeyword */:
+                    case 91 /* EnumKeyword */:
+                    case 97 /* FunctionKeyword */:
+                    case 117 /* InterfaceKeyword */:
+                    case 118 /* LetKeyword */:
+                    case 120 /* PrivateKeyword */:
+                    case 121 /* ProtectedKeyword */:
+                    case 122 /* PublicKeyword */:
+                    case 123 /* StaticKeyword */:
+                    case 112 /* VarKeyword */:
                         return true;
-                    case 126 /* AsyncKeyword */:
+                    case 129 /* AsyncKeyword */:
                         return ts.isPropertyDeclaration(contextToken.parent);
                 }
                 return ts.isDeclarationName(contextToken)
@@ -114428,7 +120316,7 @@ var ts;
                     && !(ts.isClassLike(contextToken.parent) && (contextToken !== previousToken || position > previousToken.end));
             }
             function isFunctionLikeButNotConstructor(kind) {
-                return ts.isFunctionLikeKind(kind) && kind !== 162 /* Constructor */;
+                return ts.isFunctionLikeKind(kind) && kind !== 166 /* Constructor */;
             }
             function isDotOfNumericLiteral(contextToken) {
                 if (contextToken.kind === 8 /* NumericLiteral */) {
@@ -114438,7 +120326,7 @@ var ts;
                 return false;
             }
             function isVariableDeclarationListButNotTypeArgument(node) {
-                return node.parent.kind === 243 /* VariableDeclarationList */
+                return node.parent.kind === 250 /* VariableDeclarationList */
                     && !ts.isPossiblyTypeArgumentPosition(node, sourceFile, typeChecker);
             }
             /**
@@ -114451,18 +120339,18 @@ var ts;
                 if (existingMembers.length === 0) {
                     return contextualMemberSymbols;
                 }
-                var membersDeclaredBySpreadAssignment = ts.createMap();
-                var existingMemberNames = ts.createUnderscoreEscapedMap();
+                var membersDeclaredBySpreadAssignment = new ts.Set();
+                var existingMemberNames = new ts.Set();
                 for (var _i = 0, existingMembers_1 = existingMembers; _i < existingMembers_1.length; _i++) {
                     var m = existingMembers_1[_i];
                     // Ignore omitted expressions for missing members
-                    if (m.kind !== 281 /* PropertyAssignment */ &&
-                        m.kind !== 282 /* ShorthandPropertyAssignment */ &&
-                        m.kind !== 191 /* BindingElement */ &&
-                        m.kind !== 161 /* MethodDeclaration */ &&
-                        m.kind !== 163 /* GetAccessor */ &&
-                        m.kind !== 164 /* SetAccessor */ &&
-                        m.kind !== 283 /* SpreadAssignment */) {
+                    if (m.kind !== 288 /* PropertyAssignment */ &&
+                        m.kind !== 289 /* ShorthandPropertyAssignment */ &&
+                        m.kind !== 198 /* BindingElement */ &&
+                        m.kind !== 165 /* MethodDeclaration */ &&
+                        m.kind !== 167 /* GetAccessor */ &&
+                        m.kind !== 168 /* SetAccessor */ &&
+                        m.kind !== 290 /* SpreadAssignment */) {
                         continue;
                     }
                     // If this is the current item we are editing right now, do not filter it out
@@ -114475,7 +120363,7 @@ var ts;
                     }
                     else if (ts.isBindingElement(m) && m.propertyName) {
                         // include only identifiers in completion list
-                        if (m.propertyName.kind === 75 /* Identifier */) {
+                        if (m.propertyName.kind === 78 /* Identifier */) {
                             existingName = m.propertyName.escapedText;
                         }
                     }
@@ -114486,9 +120374,11 @@ var ts;
                         var name = ts.getNameOfDeclaration(m);
                         existingName = name && ts.isPropertyNameLiteral(name) ? ts.getEscapedTextOfIdentifierOrLiteral(name) : undefined;
                     }
-                    existingMemberNames.set(existingName, true); // TODO: GH#18217
+                    if (existingName !== undefined) {
+                        existingMemberNames.add(existingName);
+                    }
                 }
-                var filteredSymbols = contextualMemberSymbols.filter(function (m) { return !existingMemberNames.get(m.escapedName); });
+                var filteredSymbols = contextualMemberSymbols.filter(function (m) { return !existingMemberNames.has(m.escapedName); });
                 setSortTextToMemberDeclaredBySpreadAssignment(membersDeclaredBySpreadAssignment, filteredSymbols);
                 return filteredSymbols;
             }
@@ -114499,7 +120389,7 @@ var ts;
                 var properties = type && type.properties;
                 if (properties) {
                     properties.forEach(function (property) {
-                        membersDeclaredBySpreadAssignment.set(property.name, true);
+                        membersDeclaredBySpreadAssignment.add(property.name);
                     });
                 }
             }
@@ -114529,14 +120419,14 @@ var ts;
              * @returns Symbols to be suggested in an class element depending on existing memebers and symbol flags
              */
             function filterClassMembersList(baseSymbols, existingMembers, currentClassElementModifierFlags) {
-                var existingMemberNames = ts.createUnderscoreEscapedMap();
+                var existingMemberNames = new ts.Set();
                 for (var _i = 0, existingMembers_2 = existingMembers; _i < existingMembers_2.length; _i++) {
                     var m = existingMembers_2[_i];
                     // Ignore omitted expressions for missing members
-                    if (m.kind !== 159 /* PropertyDeclaration */ &&
-                        m.kind !== 161 /* MethodDeclaration */ &&
-                        m.kind !== 163 /* GetAccessor */ &&
-                        m.kind !== 164 /* SetAccessor */) {
+                    if (m.kind !== 163 /* PropertyDeclaration */ &&
+                        m.kind !== 165 /* MethodDeclaration */ &&
+                        m.kind !== 167 /* GetAccessor */ &&
+                        m.kind !== 168 /* SetAccessor */) {
                         continue;
                     }
                     // If this is the current item we are editing right now, do not filter it out
@@ -114544,23 +120434,23 @@ var ts;
                         continue;
                     }
                     // Dont filter member even if the name matches if it is declared private in the list
-                    if (ts.hasModifier(m, 8 /* Private */)) {
+                    if (ts.hasEffectiveModifier(m, 8 /* Private */)) {
                         continue;
                     }
                     // do not filter it out if the static presence doesnt match
-                    if (ts.hasModifier(m, 32 /* Static */) !== !!(currentClassElementModifierFlags & 32 /* Static */)) {
+                    if (ts.hasEffectiveModifier(m, 32 /* Static */) !== !!(currentClassElementModifierFlags & 32 /* Static */)) {
                         continue;
                     }
                     var existingName = ts.getPropertyNameForPropertyNameNode(m.name);
                     if (existingName) {
-                        existingMemberNames.set(existingName, true);
+                        existingMemberNames.add(existingName);
                     }
                 }
                 return baseSymbols.filter(function (propertySymbol) {
                     return !existingMemberNames.has(propertySymbol.escapedName) &&
                         !!propertySymbol.declarations &&
                         !(ts.getDeclarationModifierFlagsFromSymbol(propertySymbol) & 8 /* Private */) &&
-                        !ts.isPrivateIdentifierPropertyDeclaration(propertySymbol.valueDeclaration);
+                        !(propertySymbol.valueDeclaration && ts.isPrivateIdentifierPropertyDeclaration(propertySymbol.valueDeclaration));
                 });
             }
             /**
@@ -114570,22 +120460,22 @@ var ts;
              *          do not occur at the current position and have not otherwise been typed.
              */
             function filterJsxAttributes(symbols, attributes) {
-                var seenNames = ts.createUnderscoreEscapedMap();
-                var membersDeclaredBySpreadAssignment = ts.createMap();
+                var seenNames = new ts.Set();
+                var membersDeclaredBySpreadAssignment = new ts.Set();
                 for (var _i = 0, attributes_1 = attributes; _i < attributes_1.length; _i++) {
                     var attr = attributes_1[_i];
                     // If this is the current item we are editing right now, do not filter it out
                     if (isCurrentlyEditingNode(attr)) {
                         continue;
                     }
-                    if (attr.kind === 273 /* JsxAttribute */) {
-                        seenNames.set(attr.name.escapedText, true);
+                    if (attr.kind === 280 /* JsxAttribute */) {
+                        seenNames.add(attr.name.escapedText);
                     }
                     else if (ts.isJsxSpreadAttribute(attr)) {
                         setMembersDeclaredBySpreadAssignment(attr, membersDeclaredBySpreadAssignment);
                     }
                 }
-                var filteredSymbols = symbols.filter(function (a) { return !seenNames.get(a.escapedName); });
+                var filteredSymbols = symbols.filter(function (a) { return !seenNames.has(a.escapedName); });
                 setSortTextToMemberDeclaredBySpreadAssignment(membersDeclaredBySpreadAssignment, filteredSymbols);
                 return filteredSymbols;
             }
@@ -114628,7 +120518,7 @@ var ts;
         var _keywordCompletions = [];
         var allKeywordsCompletions = ts.memoize(function () {
             var res = [];
-            for (var i = 77 /* FirstKeyword */; i <= 152 /* LastKeyword */; i++) {
+            for (var i = 80 /* FirstKeyword */; i <= 156 /* LastKeyword */; i++) {
                 res.push({
                     name: ts.tokenToString(i),
                     kind: "keyword" /* keyword */,
@@ -114654,12 +120544,11 @@ var ts;
                         return false;
                     case 1 /* All */:
                         return isFunctionLikeBodyKeyword(kind)
-                            || kind === 130 /* DeclareKeyword */
-                            || kind === 135 /* ModuleKeyword */
-                            || kind === 145 /* TypeKeyword */
-                            || kind === 136 /* NamespaceKeyword */
-                            || kind === 123 /* AsKeyword */
-                            || ts.isTypeKeyword(kind) && kind !== 146 /* UndefinedKeyword */;
+                            || kind === 133 /* DeclareKeyword */
+                            || kind === 139 /* ModuleKeyword */
+                            || kind === 149 /* TypeKeyword */
+                            || kind === 140 /* NamespaceKeyword */
+                            || ts.isTypeKeyword(kind) && kind !== 150 /* UndefinedKeyword */;
                     case 5 /* FunctionLikeBodyKeywords */:
                         return isFunctionLikeBodyKeyword(kind);
                     case 2 /* ClassElementKeywords */:
@@ -114669,7 +120558,7 @@ var ts;
                     case 4 /* ConstructorParameterKeywords */:
                         return ts.isParameterPropertyModifier(kind);
                     case 6 /* TypeAssertionKeywords */:
-                        return ts.isTypeKeyword(kind) || kind === 81 /* ConstKeyword */;
+                        return ts.isTypeKeyword(kind) || kind === 84 /* ConstKeyword */;
                     case 7 /* TypeKeywords */:
                         return ts.isTypeKeyword(kind);
                     default:
@@ -114679,56 +120568,57 @@ var ts;
         }
         function isTypeScriptOnlyKeyword(kind) {
             switch (kind) {
-                case 122 /* AbstractKeyword */:
-                case 125 /* AnyKeyword */:
-                case 151 /* BigIntKeyword */:
-                case 128 /* BooleanKeyword */:
-                case 130 /* DeclareKeyword */:
-                case 88 /* EnumKeyword */:
-                case 150 /* GlobalKeyword */:
-                case 113 /* ImplementsKeyword */:
-                case 132 /* InferKeyword */:
-                case 114 /* InterfaceKeyword */:
-                case 133 /* IsKeyword */:
-                case 134 /* KeyOfKeyword */:
-                case 135 /* ModuleKeyword */:
-                case 136 /* NamespaceKeyword */:
-                case 137 /* NeverKeyword */:
-                case 140 /* NumberKeyword */:
-                case 141 /* ObjectKeyword */:
-                case 117 /* PrivateKeyword */:
-                case 118 /* ProtectedKeyword */:
-                case 119 /* PublicKeyword */:
-                case 138 /* ReadonlyKeyword */:
-                case 143 /* StringKeyword */:
-                case 144 /* SymbolKeyword */:
-                case 145 /* TypeKeyword */:
-                case 147 /* UniqueKeyword */:
-                case 148 /* UnknownKeyword */:
+                case 125 /* AbstractKeyword */:
+                case 128 /* AnyKeyword */:
+                case 155 /* BigIntKeyword */:
+                case 131 /* BooleanKeyword */:
+                case 133 /* DeclareKeyword */:
+                case 91 /* EnumKeyword */:
+                case 154 /* GlobalKeyword */:
+                case 116 /* ImplementsKeyword */:
+                case 135 /* InferKeyword */:
+                case 117 /* InterfaceKeyword */:
+                case 137 /* IsKeyword */:
+                case 138 /* KeyOfKeyword */:
+                case 139 /* ModuleKeyword */:
+                case 140 /* NamespaceKeyword */:
+                case 141 /* NeverKeyword */:
+                case 144 /* NumberKeyword */:
+                case 145 /* ObjectKeyword */:
+                case 120 /* PrivateKeyword */:
+                case 121 /* ProtectedKeyword */:
+                case 122 /* PublicKeyword */:
+                case 142 /* ReadonlyKeyword */:
+                case 147 /* StringKeyword */:
+                case 148 /* SymbolKeyword */:
+                case 149 /* TypeKeyword */:
+                case 151 /* UniqueKeyword */:
+                case 152 /* UnknownKeyword */:
                     return true;
                 default:
                     return false;
             }
         }
         function isInterfaceOrTypeLiteralCompletionKeyword(kind) {
-            return kind === 138 /* ReadonlyKeyword */;
+            return kind === 142 /* ReadonlyKeyword */;
         }
         function isClassMemberCompletionKeyword(kind) {
             switch (kind) {
-                case 122 /* AbstractKeyword */:
-                case 129 /* ConstructorKeyword */:
-                case 131 /* GetKeyword */:
-                case 142 /* SetKeyword */:
-                case 126 /* AsyncKeyword */:
-                case 130 /* DeclareKeyword */:
+                case 125 /* AbstractKeyword */:
+                case 132 /* ConstructorKeyword */:
+                case 134 /* GetKeyword */:
+                case 146 /* SetKeyword */:
+                case 129 /* AsyncKeyword */:
+                case 133 /* DeclareKeyword */:
                     return true;
                 default:
                     return ts.isClassMemberModifier(kind);
             }
         }
         function isFunctionLikeBodyKeyword(kind) {
-            return kind === 126 /* AsyncKeyword */
-                || kind === 127 /* AwaitKeyword */
+            return kind === 129 /* AsyncKeyword */
+                || kind === 130 /* AwaitKeyword */
+                || kind === 126 /* AsKeyword */
                 || !ts.isContextualKeyword(kind) && !isClassMemberCompletionKeyword(kind);
         }
         function keywordForNode(node) {
@@ -114779,7 +120669,7 @@ var ts;
         function tryGetObjectTypeDeclarationCompletionContainer(sourceFile, contextToken, location, position) {
             // class c { method() { } | method2() { } }
             switch (location.kind) {
-                case 324 /* SyntaxList */:
+                case 334 /* SyntaxList */:
                     return ts.tryCast(location.parent, ts.isObjectTypeDeclaration);
                 case 1 /* EndOfFileToken */:
                     var cls = ts.tryCast(ts.lastOrUndefined(ts.cast(location.parent, ts.isSourceFile).statements), ts.isObjectTypeDeclaration);
@@ -114787,7 +120677,7 @@ var ts;
                         return cls;
                     }
                     break;
-                case 75 /* Identifier */: {
+                case 78 /* Identifier */: {
                     // class c { public prop = c| }
                     if (ts.isPropertyDeclaration(location.parent) && location.parent.initializer === location) {
                         return undefined;
@@ -114882,6 +120772,19 @@ var ts;
             }
             return false;
         }
+        function isStaticProperty(symbol) {
+            return !!(symbol.valueDeclaration && ts.getEffectiveModifierFlags(symbol.valueDeclaration) & 32 /* Static */ && ts.isClassLike(symbol.valueDeclaration.parent));
+        }
+        function tryGetObjectLiteralContextualType(node, typeChecker) {
+            var type = typeChecker.getContextualType(node);
+            if (type) {
+                return type;
+            }
+            if (ts.isBinaryExpression(node.parent) && node.parent.operatorToken.kind === 62 /* EqualsToken */) {
+                return typeChecker.getTypeAtLocation(node.parent);
+            }
+            return undefined;
+        }
     })(Completions = ts.Completions || (ts.Completions = {}));
 })(ts || (ts = {}));
 var ts;
@@ -114911,7 +120814,7 @@ var ts;
             };
         }
         function getSemanticDocumentHighlights(position, node, program, cancellationToken, sourceFilesToSearch) {
-            var sourceFilesSet = ts.arrayToSet(sourceFilesToSearch, function (f) { return f.fileName; });
+            var sourceFilesSet = new ts.Set(sourceFilesToSearch.map(function (f) { return f.fileName; }));
             var referenceEntries = ts.FindAllReferences.getReferenceEntriesForNode(position, node, program, sourceFilesToSearch, cancellationToken, /*options*/ undefined, sourceFilesSet);
             if (!referenceEntries)
                 return undefined;
@@ -114934,44 +120837,44 @@ var ts;
         }
         function getHighlightSpans(node, sourceFile) {
             switch (node.kind) {
-                case 95 /* IfKeyword */:
-                case 87 /* ElseKeyword */:
+                case 98 /* IfKeyword */:
+                case 90 /* ElseKeyword */:
                     return ts.isIfStatement(node.parent) ? getIfElseOccurrences(node.parent, sourceFile) : undefined;
-                case 101 /* ReturnKeyword */:
+                case 104 /* ReturnKeyword */:
                     return useParent(node.parent, ts.isReturnStatement, getReturnOccurrences);
-                case 105 /* ThrowKeyword */:
+                case 108 /* ThrowKeyword */:
                     return useParent(node.parent, ts.isThrowStatement, getThrowOccurrences);
-                case 107 /* TryKeyword */:
-                case 79 /* CatchKeyword */:
-                case 92 /* FinallyKeyword */:
-                    var tryStatement = node.kind === 79 /* CatchKeyword */ ? node.parent.parent : node.parent;
+                case 110 /* TryKeyword */:
+                case 82 /* CatchKeyword */:
+                case 95 /* FinallyKeyword */:
+                    var tryStatement = node.kind === 82 /* CatchKeyword */ ? node.parent.parent : node.parent;
                     return useParent(tryStatement, ts.isTryStatement, getTryCatchFinallyOccurrences);
-                case 103 /* SwitchKeyword */:
+                case 106 /* SwitchKeyword */:
                     return useParent(node.parent, ts.isSwitchStatement, getSwitchCaseDefaultOccurrences);
-                case 78 /* CaseKeyword */:
-                case 84 /* DefaultKeyword */: {
+                case 81 /* CaseKeyword */:
+                case 87 /* DefaultKeyword */: {
                     if (ts.isDefaultClause(node.parent) || ts.isCaseClause(node.parent)) {
                         return useParent(node.parent.parent.parent, ts.isSwitchStatement, getSwitchCaseDefaultOccurrences);
                     }
                     return undefined;
                 }
-                case 77 /* BreakKeyword */:
-                case 82 /* ContinueKeyword */:
+                case 80 /* BreakKeyword */:
+                case 85 /* ContinueKeyword */:
                     return useParent(node.parent, ts.isBreakOrContinueStatement, getBreakOrContinueStatementOccurrences);
-                case 93 /* ForKeyword */:
-                case 111 /* WhileKeyword */:
-                case 86 /* DoKeyword */:
+                case 96 /* ForKeyword */:
+                case 114 /* WhileKeyword */:
+                case 89 /* DoKeyword */:
                     return useParent(node.parent, function (n) { return ts.isIterationStatement(n, /*lookInLabeledStatements*/ true); }, getLoopBreakContinueOccurrences);
-                case 129 /* ConstructorKeyword */:
-                    return getFromAllDeclarations(ts.isConstructorDeclaration, [129 /* ConstructorKeyword */]);
-                case 131 /* GetKeyword */:
-                case 142 /* SetKeyword */:
-                    return getFromAllDeclarations(ts.isAccessor, [131 /* GetKeyword */, 142 /* SetKeyword */]);
-                case 127 /* AwaitKeyword */:
+                case 132 /* ConstructorKeyword */:
+                    return getFromAllDeclarations(ts.isConstructorDeclaration, [132 /* ConstructorKeyword */]);
+                case 134 /* GetKeyword */:
+                case 146 /* SetKeyword */:
+                    return getFromAllDeclarations(ts.isAccessor, [134 /* GetKeyword */, 146 /* SetKeyword */]);
+                case 130 /* AwaitKeyword */:
                     return useParent(node.parent, ts.isAwaitExpression, getAsyncAndAwaitOccurrences);
-                case 126 /* AsyncKeyword */:
+                case 129 /* AsyncKeyword */:
                     return highlightSpans(getAsyncAndAwaitOccurrences(node));
-                case 121 /* YieldKeyword */:
+                case 124 /* YieldKeyword */:
                     return highlightSpans(getYieldOccurrences(node));
                 default:
                     return ts.isModifierKind(node.kind) && (ts.isDeclaration(node.parent) || ts.isVariableStatement(node.parent))
@@ -115014,7 +120917,7 @@ var ts;
             var child = throwStatement;
             while (child.parent) {
                 var parent = child.parent;
-                if (ts.isFunctionBlock(parent) || parent.kind === 290 /* SourceFile */) {
+                if (ts.isFunctionBlock(parent) || parent.kind === 297 /* SourceFile */) {
                     return parent;
                 }
                 // A throw-statement is only owned by a try-statement if the try-statement has
@@ -115046,16 +120949,16 @@ var ts;
         function getBreakOrContinueOwner(statement) {
             return ts.findAncestor(statement, function (node) {
                 switch (node.kind) {
-                    case 237 /* SwitchStatement */:
-                        if (statement.kind === 233 /* ContinueStatement */) {
+                    case 244 /* SwitchStatement */:
+                        if (statement.kind === 240 /* ContinueStatement */) {
                             return false;
                         }
                     // falls through
-                    case 230 /* ForStatement */:
-                    case 231 /* ForInStatement */:
-                    case 232 /* ForOfStatement */:
-                    case 229 /* WhileStatement */:
-                    case 228 /* DoStatement */:
+                    case 237 /* ForStatement */:
+                    case 238 /* ForInStatement */:
+                    case 239 /* ForOfStatement */:
+                    case 236 /* WhileStatement */:
+                    case 235 /* DoStatement */:
                         return !statement.label || isLabeledBy(node, statement.label.escapedText);
                     default:
                         // Don't cross function boundaries.
@@ -115071,11 +120974,11 @@ var ts;
             // Types of node whose children might have modifiers.
             var container = declaration.parent;
             switch (container.kind) {
-                case 250 /* ModuleBlock */:
-                case 290 /* SourceFile */:
-                case 223 /* Block */:
-                case 277 /* CaseClause */:
-                case 278 /* DefaultClause */:
+                case 257 /* ModuleBlock */:
+                case 297 /* SourceFile */:
+                case 230 /* Block */:
+                case 284 /* CaseClause */:
+                case 285 /* DefaultClause */:
                     // Container is either a class declaration or the declaration is a classDeclaration
                     if (modifierFlag & 128 /* Abstract */ && ts.isClassDeclaration(declaration)) {
                         return __spreadArrays(declaration.members, [declaration]);
@@ -115083,14 +120986,14 @@ var ts;
                     else {
                         return container.statements;
                     }
-                case 162 /* Constructor */:
-                case 161 /* MethodDeclaration */:
-                case 244 /* FunctionDeclaration */:
+                case 166 /* Constructor */:
+                case 165 /* MethodDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     return __spreadArrays(container.parameters, (ts.isClassLike(container.parent) ? container.parent.members : []));
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 246 /* InterfaceDeclaration */:
-                case 173 /* TypeLiteral */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 253 /* InterfaceDeclaration */:
+                case 177 /* TypeLiteral */:
                     var nodes = container.members;
                     // If we're an accessibility modifier, we're in an instance member and should search
                     // the constructor's parameter list for instance members as well.
@@ -115104,6 +121007,9 @@ var ts;
                         return __spreadArrays(nodes, [container]);
                     }
                     return nodes;
+                // Syntactically invalid positions that the parser might produce anyway
+                case 200 /* ObjectLiteralExpression */:
+                    return undefined;
                 default:
                     ts.Debug.assertNever(container, "Invalid container kind.");
             }
@@ -115121,12 +121027,12 @@ var ts;
         }
         function getLoopBreakContinueOccurrences(loopNode) {
             var keywords = [];
-            if (pushKeywordIf(keywords, loopNode.getFirstToken(), 93 /* ForKeyword */, 111 /* WhileKeyword */, 86 /* DoKeyword */)) {
+            if (pushKeywordIf(keywords, loopNode.getFirstToken(), 96 /* ForKeyword */, 114 /* WhileKeyword */, 89 /* DoKeyword */)) {
                 // If we succeeded and got a do-while loop, then start looking for a 'while' keyword.
-                if (loopNode.kind === 228 /* DoStatement */) {
+                if (loopNode.kind === 235 /* DoStatement */) {
                     var loopTokens = loopNode.getChildren();
                     for (var i = loopTokens.length - 1; i >= 0; i--) {
-                        if (pushKeywordIf(keywords, loopTokens[i], 111 /* WhileKeyword */)) {
+                        if (pushKeywordIf(keywords, loopTokens[i], 114 /* WhileKeyword */)) {
                             break;
                         }
                     }
@@ -115134,7 +121040,7 @@ var ts;
             }
             ts.forEach(aggregateAllBreakAndContinueStatements(loopNode.statement), function (statement) {
                 if (ownsBreakOrContinueStatement(loopNode, statement)) {
-                    pushKeywordIf(keywords, statement.getFirstToken(), 77 /* BreakKeyword */, 82 /* ContinueKeyword */);
+                    pushKeywordIf(keywords, statement.getFirstToken(), 80 /* BreakKeyword */, 85 /* ContinueKeyword */);
                 }
             });
             return keywords;
@@ -115143,13 +121049,13 @@ var ts;
             var owner = getBreakOrContinueOwner(breakOrContinueStatement);
             if (owner) {
                 switch (owner.kind) {
-                    case 230 /* ForStatement */:
-                    case 231 /* ForInStatement */:
-                    case 232 /* ForOfStatement */:
-                    case 228 /* DoStatement */:
-                    case 229 /* WhileStatement */:
+                    case 237 /* ForStatement */:
+                    case 238 /* ForInStatement */:
+                    case 239 /* ForOfStatement */:
+                    case 235 /* DoStatement */:
+                    case 236 /* WhileStatement */:
                         return getLoopBreakContinueOccurrences(owner);
-                    case 237 /* SwitchStatement */:
+                    case 244 /* SwitchStatement */:
                         return getSwitchCaseDefaultOccurrences(owner);
                 }
             }
@@ -115157,13 +121063,13 @@ var ts;
         }
         function getSwitchCaseDefaultOccurrences(switchStatement) {
             var keywords = [];
-            pushKeywordIf(keywords, switchStatement.getFirstToken(), 103 /* SwitchKeyword */);
+            pushKeywordIf(keywords, switchStatement.getFirstToken(), 106 /* SwitchKeyword */);
             // Go through each clause in the switch statement, collecting the 'case'/'default' keywords.
             ts.forEach(switchStatement.caseBlock.clauses, function (clause) {
-                pushKeywordIf(keywords, clause.getFirstToken(), 78 /* CaseKeyword */, 84 /* DefaultKeyword */);
+                pushKeywordIf(keywords, clause.getFirstToken(), 81 /* CaseKeyword */, 87 /* DefaultKeyword */);
                 ts.forEach(aggregateAllBreakAndContinueStatements(clause), function (statement) {
                     if (ownsBreakOrContinueStatement(switchStatement, statement)) {
-                        pushKeywordIf(keywords, statement.getFirstToken(), 77 /* BreakKeyword */);
+                        pushKeywordIf(keywords, statement.getFirstToken(), 80 /* BreakKeyword */);
                     }
                 });
             });
@@ -115171,13 +121077,13 @@ var ts;
         }
         function getTryCatchFinallyOccurrences(tryStatement, sourceFile) {
             var keywords = [];
-            pushKeywordIf(keywords, tryStatement.getFirstToken(), 107 /* TryKeyword */);
+            pushKeywordIf(keywords, tryStatement.getFirstToken(), 110 /* TryKeyword */);
             if (tryStatement.catchClause) {
-                pushKeywordIf(keywords, tryStatement.catchClause.getFirstToken(), 79 /* CatchKeyword */);
+                pushKeywordIf(keywords, tryStatement.catchClause.getFirstToken(), 82 /* CatchKeyword */);
             }
             if (tryStatement.finallyBlock) {
-                var finallyKeyword = ts.findChildOfKind(tryStatement, 92 /* FinallyKeyword */, sourceFile);
-                pushKeywordIf(keywords, finallyKeyword, 92 /* FinallyKeyword */);
+                var finallyKeyword = ts.findChildOfKind(tryStatement, 95 /* FinallyKeyword */, sourceFile);
+                pushKeywordIf(keywords, finallyKeyword, 95 /* FinallyKeyword */);
             }
             return keywords;
         }
@@ -115188,13 +121094,13 @@ var ts;
             }
             var keywords = [];
             ts.forEach(aggregateOwnedThrowStatements(owner), function (throwStatement) {
-                keywords.push(ts.findChildOfKind(throwStatement, 105 /* ThrowKeyword */, sourceFile));
+                keywords.push(ts.findChildOfKind(throwStatement, 108 /* ThrowKeyword */, sourceFile));
             });
             // If the "owner" is a function, then we equate 'return' and 'throw' statements in their
             // ability to "jump out" of the function, and include occurrences for both.
             if (ts.isFunctionBlock(owner)) {
                 ts.forEachReturnStatement(owner, function (returnStatement) {
-                    keywords.push(ts.findChildOfKind(returnStatement, 101 /* ReturnKeyword */, sourceFile));
+                    keywords.push(ts.findChildOfKind(returnStatement, 104 /* ReturnKeyword */, sourceFile));
                 });
             }
             return keywords;
@@ -115206,11 +121112,11 @@ var ts;
             }
             var keywords = [];
             ts.forEachReturnStatement(ts.cast(func.body, ts.isBlock), function (returnStatement) {
-                keywords.push(ts.findChildOfKind(returnStatement, 101 /* ReturnKeyword */, sourceFile));
+                keywords.push(ts.findChildOfKind(returnStatement, 104 /* ReturnKeyword */, sourceFile));
             });
             // Include 'throw' statements that do not occur within a try block.
             ts.forEach(aggregateOwnedThrowStatements(func.body), function (throwStatement) {
-                keywords.push(ts.findChildOfKind(throwStatement, 105 /* ThrowKeyword */, sourceFile));
+                keywords.push(ts.findChildOfKind(throwStatement, 108 /* ThrowKeyword */, sourceFile));
             });
             return keywords;
         }
@@ -115222,13 +121128,13 @@ var ts;
             var keywords = [];
             if (func.modifiers) {
                 func.modifiers.forEach(function (modifier) {
-                    pushKeywordIf(keywords, modifier, 126 /* AsyncKeyword */);
+                    pushKeywordIf(keywords, modifier, 129 /* AsyncKeyword */);
                 });
             }
             ts.forEachChild(func, function (child) {
                 traverseWithoutCrossingFunction(child, function (node) {
                     if (ts.isAwaitExpression(node)) {
-                        pushKeywordIf(keywords, node.getFirstToken(), 127 /* AwaitKeyword */);
+                        pushKeywordIf(keywords, node.getFirstToken(), 130 /* AwaitKeyword */);
                     }
                 });
             });
@@ -115243,7 +121149,7 @@ var ts;
             ts.forEachChild(func, function (child) {
                 traverseWithoutCrossingFunction(child, function (node) {
                     if (ts.isYieldExpression(node)) {
-                        pushKeywordIf(keywords, node.getFirstToken(), 121 /* YieldKeyword */);
+                        pushKeywordIf(keywords, node.getFirstToken(), 124 /* YieldKeyword */);
                     }
                 });
             });
@@ -115262,7 +121168,7 @@ var ts;
             // We'd like to highlight else/ifs together if they are only separated by whitespace
             // (i.e. the keywords are separated by no comments, no newlines).
             for (var i = 0; i < keywords.length; i++) {
-                if (keywords[i].kind === 87 /* ElseKeyword */ && i < keywords.length - 1) {
+                if (keywords[i].kind === 90 /* ElseKeyword */ && i < keywords.length - 1) {
                     var elseKeyword = keywords[i];
                     var ifKeyword = keywords[i + 1]; // this *should* always be an 'if' keyword.
                     var shouldCombineElseAndIf = true;
@@ -115297,10 +121203,10 @@ var ts;
             // Now traverse back down through the else branches, aggregating if/else keywords of if-statements.
             while (true) {
                 var children = ifStatement.getChildren(sourceFile);
-                pushKeywordIf(keywords, children[0], 95 /* IfKeyword */);
+                pushKeywordIf(keywords, children[0], 98 /* IfKeyword */);
                 // Generally the 'else' keyword is second-to-last, so we traverse backwards.
                 for (var i = children.length - 1; i >= 0; i--) {
-                    if (pushKeywordIf(keywords, children[i], 87 /* ElseKeyword */)) {
+                    if (pushKeywordIf(keywords, children[i], 90 /* ElseKeyword */)) {
                         break;
                     }
                 }
@@ -115331,7 +121237,7 @@ var ts;
         if (currentDirectory === void 0) { currentDirectory = ""; }
         // Maps from compiler setting target (ES3, ES5, etc.) to all the cached documents we have
         // for those settings.
-        var buckets = ts.createMap();
+        var buckets = new ts.Map();
         var getCanonicalFileName = ts.createGetCanonicalFileName(!!useCaseSensitiveFileNames);
         function reportStats() {
             var bucketInfoArray = ts.arrayFrom(buckets.keys()).filter(function (name) { return name && name.charAt(0) === "_"; }).map(function (name) {
@@ -115368,7 +121274,7 @@ var ts;
             return acquireOrUpdateDocument(fileName, path, compilationSettings, key, scriptSnapshot, version, /*acquiring*/ false, scriptKind);
         }
         function acquireOrUpdateDocument(fileName, path, compilationSettings, key, scriptSnapshot, version, acquiring, scriptKind) {
-            var bucket = ts.getOrUpdate(buckets, key, ts.createMap);
+            var bucket = ts.getOrUpdate(buckets, key, function () { return new ts.Map(); });
             var entry = bucket.get(path);
             var scriptTarget = scriptKind === 6 /* JSON */ ? 100 /* JSON */ : compilationSettings.target || 1 /* ES5 */;
             if (!entry && externalCache) {
@@ -115516,46 +121422,48 @@ var ts;
                         if (cancellationToken)
                             cancellationToken.throwIfCancellationRequested();
                         switch (direct.kind) {
-                            case 196 /* CallExpression */:
+                            case 203 /* CallExpression */:
                                 if (!isAvailableThroughGlobal) {
                                     var parent = direct.parent;
-                                    if (exportKind === 2 /* ExportEquals */ && parent.kind === 242 /* VariableDeclaration */) {
+                                    if (exportKind === 2 /* ExportEquals */ && parent.kind === 249 /* VariableDeclaration */) {
                                         var name = parent.name;
-                                        if (name.kind === 75 /* Identifier */) {
+                                        if (name.kind === 78 /* Identifier */) {
                                             directImports.push(name);
                                             break;
                                         }
                                     }
-                                    // Don't support re-exporting 'require()' calls, so just add a single indirect user.
-                                    addIndirectUser(direct.getSourceFile());
                                 }
                                 break;
-                            case 75 /* Identifier */: // for 'const x = require("y");
+                            case 78 /* Identifier */: // for 'const x = require("y");
                                 break; // TODO: GH#23879
-                            case 253 /* ImportEqualsDeclaration */:
-                                handleNamespaceImport(direct, direct.name, ts.hasModifier(direct, 1 /* Export */), /*alreadyAddedDirect*/ false);
+                            case 260 /* ImportEqualsDeclaration */:
+                                handleNamespaceImport(direct, direct.name, ts.hasSyntacticModifier(direct, 1 /* Export */), /*alreadyAddedDirect*/ false);
                                 break;
-                            case 254 /* ImportDeclaration */:
+                            case 261 /* ImportDeclaration */:
                                 directImports.push(direct);
                                 var namedBindings = direct.importClause && direct.importClause.namedBindings;
-                                if (namedBindings && namedBindings.kind === 256 /* NamespaceImport */) {
+                                if (namedBindings && namedBindings.kind === 263 /* NamespaceImport */) {
                                     handleNamespaceImport(direct, namedBindings.name, /*isReExport*/ false, /*alreadyAddedDirect*/ true);
                                 }
                                 else if (!isAvailableThroughGlobal && ts.isDefaultImport(direct)) {
                                     addIndirectUser(getSourceFileLikeForImportDeclaration(direct)); // Add a check for indirect uses to handle synthetic default imports
                                 }
                                 break;
-                            case 260 /* ExportDeclaration */:
+                            case 267 /* ExportDeclaration */:
                                 if (!direct.exportClause) {
                                     // This is `export * from "foo"`, so imports of this module may import the export too.
                                     handleDirectImports(getContainingModuleSymbol(direct, checker));
                                 }
+                                else if (direct.exportClause.kind === 269 /* NamespaceExport */) {
+                                    // `export * as foo from "foo"` add to indirect uses
+                                    addIndirectUsers(getSourceFileLikeForImportDeclaration(direct));
+                                }
                                 else {
                                     // This is `export { foo } from "foo"` and creates an alias symbol, so recursive search will get handle re-exports.
                                     directImports.push(direct);
                                 }
                                 break;
-                            case 188 /* ImportType */:
+                            case 195 /* ImportType */:
                                 directImports.push(direct);
                                 break;
                             default:
@@ -115572,7 +121480,7 @@ var ts;
                 }
                 else if (!isAvailableThroughGlobal) {
                     var sourceFileLike = getSourceFileLikeForImportDeclaration(importDeclaration);
-                    ts.Debug.assert(sourceFileLike.kind === 290 /* SourceFile */ || sourceFileLike.kind === 249 /* ModuleDeclaration */);
+                    ts.Debug.assert(sourceFileLike.kind === 297 /* SourceFile */ || sourceFileLike.kind === 256 /* ModuleDeclaration */);
                     if (isReExport || findNamespaceReExports(sourceFileLike, name, checker)) {
                         addIndirectUsers(sourceFileLike);
                     }
@@ -115629,17 +121537,17 @@ var ts;
             }
             return { importSearches: importSearches, singleReferences: singleReferences };
             function handleImport(decl) {
-                if (decl.kind === 253 /* ImportEqualsDeclaration */) {
+                if (decl.kind === 260 /* ImportEqualsDeclaration */) {
                     if (isExternalModuleImportEquals(decl)) {
                         handleNamespaceImportLike(decl.name);
                     }
                     return;
                 }
-                if (decl.kind === 75 /* Identifier */) {
+                if (decl.kind === 78 /* Identifier */) {
                     handleNamespaceImportLike(decl);
                     return;
                 }
-                if (decl.kind === 188 /* ImportType */) {
+                if (decl.kind === 195 /* ImportType */) {
                     if (decl.qualifier) {
                         var firstIdentifier = ts.getFirstIdentifier(decl.qualifier);
                         if (firstIdentifier.escapedText === ts.symbolName(exportSymbol)) {
@@ -115655,7 +121563,7 @@ var ts;
                 if (decl.moduleSpecifier.kind !== 10 /* StringLiteral */) {
                     return;
                 }
-                if (decl.kind === 260 /* ExportDeclaration */) {
+                if (decl.kind === 267 /* ExportDeclaration */) {
                     if (decl.exportClause && ts.isNamedExports(decl.exportClause)) {
                         searchForNamedImport(decl.exportClause);
                     }
@@ -115664,10 +121572,10 @@ var ts;
                 var _a = decl.importClause || { name: undefined, namedBindings: undefined }, name = _a.name, namedBindings = _a.namedBindings;
                 if (namedBindings) {
                     switch (namedBindings.kind) {
-                        case 256 /* NamespaceImport */:
+                        case 263 /* NamespaceImport */:
                             handleNamespaceImportLike(namedBindings.name);
                             break;
-                        case 257 /* NamedImports */:
+                        case 264 /* NamedImports */:
                             // 'default' might be accessed as a named import `{ default as foo }`.
                             if (exportKind === 0 /* Named */ || exportKind === 1 /* Default */) {
                                 searchForNamedImport(namedBindings);
@@ -115717,7 +121625,7 @@ var ts;
                         }
                     }
                     else {
-                        var localSymbol = element.kind === 263 /* ExportSpecifier */ && element.propertyName
+                        var localSymbol = element.kind === 270 /* ExportSpecifier */ && element.propertyName
                             ? checker.getExportSpecifierLocalTargetSymbol(element) // For re-exporting under a different name, we want to get the re-exported symbol.
                             : checker.getSymbolAtLocation(name);
                         addSearch(name, localSymbol);
@@ -115746,7 +121654,7 @@ var ts;
             for (var _i = 0, sourceFiles_1 = sourceFiles; _i < sourceFiles_1.length; _i++) {
                 var referencingFile = sourceFiles_1[_i];
                 var searchSourceFile = searchModuleSymbol.valueDeclaration;
-                if (searchSourceFile.kind === 290 /* SourceFile */) {
+                if (searchSourceFile.kind === 297 /* SourceFile */) {
                     for (var _a = 0, _b = referencingFile.referencedFiles; _a < _b.length; _a++) {
                         var ref = _b[_a];
                         if (program.getSourceFileFromReference(referencingFile, ref) === searchSourceFile) {
@@ -115773,7 +121681,7 @@ var ts;
         FindAllReferences.findModuleReferences = findModuleReferences;
         /** Returns a map from a module symbol Id to all import statements that directly reference the module. */
         function getDirectImportsMap(sourceFiles, checker, cancellationToken) {
-            var map = ts.createMap();
+            var map = new ts.Map();
             for (var _i = 0, sourceFiles_2 = sourceFiles; _i < sourceFiles_2.length; _i++) {
                 var sourceFile = sourceFiles_2[_i];
                 if (cancellationToken)
@@ -115794,7 +121702,7 @@ var ts;
         }
         /** Iterates over all statements at the top level or in module declarations. Returns the first truthy result. */
         function forEachPossibleImportOrExportStatement(sourceFileLike, action) {
-            return ts.forEach(sourceFileLike.kind === 290 /* SourceFile */ ? sourceFileLike.statements : sourceFileLike.body.statements, function (statement) {
+            return ts.forEach(sourceFileLike.kind === 297 /* SourceFile */ ? sourceFileLike.statements : sourceFileLike.body.statements, function (statement) {
                 return action(statement) || (isAmbientModuleDeclaration(statement) && ts.forEach(statement.body && statement.body.statements, action));
             });
         }
@@ -115809,15 +121717,15 @@ var ts;
             else {
                 forEachPossibleImportOrExportStatement(sourceFile, function (statement) {
                     switch (statement.kind) {
-                        case 260 /* ExportDeclaration */:
-                        case 254 /* ImportDeclaration */: {
+                        case 267 /* ExportDeclaration */:
+                        case 261 /* ImportDeclaration */: {
                             var decl = statement;
                             if (decl.moduleSpecifier && ts.isStringLiteral(decl.moduleSpecifier)) {
                                 action(decl, decl.moduleSpecifier);
                             }
                             break;
                         }
-                        case 253 /* ImportEqualsDeclaration */: {
+                        case 260 /* ImportEqualsDeclaration */: {
                             var decl = statement;
                             if (isExternalModuleImportEquals(decl)) {
                                 action(decl, decl.moduleReference.expression);
@@ -115841,7 +121749,7 @@ var ts;
                 var parent = node.parent;
                 var grandParent = parent.parent;
                 if (symbol.exportSymbol) {
-                    if (parent.kind === 194 /* PropertyAccessExpression */) {
+                    if (parent.kind === 201 /* PropertyAccessExpression */) {
                         // When accessing an export of a JS module, there's no alias. The symbol will still be flagged as an export even though we're at the use.
                         // So check that we are at the declaration.
                         return symbol.declarations.some(function (d) { return d === parent; }) && ts.isBinaryExpression(grandParent)
@@ -115854,7 +121762,7 @@ var ts;
                 }
                 else {
                     var exportNode = getExportNode(parent, node);
-                    if (exportNode && ts.hasModifier(exportNode, 1 /* Export */)) {
+                    if (exportNode && ts.hasSyntacticModifier(exportNode, 1 /* Export */)) {
                         if (ts.isImportEqualsDeclaration(exportNode) && exportNode.moduleReference === node) {
                             // We're at `Y` in `export import X = Y`. This is not the exported symbol, the left-hand-side is. So treat this as an import statement.
                             if (comingFromExport) {
@@ -115867,6 +121775,9 @@ var ts;
                             return exportInfo(symbol, getExportKindForDeclaration(exportNode));
                         }
                     }
+                    else if (ts.isNamespaceExport(parent)) {
+                        return exportInfo(symbol, 0 /* Named */);
+                    }
                     // If we are in `export = a;` or `export default a;`, `parent` is the export assignment.
                     else if (ts.isExportAssignment(parent)) {
                         return getExportAssignmentExport(parent);
@@ -115936,7 +121847,7 @@ var ts;
             }
             // Not meant for use with export specifiers or export assignment.
             function getExportKindForDeclaration(node) {
-                return ts.hasModifier(node, 512 /* Default */) ? 1 /* Default */ : 0 /* Named */;
+                return ts.hasSyntacticModifier(node, 512 /* Default */) ? 1 /* Default */ : 0 /* Named */;
             }
         }
         FindAllReferences.getImportOrExportSymbol = getImportOrExportSymbol;
@@ -115971,15 +121882,17 @@ var ts;
         function isNodeImport(node) {
             var parent = node.parent;
             switch (parent.kind) {
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return parent.name === node && isExternalModuleImportEquals(parent);
-                case 258 /* ImportSpecifier */:
+                case 265 /* ImportSpecifier */:
                     // For a rename import `{ foo as bar }`, don't search for the imported symbol. Just find local uses of `bar`.
                     return !parent.propertyName;
-                case 255 /* ImportClause */:
-                case 256 /* NamespaceImport */:
+                case 262 /* ImportClause */:
+                case 263 /* NamespaceImport */:
                     ts.Debug.assert(parent.name === node);
                     return true;
+                case 198 /* BindingElement */:
+                    return ts.isInJSFile(node) && ts.isRequireVariableDeclaration(parent, /*requireStringLiteralLikeArgument*/ true);
                 default:
                     return false;
             }
@@ -116002,6 +121915,14 @@ var ts;
                     if (ts.isExportSpecifier(declaration) && !declaration.propertyName && !declaration.parent.parent.moduleSpecifier) {
                         return checker.getExportSpecifierLocalTargetSymbol(declaration);
                     }
+                    else if (ts.isPropertyAccessExpression(declaration) && ts.isModuleExportsAccessExpression(declaration.expression) && !ts.isPrivateIdentifier(declaration.name)) {
+                        return checker.getExportSpecifierLocalTargetSymbol(declaration.name);
+                    }
+                    else if (ts.isShorthandPropertyAssignment(declaration)
+                        && ts.isBinaryExpression(declaration.parent.parent)
+                        && ts.getAssignmentDeclarationKind(declaration.parent.parent) === 2 /* ModuleExports */) {
+                        return checker.getExportSpecifierLocalTargetSymbol(declaration.name);
+                    }
                 }
             }
             return symbol;
@@ -116010,21 +121931,21 @@ var ts;
             return checker.getMergedSymbol(getSourceFileLikeForImportDeclaration(importer).symbol);
         }
         function getSourceFileLikeForImportDeclaration(node) {
-            if (node.kind === 196 /* CallExpression */) {
+            if (node.kind === 203 /* CallExpression */) {
                 return node.getSourceFile();
             }
             var parent = node.parent;
-            if (parent.kind === 290 /* SourceFile */) {
+            if (parent.kind === 297 /* SourceFile */) {
                 return parent;
             }
-            ts.Debug.assert(parent.kind === 250 /* ModuleBlock */);
+            ts.Debug.assert(parent.kind === 257 /* ModuleBlock */);
             return ts.cast(parent.parent, isAmbientModuleDeclaration);
         }
         function isAmbientModuleDeclaration(node) {
-            return node.kind === 249 /* ModuleDeclaration */ && node.name.kind === 10 /* StringLiteral */;
+            return node.kind === 256 /* ModuleDeclaration */ && node.name.kind === 10 /* StringLiteral */;
         }
         function isExternalModuleImportEquals(eq) {
-            return eq.moduleReference.kind === 265 /* ExternalModuleReference */ && eq.moduleReference.expression.kind === 10 /* StringLiteral */;
+            return eq.moduleReference.kind === 272 /* ExternalModuleReference */ && eq.moduleReference.expression.kind === 10 /* StringLiteral */;
         }
     })(FindAllReferences = ts.FindAllReferences || (ts.FindAllReferences = {}));
 })(ts || (ts = {}));
@@ -116117,7 +122038,7 @@ var ts;
                 ((ts.isImportOrExportSpecifier(node.parent) || ts.isBindingElement(node.parent))
                     && node.parent.propertyName === node) ||
                 // Is default export
-                (node.kind === 84 /* DefaultKeyword */ && ts.hasModifier(node.parent, 513 /* ExportDefault */))) {
+                (node.kind === 87 /* DefaultKeyword */ && ts.hasSyntacticModifier(node.parent, 513 /* ExportDefault */))) {
                 return getContextNode(node.parent);
             }
             return undefined;
@@ -116126,7 +122047,7 @@ var ts;
             if (!node)
                 return undefined;
             switch (node.kind) {
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return !ts.isVariableDeclarationList(node.parent) || node.parent.declarations.length !== 1 ?
                         node :
                         ts.isVariableStatement(node.parent.parent) ?
@@ -116134,27 +122055,28 @@ var ts;
                             ts.isForInOrOfStatement(node.parent.parent) ?
                                 getContextNode(node.parent.parent) :
                                 node.parent;
-                case 191 /* BindingElement */:
+                case 198 /* BindingElement */:
                     return getContextNode(node.parent.parent);
-                case 258 /* ImportSpecifier */:
+                case 265 /* ImportSpecifier */:
                     return node.parent.parent.parent;
-                case 263 /* ExportSpecifier */:
-                case 256 /* NamespaceImport */:
+                case 270 /* ExportSpecifier */:
+                case 263 /* NamespaceImport */:
                     return node.parent.parent;
-                case 255 /* ImportClause */:
+                case 262 /* ImportClause */:
+                case 269 /* NamespaceExport */:
                     return node.parent;
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return ts.isExpressionStatement(node.parent) ?
                         node.parent :
                         node;
-                case 232 /* ForOfStatement */:
-                case 231 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                case 238 /* ForInStatement */:
                     return {
                         start: node.initializer,
                         end: node.expression
                     };
-                case 281 /* PropertyAssignment */:
-                case 282 /* ShorthandPropertyAssignment */:
+                case 288 /* PropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
                     return ts.isArrayLiteralOrObjectLiteralDestructuringPattern(node.parent) ?
                         getContextNode(ts.findAncestor(node.parent, function (node) {
                             return ts.isBinaryExpression(node) || ts.isForInOrOfStatement(node);
@@ -116211,15 +122133,15 @@ var ts;
             var node = ts.getTouchingPropertyName(sourceFile, position);
             var referenceEntries;
             var entries = getImplementationReferenceEntries(program, cancellationToken, sourceFiles, node, position);
-            if (node.parent.kind === 194 /* PropertyAccessExpression */
-                || node.parent.kind === 191 /* BindingElement */
-                || node.parent.kind === 195 /* ElementAccessExpression */
-                || node.kind === 102 /* SuperKeyword */) {
+            if (node.parent.kind === 201 /* PropertyAccessExpression */
+                || node.parent.kind === 198 /* BindingElement */
+                || node.parent.kind === 202 /* ElementAccessExpression */
+                || node.kind === 105 /* SuperKeyword */) {
                 referenceEntries = entries && __spreadArrays(entries);
             }
             else {
                 var queue = entries && __spreadArrays(entries);
-                var seenNodes = ts.createMap();
+                var seenNodes = new ts.Map();
                 while (queue && queue.length) {
                     var entry = queue.shift();
                     if (!ts.addToSeen(seenNodes, ts.getNodeId(entry.node))) {
@@ -116237,18 +122159,18 @@ var ts;
         }
         FindAllReferences.getImplementationsAtPosition = getImplementationsAtPosition;
         function getImplementationReferenceEntries(program, cancellationToken, sourceFiles, node, position) {
-            if (node.kind === 290 /* SourceFile */) {
+            if (node.kind === 297 /* SourceFile */) {
                 return undefined;
             }
             var checker = program.getTypeChecker();
             // If invoked directly on a shorthand property assignment, then return
             // the declaration of the symbol being assigned (not the symbol being assigned to).
-            if (node.parent.kind === 282 /* ShorthandPropertyAssignment */) {
+            if (node.parent.kind === 289 /* ShorthandPropertyAssignment */) {
                 var result_1 = [];
                 Core.getReferenceEntriesForShorthandPropertyAssignment(node, checker, function (node) { return result_1.push(nodeEntry(node)); });
                 return result_1;
             }
-            else if (node.kind === 102 /* SuperKeyword */ || ts.isSuperProperty(node.parent)) {
+            else if (node.kind === 105 /* SuperKeyword */ || ts.isSuperProperty(node.parent)) {
                 // References to and accesses on the super keyword only have one possible implementation, so no
                 // need to "Find all References"
                 var symbol = checker.getSymbolAtLocation(node);
@@ -116265,7 +122187,7 @@ var ts;
         FindAllReferences.findReferenceOrRenameEntries = findReferenceOrRenameEntries;
         function getReferenceEntriesForNode(position, node, program, sourceFiles, cancellationToken, options, sourceFilesSet) {
             if (options === void 0) { options = {}; }
-            if (sourceFilesSet === void 0) { sourceFilesSet = ts.arrayToSet(sourceFiles, function (f) { return f.fileName; }); }
+            if (sourceFilesSet === void 0) { sourceFilesSet = new ts.Set(sourceFiles.map(function (f) { return f.fileName; })); }
             return flattenEntries(Core.getReferencedSymbolsForNode(position, node, program, sourceFiles, cancellationToken, options, sourceFilesSet));
         }
         FindAllReferences.getReferenceEntriesForNode = getReferenceEntriesForNode;
@@ -116279,7 +122201,7 @@ var ts;
                         var symbol = def.symbol;
                         var _a = getDefinitionKindAndDisplayParts(symbol, checker, originalNode), displayParts_1 = _a.displayParts, kind_1 = _a.kind;
                         var name_1 = displayParts_1.map(function (p) { return p.text; }).join("");
-                        var declaration = symbol.declarations ? ts.first(symbol.declarations) : undefined;
+                        var declaration = symbol.declarations && ts.firstOrUndefined(symbol.declarations);
                         return {
                             node: declaration ?
                                 ts.getNameOfDeclaration(declaration) || declaration :
@@ -116353,23 +122275,39 @@ var ts;
         function getPrefixAndSuffixText(entry, originalNode, checker) {
             if (entry.kind !== 0 /* Span */ && ts.isIdentifier(originalNode)) {
                 var node = entry.node, kind = entry.kind;
+                var parent = node.parent;
                 var name = originalNode.text;
-                var isShorthandAssignment = ts.isShorthandPropertyAssignment(node.parent);
-                if (isShorthandAssignment || ts.isObjectBindingElementWithoutPropertyName(node.parent) && node.parent.name === node) {
+                var isShorthandAssignment = ts.isShorthandPropertyAssignment(parent);
+                if (isShorthandAssignment || (ts.isObjectBindingElementWithoutPropertyName(parent) && parent.name === node && parent.dotDotDotToken === undefined)) {
                     var prefixColon = { prefixText: name + ": " };
                     var suffixColon = { suffixText: ": " + name };
-                    return kind === 3 /* SearchedLocalFoundProperty */ ? prefixColon
-                        : kind === 4 /* SearchedPropertyFoundLocal */ ? suffixColon
-                            // In `const o = { x }; o.x`, symbolAtLocation at `x` in `{ x }` is the property symbol.
-                            // For a binding element `const { x } = o;`, symbolAtLocation at `x` is the property symbol.
-                            : isShorthandAssignment ? suffixColon : prefixColon;
+                    if (kind === 3 /* SearchedLocalFoundProperty */) {
+                        return prefixColon;
+                    }
+                    if (kind === 4 /* SearchedPropertyFoundLocal */) {
+                        return suffixColon;
+                    }
+                    // In `const o = { x }; o.x`, symbolAtLocation at `x` in `{ x }` is the property symbol.
+                    // For a binding element `const { x } = o;`, symbolAtLocation at `x` is the property symbol.
+                    if (isShorthandAssignment) {
+                        var grandParent = parent.parent;
+                        if (ts.isObjectLiteralExpression(grandParent) &&
+                            ts.isBinaryExpression(grandParent.parent) &&
+                            ts.isModuleExportsAccessExpression(grandParent.parent.left)) {
+                            return prefixColon;
+                        }
+                        return suffixColon;
+                    }
+                    else {
+                        return prefixColon;
+                    }
                 }
-                else if (ts.isImportSpecifier(entry.node.parent) && !entry.node.parent.propertyName) {
+                else if (ts.isImportSpecifier(parent) && !parent.propertyName) {
                     // If the original symbol was using this alias, just rename the alias.
                     var originalSymbol = ts.isExportSpecifier(originalNode.parent) ? checker.getExportSpecifierLocalTargetSymbol(originalNode.parent) : checker.getSymbolAtLocation(originalNode);
-                    return ts.contains(originalSymbol.declarations, entry.node.parent) ? { prefixText: name + " as " } : ts.emptyOptions;
+                    return ts.contains(originalSymbol.declarations, parent) ? { prefixText: name + " as " } : ts.emptyOptions;
                 }
-                else if (ts.isExportSpecifier(entry.node.parent) && !entry.node.parent.propertyName) {
+                else if (ts.isExportSpecifier(parent) && !parent.propertyName) {
                     // If the symbol for the node is same as declared node symbol use prefix text
                     return originalNode === entry.node || checker.getSymbolAtLocation(originalNode) === checker.getSymbolAtLocation(entry.node) ?
                         { prefixText: name + " as " } :
@@ -116393,13 +122331,13 @@ var ts;
             if (symbol) {
                 return getDefinitionKindAndDisplayParts(symbol, checker, node);
             }
-            else if (node.kind === 193 /* ObjectLiteralExpression */) {
+            else if (node.kind === 200 /* ObjectLiteralExpression */) {
                 return {
                     kind: "interface" /* interfaceElement */,
                     displayParts: [ts.punctuationPart(20 /* OpenParenToken */), ts.textPart("object literal"), ts.punctuationPart(21 /* CloseParenToken */)]
                 };
             }
-            else if (node.kind === 214 /* ClassExpression */) {
+            else if (node.kind === 221 /* ClassExpression */) {
                 return {
                     kind: "local class" /* localClassElement */,
                     displayParts: [ts.punctuationPart(20 /* OpenParenToken */), ts.textPart("anonymous local class"), ts.punctuationPart(21 /* CloseParenToken */)]
@@ -116443,13 +122381,13 @@ var ts;
         /** A node is considered a writeAccess iff it is a name of a declaration or a target of an assignment */
         function isWriteAccessForReference(node) {
             var decl = ts.getDeclarationFromName(node);
-            return !!decl && declarationIsWriteAccess(decl) || node.kind === 84 /* DefaultKeyword */ || ts.isWriteAccess(node);
+            return !!decl && declarationIsWriteAccess(decl) || node.kind === 87 /* DefaultKeyword */ || ts.isWriteAccess(node);
         }
         function isDefinitionForReference(node) {
-            return node.kind === 84 /* DefaultKeyword */
+            return node.kind === 87 /* DefaultKeyword */
                 || !!ts.getDeclarationFromName(node)
                 || ts.isLiteralComputedPropertyDeclarationName(node)
-                || (node.kind === 129 /* ConstructorKeyword */ && ts.isConstructorDeclaration(node.parent));
+                || (node.kind === 132 /* ConstructorKeyword */ && ts.isConstructorDeclaration(node.parent));
         }
         /**
          * True if 'decl' provides a value, as in `function f() {}`;
@@ -116460,47 +122398,47 @@ var ts;
             if (!!(decl.flags & 8388608 /* Ambient */))
                 return true;
             switch (decl.kind) {
-                case 209 /* BinaryExpression */:
-                case 191 /* BindingElement */:
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 84 /* DefaultKeyword */:
-                case 248 /* EnumDeclaration */:
-                case 284 /* EnumMember */:
-                case 263 /* ExportSpecifier */:
-                case 255 /* ImportClause */: // default import
-                case 253 /* ImportEqualsDeclaration */:
-                case 258 /* ImportSpecifier */:
-                case 246 /* InterfaceDeclaration */:
-                case 315 /* JSDocCallbackTag */:
-                case 322 /* JSDocTypedefTag */:
-                case 273 /* JsxAttribute */:
-                case 249 /* ModuleDeclaration */:
-                case 252 /* NamespaceExportDeclaration */:
-                case 256 /* NamespaceImport */:
-                case 262 /* NamespaceExport */:
-                case 156 /* Parameter */:
-                case 282 /* ShorthandPropertyAssignment */:
-                case 247 /* TypeAliasDeclaration */:
-                case 155 /* TypeParameter */:
+                case 216 /* BinaryExpression */:
+                case 198 /* BindingElement */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 87 /* DefaultKeyword */:
+                case 255 /* EnumDeclaration */:
+                case 291 /* EnumMember */:
+                case 270 /* ExportSpecifier */:
+                case 262 /* ImportClause */: // default import
+                case 260 /* ImportEqualsDeclaration */:
+                case 265 /* ImportSpecifier */:
+                case 253 /* InterfaceDeclaration */:
+                case 324 /* JSDocCallbackTag */:
+                case 331 /* JSDocTypedefTag */:
+                case 280 /* JsxAttribute */:
+                case 256 /* ModuleDeclaration */:
+                case 259 /* NamespaceExportDeclaration */:
+                case 263 /* NamespaceImport */:
+                case 269 /* NamespaceExport */:
+                case 160 /* Parameter */:
+                case 289 /* ShorthandPropertyAssignment */:
+                case 254 /* TypeAliasDeclaration */:
+                case 159 /* TypeParameter */:
                     return true;
-                case 281 /* PropertyAssignment */:
+                case 288 /* PropertyAssignment */:
                     // In `({ x: y } = 0);`, `x` is not a write access. (Won't call this function for `y`.)
                     return !ts.isArrayLiteralOrObjectLiteralDestructuringPattern(decl.parent);
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 162 /* Constructor */:
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 166 /* Constructor */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     return !!decl.body;
-                case 242 /* VariableDeclaration */:
-                case 159 /* PropertyDeclaration */:
+                case 249 /* VariableDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     return !!decl.initializer || ts.isCatchClause(decl.parent);
-                case 160 /* MethodSignature */:
-                case 158 /* PropertySignature */:
-                case 323 /* JSDocPropertyTag */:
-                case 317 /* JSDocParameterTag */:
+                case 164 /* MethodSignature */:
+                case 162 /* PropertySignature */:
+                case 333 /* JSDocPropertyTag */:
+                case 326 /* JSDocParameterTag */:
                     return false;
                 default:
                     return ts.Debug.failBadSyntaxKind(decl);
@@ -116512,7 +122450,7 @@ var ts;
             /** Core find-all-references algorithm. Handles special cases before delegating to `getReferencedSymbolsForSymbol`. */
             function getReferencedSymbolsForNode(position, node, program, sourceFiles, cancellationToken, options, sourceFilesSet) {
                 if (options === void 0) { options = {}; }
-                if (sourceFilesSet === void 0) { sourceFilesSet = ts.arrayToSet(sourceFiles, function (f) { return f.fileName; }); }
+                if (sourceFilesSet === void 0) { sourceFilesSet = new ts.Set(sourceFiles.map(function (f) { return f.fileName; })); }
                 if (options.use === 1 /* References */) {
                     node = ts.getAdjustedReferenceLocation(node);
                 }
@@ -116661,10 +122599,10 @@ var ts;
                     for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
                         var decl = _a[_i];
                         switch (decl.kind) {
-                            case 290 /* SourceFile */:
+                            case 297 /* SourceFile */:
                                 // Don't include the source file itself. (This may not be ideal behavior, but awkward to include an entire file as a reference.)
                                 break;
-                            case 249 /* ModuleDeclaration */:
+                            case 256 /* ModuleDeclaration */:
                                 if (sourceFilesSet.has(decl.getSourceFile().fileName)) {
                                     references.push(nodeEntry(decl.name));
                                 }
@@ -116683,7 +122621,7 @@ var ts;
                         if (sourceFilesSet.has(sourceFile.fileName)) {
                             // At `module.exports = ...`, reference node is `module`
                             var node = ts.isBinaryExpression(decl) && ts.isPropertyAccessExpression(decl.left) ? decl.left.expression :
-                                ts.isExportAssignment(decl) ? ts.Debug.checkDefined(ts.findChildOfKind(decl, 89 /* ExportKeyword */, sourceFile)) :
+                                ts.isExportAssignment(decl) ? ts.Debug.checkDefined(ts.findChildOfKind(decl, 92 /* ExportKeyword */, sourceFile)) :
                                     ts.getNameOfDeclaration(decl) || decl;
                             references.push(nodeEntry(node));
                         }
@@ -116693,25 +122631,25 @@ var ts;
             }
             /** As in a `readonly prop: any` or `constructor(readonly prop: any)`, not a `readonly any[]`. */
             function isReadonlyTypeOperator(node) {
-                return node.kind === 138 /* ReadonlyKeyword */
+                return node.kind === 142 /* ReadonlyKeyword */
                     && ts.isTypeOperatorNode(node.parent)
-                    && node.parent.operator === 138 /* ReadonlyKeyword */;
+                    && node.parent.operator === 142 /* ReadonlyKeyword */;
             }
             /** getReferencedSymbols for special node kinds. */
             function getReferencedSymbolsSpecial(node, sourceFiles, cancellationToken) {
                 if (ts.isTypeKeyword(node.kind)) {
                     // A void expression (i.e., `void foo()`) is not special, but the `void` type is.
-                    if (node.kind === 110 /* VoidKeyword */ && ts.isVoidExpression(node.parent)) {
+                    if (node.kind === 113 /* VoidKeyword */ && ts.isVoidExpression(node.parent)) {
                         return undefined;
                     }
                     // A modifier readonly (like on a property declaration) is not special;
                     // a readonly type keyword (like `readonly string[]`) is.
-                    if (node.kind === 138 /* ReadonlyKeyword */ && !isReadonlyTypeOperator(node)) {
+                    if (node.kind === 142 /* ReadonlyKeyword */ && !isReadonlyTypeOperator(node)) {
                         return undefined;
                     }
                     // Likewise, when we *are* looking for a special keyword, make sure we
                     // *don’t* include readonly member modifiers.
-                    return getAllReferencesForKeyword(sourceFiles, node.kind, cancellationToken, node.kind === 138 /* ReadonlyKeyword */ ? isReadonlyTypeOperator : undefined);
+                    return getAllReferencesForKeyword(sourceFiles, node.kind, cancellationToken, node.kind === 142 /* ReadonlyKeyword */ ? isReadonlyTypeOperator : undefined);
                 }
                 // Labels
                 if (ts.isJumpStatementTarget(node)) {
@@ -116727,7 +122665,7 @@ var ts;
                 if (ts.isThis(node)) {
                     return getReferencesForThisKeyword(node, sourceFiles, cancellationToken);
                 }
-                if (node.kind === 102 /* SuperKeyword */) {
+                if (node.kind === 105 /* SuperKeyword */) {
                     return getReferencesForSuperKeyword(node);
                 }
                 return undefined;
@@ -116744,7 +122682,7 @@ var ts;
                     // When renaming at an export specifier, rename the export and not the thing being exported.
                     getReferencesAtExportSpecifier(exportSpecifier.name, symbol, exportSpecifier, state.createSearch(node, originalSymbol, /*comingFrom*/ undefined), state, /*addReferencesHere*/ true, /*alwaysGetReferences*/ true);
                 }
-                else if (node && node.kind === 84 /* DefaultKeyword */) {
+                else if (node && node.kind === 87 /* DefaultKeyword */) {
                     addReference(node, symbol, state);
                     searchForImportsOfExport(node, symbol, { exportingModuleSymbol: ts.Debug.checkDefined(symbol.parent, "Expected export symbol to have a parent"), exportKind: 1 /* Default */ }, state);
                 }
@@ -116772,9 +122710,9 @@ var ts;
             }
             function getSpecialSearchKind(node) {
                 switch (node.kind) {
-                    case 129 /* ConstructorKeyword */:
+                    case 132 /* ConstructorKeyword */:
                         return 1 /* Constructor */;
-                    case 75 /* Identifier */:
+                    case 78 /* Identifier */:
                         if (ts.isClassLike(node.parent)) {
                             ts.Debug.assert(node.parent.name === node);
                             return 2 /* Class */;
@@ -116831,7 +122769,7 @@ var ts;
                     this.options = options;
                     this.result = result;
                     /** Cache for `explicitlyinheritsFrom`. */
-                    this.inheritsFromCache = ts.createMap();
+                    this.inheritsFromCache = new ts.Map();
                     /**
                      * Type nodes can contain multiple references to the same type. For example:
                      *      let x: Foo & (Foo & Bar) = ...
@@ -116900,11 +122838,11 @@ var ts;
                 /** Returns `true` the first time we search for a symbol in a file and `false` afterwards. */
                 State.prototype.markSearchedSymbols = function (sourceFile, symbols) {
                     var sourceId = ts.getNodeId(sourceFile);
-                    var seenSymbols = this.sourceFileToSeenSymbols[sourceId] || (this.sourceFileToSeenSymbols[sourceId] = ts.createMap());
+                    var seenSymbols = this.sourceFileToSeenSymbols[sourceId] || (this.sourceFileToSeenSymbols[sourceId] = new ts.Set());
                     var anyNewSymbols = false;
                     for (var _i = 0, symbols_3 = symbols; _i < symbols_3.length; _i++) {
                         var sym = symbols_3[_i];
-                        anyNewSymbols = ts.addToSeen(seenSymbols, ts.getSymbolId(sym)) || anyNewSymbols;
+                        anyNewSymbols = ts.tryAddToSet(seenSymbols, ts.getSymbolId(sym)) || anyNewSymbols;
                     }
                     return anyNewSymbols;
                 };
@@ -116949,7 +122887,7 @@ var ts;
                 }
             }
             function eachExportReference(sourceFiles, checker, cancellationToken, exportSymbol, exportingModuleSymbol, exportName, isDefaultExport, cb) {
-                var importTracker = FindAllReferences.createImportTracker(sourceFiles, ts.arrayToSet(sourceFiles, function (f) { return f.fileName; }), checker, cancellationToken);
+                var importTracker = FindAllReferences.createImportTracker(sourceFiles, new ts.Set(sourceFiles.map(function (f) { return f.fileName; })), checker, cancellationToken);
                 var _a = importTracker(exportSymbol, { exportKind: isDefaultExport ? 1 /* Default */ : 0 /* Named */, exportingModuleSymbol: exportingModuleSymbol }, /*isForRename*/ false), importSearches = _a.importSearches, indirectUsers = _a.indirectUsers;
                 for (var _i = 0, importSearches_2 = importSearches; _i < importSearches_2.length; _i++) {
                     var importLocation = importSearches_2[_i][0];
@@ -117012,7 +122950,7 @@ var ts;
                 // If this is the symbol of a named function expression or named class expression,
                 // then named references are limited to its own scope.
                 var declarations = symbol.declarations, flags = symbol.flags, parent = symbol.parent, valueDeclaration = symbol.valueDeclaration;
-                if (valueDeclaration && (valueDeclaration.kind === 201 /* FunctionExpression */ || valueDeclaration.kind === 214 /* ClassExpression */)) {
+                if (valueDeclaration && (valueDeclaration.kind === 208 /* FunctionExpression */ || valueDeclaration.kind === 221 /* ClassExpression */)) {
                     return valueDeclaration;
                 }
                 if (!declarations) {
@@ -117020,9 +122958,9 @@ var ts;
                 }
                 // If this is private property or method, the scope is the containing class
                 if (flags & (4 /* Property */ | 8192 /* Method */)) {
-                    var privateDeclaration = ts.find(declarations, function (d) { return ts.hasModifier(d, 8 /* Private */) || ts.isPrivateIdentifierPropertyDeclaration(d); });
+                    var privateDeclaration = ts.find(declarations, function (d) { return ts.hasEffectiveModifier(d, 8 /* Private */) || ts.isPrivateIdentifierPropertyDeclaration(d); });
                     if (privateDeclaration) {
-                        return ts.getAncestor(privateDeclaration, 245 /* ClassDeclaration */);
+                        return ts.getAncestor(privateDeclaration, 252 /* ClassDeclaration */);
                     }
                     // Else this is a public property and could be accessed from anywhere.
                     return undefined;
@@ -117051,7 +122989,7 @@ var ts;
                         // Different declarations have different containers, bail out
                         return undefined;
                     }
-                    if (!container || container.kind === 290 /* SourceFile */ && !ts.isExternalOrCommonJsModule(container)) {
+                    if (!container || container.kind === 297 /* SourceFile */ && !ts.isExternalOrCommonJsModule(container)) {
                         // This is a global variable and not an external module, any declaration defined
                         // within this scope is visible outside the file
                         return undefined;
@@ -117161,8 +123099,8 @@ var ts;
             function isValidReferencePosition(node, searchSymbolName) {
                 // Compare the length so we filter out strict superstrings of the symbol we are looking for
                 switch (node.kind) {
-                    case 76 /* PrivateIdentifier */:
-                    case 75 /* Identifier */:
+                    case 79 /* PrivateIdentifier */:
+                    case 78 /* Identifier */:
                         return node.text.length === searchSymbolName.length;
                     case 14 /* NoSubstitutionTemplateLiteral */:
                     case 10 /* StringLiteral */: {
@@ -117172,7 +123110,7 @@ var ts;
                     }
                     case 8 /* NumericLiteral */:
                         return ts.isLiteralNameOfPropertyDeclarationOrIndexAccess(node) && node.text.length === searchSymbolName.length;
-                    case 84 /* DefaultKeyword */:
+                    case 87 /* DefaultKeyword */:
                         return "default".length === searchSymbolName.length;
                     default:
                         return false;
@@ -117238,7 +123176,7 @@ var ts;
                     return;
                 }
                 if (ts.isExportSpecifier(parent)) {
-                    ts.Debug.assert(referenceLocation.kind === 75 /* Identifier */);
+                    ts.Debug.assert(referenceLocation.kind === 78 /* Identifier */);
                     getReferencesAtExportSpecifier(referenceLocation, referenceSymbol, parent, search, state, addReferencesHere);
                     return;
                 }
@@ -117294,8 +123232,8 @@ var ts;
                 }
                 // For `export { foo as bar }`, rename `foo`, but not `bar`.
                 if (!isForRenameWithPrefixAndSuffixText(state.options) || alwaysGetReferences) {
-                    var isDefaultExport = referenceLocation.originalKeywordKind === 84 /* DefaultKeyword */
-                        || exportSpecifier.name.originalKeywordKind === 84 /* DefaultKeyword */;
+                    var isDefaultExport = referenceLocation.originalKeywordKind === 87 /* DefaultKeyword */
+                        || exportSpecifier.name.originalKeywordKind === 87 /* DefaultKeyword */;
                     var exportKind = isDefaultExport ? 1 /* Default */ : 0 /* Named */;
                     var exportSymbol = ts.Debug.checkDefined(exportSpecifier.symbol);
                     var exportInfo = FindAllReferences.getExportInfo(exportSymbol, exportKind, state.checker);
@@ -117376,7 +123314,7 @@ var ts;
                 }
                 var pusher = function () { return state.referenceAdder(search.symbol); };
                 if (ts.isClassLike(referenceLocation.parent)) {
-                    ts.Debug.assert(referenceLocation.kind === 84 /* DefaultKeyword */ || referenceLocation.parent.name === referenceLocation);
+                    ts.Debug.assert(referenceLocation.kind === 87 /* DefaultKeyword */ || referenceLocation.parent.name === referenceLocation);
                     // This is the class declaration containing the constructor.
                     findOwnConstructorReferences(search.symbol, sourceFile, pusher());
                 }
@@ -117398,12 +123336,12 @@ var ts;
                 var addRef = state.referenceAdder(search.symbol);
                 for (var _i = 0, _a = classLike.members; _i < _a.length; _i++) {
                     var member = _a[_i];
-                    if (!(ts.isMethodOrAccessor(member) && ts.hasModifier(member, 32 /* Static */))) {
+                    if (!(ts.isMethodOrAccessor(member) && ts.hasSyntacticModifier(member, 32 /* Static */))) {
                         continue;
                     }
                     if (member.body) {
                         member.body.forEachChild(function cb(node) {
-                            if (node.kind === 104 /* ThisKeyword */) {
+                            if (node.kind === 107 /* ThisKeyword */) {
                                 addRef(node);
                             }
                             else if (!ts.isFunctionLike(node) && !ts.isClassLike(node)) {
@@ -117422,18 +123360,18 @@ var ts;
                 if (constructorSymbol && constructorSymbol.declarations) {
                     for (var _i = 0, _a = constructorSymbol.declarations; _i < _a.length; _i++) {
                         var decl = _a[_i];
-                        var ctrKeyword = ts.findChildOfKind(decl, 129 /* ConstructorKeyword */, sourceFile);
-                        ts.Debug.assert(decl.kind === 162 /* Constructor */ && !!ctrKeyword);
+                        var ctrKeyword = ts.findChildOfKind(decl, 132 /* ConstructorKeyword */, sourceFile);
+                        ts.Debug.assert(decl.kind === 166 /* Constructor */ && !!ctrKeyword);
                         addNode(ctrKeyword);
                     }
                 }
                 if (classSymbol.exports) {
                     classSymbol.exports.forEach(function (member) {
                         var decl = member.valueDeclaration;
-                        if (decl && decl.kind === 161 /* MethodDeclaration */) {
+                        if (decl && decl.kind === 165 /* MethodDeclaration */) {
                             var body = decl.body;
                             if (body) {
-                                forEachDescendantOfKind(body, 104 /* ThisKeyword */, function (thisKeyword) {
+                                forEachDescendantOfKind(body, 107 /* ThisKeyword */, function (thisKeyword) {
                                     if (ts.isNewExpressionTarget(thisKeyword)) {
                                         addNode(thisKeyword);
                                     }
@@ -117454,10 +123392,10 @@ var ts;
                 }
                 for (var _i = 0, _a = constructor.declarations; _i < _a.length; _i++) {
                     var decl = _a[_i];
-                    ts.Debug.assert(decl.kind === 162 /* Constructor */);
+                    ts.Debug.assert(decl.kind === 166 /* Constructor */);
                     var body = decl.body;
                     if (body) {
-                        forEachDescendantOfKind(body, 102 /* SuperKeyword */, function (node) {
+                        forEachDescendantOfKind(body, 105 /* SuperKeyword */, function (node) {
                             if (ts.isCallExpressionTarget(node)) {
                                 addNode(node);
                             }
@@ -117481,10 +123419,10 @@ var ts;
                     addReference(refNode);
                     return;
                 }
-                if (refNode.kind !== 75 /* Identifier */) {
+                if (refNode.kind !== 78 /* Identifier */) {
                     return;
                 }
-                if (refNode.parent.kind === 282 /* ShorthandPropertyAssignment */) {
+                if (refNode.parent.kind === 289 /* ShorthandPropertyAssignment */) {
                     // Go ahead and dereference the shorthand assignment by going to its definition
                     getReferenceEntriesForShorthandPropertyAssignment(refNode, state.checker, addReference);
                 }
@@ -117504,7 +123442,7 @@ var ts;
                     }
                     else if (ts.isFunctionLike(typeHavingNode) && typeHavingNode.body) {
                         var body = typeHavingNode.body;
-                        if (body.kind === 223 /* Block */) {
+                        if (body.kind === 230 /* Block */) {
                             ts.forEachReturnStatement(body, function (returnStatement) {
                                 if (returnStatement.expression)
                                     addIfImplementation(returnStatement.expression);
@@ -117532,13 +123470,13 @@ var ts;
              */
             function isImplementationExpression(node) {
                 switch (node.kind) {
-                    case 200 /* ParenthesizedExpression */:
+                    case 207 /* ParenthesizedExpression */:
                         return isImplementationExpression(node.expression);
-                    case 202 /* ArrowFunction */:
-                    case 201 /* FunctionExpression */:
-                    case 193 /* ObjectLiteralExpression */:
-                    case 214 /* ClassExpression */:
-                    case 192 /* ArrayLiteralExpression */:
+                    case 209 /* ArrowFunction */:
+                    case 208 /* FunctionExpression */:
+                    case 200 /* ObjectLiteralExpression */:
+                    case 221 /* ClassExpression */:
+                    case 199 /* ArrayLiteralExpression */:
                         return true;
                     default:
                         return false;
@@ -117591,14 +123529,14 @@ var ts;
                 // Whether 'super' occurs in a static context within a class.
                 var staticFlag = 32 /* Static */;
                 switch (searchSpaceNode.kind) {
-                    case 159 /* PropertyDeclaration */:
-                    case 158 /* PropertySignature */:
-                    case 161 /* MethodDeclaration */:
-                    case 160 /* MethodSignature */:
-                    case 162 /* Constructor */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        staticFlag &= ts.getModifierFlags(searchSpaceNode);
+                    case 163 /* PropertyDeclaration */:
+                    case 162 /* PropertySignature */:
+                    case 165 /* MethodDeclaration */:
+                    case 164 /* MethodSignature */:
+                    case 166 /* Constructor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        staticFlag &= ts.getSyntacticModifierFlags(searchSpaceNode);
                         searchSpaceNode = searchSpaceNode.parent; // re-assign to be the owning class
                         break;
                     default:
@@ -117606,53 +123544,53 @@ var ts;
                 }
                 var sourceFile = searchSpaceNode.getSourceFile();
                 var references = ts.mapDefined(getPossibleSymbolReferenceNodes(sourceFile, "super", searchSpaceNode), function (node) {
-                    if (node.kind !== 102 /* SuperKeyword */) {
+                    if (node.kind !== 105 /* SuperKeyword */) {
                         return;
                     }
                     var container = ts.getSuperContainer(node, /*stopOnFunctions*/ false);
                     // If we have a 'super' container, we must have an enclosing class.
                     // Now make sure the owning class is the same as the search-space
                     // and has the same static qualifier as the original 'super's owner.
-                    return container && (32 /* Static */ & ts.getModifierFlags(container)) === staticFlag && container.parent.symbol === searchSpaceNode.symbol ? nodeEntry(node) : undefined;
+                    return container && (32 /* Static */ & ts.getSyntacticModifierFlags(container)) === staticFlag && container.parent.symbol === searchSpaceNode.symbol ? nodeEntry(node) : undefined;
                 });
                 return [{ definition: { type: 0 /* Symbol */, symbol: searchSpaceNode.symbol }, references: references }];
             }
             function isParameterName(node) {
-                return node.kind === 75 /* Identifier */ && node.parent.kind === 156 /* Parameter */ && node.parent.name === node;
+                return node.kind === 78 /* Identifier */ && node.parent.kind === 160 /* Parameter */ && node.parent.name === node;
             }
             function getReferencesForThisKeyword(thisOrSuperKeyword, sourceFiles, cancellationToken) {
                 var searchSpaceNode = ts.getThisContainer(thisOrSuperKeyword, /* includeArrowFunctions */ false);
                 // Whether 'this' occurs in a static context within a class.
                 var staticFlag = 32 /* Static */;
                 switch (searchSpaceNode.kind) {
-                    case 161 /* MethodDeclaration */:
-                    case 160 /* MethodSignature */:
+                    case 165 /* MethodDeclaration */:
+                    case 164 /* MethodSignature */:
                         if (ts.isObjectLiteralMethod(searchSpaceNode)) {
                             break;
                         }
                     // falls through
-                    case 159 /* PropertyDeclaration */:
-                    case 158 /* PropertySignature */:
-                    case 162 /* Constructor */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        staticFlag &= ts.getModifierFlags(searchSpaceNode);
+                    case 163 /* PropertyDeclaration */:
+                    case 162 /* PropertySignature */:
+                    case 166 /* Constructor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        staticFlag &= ts.getSyntacticModifierFlags(searchSpaceNode);
                         searchSpaceNode = searchSpaceNode.parent; // re-assign to be the owning class
                         break;
-                    case 290 /* SourceFile */:
+                    case 297 /* SourceFile */:
                         if (ts.isExternalModule(searchSpaceNode) || isParameterName(thisOrSuperKeyword)) {
                             return undefined;
                         }
                     // falls through
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
                         break;
                     // Computed properties in classes are not handled here because references to this are illegal,
                     // so there is no point finding references to them.
                     default:
                         return undefined;
                 }
-                var references = ts.flatMap(searchSpaceNode.kind === 290 /* SourceFile */ ? sourceFiles : [searchSpaceNode.getSourceFile()], function (sourceFile) {
+                var references = ts.flatMap(searchSpaceNode.kind === 297 /* SourceFile */ ? sourceFiles : [searchSpaceNode.getSourceFile()], function (sourceFile) {
                     cancellationToken.throwIfCancellationRequested();
                     return getPossibleSymbolReferenceNodes(sourceFile, "this", ts.isSourceFile(searchSpaceNode) ? sourceFile : searchSpaceNode).filter(function (node) {
                         if (!ts.isThis(node)) {
@@ -117660,19 +123598,19 @@ var ts;
                         }
                         var container = ts.getThisContainer(node, /* includeArrowFunctions */ false);
                         switch (searchSpaceNode.kind) {
-                            case 201 /* FunctionExpression */:
-                            case 244 /* FunctionDeclaration */:
+                            case 208 /* FunctionExpression */:
+                            case 251 /* FunctionDeclaration */:
                                 return searchSpaceNode.symbol === container.symbol;
-                            case 161 /* MethodDeclaration */:
-                            case 160 /* MethodSignature */:
+                            case 165 /* MethodDeclaration */:
+                            case 164 /* MethodSignature */:
                                 return ts.isObjectLiteralMethod(searchSpaceNode) && searchSpaceNode.symbol === container.symbol;
-                            case 214 /* ClassExpression */:
-                            case 245 /* ClassDeclaration */:
+                            case 221 /* ClassExpression */:
+                            case 252 /* ClassDeclaration */:
                                 // Make sure the container belongs to the same class
                                 // and has the appropriate static modifier from the original container.
-                                return container.parent && searchSpaceNode.symbol === container.parent.symbol && (ts.getModifierFlags(container) & 32 /* Static */) === staticFlag;
-                            case 290 /* SourceFile */:
-                                return container.kind === 290 /* SourceFile */ && !ts.isExternalModule(container) && !isParameterName(node);
+                                return container.parent && searchSpaceNode.symbol === container.parent.symbol && (ts.getSyntacticModifierFlags(container) & 32 /* Static */) === staticFlag;
+                            case 297 /* SourceFile */:
+                                return container.kind === 297 /* SourceFile */ && !ts.isExternalModule(container) && !isParameterName(node);
                         }
                     });
                 }).map(function (n) { return nodeEntry(n); });
@@ -117698,11 +123636,27 @@ var ts;
             // This is not needed when searching for re-exports.
             function populateSearchSymbolSet(symbol, location, checker, isForRename, providePrefixAndSuffixText, implementations) {
                 var result = [];
-                forEachRelatedSymbol(symbol, location, checker, isForRename, !(isForRename && providePrefixAndSuffixText), function (sym, root, base) { result.push(base || root || sym); }, 
+                forEachRelatedSymbol(symbol, location, checker, isForRename, !(isForRename && providePrefixAndSuffixText), function (sym, root, base) {
+                    // static method/property and instance method/property might have the same name. Only include static or only include instance.
+                    if (base) {
+                        if (isStatic(symbol) !== isStatic(base)) {
+                            base = undefined;
+                        }
+                    }
+                    result.push(base || root || sym);
+                }, 
+                // when try to find implementation, implementations is true, and not allowed to find base class
                 /*allowBaseTypes*/ function () { return !implementations; });
                 return result;
             }
-            function forEachRelatedSymbol(symbol, location, checker, isForRenamePopulateSearchSymbolSet, onlyIncludeBindingElementAtReferenceLocation, cbSymbol, allowBaseTypes) {
+            /**
+             * @param allowBaseTypes return true means it would try to find in base class or interface.
+             */
+            function forEachRelatedSymbol(symbol, location, checker, isForRenamePopulateSearchSymbolSet, onlyIncludeBindingElementAtReferenceLocation, 
+            /**
+             * @param baseSymbol This symbol means one property/mehtod from base class or interface when it is not null or undefined,
+             */
+            cbSymbol, allowBaseTypes) {
                 var containingObjectLiteralElement = ts.getContainingObjectLiteralElement(location);
                 if (containingObjectLiteralElement) {
                     /* Because in short-hand property assignment, location has two meaning : property name and as value of the property
@@ -117755,7 +123709,7 @@ var ts;
                     ts.Debug.assert(paramProps.length === 2 && !!(paramProps[0].flags & 1 /* FunctionScopedVariable */) && !!(paramProps[1].flags & 4 /* Property */)); // is [parameter, property]
                     return fromRoot(symbol.flags & 1 /* FunctionScopedVariable */ ? paramProps[1] : paramProps[0]);
                 }
-                var exportSpecifier = ts.getDeclarationOfKind(symbol, 263 /* ExportSpecifier */);
+                var exportSpecifier = ts.getDeclarationOfKind(symbol, 270 /* ExportSpecifier */);
                 if (!isForRenamePopulateSearchSymbolSet || exportSpecifier && !exportSpecifier.propertyName) {
                     var localSymbol = exportSpecifier && checker.getExportSpecifierLocalTargetSymbol(exportSpecifier);
                     if (localSymbol) {
@@ -117800,19 +123754,35 @@ var ts;
                     });
                 }
                 function getPropertySymbolOfObjectBindingPatternWithoutPropertyName(symbol, checker) {
-                    var bindingElement = ts.getDeclarationOfKind(symbol, 191 /* BindingElement */);
+                    var bindingElement = ts.getDeclarationOfKind(symbol, 198 /* BindingElement */);
                     if (bindingElement && ts.isObjectBindingElementWithoutPropertyName(bindingElement)) {
                         return ts.getPropertySymbolFromBindingElement(checker, bindingElement);
                     }
                 }
             }
+            function isStatic(symbol) {
+                if (!symbol.valueDeclaration) {
+                    return false;
+                }
+                var modifierFlags = ts.getEffectiveModifierFlags(symbol.valueDeclaration);
+                return !!(modifierFlags & 32 /* Static */);
+            }
             function getRelatedSymbol(search, referenceSymbol, referenceLocation, state) {
                 var checker = state.checker;
                 return forEachRelatedSymbol(referenceSymbol, referenceLocation, checker, /*isForRenamePopulateSearchSymbolSet*/ false, 
-                /*onlyIncludeBindingElementAtReferenceLocation*/ state.options.use !== 2 /* Rename */ || !!state.options.providePrefixAndSuffixTextForRename, function (sym, rootSymbol, baseSymbol, kind) { return search.includes(baseSymbol || rootSymbol || sym)
-                    // For a base type, use the symbol for the derived type. For a synthetic (e.g. union) property, use the union symbol.
-                    ? { symbol: rootSymbol && !(ts.getCheckFlags(sym) & 6 /* Synthetic */) ? rootSymbol : sym, kind: kind }
-                    : undefined; }, 
+                /*onlyIncludeBindingElementAtReferenceLocation*/ state.options.use !== 2 /* Rename */ || !!state.options.providePrefixAndSuffixTextForRename, function (sym, rootSymbol, baseSymbol, kind) {
+                    // check whether the symbol used to search itself is just the searched one.
+                    if (baseSymbol) {
+                        // static method/property and instance method/property might have the same name. Only check static or only check instance.
+                        if (isStatic(referenceSymbol) !== isStatic(baseSymbol)) {
+                            baseSymbol = undefined;
+                        }
+                    }
+                    return search.includes(baseSymbol || rootSymbol || sym)
+                        // For a base type, use the symbol for the derived type. For a synthetic (e.g. union) property, use the union symbol.
+                        ? { symbol: rootSymbol && !(ts.getCheckFlags(sym) & 6 /* Synthetic */) ? rootSymbol : sym, kind: kind }
+                        : undefined;
+                }, 
                 /*allowBaseTypes*/ function (rootSymbol) {
                     return !(search.parents && !search.parents.some(function (parent) { return explicitlyInheritsFrom(rootSymbol.parent, parent, state.inheritsFromCache, checker); }));
                 });
@@ -117962,7 +123932,7 @@ var ts;
             return ts.Debug.checkDefined(node.modifiers && ts.find(node.modifiers, isDefaultModifier));
         }
         function isDefaultModifier(node) {
-            return node.kind === 84 /* DefaultKeyword */;
+            return node.kind === 87 /* DefaultKeyword */;
         }
         /** Gets the symbol for a call hierarchy declaration. */
         function getSymbolOfCallHierarchyDeclaration(typeChecker, node) {
@@ -118002,6 +123972,30 @@ var ts;
             }
             return { text: text, pos: declName.getStart(), end: declName.getEnd() };
         }
+        function getCallHierarchItemContainerName(node) {
+            var _a, _b;
+            if (isConstNamedExpression(node)) {
+                if (ts.isModuleBlock(node.parent.parent.parent.parent) && ts.isIdentifier(node.parent.parent.parent.parent.parent.name)) {
+                    return node.parent.parent.parent.parent.parent.name.getText();
+                }
+                return;
+            }
+            switch (node.kind) {
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 165 /* MethodDeclaration */:
+                    if (node.parent.kind === 200 /* ObjectLiteralExpression */) {
+                        return (_a = ts.getAssignedName(node.parent)) === null || _a === void 0 ? void 0 : _a.getText();
+                    }
+                    return (_b = ts.getNameOfDeclaration(node.parent)) === null || _b === void 0 ? void 0 : _b.getText();
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                    if (ts.isModuleBlock(node.parent) && ts.isIdentifier(node.parent.parent.name)) {
+                        return node.parent.parent.name.getText();
+                    }
+            }
+        }
         function findImplementation(typeChecker, node) {
             if (node.body) {
                 return node;
@@ -118090,6 +124084,10 @@ var ts;
                     }
                     return undefined;
                 }
+                // #39453
+                if (ts.isVariableDeclaration(location) && location.initializer && isConstNamedExpression(location.initializer)) {
+                    return location.initializer;
+                }
                 if (!followingSymbol) {
                     var symbol = typeChecker.getSymbolAtLocation(location);
                     if (symbol) {
@@ -118111,10 +124109,12 @@ var ts;
         function createCallHierarchyItem(program, node) {
             var sourceFile = node.getSourceFile();
             var name = getCallHierarchyItemName(program, node);
+            var containerName = getCallHierarchItemContainerName(node);
             var kind = ts.getNodeKind(node);
+            var kindModifiers = ts.getNodeModifiers(node);
             var span = ts.createTextSpanFromBounds(ts.skipTrivia(sourceFile.text, node.getFullStart(), /*stopAfterLineBreak*/ false, /*stopAtComments*/ true), node.getEnd());
             var selectionSpan = ts.createTextSpanFromBounds(name.pos, name.end);
-            return { file: sourceFile.fileName, kind: kind, name: name.text, span: span, selectionSpan: selectionSpan };
+            return { file: sourceFile.fileName, kind: kind, kindModifiers: kindModifiers, name: name.text, containerName: containerName, span: span, selectionSpan: selectionSpan };
         }
         CallHierarchy.createCallHierarchyItem = createCallHierarchyItem;
         function isDefined(x) {
@@ -118136,7 +124136,7 @@ var ts;
             }
         }
         function getCallSiteGroupKey(entry) {
-            return "" + ts.getNodeId(entry.declaration);
+            return ts.getNodeId(entry.declaration);
         }
         function createCallHierarchyIncomingCall(from, fromSpans) {
             return { from: from, fromSpans: fromSpans };
@@ -118195,56 +124195,56 @@ var ts;
                     return;
                 }
                 switch (node.kind) {
-                    case 75 /* Identifier */:
-                    case 253 /* ImportEqualsDeclaration */:
-                    case 254 /* ImportDeclaration */:
-                    case 260 /* ExportDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
+                    case 78 /* Identifier */:
+                    case 260 /* ImportEqualsDeclaration */:
+                    case 261 /* ImportDeclaration */:
+                    case 267 /* ExportDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
                         // do not descend into nodes that cannot contain callable nodes
                         return;
-                    case 199 /* TypeAssertionExpression */:
-                    case 217 /* AsExpression */:
+                    case 206 /* TypeAssertionExpression */:
+                    case 224 /* AsExpression */:
                         // do not descend into the type side of an assertion
                         collect(node.expression);
                         return;
-                    case 242 /* VariableDeclaration */:
-                    case 156 /* Parameter */:
+                    case 249 /* VariableDeclaration */:
+                    case 160 /* Parameter */:
                         // do not descend into the type of a variable or parameter declaration
                         collect(node.name);
                         collect(node.initializer);
                         return;
-                    case 196 /* CallExpression */:
+                    case 203 /* CallExpression */:
                         // do not descend into the type arguments of a call expression
                         recordCallSite(node);
                         collect(node.expression);
                         ts.forEach(node.arguments, collect);
                         return;
-                    case 197 /* NewExpression */:
+                    case 204 /* NewExpression */:
                         // do not descend into the type arguments of a new expression
                         recordCallSite(node);
                         collect(node.expression);
                         ts.forEach(node.arguments, collect);
                         return;
-                    case 198 /* TaggedTemplateExpression */:
+                    case 205 /* TaggedTemplateExpression */:
                         // do not descend into the type arguments of a tagged template expression
                         recordCallSite(node);
                         collect(node.tag);
                         collect(node.template);
                         return;
-                    case 268 /* JsxOpeningElement */:
-                    case 267 /* JsxSelfClosingElement */:
+                    case 275 /* JsxOpeningElement */:
+                    case 274 /* JsxSelfClosingElement */:
                         // do not descend into the type arguments of a JsxOpeningLikeElement
                         recordCallSite(node);
                         collect(node.tagName);
                         collect(node.attributes);
                         return;
-                    case 157 /* Decorator */:
+                    case 161 /* Decorator */:
                         recordCallSite(node);
                         collect(node.expression);
                         return;
-                    case 194 /* PropertyAccessExpression */:
-                    case 195 /* ElementAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
                         recordCallSite(node);
                         ts.forEachChild(node, collect);
                         break;
@@ -118261,7 +124261,7 @@ var ts;
             ts.forEach(node.statements, collect);
         }
         function collectCallSitesOfModuleDeclaration(node, collect) {
-            if (!ts.hasModifier(node, 2 /* Ambient */) && node.body && ts.isModuleBlock(node.body)) {
+            if (!ts.hasSyntacticModifier(node, 2 /* Ambient */) && node.body && ts.isModuleBlock(node.body)) {
                 ts.forEach(node.body.statements, collect);
             }
         }
@@ -118294,22 +124294,22 @@ var ts;
             var callSites = [];
             var collect = createCallSiteCollector(program, callSites);
             switch (node.kind) {
-                case 290 /* SourceFile */:
+                case 297 /* SourceFile */:
                     collectCallSitesOfSourceFile(node, collect);
                     break;
-                case 249 /* ModuleDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     collectCallSitesOfModuleDeclaration(node, collect);
                     break;
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                     collectCallSitesOfFunctionLikeDeclaration(program.getTypeChecker(), node, collect);
                     break;
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
                     collectCallSitesOfClassLikeDeclaration(node, collect);
                     break;
                 default:
@@ -118390,7 +124390,7 @@ var ts;
                         // If there isn't some include for this, add a new one.
                         if (ts.getRegexFromPattern(ts.Debug.checkDefined(matchers.includeFilePattern), useCaseSensitiveFileNames).test(oldFileOrDirPath) &&
                             !ts.getRegexFromPattern(ts.Debug.checkDefined(matchers.includeFilePattern), useCaseSensitiveFileNames).test(newFileOrDirPath)) {
-                            changeTracker.insertNodeAfter(configFile, ts.last(property.initializer.elements), ts.createStringLiteral(relativePath(newFileOrDirPath)));
+                            changeTracker.insertNodeAfter(configFile, ts.last(property.initializer.elements), ts.factory.createStringLiteral(relativePath(newFileOrDirPath)));
                         }
                     }
                     break;
@@ -118443,8 +124443,8 @@ var ts;
     function updateImports(program, changeTracker, oldToNew, newToOld, host, getCanonicalFileName) {
         var allFiles = program.getSourceFiles();
         var _loop_4 = function (sourceFile) {
-            var newFromOld = oldToNew(sourceFile.path);
-            var newImportFromPath = newFromOld !== undefined ? newFromOld : sourceFile.path;
+            var newFromOld = oldToNew(sourceFile.fileName);
+            var newImportFromPath = newFromOld !== null && newFromOld !== void 0 ? newFromOld : sourceFile.fileName;
             var newImportFromDirectory = ts.getDirectoryPath(newImportFromPath);
             var oldFromNew = newToOld(sourceFile.fileName);
             var oldImportFromPath = oldFromNew || sourceFile.fileName;
@@ -118468,7 +124468,7 @@ var ts;
                     : getSourceFileToImport(importedModuleSymbol, importLiteral, sourceFile, program, host, oldToNew);
                 // Need an update if the imported file moved, or the importing file moved and was using a relative path.
                 return toImport !== undefined && (toImport.updated || (importingSourceFileMoved && ts.pathIsRelative(importLiteral.text)))
-                    ? ts.moduleSpecifiers.updateModuleSpecifier(program.getCompilerOptions(), newImportFromPath, toImport.newFileName, ts.createModuleSpecifierResolutionHost(program, host), importLiteral.text)
+                    ? ts.moduleSpecifiers.updateModuleSpecifier(program.getCompilerOptions(), getCanonicalFileName(newImportFromPath), toImport.newFileName, ts.createModuleSpecifierResolutionHost(program, host), importLiteral.text)
                     : undefined;
             });
         };
@@ -118590,15 +124590,13 @@ var ts;
                 var sigInfo = createDefinitionFromSignatureDeclaration(typeChecker, calledDeclaration);
                 // For a function, if this is the original function definition, return just sigInfo.
                 // If this is the original constructor definition, parent is the class.
-                if (typeChecker.getRootSymbols(symbol).some(function (s) { return symbolMatchesSignature(s, calledDeclaration); }) ||
-                    // TODO: GH#25533 Following check shouldn't be necessary if 'require' is an alias
-                    symbol.declarations && symbol.declarations.some(function (d) { return ts.isVariableDeclaration(d) && !!d.initializer && ts.isRequireCall(d.initializer, /*checkArgumentIsStringLiteralLike*/ false); })) {
+                if (typeChecker.getRootSymbols(symbol).some(function (s) { return symbolMatchesSignature(s, calledDeclaration); })) {
                     return [sigInfo];
                 }
                 else {
                     var defs = getDefinitionFromSymbol(typeChecker, symbol, node, calledDeclaration) || ts.emptyArray;
                     // For a 'super()' call, put the signature first, else put the variable first.
-                    return node.kind === 102 /* SuperKeyword */ ? __spreadArrays([sigInfo], defs) : __spreadArrays(defs, [sigInfo]);
+                    return node.kind === 105 /* SuperKeyword */ ? __spreadArrays([sigInfo], defs) : __spreadArrays(defs, [sigInfo]);
                 }
             }
             // Because name in short-hand property assignment has two different meanings: property name and property value,
@@ -118606,7 +124604,7 @@ var ts;
             // go to the declaration of the property name (in this case stay at the same position). However, if go-to-definition
             // is performed at the location of property access, we would like to go to definition of the property in the short-hand
             // assignment. This case and others are handled by the following code.
-            if (node.parent.kind === 282 /* ShorthandPropertyAssignment */) {
+            if (node.parent.kind === 289 /* ShorthandPropertyAssignment */) {
                 var shorthandSymbol_1 = typeChecker.getShorthandAssignmentValueSymbol(symbol.valueDeclaration);
                 return shorthandSymbol_1 ? shorthandSymbol_1.declarations.map(function (decl) { return createDefinitionInfo(decl, typeChecker, shorthandSymbol_1, node); }) : [];
             }
@@ -118651,13 +124649,21 @@ var ts;
             return getDefinitionFromSymbol(typeChecker, symbol, node);
         }
         GoToDefinition.getDefinitionAtPosition = getDefinitionAtPosition;
+        function isShorthandPropertyAssignmentOfModuleExports(symbol) {
+            var shorthandProperty = ts.tryCast(symbol.valueDeclaration, ts.isShorthandPropertyAssignment);
+            var binaryExpression = ts.tryCast(shorthandProperty === null || shorthandProperty === void 0 ? void 0 : shorthandProperty.parent.parent, ts.isAssignmentExpression);
+            return !!binaryExpression && ts.getAssignmentDeclarationKind(binaryExpression) === 2 /* ModuleExports */;
+        }
         /**
          * True if we should not add definitions for both the signature symbol and the definition symbol.
          * True for `const |f = |() => 0`, false for `function |f() {} const |g = f;`.
+         * Also true for any assignment RHS.
          */
         function symbolMatchesSignature(s, calledDeclaration) {
-            return s === calledDeclaration.symbol || s === calledDeclaration.symbol.parent ||
-                !ts.isCallLikeExpression(calledDeclaration.parent) && s === calledDeclaration.parent.symbol;
+            return s === calledDeclaration.symbol
+                || s === calledDeclaration.symbol.parent
+                || ts.isAssignmentExpression(calledDeclaration.parent)
+                || (!ts.isCallLikeExpression(calledDeclaration.parent) && s === calledDeclaration.parent.symbol);
         }
         function getReferenceAtPosition(sourceFile, position, program) {
             var referencePath = findReferenceInPosition(sourceFile.referencedFiles, position);
@@ -118745,19 +124751,24 @@ var ts;
             // get the aliased symbol instead. This allows for goto def on an import e.g.
             //   import {A, B} from "mod";
             // to jump to the implementation directly.
-            if (symbol && symbol.flags & 2097152 /* Alias */ && shouldSkipAlias(node, symbol.declarations[0])) {
-                var aliased = checker.getAliasedSymbol(symbol);
-                if (aliased.declarations) {
-                    return aliased;
+            while (symbol) {
+                if (symbol.flags & 2097152 /* Alias */ && shouldSkipAlias(node, symbol.declarations[0])) {
+                    var aliased = checker.getAliasedSymbol(symbol);
+                    if (!aliased.declarations) {
+                        break;
+                    }
+                    symbol = aliased;
                 }
-            }
-            if (symbol && ts.isInJSFile(node)) {
-                var requireCall = ts.forEach(symbol.declarations, function (d) { return ts.isVariableDeclaration(d) && !!d.initializer && ts.isRequireCall(d.initializer, /*checkArgumentIsStringLiteralLike*/ true) ? d.initializer : undefined; });
-                if (requireCall) {
-                    var moduleSymbol = checker.getSymbolAtLocation(requireCall.arguments[0]);
-                    if (moduleSymbol) {
-                        return checker.resolveExternalModuleSymbol(moduleSymbol);
+                else if (isShorthandPropertyAssignmentOfModuleExports(symbol)) {
+                    // Skip past `module.exports = { Foo }` even though 'Foo' is not a real alias
+                    var shorthandTarget = checker.resolveName(symbol.name, symbol.valueDeclaration, 111551 /* Value */, /*excludeGlobals*/ false);
+                    if (!ts.some(shorthandTarget === null || shorthandTarget === void 0 ? void 0 : shorthandTarget.declarations)) {
+                        break;
                     }
+                    symbol = shorthandTarget;
+                }
+                else {
+                    break;
                 }
             }
             return symbol;
@@ -118768,18 +124779,21 @@ var ts;
         //   (2) when the aliased symbol is originating from an import.
         //
         function shouldSkipAlias(node, declaration) {
-            if (node.kind !== 75 /* Identifier */) {
+            if (node.kind !== 78 /* Identifier */) {
                 return false;
             }
             if (node.parent === declaration) {
                 return true;
             }
             switch (declaration.kind) {
-                case 255 /* ImportClause */:
-                case 253 /* ImportEqualsDeclaration */:
+                case 262 /* ImportClause */:
+                case 260 /* ImportEqualsDeclaration */:
                     return true;
-                case 258 /* ImportSpecifier */:
-                    return declaration.parent.kind === 257 /* NamedImports */;
+                case 265 /* ImportSpecifier */:
+                    return declaration.parent.kind === 264 /* NamedImports */;
+                case 198 /* BindingElement */:
+                case 249 /* VariableDeclaration */:
+                    return ts.isInJSFile(declaration) && ts.isRequireVariableDeclaration(declaration, /*requireStringLiteralLikeArgument*/ true);
                 default:
                     return false;
             }
@@ -118788,12 +124802,13 @@ var ts;
             // There are cases when you extend a function by adding properties to it afterwards,
             // we want to strip those extra properties.
             // For deduping purposes, we also want to exclude any declarationNodes if provided.
-            var filteredDeclarations = ts.filter(symbol.declarations, function (d) { return d !== declarationNode && (!ts.isAssignmentDeclaration(d) || d === symbol.valueDeclaration); }) || undefined;
+            var filteredDeclarations = ts.filter(symbol.declarations, function (d) { return d !== declarationNode && (!ts.isAssignmentDeclaration(d) || d === symbol.valueDeclaration); })
+                || undefined;
             return getConstructSignatureDefinition() || getCallSignatureDefinition() || ts.map(filteredDeclarations, function (declaration) { return createDefinitionInfo(declaration, typeChecker, symbol, node); });
             function getConstructSignatureDefinition() {
                 // Applicable only if we are in a new expression, or we are on a constructor declaration
                 // and in either case the symbol has a construct signature definition, i.e. class
-                if (symbol.flags & 32 /* Class */ && !(symbol.flags & (16 /* Function */ | 3 /* Variable */)) && (ts.isNewExpressionTarget(node) || node.kind === 129 /* ConstructorKeyword */)) {
+                if (symbol.flags & 32 /* Class */ && !(symbol.flags & (16 /* Function */ | 3 /* Variable */)) && (ts.isNewExpressionTarget(node) || node.kind === 132 /* ConstructorKeyword */)) {
                     var cls = ts.find(filteredDeclarations, ts.isClassLike) || ts.Debug.fail("Expected declaration to have at least one class-like declaration");
                     return getSignatureDefinition(cls.members, /*selectConstructors*/ true);
                 }
@@ -118851,13 +124866,10 @@ var ts;
         }
         /** Returns a CallLikeExpression where `node` is the target being invoked. */
         function getAncestorCallLikeExpression(node) {
-            var target = climbPastManyPropertyAccesses(node);
-            var callLike = target.parent;
+            var target = ts.findAncestor(node, function (n) { return !ts.isRightSideOfPropertyAccess(n); });
+            var callLike = target === null || target === void 0 ? void 0 : target.parent;
             return callLike && ts.isCallLikeExpression(callLike) && ts.getInvokedExpression(callLike) === target ? callLike : undefined;
         }
-        function climbPastManyPropertyAccesses(node) {
-            return ts.isRightSideOfPropertyAccess(node) ? climbPastManyPropertyAccesses(node.parent) : node;
-        }
         function tryGetSignatureDeclaration(typeChecker, node) {
             var callLike = getAncestorCallLikeExpression(node);
             var signature = callLike && typeChecker.getResolvedSignature(callLike);
@@ -118866,9 +124878,9 @@ var ts;
         }
         function isConstructorLike(node) {
             switch (node.kind) {
-                case 162 /* Constructor */:
-                case 171 /* ConstructorType */:
-                case 166 /* ConstructSignature */:
+                case 166 /* Constructor */:
+                case 175 /* ConstructorType */:
+                case 170 /* ConstructSignature */:
                     return true;
                 default:
                     return false;
@@ -118975,22 +124987,19 @@ var ts;
                     var comment = _a[_i].comment;
                     if (comment === undefined)
                         continue;
-                    if (documentationComment.length) {
-                        documentationComment.push(ts.lineBreakPart());
-                    }
-                    documentationComment.push(ts.textPart(comment));
+                    ts.pushIfUnique(documentationComment, comment);
                 }
             });
-            return documentationComment;
+            return ts.intersperse(ts.map(documentationComment, ts.textPart), ts.lineBreakPart());
         }
         JsDoc.getJsDocCommentsFromDeclarations = getJsDocCommentsFromDeclarations;
         function getCommentHavingNodes(declaration) {
             switch (declaration.kind) {
-                case 317 /* JSDocParameterTag */:
-                case 323 /* JSDocPropertyTag */:
+                case 326 /* JSDocParameterTag */:
+                case 333 /* JSDocPropertyTag */:
                     return [declaration];
-                case 315 /* JSDocCallbackTag */:
-                case 322 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 331 /* JSDocTypedefTag */:
                     return [declaration, declaration.parent];
                 default:
                     return ts.getJSDocCommentsAndTags(declaration);
@@ -119011,18 +125020,19 @@ var ts;
         function getCommentText(tag) {
             var comment = tag.comment;
             switch (tag.kind) {
-                case 308 /* JSDocImplementsTag */:
+                case 316 /* JSDocImplementsTag */:
                     return withNode(tag.class);
-                case 307 /* JSDocAugmentsTag */:
+                case 315 /* JSDocAugmentsTag */:
                     return withNode(tag.class);
-                case 321 /* JSDocTemplateTag */:
+                case 330 /* JSDocTemplateTag */:
                     return withList(tag.typeParameters);
-                case 320 /* JSDocTypeTag */:
+                case 329 /* JSDocTypeTag */:
                     return withNode(tag.typeExpression);
-                case 322 /* JSDocTypedefTag */:
-                case 315 /* JSDocCallbackTag */:
-                case 323 /* JSDocPropertyTag */:
-                case 317 /* JSDocParameterTag */:
+                case 331 /* JSDocTypedefTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 333 /* JSDocPropertyTag */:
+                case 326 /* JSDocParameterTag */:
+                case 332 /* JSDocSeeTag */:
                     var name = tag.name;
                     return name ? withNode(name) : comment;
                 default:
@@ -119044,7 +125054,7 @@ var ts;
                     name: tagName,
                     kind: "keyword" /* keyword */,
                     kindModifiers: "",
-                    sortText: "0",
+                    sortText: ts.Completions.SortText.LocationPriority,
                 };
             }));
         }
@@ -119056,7 +125066,7 @@ var ts;
                     name: "@" + tagName,
                     kind: "keyword" /* keyword */,
                     kindModifiers: "",
-                    sortText: "0"
+                    sortText: ts.Completions.SortText.LocationPriority
                 };
             }));
         }
@@ -119090,7 +125100,7 @@ var ts;
                     || nameThusFar !== undefined && !ts.startsWith(name, nameThusFar)) {
                     return undefined;
                 }
-                return { name: name, kind: "parameter" /* parameterElement */, kindModifiers: "", sortText: "0" };
+                return { name: name, kind: "parameter" /* parameterElement */, kindModifiers: "", sortText: ts.Completions.SortText.LocationPriority };
             });
         }
         JsDoc.getJSDocParameterNameCompletions = getJSDocParameterNameCompletions;
@@ -119181,7 +125191,7 @@ var ts;
         function parameterDocComments(parameters, isJavaScriptFile, indentationStr, newLine) {
             return parameters.map(function (_a, i) {
                 var name = _a.name, dotDotDotToken = _a.dotDotDotToken;
-                var paramName = name.kind === 75 /* Identifier */ ? name.text : "param" + i;
+                var paramName = name.kind === 78 /* Identifier */ ? name.text : "param" + i;
                 var type = isJavaScriptFile ? (dotDotDotToken ? "{...any} " : "{any} ") : "";
                 return indentationStr + " * @param " + type + paramName + newLine;
             }).join("");
@@ -119191,23 +125201,24 @@ var ts;
         }
         function getCommentOwnerInfoWorker(commentOwner) {
             switch (commentOwner.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 161 /* MethodDeclaration */:
-                case 162 /* Constructor */:
-                case 160 /* MethodSignature */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 165 /* MethodDeclaration */:
+                case 166 /* Constructor */:
+                case 164 /* MethodSignature */:
+                case 209 /* ArrowFunction */:
                     var parameters = commentOwner.parameters;
                     return { commentOwner: commentOwner, parameters: parameters };
-                case 281 /* PropertyAssignment */:
+                case 288 /* PropertyAssignment */:
                     return getCommentOwnerInfoWorker(commentOwner.initializer);
-                case 245 /* ClassDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 158 /* PropertySignature */:
-                case 248 /* EnumDeclaration */:
-                case 284 /* EnumMember */:
-                case 247 /* TypeAliasDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 162 /* PropertySignature */:
+                case 255 /* EnumDeclaration */:
+                case 291 /* EnumMember */:
+                case 254 /* TypeAliasDeclaration */:
                     return { commentOwner: commentOwner };
-                case 225 /* VariableStatement */: {
+                case 232 /* VariableStatement */: {
                     var varStatement = commentOwner;
                     var varDeclarations = varStatement.declarationList.declarations;
                     var parameters_1 = varDeclarations.length === 1 && varDeclarations[0].initializer
@@ -119215,16 +125226,16 @@ var ts;
                         : undefined;
                     return { commentOwner: commentOwner, parameters: parameters_1 };
                 }
-                case 290 /* SourceFile */:
+                case 297 /* SourceFile */:
                     return "quit";
-                case 249 /* ModuleDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     // If in walking up the tree, we hit a a nested namespace declaration,
                     // then we must be somewhere within a dotted namespace name; however we don't
                     // want to give back a JSDoc template for the 'b' or 'c' in 'namespace a.b.c { }'.
-                    return commentOwner.parent.kind === 249 /* ModuleDeclaration */ ? undefined : { commentOwner: commentOwner };
-                case 226 /* ExpressionStatement */:
+                    return commentOwner.parent.kind === 256 /* ModuleDeclaration */ ? undefined : { commentOwner: commentOwner };
+                case 233 /* ExpressionStatement */:
                     return getCommentOwnerInfoWorker(commentOwner.expression);
-                case 209 /* BinaryExpression */: {
+                case 216 /* BinaryExpression */: {
                     var be = commentOwner;
                     if (ts.getAssignmentDeclarationKind(be) === 0 /* None */) {
                         return "quit";
@@ -119232,7 +125243,7 @@ var ts;
                     var parameters_2 = ts.isFunctionLike(be.right) ? be.right.parameters : ts.emptyArray;
                     return { commentOwner: commentOwner, parameters: parameters_2 };
                 }
-                case 159 /* PropertyDeclaration */:
+                case 163 /* PropertyDeclaration */:
                     var init = commentOwner.initializer;
                     if (init && (ts.isFunctionExpression(init) || ts.isArrowFunction(init))) {
                         return { commentOwner: commentOwner, parameters: init.parameters };
@@ -119248,14 +125259,14 @@ var ts;
          * @returns the parameters of a signature found on the RHS if one exists; otherwise 'emptyArray'.
          */
         function getParametersFromRightHandSideOfAssignment(rightHandSide) {
-            while (rightHandSide.kind === 200 /* ParenthesizedExpression */) {
+            while (rightHandSide.kind === 207 /* ParenthesizedExpression */) {
                 rightHandSide = rightHandSide.expression;
             }
             switch (rightHandSide.kind) {
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
                     return rightHandSide.parameters;
-                case 214 /* ClassExpression */: {
+                case 221 /* ClassExpression */: {
                     var ctr = ts.find(rightHandSide.members, ts.isConstructorDeclaration);
                     return ctr ? ctr.parameters : ts.emptyArray;
                 }
@@ -119317,9 +125328,9 @@ var ts;
         }
         function shouldKeepItem(declaration, checker) {
             switch (declaration.kind) {
-                case 255 /* ImportClause */:
-                case 258 /* ImportSpecifier */:
-                case 253 /* ImportEqualsDeclaration */:
+                case 262 /* ImportClause */:
+                case 265 /* ImportSpecifier */:
+                case 260 /* ImportEqualsDeclaration */:
                     var importer = checker.getSymbolAtLocation(declaration.name); // TODO: GH#18217
                     var imported = checker.getAliasedSymbol(importer);
                     return importer.escapedName !== imported.escapedName;
@@ -119329,7 +125340,7 @@ var ts;
         }
         function tryAddSingleDeclarationName(declaration, containers) {
             var name = ts.getNameOfDeclaration(declaration);
-            return !!name && (pushLiteral(name, containers) || name.kind === 154 /* ComputedPropertyName */ && tryAddComputedPropertyName(name.expression, containers));
+            return !!name && (pushLiteral(name, containers) || name.kind === 158 /* ComputedPropertyName */ && tryAddComputedPropertyName(name.expression, containers));
         }
         // Only added the names of computed properties if they're simple dotted expressions, like:
         //
@@ -119346,7 +125357,7 @@ var ts;
             // First, if we started with a computed property name, then add all but the last
             // portion into the container array.
             var name = ts.getNameOfDeclaration(declaration);
-            if (name && name.kind === 154 /* ComputedPropertyName */ && !tryAddComputedPropertyName(name.expression, containers)) {
+            if (name && name.kind === 158 /* ComputedPropertyName */ && !tryAddComputedPropertyName(name.expression, containers)) {
                 return ts.emptyArray;
             }
             // Don't include the last portion.
@@ -119494,7 +125505,7 @@ var ts;
         }
         function addTrackedEs5Class(name) {
             if (!trackedEs5Classes) {
-                trackedEs5Classes = ts.createMap();
+                trackedEs5Classes = new ts.Map();
             }
             trackedEs5Classes.set(name, true);
         }
@@ -119529,6 +125540,7 @@ var ts;
             // Save the old parent
             parentsStack.push(parent);
             trackedEs5ClassesStack.push(trackedEs5Classes);
+            trackedEs5Classes = undefined;
             parent = navNode;
         }
         /** Call after calling `startNode` and adding children to it. */
@@ -119553,7 +125565,7 @@ var ts;
                 return;
             }
             switch (node.kind) {
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     // Get parameter properties, and treat them as being on the *same* level as the constructor, not under it.
                     var ctr = node;
                     addNodeWithRecursiveChild(ctr, ctr.body);
@@ -119565,21 +125577,21 @@ var ts;
                         }
                     }
                     break;
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                case 160 /* MethodSignature */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                case 164 /* MethodSignature */:
                     if (!ts.hasDynamicName(node)) {
                         addNodeWithRecursiveChild(node, node.body);
                     }
                     break;
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
                     if (!ts.hasDynamicName(node)) {
                         addLeafNode(node);
                     }
                     break;
-                case 255 /* ImportClause */:
+                case 262 /* ImportClause */:
                     var importClause = node;
                     // Handle default import case e.g.:
                     //    import d from "mod";
@@ -119591,7 +125603,7 @@ var ts;
                     //    import {a, b as B} from "mod";
                     var namedBindings = importClause.namedBindings;
                     if (namedBindings) {
-                        if (namedBindings.kind === 256 /* NamespaceImport */) {
+                        if (namedBindings.kind === 263 /* NamespaceImport */) {
                             addLeafNode(namedBindings);
                         }
                         else {
@@ -119602,17 +125614,17 @@ var ts;
                         }
                     }
                     break;
-                case 282 /* ShorthandPropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
                     addNodeWithRecursiveChild(node, node.name);
                     break;
-                case 283 /* SpreadAssignment */:
+                case 290 /* SpreadAssignment */:
                     var expression = node.expression;
                     // Use the expression as the name of the SpreadAssignment, otherwise show as <unknown>.
                     ts.isIdentifier(expression) ? addLeafNode(node, expression) : addLeafNode(node);
                     break;
-                case 191 /* BindingElement */:
-                case 281 /* PropertyAssignment */:
-                case 242 /* VariableDeclaration */:
+                case 198 /* BindingElement */:
+                case 288 /* PropertyAssignment */:
+                case 249 /* VariableDeclaration */:
                     var _e = node, name = _e.name, initializer = _e.initializer;
                     if (ts.isBindingPattern(name)) {
                         addChildrenRecursively(name);
@@ -119627,7 +125639,7 @@ var ts;
                         addNodeWithRecursiveChild(node, initializer);
                     }
                     break;
-                case 244 /* FunctionDeclaration */:
+                case 251 /* FunctionDeclaration */:
                     var nameNode = node.name;
                     // If we see a function declaration track as a possible ES5 class
                     if (nameNode && ts.isIdentifier(nameNode)) {
@@ -119635,11 +125647,11 @@ var ts;
                     }
                     addNodeWithRecursiveChild(node, node.body);
                     break;
-                case 202 /* ArrowFunction */:
-                case 201 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 208 /* FunctionExpression */:
                     addNodeWithRecursiveChild(node, node.body);
                     break;
-                case 248 /* EnumDeclaration */:
+                case 255 /* EnumDeclaration */:
                     startNode(node);
                     for (var _f = 0, _g = node.members; _f < _g.length; _f++) {
                         var member = _g[_f];
@@ -119649,9 +125661,9 @@ var ts;
                     }
                     endNode();
                     break;
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 246 /* InterfaceDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 253 /* InterfaceDeclaration */:
                     startNode(node);
                     for (var _h = 0, _j = node.members; _h < _j.length; _h++) {
                         var member = _j[_h];
@@ -119659,20 +125671,33 @@ var ts;
                     }
                     endNode();
                     break;
-                case 249 /* ModuleDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     addNodeWithRecursiveChild(node, getInteriorModule(node).body);
                     break;
-                case 259 /* ExportAssignment */:
-                case 263 /* ExportSpecifier */:
-                case 253 /* ImportEqualsDeclaration */:
-                case 167 /* IndexSignature */:
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 247 /* TypeAliasDeclaration */:
+                case 266 /* ExportAssignment */: {
+                    var expression_1 = node.expression;
+                    var child = ts.isObjectLiteralExpression(expression_1) || ts.isCallExpression(expression_1) ? expression_1 :
+                        ts.isArrowFunction(expression_1) || ts.isFunctionExpression(expression_1) ? expression_1.body : undefined;
+                    if (child) {
+                        startNode(node);
+                        addChildrenRecursively(child);
+                        endNode();
+                    }
+                    else {
+                        addLeafNode(node);
+                    }
+                    break;
+                }
+                case 270 /* ExportSpecifier */:
+                case 260 /* ImportEqualsDeclaration */:
+                case 171 /* IndexSignature */:
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 254 /* TypeAliasDeclaration */:
                     addLeafNode(node);
                     break;
-                case 196 /* CallExpression */:
-                case 209 /* BinaryExpression */: {
+                case 203 /* CallExpression */:
+                case 216 /* BinaryExpression */: {
                     var special = ts.getAssignmentDeclarationKind(node);
                     switch (special) {
                         case 1 /* ExportsProperty */:
@@ -119726,7 +125751,7 @@ var ts;
                             var memberName = defineCall.arguments[1];
                             var _k = startNestedNodes(node, className), depth = _k[0], classNameIdentifier = _k[1];
                             startNode(node, classNameIdentifier);
-                            startNode(node, ts.setTextRange(ts.createIdentifier(memberName.text), memberName));
+                            startNode(node, ts.setTextRange(ts.factory.createIdentifier(memberName.text), memberName));
                             addChildrenRecursively(node.arguments[2]);
                             endNode();
                             endNode();
@@ -119775,7 +125800,7 @@ var ts;
         }
         /** Merge declarations of the same kind. */
         function mergeChildren(children, node) {
-            var nameToItems = ts.createMap();
+            var nameToItems = new ts.Map();
             ts.filterMutate(children, function (child, index) {
                 var declName = child.name || ts.getNameOfDeclaration(child.node);
                 var name = declName && nodeText(declName);
@@ -119847,7 +125872,7 @@ var ts;
                         isPossibleConstructor(b.node) ? b.node :
                             undefined;
                     if (ctorFunction !== undefined) {
-                        var ctorNode = ts.setTextRange(ts.createConstructor(/* decorators */ undefined, /* modifiers */ undefined, [], /* body */ undefined), ctorFunction);
+                        var ctorNode = ts.setTextRange(ts.factory.createConstructorDeclaration(/* decorators */ undefined, /* modifiers */ undefined, [], /* body */ undefined), ctorFunction);
                         var ctor = emptyNavigationBarNode(ctorNode);
                         ctor.indent = a.indent + 1;
                         ctor.children = a.node === ctorFunction ? a.children : b.children;
@@ -119862,9 +125887,9 @@ var ts;
                             }
                         }
                     }
-                    lastANode = a.node = ts.setTextRange(ts.createClassDeclaration(
+                    lastANode = a.node = ts.setTextRange(ts.factory.createClassDeclaration(
                     /* decorators */ undefined, 
-                    /* modifiers */ undefined, a.name || ts.createIdentifier("__class__"), 
+                    /* modifiers */ undefined, a.name || ts.factory.createIdentifier("__class__"), 
                     /* typeParameters */ undefined, 
                     /* heritageClauses */ undefined, []), a.node);
                 }
@@ -119887,9 +125912,9 @@ var ts;
                 else {
                     if (!a.additionalNodes)
                         a.additionalNodes = [];
-                    a.additionalNodes.push(ts.setTextRange(ts.createClassDeclaration(
+                    a.additionalNodes.push(ts.setTextRange(ts.factory.createClassDeclaration(
                     /* decorators */ undefined, 
-                    /* modifiers */ undefined, a.name || ts.createIdentifier("__class__"), 
+                    /* modifiers */ undefined, a.name || ts.factory.createIdentifier("__class__"), 
                     /* typeParameters */ undefined, 
                     /* heritageClauses */ undefined, []), b.node));
                 }
@@ -119914,12 +125939,12 @@ var ts;
                 return false;
             }
             switch (a.kind) {
-                case 159 /* PropertyDeclaration */:
-                case 161 /* MethodDeclaration */:
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
-                    return ts.hasModifier(a, 32 /* Static */) === ts.hasModifier(b, 32 /* Static */);
-                case 249 /* ModuleDeclaration */:
+                case 163 /* PropertyDeclaration */:
+                case 165 /* MethodDeclaration */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
+                    return ts.hasSyntacticModifier(a, 32 /* Static */) === ts.hasSyntacticModifier(b, 32 /* Static */);
+                case 256 /* ModuleDeclaration */:
                     return areSameModule(a, b);
                 default:
                     return true;
@@ -119935,7 +125960,7 @@ var ts;
         // Only merge module nodes that have the same chain. Don't merge 'A.B.C' with 'A'!
         function areSameModule(a, b) {
             // TODO: GH#18217
-            return a.body.kind === b.body.kind && (a.body.kind !== 249 /* ModuleDeclaration */ || areSameModule(a.body, b.body));
+            return a.body.kind === b.body.kind && (a.body.kind !== 256 /* ModuleDeclaration */ || areSameModule(a.body, b.body));
         }
         /** Merge source into target. Source should be thrown away after this is called. */
         function merge(target, source) {
@@ -119965,7 +125990,7 @@ var ts;
          * So `new()` can still come before an `aardvark` method.
          */
         function tryGetName(node) {
-            if (node.kind === 249 /* ModuleDeclaration */) {
+            if (node.kind === 256 /* ModuleDeclaration */) {
                 return getModuleName(node);
             }
             var declName = ts.getNameOfDeclaration(node);
@@ -119974,16 +125999,16 @@ var ts;
                 return propertyName && ts.unescapeLeadingUnderscores(propertyName);
             }
             switch (node.kind) {
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 214 /* ClassExpression */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 221 /* ClassExpression */:
                     return getFunctionOrClassName(node);
                 default:
                     return undefined;
             }
         }
         function getItemName(node, name) {
-            if (node.kind === 249 /* ModuleDeclaration */) {
+            if (node.kind === 256 /* ModuleDeclaration */) {
                 return cleanText(getModuleName(node));
             }
             if (name) {
@@ -119995,32 +126020,32 @@ var ts;
                 }
             }
             switch (node.kind) {
-                case 290 /* SourceFile */:
+                case 297 /* SourceFile */:
                     var sourceFile = node;
                     return ts.isExternalModule(sourceFile)
                         ? "\"" + ts.escapeString(ts.getBaseFileName(ts.removeFileExtension(ts.normalizePath(sourceFile.fileName)))) + "\""
                         : "<global>";
-                case 259 /* ExportAssignment */:
+                case 266 /* ExportAssignment */:
                     return ts.isExportAssignment(node) && node.isExportEquals ? "export=" /* ExportEquals */ : "default" /* Default */;
-                case 202 /* ArrowFunction */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                    if (ts.getModifierFlags(node) & 512 /* Default */) {
+                case 209 /* ArrowFunction */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                    if (ts.getSyntacticModifierFlags(node) & 512 /* Default */) {
                         return "default";
                     }
                     // We may get a string with newlines or other whitespace in the case of an object dereference
                     // (eg: "app\n.onactivated"), so we should remove the whitespace for readabiltiy in the
                     // navigation bar.
                     return getFunctionOrClassName(node);
-                case 162 /* Constructor */:
+                case 166 /* Constructor */:
                     return "constructor";
-                case 166 /* ConstructSignature */:
+                case 170 /* ConstructSignature */:
                     return "new()";
-                case 165 /* CallSignature */:
+                case 169 /* CallSignature */:
                     return "()";
-                case 167 /* IndexSignature */:
+                case 171 /* IndexSignature */:
                     return "[]";
                 default:
                     return "<unknown>";
@@ -120053,19 +126078,19 @@ var ts;
                 }
                 // Some nodes are otherwise important enough to always include in the primary navigation menu.
                 switch (navigationBarNodeKind(item)) {
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 248 /* EnumDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 249 /* ModuleDeclaration */:
-                    case 290 /* SourceFile */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 322 /* JSDocTypedefTag */:
-                    case 315 /* JSDocCallbackTag */:
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 255 /* EnumDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 256 /* ModuleDeclaration */:
+                    case 297 /* SourceFile */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 331 /* JSDocTypedefTag */:
+                    case 324 /* JSDocCallbackTag */:
                         return true;
-                    case 202 /* ArrowFunction */:
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
                         return isTopLevelFunctionDeclaration(item);
                     default:
                         return false;
@@ -120075,10 +126100,10 @@ var ts;
                         return false;
                     }
                     switch (navigationBarNodeKind(item.parent)) {
-                        case 250 /* ModuleBlock */:
-                        case 290 /* SourceFile */:
-                        case 161 /* MethodDeclaration */:
-                        case 162 /* Constructor */:
+                        case 257 /* ModuleBlock */:
+                        case 297 /* SourceFile */:
+                        case 165 /* MethodDeclaration */:
+                        case 166 /* Constructor */:
                             return true;
                         default:
                             return false;
@@ -120136,9 +126161,8 @@ var ts;
                 return ts.getTextOfNode(moduleDeclaration.name);
             }
             // Otherwise, we need to aggregate each identifier to build up the qualified name.
-            var result = [];
-            result.push(ts.getTextOfIdentifierOrLiteral(moduleDeclaration.name));
-            while (moduleDeclaration.body && moduleDeclaration.body.kind === 249 /* ModuleDeclaration */) {
+            var result = [ts.getTextOfIdentifierOrLiteral(moduleDeclaration.name)];
+            while (moduleDeclaration.body && moduleDeclaration.body.kind === 256 /* ModuleDeclaration */) {
                 moduleDeclaration = moduleDeclaration.body;
                 result.push(ts.getTextOfIdentifierOrLiteral(moduleDeclaration.name));
             }
@@ -120152,13 +126176,13 @@ var ts;
             return decl.body && ts.isModuleDeclaration(decl.body) ? getInteriorModule(decl.body) : decl;
         }
         function isComputedProperty(member) {
-            return !member.name || member.name.kind === 154 /* ComputedPropertyName */;
+            return !member.name || member.name.kind === 158 /* ComputedPropertyName */;
         }
         function getNodeSpan(node) {
-            return node.kind === 290 /* SourceFile */ ? ts.createTextSpanFromRange(node) : ts.createTextSpanFromNode(node, curSourceFile);
+            return node.kind === 297 /* SourceFile */ ? ts.createTextSpanFromRange(node) : ts.createTextSpanFromNode(node, curSourceFile);
         }
         function getModifiers(node) {
-            if (node.parent && node.parent.kind === 242 /* VariableDeclaration */) {
+            if (node.parent && node.parent.kind === 249 /* VariableDeclaration */) {
                 node = node.parent;
             }
             return ts.getNodeModifiers(node);
@@ -120181,7 +126205,7 @@ var ts;
                 return nodeText(parent.name);
             }
             // Default exports are named "default"
-            else if (ts.getModifierFlags(node) & 512 /* Default */) {
+            else if (ts.getSyntacticModifierFlags(node) & 512 /* Default */) {
                 return "default";
             }
             else if (ts.isClassLike(node)) {
@@ -120216,9 +126240,9 @@ var ts;
         }
         function isFunctionOrClassExpression(node) {
             switch (node.kind) {
-                case 202 /* ArrowFunction */:
-                case 201 /* FunctionExpression */:
-                case 214 /* ClassExpression */:
+                case 209 /* ArrowFunction */:
+                case 208 /* FunctionExpression */:
+                case 221 /* ClassExpression */:
                     return true;
                 default:
                     return false;
@@ -120249,7 +126273,7 @@ var ts;
          */
         function organizeImports(sourceFile, formatContext, host, program, preferences) {
             var changeTracker = ts.textChanges.ChangeTracker.fromContext({ host: host, formatContext: formatContext, preferences: preferences });
-            var coalesceAndOrganizeImports = function (importGroup) { return coalesceImports(removeUnusedImports(importGroup, sourceFile, program)); };
+            var coalesceAndOrganizeImports = function (importGroup) { return ts.stableSort(coalesceImports(removeUnusedImports(importGroup, sourceFile, program)), function (s1, s2) { return compareImportsOrRequireStatements(s1, s2); }); };
             // All of the old ImportDeclarations in the file, in syntactic order.
             var topLevelImportDecls = sourceFile.statements.filter(ts.isImportDeclaration);
             organizeImportsWorker(topLevelImportDecls, coalesceAndOrganizeImports);
@@ -120333,7 +126357,7 @@ var ts;
                         var newElements = namedBindings.elements.filter(function (e) { return isDeclarationUsed(e.name); });
                         if (newElements.length < namedBindings.elements.length) {
                             namedBindings = newElements.length
-                                ? ts.updateNamedImports(namedBindings, newElements)
+                                ? ts.factory.updateNamedImports(namedBindings, newElements)
                                 : undefined;
                         }
                     }
@@ -120345,7 +126369,7 @@ var ts;
                 else if (hasModuleDeclarationMatchingSpecifier(sourceFile, moduleSpecifier)) {
                     // If we’re in a declaration file, it’s safe to remove the import clause from it
                     if (sourceFile.isDeclarationFile) {
-                        usedImports.push(ts.createImportDeclaration(importDecl.decorators, importDecl.modifiers, 
+                        usedImports.push(ts.factory.createImportDeclaration(importDecl.decorators, importDecl.modifiers, 
                         /*importClause*/ undefined, moduleSpecifier));
                     }
                     // If we’re not in a declaration file, we can’t remove the import clause even though
@@ -120419,7 +126443,7 @@ var ts;
                 else {
                     for (var _e = 0, defaultImports_1 = defaultImports; _e < defaultImports_1.length; _e++) {
                         var defaultImport = defaultImports_1[_e];
-                        newImportSpecifiers.push(ts.createImportSpecifier(ts.createIdentifier("default"), defaultImport.importClause.name)); // TODO: GH#18217
+                        newImportSpecifiers.push(ts.factory.createImportSpecifier(ts.factory.createIdentifier("default"), defaultImport.importClause.name)); // TODO: GH#18217
                     }
                 }
                 newImportSpecifiers.push.apply(newImportSpecifiers, ts.flatMap(namedImports, function (i) { return i.importClause.namedBindings.elements; })); // TODO: GH#18217
@@ -120430,10 +126454,10 @@ var ts;
                 var newNamedImports = sortedImportSpecifiers.length === 0
                     ? newDefaultImport
                         ? undefined
-                        : ts.createNamedImports(ts.emptyArray)
+                        : ts.factory.createNamedImports(ts.emptyArray)
                     : namedImports.length === 0
-                        ? ts.createNamedImports(sortedImportSpecifiers)
-                        : ts.updateNamedImports(namedImports[0].importClause.namedBindings, sortedImportSpecifiers); // TODO: GH#18217
+                        ? ts.factory.createNamedImports(sortedImportSpecifiers)
+                        : ts.factory.updateNamedImports(namedImports[0].importClause.namedBindings, sortedImportSpecifiers); // TODO: GH#18217
                 // Type-only imports are not allowed to mix default, namespace, and named imports in any combination.
                 // We could rewrite a default import as a named import (`import { default as name }`), but we currently
                 // choose not to as a stylistic preference.
@@ -120509,9 +126533,9 @@ var ts;
                 newExportSpecifiers.push.apply(newExportSpecifiers, ts.flatMap(exportGroup_1, function (i) { return i.exportClause && ts.isNamedExports(i.exportClause) ? i.exportClause.elements : ts.emptyArray; }));
                 var sortedExportSpecifiers = sortSpecifiers(newExportSpecifiers);
                 var exportDecl = exportGroup_1[0];
-                coalescedExports.push(ts.updateExportDeclaration(exportDecl, exportDecl.decorators, exportDecl.modifiers, exportDecl.exportClause && (ts.isNamedExports(exportDecl.exportClause) ?
-                    ts.updateNamedExports(exportDecl.exportClause, sortedExportSpecifiers) :
-                    ts.updateNamespaceExport(exportDecl.exportClause, exportDecl.exportClause.name)), exportDecl.moduleSpecifier, exportDecl.isTypeOnly));
+                coalescedExports.push(ts.factory.updateExportDeclaration(exportDecl, exportDecl.decorators, exportDecl.modifiers, exportDecl.isTypeOnly, exportDecl.exportClause && (ts.isNamedExports(exportDecl.exportClause) ?
+                    ts.factory.updateNamedExports(exportDecl.exportClause, sortedExportSpecifiers) :
+                    ts.factory.updateNamespaceExport(exportDecl.exportClause, exportDecl.exportClause.name)), exportDecl.moduleSpecifier));
             }
             return coalescedExports;
             /*
@@ -120546,19 +126570,21 @@ var ts;
         }
         OrganizeImports.coalesceExports = coalesceExports;
         function updateImportDeclarationAndClause(importDeclaration, name, namedBindings) {
-            return ts.updateImportDeclaration(importDeclaration, importDeclaration.decorators, importDeclaration.modifiers, ts.updateImportClause(importDeclaration.importClause, name, namedBindings, importDeclaration.importClause.isTypeOnly), // TODO: GH#18217
+            return ts.factory.updateImportDeclaration(importDeclaration, importDeclaration.decorators, importDeclaration.modifiers, ts.factory.updateImportClause(importDeclaration.importClause, importDeclaration.importClause.isTypeOnly, name, namedBindings), // TODO: GH#18217
             importDeclaration.moduleSpecifier);
         }
         function sortSpecifiers(specifiers) {
-            return ts.stableSort(specifiers, function (s1, s2) {
-                return compareIdentifiers(s1.propertyName || s1.name, s2.propertyName || s2.name) ||
-                    compareIdentifiers(s1.name, s2.name);
-            });
+            return ts.stableSort(specifiers, compareImportOrExportSpecifiers);
+        }
+        function compareImportOrExportSpecifiers(s1, s2) {
+            return compareIdentifiers(s1.propertyName || s1.name, s2.propertyName || s2.name)
+                || compareIdentifiers(s1.name, s2.name);
         }
+        OrganizeImports.compareImportOrExportSpecifiers = compareImportOrExportSpecifiers;
         /* internal */ // Exported for testing
         function compareModuleSpecifiers(m1, m2) {
-            var name1 = getExternalModuleName(m1);
-            var name2 = getExternalModuleName(m2);
+            var name1 = m1 === undefined ? undefined : getExternalModuleName(m1);
+            var name2 = m2 === undefined ? undefined : getExternalModuleName(m2);
             return ts.compareBooleans(name1 === undefined, name2 === undefined) ||
                 ts.compareBooleans(ts.isExternalModuleNameRelative(name1), ts.isExternalModuleNameRelative(name2)) ||
                 ts.compareStringsCaseInsensitive(name1, name2);
@@ -120567,6 +126593,68 @@ var ts;
         function compareIdentifiers(s1, s2) {
             return ts.compareStringsCaseInsensitive(s1.text, s2.text);
         }
+        function getModuleSpecifierExpression(declaration) {
+            var _a;
+            switch (declaration.kind) {
+                case 260 /* ImportEqualsDeclaration */:
+                    return (_a = ts.tryCast(declaration.moduleReference, ts.isExternalModuleReference)) === null || _a === void 0 ? void 0 : _a.expression;
+                case 261 /* ImportDeclaration */:
+                    return declaration.moduleSpecifier;
+                case 232 /* VariableStatement */:
+                    return declaration.declarationList.declarations[0].initializer.arguments[0];
+            }
+        }
+        function importsAreSorted(imports) {
+            return ts.arrayIsSorted(imports, compareImportsOrRequireStatements);
+        }
+        OrganizeImports.importsAreSorted = importsAreSorted;
+        function importSpecifiersAreSorted(imports) {
+            return ts.arrayIsSorted(imports, compareImportOrExportSpecifiers);
+        }
+        OrganizeImports.importSpecifiersAreSorted = importSpecifiersAreSorted;
+        function getImportDeclarationInsertionIndex(sortedImports, newImport) {
+            var index = ts.binarySearch(sortedImports, newImport, ts.identity, compareImportsOrRequireStatements);
+            return index < 0 ? ~index : index;
+        }
+        OrganizeImports.getImportDeclarationInsertionIndex = getImportDeclarationInsertionIndex;
+        function getImportSpecifierInsertionIndex(sortedImports, newImport) {
+            var index = ts.binarySearch(sortedImports, newImport, ts.identity, compareImportOrExportSpecifiers);
+            return index < 0 ? ~index : index;
+        }
+        OrganizeImports.getImportSpecifierInsertionIndex = getImportSpecifierInsertionIndex;
+        function compareImportsOrRequireStatements(s1, s2) {
+            return compareModuleSpecifiers(getModuleSpecifierExpression(s1), getModuleSpecifierExpression(s2)) || compareImportKind(s1, s2);
+        }
+        OrganizeImports.compareImportsOrRequireStatements = compareImportsOrRequireStatements;
+        function compareImportKind(s1, s2) {
+            return ts.compareValues(getImportKindOrder(s1), getImportKindOrder(s2));
+        }
+        // 1. Side-effect imports
+        // 2. Type-only imports
+        // 3. Namespace imports
+        // 4. Default imports
+        // 5. Named imports
+        // 6. ImportEqualsDeclarations
+        // 7. Require variable statements
+        function getImportKindOrder(s1) {
+            var _a;
+            switch (s1.kind) {
+                case 261 /* ImportDeclaration */:
+                    if (!s1.importClause)
+                        return 0;
+                    if (s1.importClause.isTypeOnly)
+                        return 1;
+                    if (((_a = s1.importClause.namedBindings) === null || _a === void 0 ? void 0 : _a.kind) === 263 /* NamespaceImport */)
+                        return 2;
+                    if (s1.importClause.name)
+                        return 3;
+                    return 4;
+                case 260 /* ImportEqualsDeclaration */:
+                    return 5;
+                case 232 /* VariableStatement */:
+                    return 6;
+            }
+        }
     })(OrganizeImports = ts.OrganizeImports || (ts.OrganizeImports = {}));
 })(ts || (ts = {}));
 /* @internal */
@@ -120601,7 +126689,7 @@ var ts;
                 }
                 var lastImport = current - 1;
                 if (lastImport !== firstImport) {
-                    out.push(createOutliningSpanFromBounds(ts.findChildOfKind(statements[firstImport], 96 /* ImportKeyword */, sourceFile).getStart(sourceFile), statements[lastImport].getEnd(), "imports" /* Imports */));
+                    out.push(createOutliningSpanFromBounds(ts.findChildOfKind(statements[firstImport], 99 /* ImportKeyword */, sourceFile).getStart(sourceFile), statements[lastImport].getEnd(), "imports" /* Imports */));
                 }
             }
             function visitNonImportNode(n) {
@@ -120609,11 +126697,11 @@ var ts;
                 if (depthRemaining === 0)
                     return;
                 cancellationToken.throwIfCancellationRequested();
-                if (ts.isDeclaration(n) || n.kind === 1 /* EndOfFileToken */) {
+                if (ts.isDeclaration(n) || ts.isVariableStatement(n) || n.kind === 1 /* EndOfFileToken */) {
                     addOutliningForLeadingCommentsForNode(n, sourceFile, cancellationToken, out);
                 }
-                if (isFunctionExpressionAssignedToVariable(n)) {
-                    addOutliningForLeadingCommentsForNode(n.parent.parent.parent, sourceFile, cancellationToken, out);
+                if (ts.isFunctionLike(n) && ts.isBinaryExpression(n.parent) && ts.isPropertyAccessExpression(n.parent.left)) {
+                    addOutliningForLeadingCommentsForNode(n.parent.left, sourceFile, cancellationToken, out);
                 }
                 var span = getOutliningSpanForNode(n, sourceFile);
                 if (span)
@@ -120639,13 +126727,6 @@ var ts;
                 }
                 depthRemaining++;
             }
-            function isFunctionExpressionAssignedToVariable(n) {
-                if (!ts.isFunctionExpression(n) && !ts.isArrowFunction(n)) {
-                    return false;
-                }
-                var ancestor = ts.findAncestor(n, ts.isVariableStatement);
-                return !!ancestor && ts.getSingleInitializerOfVariableStatementOrPropertyDeclaration(ancestor) === n;
-            }
         }
         function addRegionOutliningSpans(sourceFile, out) {
             var regions = [];
@@ -120726,7 +126807,7 @@ var ts;
         }
         function getOutliningSpanForNode(n, sourceFile) {
             switch (n.kind) {
-                case 223 /* Block */:
+                case 230 /* Block */:
                     if (ts.isFunctionLike(n.parent)) {
                         return functionSpan(n.parent, n, sourceFile);
                     }
@@ -120734,23 +126815,23 @@ var ts;
                     // If the latter, we want to collapse the block, but consider its hint span
                     // to be the entire span of the parent.
                     switch (n.parent.kind) {
-                        case 228 /* DoStatement */:
-                        case 231 /* ForInStatement */:
-                        case 232 /* ForOfStatement */:
-                        case 230 /* ForStatement */:
-                        case 227 /* IfStatement */:
-                        case 229 /* WhileStatement */:
-                        case 236 /* WithStatement */:
-                        case 280 /* CatchClause */:
+                        case 235 /* DoStatement */:
+                        case 238 /* ForInStatement */:
+                        case 239 /* ForOfStatement */:
+                        case 237 /* ForStatement */:
+                        case 234 /* IfStatement */:
+                        case 236 /* WhileStatement */:
+                        case 243 /* WithStatement */:
+                        case 287 /* CatchClause */:
                             return spanForNode(n.parent);
-                        case 240 /* TryStatement */:
+                        case 247 /* TryStatement */:
                             // Could be the try-block, or the finally-block.
                             var tryStatement = n.parent;
                             if (tryStatement.tryBlock === n) {
                                 return spanForNode(n.parent);
                             }
                             else if (tryStatement.finallyBlock === n) {
-                                var node = ts.findChildOfKind(tryStatement, 92 /* FinallyKeyword */, sourceFile);
+                                var node = ts.findChildOfKind(tryStatement, 95 /* FinallyKeyword */, sourceFile);
                                 if (node)
                                     return spanForNode(node);
                             }
@@ -120760,32 +126841,59 @@ var ts;
                             // the span of the block, independent of any parent span.
                             return createOutliningSpan(ts.createTextSpanFromNode(n, sourceFile), "code" /* Code */);
                     }
-                case 250 /* ModuleBlock */:
+                case 257 /* ModuleBlock */:
                     return spanForNode(n.parent);
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 246 /* InterfaceDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 251 /* CaseBlock */:
-                case 173 /* TypeLiteral */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 253 /* InterfaceDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 258 /* CaseBlock */:
+                case 177 /* TypeLiteral */:
+                case 196 /* ObjectBindingPattern */:
                     return spanForNode(n);
-                case 277 /* CaseClause */:
-                case 278 /* DefaultClause */:
+                case 179 /* TupleType */:
+                    return spanForNode(n, /*autoCollapse*/ false, /*useFullStart*/ !ts.isTupleTypeNode(n.parent), 22 /* OpenBracketToken */);
+                case 284 /* CaseClause */:
+                case 285 /* DefaultClause */:
                     return spanForNodeArray(n.statements);
-                case 193 /* ObjectLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
                     return spanForObjectOrArrayLiteral(n);
-                case 192 /* ArrayLiteralExpression */:
+                case 199 /* ArrayLiteralExpression */:
                     return spanForObjectOrArrayLiteral(n, 22 /* OpenBracketToken */);
-                case 266 /* JsxElement */:
+                case 273 /* JsxElement */:
                     return spanForJSXElement(n);
-                case 270 /* JsxFragment */:
+                case 277 /* JsxFragment */:
                     return spanForJSXFragment(n);
-                case 267 /* JsxSelfClosingElement */:
-                case 268 /* JsxOpeningElement */:
+                case 274 /* JsxSelfClosingElement */:
+                case 275 /* JsxOpeningElement */:
                     return spanForJSXAttributes(n.attributes);
-                case 211 /* TemplateExpression */:
+                case 218 /* TemplateExpression */:
                 case 14 /* NoSubstitutionTemplateLiteral */:
                     return spanForTemplateLiteral(n);
+                case 197 /* ArrayBindingPattern */:
+                    return spanForNode(n, /*autoCollapse*/ false, /*useFullStart*/ !ts.isBindingElement(n.parent), 22 /* OpenBracketToken */);
+                case 209 /* ArrowFunction */:
+                    return spanForArrowFunction(n);
+                case 203 /* CallExpression */:
+                    return spanForCallExpression(n);
+            }
+            function spanForCallExpression(node) {
+                if (!node.arguments.length) {
+                    return undefined;
+                }
+                var openToken = ts.findChildOfKind(node, 20 /* OpenParenToken */, sourceFile);
+                var closeToken = ts.findChildOfKind(node, 21 /* CloseParenToken */, sourceFile);
+                if (!openToken || !closeToken || ts.positionsAreOnSameLine(openToken.pos, closeToken.pos, sourceFile)) {
+                    return undefined;
+                }
+                return spanBetweenTokens(openToken, closeToken, node, sourceFile, /*autoCollapse*/ false, /*useFullStart*/ true);
+            }
+            function spanForArrowFunction(node) {
+                if (ts.isBlock(node.body) || ts.positionsAreOnSameLine(node.body.getFullStart(), node.body.getEnd(), sourceFile)) {
+                    return undefined;
+                }
+                var textSpan = ts.createTextSpanFromBounds(node.body.getFullStart(), node.body.getEnd());
+                return createOutliningSpan(textSpan, "code" /* Code */, ts.createTextSpanFromNode(node));
             }
             function spanForJSXElement(node) {
                 var textSpan = ts.createTextSpanFromBounds(node.openingElement.getStart(sourceFile), node.closingElement.getEnd());
@@ -120831,11 +126939,9 @@ var ts;
             }
         }
         function functionSpan(node, body, sourceFile) {
-            var openToken = ts.isNodeArrayMultiLine(node.parameters, sourceFile)
-                ? ts.findChildOfKind(node, 20 /* OpenParenToken */, sourceFile)
-                : ts.findChildOfKind(body, 18 /* OpenBraceToken */, sourceFile);
+            var openToken = tryGetFunctionOpenToken(node, body, sourceFile);
             var closeToken = ts.findChildOfKind(body, 19 /* CloseBraceToken */, sourceFile);
-            return openToken && closeToken && spanBetweenTokens(openToken, closeToken, node, sourceFile, /*autoCollapse*/ node.kind !== 202 /* ArrowFunction */);
+            return openToken && closeToken && spanBetweenTokens(openToken, closeToken, node, sourceFile, /*autoCollapse*/ node.kind !== 209 /* ArrowFunction */);
         }
         function spanBetweenTokens(openToken, closeToken, hintSpanNode, sourceFile, autoCollapse, useFullStart) {
             if (autoCollapse === void 0) { autoCollapse = false; }
@@ -120849,6 +126955,15 @@ var ts;
             if (bannerText === void 0) { bannerText = "..."; }
             return { textSpan: textSpan, kind: kind, hintSpan: hintSpan, bannerText: bannerText, autoCollapse: autoCollapse };
         }
+        function tryGetFunctionOpenToken(node, body, sourceFile) {
+            if (ts.isNodeArrayMultiLine(node.parameters, sourceFile)) {
+                var openParenToken = ts.findChildOfKind(node, 20 /* OpenParenToken */, sourceFile);
+                if (openParenToken) {
+                    return openParenToken;
+                }
+            }
+            return ts.findChildOfKind(body, 18 /* OpenBraceToken */, sourceFile);
+        }
     })(OutliningElementsCollector = ts.OutliningElementsCollector || (ts.OutliningElementsCollector = {}));
 })(ts || (ts = {}));
 /* @internal */
@@ -120873,7 +126988,7 @@ var ts;
         // we see the name of a module that is used everywhere, or the name of an overload).  As
         // such, we cache the information we compute about the candidate for the life of this
         // pattern matcher so we don't have to compute it multiple times.
-        var stringToWordSpans = ts.createMap();
+        var stringToWordSpans = new ts.Map();
         var dotSeparatedSegments = pattern.trim().split(".").map(function (p) { return createSegment(p.trim()); });
         // A segment is considered invalid if we couldn't find any words in it.
         if (dotSeparatedSegments.some(function (segment) { return !segment.subWordTextChunks.length; }))
@@ -121370,10 +127485,10 @@ var ts;
          */
         function tryConsumeDeclare() {
             var token = ts.scanner.getToken();
-            if (token === 130 /* DeclareKeyword */) {
+            if (token === 133 /* DeclareKeyword */) {
                 // declare module "mod"
                 token = nextToken();
-                if (token === 135 /* ModuleKeyword */) {
+                if (token === 139 /* ModuleKeyword */) {
                     token = nextToken();
                     if (token === 10 /* StringLiteral */) {
                         recordAmbientExternalModule();
@@ -121391,7 +127506,7 @@ var ts;
                 return false;
             }
             var token = ts.scanner.getToken();
-            if (token === 96 /* ImportKeyword */) {
+            if (token === 99 /* ImportKeyword */) {
                 token = nextToken();
                 if (token === 20 /* OpenParenToken */) {
                     token = nextToken();
@@ -121407,21 +127522,21 @@ var ts;
                     return true;
                 }
                 else {
-                    if (token === 145 /* TypeKeyword */) {
+                    if (token === 149 /* TypeKeyword */) {
                         var skipTypeKeyword = ts.scanner.lookAhead(function () {
                             var token = ts.scanner.scan();
-                            return token !== 149 /* FromKeyword */ && (token === 41 /* AsteriskToken */ ||
+                            return token !== 153 /* FromKeyword */ && (token === 41 /* AsteriskToken */ ||
                                 token === 18 /* OpenBraceToken */ ||
-                                token === 75 /* Identifier */ ||
+                                token === 78 /* Identifier */ ||
                                 ts.isKeyword(token));
                         });
                         if (skipTypeKeyword) {
                             token = nextToken();
                         }
                     }
-                    if (token === 75 /* Identifier */ || ts.isKeyword(token)) {
+                    if (token === 78 /* Identifier */ || ts.isKeyword(token)) {
                         token = nextToken();
-                        if (token === 149 /* FromKeyword */) {
+                        if (token === 153 /* FromKeyword */) {
                             token = nextToken();
                             if (token === 10 /* StringLiteral */) {
                                 // import d from "mod";
@@ -121452,7 +127567,7 @@ var ts;
                         }
                         if (token === 19 /* CloseBraceToken */) {
                             token = nextToken();
-                            if (token === 149 /* FromKeyword */) {
+                            if (token === 153 /* FromKeyword */) {
                                 token = nextToken();
                                 if (token === 10 /* StringLiteral */) {
                                     // import {a as A} from "mod";
@@ -121464,11 +127579,11 @@ var ts;
                     }
                     else if (token === 41 /* AsteriskToken */) {
                         token = nextToken();
-                        if (token === 123 /* AsKeyword */) {
+                        if (token === 126 /* AsKeyword */) {
                             token = nextToken();
-                            if (token === 75 /* Identifier */ || ts.isKeyword(token)) {
+                            if (token === 78 /* Identifier */ || ts.isKeyword(token)) {
                                 token = nextToken();
-                                if (token === 149 /* FromKeyword */) {
+                                if (token === 153 /* FromKeyword */) {
                                     token = nextToken();
                                     if (token === 10 /* StringLiteral */) {
                                         // import * as NS from "mod"
@@ -121486,10 +127601,10 @@ var ts;
         }
         function tryConsumeExport() {
             var token = ts.scanner.getToken();
-            if (token === 89 /* ExportKeyword */) {
+            if (token === 92 /* ExportKeyword */) {
                 markAsExternalModuleIfTopLevel();
                 token = nextToken();
-                if (token === 145 /* TypeKeyword */) {
+                if (token === 149 /* TypeKeyword */) {
                     var skipTypeKeyword = ts.scanner.lookAhead(function () {
                         var token = ts.scanner.scan();
                         return token === 41 /* AsteriskToken */ ||
@@ -121508,7 +127623,7 @@ var ts;
                     }
                     if (token === 19 /* CloseBraceToken */) {
                         token = nextToken();
-                        if (token === 149 /* FromKeyword */) {
+                        if (token === 153 /* FromKeyword */) {
                             token = nextToken();
                             if (token === 10 /* StringLiteral */) {
                                 // export {a as A} from "mod";
@@ -121520,7 +127635,7 @@ var ts;
                 }
                 else if (token === 41 /* AsteriskToken */) {
                     token = nextToken();
-                    if (token === 149 /* FromKeyword */) {
+                    if (token === 153 /* FromKeyword */) {
                         token = nextToken();
                         if (token === 10 /* StringLiteral */) {
                             // export * from "mod"
@@ -121528,19 +127643,19 @@ var ts;
                         }
                     }
                 }
-                else if (token === 96 /* ImportKeyword */) {
+                else if (token === 99 /* ImportKeyword */) {
                     token = nextToken();
-                    if (token === 145 /* TypeKeyword */) {
+                    if (token === 149 /* TypeKeyword */) {
                         var skipTypeKeyword = ts.scanner.lookAhead(function () {
                             var token = ts.scanner.scan();
-                            return token === 75 /* Identifier */ ||
+                            return token === 78 /* Identifier */ ||
                                 ts.isKeyword(token);
                         });
                         if (skipTypeKeyword) {
                             token = nextToken();
                         }
                     }
-                    if (token === 75 /* Identifier */ || ts.isKeyword(token)) {
+                    if (token === 78 /* Identifier */ || ts.isKeyword(token)) {
                         token = nextToken();
                         if (token === 62 /* EqualsToken */) {
                             if (tryConsumeRequireCall(/*skipCurrentToken*/ true)) {
@@ -121556,7 +127671,7 @@ var ts;
         function tryConsumeRequireCall(skipCurrentToken, allowTemplateLiterals) {
             if (allowTemplateLiterals === void 0) { allowTemplateLiterals = false; }
             var token = skipCurrentToken ? nextToken() : ts.scanner.getToken();
-            if (token === 139 /* RequireKeyword */) {
+            if (token === 143 /* RequireKeyword */) {
                 token = nextToken();
                 if (token === 20 /* OpenParenToken */) {
                     token = nextToken();
@@ -121572,7 +127687,7 @@ var ts;
         }
         function tryConsumeDefine() {
             var token = ts.scanner.getToken();
-            if (token === 75 /* Identifier */ && ts.scanner.getTokenValue() === "define") {
+            if (token === 78 /* Identifier */ && ts.scanner.getTokenValue() === "define") {
                 token = nextToken();
                 if (token !== 20 /* OpenParenToken */) {
                     return true;
@@ -121699,8 +127814,13 @@ var ts;
         Rename.getRenameInfo = getRenameInfo;
         function getRenameInfoForNode(node, typeChecker, sourceFile, isDefinedInLibraryFile, options) {
             var symbol = typeChecker.getSymbolAtLocation(node);
-            if (!symbol)
-                return;
+            if (!symbol) {
+                if (ts.isLabelName(node)) {
+                    var name = ts.getTextOfNode(node);
+                    return getRenameInfoSuccess(name, name, "label" /* label */, "" /* none */, node, sourceFile);
+                }
+                return undefined;
+            }
             // Only allow a symbol to be renamed if it actually has at least one declaration.
             var declarations = symbol.declarations;
             if (!declarations || declarations.length === 0)
@@ -121710,14 +127830,14 @@ var ts;
                 return getRenameInfoError(ts.Diagnostics.You_cannot_rename_elements_that_are_defined_in_the_standard_TypeScript_library);
             }
             // Cannot rename `default` as in `import { default as foo } from "./someModule";
-            if (ts.isIdentifier(node) && node.originalKeywordKind === 84 /* DefaultKeyword */ && symbol.parent && symbol.parent.flags & 1536 /* Module */) {
+            if (ts.isIdentifier(node) && node.originalKeywordKind === 87 /* DefaultKeyword */ && symbol.parent && symbol.parent.flags & 1536 /* Module */) {
                 return undefined;
             }
             if (ts.isStringLiteralLike(node) && ts.tryGetImportFromModuleSpecifier(node)) {
                 return options && options.allowRenameOfImportPath ? getRenameInfoForModule(node, sourceFile, symbol) : undefined;
             }
             var kind = ts.SymbolDisplay.getSymbolKind(typeChecker, symbol, node);
-            var specifierName = (ts.isImportOrExportSpecifierName(node) || ts.isStringOrNumericLiteralLike(node) && node.parent.kind === 154 /* ComputedPropertyName */)
+            var specifierName = (ts.isImportOrExportSpecifierName(node) || ts.isStringOrNumericLiteralLike(node) && node.parent.kind === 158 /* ComputedPropertyName */)
                 ? ts.stripQuotes(ts.getTextOfIdentifierOrLiteral(node))
                 : undefined;
             var displayName = specifierName || typeChecker.symbolToString(symbol);
@@ -121773,11 +127893,11 @@ var ts;
         }
         function nodeIsEligibleForRename(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
-                case 76 /* PrivateIdentifier */:
+                case 78 /* Identifier */:
+                case 79 /* PrivateIdentifier */:
                 case 10 /* StringLiteral */:
                 case 14 /* NoSubstitutionTemplateLiteral */:
-                case 104 /* ThisKeyword */:
+                case 107 /* ThisKeyword */:
                     return true;
                 case 8 /* NumericLiteral */:
                     return ts.isLiteralNameOfPropertyDeclarationOrIndexAccess(node);
@@ -121793,6 +127913,7 @@ var ts;
     var SmartSelectionRange;
     (function (SmartSelectionRange) {
         function getSmartSelectionRange(pos, sourceFile) {
+            var _a;
             var selectionRange = {
                 textSpan: ts.createTextSpanFromBounds(sourceFile.getFullStart(), sourceFile.getEnd())
             };
@@ -121805,7 +127926,7 @@ var ts;
                     var prevNode = children[i - 1];
                     var node = children[i];
                     var nextNode = children[i + 1];
-                    if (node.getStart(sourceFile) > pos) {
+                    if (ts.getTokenPosOfNode(node, sourceFile, /*includeJsDoc*/ true) > pos) {
                         break outer;
                     }
                     if (positionShouldSnapToNode(sourceFile, pos, node)) {
@@ -121814,14 +127935,14 @@ var ts;
                         //    of things that should be considered independently.
                         // 3. A VariableStatement’s children are just a VaraiableDeclarationList and a semicolon.
                         // 4. A lone VariableDeclaration in a VaraibleDeclaration feels redundant with the VariableStatement.
-                        //
                         // Dive in without pushing a selection range.
                         if (ts.isBlock(node)
                             || ts.isTemplateSpan(node) || ts.isTemplateHead(node) || ts.isTemplateTail(node)
                             || prevNode && ts.isTemplateHead(prevNode)
                             || ts.isVariableDeclarationList(node) && ts.isVariableStatement(parentNode)
                             || ts.isSyntaxList(node) && ts.isVariableDeclarationList(parentNode)
-                            || ts.isVariableDeclaration(node) && ts.isSyntaxList(parentNode) && children.length === 1) {
+                            || ts.isVariableDeclaration(node) && ts.isSyntaxList(parentNode) && children.length === 1
+                            || ts.isJSDocTypeExpression(node) || ts.isJSDocSignature(node) || ts.isJSDocTypeLiteral(node)) {
                             parentNode = node;
                             break;
                         }
@@ -121833,15 +127954,12 @@ var ts;
                         }
                         // Blocks with braces, brackets, parens, or JSX tags on separate lines should be
                         // selected from open to close, including whitespace but not including the braces/etc. themselves.
-                        var isBetweenMultiLineBookends = ts.isSyntaxList(node)
-                            && isListOpener(prevNode)
-                            && isListCloser(nextNode)
+                        var isBetweenMultiLineBookends = ts.isSyntaxList(node) && isListOpener(prevNode) && isListCloser(nextNode)
                             && !ts.positionsAreOnSameLine(prevNode.getStart(), nextNode.getStart(), sourceFile);
-                        var jsDocCommentStart = ts.hasJSDocNodes(node) && node.jsDoc[0].getStart();
                         var start = isBetweenMultiLineBookends ? prevNode.getEnd() : node.getStart();
-                        var end = isBetweenMultiLineBookends ? nextNode.getStart() : node.getEnd();
-                        if (ts.isNumber(jsDocCommentStart)) {
-                            pushSelectionRange(jsDocCommentStart, end);
+                        var end = isBetweenMultiLineBookends ? nextNode.getStart() : getEndPos(sourceFile, node);
+                        if (ts.hasJSDocNodes(node) && ((_a = node.jsDoc) === null || _a === void 0 ? void 0 : _a.length)) {
+                            pushSelectionRange(ts.first(node.jsDoc).getStart(), end);
                         }
                         pushSelectionRange(start, end);
                         // String literals should have a stop both inside and outside their quotes.
@@ -121929,14 +128047,14 @@ var ts;
                 ts.Debug.assertEqual(closeBraceToken.kind, 19 /* CloseBraceToken */);
                 // Group `-/+readonly` and `-/+?`
                 var groupedWithPlusMinusTokens = groupChildren(children, function (child) {
-                    return child === node.readonlyToken || child.kind === 138 /* ReadonlyKeyword */ ||
+                    return child === node.readonlyToken || child.kind === 142 /* ReadonlyKeyword */ ||
                         child === node.questionToken || child.kind === 57 /* QuestionToken */;
                 });
                 // Group type parameter with surrounding brackets
                 var groupedWithBrackets = groupChildren(groupedWithPlusMinusTokens, function (_a) {
                     var kind = _a.kind;
                     return kind === 22 /* OpenBracketToken */ ||
-                        kind === 155 /* TypeParameter */ ||
+                        kind === 159 /* TypeParameter */ ||
                         kind === 23 /* CloseBracketToken */;
                 });
                 return [
@@ -122043,23 +128161,33 @@ var ts;
         }
         function createSyntaxList(children) {
             ts.Debug.assertGreaterThanOrEqual(children.length, 1);
-            var syntaxList = ts.createNode(324 /* SyntaxList */, children[0].pos, ts.last(children).end);
-            syntaxList._children = children;
-            return syntaxList;
+            return ts.setTextRangePosEnd(ts.parseNodeFactory.createSyntaxList(children), children[0].pos, ts.last(children).end);
         }
         function isListOpener(token) {
             var kind = token && token.kind;
             return kind === 18 /* OpenBraceToken */
                 || kind === 22 /* OpenBracketToken */
                 || kind === 20 /* OpenParenToken */
-                || kind === 268 /* JsxOpeningElement */;
+                || kind === 275 /* JsxOpeningElement */;
         }
         function isListCloser(token) {
             var kind = token && token.kind;
             return kind === 19 /* CloseBraceToken */
                 || kind === 23 /* CloseBracketToken */
                 || kind === 21 /* CloseParenToken */
-                || kind === 269 /* JsxClosingElement */;
+                || kind === 276 /* JsxClosingElement */;
+        }
+        function getEndPos(sourceFile, node) {
+            switch (node.kind) {
+                case 326 /* JSDocParameterTag */:
+                case 324 /* JSDocCallbackTag */:
+                case 333 /* JSDocPropertyTag */:
+                case 331 /* JSDocTypedefTag */:
+                case 328 /* JSDocThisTag */:
+                    return sourceFile.getLineEndOfPosition(node.getStart());
+                default:
+                    return node.getEnd();
+            }
         }
     })(SmartSelectionRange = ts.SmartSelectionRange || (ts.SmartSelectionRange = {}));
 })(ts || (ts = {}));
@@ -122163,14 +128291,15 @@ var ts;
                 return undefined;
             // See if we can find some symbol with the call expression name that has call signatures.
             var expression = getExpressionFromInvocation(argumentInfo.invocation);
-            var name = ts.isIdentifier(expression) ? expression.text : ts.isPropertyAccessExpression(expression) ? expression.name.text : undefined;
+            var name = ts.isPropertyAccessExpression(expression) ? expression.name.text : undefined;
             var typeChecker = program.getTypeChecker();
             return name === undefined ? undefined : ts.firstDefined(program.getSourceFiles(), function (sourceFile) {
                 return ts.firstDefined(sourceFile.getNamedDeclarations().get(name), function (declaration) {
                     var type = declaration.symbol && typeChecker.getTypeOfSymbolAtLocation(declaration.symbol, declaration);
                     var callSignatures = type && type.getCallSignatures();
                     if (callSignatures && callSignatures.length) {
-                        return typeChecker.runWithCancellationToken(cancellationToken, function (typeChecker) { return createSignatureHelpItems(callSignatures, callSignatures[0], argumentInfo, sourceFile, typeChecker); });
+                        return typeChecker.runWithCancellationToken(cancellationToken, function (typeChecker) { return createSignatureHelpItems(callSignatures, callSignatures[0], argumentInfo, sourceFile, typeChecker, 
+                        /*useFullPrefix*/ true); });
                     }
                 });
             });
@@ -122264,10 +128393,10 @@ var ts;
                 }
                 return undefined;
             }
-            else if (ts.isTemplateHead(node) && parent.parent.kind === 198 /* TaggedTemplateExpression */) {
+            else if (ts.isTemplateHead(node) && parent.parent.kind === 205 /* TaggedTemplateExpression */) {
                 var templateExpression = parent;
                 var tagExpression = templateExpression.parent;
-                ts.Debug.assert(templateExpression.kind === 211 /* TemplateExpression */);
+                ts.Debug.assert(templateExpression.kind === 218 /* TemplateExpression */);
                 var argumentIndex = ts.isInsideTemplateLiteral(node, position, sourceFile) ? 0 : 1;
                 return getArgumentListInfoForTemplate(tagExpression, argumentIndex, sourceFile);
             }
@@ -122323,10 +128452,12 @@ var ts;
             if (!info)
                 return undefined;
             var contextualType = info.contextualType, argumentIndex = info.argumentIndex, argumentCount = info.argumentCount, argumentsSpan = info.argumentsSpan;
-            var signatures = contextualType.getCallSignatures();
+            // for optional function condition.
+            var nonNullableContextualType = contextualType.getNonNullableType();
+            var signatures = nonNullableContextualType.getCallSignatures();
             if (signatures.length !== 1)
                 return undefined;
-            var invocation = { kind: 2 /* Contextual */, signature: ts.first(signatures), node: startingToken, symbol: chooseBetterSymbol(contextualType.symbol) };
+            var invocation = { kind: 2 /* Contextual */, signature: ts.first(signatures), node: startingToken, symbol: chooseBetterSymbol(nonNullableContextualType.symbol) };
             return { isTypeParameterList: false, invocation: invocation, argumentsSpan: argumentsSpan, argumentIndex: argumentIndex, argumentCount: argumentCount };
         }
         function getContextualSignatureLocationInfo(startingToken, sourceFile, checker) {
@@ -122334,17 +128465,17 @@ var ts;
                 return undefined;
             var parent = startingToken.parent;
             switch (parent.kind) {
-                case 200 /* ParenthesizedExpression */:
-                case 161 /* MethodDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
+                case 207 /* ParenthesizedExpression */:
+                case 165 /* MethodDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
                     var info = getArgumentOrParameterListInfo(startingToken, sourceFile);
                     if (!info)
                         return undefined;
                     var argumentIndex = info.argumentIndex, argumentCount = info.argumentCount, argumentsSpan = info.argumentsSpan;
                     var contextualType = ts.isMethodDeclaration(parent) ? checker.getContextualTypeForObjectLiteralElement(parent) : checker.getContextualType(parent);
                     return contextualType && { contextualType: contextualType, argumentIndex: argumentIndex, argumentCount: argumentCount, argumentsSpan: argumentsSpan };
-                case 209 /* BinaryExpression */: {
+                case 216 /* BinaryExpression */: {
                     var highestBinary = getHighestBinary(parent);
                     var contextualType_1 = checker.getContextualType(highestBinary);
                     var argumentIndex_1 = startingToken.kind === 20 /* OpenParenToken */ ? 0 : countBinaryExpressionParameters(parent) - 1;
@@ -122468,7 +128599,7 @@ var ts;
             //       |       |
             // This is because a Missing node has no width. However, what we actually want is to include trivia
             // leading up to the next token in case the user is about to type in a TemplateMiddle or TemplateTail.
-            if (template.kind === 211 /* TemplateExpression */) {
+            if (template.kind === 218 /* TemplateExpression */) {
                 var lastSpan = ts.last(template.templateSpans);
                 if (lastSpan.literal.getFullWidth() === 0) {
                     applicableSpanEnd = ts.skipTrivia(sourceFile.text, applicableSpanEnd, /*stopAfterLineBreak*/ false);
@@ -122506,18 +128637,40 @@ var ts;
             return invocation.kind === 0 /* Call */ ? invocation.node : invocation.kind === 1 /* TypeArgs */ ? invocation.called : invocation.node;
         }
         var signatureHelpNodeBuilderFlags = 8192 /* OmitParameterModifiers */ | 70221824 /* IgnoreErrors */ | 16384 /* UseAliasDefinedOutsideCurrentScope */;
-        function createSignatureHelpItems(candidates, resolvedSignature, _a, sourceFile, typeChecker) {
+        function createSignatureHelpItems(candidates, resolvedSignature, _a, sourceFile, typeChecker, useFullPrefix) {
+            var _b;
             var isTypeParameterList = _a.isTypeParameterList, argumentCount = _a.argumentCount, applicableSpan = _a.argumentsSpan, invocation = _a.invocation, argumentIndex = _a.argumentIndex;
             var enclosingDeclaration = getEnclosingDeclarationFromInvocation(invocation);
-            var callTargetSymbol = invocation.kind === 2 /* Contextual */ ? invocation.symbol : typeChecker.getSymbolAtLocation(getExpressionFromInvocation(invocation));
-            var callTargetDisplayParts = callTargetSymbol ? ts.symbolToDisplayParts(typeChecker, callTargetSymbol, /*enclosingDeclaration*/ undefined, /*meaning*/ undefined) : ts.emptyArray;
-            var items = candidates.map(function (candidateSignature) { return getSignatureHelpItem(candidateSignature, callTargetDisplayParts, isTypeParameterList, typeChecker, enclosingDeclaration, sourceFile); });
+            var callTargetSymbol = invocation.kind === 2 /* Contextual */ ? invocation.symbol : (typeChecker.getSymbolAtLocation(getExpressionFromInvocation(invocation)) || useFullPrefix && ((_b = resolvedSignature.declaration) === null || _b === void 0 ? void 0 : _b.symbol));
+            var callTargetDisplayParts = callTargetSymbol ? ts.symbolToDisplayParts(typeChecker, callTargetSymbol, useFullPrefix ? sourceFile : undefined, /*meaning*/ undefined) : ts.emptyArray;
+            var items = ts.map(candidates, function (candidateSignature) { return getSignatureHelpItem(candidateSignature, callTargetDisplayParts, isTypeParameterList, typeChecker, enclosingDeclaration, sourceFile); });
             if (argumentIndex !== 0) {
                 ts.Debug.assertLessThan(argumentIndex, argumentCount);
             }
-            var selectedItemIndex = candidates.indexOf(resolvedSignature);
+            var selectedItemIndex = 0;
+            var itemsSeen = 0;
+            for (var i = 0; i < items.length; i++) {
+                var item = items[i];
+                if (candidates[i] === resolvedSignature) {
+                    selectedItemIndex = itemsSeen;
+                    if (item.length > 1) {
+                        // check to see if any items in the list better match than the first one, as the checker isn't filtering the nested lists
+                        // (those come from tuple parameter expansion)
+                        var count = 0;
+                        for (var _i = 0, item_1 = item; _i < item_1.length; _i++) {
+                            var i_1 = item_1[_i];
+                            if (i_1.isVariadic || i_1.parameters.length >= argumentCount) {
+                                selectedItemIndex = itemsSeen + count;
+                                break;
+                            }
+                            count++;
+                        }
+                    }
+                }
+                itemsSeen += item.length;
+            }
             ts.Debug.assert(selectedItemIndex !== -1); // If candidates is non-empty it should always include bestSignature. We check for an empty candidates before calling this function.
-            return { items: items, applicableSpan: applicableSpan, selectedItemIndex: selectedItemIndex, argumentIndex: argumentIndex, argumentCount: argumentCount };
+            return { items: ts.flatMapToMutable(items, ts.identity), applicableSpan: applicableSpan, selectedItemIndex: selectedItemIndex, argumentIndex: argumentIndex, argumentCount: argumentCount };
         }
         function createTypeHelpItems(symbol, _a, sourceFile, checker) {
             var argumentCount = _a.argumentCount, applicableSpan = _a.argumentsSpan, invocation = _a.invocation, argumentIndex = _a.argumentIndex;
@@ -122538,12 +128691,15 @@ var ts;
         }
         var separatorDisplayParts = [ts.punctuationPart(27 /* CommaToken */), ts.spacePart()];
         function getSignatureHelpItem(candidateSignature, callTargetDisplayParts, isTypeParameterList, checker, enclosingDeclaration, sourceFile) {
-            var _a = (isTypeParameterList ? itemInfoForTypeParameters : itemInfoForParameters)(candidateSignature, checker, enclosingDeclaration, sourceFile), isVariadic = _a.isVariadic, parameters = _a.parameters, prefix = _a.prefix, suffix = _a.suffix;
-            var prefixDisplayParts = __spreadArrays(callTargetDisplayParts, prefix);
-            var suffixDisplayParts = __spreadArrays(suffix, returnTypeToDisplayParts(candidateSignature, enclosingDeclaration, checker));
-            var documentation = candidateSignature.getDocumentationComment(checker);
-            var tags = candidateSignature.getJsDocTags();
-            return { isVariadic: isVariadic, prefixDisplayParts: prefixDisplayParts, suffixDisplayParts: suffixDisplayParts, separatorDisplayParts: separatorDisplayParts, parameters: parameters, documentation: documentation, tags: tags };
+            var infos = (isTypeParameterList ? itemInfoForTypeParameters : itemInfoForParameters)(candidateSignature, checker, enclosingDeclaration, sourceFile);
+            return ts.map(infos, function (_a) {
+                var isVariadic = _a.isVariadic, parameters = _a.parameters, prefix = _a.prefix, suffix = _a.suffix;
+                var prefixDisplayParts = __spreadArrays(callTargetDisplayParts, prefix);
+                var suffixDisplayParts = __spreadArrays(suffix, returnTypeToDisplayParts(candidateSignature, enclosingDeclaration, checker));
+                var documentation = candidateSignature.getDocumentationComment(checker);
+                var tags = candidateSignature.getJsDocTags();
+                return { isVariadic: isVariadic, prefixDisplayParts: prefixDisplayParts, suffixDisplayParts: suffixDisplayParts, separatorDisplayParts: separatorDisplayParts, parameters: parameters, documentation: documentation, tags: tags };
+            });
         }
         function returnTypeToDisplayParts(candidateSignature, enclosingDeclaration, checker) {
             return ts.mapToDisplayParts(function (writer) {
@@ -122562,24 +128718,33 @@ var ts;
             var typeParameters = (candidateSignature.target || candidateSignature).typeParameters;
             var printer = ts.createPrinter({ removeComments: true });
             var parameters = (typeParameters || ts.emptyArray).map(function (t) { return createSignatureHelpParameterForTypeParameter(t, checker, enclosingDeclaration, sourceFile, printer); });
-            var parameterParts = ts.mapToDisplayParts(function (writer) {
-                var thisParameter = candidateSignature.thisParameter ? [checker.symbolToParameterDeclaration(candidateSignature.thisParameter, enclosingDeclaration, signatureHelpNodeBuilderFlags)] : [];
-                var params = ts.createNodeArray(__spreadArrays(thisParameter, checker.getExpandedParameters(candidateSignature).map(function (param) { return checker.symbolToParameterDeclaration(param, enclosingDeclaration, signatureHelpNodeBuilderFlags); })));
-                printer.writeList(2576 /* CallExpressionArguments */, params, sourceFile, writer);
+            var thisParameter = candidateSignature.thisParameter ? [checker.symbolToParameterDeclaration(candidateSignature.thisParameter, enclosingDeclaration, signatureHelpNodeBuilderFlags)] : [];
+            return checker.getExpandedParameters(candidateSignature).map(function (paramList) {
+                var params = ts.factory.createNodeArray(__spreadArrays(thisParameter, ts.map(paramList, function (param) { return checker.symbolToParameterDeclaration(param, enclosingDeclaration, signatureHelpNodeBuilderFlags); })));
+                var parameterParts = ts.mapToDisplayParts(function (writer) {
+                    printer.writeList(2576 /* CallExpressionArguments */, params, sourceFile, writer);
+                });
+                return { isVariadic: false, parameters: parameters, prefix: [ts.punctuationPart(29 /* LessThanToken */)], suffix: __spreadArrays([ts.punctuationPart(31 /* GreaterThanToken */)], parameterParts) };
             });
-            return { isVariadic: false, parameters: parameters, prefix: [ts.punctuationPart(29 /* LessThanToken */)], suffix: __spreadArrays([ts.punctuationPart(31 /* GreaterThanToken */)], parameterParts) };
         }
         function itemInfoForParameters(candidateSignature, checker, enclosingDeclaration, sourceFile) {
             var isVariadic = checker.hasEffectiveRestParameter(candidateSignature);
             var printer = ts.createPrinter({ removeComments: true });
             var typeParameterParts = ts.mapToDisplayParts(function (writer) {
                 if (candidateSignature.typeParameters && candidateSignature.typeParameters.length) {
-                    var args = ts.createNodeArray(candidateSignature.typeParameters.map(function (p) { return checker.typeParameterToDeclaration(p, enclosingDeclaration, signatureHelpNodeBuilderFlags); }));
+                    var args = ts.factory.createNodeArray(candidateSignature.typeParameters.map(function (p) { return checker.typeParameterToDeclaration(p, enclosingDeclaration, signatureHelpNodeBuilderFlags); }));
                     printer.writeList(53776 /* TypeParameters */, args, sourceFile, writer);
                 }
             });
-            var parameters = checker.getExpandedParameters(candidateSignature).map(function (p) { return createSignatureHelpParameterForParameter(p, checker, enclosingDeclaration, sourceFile, printer); });
-            return { isVariadic: isVariadic, parameters: parameters, prefix: __spreadArrays(typeParameterParts, [ts.punctuationPart(20 /* OpenParenToken */)]), suffix: [ts.punctuationPart(21 /* CloseParenToken */)] };
+            var lists = checker.getExpandedParameters(candidateSignature);
+            return lists.map(function (parameterList) {
+                return {
+                    isVariadic: isVariadic && (lists.length === 1 || !!(parameterList[parameterList.length - 1].checkFlags & 32768 /* RestParameter */)),
+                    parameters: parameterList.map(function (p) { return createSignatureHelpParameterForParameter(p, checker, enclosingDeclaration, sourceFile, printer); }),
+                    prefix: __spreadArrays(typeParameterParts, [ts.punctuationPart(20 /* OpenParenToken */)]),
+                    suffix: [ts.punctuationPart(21 /* CloseParenToken */)]
+                };
+            });
         }
         function createSignatureHelpParameterForParameter(parameter, checker, enclosingDeclaration, sourceFile, printer) {
             var displayParts = ts.mapToDisplayParts(function (writer) {
@@ -122605,8 +128770,8 @@ var ts;
     function getSourceMapper(host) {
         var getCanonicalFileName = ts.createGetCanonicalFileName(host.useCaseSensitiveFileNames());
         var currentDirectory = host.getCurrentDirectory();
-        var sourceFileLike = ts.createMap();
-        var documentPositionMappers = ts.createMap();
+        var sourceFileLike = new ts.Map();
+        var documentPositionMappers = new ts.Map();
         return { tryGetSourcePosition: tryGetSourcePosition, tryGetGeneratedPosition: tryGetGeneratedPosition, toLineColumnOffset: toLineColumnOffset, clearCache: clearCache };
         function toPath(fileName) {
             return ts.toPath(fileName, currentDirectory, getCanonicalFileName);
@@ -122648,7 +128813,7 @@ var ts;
                 return undefined;
             }
             var options = program.getCompilerOptions();
-            var outPath = options.outFile || options.out;
+            var outPath = ts.outFile(options);
             var declarationPath = outPath ?
                 ts.removeFileExtension(outPath) + ".d.ts" /* Dts */ :
                 ts.getDeclarationEmitOutputFilePathWorker(info.fileName, program.getCompilerOptions(), currentDirectory, program.getCommonSourceDirectory(), getCanonicalFileName);
@@ -122754,7 +128919,7 @@ var ts;
 /* @internal */
 var ts;
 (function (ts) {
-    var visitedNestedConvertibleFunctions = ts.createMap();
+    var visitedNestedConvertibleFunctions = new ts.Map();
     function computeSuggestionDiagnostics(sourceFile, program, cancellationToken) {
         program.getSemanticDiagnostics(sourceFile, cancellationToken);
         var diags = [];
@@ -122786,7 +128951,7 @@ var ts;
         return diags.sort(function (d1, d2) { return d1.start - d2.start; });
         function check(node) {
             if (isJsFile) {
-                if (canBeConvertedToClass(node)) {
+                if (canBeConvertedToClass(node, checker)) {
                     diags.push(ts.createDiagnosticForNode(ts.isVariableDeclaration(node.parent) ? node.parent.name : node, ts.Diagnostics.This_constructor_function_may_be_converted_to_a_class_declaration));
                 }
             }
@@ -122815,11 +128980,11 @@ var ts;
     function containsTopLevelCommonjs(sourceFile) {
         return sourceFile.statements.some(function (statement) {
             switch (statement.kind) {
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return statement.declarationList.declarations.some(function (decl) {
                         return !!decl.initializer && ts.isRequireCall(propertyAccessLeftHandSide(decl.initializer), /*checkArgumentIsStringLiteralLike*/ true);
                     });
-                case 226 /* ExpressionStatement */: {
+                case 233 /* ExpressionStatement */: {
                     var expression = statement.expression;
                     if (!ts.isBinaryExpression(expression))
                         return ts.isRequireCall(expression, /*checkArgumentIsStringLiteralLike*/ true);
@@ -122836,12 +129001,12 @@ var ts;
     }
     function importNameForConvertToDefaultImport(node) {
         switch (node.kind) {
-            case 254 /* ImportDeclaration */:
+            case 261 /* ImportDeclaration */:
                 var importClause = node.importClause, moduleSpecifier = node.moduleSpecifier;
-                return importClause && !importClause.name && importClause.namedBindings && importClause.namedBindings.kind === 256 /* NamespaceImport */ && ts.isStringLiteral(moduleSpecifier)
+                return importClause && !importClause.name && importClause.namedBindings && importClause.namedBindings.kind === 263 /* NamespaceImport */ && ts.isStringLiteral(moduleSpecifier)
                     ? importClause.namedBindings.name
                     : undefined;
-            case 253 /* ImportEqualsDeclaration */:
+            case 260 /* ImportEqualsDeclaration */:
                 return node.name;
             default:
                 return undefined;
@@ -122903,20 +129068,20 @@ var ts;
         if (node.arguments.length < 2)
             return true;
         return ts.some(node.arguments, function (arg) {
-            return arg.kind === 100 /* NullKeyword */ ||
+            return arg.kind === 103 /* NullKeyword */ ||
                 ts.isIdentifier(arg) && arg.text === "undefined";
         });
     }
     // should be kept up to date with getTransformationBody in convertToAsyncFunction.ts
     function isFixablePromiseArgument(arg) {
         switch (arg.kind) {
-            case 244 /* FunctionDeclaration */:
-            case 201 /* FunctionExpression */:
-            case 202 /* ArrowFunction */:
+            case 251 /* FunctionDeclaration */:
+            case 208 /* FunctionExpression */:
+            case 209 /* ArrowFunction */:
                 visitedNestedConvertibleFunctions.set(getKeyFromNode(arg), true);
             // falls through
-            case 100 /* NullKeyword */:
-            case 75 /* Identifier */: // identifier includes undefined
+            case 103 /* NullKeyword */:
+            case 78 /* Identifier */: // identifier includes undefined
                 return true;
             default:
                 return false;
@@ -122925,17 +129090,16 @@ var ts;
     function getKeyFromNode(exp) {
         return exp.pos.toString() + ":" + exp.end.toString();
     }
-    function canBeConvertedToClass(node) {
+    function canBeConvertedToClass(node, checker) {
         var _a, _b, _c, _d;
-        if (node.kind === 201 /* FunctionExpression */) {
+        if (node.kind === 208 /* FunctionExpression */) {
             if (ts.isVariableDeclaration(node.parent) && ((_a = node.symbol.members) === null || _a === void 0 ? void 0 : _a.size)) {
                 return true;
             }
-            var decl = ts.getDeclarationOfExpando(node);
-            var symbol = decl === null || decl === void 0 ? void 0 : decl.symbol;
+            var symbol = checker.getSymbolOfExpando(node, /*allowDeclaration*/ false);
             return !!(symbol && (((_b = symbol.exports) === null || _b === void 0 ? void 0 : _b.size) || ((_c = symbol.members) === null || _c === void 0 ? void 0 : _c.size)));
         }
-        if (node.kind === 244 /* FunctionDeclaration */) {
+        if (node.kind === 251 /* FunctionDeclaration */) {
             return !!((_d = node.symbol.members) === null || _d === void 0 ? void 0 : _d.size);
         }
         return false;
@@ -122955,7 +129119,7 @@ var ts;
             }
             var flags = ts.getCombinedLocalAndExportSymbolFlags(symbol);
             if (flags & 32 /* Class */) {
-                return ts.getDeclarationOfKind(symbol, 214 /* ClassExpression */) ?
+                return ts.getDeclarationOfKind(symbol, 221 /* ClassExpression */) ?
                     "local class" /* localClassElement */ : "class" /* classElement */;
             }
             if (flags & 384 /* Enum */)
@@ -122990,7 +129154,7 @@ var ts;
             if (typeChecker.isArgumentsSymbol(symbol)) {
                 return "local var" /* localVariableElement */;
             }
-            if (location.kind === 104 /* ThisKeyword */ && ts.isExpression(location)) {
+            if (location.kind === 107 /* ThisKeyword */ && ts.isExpression(location)) {
                 return "parameter" /* parameterElement */;
             }
             var flags = ts.getCombinedLocalAndExportSymbolFlags(symbol);
@@ -123041,11 +129205,11 @@ var ts;
                 // If we requested completions after `x.` at the top-level, we may be at a source file location.
                 switch (location.parent && location.parent.kind) {
                     // If we've typed a character of the attribute name, will be 'JsxAttribute', else will be 'JsxOpeningElement'.
-                    case 268 /* JsxOpeningElement */:
-                    case 266 /* JsxElement */:
-                    case 267 /* JsxSelfClosingElement */:
-                        return location.kind === 75 /* Identifier */ ? "property" /* memberVariableElement */ : "JSX attribute" /* jsxAttribute */;
-                    case 273 /* JsxAttribute */:
+                    case 275 /* JsxOpeningElement */:
+                    case 273 /* JsxElement */:
+                    case 274 /* JsxSelfClosingElement */:
+                        return location.kind === 78 /* Identifier */ ? "property" /* memberVariableElement */ : "JSX attribute" /* jsxAttribute */;
+                    case 280 /* JsxAttribute */:
                         return "JSX attribute" /* jsxAttribute */;
                     default:
                         return "property" /* memberVariableElement */;
@@ -123072,14 +129236,14 @@ var ts;
             var symbolFlags = ts.getCombinedLocalAndExportSymbolFlags(symbol);
             var symbolKind = semanticMeaning & 1 /* Value */ ? getSymbolKindOfConstructorPropertyMethodAccessorFunctionOrVar(typeChecker, symbol, location) : "" /* unknown */;
             var hasAddedSymbolInfo = false;
-            var isThisExpression = location.kind === 104 /* ThisKeyword */ && ts.isInExpressionContext(location);
+            var isThisExpression = location.kind === 107 /* ThisKeyword */ && ts.isInExpressionContext(location);
             var type;
             var printer;
             var documentationFromAlias;
             var tagsFromAlias;
             var hasMultipleSignatures = false;
-            if (location.kind === 104 /* ThisKeyword */ && !isThisExpression) {
-                return { displayParts: [ts.keywordPart(104 /* ThisKeyword */)], documentation: [], symbolKind: "primitive type" /* primitiveType */, tags: undefined };
+            if (location.kind === 107 /* ThisKeyword */ && !isThisExpression) {
+                return { displayParts: [ts.keywordPart(107 /* ThisKeyword */)], documentation: [], symbolKind: "primitive type" /* primitiveType */, tags: undefined };
             }
             // Class at constructor site need to be shown as constructor apart from property,method, vars
             if (symbolKind !== "" /* unknown */ || symbolFlags & 32 /* Class */ || symbolFlags & 2097152 /* Alias */) {
@@ -123089,7 +129253,7 @@ var ts;
                 }
                 var signature = void 0;
                 type = isThisExpression ? typeChecker.getTypeAtLocation(location) : typeChecker.getTypeOfSymbolAtLocation(symbol.exportSymbol || symbol, location);
-                if (location.parent && location.parent.kind === 194 /* PropertyAccessExpression */) {
+                if (location.parent && location.parent.kind === 201 /* PropertyAccessExpression */) {
                     var right = location.parent.name;
                     // Either the location is on the right of a property access, or on the left and the right is missing
                     if (right === location || (right && right.getFullWidth() === 0)) {
@@ -123104,12 +129268,12 @@ var ts;
                 else if (ts.isCallExpressionTarget(location) || ts.isNewExpressionTarget(location)) {
                     callExpressionLike = location.parent;
                 }
-                else if (location.parent && ts.isJsxOpeningLikeElement(location.parent) && ts.isFunctionLike(symbol.valueDeclaration)) {
+                else if (location.parent && (ts.isJsxOpeningLikeElement(location.parent) || ts.isTaggedTemplateExpression(location.parent)) && ts.isFunctionLike(symbol.valueDeclaration)) {
                     callExpressionLike = location.parent;
                 }
                 if (callExpressionLike) {
                     signature = typeChecker.getResolvedSignature(callExpressionLike); // TODO: GH#18217
-                    var useConstructSignatures = callExpressionLike.kind === 197 /* NewExpression */ || (ts.isCallExpression(callExpressionLike) && callExpressionLike.expression.kind === 102 /* SuperKeyword */);
+                    var useConstructSignatures = callExpressionLike.kind === 204 /* NewExpression */ || (ts.isCallExpression(callExpressionLike) && callExpressionLike.expression.kind === 105 /* SuperKeyword */);
                     var allSignatures = useConstructSignatures ? type.getConstructSignatures() : type.getCallSignatures();
                     if (!ts.contains(allSignatures, signature.target) && !ts.contains(allSignatures, signature)) {
                         // Get the first signature if there is one -- allSignatures may contain
@@ -123127,7 +129291,7 @@ var ts;
                             pushSymbolKind(symbolKind);
                             displayParts.push(ts.spacePart());
                             if (useConstructSignatures) {
-                                displayParts.push(ts.keywordPart(99 /* NewKeyword */));
+                                displayParts.push(ts.keywordPart(102 /* NewKeyword */));
                                 displayParts.push(ts.spacePart());
                             }
                             addFullSymbolName(symbol);
@@ -123151,7 +129315,7 @@ var ts;
                                     displayParts.push(ts.lineBreakPart());
                                 }
                                 if (useConstructSignatures) {
-                                    displayParts.push(ts.keywordPart(99 /* NewKeyword */));
+                                    displayParts.push(ts.keywordPart(102 /* NewKeyword */));
                                     displayParts.push(ts.spacePart());
                                 }
                                 addSignatureDisplayParts(signature, allSignatures, 262144 /* WriteArrowStyleSignature */);
@@ -123165,29 +129329,29 @@ var ts;
                     }
                 }
                 else if ((ts.isNameOfFunctionDeclaration(location) && !(symbolFlags & 98304 /* Accessor */)) || // name of function declaration
-                    (location.kind === 129 /* ConstructorKeyword */ && location.parent.kind === 162 /* Constructor */)) { // At constructor keyword of constructor declaration
+                    (location.kind === 132 /* ConstructorKeyword */ && location.parent.kind === 166 /* Constructor */)) { // At constructor keyword of constructor declaration
                     // get the signature from the declaration and write it
                     var functionDeclaration_1 = location.parent;
                     // Use function declaration to write the signatures only if the symbol corresponding to this declaration
                     var locationIsSymbolDeclaration = symbol.declarations && ts.find(symbol.declarations, function (declaration) {
-                        return declaration === (location.kind === 129 /* ConstructorKeyword */ ? functionDeclaration_1.parent : functionDeclaration_1);
+                        return declaration === (location.kind === 132 /* ConstructorKeyword */ ? functionDeclaration_1.parent : functionDeclaration_1);
                     });
                     if (locationIsSymbolDeclaration) {
-                        var allSignatures = functionDeclaration_1.kind === 162 /* Constructor */ ? type.getNonNullableType().getConstructSignatures() : type.getNonNullableType().getCallSignatures();
+                        var allSignatures = functionDeclaration_1.kind === 166 /* Constructor */ ? type.getNonNullableType().getConstructSignatures() : type.getNonNullableType().getCallSignatures();
                         if (!typeChecker.isImplementationOfOverload(functionDeclaration_1)) {
                             signature = typeChecker.getSignatureFromDeclaration(functionDeclaration_1); // TODO: GH#18217
                         }
                         else {
                             signature = allSignatures[0];
                         }
-                        if (functionDeclaration_1.kind === 162 /* Constructor */) {
+                        if (functionDeclaration_1.kind === 166 /* Constructor */) {
                             // show (constructor) Type(...) signature
                             symbolKind = "constructor" /* constructorImplementationElement */;
                             addPrefixForAnyFunctionOrVar(type.symbol, symbolKind);
                         }
                         else {
                             // (function/method) symbol(..signature)
-                            addPrefixForAnyFunctionOrVar(functionDeclaration_1.kind === 165 /* CallSignature */ &&
+                            addPrefixForAnyFunctionOrVar(functionDeclaration_1.kind === 169 /* CallSignature */ &&
                                 !(type.symbol.flags & 2048 /* TypeLiteral */ || type.symbol.flags & 4096 /* ObjectLiteral */) ? type.symbol : symbol, symbolKind);
                         }
                         addSignatureDisplayParts(signature, allSignatures);
@@ -123198,7 +129362,7 @@ var ts;
             }
             if (symbolFlags & 32 /* Class */ && !hasAddedSymbolInfo && !isThisExpression) {
                 addAliasPrefixIfNecessary();
-                if (ts.getDeclarationOfKind(symbol, 214 /* ClassExpression */)) {
+                if (ts.getDeclarationOfKind(symbol, 221 /* ClassExpression */)) {
                     // Special case for class expressions because we would like to indicate that
                     // the class name is local to the class body (similar to function expression)
                     //      (local class) class <className>
@@ -123206,7 +129370,7 @@ var ts;
                 }
                 else {
                     // Class declaration has name which is not local.
-                    displayParts.push(ts.keywordPart(80 /* ClassKeyword */));
+                    displayParts.push(ts.keywordPart(83 /* ClassKeyword */));
                 }
                 displayParts.push(ts.spacePart());
                 addFullSymbolName(symbol);
@@ -123214,14 +129378,14 @@ var ts;
             }
             if ((symbolFlags & 64 /* Interface */) && (semanticMeaning & 2 /* Type */)) {
                 prefixNextMeaning();
-                displayParts.push(ts.keywordPart(114 /* InterfaceKeyword */));
+                displayParts.push(ts.keywordPart(117 /* InterfaceKeyword */));
                 displayParts.push(ts.spacePart());
                 addFullSymbolName(symbol);
                 writeTypeParametersOfSymbol(symbol, sourceFile);
             }
             if ((symbolFlags & 524288 /* TypeAlias */) && (semanticMeaning & 2 /* Type */)) {
                 prefixNextMeaning();
-                displayParts.push(ts.keywordPart(145 /* TypeKeyword */));
+                displayParts.push(ts.keywordPart(149 /* TypeKeyword */));
                 displayParts.push(ts.spacePart());
                 addFullSymbolName(symbol);
                 writeTypeParametersOfSymbol(symbol, sourceFile);
@@ -123233,18 +129397,18 @@ var ts;
             if (symbolFlags & 384 /* Enum */) {
                 prefixNextMeaning();
                 if (ts.some(symbol.declarations, function (d) { return ts.isEnumDeclaration(d) && ts.isEnumConst(d); })) {
-                    displayParts.push(ts.keywordPart(81 /* ConstKeyword */));
+                    displayParts.push(ts.keywordPart(84 /* ConstKeyword */));
                     displayParts.push(ts.spacePart());
                 }
-                displayParts.push(ts.keywordPart(88 /* EnumKeyword */));
+                displayParts.push(ts.keywordPart(91 /* EnumKeyword */));
                 displayParts.push(ts.spacePart());
                 addFullSymbolName(symbol);
             }
             if (symbolFlags & 1536 /* Module */ && !isThisExpression) {
                 prefixNextMeaning();
-                var declaration = ts.getDeclarationOfKind(symbol, 249 /* ModuleDeclaration */);
-                var isNamespace = declaration && declaration.name && declaration.name.kind === 75 /* Identifier */;
-                displayParts.push(ts.keywordPart(isNamespace ? 136 /* NamespaceKeyword */ : 135 /* ModuleKeyword */));
+                var declaration = ts.getDeclarationOfKind(symbol, 256 /* ModuleDeclaration */);
+                var isNamespace = declaration && declaration.name && declaration.name.kind === 78 /* Identifier */;
+                displayParts.push(ts.keywordPart(isNamespace ? 140 /* NamespaceKeyword */ : 139 /* ModuleKeyword */));
                 displayParts.push(ts.spacePart());
                 addFullSymbolName(symbol);
             }
@@ -123263,7 +129427,7 @@ var ts;
                 }
                 else {
                     // Method/function type parameter
-                    var decl = ts.getDeclarationOfKind(symbol, 155 /* TypeParameter */);
+                    var decl = ts.getDeclarationOfKind(symbol, 159 /* TypeParameter */);
                     if (decl === undefined)
                         return ts.Debug.fail();
                     var declaration = decl.parent;
@@ -123271,21 +129435,21 @@ var ts;
                         if (ts.isFunctionLikeKind(declaration.kind)) {
                             addInPrefix();
                             var signature = typeChecker.getSignatureFromDeclaration(declaration); // TODO: GH#18217
-                            if (declaration.kind === 166 /* ConstructSignature */) {
-                                displayParts.push(ts.keywordPart(99 /* NewKeyword */));
+                            if (declaration.kind === 170 /* ConstructSignature */) {
+                                displayParts.push(ts.keywordPart(102 /* NewKeyword */));
                                 displayParts.push(ts.spacePart());
                             }
-                            else if (declaration.kind !== 165 /* CallSignature */ && declaration.name) {
+                            else if (declaration.kind !== 169 /* CallSignature */ && declaration.name) {
                                 addFullSymbolName(declaration.symbol);
                             }
                             ts.addRange(displayParts, ts.signatureToDisplayParts(typeChecker, signature, sourceFile, 32 /* WriteTypeArgumentsOfSignature */));
                         }
-                        else if (declaration.kind === 247 /* TypeAliasDeclaration */) {
+                        else if (declaration.kind === 254 /* TypeAliasDeclaration */) {
                             // Type alias type parameter
                             // For example
                             //      type list<T> = T[]; // Both T will go through same code path
                             addInPrefix();
-                            displayParts.push(ts.keywordPart(145 /* TypeKeyword */));
+                            displayParts.push(ts.keywordPart(149 /* TypeKeyword */));
                             displayParts.push(ts.spacePart());
                             addFullSymbolName(declaration.symbol);
                             writeTypeParametersOfSymbol(declaration.symbol, sourceFile);
@@ -123297,7 +129461,7 @@ var ts;
                 symbolKind = "enum member" /* enumMemberElement */;
                 addPrefixForAnyFunctionOrVar(symbol, "enum member");
                 var declaration = symbol.declarations[0];
-                if (declaration.kind === 284 /* EnumMember */) {
+                if (declaration.kind === 291 /* EnumMember */) {
                     var constantValue = typeChecker.getConstantValue(declaration);
                     if (constantValue !== undefined) {
                         displayParts.push(ts.spacePart());
@@ -123307,7 +129471,8 @@ var ts;
                     }
                 }
             }
-            if (symbolFlags & 2097152 /* Alias */) {
+            // don't use symbolFlags since getAliasedSymbol requires the flag on the symbol itself
+            if (symbol.flags & 2097152 /* Alias */) {
                 prefixNextMeaning();
                 if (!hasAddedSymbolInfo) {
                     var resolvedSymbol = typeChecker.getAliasedSymbol(symbol);
@@ -123316,7 +129481,7 @@ var ts;
                         var declarationName = ts.getNameOfDeclaration(resolvedNode);
                         if (declarationName) {
                             var isExternalModuleDeclaration = ts.isModuleWithStringLiteralName(resolvedNode) &&
-                                ts.hasModifier(resolvedNode, 2 /* Ambient */);
+                                ts.hasSyntacticModifier(resolvedNode, 2 /* Ambient */);
                             var shouldUseAliasName = symbol.name !== "default" && !isExternalModuleDeclaration;
                             var resolvedInfo = getSymbolDisplayPartsDocumentationAndSymbolKind(typeChecker, resolvedSymbol, ts.getSourceFileOfNode(resolvedNode), resolvedNode, declarationName, semanticMeaning, shouldUseAliasName ? symbol : resolvedSymbol);
                             displayParts.push.apply(displayParts, resolvedInfo.displayParts);
@@ -123327,32 +129492,32 @@ var ts;
                     }
                 }
                 switch (symbol.declarations[0].kind) {
-                    case 252 /* NamespaceExportDeclaration */:
-                        displayParts.push(ts.keywordPart(89 /* ExportKeyword */));
+                    case 259 /* NamespaceExportDeclaration */:
+                        displayParts.push(ts.keywordPart(92 /* ExportKeyword */));
                         displayParts.push(ts.spacePart());
-                        displayParts.push(ts.keywordPart(136 /* NamespaceKeyword */));
+                        displayParts.push(ts.keywordPart(140 /* NamespaceKeyword */));
                         break;
-                    case 259 /* ExportAssignment */:
-                        displayParts.push(ts.keywordPart(89 /* ExportKeyword */));
+                    case 266 /* ExportAssignment */:
+                        displayParts.push(ts.keywordPart(92 /* ExportKeyword */));
                         displayParts.push(ts.spacePart());
-                        displayParts.push(ts.keywordPart(symbol.declarations[0].isExportEquals ? 62 /* EqualsToken */ : 84 /* DefaultKeyword */));
+                        displayParts.push(ts.keywordPart(symbol.declarations[0].isExportEquals ? 62 /* EqualsToken */ : 87 /* DefaultKeyword */));
                         break;
-                    case 263 /* ExportSpecifier */:
-                        displayParts.push(ts.keywordPart(89 /* ExportKeyword */));
+                    case 270 /* ExportSpecifier */:
+                        displayParts.push(ts.keywordPart(92 /* ExportKeyword */));
                         break;
                     default:
-                        displayParts.push(ts.keywordPart(96 /* ImportKeyword */));
+                        displayParts.push(ts.keywordPart(99 /* ImportKeyword */));
                 }
                 displayParts.push(ts.spacePart());
                 addFullSymbolName(symbol);
                 ts.forEach(symbol.declarations, function (declaration) {
-                    if (declaration.kind === 253 /* ImportEqualsDeclaration */) {
+                    if (declaration.kind === 260 /* ImportEqualsDeclaration */) {
                         var importEqualsDeclaration = declaration;
                         if (ts.isExternalModuleImportEqualsDeclaration(importEqualsDeclaration)) {
                             displayParts.push(ts.spacePart());
                             displayParts.push(ts.operatorPart(62 /* EqualsToken */));
                             displayParts.push(ts.spacePart());
-                            displayParts.push(ts.keywordPart(139 /* RequireKeyword */));
+                            displayParts.push(ts.keywordPart(143 /* RequireKeyword */));
                             displayParts.push(ts.punctuationPart(20 /* OpenParenToken */));
                             displayParts.push(ts.displayPart(ts.getTextOfNode(ts.getExternalModuleImportEqualsDeclarationExpression(importEqualsDeclaration)), ts.SymbolDisplayPartKind.stringLiteral));
                             displayParts.push(ts.punctuationPart(21 /* CloseParenToken */));
@@ -123375,7 +129540,7 @@ var ts;
                     if (type) {
                         if (isThisExpression) {
                             prefixNextMeaning();
-                            displayParts.push(ts.keywordPart(104 /* ThisKeyword */));
+                            displayParts.push(ts.keywordPart(107 /* ThisKeyword */));
                         }
                         else {
                             addPrefixForAnyFunctionOrVar(symbol, symbolKind);
@@ -123399,6 +129564,14 @@ var ts;
                             else {
                                 ts.addRange(displayParts, ts.typeToDisplayParts(typeChecker, type, enclosingDeclaration));
                             }
+                            if (symbol.target && symbol.target.tupleLabelDeclaration) {
+                                var labelDecl = symbol.target.tupleLabelDeclaration;
+                                ts.Debug.assertNode(labelDecl.name, ts.isIdentifier);
+                                displayParts.push(ts.spacePart());
+                                displayParts.push(ts.punctuationPart(20 /* OpenParenToken */));
+                                displayParts.push(ts.textPart(ts.idText(labelDecl.name)));
+                                displayParts.push(ts.punctuationPart(21 /* CloseParenToken */));
+                            }
                         }
                         else if (symbolFlags & 16 /* Function */ ||
                             symbolFlags & 8192 /* Method */ ||
@@ -123419,16 +129592,16 @@ var ts;
                 }
             }
             if (documentation.length === 0 && !hasMultipleSignatures) {
-                documentation = symbol.getDocumentationComment(typeChecker);
+                documentation = symbol.getContextualDocumentationComment(enclosingDeclaration, typeChecker);
             }
             if (documentation.length === 0 && symbolFlags & 4 /* Property */) {
                 // For some special property access expressions like `exports.foo = foo` or `module.exports.foo = foo`
                 // there documentation comments might be attached to the right hand side symbol of their declarations.
                 // The pattern of such special property access is that the parent symbol is the symbol of the file.
-                if (symbol.parent && ts.forEach(symbol.parent.declarations, function (declaration) { return declaration.kind === 290 /* SourceFile */; })) {
+                if (symbol.parent && ts.forEach(symbol.parent.declarations, function (declaration) { return declaration.kind === 297 /* SourceFile */; })) {
                     for (var _i = 0, _a = symbol.declarations; _i < _a.length; _i++) {
                         var declaration = _a[_i];
-                        if (!declaration.parent || declaration.parent.kind !== 209 /* BinaryExpression */) {
+                        if (!declaration.parent || declaration.parent.kind !== 216 /* BinaryExpression */) {
                             continue;
                         }
                         var rhsSymbol = typeChecker.getSymbolAtLocation(declaration.parent.right);
@@ -123473,7 +129646,7 @@ var ts;
             }
             function addInPrefix() {
                 displayParts.push(ts.spacePart());
-                displayParts.push(ts.keywordPart(97 /* InKeyword */));
+                displayParts.push(ts.keywordPart(100 /* InKeyword */));
                 displayParts.push(ts.spacePart());
             }
             function addFullSymbolName(symbolToDisplay, enclosingDeclaration) {
@@ -123546,16 +129719,16 @@ var ts;
             }
             return ts.forEach(symbol.declarations, function (declaration) {
                 // Function expressions are local
-                if (declaration.kind === 201 /* FunctionExpression */) {
+                if (declaration.kind === 208 /* FunctionExpression */) {
                     return true;
                 }
-                if (declaration.kind !== 242 /* VariableDeclaration */ && declaration.kind !== 244 /* FunctionDeclaration */) {
+                if (declaration.kind !== 249 /* VariableDeclaration */ && declaration.kind !== 251 /* FunctionDeclaration */) {
                     return false;
                 }
                 // If the parent is not sourceFile or module block it is local variable
                 for (var parent = declaration.parent; !ts.isFunctionBlock(parent); parent = parent.parent) {
                     // Reached source file or module block
-                    if (parent.kind === 290 /* SourceFile */ || parent.kind === 250 /* ModuleBlock */) {
+                    if (parent.kind === 297 /* SourceFile */ || parent.kind === 257 /* ModuleBlock */) {
                         return false;
                     }
                 }
@@ -123601,7 +129774,7 @@ var ts;
             sourceFile.moduleName = transpileOptions.moduleName;
         }
         if (transpileOptions.renamedDependencies) {
-            sourceFile.renamedDependencies = ts.createMapFromTemplate(transpileOptions.renamedDependencies);
+            sourceFile.renamedDependencies = new ts.Map(ts.getEntries(transpileOptions.renamedDependencies));
         }
         var newLine = ts.getNewLineCharacter(options);
         // Output
@@ -123807,6 +129980,7 @@ var ts;
                 getCurrentLeadingTrivia: function () { return leadingTrivia; },
                 lastTrailingTriviaWasNewLine: function () { return wasNewLine; },
                 skipToEndOf: skipToEndOf,
+                skipToStartOf: skipToStartOf,
             });
             lastTokenInfo = undefined;
             scanner.setText(undefined);
@@ -123855,12 +130029,12 @@ var ts;
             function shouldRescanJsxIdentifier(node) {
                 if (node.parent) {
                     switch (node.parent.kind) {
-                        case 273 /* JsxAttribute */:
-                        case 268 /* JsxOpeningElement */:
-                        case 269 /* JsxClosingElement */:
-                        case 267 /* JsxSelfClosingElement */:
+                        case 280 /* JsxAttribute */:
+                        case 275 /* JsxOpeningElement */:
+                        case 276 /* JsxClosingElement */:
+                        case 274 /* JsxSelfClosingElement */:
                             // May parse an identifier like `module-layout`; that will be scanned as a keyword at first, but we should parse the whole thing to get an identifier.
-                            return ts.isKeyword(node.kind) || node.kind === 75 /* Identifier */;
+                            return ts.isKeyword(node.kind) || node.kind === 78 /* Identifier */;
                     }
                 }
                 return false;
@@ -124013,6 +130187,15 @@ var ts;
                 leadingTrivia = undefined;
                 trailingTrivia = undefined;
             }
+            function skipToStartOf(node) {
+                scanner.setTextPos(node.pos);
+                savedPos = scanner.getStartPos();
+                lastScanAction = undefined;
+                lastTokenInfo = undefined;
+                wasNewLine = false;
+                leadingTrivia = undefined;
+                trailingTrivia = undefined;
+            }
         }
         formatting.getFormattingScanner = getFormattingScanner;
     })(formatting = ts.formatting || (ts.formatting = {}));
@@ -124050,7 +130233,7 @@ var ts;
     (function (formatting) {
         function getAllRules() {
             var allTokens = [];
-            for (var token = 0 /* FirstToken */; token <= 152 /* LastToken */; token++) {
+            for (var token = 0 /* FirstToken */; token <= 156 /* LastToken */; token++) {
                 if (token !== 1 /* EndOfFileToken */) {
                     allTokens.push(token);
                 }
@@ -124065,27 +130248,27 @@ var ts;
             var anyToken = { tokens: allTokens, isSpecific: false };
             var anyTokenIncludingMultilineComments = tokenRangeFrom(__spreadArrays(allTokens, [3 /* MultiLineCommentTrivia */]));
             var anyTokenIncludingEOF = tokenRangeFrom(__spreadArrays(allTokens, [1 /* EndOfFileToken */]));
-            var keywords = tokenRangeFromRange(77 /* FirstKeyword */, 152 /* LastKeyword */);
-            var binaryOperators = tokenRangeFromRange(29 /* FirstBinaryOperator */, 74 /* LastBinaryOperator */);
-            var binaryKeywordOperators = [97 /* InKeyword */, 98 /* InstanceOfKeyword */, 152 /* OfKeyword */, 123 /* AsKeyword */, 133 /* IsKeyword */];
+            var keywords = tokenRangeFromRange(80 /* FirstKeyword */, 156 /* LastKeyword */);
+            var binaryOperators = tokenRangeFromRange(29 /* FirstBinaryOperator */, 77 /* LastBinaryOperator */);
+            var binaryKeywordOperators = [100 /* InKeyword */, 101 /* InstanceOfKeyword */, 156 /* OfKeyword */, 126 /* AsKeyword */, 137 /* IsKeyword */];
             var unaryPrefixOperators = [45 /* PlusPlusToken */, 46 /* MinusMinusToken */, 54 /* TildeToken */, 53 /* ExclamationToken */];
             var unaryPrefixExpressions = [
-                8 /* NumericLiteral */, 9 /* BigIntLiteral */, 75 /* Identifier */, 20 /* OpenParenToken */,
-                22 /* OpenBracketToken */, 18 /* OpenBraceToken */, 104 /* ThisKeyword */, 99 /* NewKeyword */
+                8 /* NumericLiteral */, 9 /* BigIntLiteral */, 78 /* Identifier */, 20 /* OpenParenToken */,
+                22 /* OpenBracketToken */, 18 /* OpenBraceToken */, 107 /* ThisKeyword */, 102 /* NewKeyword */
             ];
-            var unaryPreincrementExpressions = [75 /* Identifier */, 20 /* OpenParenToken */, 104 /* ThisKeyword */, 99 /* NewKeyword */];
-            var unaryPostincrementExpressions = [75 /* Identifier */, 21 /* CloseParenToken */, 23 /* CloseBracketToken */, 99 /* NewKeyword */];
-            var unaryPredecrementExpressions = [75 /* Identifier */, 20 /* OpenParenToken */, 104 /* ThisKeyword */, 99 /* NewKeyword */];
-            var unaryPostdecrementExpressions = [75 /* Identifier */, 21 /* CloseParenToken */, 23 /* CloseBracketToken */, 99 /* NewKeyword */];
+            var unaryPreincrementExpressions = [78 /* Identifier */, 20 /* OpenParenToken */, 107 /* ThisKeyword */, 102 /* NewKeyword */];
+            var unaryPostincrementExpressions = [78 /* Identifier */, 21 /* CloseParenToken */, 23 /* CloseBracketToken */, 102 /* NewKeyword */];
+            var unaryPredecrementExpressions = [78 /* Identifier */, 20 /* OpenParenToken */, 107 /* ThisKeyword */, 102 /* NewKeyword */];
+            var unaryPostdecrementExpressions = [78 /* Identifier */, 21 /* CloseParenToken */, 23 /* CloseBracketToken */, 102 /* NewKeyword */];
             var comments = [2 /* SingleLineCommentTrivia */, 3 /* MultiLineCommentTrivia */];
-            var typeNames = __spreadArrays([75 /* Identifier */], ts.typeKeywords);
+            var typeNames = __spreadArrays([78 /* Identifier */], ts.typeKeywords);
             // Place a space before open brace in a function declaration
             // TypeScript: Function can have return types, which can be made of tons of different token kinds
             var functionOpenBraceLeftTokenRange = anyTokenIncludingMultilineComments;
             // Place a space before open brace in a TypeScript declaration that has braces as children (class, module, enum, etc)
-            var typeScriptOpenBraceLeftTokenRange = tokenRangeFrom([75 /* Identifier */, 3 /* MultiLineCommentTrivia */, 80 /* ClassKeyword */, 89 /* ExportKeyword */, 96 /* ImportKeyword */]);
+            var typeScriptOpenBraceLeftTokenRange = tokenRangeFrom([78 /* Identifier */, 3 /* MultiLineCommentTrivia */, 83 /* ClassKeyword */, 92 /* ExportKeyword */, 99 /* ImportKeyword */]);
             // Place a space before open brace in a control flow construct
-            var controlOpenBraceLeftTokenRange = tokenRangeFrom([21 /* CloseParenToken */, 3 /* MultiLineCommentTrivia */, 86 /* DoKeyword */, 107 /* TryKeyword */, 92 /* FinallyKeyword */, 87 /* ElseKeyword */]);
+            var controlOpenBraceLeftTokenRange = tokenRangeFrom([21 /* CloseParenToken */, 3 /* MultiLineCommentTrivia */, 89 /* DoKeyword */, 110 /* TryKeyword */, 95 /* FinallyKeyword */, 90 /* ElseKeyword */]);
             // These rules are higher in priority than user-configurable
             var highPriorityCommonRules = [
                 // Leave comments alone
@@ -124093,14 +130276,14 @@ var ts;
                 rule("IgnoreAfterLineComment", 2 /* SingleLineCommentTrivia */, anyToken, formatting.anyContext, 1 /* StopProcessingSpaceActions */),
                 rule("NotSpaceBeforeColon", anyToken, 58 /* ColonToken */, [isNonJsxSameLineTokenContext, isNotBinaryOpContext, isNotTypeAnnotationContext], 16 /* DeleteSpace */),
                 rule("SpaceAfterColon", 58 /* ColonToken */, anyToken, [isNonJsxSameLineTokenContext, isNotBinaryOpContext], 4 /* InsertSpace */),
-                rule("NoSpaceBeforeQuestionMark", anyToken, 57 /* QuestionToken */, [isNonJsxSameLineTokenContext, isNotBinaryOpContext], 16 /* DeleteSpace */),
+                rule("NoSpaceBeforeQuestionMark", anyToken, 57 /* QuestionToken */, [isNonJsxSameLineTokenContext, isNotBinaryOpContext, isNotTypeAnnotationContext], 16 /* DeleteSpace */),
                 // insert space after '?' only when it is used in conditional operator
                 rule("SpaceAfterQuestionMarkInConditionalOperator", 57 /* QuestionToken */, anyToken, [isNonJsxSameLineTokenContext, isConditionalOperatorContext], 4 /* InsertSpace */),
                 // in other cases there should be no space between '?' and next token
                 rule("NoSpaceAfterQuestionMark", 57 /* QuestionToken */, anyToken, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("NoSpaceBeforeDot", anyToken, [24 /* DotToken */, 28 /* QuestionDotToken */], [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("NoSpaceAfterDot", [24 /* DotToken */, 28 /* QuestionDotToken */], anyToken, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
-                rule("NoSpaceBetweenImportParenInImportType", 96 /* ImportKeyword */, 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext, isImportTypeContext], 16 /* DeleteSpace */),
+                rule("NoSpaceBetweenImportParenInImportType", 99 /* ImportKeyword */, 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext, isImportTypeContext], 16 /* DeleteSpace */),
                 // Special handling of unary operators.
                 // Prefix operators generally shouldn't have a space between
                 // them and their target unary expression.
@@ -124127,79 +130310,79 @@ var ts;
                 rule("SpaceAfterCloseBrace", 19 /* CloseBraceToken */, anyTokenExcept(21 /* CloseParenToken */), [isNonJsxSameLineTokenContext, isAfterCodeBlockContext], 4 /* InsertSpace */),
                 // Special case for (}, else) and (}, while) since else & while tokens are not part of the tree which makes SpaceAfterCloseBrace rule not applied
                 // Also should not apply to })
-                rule("SpaceBetweenCloseBraceAndElse", 19 /* CloseBraceToken */, 87 /* ElseKeyword */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
-                rule("SpaceBetweenCloseBraceAndWhile", 19 /* CloseBraceToken */, 111 /* WhileKeyword */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBetweenCloseBraceAndElse", 19 /* CloseBraceToken */, 90 /* ElseKeyword */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBetweenCloseBraceAndWhile", 19 /* CloseBraceToken */, 114 /* WhileKeyword */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 rule("NoSpaceBetweenEmptyBraceBrackets", 18 /* OpenBraceToken */, 19 /* CloseBraceToken */, [isNonJsxSameLineTokenContext, isObjectContext], 16 /* DeleteSpace */),
                 // Add a space after control dec context if the next character is an open bracket ex: 'if (false)[a, b] = [1, 2];' -> 'if (false) [a, b] = [1, 2];'
                 rule("SpaceAfterConditionalClosingParen", 21 /* CloseParenToken */, 22 /* OpenBracketToken */, [isControlDeclContext], 4 /* InsertSpace */),
-                rule("NoSpaceBetweenFunctionKeywordAndStar", 94 /* FunctionKeyword */, 41 /* AsteriskToken */, [isFunctionDeclarationOrFunctionExpressionContext], 16 /* DeleteSpace */),
-                rule("SpaceAfterStarInGeneratorDeclaration", 41 /* AsteriskToken */, 75 /* Identifier */, [isFunctionDeclarationOrFunctionExpressionContext], 4 /* InsertSpace */),
-                rule("SpaceAfterFunctionInFuncDecl", 94 /* FunctionKeyword */, anyToken, [isFunctionDeclContext], 4 /* InsertSpace */),
+                rule("NoSpaceBetweenFunctionKeywordAndStar", 97 /* FunctionKeyword */, 41 /* AsteriskToken */, [isFunctionDeclarationOrFunctionExpressionContext], 16 /* DeleteSpace */),
+                rule("SpaceAfterStarInGeneratorDeclaration", 41 /* AsteriskToken */, 78 /* Identifier */, [isFunctionDeclarationOrFunctionExpressionContext], 4 /* InsertSpace */),
+                rule("SpaceAfterFunctionInFuncDecl", 97 /* FunctionKeyword */, anyToken, [isFunctionDeclContext], 4 /* InsertSpace */),
                 // Insert new line after { and before } in multi-line contexts.
                 rule("NewLineAfterOpenBraceInBlockContext", 18 /* OpenBraceToken */, anyToken, [isMultilineBlockContext], 8 /* InsertNewLine */),
                 // For get/set members, we check for (identifier,identifier) since get/set don't have tokens and they are represented as just an identifier token.
                 // Though, we do extra check on the context to make sure we are dealing with get/set node. Example:
                 //      get x() {}
                 //      set x(val) {}
-                rule("SpaceAfterGetSetInMember", [131 /* GetKeyword */, 142 /* SetKeyword */], 75 /* Identifier */, [isFunctionDeclContext], 4 /* InsertSpace */),
-                rule("NoSpaceBetweenYieldKeywordAndStar", 121 /* YieldKeyword */, 41 /* AsteriskToken */, [isNonJsxSameLineTokenContext, isYieldOrYieldStarWithOperand], 16 /* DeleteSpace */),
-                rule("SpaceBetweenYieldOrYieldStarAndOperand", [121 /* YieldKeyword */, 41 /* AsteriskToken */], anyToken, [isNonJsxSameLineTokenContext, isYieldOrYieldStarWithOperand], 4 /* InsertSpace */),
-                rule("NoSpaceBetweenReturnAndSemicolon", 101 /* ReturnKeyword */, 26 /* SemicolonToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
-                rule("SpaceAfterCertainKeywords", [109 /* VarKeyword */, 105 /* ThrowKeyword */, 99 /* NewKeyword */, 85 /* DeleteKeyword */, 101 /* ReturnKeyword */, 108 /* TypeOfKeyword */, 127 /* AwaitKeyword */], anyToken, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
-                rule("SpaceAfterLetConstInVariableDeclaration", [115 /* LetKeyword */, 81 /* ConstKeyword */], anyToken, [isNonJsxSameLineTokenContext, isStartOfVariableDeclarationList], 4 /* InsertSpace */),
+                rule("SpaceAfterGetSetInMember", [134 /* GetKeyword */, 146 /* SetKeyword */], 78 /* Identifier */, [isFunctionDeclContext], 4 /* InsertSpace */),
+                rule("NoSpaceBetweenYieldKeywordAndStar", 124 /* YieldKeyword */, 41 /* AsteriskToken */, [isNonJsxSameLineTokenContext, isYieldOrYieldStarWithOperand], 16 /* DeleteSpace */),
+                rule("SpaceBetweenYieldOrYieldStarAndOperand", [124 /* YieldKeyword */, 41 /* AsteriskToken */], anyToken, [isNonJsxSameLineTokenContext, isYieldOrYieldStarWithOperand], 4 /* InsertSpace */),
+                rule("NoSpaceBetweenReturnAndSemicolon", 104 /* ReturnKeyword */, 26 /* SemicolonToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("SpaceAfterCertainKeywords", [112 /* VarKeyword */, 108 /* ThrowKeyword */, 102 /* NewKeyword */, 88 /* DeleteKeyword */, 104 /* ReturnKeyword */, 111 /* TypeOfKeyword */, 130 /* AwaitKeyword */], anyToken, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceAfterLetConstInVariableDeclaration", [118 /* LetKeyword */, 84 /* ConstKeyword */], anyToken, [isNonJsxSameLineTokenContext, isStartOfVariableDeclarationList], 4 /* InsertSpace */),
                 rule("NoSpaceBeforeOpenParenInFuncCall", anyToken, 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext, isFunctionCallOrNewContext, isPreviousTokenNotComma], 16 /* DeleteSpace */),
                 // Special case for binary operators (that are keywords). For these we have to add a space and shouldn't follow any user options.
                 rule("SpaceBeforeBinaryKeywordOperator", anyToken, binaryKeywordOperators, [isNonJsxSameLineTokenContext, isBinaryOpContext], 4 /* InsertSpace */),
                 rule("SpaceAfterBinaryKeywordOperator", binaryKeywordOperators, anyToken, [isNonJsxSameLineTokenContext, isBinaryOpContext], 4 /* InsertSpace */),
-                rule("SpaceAfterVoidOperator", 110 /* VoidKeyword */, anyToken, [isNonJsxSameLineTokenContext, isVoidOpContext], 4 /* InsertSpace */),
+                rule("SpaceAfterVoidOperator", 113 /* VoidKeyword */, anyToken, [isNonJsxSameLineTokenContext, isVoidOpContext], 4 /* InsertSpace */),
                 // Async-await
-                rule("SpaceBetweenAsyncAndOpenParen", 126 /* AsyncKeyword */, 20 /* OpenParenToken */, [isArrowFunctionContext, isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
-                rule("SpaceBetweenAsyncAndFunctionKeyword", 126 /* AsyncKeyword */, [94 /* FunctionKeyword */, 75 /* Identifier */], [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBetweenAsyncAndOpenParen", 129 /* AsyncKeyword */, 20 /* OpenParenToken */, [isArrowFunctionContext, isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBetweenAsyncAndFunctionKeyword", 129 /* AsyncKeyword */, [97 /* FunctionKeyword */, 78 /* Identifier */], [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 // Template string
-                rule("NoSpaceBetweenTagAndTemplateString", [75 /* Identifier */, 21 /* CloseParenToken */], [14 /* NoSubstitutionTemplateLiteral */, 15 /* TemplateHead */], [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("NoSpaceBetweenTagAndTemplateString", [78 /* Identifier */, 21 /* CloseParenToken */], [14 /* NoSubstitutionTemplateLiteral */, 15 /* TemplateHead */], [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // JSX opening elements
-                rule("SpaceBeforeJsxAttribute", anyToken, 75 /* Identifier */, [isNextTokenParentJsxAttribute, isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBeforeJsxAttribute", anyToken, 78 /* Identifier */, [isNextTokenParentJsxAttribute, isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 rule("SpaceBeforeSlashInJsxOpeningElement", anyToken, 43 /* SlashToken */, [isJsxSelfClosingElementContext, isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 rule("NoSpaceBeforeGreaterThanTokenInJsxOpeningElement", 43 /* SlashToken */, 31 /* GreaterThanToken */, [isJsxSelfClosingElementContext, isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("NoSpaceBeforeEqualInJsxAttribute", anyToken, 62 /* EqualsToken */, [isJsxAttributeContext, isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("NoSpaceAfterEqualInJsxAttribute", 62 /* EqualsToken */, anyToken, [isJsxAttributeContext, isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // TypeScript-specific rules
                 // Use of module as a function call. e.g.: import m2 = module("m2");
-                rule("NoSpaceAfterModuleImport", [135 /* ModuleKeyword */, 139 /* RequireKeyword */], 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("NoSpaceAfterModuleImport", [139 /* ModuleKeyword */, 143 /* RequireKeyword */], 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // Add a space around certain TypeScript keywords
                 rule("SpaceAfterCertainTypeScriptKeywords", [
-                    122 /* AbstractKeyword */,
-                    80 /* ClassKeyword */,
-                    130 /* DeclareKeyword */,
-                    84 /* DefaultKeyword */,
-                    88 /* EnumKeyword */,
-                    89 /* ExportKeyword */,
-                    90 /* ExtendsKeyword */,
-                    131 /* GetKeyword */,
-                    113 /* ImplementsKeyword */,
-                    96 /* ImportKeyword */,
-                    114 /* InterfaceKeyword */,
-                    135 /* ModuleKeyword */,
-                    136 /* NamespaceKeyword */,
-                    117 /* PrivateKeyword */,
-                    119 /* PublicKeyword */,
-                    118 /* ProtectedKeyword */,
-                    138 /* ReadonlyKeyword */,
-                    142 /* SetKeyword */,
-                    120 /* StaticKeyword */,
-                    145 /* TypeKeyword */,
-                    149 /* FromKeyword */,
-                    134 /* KeyOfKeyword */,
-                    132 /* InferKeyword */,
+                    125 /* AbstractKeyword */,
+                    83 /* ClassKeyword */,
+                    133 /* DeclareKeyword */,
+                    87 /* DefaultKeyword */,
+                    91 /* EnumKeyword */,
+                    92 /* ExportKeyword */,
+                    93 /* ExtendsKeyword */,
+                    134 /* GetKeyword */,
+                    116 /* ImplementsKeyword */,
+                    99 /* ImportKeyword */,
+                    117 /* InterfaceKeyword */,
+                    139 /* ModuleKeyword */,
+                    140 /* NamespaceKeyword */,
+                    120 /* PrivateKeyword */,
+                    122 /* PublicKeyword */,
+                    121 /* ProtectedKeyword */,
+                    142 /* ReadonlyKeyword */,
+                    146 /* SetKeyword */,
+                    123 /* StaticKeyword */,
+                    149 /* TypeKeyword */,
+                    153 /* FromKeyword */,
+                    138 /* KeyOfKeyword */,
+                    135 /* InferKeyword */,
                 ], anyToken, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
-                rule("SpaceBeforeCertainTypeScriptKeywords", anyToken, [90 /* ExtendsKeyword */, 113 /* ImplementsKeyword */, 149 /* FromKeyword */], [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBeforeCertainTypeScriptKeywords", anyToken, [93 /* ExtendsKeyword */, 116 /* ImplementsKeyword */, 153 /* FromKeyword */], [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 // Treat string literals in module names as identifiers, and add a space between the literal and the opening Brace braces, e.g.: module "m2" {
                 rule("SpaceAfterModuleName", 10 /* StringLiteral */, 18 /* OpenBraceToken */, [isModuleDeclContext], 4 /* InsertSpace */),
                 // Lambda expressions
                 rule("SpaceBeforeArrow", anyToken, 38 /* EqualsGreaterThanToken */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 rule("SpaceAfterArrow", 38 /* EqualsGreaterThanToken */, anyToken, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 // Optional parameters and let args
-                rule("NoSpaceAfterEllipsis", 25 /* DotDotDotToken */, 75 /* Identifier */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("NoSpaceAfterEllipsis", 25 /* DotDotDotToken */, 78 /* Identifier */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("NoSpaceAfterOptionalParameters", 57 /* QuestionToken */, [21 /* CloseParenToken */, 27 /* CommaToken */], [isNonJsxSameLineTokenContext, isNotBinaryOpContext], 16 /* DeleteSpace */),
                 // Remove spaces in empty interface literals. e.g.: x: {}
                 rule("NoSpaceBetweenEmptyInterfaceBraceBrackets", 18 /* OpenBraceToken */, 19 /* CloseBraceToken */, [isNonJsxSameLineTokenContext, isObjectTypeContext], 16 /* DeleteSpace */),
@@ -124210,38 +130393,38 @@ var ts;
                 rule("NoSpaceBeforeCloseAngularBracket", anyToken, 31 /* GreaterThanToken */, [isNonJsxSameLineTokenContext, isTypeArgumentOrParameterOrAssertionContext], 16 /* DeleteSpace */),
                 rule("NoSpaceAfterCloseAngularBracket", 31 /* GreaterThanToken */, [20 /* OpenParenToken */, 22 /* OpenBracketToken */, 31 /* GreaterThanToken */, 27 /* CommaToken */], [isNonJsxSameLineTokenContext, isTypeArgumentOrParameterOrAssertionContext, isNotFunctionDeclContext /*To prevent an interference with the SpaceBeforeOpenParenInFuncDecl rule*/], 16 /* DeleteSpace */),
                 // decorators
-                rule("SpaceBeforeAt", [21 /* CloseParenToken */, 75 /* Identifier */], 59 /* AtToken */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBeforeAt", [21 /* CloseParenToken */, 78 /* Identifier */], 59 /* AtToken */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 rule("NoSpaceAfterAt", 59 /* AtToken */, anyToken, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // Insert space after @ in decorator
                 rule("SpaceAfterDecorator", anyToken, [
-                    122 /* AbstractKeyword */,
-                    75 /* Identifier */,
-                    89 /* ExportKeyword */,
-                    84 /* DefaultKeyword */,
-                    80 /* ClassKeyword */,
-                    120 /* StaticKeyword */,
-                    119 /* PublicKeyword */,
-                    117 /* PrivateKeyword */,
-                    118 /* ProtectedKeyword */,
-                    131 /* GetKeyword */,
-                    142 /* SetKeyword */,
+                    125 /* AbstractKeyword */,
+                    78 /* Identifier */,
+                    92 /* ExportKeyword */,
+                    87 /* DefaultKeyword */,
+                    83 /* ClassKeyword */,
+                    123 /* StaticKeyword */,
+                    122 /* PublicKeyword */,
+                    120 /* PrivateKeyword */,
+                    121 /* ProtectedKeyword */,
+                    134 /* GetKeyword */,
+                    146 /* SetKeyword */,
                     22 /* OpenBracketToken */,
                     41 /* AsteriskToken */,
                 ], [isEndOfDecoratorContextOnSameLine], 4 /* InsertSpace */),
                 rule("NoSpaceBeforeNonNullAssertionOperator", anyToken, 53 /* ExclamationToken */, [isNonJsxSameLineTokenContext, isNonNullAssertionContext], 16 /* DeleteSpace */),
-                rule("NoSpaceAfterNewKeywordOnConstructorSignature", 99 /* NewKeyword */, 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext, isConstructorSignatureContext], 16 /* DeleteSpace */),
+                rule("NoSpaceAfterNewKeywordOnConstructorSignature", 102 /* NewKeyword */, 20 /* OpenParenToken */, [isNonJsxSameLineTokenContext, isConstructorSignatureContext], 16 /* DeleteSpace */),
                 rule("SpaceLessThanAndNonJSXTypeAnnotation", 29 /* LessThanToken */, 29 /* LessThanToken */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
             ];
             // These rules are applied after high priority
             var userConfigurableRules = [
                 // Treat constructor as an identifier in a function declaration, and remove spaces between constructor and following left parentheses
-                rule("SpaceAfterConstructor", 129 /* ConstructorKeyword */, 20 /* OpenParenToken */, [isOptionEnabled("insertSpaceAfterConstructor"), isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
-                rule("NoSpaceAfterConstructor", 129 /* ConstructorKeyword */, 20 /* OpenParenToken */, [isOptionDisabledOrUndefined("insertSpaceAfterConstructor"), isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("SpaceAfterConstructor", 132 /* ConstructorKeyword */, 20 /* OpenParenToken */, [isOptionEnabled("insertSpaceAfterConstructor"), isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("NoSpaceAfterConstructor", 132 /* ConstructorKeyword */, 20 /* OpenParenToken */, [isOptionDisabledOrUndefined("insertSpaceAfterConstructor"), isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("SpaceAfterComma", 27 /* CommaToken */, anyToken, [isOptionEnabled("insertSpaceAfterCommaDelimiter"), isNonJsxSameLineTokenContext, isNonJsxElementOrFragmentContext, isNextTokenNotCloseBracket, isNextTokenNotCloseParen], 4 /* InsertSpace */),
                 rule("NoSpaceAfterComma", 27 /* CommaToken */, anyToken, [isOptionDisabledOrUndefined("insertSpaceAfterCommaDelimiter"), isNonJsxSameLineTokenContext, isNonJsxElementOrFragmentContext], 16 /* DeleteSpace */),
                 // Insert space after function keyword for anonymous functions
-                rule("SpaceAfterAnonymousFunctionKeyword", [94 /* FunctionKeyword */, 41 /* AsteriskToken */], 20 /* OpenParenToken */, [isOptionEnabled("insertSpaceAfterFunctionKeywordForAnonymousFunctions"), isFunctionDeclContext], 4 /* InsertSpace */),
-                rule("NoSpaceAfterAnonymousFunctionKeyword", [94 /* FunctionKeyword */, 41 /* AsteriskToken */], 20 /* OpenParenToken */, [isOptionDisabledOrUndefined("insertSpaceAfterFunctionKeywordForAnonymousFunctions"), isFunctionDeclContext], 16 /* DeleteSpace */),
+                rule("SpaceAfterAnonymousFunctionKeyword", [97 /* FunctionKeyword */, 41 /* AsteriskToken */], 20 /* OpenParenToken */, [isOptionEnabled("insertSpaceAfterFunctionKeywordForAnonymousFunctions"), isFunctionDeclContext], 4 /* InsertSpace */),
+                rule("NoSpaceAfterAnonymousFunctionKeyword", [97 /* FunctionKeyword */, 41 /* AsteriskToken */], 20 /* OpenParenToken */, [isOptionDisabledOrUndefined("insertSpaceAfterFunctionKeywordForAnonymousFunctions"), isFunctionDeclContext], 16 /* DeleteSpace */),
                 // Insert space after keywords in control flow statements
                 rule("SpaceAfterKeywordInControl", keywords, 20 /* OpenParenToken */, [isOptionEnabled("insertSpaceAfterKeywordsInControlFlowStatements"), isControlDeclContext], 4 /* InsertSpace */),
                 rule("NoSpaceAfterKeywordInControl", keywords, 20 /* OpenParenToken */, [isOptionDisabledOrUndefined("insertSpaceAfterKeywordsInControlFlowStatements"), isControlDeclContext], 16 /* DeleteSpace */),
@@ -124268,9 +130451,9 @@ var ts;
                 rule("SpaceBetweenEmptyBraceBrackets", 18 /* OpenBraceToken */, 19 /* CloseBraceToken */, [isOptionEnabled("insertSpaceAfterOpeningAndBeforeClosingEmptyBraces")], 4 /* InsertSpace */),
                 rule("NoSpaceBetweenEmptyBraceBrackets", 18 /* OpenBraceToken */, 19 /* CloseBraceToken */, [isOptionDisabled("insertSpaceAfterOpeningAndBeforeClosingEmptyBraces"), isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // Insert space after opening and before closing template string braces
-                rule("SpaceAfterTemplateHeadAndMiddle", [15 /* TemplateHead */, 16 /* TemplateMiddle */], anyToken, [isOptionEnabled("insertSpaceAfterOpeningAndBeforeClosingTemplateStringBraces"), isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceAfterTemplateHeadAndMiddle", [15 /* TemplateHead */, 16 /* TemplateMiddle */], anyToken, [isOptionEnabled("insertSpaceAfterOpeningAndBeforeClosingTemplateStringBraces"), isNonJsxTextContext], 4 /* InsertSpace */, 1 /* CanDeleteNewLines */),
                 rule("SpaceBeforeTemplateMiddleAndTail", anyToken, [16 /* TemplateMiddle */, 17 /* TemplateTail */], [isOptionEnabled("insertSpaceAfterOpeningAndBeforeClosingTemplateStringBraces"), isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
-                rule("NoSpaceAfterTemplateHeadAndMiddle", [15 /* TemplateHead */, 16 /* TemplateMiddle */], anyToken, [isOptionDisabledOrUndefined("insertSpaceAfterOpeningAndBeforeClosingTemplateStringBraces"), isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("NoSpaceAfterTemplateHeadAndMiddle", [15 /* TemplateHead */, 16 /* TemplateMiddle */], anyToken, [isOptionDisabledOrUndefined("insertSpaceAfterOpeningAndBeforeClosingTemplateStringBraces"), isNonJsxTextContext], 16 /* DeleteSpace */, 1 /* CanDeleteNewLines */),
                 rule("NoSpaceBeforeTemplateMiddleAndTail", anyToken, [16 /* TemplateMiddle */, 17 /* TemplateTail */], [isOptionDisabledOrUndefined("insertSpaceAfterOpeningAndBeforeClosingTemplateStringBraces"), isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // No space after { and before } in JSX expression
                 rule("SpaceAfterOpenBraceInJsxExpression", 18 /* OpenBraceToken */, anyToken, [isOptionEnabled("insertSpaceAfterOpeningAndBeforeClosingJsxExpressionBraces"), isNonJsxSameLineTokenContext, isJsxExpressionContext], 4 /* InsertSpace */),
@@ -124296,8 +130479,8 @@ var ts;
                 rule("NewLineBeforeOpenBraceInTypeScriptDeclWithBlock", typeScriptOpenBraceLeftTokenRange, 18 /* OpenBraceToken */, [isOptionEnabled("placeOpenBraceOnNewLineForFunctions"), isTypeScriptDeclWithBlockContext, isBeforeMultilineBlockContext], 8 /* InsertNewLine */, 1 /* CanDeleteNewLines */),
                 rule("SpaceAfterTypeAssertion", 31 /* GreaterThanToken */, anyToken, [isOptionEnabled("insertSpaceAfterTypeAssertion"), isNonJsxSameLineTokenContext, isTypeAssertionContext], 4 /* InsertSpace */),
                 rule("NoSpaceAfterTypeAssertion", 31 /* GreaterThanToken */, anyToken, [isOptionDisabledOrUndefined("insertSpaceAfterTypeAssertion"), isNonJsxSameLineTokenContext, isTypeAssertionContext], 16 /* DeleteSpace */),
-                rule("SpaceBeforeTypeAnnotation", anyToken, 58 /* ColonToken */, [isOptionEnabled("insertSpaceBeforeTypeAnnotation"), isNonJsxSameLineTokenContext, isTypeAnnotationContext], 4 /* InsertSpace */),
-                rule("NoSpaceBeforeTypeAnnotation", anyToken, 58 /* ColonToken */, [isOptionDisabledOrUndefined("insertSpaceBeforeTypeAnnotation"), isNonJsxSameLineTokenContext, isTypeAnnotationContext], 16 /* DeleteSpace */),
+                rule("SpaceBeforeTypeAnnotation", anyToken, [57 /* QuestionToken */, 58 /* ColonToken */], [isOptionEnabled("insertSpaceBeforeTypeAnnotation"), isNonJsxSameLineTokenContext, isTypeAnnotationContext], 4 /* InsertSpace */),
+                rule("NoSpaceBeforeTypeAnnotation", anyToken, [57 /* QuestionToken */, 58 /* ColonToken */], [isOptionDisabledOrUndefined("insertSpaceBeforeTypeAnnotation"), isNonJsxSameLineTokenContext, isTypeAnnotationContext], 16 /* DeleteSpace */),
                 rule("NoOptionalSemicolon", 26 /* SemicolonToken */, anyTokenIncludingEOF, [optionEquals("semicolons", ts.SemicolonPreference.Remove), isSemicolonDeletionContext], 32 /* DeleteToken */),
                 rule("OptionalSemicolon", anyToken, anyTokenIncludingEOF, [optionEquals("semicolons", ts.SemicolonPreference.Insert), isSemicolonInsertionContext], 64 /* InsertTrailingSemicolon */),
             ];
@@ -124310,16 +130493,16 @@ var ts;
                 rule("SpaceBeforeOpenBraceInTypeScriptDeclWithBlock", typeScriptOpenBraceLeftTokenRange, 18 /* OpenBraceToken */, [isOptionDisabledOrUndefinedOrTokensOnSameLine("placeOpenBraceOnNewLineForFunctions"), isTypeScriptDeclWithBlockContext, isNotFormatOnEnter, isSameLineTokenOrBeforeBlockContext], 4 /* InsertSpace */, 1 /* CanDeleteNewLines */),
                 rule("NoSpaceBeforeComma", anyToken, 27 /* CommaToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 // No space before and after indexer `x[]`
-                rule("NoSpaceBeforeOpenBracket", anyTokenExcept(126 /* AsyncKeyword */, 78 /* CaseKeyword */), 22 /* OpenBracketToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
+                rule("NoSpaceBeforeOpenBracket", anyTokenExcept(129 /* AsyncKeyword */, 81 /* CaseKeyword */), 22 /* OpenBracketToken */, [isNonJsxSameLineTokenContext], 16 /* DeleteSpace */),
                 rule("NoSpaceAfterCloseBracket", 23 /* CloseBracketToken */, anyToken, [isNonJsxSameLineTokenContext, isNotBeforeBlockInFunctionDeclarationContext], 16 /* DeleteSpace */),
                 rule("SpaceAfterSemicolon", 26 /* SemicolonToken */, anyToken, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 // Remove extra space between for and await
-                rule("SpaceBetweenForAndAwaitKeyword", 93 /* ForKeyword */, 127 /* AwaitKeyword */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBetweenForAndAwaitKeyword", 96 /* ForKeyword */, 130 /* AwaitKeyword */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
                 // Add a space between statements. All keywords except (do,else,case) has open/close parens after them.
                 // So, we have a rule to add a space for [),Any], [do,Any], [else,Any], and [case,Any]
-                rule("SpaceBetweenStatements", [21 /* CloseParenToken */, 86 /* DoKeyword */, 87 /* ElseKeyword */, 78 /* CaseKeyword */], anyToken, [isNonJsxSameLineTokenContext, isNonJsxElementOrFragmentContext, isNotForContext], 4 /* InsertSpace */),
-                // This low-pri rule takes care of "try {" and "finally {" in case the rule SpaceBeforeOpenBraceInControl didn't execute on FormatOnEnter.
-                rule("SpaceAfterTryFinally", [107 /* TryKeyword */, 92 /* FinallyKeyword */], 18 /* OpenBraceToken */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
+                rule("SpaceBetweenStatements", [21 /* CloseParenToken */, 89 /* DoKeyword */, 90 /* ElseKeyword */, 81 /* CaseKeyword */], anyToken, [isNonJsxSameLineTokenContext, isNonJsxElementOrFragmentContext, isNotForContext], 4 /* InsertSpace */),
+                // This low-pri rule takes care of "try {", "catch {" and "finally {" in case the rule SpaceBeforeOpenBraceInControl didn't execute on FormatOnEnter.
+                rule("SpaceAfterTryCatchFinally", [110 /* TryKeyword */, 82 /* CatchKeyword */, 95 /* FinallyKeyword */], 18 /* OpenBraceToken */, [isNonJsxSameLineTokenContext], 4 /* InsertSpace */),
             ];
             return __spreadArrays(highPriorityCommonRules, userConfigurableRules, lowPriorityCommonRules);
         }
@@ -124378,51 +130561,51 @@ var ts;
             return function (context) { return !context.options || !context.options.hasOwnProperty(optionName) || !!context.options[optionName]; };
         }
         function isForContext(context) {
-            return context.contextNode.kind === 230 /* ForStatement */;
+            return context.contextNode.kind === 237 /* ForStatement */;
         }
         function isNotForContext(context) {
             return !isForContext(context);
         }
         function isBinaryOpContext(context) {
             switch (context.contextNode.kind) {
-                case 209 /* BinaryExpression */:
+                case 216 /* BinaryExpression */:
                     return context.contextNode.operatorToken.kind !== 27 /* CommaToken */;
-                case 210 /* ConditionalExpression */:
-                case 180 /* ConditionalType */:
-                case 217 /* AsExpression */:
-                case 263 /* ExportSpecifier */:
-                case 258 /* ImportSpecifier */:
-                case 168 /* TypePredicate */:
-                case 178 /* UnionType */:
-                case 179 /* IntersectionType */:
+                case 217 /* ConditionalExpression */:
+                case 184 /* ConditionalType */:
+                case 224 /* AsExpression */:
+                case 270 /* ExportSpecifier */:
+                case 265 /* ImportSpecifier */:
+                case 172 /* TypePredicate */:
+                case 182 /* UnionType */:
+                case 183 /* IntersectionType */:
                     return true;
                 // equals in binding elements: function foo([[x, y] = [1, 2]])
-                case 191 /* BindingElement */:
+                case 198 /* BindingElement */:
                 // equals in type X = ...
                 // falls through
-                case 247 /* TypeAliasDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
                 // equal in import a = module('a');
                 // falls through
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                 // equal in let a = 0
                 // falls through
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                 // equal in p = 0
                 // falls through
-                case 156 /* Parameter */:
-                case 284 /* EnumMember */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
+                case 160 /* Parameter */:
+                case 291 /* EnumMember */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
                     return context.currentTokenSpan.kind === 62 /* EqualsToken */ || context.nextTokenSpan.kind === 62 /* EqualsToken */;
                 // "in" keyword in for (let x in []) { }
-                case 231 /* ForInStatement */:
+                case 238 /* ForInStatement */:
                 // "in" keyword in [P in keyof T]: T[P]
                 // falls through
-                case 155 /* TypeParameter */:
-                    return context.currentTokenSpan.kind === 97 /* InKeyword */ || context.nextTokenSpan.kind === 97 /* InKeyword */ || context.currentTokenSpan.kind === 62 /* EqualsToken */ || context.nextTokenSpan.kind === 62 /* EqualsToken */;
+                case 159 /* TypeParameter */:
+                    return context.currentTokenSpan.kind === 100 /* InKeyword */ || context.nextTokenSpan.kind === 100 /* InKeyword */ || context.currentTokenSpan.kind === 62 /* EqualsToken */ || context.nextTokenSpan.kind === 62 /* EqualsToken */;
                 // Technically, "of" is not a binary operator, but format it the same way as "in"
-                case 232 /* ForOfStatement */:
-                    return context.currentTokenSpan.kind === 152 /* OfKeyword */ || context.nextTokenSpan.kind === 152 /* OfKeyword */;
+                case 239 /* ForOfStatement */:
+                    return context.currentTokenSpan.kind === 156 /* OfKeyword */ || context.nextTokenSpan.kind === 156 /* OfKeyword */;
             }
             return false;
         }
@@ -124434,22 +130617,22 @@ var ts;
         }
         function isTypeAnnotationContext(context) {
             var contextKind = context.contextNode.kind;
-            return contextKind === 159 /* PropertyDeclaration */ ||
-                contextKind === 158 /* PropertySignature */ ||
-                contextKind === 156 /* Parameter */ ||
-                contextKind === 242 /* VariableDeclaration */ ||
+            return contextKind === 163 /* PropertyDeclaration */ ||
+                contextKind === 162 /* PropertySignature */ ||
+                contextKind === 160 /* Parameter */ ||
+                contextKind === 249 /* VariableDeclaration */ ||
                 ts.isFunctionLikeKind(contextKind);
         }
         function isConditionalOperatorContext(context) {
-            return context.contextNode.kind === 210 /* ConditionalExpression */ ||
-                context.contextNode.kind === 180 /* ConditionalType */;
+            return context.contextNode.kind === 217 /* ConditionalExpression */ ||
+                context.contextNode.kind === 184 /* ConditionalType */;
         }
         function isSameLineTokenOrBeforeBlockContext(context) {
             return context.TokensAreOnSameLine() || isBeforeBlockContext(context);
         }
         function isBraceWrappedContext(context) {
-            return context.contextNode.kind === 189 /* ObjectBindingPattern */ ||
-                context.contextNode.kind === 186 /* MappedType */ ||
+            return context.contextNode.kind === 196 /* ObjectBindingPattern */ ||
+                context.contextNode.kind === 190 /* MappedType */ ||
                 isSingleLineBlockContext(context);
         }
         // This check is done before an open brace in a control construct, a function, or a typescript block declaration
@@ -124475,34 +130658,34 @@ var ts;
                 return true;
             }
             switch (node.kind) {
-                case 223 /* Block */:
-                case 251 /* CaseBlock */:
-                case 193 /* ObjectLiteralExpression */:
-                case 250 /* ModuleBlock */:
+                case 230 /* Block */:
+                case 258 /* CaseBlock */:
+                case 200 /* ObjectLiteralExpression */:
+                case 257 /* ModuleBlock */:
                     return true;
             }
             return false;
         }
         function isFunctionDeclContext(context) {
             switch (context.contextNode.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
+                case 251 /* FunctionDeclaration */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
                 // case SyntaxKind.MemberFunctionDeclaration:
                 // falls through
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */:
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */:
                 // case SyntaxKind.MethodSignature:
                 // falls through
-                case 165 /* CallSignature */:
-                case 201 /* FunctionExpression */:
-                case 162 /* Constructor */:
-                case 202 /* ArrowFunction */:
+                case 169 /* CallSignature */:
+                case 208 /* FunctionExpression */:
+                case 166 /* Constructor */:
+                case 209 /* ArrowFunction */:
                 // case SyntaxKind.ConstructorDeclaration:
                 // case SyntaxKind.SimpleArrowFunctionExpression:
                 // case SyntaxKind.ParenthesizedArrowFunctionExpression:
                 // falls through
-                case 246 /* InterfaceDeclaration */: // This one is not truly a function, but for formatting purposes, it acts just like one
+                case 253 /* InterfaceDeclaration */: // This one is not truly a function, but for formatting purposes, it acts just like one
                     return true;
             }
             return false;
@@ -124511,40 +130694,40 @@ var ts;
             return !isFunctionDeclContext(context);
         }
         function isFunctionDeclarationOrFunctionExpressionContext(context) {
-            return context.contextNode.kind === 244 /* FunctionDeclaration */ || context.contextNode.kind === 201 /* FunctionExpression */;
+            return context.contextNode.kind === 251 /* FunctionDeclaration */ || context.contextNode.kind === 208 /* FunctionExpression */;
         }
         function isTypeScriptDeclWithBlockContext(context) {
             return nodeIsTypeScriptDeclWithBlockContext(context.contextNode);
         }
         function nodeIsTypeScriptDeclWithBlockContext(node) {
             switch (node.kind) {
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 246 /* InterfaceDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 173 /* TypeLiteral */:
-                case 249 /* ModuleDeclaration */:
-                case 260 /* ExportDeclaration */:
-                case 261 /* NamedExports */:
-                case 254 /* ImportDeclaration */:
-                case 257 /* NamedImports */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 253 /* InterfaceDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 177 /* TypeLiteral */:
+                case 256 /* ModuleDeclaration */:
+                case 267 /* ExportDeclaration */:
+                case 268 /* NamedExports */:
+                case 261 /* ImportDeclaration */:
+                case 264 /* NamedImports */:
                     return true;
             }
             return false;
         }
         function isAfterCodeBlockContext(context) {
             switch (context.currentTokenParent.kind) {
-                case 245 /* ClassDeclaration */:
-                case 249 /* ModuleDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 280 /* CatchClause */:
-                case 250 /* ModuleBlock */:
-                case 237 /* SwitchStatement */:
+                case 252 /* ClassDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 287 /* CatchClause */:
+                case 257 /* ModuleBlock */:
+                case 244 /* SwitchStatement */:
                     return true;
-                case 223 /* Block */: {
+                case 230 /* Block */: {
                     var blockParent = context.currentTokenParent.parent;
                     // In a codefix scenario, we can't rely on parents being set. So just always return true.
-                    if (!blockParent || blockParent.kind !== 202 /* ArrowFunction */ && blockParent.kind !== 201 /* FunctionExpression */) {
+                    if (!blockParent || blockParent.kind !== 209 /* ArrowFunction */ && blockParent.kind !== 208 /* FunctionExpression */) {
                         return true;
                     }
                 }
@@ -124553,32 +130736,32 @@ var ts;
         }
         function isControlDeclContext(context) {
             switch (context.contextNode.kind) {
-                case 227 /* IfStatement */:
-                case 237 /* SwitchStatement */:
-                case 230 /* ForStatement */:
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
-                case 229 /* WhileStatement */:
-                case 240 /* TryStatement */:
-                case 228 /* DoStatement */:
-                case 236 /* WithStatement */:
+                case 234 /* IfStatement */:
+                case 244 /* SwitchStatement */:
+                case 237 /* ForStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                case 236 /* WhileStatement */:
+                case 247 /* TryStatement */:
+                case 235 /* DoStatement */:
+                case 243 /* WithStatement */:
                 // TODO
                 // case SyntaxKind.ElseClause:
                 // falls through
-                case 280 /* CatchClause */:
+                case 287 /* CatchClause */:
                     return true;
                 default:
                     return false;
             }
         }
         function isObjectContext(context) {
-            return context.contextNode.kind === 193 /* ObjectLiteralExpression */;
+            return context.contextNode.kind === 200 /* ObjectLiteralExpression */;
         }
         function isFunctionCallContext(context) {
-            return context.contextNode.kind === 196 /* CallExpression */;
+            return context.contextNode.kind === 203 /* CallExpression */;
         }
         function isNewContext(context) {
-            return context.contextNode.kind === 197 /* NewExpression */;
+            return context.contextNode.kind === 204 /* NewExpression */;
         }
         function isFunctionCallOrNewContext(context) {
             return isFunctionCallContext(context) || isNewContext(context);
@@ -124593,28 +130776,31 @@ var ts;
             return context.nextTokenSpan.kind !== 21 /* CloseParenToken */;
         }
         function isArrowFunctionContext(context) {
-            return context.contextNode.kind === 202 /* ArrowFunction */;
+            return context.contextNode.kind === 209 /* ArrowFunction */;
         }
         function isImportTypeContext(context) {
-            return context.contextNode.kind === 188 /* ImportType */;
+            return context.contextNode.kind === 195 /* ImportType */;
         }
         function isNonJsxSameLineTokenContext(context) {
             return context.TokensAreOnSameLine() && context.contextNode.kind !== 11 /* JsxText */;
         }
+        function isNonJsxTextContext(context) {
+            return context.contextNode.kind !== 11 /* JsxText */;
+        }
         function isNonJsxElementOrFragmentContext(context) {
-            return context.contextNode.kind !== 266 /* JsxElement */ && context.contextNode.kind !== 270 /* JsxFragment */;
+            return context.contextNode.kind !== 273 /* JsxElement */ && context.contextNode.kind !== 277 /* JsxFragment */;
         }
         function isJsxExpressionContext(context) {
-            return context.contextNode.kind === 276 /* JsxExpression */ || context.contextNode.kind === 275 /* JsxSpreadAttribute */;
+            return context.contextNode.kind === 283 /* JsxExpression */ || context.contextNode.kind === 282 /* JsxSpreadAttribute */;
         }
         function isNextTokenParentJsxAttribute(context) {
-            return context.nextTokenParent.kind === 273 /* JsxAttribute */;
+            return context.nextTokenParent.kind === 280 /* JsxAttribute */;
         }
         function isJsxAttributeContext(context) {
-            return context.contextNode.kind === 273 /* JsxAttribute */;
+            return context.contextNode.kind === 280 /* JsxAttribute */;
         }
         function isJsxSelfClosingElementContext(context) {
-            return context.contextNode.kind === 267 /* JsxSelfClosingElement */;
+            return context.contextNode.kind === 274 /* JsxSelfClosingElement */;
         }
         function isNotBeforeBlockInFunctionDeclarationContext(context) {
             return !isFunctionDeclContext(context) && !isBeforeBlockContext(context);
@@ -124629,45 +130815,45 @@ var ts;
             while (ts.isExpressionNode(node)) {
                 node = node.parent;
             }
-            return node.kind === 157 /* Decorator */;
+            return node.kind === 161 /* Decorator */;
         }
         function isStartOfVariableDeclarationList(context) {
-            return context.currentTokenParent.kind === 243 /* VariableDeclarationList */ &&
+            return context.currentTokenParent.kind === 250 /* VariableDeclarationList */ &&
                 context.currentTokenParent.getStart(context.sourceFile) === context.currentTokenSpan.pos;
         }
         function isNotFormatOnEnter(context) {
             return context.formattingRequestKind !== 2 /* FormatOnEnter */;
         }
         function isModuleDeclContext(context) {
-            return context.contextNode.kind === 249 /* ModuleDeclaration */;
+            return context.contextNode.kind === 256 /* ModuleDeclaration */;
         }
         function isObjectTypeContext(context) {
-            return context.contextNode.kind === 173 /* TypeLiteral */; // && context.contextNode.parent.kind !== SyntaxKind.InterfaceDeclaration;
+            return context.contextNode.kind === 177 /* TypeLiteral */; // && context.contextNode.parent.kind !== SyntaxKind.InterfaceDeclaration;
         }
         function isConstructorSignatureContext(context) {
-            return context.contextNode.kind === 166 /* ConstructSignature */;
+            return context.contextNode.kind === 170 /* ConstructSignature */;
         }
         function isTypeArgumentOrParameterOrAssertion(token, parent) {
             if (token.kind !== 29 /* LessThanToken */ && token.kind !== 31 /* GreaterThanToken */) {
                 return false;
             }
             switch (parent.kind) {
-                case 169 /* TypeReference */:
-                case 199 /* TypeAssertionExpression */:
-                case 247 /* TypeAliasDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 214 /* ClassExpression */:
-                case 246 /* InterfaceDeclaration */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 196 /* CallExpression */:
-                case 197 /* NewExpression */:
-                case 216 /* ExpressionWithTypeArguments */:
+                case 173 /* TypeReference */:
+                case 206 /* TypeAssertionExpression */:
+                case 254 /* TypeAliasDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 221 /* ClassExpression */:
+                case 253 /* InterfaceDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 203 /* CallExpression */:
+                case 204 /* NewExpression */:
+                case 223 /* ExpressionWithTypeArguments */:
                     return true;
                 default:
                     return false;
@@ -124678,28 +130864,28 @@ var ts;
                 isTypeArgumentOrParameterOrAssertion(context.nextTokenSpan, context.nextTokenParent);
         }
         function isTypeAssertionContext(context) {
-            return context.contextNode.kind === 199 /* TypeAssertionExpression */;
+            return context.contextNode.kind === 206 /* TypeAssertionExpression */;
         }
         function isVoidOpContext(context) {
-            return context.currentTokenSpan.kind === 110 /* VoidKeyword */ && context.currentTokenParent.kind === 205 /* VoidExpression */;
+            return context.currentTokenSpan.kind === 113 /* VoidKeyword */ && context.currentTokenParent.kind === 212 /* VoidExpression */;
         }
         function isYieldOrYieldStarWithOperand(context) {
-            return context.contextNode.kind === 212 /* YieldExpression */ && context.contextNode.expression !== undefined;
+            return context.contextNode.kind === 219 /* YieldExpression */ && context.contextNode.expression !== undefined;
         }
         function isNonNullAssertionContext(context) {
-            return context.contextNode.kind === 218 /* NonNullExpression */;
+            return context.contextNode.kind === 225 /* NonNullExpression */;
         }
         function isNotStatementConditionContext(context) {
             return !isStatementConditionContext(context);
         }
         function isStatementConditionContext(context) {
             switch (context.contextNode.kind) {
-                case 227 /* IfStatement */:
-                case 230 /* ForStatement */:
-                case 231 /* ForInStatement */:
-                case 232 /* ForOfStatement */:
-                case 228 /* DoStatement */:
-                case 229 /* WhileStatement */:
+                case 234 /* IfStatement */:
+                case 237 /* ForStatement */:
+                case 238 /* ForInStatement */:
+                case 239 /* ForOfStatement */:
+                case 235 /* DoStatement */:
+                case 236 /* WhileStatement */:
                     return true;
                 default:
                     return false;
@@ -124724,12 +130910,12 @@ var ts;
                 return nextTokenKind === 19 /* CloseBraceToken */
                     || nextTokenKind === 1 /* EndOfFileToken */;
             }
-            if (nextTokenKind === 222 /* SemicolonClassElement */ ||
+            if (nextTokenKind === 229 /* SemicolonClassElement */ ||
                 nextTokenKind === 26 /* SemicolonToken */) {
                 return false;
             }
-            if (context.contextNode.kind === 246 /* InterfaceDeclaration */ ||
-                context.contextNode.kind === 247 /* TypeAliasDeclaration */) {
+            if (context.contextNode.kind === 253 /* InterfaceDeclaration */ ||
+                context.contextNode.kind === 254 /* TypeAliasDeclaration */) {
                 // Can’t remove semicolon after `foo`; it would parse as a method declaration:
                 //
                 // interface I {
@@ -124743,9 +130929,9 @@ var ts;
             if (ts.isPropertyDeclaration(context.currentTokenParent)) {
                 return !context.currentTokenParent.initializer;
             }
-            return context.currentTokenParent.kind !== 230 /* ForStatement */
-                && context.currentTokenParent.kind !== 224 /* EmptyStatement */
-                && context.currentTokenParent.kind !== 222 /* SemicolonClassElement */
+            return context.currentTokenParent.kind !== 237 /* ForStatement */
+                && context.currentTokenParent.kind !== 231 /* EmptyStatement */
+                && context.currentTokenParent.kind !== 229 /* SemicolonClassElement */
                 && nextTokenKind !== 22 /* OpenBracketToken */
                 && nextTokenKind !== 20 /* OpenParenToken */
                 && nextTokenKind !== 39 /* PlusToken */
@@ -124753,7 +130939,7 @@ var ts;
                 && nextTokenKind !== 43 /* SlashToken */
                 && nextTokenKind !== 13 /* RegularExpressionLiteral */
                 && nextTokenKind !== 27 /* CommaToken */
-                && nextTokenKind !== 211 /* TemplateExpression */
+                && nextTokenKind !== 218 /* TemplateExpression */
                 && nextTokenKind !== 15 /* TemplateHead */
                 && nextTokenKind !== 14 /* NoSubstitutionTemplateLiteral */
                 && nextTokenKind !== 24 /* DotToken */;
@@ -124844,12 +131030,12 @@ var ts;
             return map;
         }
         function getRuleBucketIndex(row, column) {
-            ts.Debug.assert(row <= 152 /* LastKeyword */ && column <= 152 /* LastKeyword */, "Must compute formatting context from tokens");
+            ts.Debug.assert(row <= 156 /* LastKeyword */ && column <= 156 /* LastKeyword */, "Must compute formatting context from tokens");
             return (row * mapRowLength) + column;
         }
         var maskBitSize = 5;
         var mask = 31; // MaskBitSize bits
-        var mapRowLength = 152 /* LastToken */ + 1;
+        var mapRowLength = 156 /* LastToken */ + 1;
         var RulesPosition;
         (function (RulesPosition) {
             RulesPosition[RulesPosition["StopRulesSpecific"] = 0] = "StopRulesSpecific";
@@ -125037,17 +131223,17 @@ var ts;
         // i.e. parent is class declaration with the list of members and node is one of members.
         function isListElement(parent, node) {
             switch (parent.kind) {
-                case 245 /* ClassDeclaration */:
-                case 246 /* InterfaceDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 253 /* InterfaceDeclaration */:
                     return ts.rangeContainsRange(parent.members, node);
-                case 249 /* ModuleDeclaration */:
+                case 256 /* ModuleDeclaration */:
                     var body = parent.body;
-                    return !!body && body.kind === 250 /* ModuleBlock */ && ts.rangeContainsRange(body.statements, node);
-                case 290 /* SourceFile */:
-                case 223 /* Block */:
-                case 250 /* ModuleBlock */:
+                    return !!body && body.kind === 257 /* ModuleBlock */ && ts.rangeContainsRange(body.statements, node);
+                case 297 /* SourceFile */:
+                case 230 /* Block */:
+                case 257 /* ModuleBlock */:
                     return ts.rangeContainsRange(parent.statements, node);
-                case 280 /* CatchClause */:
+                case 287 /* CatchClause */:
                     return ts.rangeContainsRange(parent.block.statements, node);
             }
             return false;
@@ -125274,19 +131460,19 @@ var ts;
                     return node.modifiers[0].kind;
                 }
                 switch (node.kind) {
-                    case 245 /* ClassDeclaration */: return 80 /* ClassKeyword */;
-                    case 246 /* InterfaceDeclaration */: return 114 /* InterfaceKeyword */;
-                    case 244 /* FunctionDeclaration */: return 94 /* FunctionKeyword */;
-                    case 248 /* EnumDeclaration */: return 248 /* EnumDeclaration */;
-                    case 163 /* GetAccessor */: return 131 /* GetKeyword */;
-                    case 164 /* SetAccessor */: return 142 /* SetKeyword */;
-                    case 161 /* MethodDeclaration */:
+                    case 252 /* ClassDeclaration */: return 83 /* ClassKeyword */;
+                    case 253 /* InterfaceDeclaration */: return 117 /* InterfaceKeyword */;
+                    case 251 /* FunctionDeclaration */: return 97 /* FunctionKeyword */;
+                    case 255 /* EnumDeclaration */: return 255 /* EnumDeclaration */;
+                    case 167 /* GetAccessor */: return 134 /* GetKeyword */;
+                    case 168 /* SetAccessor */: return 146 /* SetKeyword */;
+                    case 165 /* MethodDeclaration */:
                         if (node.asteriskToken) {
                             return 41 /* AsteriskToken */;
                         }
                     // falls through
-                    case 159 /* PropertyDeclaration */:
-                    case 156 /* Parameter */:
+                    case 163 /* PropertyDeclaration */:
+                    case 160 /* Parameter */:
                         var name = ts.getNameOfDeclaration(node);
                         if (name) {
                             return name.kind;
@@ -125336,22 +131522,22 @@ var ts;
                         case 18 /* OpenBraceToken */:
                         case 19 /* CloseBraceToken */:
                         case 21 /* CloseParenToken */:
-                        case 87 /* ElseKeyword */:
-                        case 111 /* WhileKeyword */:
+                        case 90 /* ElseKeyword */:
+                        case 114 /* WhileKeyword */:
                         case 59 /* AtToken */:
                             return false;
                         case 43 /* SlashToken */:
                         case 31 /* GreaterThanToken */:
                             switch (container.kind) {
-                                case 268 /* JsxOpeningElement */:
-                                case 269 /* JsxClosingElement */:
-                                case 267 /* JsxSelfClosingElement */:
+                                case 275 /* JsxOpeningElement */:
+                                case 276 /* JsxClosingElement */:
+                                case 274 /* JsxSelfClosingElement */:
                                     return false;
                             }
                             break;
                         case 22 /* OpenBracketToken */:
                         case 23 /* CloseBracketToken */:
-                            if (container.kind !== 186 /* MappedType */) {
+                            if (container.kind !== 190 /* MappedType */) {
                                 return false;
                             }
                             break;
@@ -125438,6 +131624,9 @@ var ts;
                         // proceed any parent tokens that are located prior to child.getStart()
                         var tokenInfo = formattingScanner.readTokenInfo(node);
                         if (tokenInfo.token.end > childStartPos) {
+                            if (tokenInfo.token.pos > childStartPos) {
+                                formattingScanner.skipToStartOf(child);
+                            }
                             // stop when formatting scanner advances past the beginning of the child
                             break;
                         }
@@ -125446,15 +131635,17 @@ var ts;
                     if (!formattingScanner.isOnToken()) {
                         return inheritedIndentation;
                     }
-                    // JSX text shouldn't affect indenting
-                    if (ts.isToken(child) && child.kind !== 11 /* JsxText */) {
+                    if (ts.isToken(child)) {
                         // if child node is a token, it does not impact indentation, proceed it using parent indentation scope rules
                         var tokenInfo = formattingScanner.readTokenInfo(child);
-                        ts.Debug.assert(tokenInfo.token.end === child.end, "Token end is child end");
-                        consumeTokenAndAdvanceScanner(tokenInfo, node, parentDynamicIndentation, child);
-                        return inheritedIndentation;
+                        // JSX text shouldn't affect indenting
+                        if (child.kind !== 11 /* JsxText */) {
+                            ts.Debug.assert(tokenInfo.token.end === child.end, "Token end is child end");
+                            consumeTokenAndAdvanceScanner(tokenInfo, node, parentDynamicIndentation, child);
+                            return inheritedIndentation;
+                        }
                     }
-                    var effectiveParentStartLine = child.kind === 157 /* Decorator */ ? childStartLine : undecoratedParentStartLine;
+                    var effectiveParentStartLine = child.kind === 161 /* Decorator */ ? childStartLine : undecoratedParentStartLine;
                     var childIndentation = computeIndentation(child, childStartLine, childIndentationAmount, node, parentDynamicIndentation, effectiveParentStartLine);
                     processNode(child, childContextNode, childStartLine, undecoratedChildStartLine, childIndentation.indentation, childIndentation.delta);
                     if (child.kind === 11 /* JsxText */) {
@@ -125474,7 +131665,7 @@ var ts;
                         }
                     }
                     childContextNode = node;
-                    if (isFirstListItem && parent.kind === 192 /* ArrayLiteralExpression */ && inheritedIndentation === -1 /* Unknown */) {
+                    if (isFirstListItem && parent.kind === 199 /* ArrayLiteralExpression */ && inheritedIndentation === -1 /* Unknown */) {
                         inheritedIndentation = childIndentation.indentation;
                     }
                     return inheritedIndentation;
@@ -125741,9 +131932,6 @@ var ts;
                     return;
                 var startLinePos = ts.getStartPositionOfLine(startLine, sourceFile);
                 var nonWhitespaceColumnInFirstPart = formatting.SmartIndenter.findFirstNonWhitespaceCharacterAndColumn(startLinePos, parts[0].pos, sourceFile, options);
-                if (indentation === nonWhitespaceColumnInFirstPart.column && !jsxTextStyleIndent) {
-                    return;
-                }
                 var startIndex = 0;
                 if (firstLineIsIndented) {
                     startIndex = 1;
@@ -125917,12 +132105,12 @@ var ts;
         formatting.getRangeOfEnclosingComment = getRangeOfEnclosingComment;
         function getOpenTokenForList(node, list) {
             switch (node.kind) {
-                case 162 /* Constructor */:
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 202 /* ArrowFunction */:
+                case 166 /* Constructor */:
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 209 /* ArrowFunction */:
                     if (node.typeParameters === list) {
                         return 29 /* LessThanToken */;
                     }
@@ -125930,8 +132118,8 @@ var ts;
                         return 20 /* OpenParenToken */;
                     }
                     break;
-                case 196 /* CallExpression */:
-                case 197 /* NewExpression */:
+                case 203 /* CallExpression */:
+                case 204 /* NewExpression */:
                     if (node.typeArguments === list) {
                         return 29 /* LessThanToken */;
                     }
@@ -125939,12 +132127,12 @@ var ts;
                         return 20 /* OpenParenToken */;
                     }
                     break;
-                case 169 /* TypeReference */:
+                case 173 /* TypeReference */:
                     if (node.typeArguments === list) {
                         return 29 /* LessThanToken */;
                     }
                     break;
-                case 173 /* TypeLiteral */:
+                case 177 /* TypeLiteral */:
                     return 18 /* OpenBraceToken */;
             }
             return 0 /* Unknown */;
@@ -126062,7 +132250,7 @@ var ts;
                 if (options.indentStyle === ts.IndentStyle.Block) {
                     return getBlockIndent(sourceFile, position, options);
                 }
-                if (precedingToken.kind === 27 /* CommaToken */ && precedingToken.parent.kind !== 209 /* BinaryExpression */) {
+                if (precedingToken.kind === 27 /* CommaToken */ && precedingToken.parent.kind !== 216 /* BinaryExpression */) {
                     // previous token is comma that separates items in list - find the previous item and try to derive indentation from it
                     var actualIndentation = getActualIndentationForListItemBeforeComma(precedingToken, sourceFile, options);
                     if (actualIndentation !== -1 /* Unknown */) {
@@ -126216,7 +132404,7 @@ var ts;
                 // - parent is SourceFile - by default immediate children of SourceFile are not indented except when user indents them manually
                 // - parent and child are not on the same line
                 var useActualIndentation = (ts.isDeclaration(current) || ts.isStatementButNotDeclaration(current)) &&
-                    (parent.kind === 290 /* SourceFile */ || !parentAndChildShareLine);
+                    (parent.kind === 297 /* SourceFile */ || !parentAndChildShareLine);
                 if (!useActualIndentation) {
                     return -1 /* Unknown */;
                 }
@@ -126264,8 +132452,8 @@ var ts;
             }
             SmartIndenter.isArgumentAndStartLineOverlapsExpressionBeingCalled = isArgumentAndStartLineOverlapsExpressionBeingCalled;
             function childStartsOnTheSameLineWithElseInIfStatement(parent, child, childStartLine, sourceFile) {
-                if (parent.kind === 227 /* IfStatement */ && parent.elseStatement === child) {
-                    var elseKeyword = ts.findChildOfKind(parent, 87 /* ElseKeyword */, sourceFile);
+                if (parent.kind === 234 /* IfStatement */ && parent.elseStatement === child) {
+                    var elseKeyword = ts.findChildOfKind(parent, 90 /* ElseKeyword */, sourceFile);
                     ts.Debug.assert(elseKeyword !== undefined);
                     var elseKeywordStartLine = getStartLineAndCharacterForNode(elseKeyword, sourceFile).line;
                     return elseKeywordStartLine === childStartLine;
@@ -126302,40 +132490,40 @@ var ts;
             }
             function getListByRange(start, end, node, sourceFile) {
                 switch (node.kind) {
-                    case 169 /* TypeReference */:
+                    case 173 /* TypeReference */:
                         return getList(node.typeArguments);
-                    case 193 /* ObjectLiteralExpression */:
+                    case 200 /* ObjectLiteralExpression */:
                         return getList(node.properties);
-                    case 192 /* ArrayLiteralExpression */:
+                    case 199 /* ArrayLiteralExpression */:
                         return getList(node.elements);
-                    case 173 /* TypeLiteral */:
+                    case 177 /* TypeLiteral */:
                         return getList(node.members);
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
-                    case 202 /* ArrowFunction */:
-                    case 161 /* MethodDeclaration */:
-                    case 160 /* MethodSignature */:
-                    case 165 /* CallSignature */:
-                    case 162 /* Constructor */:
-                    case 171 /* ConstructorType */:
-                    case 166 /* ConstructSignature */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
+                    case 165 /* MethodDeclaration */:
+                    case 164 /* MethodSignature */:
+                    case 169 /* CallSignature */:
+                    case 166 /* Constructor */:
+                    case 175 /* ConstructorType */:
+                    case 170 /* ConstructSignature */:
                         return getList(node.typeParameters) || getList(node.parameters);
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 321 /* JSDocTemplateTag */:
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 330 /* JSDocTemplateTag */:
                         return getList(node.typeParameters);
-                    case 197 /* NewExpression */:
-                    case 196 /* CallExpression */:
+                    case 204 /* NewExpression */:
+                    case 203 /* CallExpression */:
                         return getList(node.typeArguments) || getList(node.arguments);
-                    case 243 /* VariableDeclarationList */:
+                    case 250 /* VariableDeclarationList */:
                         return getList(node.declarations);
-                    case 257 /* NamedImports */:
-                    case 261 /* NamedExports */:
+                    case 264 /* NamedImports */:
+                    case 268 /* NamedExports */:
                         return getList(node.elements);
-                    case 189 /* ObjectBindingPattern */:
-                    case 190 /* ArrayBindingPattern */:
+                    case 196 /* ObjectBindingPattern */:
+                    case 197 /* ArrayBindingPattern */:
                         return getList(node.elements);
                 }
                 function getList(list) {
@@ -126358,7 +132546,7 @@ var ts;
                 return findColumnForFirstNonWhitespaceCharacterInLine(sourceFile.getLineAndCharacterOfPosition(list.pos), sourceFile, options);
             }
             function getActualIndentationForListItem(node, sourceFile, options, listIndentsChild) {
-                if (node.parent && node.parent.kind === 243 /* VariableDeclarationList */) {
+                if (node.parent && node.parent.kind === 250 /* VariableDeclarationList */) {
                     // VariableDeclarationList has no wrapping tokens
                     return -1 /* Unknown */;
                 }
@@ -126431,87 +132619,91 @@ var ts;
             function nodeWillIndentChild(settings, parent, child, sourceFile, indentByDefault) {
                 var childKind = child ? child.kind : 0 /* Unknown */;
                 switch (parent.kind) {
-                    case 226 /* ExpressionStatement */:
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 248 /* EnumDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 192 /* ArrayLiteralExpression */:
-                    case 223 /* Block */:
-                    case 250 /* ModuleBlock */:
-                    case 193 /* ObjectLiteralExpression */:
-                    case 173 /* TypeLiteral */:
-                    case 186 /* MappedType */:
-                    case 175 /* TupleType */:
-                    case 251 /* CaseBlock */:
-                    case 278 /* DefaultClause */:
-                    case 277 /* CaseClause */:
-                    case 200 /* ParenthesizedExpression */:
-                    case 194 /* PropertyAccessExpression */:
-                    case 196 /* CallExpression */:
-                    case 197 /* NewExpression */:
-                    case 225 /* VariableStatement */:
-                    case 259 /* ExportAssignment */:
-                    case 235 /* ReturnStatement */:
-                    case 210 /* ConditionalExpression */:
-                    case 190 /* ArrayBindingPattern */:
-                    case 189 /* ObjectBindingPattern */:
-                    case 268 /* JsxOpeningElement */:
-                    case 271 /* JsxOpeningFragment */:
-                    case 267 /* JsxSelfClosingElement */:
-                    case 276 /* JsxExpression */:
-                    case 160 /* MethodSignature */:
-                    case 165 /* CallSignature */:
-                    case 166 /* ConstructSignature */:
-                    case 156 /* Parameter */:
-                    case 170 /* FunctionType */:
-                    case 171 /* ConstructorType */:
-                    case 182 /* ParenthesizedType */:
-                    case 198 /* TaggedTemplateExpression */:
-                    case 206 /* AwaitExpression */:
-                    case 261 /* NamedExports */:
-                    case 257 /* NamedImports */:
-                    case 263 /* ExportSpecifier */:
-                    case 258 /* ImportSpecifier */:
-                    case 159 /* PropertyDeclaration */:
+                    case 233 /* ExpressionStatement */:
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 255 /* EnumDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 199 /* ArrayLiteralExpression */:
+                    case 230 /* Block */:
+                    case 257 /* ModuleBlock */:
+                    case 200 /* ObjectLiteralExpression */:
+                    case 177 /* TypeLiteral */:
+                    case 190 /* MappedType */:
+                    case 179 /* TupleType */:
+                    case 258 /* CaseBlock */:
+                    case 285 /* DefaultClause */:
+                    case 284 /* CaseClause */:
+                    case 207 /* ParenthesizedExpression */:
+                    case 201 /* PropertyAccessExpression */:
+                    case 203 /* CallExpression */:
+                    case 204 /* NewExpression */:
+                    case 232 /* VariableStatement */:
+                    case 266 /* ExportAssignment */:
+                    case 242 /* ReturnStatement */:
+                    case 217 /* ConditionalExpression */:
+                    case 197 /* ArrayBindingPattern */:
+                    case 196 /* ObjectBindingPattern */:
+                    case 275 /* JsxOpeningElement */:
+                    case 278 /* JsxOpeningFragment */:
+                    case 274 /* JsxSelfClosingElement */:
+                    case 283 /* JsxExpression */:
+                    case 164 /* MethodSignature */:
+                    case 169 /* CallSignature */:
+                    case 170 /* ConstructSignature */:
+                    case 160 /* Parameter */:
+                    case 174 /* FunctionType */:
+                    case 175 /* ConstructorType */:
+                    case 186 /* ParenthesizedType */:
+                    case 205 /* TaggedTemplateExpression */:
+                    case 213 /* AwaitExpression */:
+                    case 268 /* NamedExports */:
+                    case 264 /* NamedImports */:
+                    case 270 /* ExportSpecifier */:
+                    case 265 /* ImportSpecifier */:
+                    case 163 /* PropertyDeclaration */:
                         return true;
-                    case 242 /* VariableDeclaration */:
-                    case 281 /* PropertyAssignment */:
-                    case 209 /* BinaryExpression */:
-                        if (!settings.indentMultiLineObjectLiteralBeginningOnBlankLine && sourceFile && childKind === 193 /* ObjectLiteralExpression */) { // TODO: GH#18217
+                    case 249 /* VariableDeclaration */:
+                    case 288 /* PropertyAssignment */:
+                    case 216 /* BinaryExpression */:
+                        if (!settings.indentMultiLineObjectLiteralBeginningOnBlankLine && sourceFile && childKind === 200 /* ObjectLiteralExpression */) { // TODO: GH#18217
                             return rangeIsOnOneLine(sourceFile, child);
                         }
-                        if (parent.kind !== 209 /* BinaryExpression */) {
+                        if (parent.kind !== 216 /* BinaryExpression */) {
                             return true;
                         }
                         break;
-                    case 228 /* DoStatement */:
-                    case 229 /* WhileStatement */:
-                    case 231 /* ForInStatement */:
-                    case 232 /* ForOfStatement */:
-                    case 230 /* ForStatement */:
-                    case 227 /* IfStatement */:
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
-                    case 161 /* MethodDeclaration */:
-                    case 202 /* ArrowFunction */:
-                    case 162 /* Constructor */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                        return childKind !== 223 /* Block */;
-                    case 260 /* ExportDeclaration */:
-                        return childKind !== 261 /* NamedExports */;
-                    case 254 /* ImportDeclaration */:
-                        return childKind !== 255 /* ImportClause */ ||
-                            (!!child.namedBindings && child.namedBindings.kind !== 257 /* NamedImports */);
-                    case 266 /* JsxElement */:
-                        return childKind !== 269 /* JsxClosingElement */;
-                    case 270 /* JsxFragment */:
-                        return childKind !== 272 /* JsxClosingFragment */;
-                    case 179 /* IntersectionType */:
-                    case 178 /* UnionType */:
-                        if (childKind === 173 /* TypeLiteral */) {
+                    case 235 /* DoStatement */:
+                    case 236 /* WhileStatement */:
+                    case 238 /* ForInStatement */:
+                    case 239 /* ForOfStatement */:
+                    case 237 /* ForStatement */:
+                    case 234 /* IfStatement */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
+                    case 165 /* MethodDeclaration */:
+                    case 166 /* Constructor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                        return childKind !== 230 /* Block */;
+                    case 209 /* ArrowFunction */:
+                        if (sourceFile && childKind === 207 /* ParenthesizedExpression */) {
+                            return rangeIsOnOneLine(sourceFile, child);
+                        }
+                        return childKind !== 230 /* Block */;
+                    case 267 /* ExportDeclaration */:
+                        return childKind !== 268 /* NamedExports */;
+                    case 261 /* ImportDeclaration */:
+                        return childKind !== 262 /* ImportClause */ ||
+                            (!!child.namedBindings && child.namedBindings.kind !== 264 /* NamedImports */);
+                    case 273 /* JsxElement */:
+                        return childKind !== 276 /* JsxClosingElement */;
+                    case 277 /* JsxFragment */:
+                        return childKind !== 279 /* JsxClosingFragment */;
+                    case 183 /* IntersectionType */:
+                    case 182 /* UnionType */:
+                        if (childKind === 177 /* TypeLiteral */ || childKind === 179 /* TupleType */) {
                             return false;
                         }
                     // falls through
@@ -126522,11 +132714,11 @@ var ts;
             SmartIndenter.nodeWillIndentChild = nodeWillIndentChild;
             function isControlFlowEndingStatement(kind, parent) {
                 switch (kind) {
-                    case 235 /* ReturnStatement */:
-                    case 239 /* ThrowStatement */:
-                    case 233 /* ContinueStatement */:
-                    case 234 /* BreakStatement */:
-                        return parent.kind !== 223 /* Block */;
+                    case 242 /* ReturnStatement */:
+                    case 246 /* ThrowStatement */:
+                    case 240 /* ContinueStatement */:
+                    case 241 /* BreakStatement */:
+                        return parent.kind !== 230 /* Block */;
                     default:
                         return false;
                 }
@@ -126600,8 +132792,10 @@ var ts;
         (function (TrailingTriviaOption) {
             /** Exclude all trailing trivia (use getEnd()) */
             TrailingTriviaOption[TrailingTriviaOption["Exclude"] = 0] = "Exclude";
+            /** Doesn't include whitespace, but does strip comments */
+            TrailingTriviaOption[TrailingTriviaOption["ExcludeWhitespace"] = 1] = "ExcludeWhitespace";
             /** Include trailing trivia */
-            TrailingTriviaOption[TrailingTriviaOption["Include"] = 1] = "Include";
+            TrailingTriviaOption[TrailingTriviaOption["Include"] = 2] = "Include";
         })(TrailingTriviaOption = textChanges_3.TrailingTriviaOption || (textChanges_3.TrailingTriviaOption = {}));
         function skipWhitespacesAndLineBreaks(text, start) {
             return ts.skipTrivia(text, start, /*stopAfterLineBreak*/ false, /*stopAtComments*/ true);
@@ -126672,9 +132866,18 @@ var ts;
             return ts.getStartPositionOfLine(ts.getLineOfLocalPosition(sourceFile, adjustedStartPosition), sourceFile);
         }
         function getAdjustedEndPosition(sourceFile, node, options) {
+            var _a;
             var end = node.end;
             var trailingTriviaOption = options.trailingTriviaOption;
-            if (trailingTriviaOption === TrailingTriviaOption.Exclude || (ts.isExpression(node) && trailingTriviaOption !== TrailingTriviaOption.Include)) {
+            if (trailingTriviaOption === TrailingTriviaOption.Exclude) {
+                return end;
+            }
+            if (trailingTriviaOption === TrailingTriviaOption.ExcludeWhitespace) {
+                var comments = ts.concatenate(ts.getTrailingCommentRanges(sourceFile.text, end), ts.getLeadingCommentRanges(sourceFile.text, end));
+                var realEnd = (_a = comments === null || comments === void 0 ? void 0 : comments[comments.length - 1]) === null || _a === void 0 ? void 0 : _a.end;
+                if (realEnd) {
+                    return realEnd;
+                }
                 return end;
             }
             var newEnd = ts.skipTrivia(sourceFile.text, end, /*stopAfterLineBreak*/ true);
@@ -126686,7 +132889,7 @@ var ts;
          * Checks if 'candidate' argument is a legal separator in the list that contains 'node' as an element
          */
         function isSeparator(node, candidate) {
-            return !!candidate && !!node.parent && (candidate.kind === 27 /* CommaToken */ || (candidate.kind === 26 /* SemicolonToken */ && node.parent.kind === 193 /* ObjectLiteralExpression */));
+            return !!candidate && !!node.parent && (candidate.kind === 27 /* CommaToken */ || (candidate.kind === 26 /* SemicolonToken */ && node.parent.kind === 200 /* ObjectLiteralExpression */));
         }
         function spaces(count) {
             var s = "";
@@ -126706,7 +132909,7 @@ var ts;
                 this.formatContext = formatContext;
                 this.changes = [];
                 this.newFiles = [];
-                this.classesWithNodesInsertedAtStart = ts.createMap(); // Set<ClassDeclaration> implemented as Map<node id, ClassDeclaration>
+                this.classesWithNodesInsertedAtStart = new ts.Map(); // Set<ClassDeclaration> implemented as Map<node id, ClassDeclaration>
                 this.deletedNodes = [];
             }
             ChangeTracker.fromContext = function (context) {
@@ -126825,13 +133028,14 @@ var ts;
                     this.insertNodeAt(sourceFile, parameters.pos, newParam);
                 }
             };
-            ChangeTracker.prototype.insertNodeBefore = function (sourceFile, before, newNode, blankLineBetween) {
+            ChangeTracker.prototype.insertNodeBefore = function (sourceFile, before, newNode, blankLineBetween, options) {
                 if (blankLineBetween === void 0) { blankLineBetween = false; }
-                this.insertNodeAt(sourceFile, getAdjustedStartPosition(sourceFile, before, {}), newNode, this.getOptionsForInsertNodeBefore(before, newNode, blankLineBetween));
+                if (options === void 0) { options = {}; }
+                this.insertNodeAt(sourceFile, getAdjustedStartPosition(sourceFile, before, options), newNode, this.getOptionsForInsertNodeBefore(before, newNode, blankLineBetween));
             };
             ChangeTracker.prototype.insertModifierBefore = function (sourceFile, modifier, before) {
                 var pos = before.getStart(sourceFile);
-                this.insertNodeAt(sourceFile, pos, ts.createToken(modifier), { suffix: " " });
+                this.insertNodeAt(sourceFile, pos, ts.factory.createToken(modifier), { suffix: " " });
             };
             ChangeTracker.prototype.insertLastModifierBefore = function (sourceFile, modifier, before) {
                 if (!before.modifiers) {
@@ -126839,7 +133043,7 @@ var ts;
                     return;
                 }
                 var pos = before.modifiers.end;
-                this.insertNodeAt(sourceFile, pos, ts.createToken(modifier), { prefix: " " });
+                this.insertNodeAt(sourceFile, pos, ts.factory.createToken(modifier), { prefix: " " });
             };
             ChangeTracker.prototype.insertCommentBeforeLine = function (sourceFile, lineNumber, position, commentText) {
                 var lineStartPosition = ts.getStartPositionOfLine(lineNumber, sourceFile);
@@ -126889,7 +133093,7 @@ var ts;
                     }
                 }
                 else {
-                    endNode = (_a = (node.kind === 242 /* VariableDeclaration */ ? node.exclamationToken : node.questionToken)) !== null && _a !== void 0 ? _a : node.name;
+                    endNode = (_a = (node.kind === 249 /* VariableDeclaration */ ? node.exclamationToken : node.questionToken)) !== null && _a !== void 0 ? _a : node.name;
                 }
                 this.insertNodeAt(sourceFile, endNode.end, type, { prefix: ": " });
                 return true;
@@ -126902,11 +133106,11 @@ var ts;
             ChangeTracker.prototype.insertTypeParameters = function (sourceFile, node, typeParameters) {
                 // If no `(`, is an arrow function `x => x`, so use the pos of the first parameter
                 var start = (ts.findChildOfKind(node, 20 /* OpenParenToken */, sourceFile) || ts.first(node.parameters)).getStart(sourceFile);
-                this.insertNodesAt(sourceFile, start, typeParameters, { prefix: "<", suffix: ">" });
+                this.insertNodesAt(sourceFile, start, typeParameters, { prefix: "<", suffix: ">", joiner: ", " });
             };
-            ChangeTracker.prototype.getOptionsForInsertNodeBefore = function (before, inserted, doubleNewlines) {
+            ChangeTracker.prototype.getOptionsForInsertNodeBefore = function (before, inserted, blankLineBetween) {
                 if (ts.isStatement(before) || ts.isClassElement(before)) {
-                    return { suffix: doubleNewlines ? this.newLineCharacter + this.newLineCharacter : this.newLineCharacter };
+                    return { suffix: blankLineBetween ? this.newLineCharacter + this.newLineCharacter : this.newLineCharacter };
                 }
                 else if (ts.isVariableDeclaration(before)) { // insert `x = 1, ` into `const x = 1, y = 2;
                     return { suffix: ", " };
@@ -126917,6 +133121,9 @@ var ts;
                 else if (ts.isStringLiteral(before) && ts.isImportDeclaration(before.parent) || ts.isNamedImports(before)) {
                     return { suffix: ", " };
                 }
+                else if (ts.isImportSpecifier(before)) {
+                    return { suffix: "," + (blankLineBetween ? this.newLineCharacter : " ") };
+                }
                 return ts.Debug.failBadSyntaxKind(before); // We haven't handled this kind of node yet -- add it
             };
             ChangeTracker.prototype.insertNodeAtConstructorStart = function (sourceFile, ctr, newStatement) {
@@ -126938,7 +133145,7 @@ var ts;
                 }
             };
             ChangeTracker.prototype.replaceConstructorBody = function (sourceFile, ctr, statements) {
-                this.replaceNode(sourceFile, ctr.body, ts.createBlock(statements, /*multiLine*/ true));
+                this.replaceNode(sourceFile, ctr.body, ts.factory.createBlock(statements, /*multiLine*/ true));
             };
             ChangeTracker.prototype.insertNodeAtEndOfScope = function (sourceFile, scope, newNode) {
                 var pos = getAdjustedStartPosition(sourceFile, scope.getLastToken(), {});
@@ -127031,7 +133238,7 @@ var ts;
                     // check if previous statement ends with semicolon
                     // if not - insert semicolon to preserve the code from changing the meaning due to ASI
                     if (sourceFile.text.charCodeAt(after.end - 1) !== 59 /* semicolon */) {
-                        this.replaceRange(sourceFile, ts.createRange(after.end), ts.createToken(26 /* SemicolonToken */));
+                        this.replaceRange(sourceFile, ts.createRange(after.end), ts.factory.createToken(26 /* SemicolonToken */));
                     }
                 }
                 var endPosition = getAdjustedEndPosition(sourceFile, after, {});
@@ -127043,18 +133250,18 @@ var ts;
             };
             ChangeTracker.prototype.getInsertNodeAfterOptionsWorker = function (node) {
                 switch (node.kind) {
-                    case 245 /* ClassDeclaration */:
-                    case 249 /* ModuleDeclaration */:
+                    case 252 /* ClassDeclaration */:
+                    case 256 /* ModuleDeclaration */:
                         return { prefix: this.newLineCharacter, suffix: this.newLineCharacter };
-                    case 242 /* VariableDeclaration */:
+                    case 249 /* VariableDeclaration */:
                     case 10 /* StringLiteral */:
-                    case 75 /* Identifier */:
+                    case 78 /* Identifier */:
                         return { prefix: ", " };
-                    case 281 /* PropertyAssignment */:
+                    case 288 /* PropertyAssignment */:
                         return { suffix: "," + this.newLineCharacter };
-                    case 89 /* ExportKeyword */:
+                    case 92 /* ExportKeyword */:
                         return { prefix: " " };
-                    case 156 /* Parameter */:
+                    case 160 /* Parameter */:
                         return {};
                     default:
                         ts.Debug.assert(ts.isStatement(node) || ts.isClassOrTypeElement(node)); // Else we haven't handled this kind of node yet -- add it
@@ -127063,29 +133270,29 @@ var ts;
             };
             ChangeTracker.prototype.insertName = function (sourceFile, node, name) {
                 ts.Debug.assert(!node.name);
-                if (node.kind === 202 /* ArrowFunction */) {
+                if (node.kind === 209 /* ArrowFunction */) {
                     var arrow = ts.findChildOfKind(node, 38 /* EqualsGreaterThanToken */, sourceFile);
                     var lparen = ts.findChildOfKind(node, 20 /* OpenParenToken */, sourceFile);
                     if (lparen) {
                         // `() => {}` --> `function f() {}`
-                        this.insertNodesAt(sourceFile, lparen.getStart(sourceFile), [ts.createToken(94 /* FunctionKeyword */), ts.createIdentifier(name)], { joiner: " " });
+                        this.insertNodesAt(sourceFile, lparen.getStart(sourceFile), [ts.factory.createToken(97 /* FunctionKeyword */), ts.factory.createIdentifier(name)], { joiner: " " });
                         deleteNode(this, sourceFile, arrow);
                     }
                     else {
                         // `x => {}` -> `function f(x) {}`
                         this.insertText(sourceFile, ts.first(node.parameters).getStart(sourceFile), "function " + name + "(");
                         // Replacing full range of arrow to get rid of the leading space -- replace ` =>` with `)`
-                        this.replaceRange(sourceFile, arrow, ts.createToken(21 /* CloseParenToken */));
+                        this.replaceRange(sourceFile, arrow, ts.factory.createToken(21 /* CloseParenToken */));
                     }
-                    if (node.body.kind !== 223 /* Block */) {
+                    if (node.body.kind !== 230 /* Block */) {
                         // `() => 0` => `function f() { return 0; }`
-                        this.insertNodesAt(sourceFile, node.body.getStart(sourceFile), [ts.createToken(18 /* OpenBraceToken */), ts.createToken(101 /* ReturnKeyword */)], { joiner: " ", suffix: " " });
-                        this.insertNodesAt(sourceFile, node.body.end, [ts.createToken(26 /* SemicolonToken */), ts.createToken(19 /* CloseBraceToken */)], { joiner: " " });
+                        this.insertNodesAt(sourceFile, node.body.getStart(sourceFile), [ts.factory.createToken(18 /* OpenBraceToken */), ts.factory.createToken(104 /* ReturnKeyword */)], { joiner: " ", suffix: " " });
+                        this.insertNodesAt(sourceFile, node.body.end, [ts.factory.createToken(26 /* SemicolonToken */), ts.factory.createToken(19 /* CloseBraceToken */)], { joiner: " " });
                     }
                 }
                 else {
-                    var pos = ts.findChildOfKind(node, node.kind === 201 /* FunctionExpression */ ? 94 /* FunctionKeyword */ : 80 /* ClassKeyword */, sourceFile).end;
-                    this.insertNodeAt(sourceFile, pos, ts.createIdentifier(name), { prefix: " " });
+                    var pos = ts.findChildOfKind(node, node.kind === 208 /* FunctionExpression */ ? 97 /* FunctionKeyword */ : 83 /* ClassKeyword */, sourceFile).end;
+                    this.insertNodeAt(sourceFile, pos, ts.factory.createIdentifier(name), { prefix: " " });
                 }
             };
             ChangeTracker.prototype.insertExportModifier = function (sourceFile, node) {
@@ -127188,12 +133395,13 @@ var ts;
                     }
                     if (multilineList) {
                         // insert separator immediately following the 'after' node to preserve comments in trailing trivia
-                        this.replaceRange(sourceFile, ts.createRange(end), ts.createToken(separator));
+                        this.replaceRange(sourceFile, ts.createRange(end), ts.factory.createToken(separator));
                         // use the same indentation as 'after' item
                         var indentation = ts.formatting.SmartIndenter.findFirstNonWhitespaceColumn(afterStartLinePosition, afterStart, sourceFile, this.formatContext.options);
                         // insert element before the line break on the line that contains 'after' element
                         var insertPos = ts.skipTrivia(sourceFile.text, end, /*stopAfterLineBreak*/ true, /*stopAtComments*/ false);
-                        if (insertPos !== end && ts.isLineBreak(sourceFile.text.charCodeAt(insertPos - 1))) {
+                        // find position before "\n" or "\r\n"
+                        while (insertPos !== end && ts.isLineBreak(sourceFile.text.charCodeAt(insertPos - 1))) {
                             insertPos--;
                         }
                         this.replaceRange(sourceFile, ts.createRange(insertPos), newNode, { indentation: indentation, prefix: this.newLineCharacter });
@@ -127204,7 +133412,7 @@ var ts;
                 }
             };
             ChangeTracker.prototype.parenthesizeExpression = function (sourceFile, expression) {
-                this.replaceRange(sourceFile, ts.rangeOfNode(expression), ts.createParen(expression));
+                this.replaceRange(sourceFile, ts.rangeOfNode(expression), ts.factory.createParenthesizedExpression(expression));
             };
             ChangeTracker.prototype.finishClassesWithNodesInsertedAtStart = function () {
                 var _this = this;
@@ -127226,7 +133434,7 @@ var ts;
             };
             ChangeTracker.prototype.finishDeleteDeclarations = function () {
                 var _this = this;
-                var deletedNodesInLists = new ts.NodeSet(); // Stores nodes in lists that we already deleted. Used to avoid deleting `, ` twice in `a, b`.
+                var deletedNodesInLists = new ts.Set(); // Stores nodes in lists that we already deleted. Used to avoid deleting `, ` twice in `a, b`.
                 var _loop_9 = function (sourceFile, node) {
                     if (!this_1.deletedNodes.some(function (d) { return d.sourceFile === sourceFile && ts.rangeContainsRangeExclusive(d.node, node); })) {
                         if (ts.isArray(node)) {
@@ -127328,7 +133536,7 @@ var ts;
             changesToText.newFileChanges = newFileChanges;
             function newFileChangesWorker(oldFile, scriptKind, statements, newLineCharacter, formatContext) {
                 // TODO: this emits the file, parses it back, then formats it that -- may be a less roundabout way to do this
-                var nonFormattedText = statements.map(function (s) { return getNonformattedText(s, oldFile, newLineCharacter).text; }).join(newLineCharacter);
+                var nonFormattedText = statements.map(function (s) { return s === 4 /* NewLineTrivia */ ? "" : getNonformattedText(s, oldFile, newLineCharacter).text; }).join(newLineCharacter);
                 var sourceFile = ts.createSourceFile("any file name", nonFormattedText, 99 /* ESNext */, /*setParentNodes*/ true, scriptKind);
                 var changes = ts.formatting.formatDocument(sourceFile, formatContext);
                 return applyChanges(nonFormattedText, changes) + newLineCharacter;
@@ -127379,7 +133587,12 @@ var ts;
             function getNonformattedText(node, sourceFile, newLineCharacter) {
                 var writer = createWriter(newLineCharacter);
                 var newLine = newLineCharacter === "\n" ? 1 /* LineFeed */ : 0 /* CarriageReturnLineFeed */;
-                ts.createPrinter({ newLine: newLine, neverAsciiEscape: true, preserveSourceNewlines: true }, writer).writeNode(4 /* Unspecified */, node, sourceFile, writer);
+                ts.createPrinter({
+                    newLine: newLine,
+                    neverAsciiEscape: true,
+                    preserveSourceNewlines: true,
+                    terminateUnterminatedLiterals: true
+                }, writer).writeNode(4 /* Unspecified */, node, sourceFile, writer);
                 return { text: writer.getText(), node: assignPositionsToNode(node) };
             }
             changesToText.getNonformattedText = getNonformattedText;
@@ -127399,8 +133612,7 @@ var ts;
             var visited = ts.visitEachChild(node, assignPositionsToNode, ts.nullTransformationContext, assignPositionsToNodeArray, assignPositionsToNode); // TODO: GH#18217
             // create proxy node for non synthesized nodes
             var newNode = ts.nodeIsSynthesized(visited) ? visited : Object.create(visited);
-            newNode.pos = getPos(node);
-            newNode.end = getEnd(node);
+            ts.setTextRangePosEnd(newNode, getPos(node), getEnd(node));
             return newNode;
         }
         function assignPositionsToNodeArray(nodes, visitor, test, start, count) {
@@ -127409,9 +133621,8 @@ var ts;
                 return visited;
             }
             // clone nodearray if necessary
-            var nodeArray = visited === nodes ? ts.createNodeArray(visited.slice(0)) : visited;
-            nodeArray.pos = getPos(nodes);
-            nodeArray.end = getEnd(nodes);
+            var nodeArray = visited === nodes ? ts.factory.createNodeArray(visited.slice(0)) : visited;
+            ts.setTextRangePosEnd(nodeArray, getPos(nodes), getEnd(nodes));
             return nodeArray;
         }
         function createWriter(newLine) {
@@ -127655,14 +133866,14 @@ var ts;
         }
         textChanges_3.isValidLocationToAddComment = isValidLocationToAddComment;
         function needSemicolonBetween(a, b) {
-            return (ts.isPropertySignature(a) || ts.isPropertyDeclaration(a)) && ts.isClassOrTypeElement(b) && b.name.kind === 154 /* ComputedPropertyName */
+            return (ts.isPropertySignature(a) || ts.isPropertyDeclaration(a)) && ts.isClassOrTypeElement(b) && b.name.kind === 158 /* ComputedPropertyName */
                 || ts.isStatementButNotDeclaration(a) && ts.isStatementButNotDeclaration(b); // TODO: only if b would start with a `(` or `[`
         }
         var deleteDeclaration;
         (function (deleteDeclaration_1) {
             function deleteDeclaration(changes, deletedNodesInLists, sourceFile, node) {
                 switch (node.kind) {
-                    case 156 /* Parameter */: {
+                    case 160 /* Parameter */: {
                         var oldFunction = node.parent;
                         if (ts.isArrowFunction(oldFunction) &&
                             oldFunction.parameters.length === 1 &&
@@ -127677,15 +133888,15 @@ var ts;
                         }
                         break;
                     }
-                    case 254 /* ImportDeclaration */:
-                    case 253 /* ImportEqualsDeclaration */:
+                    case 261 /* ImportDeclaration */:
+                    case 260 /* ImportEqualsDeclaration */:
                         var isFirstImport = sourceFile.imports.length && node === ts.first(sourceFile.imports).parent || node === ts.find(sourceFile.statements, ts.isAnyImportSyntax);
                         // For first import, leave header comment in place, otherwise only delete JSDoc comments
                         deleteNode(changes, sourceFile, node, { leadingTriviaOption: isFirstImport ? LeadingTriviaOption.Exclude : ts.hasJSDocNodes(node) ? LeadingTriviaOption.JSDoc : LeadingTriviaOption.StartLine });
                         break;
-                    case 191 /* BindingElement */:
+                    case 198 /* BindingElement */:
                         var pattern = node.parent;
-                        var preserveComma = pattern.kind === 190 /* ArrayBindingPattern */ && node !== ts.last(pattern.elements);
+                        var preserveComma = pattern.kind === 197 /* ArrayBindingPattern */ && node !== ts.last(pattern.elements);
                         if (preserveComma) {
                             deleteNode(changes, sourceFile, node);
                         }
@@ -127693,13 +133904,13 @@ var ts;
                             deleteNodeInList(changes, deletedNodesInLists, sourceFile, node);
                         }
                         break;
-                    case 242 /* VariableDeclaration */:
+                    case 249 /* VariableDeclaration */:
                         deleteVariableDeclaration(changes, deletedNodesInLists, sourceFile, node);
                         break;
-                    case 155 /* TypeParameter */:
+                    case 159 /* TypeParameter */:
                         deleteNodeInList(changes, deletedNodesInLists, sourceFile, node);
                         break;
-                    case 258 /* ImportSpecifier */:
+                    case 265 /* ImportSpecifier */:
                         var namedImports = node.parent;
                         if (namedImports.elements.length === 1) {
                             deleteImportBinding(changes, sourceFile, namedImports);
@@ -127708,17 +133919,17 @@ var ts;
                             deleteNodeInList(changes, deletedNodesInLists, sourceFile, node);
                         }
                         break;
-                    case 256 /* NamespaceImport */:
+                    case 263 /* NamespaceImport */:
                         deleteImportBinding(changes, sourceFile, node);
                         break;
                     case 26 /* SemicolonToken */:
                         deleteNode(changes, sourceFile, node, { trailingTriviaOption: TrailingTriviaOption.Exclude });
                         break;
-                    case 94 /* FunctionKeyword */:
+                    case 97 /* FunctionKeyword */:
                         deleteNode(changes, sourceFile, node, { leadingTriviaOption: LeadingTriviaOption.Exclude });
                         break;
-                    case 245 /* ClassDeclaration */:
-                    case 244 /* FunctionDeclaration */:
+                    case 252 /* ClassDeclaration */:
+                    case 251 /* FunctionDeclaration */:
                         deleteNode(changes, sourceFile, node, { leadingTriviaOption: ts.hasJSDocNodes(node) ? LeadingTriviaOption.JSDoc : LeadingTriviaOption.StartLine });
                         break;
                     default:
@@ -127765,13 +133976,13 @@ var ts;
                     // Delete the entire import declaration
                     // |import * as ns from './file'|
                     // |import { a } from './file'|
-                    var importDecl = ts.getAncestor(node, 254 /* ImportDeclaration */);
+                    var importDecl = ts.getAncestor(node, 261 /* ImportDeclaration */);
                     deleteNode(changes, sourceFile, importDecl);
                 }
             }
             function deleteVariableDeclaration(changes, deletedNodesInLists, sourceFile, node) {
                 var parent = node.parent;
-                if (parent.kind === 280 /* CatchClause */) {
+                if (parent.kind === 287 /* CatchClause */) {
                     // TODO: There's currently no unused diagnostic for this, could be a suggestion
                     changes.deleteNodeRange(sourceFile, ts.findChildOfKind(parent, 20 /* OpenParenToken */, sourceFile), ts.findChildOfKind(parent, 21 /* CloseParenToken */, sourceFile));
                     return;
@@ -127782,14 +133993,14 @@ var ts;
                 }
                 var gp = parent.parent;
                 switch (gp.kind) {
-                    case 232 /* ForOfStatement */:
-                    case 231 /* ForInStatement */:
-                        changes.replaceNode(sourceFile, node, ts.createObjectLiteral());
+                    case 239 /* ForOfStatement */:
+                    case 238 /* ForInStatement */:
+                        changes.replaceNode(sourceFile, node, ts.factory.createObjectLiteralExpression());
                         break;
-                    case 230 /* ForStatement */:
+                    case 237 /* ForStatement */:
                         deleteNode(changes, sourceFile, parent);
                         break;
-                    case 225 /* VariableStatement */:
+                    case 232 /* VariableStatement */:
                         deleteNode(changes, sourceFile, gp, { leadingTriviaOption: ts.hasJSDocNodes(gp) ? LeadingTriviaOption.JSDoc : LeadingTriviaOption.StartLine });
                         break;
                     default:
@@ -127831,7 +134042,7 @@ var ts;
     var codefix;
     (function (codefix) {
         var errorCodeToFixes = ts.createMultiMap();
-        var fixIdToRegistration = ts.createMap();
+        var fixIdToRegistration = new ts.Map();
         function diagnosticToString(diag) {
             return ts.isArray(diag)
                 ? ts.formatStringFromArgs(ts.getLocaleSpecificMessage(diag[0]), diag.slice(1))
@@ -127929,7 +134140,7 @@ var ts;
     (function (refactor_1) {
         // A map with the refactor code as key, the refactor itself as value
         // e.g.  nonSuggestableRefactors[refactorCode] -> the refactor you want
-        var refactors = ts.createMap();
+        var refactors = new ts.Map();
         /** @param name An unique code associated with each refactor. Does not have to be human-readable. */
         function registerRefactor(name, refactor) {
             refactors.set(name, refactor);
@@ -127966,10 +134177,10 @@ var ts;
         });
         function makeChange(changeTracker, sourceFile, pos) {
             var token = ts.getTokenAtPosition(sourceFile, pos);
-            var assertion = ts.Debug.checkDefined(ts.findAncestor(token, function (n) { return ts.isAsExpression(n) || ts.isTypeAssertion(n); }), "Expected to find an assertion expression");
+            var assertion = ts.Debug.checkDefined(ts.findAncestor(token, function (n) { return ts.isAsExpression(n) || ts.isTypeAssertionExpression(n); }), "Expected to find an assertion expression");
             var replacement = ts.isAsExpression(assertion)
-                ? ts.createAsExpression(assertion.expression, ts.createKeywordTypeNode(148 /* UnknownKeyword */))
-                : ts.createTypeAssertion(ts.createKeywordTypeNode(148 /* UnknownKeyword */), assertion.expression);
+                ? ts.factory.createAsExpression(assertion.expression, ts.factory.createKeywordTypeNode(152 /* UnknownKeyword */))
+                : ts.factory.createTypeAssertion(ts.factory.createKeywordTypeNode(152 /* UnknownKeyword */), assertion.expression);
             changeTracker.replaceNode(sourceFile, assertion.expression, replacement);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -127984,11 +134195,11 @@ var ts;
             getCodeActions: function (context) {
                 var sourceFile = context.sourceFile;
                 var changes = ts.textChanges.ChangeTracker.with(context, function (changes) {
-                    var exportDeclaration = ts.createExportDeclaration(
+                    var exportDeclaration = ts.factory.createExportDeclaration(
                     /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, ts.createNamedExports([]), 
-                    /*moduleSpecifier*/ undefined
-                    /*isTypeOnly*/ false);
+                    /*modifiers*/ undefined, 
+                    /*isTypeOnly*/ false, ts.factory.createNamedExports([])
+                    /*moduleSpecifier*/ undefined);
                     changes.insertNodeAtEndOfScope(sourceFile, sourceFile, exportDeclaration);
                 });
                 return [codefix.createCodeFixActionWithoutFixAll("addEmptyExportDeclaration", changes, ts.Diagnostics.Add_export_to_make_this_file_into_a_module)];
@@ -128023,7 +134234,7 @@ var ts;
             },
             getAllCodeActions: function (context) {
                 var sourceFile = context.sourceFile;
-                var fixedDeclarations = ts.createMap();
+                var fixedDeclarations = new ts.Set();
                 return codefix.codeFixAll(context, errorCodes, function (t, diagnostic) {
                     var span = diagnostic.relatedInformation && ts.find(diagnostic.relatedInformation, function (r) { return r.code === ts.Diagnostics.Did_you_mean_to_mark_this_function_as_async.code; });
                     var decl = getFixableErrorSpanDeclaration(sourceFile, span);
@@ -128041,14 +134252,12 @@ var ts;
         }
         function makeChange(changeTracker, sourceFile, insertionSite, fixedDeclarations) {
             if (fixedDeclarations) {
-                if (fixedDeclarations.has(ts.getNodeId(insertionSite).toString())) {
+                if (fixedDeclarations.has(ts.getNodeId(insertionSite))) {
                     return;
                 }
             }
-            fixedDeclarations === null || fixedDeclarations === void 0 ? void 0 : fixedDeclarations.set(ts.getNodeId(insertionSite).toString(), true);
-            var cloneWithModifier = ts.getSynthesizedDeepClone(insertionSite, /*includeTrivia*/ true);
-            cloneWithModifier.modifiers = ts.createNodeArray(ts.createModifiersFromModifierFlags(ts.getModifierFlags(insertionSite) | 256 /* Async */));
-            cloneWithModifier.modifierFlagsCache = 0;
+            fixedDeclarations === null || fixedDeclarations === void 0 ? void 0 : fixedDeclarations.add(ts.getNodeId(insertionSite));
+            var cloneWithModifier = ts.factory.updateModifiers(ts.getSynthesizedDeepClone(insertionSite, /*includeTrivia*/ true), ts.factory.createNodeArray(ts.factory.createModifiersFromModifierFlags(ts.getSyntacticModifierFlags(insertionSite) | 256 /* Async */)));
             changeTracker.replaceNode(sourceFile, insertionSite, cloneWithModifier);
         }
         function getFixableErrorSpanDeclaration(sourceFile, span) {
@@ -128124,7 +134333,7 @@ var ts;
             getAllCodeActions: function (context) {
                 var sourceFile = context.sourceFile, program = context.program, cancellationToken = context.cancellationToken;
                 var checker = context.program.getTypeChecker();
-                var fixedDeclarations = ts.createMap();
+                var fixedDeclarations = new ts.Set();
                 return codefix.codeFixAll(context, errorCodes, function (t, diagnostic) {
                     var expression = getFixableErrorSpanExpression(sourceFile, diagnostic.code, diagnostic, cancellationToken, program);
                     if (!expression) {
@@ -128200,12 +134409,12 @@ var ts;
                 }
                 var declaration = ts.tryCast(symbol.valueDeclaration, ts.isVariableDeclaration);
                 var variableName = declaration && ts.tryCast(declaration.name, ts.isIdentifier);
-                var variableStatement = ts.getAncestor(declaration, 225 /* VariableStatement */);
+                var variableStatement = ts.getAncestor(declaration, 232 /* VariableStatement */);
                 if (!declaration || !variableStatement ||
                     declaration.type ||
                     !declaration.initializer ||
                     variableStatement.getSourceFile() !== sourceFile ||
-                    ts.hasModifier(variableStatement, 1 /* Export */) ||
+                    ts.hasSyntacticModifier(variableStatement, 1 /* Export */) ||
                     !variableName ||
                     !isInsideAwaitableBody(declaration.initializer)) {
                     isCompleteFix = false;
@@ -128278,10 +134487,10 @@ var ts;
         function isInsideAwaitableBody(node) {
             return node.kind & 32768 /* AwaitContext */ || !!ts.findAncestor(node, function (ancestor) {
                 return ancestor.parent && ts.isArrowFunction(ancestor.parent) && ancestor.parent.body === ancestor ||
-                    ts.isBlock(ancestor) && (ancestor.parent.kind === 244 /* FunctionDeclaration */ ||
-                        ancestor.parent.kind === 201 /* FunctionExpression */ ||
-                        ancestor.parent.kind === 202 /* ArrowFunction */ ||
-                        ancestor.parent.kind === 161 /* MethodDeclaration */);
+                    ts.isBlock(ancestor) && (ancestor.parent.kind === 251 /* FunctionDeclaration */ ||
+                        ancestor.parent.kind === 208 /* FunctionExpression */ ||
+                        ancestor.parent.kind === 209 /* ArrowFunction */ ||
+                        ancestor.parent.kind === 165 /* MethodDeclaration */);
             });
         }
         function makeChange(changeTracker, errorCode, sourceFile, checker, insertionSite, fixedDeclarations) {
@@ -128290,43 +134499,43 @@ var ts;
                     var side = _a[_i];
                     if (fixedDeclarations && ts.isIdentifier(side)) {
                         var symbol = checker.getSymbolAtLocation(side);
-                        if (symbol && fixedDeclarations.has(ts.getSymbolId(symbol).toString())) {
+                        if (symbol && fixedDeclarations.has(ts.getSymbolId(symbol))) {
                             continue;
                         }
                     }
                     var type = checker.getTypeAtLocation(side);
-                    var newNode = checker.getPromisedTypeOfPromise(type) ? ts.createAwait(side) : side;
+                    var newNode = checker.getPromisedTypeOfPromise(type) ? ts.factory.createAwaitExpression(side) : side;
                     changeTracker.replaceNode(sourceFile, side, newNode);
                 }
             }
             else if (errorCode === propertyAccessCode && ts.isPropertyAccessExpression(insertionSite.parent)) {
                 if (fixedDeclarations && ts.isIdentifier(insertionSite.parent.expression)) {
                     var symbol = checker.getSymbolAtLocation(insertionSite.parent.expression);
-                    if (symbol && fixedDeclarations.has(ts.getSymbolId(symbol).toString())) {
+                    if (symbol && fixedDeclarations.has(ts.getSymbolId(symbol))) {
                         return;
                     }
                 }
-                changeTracker.replaceNode(sourceFile, insertionSite.parent.expression, ts.createParen(ts.createAwait(insertionSite.parent.expression)));
+                changeTracker.replaceNode(sourceFile, insertionSite.parent.expression, ts.factory.createParenthesizedExpression(ts.factory.createAwaitExpression(insertionSite.parent.expression)));
                 insertLeadingSemicolonIfNeeded(changeTracker, insertionSite.parent.expression, sourceFile);
             }
             else if (ts.contains(callableConstructableErrorCodes, errorCode) && ts.isCallOrNewExpression(insertionSite.parent)) {
                 if (fixedDeclarations && ts.isIdentifier(insertionSite)) {
                     var symbol = checker.getSymbolAtLocation(insertionSite);
-                    if (symbol && fixedDeclarations.has(ts.getSymbolId(symbol).toString())) {
+                    if (symbol && fixedDeclarations.has(ts.getSymbolId(symbol))) {
                         return;
                     }
                 }
-                changeTracker.replaceNode(sourceFile, insertionSite, ts.createParen(ts.createAwait(insertionSite)));
+                changeTracker.replaceNode(sourceFile, insertionSite, ts.factory.createParenthesizedExpression(ts.factory.createAwaitExpression(insertionSite)));
                 insertLeadingSemicolonIfNeeded(changeTracker, insertionSite, sourceFile);
             }
             else {
                 if (fixedDeclarations && ts.isVariableDeclaration(insertionSite.parent) && ts.isIdentifier(insertionSite.parent.name)) {
                     var symbol = checker.getSymbolAtLocation(insertionSite.parent.name);
-                    if (symbol && !ts.addToSeen(fixedDeclarations, ts.getSymbolId(symbol))) {
+                    if (symbol && !ts.tryAddToSet(fixedDeclarations, ts.getSymbolId(symbol))) {
                         return;
                     }
                 }
-                changeTracker.replaceNode(sourceFile, insertionSite, ts.createAwait(insertionSite));
+                changeTracker.replaceNode(sourceFile, insertionSite, ts.factory.createAwaitExpression(insertionSite));
             }
         }
         function insertLeadingSemicolonIfNeeded(changeTracker, beforeNode, sourceFile) {
@@ -128357,7 +134566,7 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var fixedNodes = new ts.NodeSet();
+                var fixedNodes = new ts.Set();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) { return makeChange(changes, diag.file, diag.start, context.program, fixedNodes); });
             },
         });
@@ -128393,17 +134602,17 @@ var ts;
             }
         }
         function applyChange(changeTracker, initializer, sourceFile, fixedNodes) {
-            if (!fixedNodes || fixedNodes.tryAdd(initializer)) {
-                changeTracker.insertModifierBefore(sourceFile, 81 /* ConstKeyword */, initializer);
+            if (!fixedNodes || ts.tryAddToSet(fixedNodes, initializer)) {
+                changeTracker.insertModifierBefore(sourceFile, 84 /* ConstKeyword */, initializer);
             }
         }
         function isPossiblyPartOfDestructuring(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
-                case 192 /* ArrayLiteralExpression */:
-                case 193 /* ObjectLiteralExpression */:
-                case 281 /* PropertyAssignment */:
-                case 282 /* ShorthandPropertyAssignment */:
+                case 78 /* Identifier */:
+                case 199 /* ArrayLiteralExpression */:
+                case 200 /* ObjectLiteralExpression */:
+                case 288 /* PropertyAssignment */:
+                case 289 /* ShorthandPropertyAssignment */:
                     return true;
                 default:
                     return false;
@@ -128417,8 +134626,8 @@ var ts;
         }
         function isPossiblyPartOfCommaSeperatedInitializer(node) {
             switch (node.kind) {
-                case 75 /* Identifier */:
-                case 209 /* BinaryExpression */:
+                case 78 /* Identifier */:
+                case 216 /* BinaryExpression */:
                 case 27 /* CommaToken */:
                     return true;
                 default:
@@ -128457,7 +134666,7 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var fixedNodes = new ts.NodeSet();
+                var fixedNodes = new ts.Set();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) { return makeChange(changes, diag.file, diag.start, fixedNodes); });
             },
         });
@@ -128467,9 +134676,9 @@ var ts;
                 return;
             }
             var declaration = token.parent;
-            if (declaration.kind === 159 /* PropertyDeclaration */ &&
-                (!fixedNodes || fixedNodes.tryAdd(declaration))) {
-                changeTracker.insertModifierBefore(sourceFile, 130 /* DeclareKeyword */, declaration);
+            if (declaration.kind === 163 /* PropertyDeclaration */ &&
+                (!fixedNodes || ts.tryAddToSet(fixedNodes, declaration))) {
+                changeTracker.insertModifierBefore(sourceFile, 133 /* DeclareKeyword */, declaration);
             }
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -128494,7 +134703,7 @@ var ts;
             var token = ts.getTokenAtPosition(sourceFile, pos);
             var decorator = ts.findAncestor(token, ts.isDecorator);
             ts.Debug.assert(!!decorator, "Expected position to be owned by a decorator.");
-            var replacement = ts.createCall(decorator.expression, /*typeArguments*/ undefined, /*argumentsArray*/ undefined);
+            var replacement = ts.factory.createCallExpression(decorator.expression, /*typeArguments*/ undefined, /*argumentsArray*/ undefined);
             changeTracker.replaceNode(sourceFile, decorator.expression, replacement);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -128527,8 +134736,8 @@ var ts;
             var i = param.parent.parameters.indexOf(param);
             ts.Debug.assert(!param.type, "Tried to add a parameter name to a parameter that already had one.");
             ts.Debug.assert(i > -1, "Parameter not found in parent parameter list.");
-            var replacement = ts.createParameter(
-            /*decorators*/ undefined, param.modifiers, param.dotDotDotToken, "arg" + i, param.questionToken, ts.createTypeReferenceNode(token, /*typeArguments*/ undefined), param.initializer);
+            var replacement = ts.factory.createParameterDeclaration(
+            /*decorators*/ undefined, param.modifiers, param.dotDotDotToken, "arg" + i, param.questionToken, ts.factory.createTypeReferenceNode(token, /*typeArguments*/ undefined), param.initializer);
             changeTracker.replaceNode(sourceFile, token, replacement);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -128579,7 +134788,7 @@ var ts;
                 }
                 var needParens = ts.isArrowFunction(decl) && !ts.findChildOfKind(decl, 20 /* OpenParenToken */, sourceFile);
                 if (needParens)
-                    changes.insertNodeBefore(sourceFile, ts.first(decl.parameters), ts.createToken(20 /* OpenParenToken */));
+                    changes.insertNodeBefore(sourceFile, ts.first(decl.parameters), ts.factory.createToken(20 /* OpenParenToken */));
                 for (var _i = 0, _a = decl.parameters; _i < _a.length; _i++) {
                     var param = _a[_i];
                     if (!param.type) {
@@ -128589,7 +134798,7 @@ var ts;
                     }
                 }
                 if (needParens)
-                    changes.insertNodeAfter(sourceFile, ts.last(decl.parameters), ts.createToken(21 /* CloseParenToken */));
+                    changes.insertNodeAfter(sourceFile, ts.last(decl.parameters), ts.factory.createToken(21 /* CloseParenToken */));
                 if (!decl.type) {
                     var returnType = ts.getJSDocReturnType(decl);
                     if (returnType)
@@ -128604,51 +134813,54 @@ var ts;
         }
         function isDeclarationWithType(node) {
             return ts.isFunctionLikeDeclaration(node) ||
-                node.kind === 242 /* VariableDeclaration */ ||
-                node.kind === 158 /* PropertySignature */ ||
-                node.kind === 159 /* PropertyDeclaration */;
+                node.kind === 249 /* VariableDeclaration */ ||
+                node.kind === 162 /* PropertySignature */ ||
+                node.kind === 163 /* PropertyDeclaration */;
         }
         function transformJSDocType(node) {
             switch (node.kind) {
-                case 295 /* JSDocAllType */:
-                case 296 /* JSDocUnknownType */:
-                    return ts.createTypeReferenceNode("any", ts.emptyArray);
-                case 299 /* JSDocOptionalType */:
+                case 303 /* JSDocAllType */:
+                case 304 /* JSDocUnknownType */:
+                    return ts.factory.createTypeReferenceNode("any", ts.emptyArray);
+                case 307 /* JSDocOptionalType */:
                     return transformJSDocOptionalType(node);
-                case 298 /* JSDocNonNullableType */:
+                case 306 /* JSDocNonNullableType */:
                     return transformJSDocType(node.type);
-                case 297 /* JSDocNullableType */:
+                case 305 /* JSDocNullableType */:
                     return transformJSDocNullableType(node);
-                case 301 /* JSDocVariadicType */:
+                case 309 /* JSDocVariadicType */:
                     return transformJSDocVariadicType(node);
-                case 300 /* JSDocFunctionType */:
+                case 308 /* JSDocFunctionType */:
                     return transformJSDocFunctionType(node);
-                case 169 /* TypeReference */:
+                case 173 /* TypeReference */:
                     return transformJSDocTypeReference(node);
                 default:
-                    var visited = ts.visitEachChild(node, transformJSDocType, /*context*/ undefined); // TODO: GH#18217
+                    var visited = ts.visitEachChild(node, transformJSDocType, ts.nullTransformationContext);
                     ts.setEmitFlags(visited, 1 /* SingleLine */);
                     return visited;
             }
         }
         function transformJSDocOptionalType(node) {
-            return ts.createUnionTypeNode([ts.visitNode(node.type, transformJSDocType), ts.createTypeReferenceNode("undefined", ts.emptyArray)]);
+            return ts.factory.createUnionTypeNode([ts.visitNode(node.type, transformJSDocType), ts.factory.createTypeReferenceNode("undefined", ts.emptyArray)]);
         }
         function transformJSDocNullableType(node) {
-            return ts.createUnionTypeNode([ts.visitNode(node.type, transformJSDocType), ts.createTypeReferenceNode("null", ts.emptyArray)]);
+            return ts.factory.createUnionTypeNode([ts.visitNode(node.type, transformJSDocType), ts.factory.createTypeReferenceNode("null", ts.emptyArray)]);
         }
         function transformJSDocVariadicType(node) {
-            return ts.createArrayTypeNode(ts.visitNode(node.type, transformJSDocType));
+            return ts.factory.createArrayTypeNode(ts.visitNode(node.type, transformJSDocType));
         }
         function transformJSDocFunctionType(node) {
-            return ts.createFunctionTypeNode(ts.emptyArray, node.parameters.map(transformJSDocParameter), node.type);
+            var _a;
+            // TODO: This does not properly handle `function(new:C, string)` per https://github.com/google/closure-compiler/wiki/Types-in-the-Closure-Type-System#the-javascript-type-language
+            //       however we do handle it correctly in `serializeTypeForDeclaration` in checker.ts
+            return ts.factory.createFunctionTypeNode(ts.emptyArray, node.parameters.map(transformJSDocParameter), (_a = node.type) !== null && _a !== void 0 ? _a : ts.factory.createKeywordTypeNode(128 /* AnyKeyword */));
         }
         function transformJSDocParameter(node) {
             var index = node.parent.parameters.indexOf(node);
-            var isRest = node.type.kind === 301 /* JSDocVariadicType */ && index === node.parent.parameters.length - 1; // TODO: GH#18217
+            var isRest = node.type.kind === 309 /* JSDocVariadicType */ && index === node.parent.parameters.length - 1; // TODO: GH#18217
             var name = node.name || (isRest ? "rest" : "arg" + index);
-            var dotdotdot = isRest ? ts.createToken(25 /* DotDotDotToken */) : node.dotDotDotToken;
-            return ts.createParameter(node.decorators, node.modifiers, dotdotdot, name, node.questionToken, ts.visitNode(node.type, transformJSDocType), node.initializer);
+            var dotdotdot = isRest ? ts.factory.createToken(25 /* DotDotDotToken */) : node.dotDotDotToken;
+            return ts.factory.createParameterDeclaration(node.decorators, node.modifiers, dotdotdot, name, node.questionToken, ts.visitNode(node.type, transformJSDocType), node.initializer);
         }
         function transformJSDocTypeReference(node) {
             var name = node.typeName;
@@ -128671,24 +134883,24 @@ var ts;
                         text = text[0].toUpperCase() + text.slice(1);
                         break;
                 }
-                name = ts.createIdentifier(text);
+                name = ts.factory.createIdentifier(text);
                 if ((text === "Array" || text === "Promise") && !node.typeArguments) {
-                    args = ts.createNodeArray([ts.createTypeReferenceNode("any", ts.emptyArray)]);
+                    args = ts.factory.createNodeArray([ts.factory.createTypeReferenceNode("any", ts.emptyArray)]);
                 }
                 else {
                     args = ts.visitNodes(node.typeArguments, transformJSDocType);
                 }
             }
-            return ts.createTypeReferenceNode(name, args);
+            return ts.factory.createTypeReferenceNode(name, args);
         }
         function transformJSDocIndexSignature(node) {
-            var index = ts.createParameter(
+            var index = ts.factory.createParameterDeclaration(
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, 
-            /*dotDotDotToken*/ undefined, node.typeArguments[0].kind === 140 /* NumberKeyword */ ? "n" : "s", 
-            /*questionToken*/ undefined, ts.createTypeReferenceNode(node.typeArguments[0].kind === 140 /* NumberKeyword */ ? "number" : "string", []), 
+            /*dotDotDotToken*/ undefined, node.typeArguments[0].kind === 144 /* NumberKeyword */ ? "n" : "s", 
+            /*questionToken*/ undefined, ts.factory.createTypeReferenceNode(node.typeArguments[0].kind === 144 /* NumberKeyword */ ? "number" : "string", []), 
             /*initializer*/ undefined);
-            var indexSignature = ts.createTypeLiteralNode([ts.createIndexSignature(/*decorators*/ undefined, /*modifiers*/ undefined, [index], node.typeArguments[1])]);
+            var indexSignature = ts.factory.createTypeLiteralNode([ts.factory.createIndexSignature(/*decorators*/ undefined, /*modifiers*/ undefined, [index], node.typeArguments[1])]);
             ts.setEmitFlags(indexSignature, 1 /* SingleLine */);
             return indexSignature;
         }
@@ -128717,133 +134929,186 @@ var ts;
                 return undefined;
             }
             var ctorDeclaration = ctorSymbol.valueDeclaration;
-            var precedingNode;
-            var newClassDeclaration;
-            switch (ctorDeclaration.kind) {
-                case 244 /* FunctionDeclaration */:
-                    precedingNode = ctorDeclaration;
-                    changes.delete(sourceFile, ctorDeclaration);
-                    newClassDeclaration = createClassFromFunctionDeclaration(ctorDeclaration);
-                    break;
-                case 242 /* VariableDeclaration */:
-                    precedingNode = ctorDeclaration.parent.parent;
-                    newClassDeclaration = createClassFromVariableDeclaration(ctorDeclaration);
-                    if (ctorDeclaration.parent.declarations.length === 1) {
-                        ts.copyLeadingComments(precedingNode, newClassDeclaration, sourceFile); // TODO: GH#18217
-                        changes.delete(sourceFile, precedingNode);
-                    }
-                    else {
-                        changes.delete(sourceFile, ctorDeclaration);
-                    }
-                    break;
-            }
-            if (!newClassDeclaration) {
-                return undefined;
+            if (ts.isFunctionDeclaration(ctorDeclaration)) {
+                changes.replaceNode(sourceFile, ctorDeclaration, createClassFromFunctionDeclaration(ctorDeclaration));
             }
-            // Deleting a declaration only deletes JSDoc style comments, so only copy those to the new node.
-            if (ts.hasJSDocNodes(ctorDeclaration)) {
-                ts.copyLeadingComments(ctorDeclaration, newClassDeclaration, sourceFile);
+            else if (ts.isVariableDeclaration(ctorDeclaration)) {
+                var classDeclaration = createClassFromVariableDeclaration(ctorDeclaration);
+                if (!classDeclaration) {
+                    return undefined;
+                }
+                var ancestor = ctorDeclaration.parent.parent;
+                if (ts.isVariableDeclarationList(ctorDeclaration.parent) && ctorDeclaration.parent.declarations.length > 1) {
+                    changes.delete(sourceFile, ctorDeclaration);
+                    changes.insertNodeAfter(sourceFile, ancestor, classDeclaration);
+                }
+                else {
+                    changes.replaceNode(sourceFile, ancestor, classDeclaration);
+                }
             }
-            // Because the preceding node could be touched, we need to insert nodes before delete nodes.
-            changes.insertNodeAfter(sourceFile, precedingNode, newClassDeclaration);
             function createClassElementsFromSymbol(symbol) {
                 var memberElements = [];
                 // all instance members are stored in the "member" array of symbol
                 if (symbol.members) {
-                    symbol.members.forEach(function (member) {
+                    symbol.members.forEach(function (member, key) {
+                        if (key === "constructor") {
+                            // fn.prototype.constructor = fn
+                            changes.delete(sourceFile, member.valueDeclaration.parent);
+                            return;
+                        }
                         var memberElement = createClassElement(member, /*modifiers*/ undefined);
                         if (memberElement) {
-                            memberElements.push(memberElement);
+                            memberElements.push.apply(memberElements, memberElement);
                         }
                     });
                 }
                 // all static members are stored in the "exports" array of symbol
                 if (symbol.exports) {
                     symbol.exports.forEach(function (member) {
-                        var memberElement = createClassElement(member, [ts.createToken(120 /* StaticKeyword */)]);
-                        if (memberElement) {
-                            memberElements.push(memberElement);
+                        if (member.name === "prototype") {
+                            var firstDeclaration = member.declarations[0];
+                            // only one "x.prototype = { ... }" will pass
+                            if (member.declarations.length === 1 &&
+                                ts.isPropertyAccessExpression(firstDeclaration) &&
+                                ts.isBinaryExpression(firstDeclaration.parent) &&
+                                firstDeclaration.parent.operatorToken.kind === 62 /* EqualsToken */ &&
+                                ts.isObjectLiteralExpression(firstDeclaration.parent.right)) {
+                                var prototypes = firstDeclaration.parent.right;
+                                var memberElement = createClassElement(prototypes.symbol, /** modifiers */ undefined);
+                                if (memberElement) {
+                                    memberElements.push.apply(memberElements, memberElement);
+                                }
+                            }
+                        }
+                        else {
+                            var memberElement = createClassElement(member, [ts.factory.createToken(123 /* StaticKeyword */)]);
+                            if (memberElement) {
+                                memberElements.push.apply(memberElements, memberElement);
+                            }
                         }
                     });
                 }
                 return memberElements;
                 function shouldConvertDeclaration(_target, source) {
-                    // Right now the only thing we can convert are function expressions - other values shouldn't get
-                    // transformed. We can update this once ES public class properties are available.
-                    return ts.isFunctionLike(source);
+                    // Right now the only thing we can convert are function expressions, get/set accessors and methods
+                    // other values like normal value fields ({a: 1}) shouldn't get transformed.
+                    // We can update this once ES public class properties are available.
+                    if (ts.isPropertyAccessExpression(_target)) {
+                        if (isConstructorAssignment(_target))
+                            return true;
+                        return ts.isFunctionLike(source);
+                    }
+                    else {
+                        return ts.every(_target.properties, function (property) {
+                            // a() {}
+                            if (ts.isMethodDeclaration(property) || ts.isGetOrSetAccessorDeclaration(property))
+                                return true;
+                            // a: function() {}
+                            if (ts.isPropertyAssignment(property) && ts.isFunctionExpression(property.initializer) && !!property.name)
+                                return true;
+                            // x.prototype.constructor = fn
+                            if (isConstructorAssignment(property))
+                                return true;
+                            return false;
+                        });
+                    }
                 }
                 function createClassElement(symbol, modifiers) {
                     // Right now the only thing we can convert are function expressions, which are marked as methods
-                    if (!(symbol.flags & 8192 /* Method */)) {
-                        return;
+                    // or { x: y } type prototype assignments, which are marked as ObjectLiteral
+                    var members = [];
+                    if (!(symbol.flags & 8192 /* Method */) && !(symbol.flags & 4096 /* ObjectLiteral */)) {
+                        return members;
                     }
                     var memberDeclaration = symbol.valueDeclaration;
                     var assignmentBinaryExpression = memberDeclaration.parent;
-                    if (!shouldConvertDeclaration(memberDeclaration, assignmentBinaryExpression.right)) {
-                        return;
+                    var assignmentExpr = assignmentBinaryExpression.right;
+                    if (!shouldConvertDeclaration(memberDeclaration, assignmentExpr)) {
+                        return members;
                     }
                     // delete the entire statement if this expression is the sole expression to take care of the semicolon at the end
-                    var nodeToDelete = assignmentBinaryExpression.parent && assignmentBinaryExpression.parent.kind === 226 /* ExpressionStatement */
+                    var nodeToDelete = assignmentBinaryExpression.parent && assignmentBinaryExpression.parent.kind === 233 /* ExpressionStatement */
                         ? assignmentBinaryExpression.parent : assignmentBinaryExpression;
                     changes.delete(sourceFile, nodeToDelete);
-                    if (!assignmentBinaryExpression.right) {
-                        return ts.createProperty([], modifiers, symbol.name, /*questionToken*/ undefined, 
-                        /*type*/ undefined, /*initializer*/ undefined);
-                    }
-                    switch (assignmentBinaryExpression.right.kind) {
-                        case 201 /* FunctionExpression */: {
-                            var functionExpression = assignmentBinaryExpression.right;
-                            var fullModifiers = ts.concatenate(modifiers, getModifierKindFromSource(functionExpression, 126 /* AsyncKeyword */));
-                            var method = ts.createMethod(/*decorators*/ undefined, fullModifiers, /*asteriskToken*/ undefined, memberDeclaration.name, /*questionToken*/ undefined, 
-                            /*typeParameters*/ undefined, functionExpression.parameters, /*type*/ undefined, functionExpression.body);
-                            ts.copyLeadingComments(assignmentBinaryExpression, method, sourceFile);
-                            return method;
-                        }
-                        case 202 /* ArrowFunction */: {
-                            var arrowFunction = assignmentBinaryExpression.right;
-                            var arrowFunctionBody = arrowFunction.body;
-                            var bodyBlock = void 0;
-                            // case 1: () => { return [1,2,3] }
-                            if (arrowFunctionBody.kind === 223 /* Block */) {
-                                bodyBlock = arrowFunctionBody;
-                            }
-                            // case 2: () => [1,2,3]
-                            else {
-                                bodyBlock = ts.createBlock([ts.createReturn(arrowFunctionBody)]);
-                            }
-                            var fullModifiers = ts.concatenate(modifiers, getModifierKindFromSource(arrowFunction, 126 /* AsyncKeyword */));
-                            var method = ts.createMethod(/*decorators*/ undefined, fullModifiers, /*asteriskToken*/ undefined, memberDeclaration.name, /*questionToken*/ undefined, 
-                            /*typeParameters*/ undefined, arrowFunction.parameters, /*type*/ undefined, bodyBlock);
-                            ts.copyLeadingComments(assignmentBinaryExpression, method, sourceFile);
-                            return method;
-                        }
-                        default: {
-                            // Don't try to declare members in JavaScript files
-                            if (ts.isSourceFileJS(sourceFile)) {
-                                return;
-                            }
-                            var prop = ts.createProperty(/*decorators*/ undefined, modifiers, memberDeclaration.name, /*questionToken*/ undefined, 
-                            /*type*/ undefined, assignmentBinaryExpression.right);
-                            ts.copyLeadingComments(assignmentBinaryExpression.parent, prop, sourceFile);
-                            return prop;
+                    if (!assignmentExpr) {
+                        members.push(ts.factory.createPropertyDeclaration([], modifiers, symbol.name, /*questionToken*/ undefined, 
+                        /*type*/ undefined, /*initializer*/ undefined));
+                        return members;
+                    }
+                    // f.x = expr
+                    if (ts.isPropertyAccessExpression(memberDeclaration) && (ts.isFunctionExpression(assignmentExpr) || ts.isArrowFunction(assignmentExpr))) {
+                        return createFunctionLikeExpressionMember(members, assignmentExpr, memberDeclaration.name);
+                    }
+                    // f.prototype = { ... }
+                    else if (ts.isObjectLiteralExpression(assignmentExpr)) {
+                        return ts.flatMap(assignmentExpr.properties, function (property) {
+                            if (ts.isMethodDeclaration(property) || ts.isGetOrSetAccessorDeclaration(property)) {
+                                // MethodDeclaration and AccessorDeclaration can appear in a class directly
+                                return members.concat(property);
+                            }
+                            if (ts.isPropertyAssignment(property) && ts.isFunctionExpression(property.initializer)) {
+                                return createFunctionLikeExpressionMember(members, property.initializer, property.name);
+                            }
+                            // Drop constructor assignments
+                            if (isConstructorAssignment(property))
+                                return members;
+                            return [];
+                        });
+                    }
+                    else {
+                        // Don't try to declare members in JavaScript files
+                        if (ts.isSourceFileJS(sourceFile))
+                            return members;
+                        if (!ts.isPropertyAccessExpression(memberDeclaration))
+                            return members;
+                        var prop = ts.factory.createPropertyDeclaration(/*decorators*/ undefined, modifiers, memberDeclaration.name, /*questionToken*/ undefined, /*type*/ undefined, assignmentExpr);
+                        ts.copyLeadingComments(assignmentBinaryExpression.parent, prop, sourceFile);
+                        members.push(prop);
+                        return members;
+                    }
+                    function createFunctionLikeExpressionMember(members, expression, name) {
+                        if (ts.isFunctionExpression(expression))
+                            return createFunctionExpressionMember(members, expression, name);
+                        else
+                            return createArrowFunctionExpressionMember(members, expression, name);
+                    }
+                    function createFunctionExpressionMember(members, functionExpression, name) {
+                        var fullModifiers = ts.concatenate(modifiers, getModifierKindFromSource(functionExpression, 129 /* AsyncKeyword */));
+                        var method = ts.factory.createMethodDeclaration(/*decorators*/ undefined, fullModifiers, /*asteriskToken*/ undefined, name, /*questionToken*/ undefined, 
+                        /*typeParameters*/ undefined, functionExpression.parameters, /*type*/ undefined, functionExpression.body);
+                        ts.copyLeadingComments(assignmentBinaryExpression, method, sourceFile);
+                        return members.concat(method);
+                    }
+                    function createArrowFunctionExpressionMember(members, arrowFunction, name) {
+                        var arrowFunctionBody = arrowFunction.body;
+                        var bodyBlock;
+                        // case 1: () => { return [1,2,3] }
+                        if (arrowFunctionBody.kind === 230 /* Block */) {
+                            bodyBlock = arrowFunctionBody;
+                        }
+                        // case 2: () => [1,2,3]
+                        else {
+                            bodyBlock = ts.factory.createBlock([ts.factory.createReturnStatement(arrowFunctionBody)]);
                         }
+                        var fullModifiers = ts.concatenate(modifiers, getModifierKindFromSource(arrowFunction, 129 /* AsyncKeyword */));
+                        var method = ts.factory.createMethodDeclaration(/*decorators*/ undefined, fullModifiers, /*asteriskToken*/ undefined, name, /*questionToken*/ undefined, 
+                        /*typeParameters*/ undefined, arrowFunction.parameters, /*type*/ undefined, bodyBlock);
+                        ts.copyLeadingComments(assignmentBinaryExpression, method, sourceFile);
+                        return members.concat(method);
                     }
                 }
             }
             function createClassFromVariableDeclaration(node) {
                 var initializer = node.initializer;
-                if (!initializer || initializer.kind !== 201 /* FunctionExpression */) {
-                    return undefined;
-                }
-                if (node.name.kind !== 75 /* Identifier */) {
+                if (!initializer || !ts.isFunctionExpression(initializer) || !ts.isIdentifier(node.name)) {
                     return undefined;
                 }
                 var memberElements = createClassElementsFromSymbol(node.symbol);
                 if (initializer.body) {
-                    memberElements.unshift(ts.createConstructor(/*decorators*/ undefined, /*modifiers*/ undefined, initializer.parameters, initializer.body));
+                    memberElements.unshift(ts.factory.createConstructorDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, initializer.parameters, initializer.body));
                 }
-                var modifiers = getModifierKindFromSource(precedingNode, 89 /* ExportKeyword */);
-                var cls = ts.createClassDeclaration(/*decorators*/ undefined, modifiers, node.name, 
+                var modifiers = getModifierKindFromSource(node.parent.parent, 92 /* ExportKeyword */);
+                var cls = ts.factory.createClassDeclaration(/*decorators*/ undefined, modifiers, node.name, 
                 /*typeParameters*/ undefined, /*heritageClauses*/ undefined, memberElements);
                 // Don't call copyComments here because we'll already leave them in place
                 return cls;
@@ -128851,10 +135116,10 @@ var ts;
             function createClassFromFunctionDeclaration(node) {
                 var memberElements = createClassElementsFromSymbol(ctorSymbol);
                 if (node.body) {
-                    memberElements.unshift(ts.createConstructor(/*decorators*/ undefined, /*modifiers*/ undefined, node.parameters, node.body));
+                    memberElements.unshift(ts.factory.createConstructorDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, node.parameters, node.body));
                 }
-                var modifiers = getModifierKindFromSource(node, 89 /* ExportKeyword */);
-                var cls = ts.createClassDeclaration(/*decorators*/ undefined, modifiers, node.name, 
+                var modifiers = getModifierKindFromSource(node, 92 /* ExportKeyword */);
+                var cls = ts.factory.createClassDeclaration(/*decorators*/ undefined, modifiers, node.name, 
                 /*typeParameters*/ undefined, /*heritageClauses*/ undefined, memberElements);
                 // Don't call copyComments here because we'll already leave them in place
                 return cls;
@@ -128863,6 +135128,13 @@ var ts;
         function getModifierKindFromSource(source, kind) {
             return ts.filter(source.modifiers, function (modifier) { return modifier.kind === kind; });
         }
+        function isConstructorAssignment(x) {
+            if (!x.name)
+                return false;
+            if (ts.isIdentifier(x.name) && x.name.text === "constructor")
+                return true;
+            return false;
+        }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
 /* @internal */
@@ -128903,7 +135175,7 @@ var ts;
             if (!functionToConvert) {
                 return;
             }
-            var synthNamesMap = ts.createMap();
+            var synthNamesMap = new ts.Map();
             var isInJavascript = ts.isInJSFile(functionToConvert);
             var setOfExpressionsToReturn = getAllPromiseExpressionsToReturn(functionToConvert, checker);
             var functionToConvertRenamed = renameCollidingVarNames(functionToConvert, checker, synthNamesMap, context.sourceFile);
@@ -128913,7 +135185,7 @@ var ts;
                 return;
             }
             // add the async keyword
-            changes.insertLastModifierBefore(sourceFile, 126 /* AsyncKeyword */, functionToConvert);
+            changes.insertLastModifierBefore(sourceFile, 129 /* AsyncKeyword */, functionToConvert);
             var _loop_12 = function (returnStatement) {
                 ts.forEachChild(returnStatement, function visit(node) {
                     if (ts.isCallExpression(node)) {
@@ -128943,21 +135215,21 @@ var ts;
         */
         function getAllPromiseExpressionsToReturn(func, checker) {
             if (!func.body) {
-                return ts.createMap();
+                return new ts.Set();
             }
-            var setOfExpressionsToReturn = ts.createMap();
+            var setOfExpressionsToReturn = new ts.Set();
             ts.forEachChild(func.body, function visit(node) {
                 if (isPromiseReturningCallExpression(node, checker, "then")) {
-                    setOfExpressionsToReturn.set(ts.getNodeId(node).toString(), true);
+                    setOfExpressionsToReturn.add(ts.getNodeId(node));
                     ts.forEach(node.arguments, visit);
                 }
                 else if (isPromiseReturningCallExpression(node, checker, "catch")) {
-                    setOfExpressionsToReturn.set(ts.getNodeId(node).toString(), true);
+                    setOfExpressionsToReturn.add(ts.getNodeId(node));
                     // if .catch() is the last call in the chain, move leftward in the chain until we hit something else that should be returned
                     ts.forEachChild(node, visit);
                 }
                 else if (isPromiseTypedExpression(node, checker)) {
-                    setOfExpressionsToReturn.set(ts.getNodeId(node).toString(), true);
+                    setOfExpressionsToReturn.add(ts.getNodeId(node));
                     // don't recurse here, since we won't refactor any children or arguments of the expression
                 }
                 else {
@@ -128987,7 +135259,7 @@ var ts;
             It then checks for any collisions and renames them through getSynthesizedDeepClone
         */
         function renameCollidingVarNames(nodeToRename, checker, synthNamesMap, sourceFile) {
-            var identsToRenameMap = ts.createMap(); // key is the symbol id
+            var identsToRenameMap = new ts.Map(); // key is the symbol id
             var collidingSymbolMap = ts.createMultiMap();
             ts.forEachChild(nodeToRename, function visit(node) {
                 if (!ts.isIdentifier(node)) {
@@ -129006,9 +135278,9 @@ var ts;
                     // will eventually become
                     //   const response = await fetch('...')
                     // so we push an entry for 'response'.
-                    if (lastCallSignature && !ts.isFunctionLikeDeclaration(node.parent) && !synthNamesMap.has(symbolIdString)) {
+                    if (lastCallSignature && !ts.isParameter(node.parent) && !ts.isFunctionLikeDeclaration(node.parent) && !synthNamesMap.has(symbolIdString)) {
                         var firstParameter = ts.firstOrUndefined(lastCallSignature.parameters);
-                        var ident = firstParameter && ts.isParameter(firstParameter.valueDeclaration) && ts.tryCast(firstParameter.valueDeclaration.name, ts.isIdentifier) || ts.createOptimisticUniqueName("result");
+                        var ident = firstParameter && ts.isParameter(firstParameter.valueDeclaration) && ts.tryCast(firstParameter.valueDeclaration.name, ts.isIdentifier) || ts.factory.createUniqueName("result", 16 /* Optimistic */);
                         var synthName = getNewNameIfConflict(ident, collidingSymbolMap);
                         synthNamesMap.set(symbolIdString, synthName);
                         collidingSymbolMap.add(ident.text, symbol);
@@ -129032,11 +135304,26 @@ var ts;
                     }
                 }
             });
-            return ts.getSynthesizedDeepCloneWithRenames(nodeToRename, /*includeTrivia*/ true, identsToRenameMap, checker);
+            return ts.getSynthesizedDeepCloneWithReplacements(nodeToRename, /*includeTrivia*/ true, function (original) {
+                if (ts.isBindingElement(original) && ts.isIdentifier(original.name) && ts.isObjectBindingPattern(original.parent)) {
+                    var symbol = checker.getSymbolAtLocation(original.name);
+                    var renameInfo = symbol && identsToRenameMap.get(String(ts.getSymbolId(symbol)));
+                    if (renameInfo && renameInfo.text !== (original.name || original.propertyName).getText()) {
+                        return ts.factory.createBindingElement(original.dotDotDotToken, original.propertyName || original.name, renameInfo, original.initializer);
+                    }
+                }
+                else if (ts.isIdentifier(original)) {
+                    var symbol = checker.getSymbolAtLocation(original);
+                    var renameInfo = symbol && identsToRenameMap.get(String(ts.getSymbolId(symbol)));
+                    if (renameInfo) {
+                        return ts.factory.createIdentifier(renameInfo.text);
+                    }
+                }
+            });
         }
         function getNewNameIfConflict(name, originalNames) {
             var numVarsSameName = (originalNames.get(name.text) || ts.emptyArray).length;
-            var identifier = numVarsSameName === 0 ? name : ts.createIdentifier(name.text + "_" + numVarsSameName);
+            var identifier = numVarsSameName === 0 ? name : ts.factory.createIdentifier(name.text + "_" + numVarsSameName);
             return createSynthIdentifier(identifier);
         }
         function silentFail() {
@@ -129086,18 +135373,18 @@ var ts;
                     });
                 }
                 else {
-                    possibleNameForVarDecl = createSynthIdentifier(ts.createOptimisticUniqueName("result"), prevArgName.types);
+                    possibleNameForVarDecl = createSynthIdentifier(ts.factory.createUniqueName("result", 16 /* Optimistic */), prevArgName.types);
                 }
                 // We are about to write a 'let' variable declaration, but `transformExpression` for both
                 // the try block and catch block will assign to this name. Setting this flag indicates
                 // that future assignments should be written as `name = value` instead of `const name = value`.
                 possibleNameForVarDecl.hasBeenDeclared = true;
             }
-            var tryBlock = ts.createBlock(transformExpression(node.expression, transformer, possibleNameForVarDecl));
+            var tryBlock = ts.factory.createBlock(transformExpression(node.expression, transformer, possibleNameForVarDecl));
             var transformationBody = getTransformationBody(func, possibleNameForVarDecl, argName, node, transformer);
             var catchArg = argName ? isSynthIdentifier(argName) ? argName.identifier.text : argName.bindingPattern : "e";
-            var catchVariableDeclaration = ts.createVariableDeclaration(catchArg);
-            var catchClause = ts.createCatchClause(catchVariableDeclaration, ts.createBlock(transformationBody));
+            var catchVariableDeclaration = ts.factory.createVariableDeclaration(catchArg);
+            var catchClause = ts.factory.createCatchClause(catchVariableDeclaration, ts.factory.createBlock(transformationBody));
             /*
                 In order to avoid an implicit any, we will synthesize a type for the declaration using the unions of the types of both paths (try block and catch block)
             */
@@ -129108,16 +135395,16 @@ var ts;
                 var typeArray = possibleNameForVarDecl.types;
                 var unionType = transformer.checker.getUnionType(typeArray, 2 /* Subtype */);
                 var unionTypeNode = transformer.isInJSFile ? undefined : transformer.checker.typeToTypeNode(unionType, /*enclosingDeclaration*/ undefined, /*flags*/ undefined);
-                var varDecl = [ts.createVariableDeclaration(varDeclIdentifier, unionTypeNode)];
-                varDeclList = ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList(varDecl, 1 /* Let */));
+                var varDecl = [ts.factory.createVariableDeclaration(varDeclIdentifier, /*exclamationToken*/ undefined, unionTypeNode)];
+                varDeclList = ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList(varDecl, 1 /* Let */));
             }
-            var tryStatement = ts.createTry(tryBlock, catchClause, /*finallyBlock*/ undefined);
+            var tryStatement = ts.factory.createTryStatement(tryBlock, catchClause, /*finallyBlock*/ undefined);
             var destructuredResult = prevArgName && varDeclIdentifier && isSynthBindingPattern(prevArgName)
-                && ts.createVariableStatement(/* modifiers */ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(ts.getSynthesizedDeepCloneWithRenames(prevArgName.bindingPattern), /* type */ undefined, varDeclIdentifier)], 2 /* Const */));
+                && ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(ts.getSynthesizedDeepClone(prevArgName.bindingPattern), /*exclamationToken*/ undefined, /*type*/ undefined, varDeclIdentifier)], 2 /* Const */));
             return ts.compact([varDeclList, tryStatement, destructuredResult]);
         }
         function createUniqueSynthName(prevArgName) {
-            var renamedPrevArg = ts.createOptimisticUniqueName(prevArgName.identifier.text);
+            var renamedPrevArg = ts.factory.createUniqueName(prevArgName.identifier.text, 16 /* Optimistic */);
             return createSynthIdentifier(renamedPrevArg);
         }
         function transformThen(node, transformer, prevArgName) {
@@ -129126,12 +135413,12 @@ var ts;
             var transformationBody = getTransformationBody(onFulfilled, prevArgName, onFulfilledArgumentName, node, transformer);
             if (onRejected) {
                 var onRejectedArgumentName = getArgBindingName(onRejected, transformer);
-                var tryBlock = ts.createBlock(transformExpression(node.expression, transformer, onFulfilledArgumentName).concat(transformationBody));
+                var tryBlock = ts.factory.createBlock(transformExpression(node.expression, transformer, onFulfilledArgumentName).concat(transformationBody));
                 var transformationBody2 = getTransformationBody(onRejected, prevArgName, onRejectedArgumentName, node, transformer);
                 var catchArg = onRejectedArgumentName ? isSynthIdentifier(onRejectedArgumentName) ? onRejectedArgumentName.identifier.text : onRejectedArgumentName.bindingPattern : "e";
-                var catchVariableDeclaration = ts.createVariableDeclaration(catchArg);
-                var catchClause = ts.createCatchClause(catchVariableDeclaration, ts.createBlock(transformationBody2));
-                return [ts.createTry(tryBlock, catchClause, /* finallyBlock */ undefined)];
+                var catchVariableDeclaration = ts.factory.createVariableDeclaration(catchArg);
+                var catchClause = ts.factory.createCatchClause(catchVariableDeclaration, ts.factory.createBlock(transformationBody2));
+                return [ts.factory.createTryStatement(tryBlock, catchClause, /* finallyBlock */ undefined)];
             }
             return transformExpression(node.expression, transformer, onFulfilledArgumentName).concat(transformationBody);
         }
@@ -129140,48 +135427,49 @@ var ts;
          */
         function transformPromiseExpressionOfPropertyAccess(node, transformer, prevArgName) {
             if (shouldReturn(node, transformer)) {
-                return [ts.createReturn(ts.getSynthesizedDeepClone(node))];
+                return [ts.factory.createReturnStatement(ts.getSynthesizedDeepClone(node))];
             }
-            return createVariableOrAssignmentOrExpressionStatement(prevArgName, ts.createAwait(node), /*typeAnnotation*/ undefined);
+            return createVariableOrAssignmentOrExpressionStatement(prevArgName, ts.factory.createAwaitExpression(node), /*typeAnnotation*/ undefined);
         }
         function createVariableOrAssignmentOrExpressionStatement(variableName, rightHandSide, typeAnnotation) {
             if (!variableName || isEmptyBindingName(variableName)) {
                 // if there's no argName to assign to, there still might be side effects
-                return [ts.createExpressionStatement(rightHandSide)];
+                return [ts.factory.createExpressionStatement(rightHandSide)];
             }
             if (isSynthIdentifier(variableName) && variableName.hasBeenDeclared) {
                 // if the variable has already been declared, we don't need "let" or "const"
-                return [ts.createExpressionStatement(ts.createAssignment(ts.getSynthesizedDeepClone(variableName.identifier), rightHandSide))];
+                return [ts.factory.createExpressionStatement(ts.factory.createAssignment(ts.getSynthesizedDeepClone(variableName.identifier), rightHandSide))];
             }
             return [
-                ts.createVariableStatement(
-                /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                    ts.createVariableDeclaration(ts.getSynthesizedDeepClone(getNode(variableName)), typeAnnotation, rightHandSide)
+                ts.factory.createVariableStatement(
+                /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([
+                    ts.factory.createVariableDeclaration(ts.getSynthesizedDeepClone(getNode(variableName)), 
+                    /*exclamationToken*/ undefined, typeAnnotation, rightHandSide)
                 ], 2 /* Const */))
             ];
         }
         function maybeAnnotateAndReturn(expressionToReturn, typeAnnotation) {
             if (typeAnnotation && expressionToReturn) {
-                var name = ts.createOptimisticUniqueName("result");
+                var name = ts.factory.createUniqueName("result", 16 /* Optimistic */);
                 return __spreadArrays(createVariableOrAssignmentOrExpressionStatement(createSynthIdentifier(name), expressionToReturn, typeAnnotation), [
-                    ts.createReturn(name)
+                    ts.factory.createReturnStatement(name)
                 ]);
             }
-            return [ts.createReturn(expressionToReturn)];
+            return [ts.factory.createReturnStatement(expressionToReturn)];
         }
         // should be kept up to date with isFixablePromiseArgument in suggestionDiagnostics.ts
         function getTransformationBody(func, prevArgName, argName, parent, transformer) {
-            var _a, _b, _c, _d;
+            var _a, _b, _c, _d, _e;
             switch (func.kind) {
-                case 100 /* NullKeyword */:
+                case 103 /* NullKeyword */:
                     // do not produce a transformed statement for a null argument
                     break;
-                case 75 /* Identifier */: // identifier includes undefined
+                case 78 /* Identifier */: // identifier includes undefined
                     if (!argName) {
                         // undefined was argument passed to promise handler
                         break;
                     }
-                    var synthCall = ts.createCall(ts.getSynthesizedDeepClone(func), /*typeArguments*/ undefined, isSynthIdentifier(argName) ? [argName.identifier] : []);
+                    var synthCall = ts.factory.createCallExpression(ts.getSynthesizedDeepClone(func), /*typeArguments*/ undefined, isSynthIdentifier(argName) ? [argName.identifier] : []);
                     if (shouldReturn(parent, transformer)) {
                         return maybeAnnotateAndReturn(synthCall, (_a = parent.typeArguments) === null || _a === void 0 ? void 0 : _a[0]);
                     }
@@ -129192,27 +135480,29 @@ var ts;
                         return silentFail();
                     }
                     var returnType = callSignatures[0].getReturnType();
-                    var varDeclOrAssignment = createVariableOrAssignmentOrExpressionStatement(prevArgName, ts.createAwait(synthCall), (_b = parent.typeArguments) === null || _b === void 0 ? void 0 : _b[0]);
+                    var varDeclOrAssignment = createVariableOrAssignmentOrExpressionStatement(prevArgName, ts.factory.createAwaitExpression(synthCall), (_b = parent.typeArguments) === null || _b === void 0 ? void 0 : _b[0]);
                     if (prevArgName) {
                         prevArgName.types.push(returnType);
                     }
                     return varDeclOrAssignment;
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */: {
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */: {
                     var funcBody = func.body;
+                    var returnType_1 = (_c = getLastCallSignature(transformer.checker.getTypeAtLocation(func), transformer.checker)) === null || _c === void 0 ? void 0 : _c.getReturnType();
                     // Arrow functions with block bodies { } will enter this control flow
                     if (ts.isBlock(funcBody)) {
                         var refactoredStmts = [];
                         var seenReturnStatement = false;
-                        for (var _i = 0, _e = funcBody.statements; _i < _e.length; _i++) {
-                            var statement = _e[_i];
+                        for (var _i = 0, _f = funcBody.statements; _i < _f.length; _i++) {
+                            var statement = _f[_i];
                             if (ts.isReturnStatement(statement)) {
                                 seenReturnStatement = true;
                                 if (ts.isReturnStatementWithFixablePromiseHandler(statement)) {
                                     refactoredStmts = refactoredStmts.concat(getInnerTransformationBody(transformer, [statement], prevArgName));
                                 }
                                 else {
-                                    refactoredStmts.push.apply(refactoredStmts, maybeAnnotateAndReturn(statement.expression, (_c = parent.typeArguments) === null || _c === void 0 ? void 0 : _c[0]));
+                                    var possiblyAwaitedRightHandSide = returnType_1 && statement.expression ? getPossiblyAwaitedRightHandSide(transformer.checker, returnType_1, statement.expression) : statement.expression;
+                                    refactoredStmts.push.apply(refactoredStmts, maybeAnnotateAndReturn(possiblyAwaitedRightHandSide, (_d = parent.typeArguments) === null || _d === void 0 ? void 0 : _d[0]));
                                 }
                             }
                             else {
@@ -129224,24 +135514,26 @@ var ts;
                             : removeReturns(refactoredStmts, prevArgName, transformer, seenReturnStatement);
                     }
                     else {
-                        var innerRetStmts = ts.isFixablePromiseHandler(funcBody) ? [ts.createReturn(funcBody)] : ts.emptyArray;
+                        var innerRetStmts = ts.isFixablePromiseHandler(funcBody) ? [ts.factory.createReturnStatement(funcBody)] : ts.emptyArray;
                         var innerCbBody = getInnerTransformationBody(transformer, innerRetStmts, prevArgName);
                         if (innerCbBody.length > 0) {
                             return innerCbBody;
                         }
-                        var type_1 = transformer.checker.getTypeAtLocation(func);
-                        var returnType_1 = getLastCallSignature(type_1, transformer.checker).getReturnType();
-                        var rightHandSide = ts.getSynthesizedDeepClone(funcBody);
-                        var possiblyAwaitedRightHandSide = !!transformer.checker.getPromisedTypeOfPromise(returnType_1) ? ts.createAwait(rightHandSide) : rightHandSide;
-                        if (!shouldReturn(parent, transformer)) {
-                            var transformedStatement = createVariableOrAssignmentOrExpressionStatement(prevArgName, possiblyAwaitedRightHandSide, /*typeAnnotation*/ undefined);
-                            if (prevArgName) {
-                                prevArgName.types.push(returnType_1);
+                        if (returnType_1) {
+                            var possiblyAwaitedRightHandSide = getPossiblyAwaitedRightHandSide(transformer.checker, returnType_1, funcBody);
+                            if (!shouldReturn(parent, transformer)) {
+                                var transformedStatement = createVariableOrAssignmentOrExpressionStatement(prevArgName, possiblyAwaitedRightHandSide, /*typeAnnotation*/ undefined);
+                                if (prevArgName) {
+                                    prevArgName.types.push(returnType_1);
+                                }
+                                return transformedStatement;
+                            }
+                            else {
+                                return maybeAnnotateAndReturn(possiblyAwaitedRightHandSide, (_e = parent.typeArguments) === null || _e === void 0 ? void 0 : _e[0]);
                             }
-                            return transformedStatement;
                         }
                         else {
-                            return maybeAnnotateAndReturn(possiblyAwaitedRightHandSide, (_d = parent.typeArguments) === null || _d === void 0 ? void 0 : _d[0]);
+                            return silentFail();
                         }
                     }
                 }
@@ -129251,6 +135543,10 @@ var ts;
             }
             return ts.emptyArray;
         }
+        function getPossiblyAwaitedRightHandSide(checker, type, expr) {
+            var rightHandSide = ts.getSynthesizedDeepClone(expr);
+            return !!checker.getPromisedTypeOfPromise(type) ? ts.factory.createAwaitExpression(rightHandSide) : rightHandSide;
+        }
         function getLastCallSignature(type, checker) {
             var callSignatures = checker.getSignaturesOfType(type, 0 /* Call */);
             return ts.lastOrUndefined(callSignatures);
@@ -129261,12 +135557,12 @@ var ts;
                 var stmt = stmts_1[_i];
                 if (ts.isReturnStatement(stmt)) {
                     if (stmt.expression) {
-                        var possiblyAwaitedExpression = isPromiseTypedExpression(stmt.expression, transformer.checker) ? ts.createAwait(stmt.expression) : stmt.expression;
+                        var possiblyAwaitedExpression = isPromiseTypedExpression(stmt.expression, transformer.checker) ? ts.factory.createAwaitExpression(stmt.expression) : stmt.expression;
                         if (prevArgName === undefined) {
-                            ret.push(ts.createExpressionStatement(possiblyAwaitedExpression));
+                            ret.push(ts.factory.createExpressionStatement(possiblyAwaitedExpression));
                         }
                         else {
-                            ret.push(ts.createVariableStatement(/*modifiers*/ undefined, (ts.createVariableDeclarationList([ts.createVariableDeclaration(getNode(prevArgName), /*type*/ undefined, possiblyAwaitedExpression)], 2 /* Const */))));
+                            ret.push(ts.factory.createVariableStatement(/*modifiers*/ undefined, (ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(getNode(prevArgName), /*exclamationToken*/ undefined, /*type*/ undefined, possiblyAwaitedExpression)], 2 /* Const */))));
                         }
                     }
                 }
@@ -129276,7 +135572,7 @@ var ts;
             }
             // if block has no return statement, need to define prevArgName as undefined to prevent undeclared variables
             if (!seenReturnStatement && prevArgName !== undefined) {
-                ret.push(ts.createVariableStatement(/*modifiers*/ undefined, (ts.createVariableDeclarationList([ts.createVariableDeclaration(getNode(prevArgName), /*type*/ undefined, ts.createIdentifier("undefined"))], 2 /* Const */))));
+                ret.push(ts.factory.createVariableStatement(/*modifiers*/ undefined, (ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(getNode(prevArgName), /*exclamationToken*/ undefined, /*type*/ undefined, ts.factory.createIdentifier("undefined"))], 2 /* Const */))));
             }
             return ret;
         }
@@ -129371,7 +135667,7 @@ var ts;
             return bindingName.kind === 1 /* BindingPattern */;
         }
         function shouldReturn(expression, transformer) {
-            return !!expression.original && transformer.setOfExpressionsToReturn.has(ts.getNodeId(expression.original).toString());
+            return !!expression.original && transformer.setOfExpressionsToReturn.has(ts.getNodeId(expression.original));
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -129406,12 +135702,12 @@ var ts;
                 }
                 var importNode = ts.importFromModuleSpecifier(moduleSpecifier);
                 switch (importNode.kind) {
-                    case 253 /* ImportEqualsDeclaration */:
+                    case 260 /* ImportEqualsDeclaration */:
                         changes.replaceNode(importingFile, importNode, ts.makeImport(importNode.name, /*namedImports*/ undefined, moduleSpecifier, quotePreference));
                         break;
-                    case 196 /* CallExpression */:
+                    case 203 /* CallExpression */:
                         if (ts.isRequireCall(importNode, /*checkArgumentIsStringLiteralLike*/ false)) {
-                            changes.replaceNode(importingFile, importNode, ts.createPropertyAccess(ts.getSynthesizedDeepClone(importNode), "default"));
+                            changes.replaceNode(importingFile, importNode, ts.factory.createPropertyAccessExpression(ts.getSynthesizedDeepClone(importNode), "default"));
                         }
                         break;
                 }
@@ -129419,19 +135715,33 @@ var ts;
         }
         /** @returns Whether we converted a `module.exports =` to a default export. */
         function convertFileToEs6Module(sourceFile, checker, changes, target, quotePreference) {
-            var identifiers = { original: collectFreeIdentifiers(sourceFile), additional: ts.createMap() };
+            var identifiers = { original: collectFreeIdentifiers(sourceFile), additional: new ts.Set() };
             var exports = collectExportRenames(sourceFile, checker, identifiers);
             convertExportsAccesses(sourceFile, exports, changes);
             var moduleExportsChangedToDefault = false;
-            for (var _i = 0, _a = sourceFile.statements; _i < _a.length; _i++) {
+            var useSitesToUnqualify;
+            // Process variable statements first to collect use sites that need to be updated inside other transformations
+            for (var _i = 0, _a = ts.filter(sourceFile.statements, ts.isVariableStatement); _i < _a.length; _i++) {
                 var statement = _a[_i];
-                var moduleExportsChanged = convertStatement(sourceFile, statement, checker, changes, identifiers, target, exports, quotePreference);
+                var newUseSites = convertVariableStatement(sourceFile, statement, changes, checker, identifiers, target, quotePreference);
+                if (newUseSites) {
+                    ts.copyEntries(newUseSites, useSitesToUnqualify !== null && useSitesToUnqualify !== void 0 ? useSitesToUnqualify : (useSitesToUnqualify = new ts.Map()));
+                }
+            }
+            // `convertStatement` will delete entries from `useSitesToUnqualify` when containing statements are replaced
+            for (var _b = 0, _c = ts.filter(sourceFile.statements, function (s) { return !ts.isVariableStatement(s); }); _b < _c.length; _b++) {
+                var statement = _c[_b];
+                var moduleExportsChanged = convertStatement(sourceFile, statement, checker, changes, identifiers, target, exports, useSitesToUnqualify, quotePreference);
                 moduleExportsChangedToDefault = moduleExportsChangedToDefault || moduleExportsChanged;
             }
+            // Remaining use sites can be changed directly
+            useSitesToUnqualify === null || useSitesToUnqualify === void 0 ? void 0 : useSitesToUnqualify.forEach(function (replacement, original) {
+                changes.replaceNode(sourceFile, original, replacement);
+            });
             return moduleExportsChangedToDefault;
         }
         function collectExportRenames(sourceFile, checker, identifiers) {
-            var res = ts.createMap();
+            var res = new ts.Map();
             forEachExportReference(sourceFile, function (node) {
                 var _a = node.name, text = _a.text, originalKeywordKind = _a.originalKeywordKind;
                 if (!res.has(text) && (originalKeywordKind !== undefined && ts.isNonContextualKeyword(originalKeywordKind)
@@ -129448,7 +135758,7 @@ var ts;
                     return;
                 }
                 var text = node.name.text;
-                changes.replaceNode(sourceFile, node, ts.createIdentifier(exports.get(text) || text));
+                changes.replaceNode(sourceFile, node, ts.factory.createIdentifier(exports.get(text) || text));
             });
         }
         function forEachExportReference(sourceFile, cb) {
@@ -129460,24 +135770,24 @@ var ts;
                 node.forEachChild(recur);
             });
         }
-        function convertStatement(sourceFile, statement, checker, changes, identifiers, target, exports, quotePreference) {
+        function convertStatement(sourceFile, statement, checker, changes, identifiers, target, exports, useSitesToUnqualify, quotePreference) {
             switch (statement.kind) {
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     convertVariableStatement(sourceFile, statement, changes, checker, identifiers, target, quotePreference);
                     return false;
-                case 226 /* ExpressionStatement */: {
+                case 233 /* ExpressionStatement */: {
                     var expression = statement.expression;
                     switch (expression.kind) {
-                        case 196 /* CallExpression */: {
+                        case 203 /* CallExpression */: {
                             if (ts.isRequireCall(expression, /*checkArgumentIsStringLiteralLike*/ true)) {
                                 // For side-effecting require() call, just make a side-effecting import.
                                 changes.replaceNode(sourceFile, statement, ts.makeImport(/*name*/ undefined, /*namedImports*/ undefined, expression.arguments[0], quotePreference));
                             }
                             return false;
                         }
-                        case 209 /* BinaryExpression */: {
+                        case 216 /* BinaryExpression */: {
                             var operatorToken = expression.operatorToken;
-                            return operatorToken.kind === 62 /* EqualsToken */ && convertAssignment(sourceFile, checker, expression, changes, exports);
+                            return operatorToken.kind === 62 /* EqualsToken */ && convertAssignment(sourceFile, checker, expression, changes, exports, useSitesToUnqualify);
                         }
                     }
                 }
@@ -129489,17 +135799,17 @@ var ts;
         function convertVariableStatement(sourceFile, statement, changes, checker, identifiers, target, quotePreference) {
             var declarationList = statement.declarationList;
             var foundImport = false;
-            var newNodes = ts.flatMap(declarationList.declarations, function (decl) {
+            var converted = ts.map(declarationList.declarations, function (decl) {
                 var name = decl.name, initializer = decl.initializer;
                 if (initializer) {
                     if (ts.isExportsOrModuleExportsOrAlias(sourceFile, initializer)) {
                         // `const alias = module.exports;` can be removed.
                         foundImport = true;
-                        return [];
+                        return convertedImports([]);
                     }
                     else if (ts.isRequireCall(initializer, /*checkArgumentIsStringLiteralLike*/ true)) {
                         foundImport = true;
-                        return convertSingleImport(sourceFile, name, initializer.arguments[0], changes, checker, identifiers, target, quotePreference);
+                        return convertSingleImport(name, initializer.arguments[0], checker, identifiers, target, quotePreference);
                     }
                     else if (ts.isPropertyAccessExpression(initializer) && ts.isRequireCall(initializer.expression, /*checkArgumentIsStringLiteralLike*/ true)) {
                         foundImport = true;
@@ -129507,33 +135817,40 @@ var ts;
                     }
                 }
                 // Move it out to its own variable statement. (This will not be used if `!foundImport`)
-                return ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([decl], declarationList.flags));
+                return convertedImports([ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList([decl], declarationList.flags))]);
             });
             if (foundImport) {
                 // useNonAdjustedEndPosition to ensure we don't eat the newline after the statement.
-                changes.replaceNodeWithNodes(sourceFile, statement, newNodes);
+                changes.replaceNodeWithNodes(sourceFile, statement, ts.flatMap(converted, function (c) { return c.newImports; }));
+                var combinedUseSites_1;
+                ts.forEach(converted, function (c) {
+                    if (c.useSitesToUnqualify) {
+                        ts.copyEntries(c.useSitesToUnqualify, combinedUseSites_1 !== null && combinedUseSites_1 !== void 0 ? combinedUseSites_1 : (combinedUseSites_1 = new ts.Map()));
+                    }
+                });
+                return combinedUseSites_1;
             }
         }
         /** Converts `const name = require("moduleSpecifier").propertyName` */
         function convertPropertyAccessImport(name, propertyName, moduleSpecifier, identifiers, quotePreference) {
             switch (name.kind) {
-                case 189 /* ObjectBindingPattern */:
-                case 190 /* ArrayBindingPattern */: {
+                case 196 /* ObjectBindingPattern */:
+                case 197 /* ArrayBindingPattern */: {
                     // `const [a, b] = require("c").d` --> `import { d } from "c"; const [a, b] = d;`
                     var tmp = makeUniqueName(propertyName, identifiers);
-                    return [
+                    return convertedImports([
                         makeSingleImport(tmp, propertyName, moduleSpecifier, quotePreference),
-                        makeConst(/*modifiers*/ undefined, name, ts.createIdentifier(tmp)),
-                    ];
+                        makeConst(/*modifiers*/ undefined, name, ts.factory.createIdentifier(tmp)),
+                    ]);
                 }
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     // `const a = require("b").c` --> `import { c as a } from "./b";
-                    return [makeSingleImport(name.text, propertyName, moduleSpecifier, quotePreference)];
+                    return convertedImports([makeSingleImport(name.text, propertyName, moduleSpecifier, quotePreference)]);
                 default:
                     return ts.Debug.assertNever(name, "Convert to ES6 module got invalid syntax form " + name.kind);
             }
         }
-        function convertAssignment(sourceFile, checker, assignment, changes, exports) {
+        function convertAssignment(sourceFile, checker, assignment, changes, exports, useSitesToUnqualify) {
             var left = assignment.left, right = assignment.right;
             if (!ts.isPropertyAccessExpression(left)) {
                 return false;
@@ -129544,7 +135861,7 @@ var ts;
                     changes.delete(sourceFile, assignment.parent);
                 }
                 else {
-                    var replacement = ts.isObjectLiteralExpression(right) ? tryChangeModuleExportsObject(right)
+                    var replacement = ts.isObjectLiteralExpression(right) ? tryChangeModuleExportsObject(right, useSitesToUnqualify)
                         : ts.isRequireCall(right, /*checkArgumentIsStringLiteralLike*/ true) ? convertReExportAll(right.arguments[0], checker)
                             : undefined;
                     if (replacement) {
@@ -129566,20 +135883,20 @@ var ts;
          * Convert `module.exports = { ... }` to individual exports..
          * We can't always do this if the module has interesting members -- then it will be a default export instead.
          */
-        function tryChangeModuleExportsObject(object) {
+        function tryChangeModuleExportsObject(object, useSitesToUnqualify) {
             var statements = ts.mapAllOrFail(object.properties, function (prop) {
                 switch (prop.kind) {
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
                     // TODO: Maybe we should handle this? See fourslash test `refactorConvertToEs6Module_export_object_shorthand.ts`.
                     // falls through
-                    case 282 /* ShorthandPropertyAssignment */:
-                    case 283 /* SpreadAssignment */:
+                    case 289 /* ShorthandPropertyAssignment */:
+                    case 290 /* SpreadAssignment */:
                         return undefined;
-                    case 281 /* PropertyAssignment */:
-                        return !ts.isIdentifier(prop.name) ? undefined : convertExportsDotXEquals_replaceNode(prop.name.text, prop.initializer);
-                    case 161 /* MethodDeclaration */:
-                        return !ts.isIdentifier(prop.name) ? undefined : functionExpressionToDeclaration(prop.name.text, [ts.createToken(89 /* ExportKeyword */)], prop);
+                    case 288 /* PropertyAssignment */:
+                        return !ts.isIdentifier(prop.name) ? undefined : convertExportsDotXEquals_replaceNode(prop.name.text, prop.initializer, useSitesToUnqualify);
+                    case 165 /* MethodDeclaration */:
+                        return !ts.isIdentifier(prop.name) ? undefined : functionExpressionToDeclaration(prop.name.text, [ts.factory.createToken(92 /* ExportKeyword */)], prop, useSitesToUnqualify);
                     default:
                         ts.Debug.assertNever(prop, "Convert to ES6 got invalid prop kind " + prop.kind);
                 }
@@ -129597,7 +135914,7 @@ var ts;
                 */
                 var newNodes = [
                     makeConst(/*modifiers*/ undefined, rename, assignment.right),
-                    makeExportDeclaration([ts.createExportSpecifier(rename, text)]),
+                    makeExportDeclaration([ts.factory.createExportSpecifier(rename, text)]),
                 ];
                 changes.replaceNodeWithNodes(sourceFile, assignment.parent, newNodes);
             }
@@ -129609,9 +135926,9 @@ var ts;
             // `module.exports = require("x");` ==> `export * from "x"; export { default } from "x";`
             var moduleSpecifier = reExported.text;
             var moduleSymbol = checker.getSymbolAtLocation(reExported);
-            var exports = moduleSymbol ? moduleSymbol.exports : ts.emptyUnderscoreEscapedMap;
-            return exports.has("export=") ? [[reExportDefault(moduleSpecifier)], true] :
-                !exports.has("default") ? [[reExportStar(moduleSpecifier)], false] :
+            var exports = moduleSymbol ? moduleSymbol.exports : ts.emptyMap;
+            return exports.has("export=" /* ExportEquals */) ? [[reExportDefault(moduleSpecifier)], true] :
+                !exports.has("default" /* Default */) ? [[reExportStar(moduleSpecifier)], false] :
                     // If there's some non-default export, must include both `export *` and `export default`.
                     exports.size > 1 ? [[reExportStar(moduleSpecifier), reExportDefault(moduleSpecifier)], true] : [[reExportDefault(moduleSpecifier)], true];
         }
@@ -129619,14 +135936,14 @@ var ts;
             return makeExportDeclaration(/*exportClause*/ undefined, moduleSpecifier);
         }
         function reExportDefault(moduleSpecifier) {
-            return makeExportDeclaration([ts.createExportSpecifier(/*propertyName*/ undefined, "default")], moduleSpecifier);
+            return makeExportDeclaration([ts.factory.createExportSpecifier(/*propertyName*/ undefined, "default")], moduleSpecifier);
         }
         function convertExportsPropertyAssignment(_a, sourceFile, changes) {
             var left = _a.left, right = _a.right, parent = _a.parent;
             var name = left.name.text;
             if ((ts.isFunctionExpression(right) || ts.isArrowFunction(right) || ts.isClassExpression(right)) && (!right.name || right.name.text === name)) {
                 // `exports.f = function() {}` -> `export function f() {}` -- Replace `exports.f = ` with `export `, and insert the name after `function`.
-                changes.replaceRange(sourceFile, { pos: left.getStart(sourceFile), end: right.getStart(sourceFile) }, ts.createToken(89 /* ExportKeyword */), { suffix: " " });
+                changes.replaceRange(sourceFile, { pos: left.getStart(sourceFile), end: right.getStart(sourceFile) }, ts.factory.createToken(92 /* ExportKeyword */), { suffix: " " });
                 if (!right.name)
                     changes.insertName(sourceFile, right, name);
                 var semi = ts.findChildOfKind(parent, 26 /* SemicolonToken */, sourceFile);
@@ -129635,14 +135952,14 @@ var ts;
             }
             else {
                 // `exports.f = function g() {}` -> `export const f = function g() {}` -- just replace `exports.` with `export const `
-                changes.replaceNodeRangeWithNodes(sourceFile, left.expression, ts.findChildOfKind(left, 24 /* DotToken */, sourceFile), [ts.createToken(89 /* ExportKeyword */), ts.createToken(81 /* ConstKeyword */)], { joiner: " ", suffix: " " });
+                changes.replaceNodeRangeWithNodes(sourceFile, left.expression, ts.findChildOfKind(left, 24 /* DotToken */, sourceFile), [ts.factory.createToken(92 /* ExportKeyword */), ts.factory.createToken(84 /* ConstKeyword */)], { joiner: " ", suffix: " " });
             }
         }
         // TODO: GH#22492 this will cause an error if a change has been made inside the body of the node.
-        function convertExportsDotXEquals_replaceNode(name, exported) {
-            var modifiers = [ts.createToken(89 /* ExportKeyword */)];
+        function convertExportsDotXEquals_replaceNode(name, exported, useSitesToUnqualify) {
+            var modifiers = [ts.factory.createToken(92 /* ExportKeyword */)];
             switch (exported.kind) {
-                case 201 /* FunctionExpression */: {
+                case 208 /* FunctionExpression */: {
                     var expressionName = exported.name;
                     if (expressionName && expressionName.text !== name) {
                         // `exports.f = function g() {}` -> `export const f = function g() {}`
@@ -129650,18 +135967,35 @@ var ts;
                     }
                 }
                 // falls through
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     // `exports.f = function() {}` --> `export function f() {}`
-                    return functionExpressionToDeclaration(name, modifiers, exported);
-                case 214 /* ClassExpression */:
+                    return functionExpressionToDeclaration(name, modifiers, exported, useSitesToUnqualify);
+                case 221 /* ClassExpression */:
                     // `exports.C = class {}` --> `export class C {}`
-                    return classExpressionToDeclaration(name, modifiers, exported);
+                    return classExpressionToDeclaration(name, modifiers, exported, useSitesToUnqualify);
                 default:
                     return exportConst();
             }
             function exportConst() {
                 // `exports.x = 0;` --> `export const x = 0;`
-                return makeConst(modifiers, ts.createIdentifier(name), exported); // TODO: GH#18217
+                return makeConst(modifiers, ts.factory.createIdentifier(name), replaceImportUseSites(exported, useSitesToUnqualify)); // TODO: GH#18217
+            }
+        }
+        function replaceImportUseSites(nodeOrNodes, useSitesToUnqualify) {
+            if (!useSitesToUnqualify || !ts.some(ts.arrayFrom(useSitesToUnqualify.keys()), function (original) { return ts.rangeContainsRange(nodeOrNodes, original); })) {
+                return nodeOrNodes;
+            }
+            return ts.isArray(nodeOrNodes)
+                ? ts.getSynthesizedDeepClonesWithReplacements(nodeOrNodes, /*includeTrivia*/ true, replaceNode)
+                : ts.getSynthesizedDeepCloneWithReplacements(nodeOrNodes, /*includeTrivia*/ true, replaceNode);
+            function replaceNode(original) {
+                // We are replacing `mod.SomeExport` wih `SomeExport`, so we only need to look at PropertyAccessExpressions
+                if (original.kind === 201 /* PropertyAccessExpression */) {
+                    var replacement = useSitesToUnqualify.get(original);
+                    // Remove entry from `useSitesToUnqualify` so the refactor knows it's taken care of by the parent statement we're replacing
+                    useSitesToUnqualify.delete(original);
+                    return replacement;
+                }
             }
         }
         /**
@@ -129669,9 +136003,9 @@ var ts;
          * Returns nodes that will replace the variable declaration for the commonjs import.
          * May also make use `changes` to remove qualifiers at the use sites of imports, to change `mod.x` to `x`.
          */
-        function convertSingleImport(file, name, moduleSpecifier, changes, checker, identifiers, target, quotePreference) {
+        function convertSingleImport(name, moduleSpecifier, checker, identifiers, target, quotePreference) {
             switch (name.kind) {
-                case 189 /* ObjectBindingPattern */: {
+                case 196 /* ObjectBindingPattern */: {
                     var importSpecifiers = ts.mapAllOrFail(name.elements, function (e) {
                         return e.dotDotDotToken || e.initializer || e.propertyName && !ts.isIdentifier(e.propertyName) || !ts.isIdentifier(e.name)
                             ? undefined
@@ -129680,23 +136014,23 @@ var ts;
                             : makeImportSpecifier(e.propertyName && e.propertyName.text, e.name.text);
                     });
                     if (importSpecifiers) {
-                        return [ts.makeImport(/*name*/ undefined, importSpecifiers, moduleSpecifier, quotePreference)];
+                        return convertedImports([ts.makeImport(/*name*/ undefined, importSpecifiers, moduleSpecifier, quotePreference)]);
                     }
                 }
                 // falls through -- object destructuring has an interesting pattern and must be a variable declaration
-                case 190 /* ArrayBindingPattern */: {
+                case 197 /* ArrayBindingPattern */: {
                     /*
                     import x from "x";
                     const [a, b, c] = x;
                     */
                     var tmp = makeUniqueName(codefix.moduleSpecifierToValidIdentifier(moduleSpecifier.text, target), identifiers);
-                    return [
-                        ts.makeImport(ts.createIdentifier(tmp), /*namedImports*/ undefined, moduleSpecifier, quotePreference),
-                        makeConst(/*modifiers*/ undefined, ts.getSynthesizedDeepClone(name), ts.createIdentifier(tmp)),
-                    ];
+                    return convertedImports([
+                        ts.makeImport(ts.factory.createIdentifier(tmp), /*namedImports*/ undefined, moduleSpecifier, quotePreference),
+                        makeConst(/*modifiers*/ undefined, ts.getSynthesizedDeepClone(name), ts.factory.createIdentifier(tmp)),
+                    ]);
                 }
-                case 75 /* Identifier */:
-                    return convertSingleIdentifierImport(file, name, moduleSpecifier, changes, checker, identifiers, quotePreference);
+                case 78 /* Identifier */:
+                    return convertSingleIdentifierImport(name, moduleSpecifier, checker, identifiers, quotePreference);
                 default:
                     return ts.Debug.assertNever(name, "Convert to ES6 module got invalid name kind " + name.kind);
             }
@@ -129705,12 +136039,13 @@ var ts;
          * Convert `import x = require("x").`
          * Also converts uses like `x.y()` to `y()` and uses a named import.
          */
-        function convertSingleIdentifierImport(file, name, moduleSpecifier, changes, checker, identifiers, quotePreference) {
+        function convertSingleIdentifierImport(name, moduleSpecifier, checker, identifiers, quotePreference) {
             var nameSymbol = checker.getSymbolAtLocation(name);
             // Maps from module property name to name actually used. (The same if there isn't shadowing.)
-            var namedBindingsNames = ts.createMap();
+            var namedBindingsNames = new ts.Map();
             // True if there is some non-property use like `x()` or `f(x)`.
             var needDefaultImport = false;
+            var useSitesToUnqualify;
             for (var _i = 0, _a = identifiers.original.get(name.text); _i < _a.length; _i++) {
                 var use = _a[_i];
                 if (checker.getSymbolAtLocation(use) !== nameSymbol || use === name) {
@@ -129726,7 +136061,7 @@ var ts;
                         idName = makeUniqueName(propertyName, identifiers);
                         namedBindingsNames.set(propertyName, idName);
                     }
-                    changes.replaceNode(file, parent, ts.createIdentifier(idName));
+                    (useSitesToUnqualify !== null && useSitesToUnqualify !== void 0 ? useSitesToUnqualify : (useSitesToUnqualify = new ts.Map())).set(parent, ts.factory.createIdentifier(idName));
                 }
                 else {
                     needDefaultImport = true;
@@ -129734,20 +136069,20 @@ var ts;
             }
             var namedBindings = namedBindingsNames.size === 0 ? undefined : ts.arrayFrom(ts.mapIterator(namedBindingsNames.entries(), function (_a) {
                 var propertyName = _a[0], idName = _a[1];
-                return ts.createImportSpecifier(propertyName === idName ? undefined : ts.createIdentifier(propertyName), ts.createIdentifier(idName));
+                return ts.factory.createImportSpecifier(propertyName === idName ? undefined : ts.factory.createIdentifier(propertyName), ts.factory.createIdentifier(idName));
             }));
             if (!namedBindings) {
                 // If it was unused, ensure that we at least import *something*.
                 needDefaultImport = true;
             }
-            return [ts.makeImport(needDefaultImport ? ts.getSynthesizedDeepClone(name) : undefined, namedBindings, moduleSpecifier, quotePreference)];
+            return convertedImports([ts.makeImport(needDefaultImport ? ts.getSynthesizedDeepClone(name) : undefined, namedBindings, moduleSpecifier, quotePreference)], useSitesToUnqualify);
         }
         // Identifiers helpers
         function makeUniqueName(name, identifiers) {
             while (identifiers.original.has(name) || identifiers.additional.has(name)) {
                 name = "_" + name;
             }
-            identifiers.additional.set(name, true);
+            identifiers.additional.add(name);
             return name;
         }
         function collectFreeIdentifiers(file) {
@@ -129767,40 +136102,47 @@ var ts;
         function isFreeIdentifier(node) {
             var parent = node.parent;
             switch (parent.kind) {
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     return parent.name !== node;
-                case 191 /* BindingElement */:
+                case 198 /* BindingElement */:
                     return parent.propertyName !== node;
-                case 258 /* ImportSpecifier */:
+                case 265 /* ImportSpecifier */:
                     return parent.propertyName !== node;
                 default:
                     return true;
             }
         }
         // Node helpers
-        function functionExpressionToDeclaration(name, additionalModifiers, fn) {
-            return ts.createFunctionDeclaration(ts.getSynthesizedDeepClones(fn.decorators), // TODO: GH#19915 Don't think this is even legal.
-            ts.concatenate(additionalModifiers, ts.getSynthesizedDeepClones(fn.modifiers)), ts.getSynthesizedDeepClone(fn.asteriskToken), name, ts.getSynthesizedDeepClones(fn.typeParameters), ts.getSynthesizedDeepClones(fn.parameters), ts.getSynthesizedDeepClone(fn.type), ts.convertToFunctionBody(ts.getSynthesizedDeepClone(fn.body)));
+        function functionExpressionToDeclaration(name, additionalModifiers, fn, useSitesToUnqualify) {
+            return ts.factory.createFunctionDeclaration(ts.getSynthesizedDeepClones(fn.decorators), // TODO: GH#19915 Don't think this is even legal.
+            ts.concatenate(additionalModifiers, ts.getSynthesizedDeepClones(fn.modifiers)), ts.getSynthesizedDeepClone(fn.asteriskToken), name, ts.getSynthesizedDeepClones(fn.typeParameters), ts.getSynthesizedDeepClones(fn.parameters), ts.getSynthesizedDeepClone(fn.type), ts.factory.converters.convertToFunctionBlock(replaceImportUseSites(fn.body, useSitesToUnqualify)));
         }
-        function classExpressionToDeclaration(name, additionalModifiers, cls) {
-            return ts.createClassDeclaration(ts.getSynthesizedDeepClones(cls.decorators), // TODO: GH#19915 Don't think this is even legal.
-            ts.concatenate(additionalModifiers, ts.getSynthesizedDeepClones(cls.modifiers)), name, ts.getSynthesizedDeepClones(cls.typeParameters), ts.getSynthesizedDeepClones(cls.heritageClauses), ts.getSynthesizedDeepClones(cls.members));
+        function classExpressionToDeclaration(name, additionalModifiers, cls, useSitesToUnqualify) {
+            return ts.factory.createClassDeclaration(ts.getSynthesizedDeepClones(cls.decorators), // TODO: GH#19915 Don't think this is even legal.
+            ts.concatenate(additionalModifiers, ts.getSynthesizedDeepClones(cls.modifiers)), name, ts.getSynthesizedDeepClones(cls.typeParameters), ts.getSynthesizedDeepClones(cls.heritageClauses), replaceImportUseSites(cls.members, useSitesToUnqualify));
         }
         function makeSingleImport(localName, propertyName, moduleSpecifier, quotePreference) {
             return propertyName === "default"
-                ? ts.makeImport(ts.createIdentifier(localName), /*namedImports*/ undefined, moduleSpecifier, quotePreference)
+                ? ts.makeImport(ts.factory.createIdentifier(localName), /*namedImports*/ undefined, moduleSpecifier, quotePreference)
                 : ts.makeImport(/*name*/ undefined, [makeImportSpecifier(propertyName, localName)], moduleSpecifier, quotePreference);
         }
         function makeImportSpecifier(propertyName, name) {
-            return ts.createImportSpecifier(propertyName !== undefined && propertyName !== name ? ts.createIdentifier(propertyName) : undefined, ts.createIdentifier(name));
+            return ts.factory.createImportSpecifier(propertyName !== undefined && propertyName !== name ? ts.factory.createIdentifier(propertyName) : undefined, ts.factory.createIdentifier(name));
         }
         function makeConst(modifiers, name, init) {
-            return ts.createVariableStatement(modifiers, ts.createVariableDeclarationList([ts.createVariableDeclaration(name, /*type*/ undefined, init)], 2 /* Const */));
+            return ts.factory.createVariableStatement(modifiers, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(name, /*exclamationToken*/ undefined, /*type*/ undefined, init)], 2 /* Const */));
         }
         function makeExportDeclaration(exportSpecifiers, moduleSpecifier) {
-            return ts.createExportDeclaration(
+            return ts.factory.createExportDeclaration(
             /*decorators*/ undefined, 
-            /*modifiers*/ undefined, exportSpecifiers && ts.createNamedExports(exportSpecifiers), moduleSpecifier === undefined ? undefined : ts.createLiteral(moduleSpecifier));
+            /*modifiers*/ undefined, 
+            /*isTypeOnly*/ false, exportSpecifiers && ts.factory.createNamedExports(exportSpecifiers), moduleSpecifier === undefined ? undefined : ts.factory.createStringLiteral(moduleSpecifier));
+        }
+        function convertedImports(newImports, useSitesToUnqualify) {
+            return {
+                newImports: newImports,
+                useSitesToUnqualify: useSitesToUnqualify
+            };
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -129836,7 +136178,7 @@ var ts;
         }
         function doChange(changeTracker, sourceFile, qualifiedName) {
             var rightText = qualifiedName.right.text;
-            var replacement = ts.createIndexedAccessTypeNode(ts.createTypeReferenceNode(qualifiedName.left, /*typeArguments*/ undefined), ts.createLiteralTypeNode(ts.createLiteral(rightText)));
+            var replacement = ts.factory.createIndexedAccessTypeNode(ts.factory.createTypeReferenceNode(qualifiedName.left, /*typeArguments*/ undefined), ts.factory.createLiteralTypeNode(ts.factory.createStringLiteral(rightText)));
             changeTracker.replaceNode(sourceFile, qualifiedName, replacement);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -129858,10 +136200,10 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var fixedExportDeclarations = ts.createMap();
+                var fixedExportDeclarations = new ts.Map();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
                     var exportSpecifier = getExportSpecifierForDiagnosticSpan(diag, context.sourceFile);
-                    if (exportSpecifier && !ts.addToSeen(fixedExportDeclarations, ts.getNodeId(exportSpecifier.parent.parent))) {
+                    if (exportSpecifier && ts.addToSeen(fixedExportDeclarations, ts.getNodeId(exportSpecifier.parent.parent))) {
                         fixSingleExportDeclaration(changes, exportSpecifier, context);
                     }
                 });
@@ -129878,17 +136220,19 @@ var ts;
             var exportDeclaration = exportClause.parent;
             var typeExportSpecifiers = getTypeExportSpecifiers(exportSpecifier, context);
             if (typeExportSpecifiers.length === exportClause.elements.length) {
-                changes.replaceNode(context.sourceFile, exportDeclaration, ts.updateExportDeclaration(exportDeclaration, exportDeclaration.decorators, exportDeclaration.modifiers, exportClause, exportDeclaration.moduleSpecifier, 
-                /*isTypeOnly*/ true));
+                changes.insertModifierBefore(context.sourceFile, 149 /* TypeKeyword */, exportClause);
             }
             else {
-                var valueExportDeclaration = ts.updateExportDeclaration(exportDeclaration, exportDeclaration.decorators, exportDeclaration.modifiers, ts.updateNamedExports(exportClause, ts.filter(exportClause.elements, function (e) { return !ts.contains(typeExportSpecifiers, e); })), exportDeclaration.moduleSpecifier
-                /*isTypeOnly*/ false);
-                var typeExportDeclaration = ts.createExportDeclaration(
+                var valueExportDeclaration = ts.factory.updateExportDeclaration(exportDeclaration, exportDeclaration.decorators, exportDeclaration.modifiers
+                /*isTypeOnly*/ false, ts.factory.updateNamedExports(exportClause, ts.filter(exportClause.elements, function (e) { return !ts.contains(typeExportSpecifiers, e); })), exportDeclaration.moduleSpecifier);
+                var typeExportDeclaration = ts.factory.createExportDeclaration(
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, ts.createNamedExports(typeExportSpecifiers), exportDeclaration.moduleSpecifier, 
-                /*isTypeOnly*/ true);
-                changes.replaceNode(context.sourceFile, exportDeclaration, valueExportDeclaration);
+                /*modifiers*/ undefined, 
+                /*isTypeOnly*/ true, ts.factory.createNamedExports(typeExportSpecifiers), exportDeclaration.moduleSpecifier);
+                changes.replaceNode(context.sourceFile, exportDeclaration, valueExportDeclaration, {
+                    leadingTriviaOption: ts.textChanges.LeadingTriviaOption.IncludeAll,
+                    trailingTriviaOption: ts.textChanges.TrailingTriviaOption.Exclude
+                });
                 changes.insertNodeAfter(context.sourceFile, exportDeclaration, typeExportDeclaration);
             }
         }
@@ -129944,17 +136288,64 @@ var ts;
             // `import type foo, { Bar }` is not allowed, so move `foo` to new declaration
             if (importClause.name && importClause.namedBindings) {
                 changes.deleteNodeRangeExcludingEnd(context.sourceFile, importClause.name, importDeclaration.importClause.namedBindings);
-                changes.insertNodeBefore(context.sourceFile, importDeclaration, ts.updateImportDeclaration(importDeclaration, 
+                changes.insertNodeBefore(context.sourceFile, importDeclaration, ts.factory.updateImportDeclaration(importDeclaration, 
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, ts.createImportClause(importClause.name, 
-                /*namedBindings*/ undefined
-                /*isTypeOnly*/ true), importDeclaration.moduleSpecifier));
+                /*modifiers*/ undefined, ts.factory.createImportClause(
+                /*isTypeOnly*/ true, importClause.name
+                /*namedBindings*/ undefined), importDeclaration.moduleSpecifier));
             }
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixId = "convertLiteralTypeToMappedType";
+        var errorCodes = [ts.Diagnostics._0_only_refers_to_a_type_but_is_being_used_as_a_value_here_Did_you_mean_to_use_1_in_0.code];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            getCodeActions: function (context) {
+                var sourceFile = context.sourceFile, span = context.span;
+                var info = getInfo(sourceFile, span.start);
+                if (!info) {
+                    return undefined;
+                }
+                var name = info.name, constraint = info.constraint;
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, sourceFile, info); });
+                return [codefix.createCodeFixAction(fixId, changes, [ts.Diagnostics.Convert_0_to_1_in_0, constraint, name], fixId, ts.Diagnostics.Convert_all_type_literals_to_mapped_type)];
+            },
+            fixIds: [fixId],
+            getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
+                var info = getInfo(diag.file, diag.start);
+                if (info) {
+                    doChange(changes, diag.file, info);
+                }
+            }); }
+        });
+        function getInfo(sourceFile, pos) {
+            var token = ts.getTokenAtPosition(sourceFile, pos);
+            if (ts.isIdentifier(token)) {
+                var propertySignature = ts.cast(token.parent.parent, ts.isPropertySignature);
+                var propertyName = token.getText(sourceFile);
+                return {
+                    container: ts.cast(propertySignature.parent, ts.isTypeLiteralNode),
+                    typeNode: propertySignature.type,
+                    constraint: propertyName,
+                    name: propertyName === "K" ? "P" : "K",
+                };
+            }
+            return undefined;
+        }
+        function doChange(changes, sourceFile, _a) {
+            var container = _a.container, typeNode = _a.typeNode, constraint = _a.constraint, name = _a.name;
+            changes.replaceNode(sourceFile, container, ts.factory.createMappedTypeNode(/*readonlyToken*/ undefined, ts.factory.createTypeParameterDeclaration(name, ts.factory.createTypeReferenceNode(constraint)), /*nameType*/ undefined, /*questionToken*/ undefined, typeNode));
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var codefix;
     (function (codefix) {
@@ -129975,7 +136366,7 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var seenClassDeclarations = ts.createMap();
+                var seenClassDeclarations = new ts.Map();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
                     var classDeclaration = getClass(diag.file, diag.start);
                     if (ts.addToSeen(seenClassDeclarations, ts.getNodeId(classDeclaration))) {
@@ -129991,7 +136382,7 @@ var ts;
             return ts.Debug.checkDefined(ts.getContainingClass(ts.getTokenAtPosition(sourceFile, pos)), "There should be a containing class");
         }
         function symbolPointsToNonPrivateMember(symbol) {
-            return !symbol.valueDeclaration || !(ts.getModifierFlags(symbol.valueDeclaration) & 8 /* Private */);
+            return !symbol.valueDeclaration || !(ts.getEffectiveModifierFlags(symbol.valueDeclaration) & 8 /* Private */);
         }
         function addMissingDeclarations(context, implementedTypeNode, sourceFile, classDeclaration, changeTracker, preferences) {
             var checker = context.program.getTypeChecker();
@@ -130010,7 +136401,7 @@ var ts;
                 createMissingIndexSignatureDeclaration(implementedType, 0 /* String */);
             }
             var importAdder = codefix.createImportAdder(sourceFile, context.program, preferences, context.host);
-            codefix.createMissingMemberNodes(classDeclaration, nonPrivateAndNotExistedInHeritageClauseMembers, context, preferences, importAdder, function (member) { return insertInterfaceMemberNode(sourceFile, classDeclaration, member); });
+            codefix.createMissingMemberNodes(classDeclaration, nonPrivateAndNotExistedInHeritageClauseMembers, sourceFile, context, preferences, importAdder, function (member) { return insertInterfaceMemberNode(sourceFile, classDeclaration, member); });
             importAdder.writeFixes(changeTracker);
             function createMissingIndexSignatureDeclaration(type, kind) {
                 var indexInfoOfKind = checker.getIndexInfoOfType(type, kind);
@@ -130058,7 +136449,7 @@ var ts;
             errorCodes: errorCodes,
             getCodeActions: function (context) {
                 var errorCode = context.errorCode, preferences = context.preferences, sourceFile = context.sourceFile, span = context.span;
-                var info = getFixesInfo(context, errorCode, span.start);
+                var info = getFixesInfo(context, errorCode, span.start, /*useAutoImportProvider*/ true);
                 if (!info)
                     return undefined;
                 var fixes = info.fixes, symbolName = info.symbolName;
@@ -130068,22 +136459,26 @@ var ts;
             fixIds: [importFixId],
             getAllCodeActions: function (context) {
                 var sourceFile = context.sourceFile, program = context.program, preferences = context.preferences, host = context.host;
-                var importAdder = createImportAdder(sourceFile, program, preferences, host);
+                var importAdder = createImportAdderWorker(sourceFile, program, /*useAutoImportProvider*/ true, preferences, host);
                 codefix.eachDiagnostic(context, errorCodes, function (diag) { return importAdder.addImportFromDiagnostic(diag, context); });
                 return codefix.createCombinedCodeActions(ts.textChanges.ChangeTracker.with(context, importAdder.writeFixes));
             },
         });
         function createImportAdder(sourceFile, program, preferences, host) {
+            return createImportAdderWorker(sourceFile, program, /*useAutoImportProvider*/ false, preferences, host);
+        }
+        codefix.createImportAdder = createImportAdder;
+        function createImportAdderWorker(sourceFile, program, useAutoImportProvider, preferences, host) {
             var compilerOptions = program.getCompilerOptions();
             // Namespace fixes don't conflict, so just build a list.
             var addToNamespace = [];
             var importType = [];
             // Keys are import clause node IDs.
-            var addToExisting = ts.createMap();
-            var newImports = ts.createMap();
+            var addToExisting = new ts.Map();
+            var newImports = new ts.Map();
             return { addImportFromDiagnostic: addImportFromDiagnostic, addImportFromExportedSymbol: addImportFromExportedSymbol, writeFixes: writeFixes };
             function addImportFromDiagnostic(diagnostic, context) {
-                var info = getFixesInfo(context, diagnostic.code, diagnostic.start);
+                var info = getFixesInfo(context, diagnostic.code, diagnostic.start, useAutoImportProvider);
                 if (!info || !info.fixes.length)
                     return;
                 addImport(info);
@@ -130093,9 +136488,9 @@ var ts;
                 var symbolName = ts.getNameForExportedSymbol(exportedSymbol, ts.getEmitScriptTarget(compilerOptions));
                 var checker = program.getTypeChecker();
                 var symbol = checker.getMergedSymbol(ts.skipAlias(exportedSymbol, checker));
-                var exportInfos = getAllReExportingModules(sourceFile, symbol, moduleSymbol, symbolName, sourceFile, compilerOptions, checker, program.getSourceFiles());
+                var exportInfos = getAllReExportingModules(sourceFile, symbol, moduleSymbol, symbolName, host, program, useAutoImportProvider);
                 var preferTypeOnlyImport = !!usageIsTypeOnly && compilerOptions.importsNotUsedAsValues === 2 /* Error */;
-                var useRequire = shouldUseRequire(sourceFile, compilerOptions);
+                var useRequire = shouldUseRequire(sourceFile, program);
                 var fix = getImportFixForSymbol(sourceFile, exportInfos, moduleSymbol, symbolName, program, /*position*/ undefined, preferTypeOnlyImport, useRequire, host, preferences);
                 addImport({ fixes: [fix], symbolName: symbolName });
             }
@@ -130180,7 +136575,6 @@ var ts;
                 }
             }
         }
-        codefix.createImportAdder = createImportAdder;
         // Sorted with the preferred fix coming first.
         var ImportFixKind;
         (function (ImportFixKind) {
@@ -130198,8 +136592,8 @@ var ts;
         })(ImportKind || (ImportKind = {}));
         function getImportCompletionAction(exportedSymbol, moduleSymbol, sourceFile, symbolName, host, program, formatContext, position, preferences) {
             var compilerOptions = program.getCompilerOptions();
-            var exportInfos = getAllReExportingModules(sourceFile, exportedSymbol, moduleSymbol, symbolName, sourceFile, compilerOptions, program.getTypeChecker(), program.getSourceFiles());
-            var useRequire = shouldUseRequire(sourceFile, compilerOptions);
+            var exportInfos = getAllReExportingModules(sourceFile, exportedSymbol, moduleSymbol, symbolName, host, program, /*useAutoImportProvider*/ true);
+            var useRequire = shouldUseRequire(sourceFile, program);
             var preferTypeOnlyImport = compilerOptions.importsNotUsedAsValues === 2 /* Error */ && !ts.isSourceFileJS(sourceFile) && ts.isValidTypeOnlyAliasUseSite(ts.getTokenAtPosition(sourceFile, position));
             var moduleSpecifier = ts.first(getNewImportInfos(program, sourceFile, position, preferTypeOnlyImport, useRequire, exportInfos, host, preferences)).moduleSpecifier;
             var fix = getImportFixForSymbol(sourceFile, exportInfos, moduleSymbol, symbolName, program, position, preferTypeOnlyImport, useRequire, host, preferences);
@@ -130215,15 +136609,17 @@ var ts;
             var description = _a.description, changes = _a.changes, commands = _a.commands;
             return { description: description, changes: changes, commands: commands };
         }
-        function getAllReExportingModules(importingFile, exportedSymbol, exportingModuleSymbol, symbolName, sourceFile, compilerOptions, checker, allSourceFiles) {
+        function getAllReExportingModules(importingFile, exportedSymbol, exportingModuleSymbol, symbolName, host, program, useAutoImportProvider) {
             var result = [];
-            forEachExternalModule(checker, allSourceFiles, function (moduleSymbol, moduleFile) {
+            var compilerOptions = program.getCompilerOptions();
+            forEachExternalModuleToImportFrom(program, host, importingFile, /*filterByPackageJson*/ false, useAutoImportProvider, function (moduleSymbol, moduleFile, program) {
+                var checker = program.getTypeChecker();
                 // Don't import from a re-export when looking "up" like to `./index` or `../index`.
-                if (moduleFile && moduleSymbol !== exportingModuleSymbol && ts.startsWith(sourceFile.fileName, ts.getDirectoryPath(moduleFile.fileName))) {
+                if (moduleFile && moduleSymbol !== exportingModuleSymbol && ts.startsWith(importingFile.fileName, ts.getDirectoryPath(moduleFile.fileName))) {
                     return;
                 }
                 var defaultInfo = getDefaultLikeExportInfo(importingFile, moduleSymbol, checker, compilerOptions);
-                if (defaultInfo && defaultInfo.name === symbolName && ts.skipAlias(defaultInfo.symbol, checker) === exportedSymbol) {
+                if (defaultInfo && (defaultInfo.name === symbolName || moduleSymbolToValidIdentifier(moduleSymbol, compilerOptions.target) === symbolName) && ts.skipAlias(defaultInfo.symbol, checker) === exportedSymbol) {
                     result.push({ moduleSymbol: moduleSymbol, importKind: defaultInfo.kind, exportedSymbolIsTypeOnly: isTypeOnlySymbol(defaultInfo.symbol, checker) });
                 }
                 for (var _i = 0, _a = checker.getExportsAndPropertiesOfModule(moduleSymbol); _i < _a.length; _i++) {
@@ -130279,11 +136675,11 @@ var ts;
         function getTargetModuleFromNamespaceLikeImport(declaration, checker) {
             var _a;
             switch (declaration.kind) {
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return checker.resolveExternalModuleName(declaration.initializer.arguments[0]);
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return checker.getAliasedSymbol(declaration.symbol);
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     var namespaceImport = ts.tryCast((_a = declaration.importClause) === null || _a === void 0 ? void 0 : _a.namedBindings, ts.isNamespaceImport);
                     return namespaceImport && checker.getAliasedSymbol(namespaceImport.symbol);
                 default:
@@ -130293,11 +136689,11 @@ var ts;
         function getNamespaceLikeImportText(declaration) {
             var _a, _b, _c;
             switch (declaration.kind) {
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return (_a = ts.tryCast(declaration.name, ts.isIdentifier)) === null || _a === void 0 ? void 0 : _a.text;
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return declaration.name.text;
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     return (_c = ts.tryCast((_b = declaration.importClause) === null || _b === void 0 ? void 0 : _b.namedBindings, ts.isNamespaceImport)) === null || _c === void 0 ? void 0 : _c.name.text;
                 default:
                     return ts.Debug.assertNever(declaration);
@@ -130306,10 +136702,10 @@ var ts;
         function tryAddToExistingImport(existingImports, canUseTypeOnlyImport) {
             return ts.firstDefined(existingImports, function (_a) {
                 var declaration = _a.declaration, importKind = _a.importKind;
-                if (declaration.kind === 253 /* ImportEqualsDeclaration */)
+                if (declaration.kind === 260 /* ImportEqualsDeclaration */)
                     return undefined;
-                if (declaration.kind === 242 /* VariableDeclaration */) {
-                    return (importKind === 0 /* Named */ || importKind === 1 /* Default */) && declaration.name.kind === 189 /* ObjectBindingPattern */
+                if (declaration.kind === 249 /* VariableDeclaration */) {
+                    return (importKind === 0 /* Named */ || importKind === 1 /* Default */) && declaration.name.kind === 196 /* ObjectBindingPattern */
                         ? { kind: 2 /* AddToExisting */, importClauseOrBindingPattern: declaration.name, importKind: importKind, moduleSpecifier: declaration.initializer.arguments[0].text, canUseTypeOnlyImport: false }
                         : undefined;
                 }
@@ -130317,7 +136713,7 @@ var ts;
                 if (!importClause)
                     return undefined;
                 var name = importClause.name, namedBindings = importClause.namedBindings;
-                return importKind === 1 /* Default */ && !name || importKind === 0 /* Named */ && (!namedBindings || namedBindings.kind === 257 /* NamedImports */)
+                return importKind === 1 /* Default */ && !name || importKind === 0 /* Named */ && (!namedBindings || namedBindings.kind === 264 /* NamedImports */)
                     ? { kind: 2 /* AddToExisting */, importClauseOrBindingPattern: importClause, importKind: importKind, moduleSpecifier: declaration.moduleSpecifier.getText(), canUseTypeOnlyImport: canUseTypeOnlyImport }
                     : undefined;
             });
@@ -130330,15 +136726,38 @@ var ts;
                 if (ts.isRequireVariableDeclaration(i.parent, /*requireStringLiteralLikeArgument*/ true)) {
                     return checker.resolveExternalModuleName(moduleSpecifier) === moduleSymbol ? { declaration: i.parent, importKind: importKind } : undefined;
                 }
-                if (i.kind === 254 /* ImportDeclaration */ || i.kind === 253 /* ImportEqualsDeclaration */) {
+                if (i.kind === 261 /* ImportDeclaration */ || i.kind === 260 /* ImportEqualsDeclaration */) {
                     return checker.getSymbolAtLocation(moduleSpecifier) === moduleSymbol ? { declaration: i, importKind: importKind } : undefined;
                 }
             });
         }
-        function shouldUseRequire(sourceFile, compilerOptions) {
-            return ts.isSourceFileJS(sourceFile)
-                && !sourceFile.externalModuleIndicator
-                && (!!sourceFile.commonJsModuleIndicator || ts.getEmitModuleKind(compilerOptions) < ts.ModuleKind.ES2015);
+        function shouldUseRequire(sourceFile, program) {
+            // 1. TypeScript files don't use require variable declarations
+            if (!ts.isSourceFileJS(sourceFile)) {
+                return false;
+            }
+            // 2. If the current source file is unambiguously CJS or ESM, go with that
+            if (sourceFile.commonJsModuleIndicator && !sourceFile.externalModuleIndicator)
+                return true;
+            if (sourceFile.externalModuleIndicator && !sourceFile.commonJsModuleIndicator)
+                return false;
+            // 3. If there's a tsconfig/jsconfig, use its module setting
+            var compilerOptions = program.getCompilerOptions();
+            if (compilerOptions.configFile) {
+                return ts.getEmitModuleKind(compilerOptions) < ts.ModuleKind.ES2015;
+            }
+            // 4. Match the first other JS file in the program that's unambiguously CJS or ESM
+            for (var _i = 0, _a = program.getSourceFiles(); _i < _a.length; _i++) {
+                var otherFile = _a[_i];
+                if (otherFile === sourceFile || !ts.isSourceFileJS(otherFile) || program.isSourceFileFromExternalLibrary(otherFile))
+                    continue;
+                if (otherFile.commonJsModuleIndicator && !otherFile.externalModuleIndicator)
+                    return true;
+                if (otherFile.externalModuleIndicator && !otherFile.commonJsModuleIndicator)
+                    return false;
+            }
+            // 5. Literally nothing to go on
+            return true;
         }
         function getNewImportInfos(program, sourceFile, position, preferTypeOnlyImport, useRequire, moduleSymbols, host, preferences) {
             var isJs = ts.isSourceFileJS(sourceFile);
@@ -130373,19 +136792,19 @@ var ts;
         }
         function newImportInfoFromExistingSpecifier(_a, preferTypeOnlyImport, useRequire) {
             var declaration = _a.declaration, importKind = _a.importKind;
-            var moduleSpecifier = declaration.kind === 254 /* ImportDeclaration */ ? declaration.moduleSpecifier :
-                declaration.kind === 242 /* VariableDeclaration */ ? declaration.initializer.arguments[0] :
-                    declaration.moduleReference.kind === 265 /* ExternalModuleReference */ ? declaration.moduleReference.expression :
+            var moduleSpecifier = declaration.kind === 261 /* ImportDeclaration */ ? declaration.moduleSpecifier :
+                declaration.kind === 249 /* VariableDeclaration */ ? declaration.initializer.arguments[0] :
+                    declaration.moduleReference.kind === 272 /* ExternalModuleReference */ ? declaration.moduleReference.expression :
                         undefined;
             return moduleSpecifier && ts.isStringLiteral(moduleSpecifier)
                 ? { kind: 3 /* AddNew */, moduleSpecifier: moduleSpecifier.text, importKind: importKind, typeOnly: preferTypeOnlyImport, useRequire: useRequire }
                 : undefined;
         }
-        function getFixesInfo(context, errorCode, pos) {
+        function getFixesInfo(context, errorCode, pos, useAutoImportProvider) {
             var symbolToken = ts.getTokenAtPosition(context.sourceFile, pos);
             var info = errorCode === ts.Diagnostics._0_refers_to_a_UMD_global_but_the_current_file_is_a_module_Consider_adding_an_import_instead.code
                 ? getFixesInfoForUMDImport(context, symbolToken)
-                : ts.isIdentifier(symbolToken) ? getFixesInfoForNonUMDImport(context, symbolToken) : undefined;
+                : ts.isIdentifier(symbolToken) ? getFixesInfoForNonUMDImport(context, symbolToken, useAutoImportProvider) : undefined;
             return info && __assign(__assign({}, info), { fixes: ts.sort(info.fixes, function (a, b) { return a.kind - b.kind; }) });
         }
         function getFixesInfoForUMDImport(_a, token) {
@@ -130397,7 +136816,7 @@ var ts;
             var symbol = checker.getAliasedSymbol(umdSymbol);
             var symbolName = umdSymbol.name;
             var exportInfos = [{ moduleSymbol: symbol, importKind: getUmdImportKind(sourceFile, program.getCompilerOptions()), exportedSymbolIsTypeOnly: false }];
-            var useRequire = shouldUseRequire(sourceFile, program.getCompilerOptions());
+            var useRequire = shouldUseRequire(sourceFile, program);
             var fixes = getFixForImport(exportInfos, symbolName, ts.isIdentifier(token) ? token.getStart(sourceFile) : undefined, /*preferTypeOnlyImport*/ false, useRequire, program, sourceFile, host, preferences);
             return { fixes: fixes, symbolName: symbolName };
         }
@@ -130438,45 +136857,52 @@ var ts;
                     return ts.Debug.assertNever(moduleKind, "Unexpected moduleKind " + moduleKind);
             }
         }
-        function getFixesInfoForNonUMDImport(_a, symbolToken) {
+        function getFixesInfoForNonUMDImport(_a, symbolToken, useAutoImportProvider) {
             var sourceFile = _a.sourceFile, program = _a.program, cancellationToken = _a.cancellationToken, host = _a.host, preferences = _a.preferences;
             var checker = program.getTypeChecker();
-            // If we're at `<Foo/>`, we must check if `Foo` is already in scope, and if so, get an import for `React` instead.
-            var symbolName = ts.isJsxOpeningLikeElement(symbolToken.parent)
-                && symbolToken.parent.tagName === symbolToken
-                && (ts.isIntrinsicJsxName(symbolToken.text) || checker.resolveName(symbolToken.text, symbolToken, 67108863 /* All */, /*excludeGlobals*/ false))
-                ? checker.getJsxNamespace(sourceFile)
-                : symbolToken.text;
+            var symbolName = getSymbolName(sourceFile, checker, symbolToken);
             // "default" is a keyword and not a legal identifier for the import, so we don't expect it here
             ts.Debug.assert(symbolName !== "default" /* Default */, "'default' isn't a legal identifier and couldn't occur here");
             var compilerOptions = program.getCompilerOptions();
             var preferTypeOnlyImport = compilerOptions.importsNotUsedAsValues === 2 /* Error */ && ts.isValidTypeOnlyAliasUseSite(symbolToken);
-            var useRequire = shouldUseRequire(sourceFile, compilerOptions);
-            var exportInfos = getExportInfos(symbolName, ts.getMeaningFromLocation(symbolToken), cancellationToken, sourceFile, checker, program, host);
+            var useRequire = shouldUseRequire(sourceFile, program);
+            var exportInfos = getExportInfos(symbolName, ts.getMeaningFromLocation(symbolToken), cancellationToken, sourceFile, program, useAutoImportProvider, host);
             var fixes = ts.arrayFrom(ts.flatMapIterator(exportInfos.entries(), function (_a) {
                 var _ = _a[0], exportInfos = _a[1];
                 return getFixForImport(exportInfos, symbolName, symbolToken.getStart(sourceFile), preferTypeOnlyImport, useRequire, program, sourceFile, host, preferences);
             }));
             return { fixes: fixes, symbolName: symbolName };
         }
+        function getSymbolName(sourceFile, checker, symbolToken) {
+            var parent = symbolToken.parent;
+            if ((ts.isJsxOpeningLikeElement(parent) || ts.isJsxClosingElement(parent)) && parent.tagName === symbolToken) {
+                var jsxNamespace = checker.getJsxNamespace(sourceFile);
+                if (ts.isIntrinsicJsxName(symbolToken.text) || !checker.resolveName(jsxNamespace, parent, 111551 /* Value */, /*excludeGlobals*/ true)) {
+                    return jsxNamespace;
+                }
+            }
+            return symbolToken.text;
+        }
         // Returns a map from an exported symbol's ID to a list of every way it's (re-)exported.
-        function getExportInfos(symbolName, currentTokenMeaning, cancellationToken, sourceFile, checker, program, host) {
+        function getExportInfos(symbolName, currentTokenMeaning, cancellationToken, sourceFile, program, useAutoImportProvider, host) {
             // For each original symbol, keep all re-exports of that symbol together so we can call `getCodeActionsForImport` on the whole group at once.
             // Maps symbol id to info for modules providing that symbol (original export + re-exports).
             var originalSymbolToExportInfos = ts.createMultiMap();
-            function addSymbol(moduleSymbol, exportedSymbol, importKind) {
+            function addSymbol(moduleSymbol, exportedSymbol, importKind, checker) {
                 originalSymbolToExportInfos.add(ts.getUniqueSymbolId(exportedSymbol, checker).toString(), { moduleSymbol: moduleSymbol, importKind: importKind, exportedSymbolIsTypeOnly: isTypeOnlySymbol(exportedSymbol, checker) });
             }
-            forEachExternalModuleToImportFrom(program, host, sourceFile, /*filterByPackageJson*/ true, function (moduleSymbol) {
+            forEachExternalModuleToImportFrom(program, host, sourceFile, /*filterByPackageJson*/ true, useAutoImportProvider, function (moduleSymbol, _, program) {
+                var checker = program.getTypeChecker();
                 cancellationToken.throwIfCancellationRequested();
-                var defaultInfo = getDefaultLikeExportInfo(sourceFile, moduleSymbol, checker, program.getCompilerOptions());
-                if (defaultInfo && defaultInfo.name === symbolName && symbolHasMeaning(defaultInfo.symbolForMeaning, currentTokenMeaning)) {
-                    addSymbol(moduleSymbol, defaultInfo.symbol, defaultInfo.kind);
+                var compilerOptions = program.getCompilerOptions();
+                var defaultInfo = getDefaultLikeExportInfo(sourceFile, moduleSymbol, checker, compilerOptions);
+                if (defaultInfo && (defaultInfo.name === symbolName || moduleSymbolToValidIdentifier(moduleSymbol, compilerOptions.target) === symbolName) && symbolHasMeaning(defaultInfo.symbolForMeaning, currentTokenMeaning)) {
+                    addSymbol(moduleSymbol, defaultInfo.symbol, defaultInfo.kind, checker);
                 }
                 // check exports with the same name
                 var exportSymbolWithIdenticalName = checker.tryGetMemberInModuleExportsAndProperties(symbolName, moduleSymbol);
                 if (exportSymbolWithIdenticalName && symbolHasMeaning(exportSymbolWithIdenticalName, currentTokenMeaning)) {
-                    addSymbol(moduleSymbol, exportSymbolWithIdenticalName, 0 /* Named */);
+                    addSymbol(moduleSymbol, exportSymbolWithIdenticalName, 0 /* Named */, checker);
                 }
             });
             return originalSymbolToExportInfos;
@@ -130530,7 +136956,13 @@ var ts;
                 return { symbolForMeaning: defaultExport, name: name };
             if (defaultExport.flags & 2097152 /* Alias */) {
                 var aliased = checker.getImmediateAliasedSymbol(defaultExport);
-                return aliased && getDefaultExportInfoWorker(aliased, ts.Debug.checkDefined(aliased.parent, "Alias targets of default exports must have a parent"), checker, compilerOptions);
+                if (aliased && aliased.parent) {
+                    // - `aliased` will be undefined if the module is exporting an unresolvable name,
+                    //    but we can still offer completions for it.
+                    // - `aliased.parent` will be undefined if the module is exporting `globalThis.something`,
+                    //    or another expression that resolves to a global.
+                    return getDefaultExportInfoWorker(aliased, aliased.parent, checker, compilerOptions);
+                }
             }
             if (defaultExport.escapedName !== "default" /* Default */ &&
                 defaultExport.escapedName !== "export=" /* ExportEquals */) {
@@ -130586,7 +137018,7 @@ var ts;
             }
         }
         function doAddExistingFix(changes, sourceFile, clause, defaultImport, namedImports, canUseTypeOnlyImport) {
-            if (clause.kind === 189 /* ObjectBindingPattern */) {
+            if (clause.kind === 196 /* ObjectBindingPattern */) {
                 if (defaultImport) {
                     addElementToBindingPattern(clause, defaultImport, "default");
                 }
@@ -130599,19 +137031,33 @@ var ts;
             var convertTypeOnlyToRegular = !canUseTypeOnlyImport && clause.isTypeOnly;
             if (defaultImport) {
                 ts.Debug.assert(!clause.name, "Cannot add a default import to an import clause that already has one");
-                changes.insertNodeAt(sourceFile, clause.getStart(sourceFile), ts.createIdentifier(defaultImport), { suffix: ", " });
+                changes.insertNodeAt(sourceFile, clause.getStart(sourceFile), ts.factory.createIdentifier(defaultImport), { suffix: ", " });
             }
             if (namedImports.length) {
-                var specifiers = namedImports.map(function (name) { return ts.createImportSpecifier(/*propertyName*/ undefined, ts.createIdentifier(name)); });
-                if (clause.namedBindings && ts.cast(clause.namedBindings, ts.isNamedImports).elements.length) {
-                    for (var _a = 0, specifiers_1 = specifiers; _a < specifiers_1.length; _a++) {
-                        var spec = specifiers_1[_a];
-                        changes.insertNodeInListAfter(sourceFile, ts.last(ts.cast(clause.namedBindings, ts.isNamedImports).elements), spec);
+                var existingSpecifiers = clause.namedBindings && ts.cast(clause.namedBindings, ts.isNamedImports).elements;
+                var newSpecifiers = ts.stableSort(namedImports.map(function (name) { return ts.factory.createImportSpecifier(/*propertyName*/ undefined, ts.factory.createIdentifier(name)); }), ts.OrganizeImports.compareImportOrExportSpecifiers);
+                if ((existingSpecifiers === null || existingSpecifiers === void 0 ? void 0 : existingSpecifiers.length) && ts.OrganizeImports.importSpecifiersAreSorted(existingSpecifiers)) {
+                    for (var _a = 0, newSpecifiers_1 = newSpecifiers; _a < newSpecifiers_1.length; _a++) {
+                        var spec = newSpecifiers_1[_a];
+                        var insertionIndex = ts.OrganizeImports.getImportSpecifierInsertionIndex(existingSpecifiers, spec);
+                        var prevSpecifier = clause.namedBindings.elements[insertionIndex - 1];
+                        if (prevSpecifier) {
+                            changes.insertNodeInListAfter(sourceFile, prevSpecifier, spec);
+                        }
+                        else {
+                            changes.insertNodeBefore(sourceFile, existingSpecifiers[0], spec, !ts.positionsAreOnSameLine(existingSpecifiers[0].getStart(), clause.parent.getStart(), sourceFile));
+                        }
+                    }
+                }
+                else if (existingSpecifiers === null || existingSpecifiers === void 0 ? void 0 : existingSpecifiers.length) {
+                    for (var _b = 0, newSpecifiers_2 = newSpecifiers; _b < newSpecifiers_2.length; _b++) {
+                        var spec = newSpecifiers_2[_b];
+                        changes.insertNodeInListAfter(sourceFile, ts.last(existingSpecifiers), spec, existingSpecifiers);
                     }
                 }
                 else {
-                    if (specifiers.length) {
-                        var namedImports_2 = ts.createNamedImports(specifiers);
+                    if (newSpecifiers.length) {
+                        var namedImports_2 = ts.factory.createNamedImports(newSpecifiers);
                         if (clause.namedBindings) {
                             changes.replaceNode(sourceFile, clause.namedBindings, namedImports_2);
                         }
@@ -130625,12 +137071,12 @@ var ts;
                 changes.delete(sourceFile, ts.getTypeKeywordOfTypeOnlyImport(clause, sourceFile));
             }
             function addElementToBindingPattern(bindingPattern, name, propertyName) {
-                var element = ts.createBindingElement(/*dotDotDotToken*/ undefined, propertyName, name);
+                var element = ts.factory.createBindingElement(/*dotDotDotToken*/ undefined, propertyName, name);
                 if (bindingPattern.elements.length) {
                     changes.insertNodeInListAfter(sourceFile, ts.last(bindingPattern.elements), element);
                 }
                 else {
-                    changes.replaceNode(sourceFile, bindingPattern, ts.createObjectBindingPattern([element]));
+                    changes.replaceNode(sourceFile, bindingPattern, ts.factory.createObjectBindingPattern([element]));
                 }
             }
         }
@@ -130651,18 +137097,18 @@ var ts;
             var quotedModuleSpecifier = ts.makeStringLiteral(moduleSpecifier, quotePreference);
             var statements;
             if (imports.defaultImport !== undefined || ((_a = imports.namedImports) === null || _a === void 0 ? void 0 : _a.length)) {
-                statements = ts.combine(statements, ts.makeImport(imports.defaultImport === undefined ? undefined : ts.createIdentifier(imports.defaultImport), (_b = imports.namedImports) === null || _b === void 0 ? void 0 : _b.map(function (n) { return ts.createImportSpecifier(/*propertyName*/ undefined, ts.createIdentifier(n)); }), moduleSpecifier, quotePreference, imports.typeOnly));
+                statements = ts.combine(statements, ts.makeImport(imports.defaultImport === undefined ? undefined : ts.factory.createIdentifier(imports.defaultImport), (_b = imports.namedImports) === null || _b === void 0 ? void 0 : _b.map(function (n) { return ts.factory.createImportSpecifier(/*propertyName*/ undefined, ts.factory.createIdentifier(n)); }), moduleSpecifier, quotePreference, imports.typeOnly));
             }
             var namespaceLikeImport = imports.namespaceLikeImport, typeOnly = imports.typeOnly;
             if (namespaceLikeImport) {
                 var declaration = namespaceLikeImport.importKind === 3 /* CommonJS */
-                    ? ts.createImportEqualsDeclaration(
+                    ? ts.factory.createImportEqualsDeclaration(
                     /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, ts.createIdentifier(namespaceLikeImport.name), ts.createExternalModuleReference(quotedModuleSpecifier))
-                    : ts.createImportDeclaration(
+                    /*modifiers*/ undefined, ts.factory.createIdentifier(namespaceLikeImport.name), ts.factory.createExternalModuleReference(quotedModuleSpecifier))
+                    : ts.factory.createImportDeclaration(
                     /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, ts.createImportClause(
-                    /*name*/ undefined, ts.createNamespaceImport(ts.createIdentifier(namespaceLikeImport.name)), typeOnly), quotedModuleSpecifier);
+                    /*modifiers*/ undefined, ts.factory.createImportClause(typeOnly, 
+                    /*name*/ undefined, ts.factory.createNamespaceImport(ts.factory.createIdentifier(namespaceLikeImport.name))), quotedModuleSpecifier);
                 statements = ts.combine(statements, declaration);
             }
             return ts.Debug.checkDefined(statements);
@@ -130673,11 +137119,11 @@ var ts;
             var statements;
             // const { default: foo, bar, etc } = require('./mod');
             if (imports.defaultImport || ((_a = imports.namedImports) === null || _a === void 0 ? void 0 : _a.length)) {
-                var bindingElements = ((_b = imports.namedImports) === null || _b === void 0 ? void 0 : _b.map(function (name) { return ts.createBindingElement(/*dotDotDotToken*/ undefined, /*propertyName*/ undefined, name); })) || [];
+                var bindingElements = ((_b = imports.namedImports) === null || _b === void 0 ? void 0 : _b.map(function (name) { return ts.factory.createBindingElement(/*dotDotDotToken*/ undefined, /*propertyName*/ undefined, name); })) || [];
                 if (imports.defaultImport) {
-                    bindingElements.unshift(ts.createBindingElement(/*dotDotDotToken*/ undefined, "default", imports.defaultImport));
+                    bindingElements.unshift(ts.factory.createBindingElement(/*dotDotDotToken*/ undefined, "default", imports.defaultImport));
                 }
-                var declaration = createConstEqualsRequireDeclaration(ts.createObjectBindingPattern(bindingElements), quotedModuleSpecifier);
+                var declaration = createConstEqualsRequireDeclaration(ts.factory.createObjectBindingPattern(bindingElements), quotedModuleSpecifier);
                 statements = ts.combine(statements, declaration);
             }
             // const foo = require('./mod');
@@ -130688,24 +137134,37 @@ var ts;
             return ts.Debug.checkDefined(statements);
         }
         function createConstEqualsRequireDeclaration(name, quotedModuleSpecifier) {
-            return ts.createVariableStatement(
-            /*modifiers*/ undefined, ts.createVariableDeclarationList([
-                ts.createVariableDeclaration(typeof name === "string" ? ts.createIdentifier(name) : name, 
-                /*type*/ undefined, ts.createCall(ts.createIdentifier("require"), /*typeArguments*/ undefined, [quotedModuleSpecifier]))
+            return ts.factory.createVariableStatement(
+            /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([
+                ts.factory.createVariableDeclaration(typeof name === "string" ? ts.factory.createIdentifier(name) : name, 
+                /*exclamationToken*/ undefined, 
+                /*type*/ undefined, ts.factory.createCallExpression(ts.factory.createIdentifier("require"), /*typeArguments*/ undefined, [quotedModuleSpecifier]))
             ], 2 /* Const */));
         }
         function symbolHasMeaning(_a, meaning) {
             var declarations = _a.declarations;
             return ts.some(declarations, function (decl) { return !!(ts.getMeaningFromDeclaration(decl) & meaning); });
         }
-        function forEachExternalModuleToImportFrom(program, host, from, filterByPackageJson, cb) {
+        function forEachExternalModuleToImportFrom(program, host, from, filterByPackageJson, useAutoImportProvider, cb) {
+            var _a, _b;
+            forEachExternalModuleToImportFromInProgram(program, host, from, filterByPackageJson, function (module, file) { return cb(module, file, program, /*isFromPackageJson*/ false); });
+            var autoImportProvider = useAutoImportProvider && ((_a = host.getPackageJsonAutoImportProvider) === null || _a === void 0 ? void 0 : _a.call(host));
+            if (autoImportProvider) {
+                var start = ts.timestamp();
+                forEachExternalModuleToImportFromInProgram(autoImportProvider, host, from, filterByPackageJson, function (module, file) { return cb(module, file, autoImportProvider, /*isFromPackageJson*/ true); });
+                (_b = host.log) === null || _b === void 0 ? void 0 : _b.call(host, "forEachExternalModuleToImportFrom autoImportProvider: " + (ts.timestamp() - start));
+            }
+        }
+        codefix.forEachExternalModuleToImportFrom = forEachExternalModuleToImportFrom;
+        function forEachExternalModuleToImportFromInProgram(program, host, from, filterByPackageJson, cb) {
+            var _a;
             var filteredCount = 0;
             var moduleSpecifierResolutionHost = ts.createModuleSpecifierResolutionHost(program, host);
             var packageJson = filterByPackageJson && createAutoImportFilter(from, program, host, moduleSpecifierResolutionHost);
             forEachExternalModule(program.getTypeChecker(), program.getSourceFiles(), function (module, sourceFile) {
                 if (sourceFile === undefined) {
                     if (!packageJson || packageJson.allowsImportingAmbientModule(module)) {
-                        cb(module);
+                        cb(module, sourceFile);
                     }
                     else if (packageJson) {
                         filteredCount++;
@@ -130715,18 +137174,15 @@ var ts;
                     sourceFile !== from &&
                     isImportableFile(program, from, sourceFile, moduleSpecifierResolutionHost)) {
                     if (!packageJson || packageJson.allowsImportingSourceFile(sourceFile)) {
-                        cb(module);
+                        cb(module, sourceFile);
                     }
                     else if (packageJson) {
                         filteredCount++;
                     }
                 }
             });
-            if (host.log) {
-                host.log("forEachExternalModuleToImportFrom: filtered out " + filteredCount + " modules by package.json contents");
-            }
+            (_a = host.log) === null || _a === void 0 ? void 0 : _a.call(host, "forEachExternalModuleToImportFrom: filtered out " + filteredCount + " modules by package.json contents");
         }
-        codefix.forEachExternalModuleToImportFrom = forEachExternalModuleToImportFrom;
         function forEachExternalModule(checker, allSourceFiles, cb) {
             for (var _i = 0, _a = checker.getAmbientModules(); _i < _a.length; _i++) {
                 var ambient = _a[_i];
@@ -130797,15 +137253,14 @@ var ts;
         codefix.moduleSpecifierToValidIdentifier = moduleSpecifierToValidIdentifier;
         function createAutoImportFilter(fromFile, program, host, moduleSpecifierResolutionHost) {
             if (moduleSpecifierResolutionHost === void 0) { moduleSpecifierResolutionHost = ts.createModuleSpecifierResolutionHost(program, host); }
-            var packageJsons = host.getPackageJsonsVisibleToFile && host.getPackageJsonsVisibleToFile(fromFile.fileName) || ts.getPackageJsonsVisibleToFile(fromFile.fileName, host);
-            var dependencyGroups = 1 /* Dependencies */ | 2 /* DevDependencies */ | 8 /* OptionalDependencies */;
+            var packageJsons = ((host.getPackageJsonsVisibleToFile && host.getPackageJsonsVisibleToFile(fromFile.fileName)) || ts.getPackageJsonsVisibleToFile(fromFile.fileName, host)).filter(function (p) { return p.parseable; });
             var usesNodeCoreModules;
             return { allowsImportingAmbientModule: allowsImportingAmbientModule, allowsImportingSourceFile: allowsImportingSourceFile, allowsImportingSpecifier: allowsImportingSpecifier, moduleSpecifierResolutionHost: moduleSpecifierResolutionHost };
             function moduleSpecifierIsCoveredByPackageJson(specifier) {
                 var packageName = getNodeModuleRootSpecifier(specifier);
                 for (var _i = 0, packageJsons_1 = packageJsons; _i < packageJsons_1.length; _i++) {
                     var packageJson = packageJsons_1[_i];
-                    if (packageJson.has(packageName, dependencyGroups) || packageJson.has(ts.getTypesPackageName(packageName), dependencyGroups)) {
+                    if (packageJson.has(packageName) || packageJson.has(ts.getTypesPackageName(packageName))) {
                         return true;
                     }
                 }
@@ -130915,12 +137370,12 @@ var ts;
         });
         function doChange(changes, sourceFile, pos, checker) {
             var token = ts.getTokenAtPosition(sourceFile, pos);
-            ts.Debug.assert(token.kind === 104 /* ThisKeyword */);
+            ts.Debug.assert(token.kind === 107 /* ThisKeyword */);
             var fn = ts.getThisContainer(token, /*includeArrowFunctions*/ false);
             if (!ts.isFunctionDeclaration(fn) && !ts.isFunctionExpression(fn))
                 return undefined;
             if (!ts.isSourceFile(ts.getThisContainer(fn, /*includeArrowFunctions*/ false))) { // 'this' is defined outside, convert to arrow function
-                var fnKeyword = ts.Debug.assertDefined(ts.findChildOfKind(fn, 94 /* FunctionKeyword */, sourceFile));
+                var fnKeyword = ts.Debug.assertDefined(ts.findChildOfKind(fn, 97 /* FunctionKeyword */, sourceFile));
                 var name = fn.name;
                 var body = ts.Debug.assertDefined(fn.body); // Should be defined because the function contained a 'this' expression
                 if (ts.isFunctionExpression(fn)) {
@@ -130939,17 +137394,60 @@ var ts;
                 else {
                     // `function f() {}` => `const f = () => {}`
                     // `name` should be defined because we only do this in inner contexts, and name is only undefined for `export default function() {}`.
-                    changes.replaceNode(sourceFile, fnKeyword, ts.createToken(81 /* ConstKeyword */));
+                    changes.replaceNode(sourceFile, fnKeyword, ts.factory.createToken(84 /* ConstKeyword */));
                     changes.insertText(sourceFile, name.end, " = ");
                     changes.insertText(sourceFile, body.pos, " =>");
                     return [ts.Diagnostics.Convert_function_declaration_0_to_arrow_function, name.text];
                 }
             }
-            // No outer 'this', add a @class tag if in a JS constructor function
-            else if (ts.isSourceFileJS(sourceFile) && ts.isPropertyAccessExpression(token.parent) && ts.isAssignmentExpression(token.parent.parent)) {
-                codefix.addJSDocTags(changes, sourceFile, fn, [ts.createJSDocClassTag()]);
-                return ts.Diagnostics.Add_class_tag;
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixId = "fixIncorrectNamedTupleSyntax";
+        var errorCodes = [
+            ts.Diagnostics.A_labeled_tuple_element_is_declared_as_optional_with_a_question_mark_after_the_name_and_before_the_colon_rather_than_after_the_type.code,
+            ts.Diagnostics.A_labeled_tuple_element_is_declared_as_rest_with_a_before_the_name_rather_than_before_the_type.code
+        ];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            getCodeActions: function (context) {
+                var sourceFile = context.sourceFile, span = context.span;
+                var namedTupleMember = getNamedTupleMember(sourceFile, span.start);
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, sourceFile, namedTupleMember); });
+                return [codefix.createCodeFixAction(fixId, changes, ts.Diagnostics.Move_labeled_tuple_element_modifiers_to_labels, fixId, ts.Diagnostics.Move_labeled_tuple_element_modifiers_to_labels)];
+            },
+            fixIds: [fixId]
+        });
+        function getNamedTupleMember(sourceFile, pos) {
+            var token = ts.getTokenAtPosition(sourceFile, pos);
+            return ts.findAncestor(token, function (t) { return t.kind === 192 /* NamedTupleMember */; });
+        }
+        function doChange(changes, sourceFile, namedTupleMember) {
+            if (!namedTupleMember) {
+                return;
+            }
+            var unwrappedType = namedTupleMember.type;
+            var sawOptional = false;
+            var sawRest = false;
+            while (unwrappedType.kind === 180 /* OptionalType */ || unwrappedType.kind === 181 /* RestType */ || unwrappedType.kind === 186 /* ParenthesizedType */) {
+                if (unwrappedType.kind === 180 /* OptionalType */) {
+                    sawOptional = true;
+                }
+                else if (unwrappedType.kind === 181 /* RestType */) {
+                    sawRest = true;
+                }
+                unwrappedType = unwrappedType.type;
             }
+            var updated = ts.factory.updateNamedTupleMember(namedTupleMember, namedTupleMember.dotDotDotToken || (sawRest ? ts.factory.createToken(25 /* DotDotDotToken */) : undefined), namedTupleMember.name, namedTupleMember.questionToken || (sawOptional ? ts.factory.createToken(57 /* QuestionToken */) : undefined), unwrappedType);
+            if (updated === namedTupleMember) {
+                return;
+            }
+            changes.replaceNode(sourceFile, namedTupleMember, updated);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -130964,13 +137462,17 @@ var ts;
             ts.Diagnostics.Cannot_find_name_0_Did_you_mean_1.code,
             ts.Diagnostics.Cannot_find_name_0_Did_you_mean_the_instance_member_this_0.code,
             ts.Diagnostics.Cannot_find_name_0_Did_you_mean_the_static_member_1_0.code,
-            ts.Diagnostics.Module_0_has_no_exported_member_1_Did_you_mean_2.code,
+            ts.Diagnostics._0_has_no_exported_member_named_1_Did_you_mean_2.code,
+            // for JSX class components
+            ts.Diagnostics.No_overload_matches_this_call.code,
+            // for JSX FC
+            ts.Diagnostics.Type_0_is_not_assignable_to_type_1.code,
         ];
         codefix.registerCodeFix({
             errorCodes: errorCodes,
             getCodeActions: function (context) {
-                var sourceFile = context.sourceFile;
-                var info = getInfo(sourceFile, context.span.start, context);
+                var sourceFile = context.sourceFile, errorCode = context.errorCode;
+                var info = getInfo(sourceFile, context.span.start, context, errorCode);
                 if (!info)
                     return undefined;
                 var node = info.node, suggestedSymbol = info.suggestedSymbol;
@@ -130980,18 +137482,23 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
-                var info = getInfo(diag.file, diag.start, context);
+                var info = getInfo(diag.file, diag.start, context, diag.code);
                 var target = context.host.getCompilationSettings().target;
                 if (info)
                     doChange(changes, context.sourceFile, info.node, info.suggestedSymbol, target);
             }); },
         });
-        function getInfo(sourceFile, pos, context) {
+        function getInfo(sourceFile, pos, context, errorCode) {
             // This is the identifier of the misspelled word. eg:
             // this.speling = 1;
             //      ^^^^^^^
             var node = ts.getTokenAtPosition(sourceFile, pos);
             var parent = node.parent;
+            // Only fix spelling for No_overload_matches_this_call emitted on the React class component
+            if ((errorCode === ts.Diagnostics.No_overload_matches_this_call.code ||
+                errorCode === ts.Diagnostics.Type_0_is_not_assignable_to_type_1.code) &&
+                !ts.isJsxAttribute(parent))
+                return undefined;
             var checker = context.program.getTypeChecker();
             var suggestedSymbol;
             if (ts.isPropertyAccessExpression(parent) && parent.name === node) {
@@ -131002,6 +137509,12 @@ var ts;
                 }
                 suggestedSymbol = checker.getSuggestedSymbolForNonexistentProperty(node, containingType);
             }
+            else if (ts.isQualifiedName(parent) && parent.right === node) {
+                var symbol = checker.getSymbolAtLocation(parent.left);
+                if (symbol && symbol.flags & 1536 /* Module */) {
+                    suggestedSymbol = checker.getSuggestedSymbolForNonexistentModule(parent.right, symbol);
+                }
+            }
             else if (ts.isImportSpecifier(parent) && parent.name === node) {
                 ts.Debug.assertNode(node, ts.isIdentifier, "Expected an identifier for spelling (import)");
                 var importDeclaration = ts.findAncestor(node, ts.isImportDeclaration);
@@ -131010,6 +137523,12 @@ var ts;
                     suggestedSymbol = checker.getSuggestedSymbolForNonexistentModule(node, resolvedSourceFile.symbol);
                 }
             }
+            else if (ts.isJsxAttribute(parent) && parent.name === node) {
+                ts.Debug.assertNode(node, ts.isIdentifier, "Expected an identifier for JSX attribute");
+                var tag = ts.findAncestor(node, ts.isJsxOpeningLikeElement);
+                var props = checker.getContextualTypeForArgumentAtIndex(tag, 0);
+                suggestedSymbol = checker.getSuggestedSymbolForNonexistentJSXAttribute(node, props);
+            }
             else {
                 var meaning = ts.getMeaningFromLocation(node);
                 var name = ts.getTextOfNode(node);
@@ -131023,14 +137542,14 @@ var ts;
             if (!ts.isIdentifierText(suggestion, target) && ts.isPropertyAccessExpression(node.parent)) {
                 var valDecl = suggestedSymbol.valueDeclaration;
                 if (ts.isNamedDeclaration(valDecl) && ts.isPrivateIdentifier(valDecl.name)) {
-                    changes.replaceNode(sourceFile, node, ts.createIdentifier(suggestion));
+                    changes.replaceNode(sourceFile, node, ts.factory.createIdentifier(suggestion));
                 }
                 else {
-                    changes.replaceNode(sourceFile, node.parent, ts.createElementAccess(node.parent.expression, ts.createLiteral(suggestion)));
+                    changes.replaceNode(sourceFile, node.parent, ts.factory.createElementAccessExpression(node.parent.expression, ts.factory.createStringLiteral(suggestion)));
                 }
             }
             else {
-                changes.replaceNode(sourceFile, node, ts.createIdentifier(suggestion));
+                changes.replaceNode(sourceFile, node, ts.factory.createIdentifier(suggestion));
             }
         }
         function convertSemanticMeaningToSymbolFlags(meaning) {
@@ -131063,7 +137582,7 @@ var ts;
     (function (codefix) {
         var fixId = "returnValueCorrect";
         var fixIdAddReturnStatement = "fixAddReturnStatement";
-        var fixIdRemoveBlockBodyBrace = "fixRemoveBlockBodyBrace";
+        var fixRemoveBracesFromArrowFunctionBody = "fixRemoveBracesFromArrowFunctionBody";
         var fixIdWrapTheBlockWithParen = "fixWrapTheBlockWithParen";
         var errorCodes = [
             ts.Diagnostics.A_function_whose_declared_type_is_neither_void_nor_any_must_return_a_value.code,
@@ -131077,14 +137596,14 @@ var ts;
         })(ProblemKind || (ProblemKind = {}));
         codefix.registerCodeFix({
             errorCodes: errorCodes,
-            fixIds: [fixIdAddReturnStatement, fixIdRemoveBlockBodyBrace, fixIdWrapTheBlockWithParen],
+            fixIds: [fixIdAddReturnStatement, fixRemoveBracesFromArrowFunctionBody, fixIdWrapTheBlockWithParen],
             getCodeActions: function (context) {
                 var program = context.program, sourceFile = context.sourceFile, start = context.span.start, errorCode = context.errorCode;
                 var info = getInfo(program.getTypeChecker(), sourceFile, start, errorCode);
                 if (!info)
                     return undefined;
                 if (info.kind === ProblemKind.MissingReturnStatement) {
-                    return ts.append([getActionForfixAddReturnStatement(context, info.expression, info.statement)], ts.isArrowFunction(info.declaration) ? getActionForfixRemoveBlockBodyBrace(context, info.declaration, info.expression, info.commentSource) : undefined);
+                    return ts.append([getActionForfixAddReturnStatement(context, info.expression, info.statement)], ts.isArrowFunction(info.declaration) ? getActionForFixRemoveBracesFromArrowFunctionBody(context, info.declaration, info.expression, info.commentSource) : undefined);
                 }
                 else {
                     return [getActionForfixWrapTheBlockWithParen(context, info.declaration, info.expression)];
@@ -131098,7 +137617,7 @@ var ts;
                     case fixIdAddReturnStatement:
                         addReturnStatement(changes, diag.file, info.expression, info.statement);
                         break;
-                    case fixIdRemoveBlockBodyBrace:
+                    case fixRemoveBracesFromArrowFunctionBody:
                         if (!ts.isArrowFunction(info.declaration))
                             return undefined;
                         removeBlockBodyBrace(changes, diag.file, info.declaration, info.expression, info.commentSource, /* withParen */ false);
@@ -131113,11 +137632,17 @@ var ts;
                 }
             }); },
         });
+        function createObjectTypeFromLabeledExpression(checker, label, expression) {
+            var member = checker.createSymbol(4 /* Property */, label.escapedText);
+            member.type = checker.getTypeAtLocation(expression);
+            var members = ts.createSymbolTable([member]);
+            return checker.createAnonymousType(/*symbol*/ undefined, members, [], [], /*stringIndexInfo*/ undefined, /*numberIndexInfo*/ undefined);
+        }
         function getFixInfo(checker, declaration, expectType, isFunctionType) {
             if (!declaration.body || !ts.isBlock(declaration.body) || ts.length(declaration.body.statements) !== 1)
                 return undefined;
             var firstStatement = ts.first(declaration.body.statements);
-            if (ts.isExpressionStatement(firstStatement) && checkFixedAssignableTo(checker, declaration, firstStatement.expression, expectType, isFunctionType)) {
+            if (ts.isExpressionStatement(firstStatement) && checkFixedAssignableTo(checker, declaration, checker.getTypeAtLocation(firstStatement.expression), expectType, isFunctionType)) {
                 return {
                     declaration: declaration,
                     kind: ProblemKind.MissingReturnStatement,
@@ -131127,8 +137652,9 @@ var ts;
                 };
             }
             else if (ts.isLabeledStatement(firstStatement) && ts.isExpressionStatement(firstStatement.statement)) {
-                var node = ts.createObjectLiteral([ts.createPropertyAssignment(firstStatement.label, firstStatement.statement.expression)]);
-                if (checkFixedAssignableTo(checker, declaration, node, expectType, isFunctionType)) {
+                var node = ts.factory.createObjectLiteralExpression([ts.factory.createPropertyAssignment(firstStatement.label, firstStatement.statement.expression)]);
+                var nodeType = createObjectTypeFromLabeledExpression(checker, firstStatement.label, firstStatement.statement.expression);
+                if (checkFixedAssignableTo(checker, declaration, nodeType, expectType, isFunctionType)) {
                     return ts.isArrowFunction(declaration) ? {
                         declaration: declaration,
                         kind: ProblemKind.MissingParentheses,
@@ -131147,8 +137673,9 @@ var ts;
             else if (ts.isBlock(firstStatement) && ts.length(firstStatement.statements) === 1) {
                 var firstBlockStatement = ts.first(firstStatement.statements);
                 if (ts.isLabeledStatement(firstBlockStatement) && ts.isExpressionStatement(firstBlockStatement.statement)) {
-                    var node = ts.createObjectLiteral([ts.createPropertyAssignment(firstBlockStatement.label, firstBlockStatement.statement.expression)]);
-                    if (checkFixedAssignableTo(checker, declaration, node, expectType, isFunctionType)) {
+                    var node = ts.factory.createObjectLiteralExpression([ts.factory.createPropertyAssignment(firstBlockStatement.label, firstBlockStatement.statement.expression)]);
+                    var nodeType = createObjectTypeFromLabeledExpression(checker, firstBlockStatement.label, firstBlockStatement.statement.expression);
+                    if (checkFixedAssignableTo(checker, declaration, nodeType, expectType, isFunctionType)) {
                         return {
                             declaration: declaration,
                             kind: ProblemKind.MissingReturnStatement,
@@ -131161,8 +137688,25 @@ var ts;
             }
             return undefined;
         }
-        function checkFixedAssignableTo(checker, declaration, expr, type, isFunctionType) {
-            return checker.isTypeAssignableTo(checker.getTypeAtLocation(isFunctionType ? ts.updateFunctionLikeBody(declaration, ts.createBlock([ts.createReturn(expr)])) : expr), type);
+        function checkFixedAssignableTo(checker, declaration, exprType, type, isFunctionType) {
+            if (isFunctionType) {
+                var sig = checker.getSignatureFromDeclaration(declaration);
+                if (sig) {
+                    if (ts.hasSyntacticModifier(declaration, 256 /* Async */)) {
+                        exprType = checker.createPromiseType(exprType);
+                    }
+                    var newSig = checker.createSignature(declaration, sig.typeParameters, sig.thisParameter, sig.parameters, exprType, 
+                    /*typePredicate*/ undefined, sig.minArgumentCount, sig.flags);
+                    exprType = checker.createAnonymousType(
+                    /*symbol*/ undefined, ts.createSymbolTable(), [newSig], [], 
+                    /*stringIndexInfo*/ undefined, 
+                    /*numberIndexInfo*/ undefined);
+                }
+                else {
+                    exprType = checker.getAnyType();
+                }
+            }
+            return checker.isTypeAssignableTo(exprType, type);
         }
         function getInfo(checker, sourceFile, position, errorCode) {
             var node = ts.getTokenAtPosition(sourceFile, position);
@@ -131194,47 +137738,47 @@ var ts;
         }
         function getVariableLikeInitializer(declaration) {
             switch (declaration.kind) {
-                case 242 /* VariableDeclaration */:
-                case 156 /* Parameter */:
-                case 191 /* BindingElement */:
-                case 159 /* PropertyDeclaration */:
-                case 281 /* PropertyAssignment */:
+                case 249 /* VariableDeclaration */:
+                case 160 /* Parameter */:
+                case 198 /* BindingElement */:
+                case 163 /* PropertyDeclaration */:
+                case 288 /* PropertyAssignment */:
                     return declaration.initializer;
-                case 273 /* JsxAttribute */:
+                case 280 /* JsxAttribute */:
                     return declaration.initializer && (ts.isJsxExpression(declaration.initializer) ? declaration.initializer.expression : undefined);
-                case 282 /* ShorthandPropertyAssignment */:
-                case 158 /* PropertySignature */:
-                case 284 /* EnumMember */:
-                case 323 /* JSDocPropertyTag */:
-                case 317 /* JSDocParameterTag */:
+                case 289 /* ShorthandPropertyAssignment */:
+                case 162 /* PropertySignature */:
+                case 291 /* EnumMember */:
+                case 333 /* JSDocPropertyTag */:
+                case 326 /* JSDocParameterTag */:
                     return undefined;
             }
         }
         function addReturnStatement(changes, sourceFile, expression, statement) {
             ts.suppressLeadingAndTrailingTrivia(expression);
             var probablyNeedSemi = ts.probablyUsesSemicolons(sourceFile);
-            changes.replaceNode(sourceFile, statement, ts.createReturn(expression), {
+            changes.replaceNode(sourceFile, statement, ts.factory.createReturnStatement(expression), {
                 leadingTriviaOption: ts.textChanges.LeadingTriviaOption.Exclude,
                 trailingTriviaOption: ts.textChanges.TrailingTriviaOption.Exclude,
                 suffix: probablyNeedSemi ? ";" : undefined
             });
         }
         function removeBlockBodyBrace(changes, sourceFile, declaration, expression, commentSource, withParen) {
-            var newBody = (withParen || ts.needsParentheses(expression)) ? ts.createParen(expression) : expression;
+            var newBody = (withParen || ts.needsParentheses(expression)) ? ts.factory.createParenthesizedExpression(expression) : expression;
             ts.suppressLeadingAndTrailingTrivia(commentSource);
             ts.copyComments(commentSource, newBody);
             changes.replaceNode(sourceFile, declaration.body, newBody);
         }
         function wrapBlockWithParen(changes, sourceFile, declaration, expression) {
-            changes.replaceNode(sourceFile, declaration.body, ts.createParen(expression));
+            changes.replaceNode(sourceFile, declaration.body, ts.factory.createParenthesizedExpression(expression));
         }
         function getActionForfixAddReturnStatement(context, expression, statement) {
             var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return addReturnStatement(t, context.sourceFile, expression, statement); });
             return codefix.createCodeFixAction(fixId, changes, ts.Diagnostics.Add_a_return_statement, fixIdAddReturnStatement, ts.Diagnostics.Add_all_missing_return_statement);
         }
-        function getActionForfixRemoveBlockBodyBrace(context, declaration, expression, commentSource) {
+        function getActionForFixRemoveBracesFromArrowFunctionBody(context, declaration, expression, commentSource) {
             var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return removeBlockBodyBrace(t, context.sourceFile, declaration, expression, commentSource, /* withParen */ false); });
-            return codefix.createCodeFixAction(fixId, changes, ts.Diagnostics.Remove_block_body_braces, fixIdRemoveBlockBodyBrace, ts.Diagnostics.Remove_all_incorrect_body_block_braces);
+            return codefix.createCodeFixAction(fixId, changes, ts.Diagnostics.Remove_braces_from_arrow_function_body, fixRemoveBracesFromArrowFunctionBody, ts.Diagnostics.Remove_braces_from_all_arrow_function_bodies_with_relevant_issues);
         }
         function getActionForfixWrapTheBlockWithParen(context, declaration, expression) {
             var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return wrapBlockWithParen(t, context.sourceFile, declaration, expression); });
@@ -131260,26 +137804,22 @@ var ts;
             errorCodes: errorCodes,
             getCodeActions: function (context) {
                 var info = getInfo(context.sourceFile, context.span.start, context.program.getTypeChecker(), context.program);
-                if (!info)
+                if (!info) {
                     return undefined;
+                }
                 if (info.kind === 0 /* Enum */) {
                     var token_1 = info.token, parentDeclaration_1 = info.parentDeclaration;
                     var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return addEnumMemberDeclaration(t, context.program.getTypeChecker(), token_1, parentDeclaration_1); });
                     return [codefix.createCodeFixAction(fixName, changes, [ts.Diagnostics.Add_missing_enum_member_0, token_1.text], fixId, ts.Diagnostics.Add_all_missing_members)];
                 }
-                var parentDeclaration = info.parentDeclaration, declSourceFile = info.declSourceFile, inJs = info.inJs, makeStatic = info.makeStatic, token = info.token, call = info.call;
-                var methodCodeAction = call && getActionForMethodDeclaration(context, declSourceFile, parentDeclaration, token, call, makeStatic, inJs);
-                var addMember = inJs && !ts.isInterfaceDeclaration(parentDeclaration) ?
-                    ts.singleElementArray(getActionsForAddMissingMemberInJavascriptFile(context, declSourceFile, parentDeclaration, token, makeStatic)) :
-                    getActionsForAddMissingMemberInTypeScriptFile(context, declSourceFile, parentDeclaration, token, makeStatic);
-                return ts.concatenate(ts.singleElementArray(methodCodeAction), addMember);
+                return ts.concatenate(getActionsForMissingMethodDeclaration(context, info), getActionsForMissingMemberDeclaration(context, info));
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
                 var program = context.program;
                 var checker = program.getTypeChecker();
-                var seen = ts.createMap();
-                var typeDeclToMembers = new ts.NodeMap();
+                var seen = new ts.Map();
+                var typeDeclToMembers = new ts.Map();
                 return codefix.createCombinedCodeActions(ts.textChanges.ChangeTracker.with(context, function (changes) {
                     codefix.eachDiagnostic(context, errorCodes, function (diag) {
                         var info = getInfo(diag.file, diag.start, checker, context.program);
@@ -131292,13 +137832,13 @@ var ts;
                         }
                         else {
                             var parentDeclaration = info.parentDeclaration, token_2 = info.token;
-                            var infos = typeDeclToMembers.getOrUpdate(parentDeclaration, function () { return []; });
+                            var infos = ts.getOrUpdate(typeDeclToMembers, parentDeclaration, function () { return []; });
                             if (!infos.some(function (i) { return i.token.text === token_2.text; }))
                                 infos.push(info);
                         }
                     });
                     typeDeclToMembers.forEach(function (infos, classDeclaration) {
-                        var supers = getAllSupers(classDeclaration, checker);
+                        var supers = codefix.getAllSupers(classDeclaration, checker);
                         var _loop_13 = function (info) {
                             // If some superclass added this property, don't add it again.
                             if (supers.some(function (superClassOrInterface) {
@@ -131309,18 +137849,18 @@ var ts;
                                 });
                             }))
                                 return "continue";
-                            var parentDeclaration = info.parentDeclaration, declSourceFile = info.declSourceFile, inJs = info.inJs, makeStatic = info.makeStatic, token = info.token, call = info.call;
+                            var parentDeclaration = info.parentDeclaration, declSourceFile = info.declSourceFile, modifierFlags = info.modifierFlags, token = info.token, call = info.call, isJSFile = info.isJSFile;
                             // Always prefer to add a method declaration if possible.
                             if (call && !ts.isPrivateIdentifier(token)) {
-                                addMethodDeclaration(context, changes, declSourceFile, parentDeclaration, token, call, makeStatic, inJs);
+                                addMethodDeclaration(context, changes, call, token.text, modifierFlags & 32 /* Static */, parentDeclaration, declSourceFile, isJSFile);
                             }
                             else {
-                                if (inJs && !ts.isInterfaceDeclaration(parentDeclaration)) {
-                                    addMissingMemberInJs(changes, declSourceFile, parentDeclaration, token, makeStatic);
+                                if (isJSFile && !ts.isInterfaceDeclaration(parentDeclaration)) {
+                                    addMissingMemberInJs(changes, declSourceFile, parentDeclaration, token, !!(modifierFlags & 32 /* Static */));
                                 }
                                 else {
                                     var typeNode = getTypeNode(program.getTypeChecker(), parentDeclaration, token);
-                                    addPropertyDeclaration(changes, declSourceFile, parentDeclaration, token.text, typeNode, makeStatic ? 32 /* Static */ : 0);
+                                    addPropertyDeclaration(changes, declSourceFile, parentDeclaration, token.text, typeNode, modifierFlags & 32 /* Static */);
                                 }
                             }
                         };
@@ -131332,19 +137872,6 @@ var ts;
                 }));
             },
         });
-        function getAllSupers(decl, checker) {
-            var res = [];
-            while (decl) {
-                var superElement = ts.getClassExtendsHeritageElement(decl);
-                var superSymbol = superElement && checker.getSymbolAtLocation(superElement.expression);
-                var superDecl = superSymbol && ts.find(superSymbol.declarations, ts.isClassLike);
-                if (superDecl) {
-                    res.push(superDecl);
-                }
-                decl = superDecl;
-            }
-            return res;
-        }
         var InfoKind;
         (function (InfoKind) {
             InfoKind[InfoKind["Enum"] = 0] = "Enum";
@@ -131378,9 +137905,10 @@ var ts;
                     return undefined;
                 }
                 var declSourceFile = classOrInterface.getSourceFile();
-                var inJs = ts.isSourceFileJS(declSourceFile);
+                var modifierFlags = (makeStatic ? 32 /* Static */ : 0) | (ts.startsWithUnderscore(token.text) ? 8 /* Private */ : 0);
+                var isJSFile = ts.isSourceFileJS(declSourceFile);
                 var call = ts.tryCast(parent.parent, ts.isCallExpression);
-                return { kind: 1 /* ClassOrInterface */, token: token, parentDeclaration: classOrInterface, makeStatic: makeStatic, declSourceFile: declSourceFile, inJs: inJs, call: call };
+                return { kind: 1 /* ClassOrInterface */, token: token, call: call, modifierFlags: modifierFlags, parentDeclaration: classOrInterface, declSourceFile: declSourceFile, isJSFile: isJSFile };
             }
             var enumDeclaration = ts.find(symbol.declarations, ts.isEnumDeclaration);
             if (enumDeclaration && !ts.isPrivateIdentifier(token) && !program.isSourceFileFromExternalLibrary(enumDeclaration.getSourceFile())) {
@@ -131388,27 +137916,35 @@ var ts;
             }
             return undefined;
         }
-        function getActionsForAddMissingMemberInJavascriptFile(context, declSourceFile, classDeclaration, token, makeStatic) {
-            var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return addMissingMemberInJs(t, declSourceFile, classDeclaration, token, makeStatic); });
+        function getActionsForMissingMemberDeclaration(context, info) {
+            return info.isJSFile ? ts.singleElementArray(createActionForAddMissingMemberInJavascriptFile(context, info)) :
+                createActionsForAddMissingMemberInTypeScriptFile(context, info);
+        }
+        function createActionForAddMissingMemberInJavascriptFile(context, _a) {
+            var parentDeclaration = _a.parentDeclaration, declSourceFile = _a.declSourceFile, modifierFlags = _a.modifierFlags, token = _a.token;
+            if (ts.isInterfaceDeclaration(parentDeclaration)) {
+                return undefined;
+            }
+            var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return addMissingMemberInJs(t, declSourceFile, parentDeclaration, token, !!(modifierFlags & 32 /* Static */)); });
             if (changes.length === 0) {
                 return undefined;
             }
-            var diagnostic = makeStatic ? ts.Diagnostics.Initialize_static_property_0 :
+            var diagnostic = modifierFlags & 32 /* Static */ ? ts.Diagnostics.Initialize_static_property_0 :
                 ts.isPrivateIdentifier(token) ? ts.Diagnostics.Declare_a_private_field_named_0 : ts.Diagnostics.Initialize_property_0_in_the_constructor;
             return codefix.createCodeFixAction(fixName, changes, [diagnostic, token.text], fixId, ts.Diagnostics.Add_all_missing_members);
         }
         function addMissingMemberInJs(changeTracker, declSourceFile, classDeclaration, token, makeStatic) {
             var tokenName = token.text;
             if (makeStatic) {
-                if (classDeclaration.kind === 214 /* ClassExpression */) {
+                if (classDeclaration.kind === 221 /* ClassExpression */) {
                     return;
                 }
                 var className = classDeclaration.name.getText();
-                var staticInitialization = initializePropertyToUndefined(ts.createIdentifier(className), tokenName);
+                var staticInitialization = initializePropertyToUndefined(ts.factory.createIdentifier(className), tokenName);
                 changeTracker.insertNodeAfter(declSourceFile, classDeclaration, staticInitialization);
             }
             else if (ts.isPrivateIdentifier(token)) {
-                var property = ts.createProperty(
+                var property = ts.factory.createPropertyDeclaration(
                 /*decorators*/ undefined, 
                 /*modifiers*/ undefined, tokenName, 
                 /*questionToken*/ undefined, 
@@ -131427,28 +137963,32 @@ var ts;
                 if (!classConstructor) {
                     return;
                 }
-                var propertyInitialization = initializePropertyToUndefined(ts.createThis(), tokenName);
+                var propertyInitialization = initializePropertyToUndefined(ts.factory.createThis(), tokenName);
                 changeTracker.insertNodeAtConstructorEnd(declSourceFile, classConstructor, propertyInitialization);
             }
         }
         function initializePropertyToUndefined(obj, propertyName) {
-            return ts.createStatement(ts.createAssignment(ts.createPropertyAccess(obj, propertyName), ts.createIdentifier("undefined")));
-        }
-        function getActionsForAddMissingMemberInTypeScriptFile(context, declSourceFile, classDeclaration, token, makeStatic) {
-            var typeNode = getTypeNode(context.program.getTypeChecker(), classDeclaration, token);
-            var actions = [createAddPropertyDeclarationAction(context, declSourceFile, classDeclaration, token.text, typeNode, makeStatic ? 32 /* Static */ : 0)];
-            if (makeStatic || ts.isPrivateIdentifier(token)) {
+            return ts.factory.createExpressionStatement(ts.factory.createAssignment(ts.factory.createPropertyAccessExpression(obj, propertyName), ts.factory.createIdentifier("undefined")));
+        }
+        function createActionsForAddMissingMemberInTypeScriptFile(context, _a) {
+            var parentDeclaration = _a.parentDeclaration, declSourceFile = _a.declSourceFile, modifierFlags = _a.modifierFlags, token = _a.token;
+            var memberName = token.text;
+            var isStatic = modifierFlags & 32 /* Static */;
+            var typeNode = getTypeNode(context.program.getTypeChecker(), parentDeclaration, token);
+            var addPropertyDeclarationChanges = function (modifierFlags) { return ts.textChanges.ChangeTracker.with(context, function (t) { return addPropertyDeclaration(t, declSourceFile, parentDeclaration, memberName, typeNode, modifierFlags); }); };
+            var actions = [codefix.createCodeFixAction(fixName, addPropertyDeclarationChanges(modifierFlags & 32 /* Static */), [isStatic ? ts.Diagnostics.Declare_static_property_0 : ts.Diagnostics.Declare_property_0, memberName], fixId, ts.Diagnostics.Add_all_missing_members)];
+            if (isStatic || ts.isPrivateIdentifier(token)) {
                 return actions;
             }
-            if (ts.startsWithUnderscore(token.text)) {
-                actions.unshift(createAddPropertyDeclarationAction(context, declSourceFile, classDeclaration, token.text, typeNode, 8 /* Private */));
+            if (modifierFlags & 8 /* Private */) {
+                actions.unshift(codefix.createCodeFixActionWithoutFixAll(fixName, addPropertyDeclarationChanges(8 /* Private */), [ts.Diagnostics.Declare_private_property_0, memberName]));
             }
-            actions.push(createAddIndexSignatureAction(context, declSourceFile, classDeclaration, token.text, typeNode));
+            actions.push(createAddIndexSignatureAction(context, declSourceFile, parentDeclaration, token.text, typeNode));
             return actions;
         }
         function getTypeNode(checker, classDeclaration, token) {
             var typeNode;
-            if (token.parent.parent.kind === 209 /* BinaryExpression */) {
+            if (token.parent.parent.kind === 216 /* BinaryExpression */) {
                 var binaryExpression = token.parent.parent;
                 var otherExpression = token.parent === binaryExpression.left ? binaryExpression.right : binaryExpression.left;
                 var widenedType = checker.getWidenedType(checker.getBaseTypeOfLiteralType(checker.getTypeAtLocation(otherExpression)));
@@ -131458,19 +137998,12 @@ var ts;
                 var contextualType = checker.getContextualType(token.parent);
                 typeNode = contextualType ? checker.typeToTypeNode(contextualType, /*enclosingDeclaration*/ undefined, /*flags*/ undefined) : undefined;
             }
-            return typeNode || ts.createKeywordTypeNode(125 /* AnyKeyword */);
-        }
-        function createAddPropertyDeclarationAction(context, declSourceFile, classDeclaration, tokenName, typeNode, modifierFlags) {
-            var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return addPropertyDeclaration(t, declSourceFile, classDeclaration, tokenName, typeNode, modifierFlags); });
-            if (modifierFlags & 8 /* Private */) {
-                return codefix.createCodeFixActionWithoutFixAll(fixName, changes, [ts.Diagnostics.Declare_private_property_0, tokenName]);
-            }
-            return codefix.createCodeFixAction(fixName, changes, [modifierFlags & 32 /* Static */ ? ts.Diagnostics.Declare_static_property_0 : ts.Diagnostics.Declare_property_0, tokenName], fixId, ts.Diagnostics.Add_all_missing_members);
+            return typeNode || ts.factory.createKeywordTypeNode(128 /* AnyKeyword */);
         }
         function addPropertyDeclaration(changeTracker, declSourceFile, classDeclaration, tokenName, typeNode, modifierFlags) {
-            var property = ts.createProperty(
+            var property = ts.factory.createPropertyDeclaration(
             /*decorators*/ undefined, 
-            /*modifiers*/ modifierFlags ? ts.createNodeArray(ts.createModifiersFromModifierFlags(modifierFlags)) : undefined, tokenName, 
+            /*modifiers*/ modifierFlags ? ts.factory.createNodeArray(ts.factory.createModifiersFromModifierFlags(modifierFlags)) : undefined, tokenName, 
             /*questionToken*/ undefined, typeNode, 
             /*initializer*/ undefined);
             var lastProp = getNodeToInsertPropertyAfter(classDeclaration);
@@ -131494,39 +138027,48 @@ var ts;
         }
         function createAddIndexSignatureAction(context, declSourceFile, classDeclaration, tokenName, typeNode) {
             // Index signatures cannot have the static modifier.
-            var stringTypeNode = ts.createKeywordTypeNode(143 /* StringKeyword */);
-            var indexingParameter = ts.createParameter(
+            var stringTypeNode = ts.factory.createKeywordTypeNode(147 /* StringKeyword */);
+            var indexingParameter = ts.factory.createParameterDeclaration(
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, 
             /*dotDotDotToken*/ undefined, "x", 
             /*questionToken*/ undefined, stringTypeNode, 
             /*initializer*/ undefined);
-            var indexSignature = ts.createIndexSignature(
+            var indexSignature = ts.factory.createIndexSignature(
             /*decorators*/ undefined, 
             /*modifiers*/ undefined, [indexingParameter], typeNode);
             var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return t.insertNodeAtClassStart(declSourceFile, classDeclaration, indexSignature); });
             // No fixId here because code-fix-all currently only works on adding individual named properties.
             return codefix.createCodeFixActionWithoutFixAll(fixName, changes, [ts.Diagnostics.Add_index_signature_for_property_0, tokenName]);
         }
-        function getActionForMethodDeclaration(context, declSourceFile, classDeclaration, token, callExpression, makeStatic, inJs) {
+        function getActionsForMissingMethodDeclaration(context, info) {
+            var parentDeclaration = info.parentDeclaration, declSourceFile = info.declSourceFile, modifierFlags = info.modifierFlags, token = info.token, call = info.call, isJSFile = info.isJSFile;
+            if (call === undefined) {
+                return undefined;
+            }
             // Private methods are not implemented yet.
             if (ts.isPrivateIdentifier(token)) {
                 return undefined;
             }
-            var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return addMethodDeclaration(context, t, declSourceFile, classDeclaration, token, callExpression, makeStatic, inJs); });
-            return codefix.createCodeFixAction(fixName, changes, [makeStatic ? ts.Diagnostics.Declare_static_method_0 : ts.Diagnostics.Declare_method_0, token.text], fixId, ts.Diagnostics.Add_all_missing_members);
+            var methodName = token.text;
+            var addMethodDeclarationChanges = function (modifierFlags) { return ts.textChanges.ChangeTracker.with(context, function (t) { return addMethodDeclaration(context, t, call, methodName, modifierFlags, parentDeclaration, declSourceFile, isJSFile); }); };
+            var actions = [codefix.createCodeFixAction(fixName, addMethodDeclarationChanges(modifierFlags & 32 /* Static */), [modifierFlags & 32 /* Static */ ? ts.Diagnostics.Declare_static_method_0 : ts.Diagnostics.Declare_method_0, methodName], fixId, ts.Diagnostics.Add_all_missing_members)];
+            if (modifierFlags & 8 /* Private */) {
+                actions.unshift(codefix.createCodeFixActionWithoutFixAll(fixName, addMethodDeclarationChanges(8 /* Private */), [ts.Diagnostics.Declare_private_method_0, methodName]));
+            }
+            return actions;
         }
-        function addMethodDeclaration(context, changeTracker, declSourceFile, typeDecl, token, callExpression, makeStatic, inJs) {
-            var importAdder = codefix.createImportAdder(declSourceFile, context.program, context.preferences, context.host);
-            var methodDeclaration = codefix.createMethodFromCallExpression(context, callExpression, token.text, inJs, makeStatic, typeDecl, importAdder);
-            var containingMethodDeclaration = ts.getAncestor(callExpression, 161 /* MethodDeclaration */);
-            if (containingMethodDeclaration && containingMethodDeclaration.parent === typeDecl) {
-                changeTracker.insertNodeAfter(declSourceFile, containingMethodDeclaration, methodDeclaration);
+        function addMethodDeclaration(context, changes, callExpression, methodName, modifierFlags, parentDeclaration, sourceFile, isJSFile) {
+            var importAdder = codefix.createImportAdder(sourceFile, context.program, context.preferences, context.host);
+            var methodDeclaration = codefix.createMethodFromCallExpression(context, importAdder, callExpression, methodName, modifierFlags, parentDeclaration, isJSFile);
+            var containingMethodDeclaration = ts.findAncestor(callExpression, function (n) { return ts.isMethodDeclaration(n) || ts.isConstructorDeclaration(n); });
+            if (containingMethodDeclaration && containingMethodDeclaration.parent === parentDeclaration) {
+                changes.insertNodeAfter(sourceFile, containingMethodDeclaration, methodDeclaration);
             }
             else {
-                changeTracker.insertNodeAtClassStart(declSourceFile, typeDecl, methodDeclaration);
+                changes.insertNodeAtClassStart(sourceFile, parentDeclaration, methodDeclaration);
             }
-            importAdder.writeFixes(changeTracker);
+            importAdder.writeFixes(changes);
         }
         function addEnumMemberDeclaration(changes, checker, token, enumDeclaration) {
             /**
@@ -131536,10 +138078,10 @@ var ts;
              */
             var hasStringInitializer = ts.some(enumDeclaration.members, function (member) {
                 var type = checker.getTypeAtLocation(member);
-                return !!(type && type.flags & 132 /* StringLike */);
+                return !!(type && type.flags & 402653316 /* StringLike */);
             });
-            var enumMember = ts.createEnumMember(token, hasStringInitializer ? ts.createStringLiteral(token.text) : undefined);
-            changes.replaceNode(enumDeclaration.getSourceFile(), enumDeclaration, ts.updateEnumDeclaration(enumDeclaration, enumDeclaration.decorators, enumDeclaration.modifiers, enumDeclaration.name, ts.concatenate(enumDeclaration.members, ts.singleElementArray(enumMember))), {
+            var enumMember = ts.factory.createEnumMember(token, hasStringInitializer ? ts.factory.createStringLiteral(token.text) : undefined);
+            changes.replaceNode(enumDeclaration.getSourceFile(), enumDeclaration, ts.factory.updateEnumDeclaration(enumDeclaration, enumDeclaration.decorators, enumDeclaration.modifiers, enumDeclaration.name, ts.concatenate(enumDeclaration.members, ts.singleElementArray(enumMember))), {
                 leadingTriviaOption: ts.textChanges.LeadingTriviaOption.IncludeAll,
                 trailingTriviaOption: ts.textChanges.TrailingTriviaOption.Exclude
             });
@@ -131567,7 +138109,7 @@ var ts;
         });
         function addMissingNewOperator(changes, sourceFile, span) {
             var call = ts.cast(findAncestorMatchingSpan(sourceFile, span), ts.isCallExpression);
-            var newExpression = ts.createNew(call.expression, call.typeArguments, call.arguments);
+            var newExpression = ts.factory.createNewExpression(call.expression, call.typeArguments, call.arguments);
             changes.replaceNode(sourceFile, call, newExpression);
         }
         function findAncestorMatchingSpan(sourceFile, span) {
@@ -131660,7 +138202,7 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var seenClassDeclarations = ts.createMap();
+                var seenClassDeclarations = new ts.Map();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
                     var classDeclaration = getClass(diag.file, diag.start);
                     if (ts.addToSeen(seenClassDeclarations, ts.getNodeId(classDeclaration))) {
@@ -131683,13 +138225,13 @@ var ts;
             // so duplicates cannot occur.
             var abstractAndNonPrivateExtendsSymbols = checker.getPropertiesOfType(instantiatedExtendsType).filter(symbolPointsToNonPrivateAndAbstractMember);
             var importAdder = codefix.createImportAdder(sourceFile, context.program, preferences, context.host);
-            codefix.createMissingMemberNodes(classDeclaration, abstractAndNonPrivateExtendsSymbols, context, preferences, importAdder, function (member) { return changeTracker.insertNodeAtClassStart(sourceFile, classDeclaration, member); });
+            codefix.createMissingMemberNodes(classDeclaration, abstractAndNonPrivateExtendsSymbols, sourceFile, context, preferences, importAdder, function (member) { return changeTracker.insertNodeAtClassStart(sourceFile, classDeclaration, member); });
             importAdder.writeFixes(changeTracker);
         }
         function symbolPointsToNonPrivateAndAbstractMember(symbol) {
             // See `codeFixClassExtendAbstractProtectedProperty.ts` in https://github.com/Microsoft/TypeScript/pull/11547/files
             // (now named `codeFixClassExtendAbstractPrivateProperty.ts`)
-            var flags = ts.getModifierFlags(ts.first(symbol.getDeclarations()));
+            var flags = ts.getSyntacticModifierFlags(ts.first(symbol.getDeclarations()));
             return !(flags & 8 /* Private */) && !!(flags & 128 /* Abstract */);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -131715,7 +138257,7 @@ var ts;
             fixIds: [fixId],
             getAllCodeActions: function (context) {
                 var sourceFile = context.sourceFile;
-                var seenClasses = ts.createMap(); // Ensure we only do this once per class.
+                var seenClasses = new ts.Map(); // Ensure we only do this once per class.
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
                     var nodes = getNodes(diag.file, diag.start);
                     if (!nodes)
@@ -131733,7 +138275,7 @@ var ts;
         }
         function getNodes(sourceFile, pos) {
             var token = ts.getTokenAtPosition(sourceFile, pos);
-            if (token.kind !== 104 /* ThisKeyword */)
+            if (token.kind !== 107 /* ThisKeyword */)
                 return undefined;
             var constructor = ts.getContainingFunction(token);
             var superCall = findSuperCall(constructor.body);
@@ -131776,7 +138318,7 @@ var ts;
             return token.parent;
         }
         function doChange(changes, sourceFile, ctr) {
-            var superCall = ts.createStatement(ts.createCall(ts.createSuper(), /*typeArguments*/ undefined, /*argumentsArray*/ ts.emptyArray));
+            var superCall = ts.factory.createExpressionStatement(ts.factory.createCallExpression(ts.factory.createSuper(), /*typeArguments*/ undefined, /*argumentsArray*/ ts.emptyArray));
             changes.insertNodeAtConstructorStart(sourceFile, ctr, superCall);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -131810,7 +138352,7 @@ var ts;
             }); },
         });
         function doChange(changeTracker, configFile) {
-            codefix.setJsonCompilerOptionValue(changeTracker, configFile, "experimentalDecorators", ts.createTrue());
+            codefix.setJsonCompilerOptionValue(changeTracker, configFile, "experimentalDecorators", ts.factory.createTrue());
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -131847,7 +138389,7 @@ var ts;
             }
         });
         function doChange(changeTracker, configFile) {
-            codefix.setJsonCompilerOptionValue(changeTracker, configFile, "jsx", ts.createStringLiteral("react"));
+            codefix.setJsonCompilerOptionValue(changeTracker, configFile, "jsx", ts.factory.createStringLiteral("react"));
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -131869,7 +138411,7 @@ var ts;
                 var moduleOutOfRange = moduleKind >= ts.ModuleKind.ES2015 && moduleKind < ts.ModuleKind.ESNext;
                 if (moduleOutOfRange) {
                     var changes = ts.textChanges.ChangeTracker.with(context, function (changes) {
-                        codefix.setJsonCompilerOptionValue(changes, configFile, "module", ts.createStringLiteral("esnext"));
+                        codefix.setJsonCompilerOptionValue(changes, configFile, "module", ts.factory.createStringLiteral("esnext"));
                     });
                     codeFixes.push(codefix.createCodeFixActionWithoutFixAll("fixModuleOption", changes, [ts.Diagnostics.Set_the_module_option_in_your_configuration_file_to_0, "esnext"]));
                 }
@@ -131880,10 +138422,10 @@ var ts;
                         var configObject = ts.getTsConfigObjectLiteralExpression(configFile);
                         if (!configObject)
                             return;
-                        var options = [["target", ts.createStringLiteral("es2017")]];
+                        var options = [["target", ts.factory.createStringLiteral("es2017")]];
                         if (moduleKind === ts.ModuleKind.CommonJS) {
                             // Ensure we preserve the default module kind (commonjs), as targets >= ES2015 have a default module kind of es2015.
-                            options.push(["module", ts.createStringLiteral("commonjs")]);
+                            options.push(["module", ts.factory.createStringLiteral("commonjs")]);
                         }
                         codefix.setJsonCompilerOptionValues(tracker, configFile, options);
                     });
@@ -131896,6 +138438,36 @@ var ts;
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixId = "fixPropertyAssignment";
+        var errorCodes = [
+            ts.Diagnostics.Did_you_mean_to_use_a_Colon_An_can_only_follow_a_property_name_when_the_containing_object_literal_is_part_of_a_destructuring_pattern.code
+        ];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            fixIds: [fixId],
+            getCodeActions: function (context) {
+                var sourceFile = context.sourceFile, span = context.span;
+                var property = getProperty(sourceFile, span.start);
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, context.sourceFile, property); });
+                return [codefix.createCodeFixAction(fixId, changes, [ts.Diagnostics.Change_0_to_1, "=", ":"], fixId, [ts.Diagnostics.Switch_each_misused_0_to_1, "=", ":"])];
+            },
+            getAllCodeActions: function (context) {
+                return codefix.codeFixAll(context, errorCodes, function (changes, diag) { return doChange(changes, diag.file, getProperty(diag.file, diag.start)); });
+            }
+        });
+        function doChange(changes, sourceFile, node) {
+            changes.replaceNode(sourceFile, node, ts.factory.createPropertyAssignment(node.name, node.objectAssignmentInitializer));
+        }
+        function getProperty(sourceFile, pos) {
+            return ts.cast(ts.getTokenAtPosition(sourceFile, pos).parent, ts.isShorthandPropertyAssignment);
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var codefix;
     (function (codefix) {
@@ -131923,17 +138495,17 @@ var ts;
             var token = ts.getTokenAtPosition(sourceFile, pos);
             var heritageClauses = ts.getContainingClass(token).heritageClauses;
             var extendsToken = heritageClauses[0].getFirstToken();
-            return extendsToken.kind === 90 /* ExtendsKeyword */ ? { extendsToken: extendsToken, heritageClauses: heritageClauses } : undefined;
+            return extendsToken.kind === 93 /* ExtendsKeyword */ ? { extendsToken: extendsToken, heritageClauses: heritageClauses } : undefined;
         }
         function doChanges(changes, sourceFile, extendsToken, heritageClauses) {
-            changes.replaceNode(sourceFile, extendsToken, ts.createToken(113 /* ImplementsKeyword */));
+            changes.replaceNode(sourceFile, extendsToken, ts.factory.createToken(116 /* ImplementsKeyword */));
             // If there is already an implements clause, replace the implements keyword with a comma.
             if (heritageClauses.length === 2 &&
-                heritageClauses[0].token === 90 /* ExtendsKeyword */ &&
-                heritageClauses[1].token === 113 /* ImplementsKeyword */) {
+                heritageClauses[0].token === 93 /* ExtendsKeyword */ &&
+                heritageClauses[1].token === 116 /* ImplementsKeyword */) {
                 var implementsToken = heritageClauses[1].getFirstToken();
                 var implementsFullStart = implementsToken.getFullStart();
-                changes.replaceRange(sourceFile, { pos: implementsFullStart, end: implementsFullStart }, ts.createToken(27 /* CommaToken */));
+                changes.replaceRange(sourceFile, { pos: implementsFullStart, end: implementsFullStart }, ts.factory.createToken(27 /* CommaToken */));
                 // Rough heuristic: delete trailing whitespace after keyword so that it's not excessive.
                 // (Trailing because leading might be indentation, which is more sensitive.)
                 var text = sourceFile.text;
@@ -131986,7 +138558,7 @@ var ts;
             var node = _a.node, className = _a.className;
             // TODO (https://github.com/Microsoft/TypeScript/issues/21246): use shared helper
             ts.suppressLeadingAndTrailingTrivia(node);
-            changes.replaceNode(sourceFile, node, ts.createPropertyAccess(className ? ts.createIdentifier(className) : ts.createThis(), node));
+            changes.replaceNode(sourceFile, node, ts.factory.createPropertyAccessExpression(className ? ts.factory.createIdentifier(className) : ts.factory.createThis(), node));
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -132030,7 +138602,7 @@ var ts;
             if (!isValidCharacter(character)) {
                 return;
             }
-            var replacement = useHtmlEntity ? htmlEntity[character] : "{" + ts.quote(character, preferences) + "}";
+            var replacement = useHtmlEntity ? htmlEntity[character] : "{" + ts.quote(sourceFile, preferences, character) + "}";
             changes.replaceRangeWithText(sourceFile, { pos: start, end: start + 1 }, replacement);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
@@ -132072,18 +138644,34 @@ var ts;
                     var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return t.delete(sourceFile, importDecl); });
                     return [createDeleteFix(changes, [ts.Diagnostics.Remove_import_from_0, ts.showModuleSpecifier(importDecl)])];
                 }
-                var delDestructure = ts.textChanges.ChangeTracker.with(context, function (t) {
-                    return tryDeleteFullDestructure(token, t, sourceFile, checker, sourceFiles, /*isFixAll*/ false);
-                });
-                if (delDestructure.length) {
-                    return [createDeleteFix(delDestructure, ts.Diagnostics.Remove_destructuring)];
+                if (ts.isObjectBindingPattern(token.parent)) {
+                    if (ts.isParameter(token.parent.parent)) {
+                        var elements = token.parent.elements;
+                        var diagnostic = [
+                            elements.length > 1 ? ts.Diagnostics.Remove_unused_declarations_for_Colon_0 : ts.Diagnostics.Remove_unused_declaration_for_Colon_0,
+                            ts.map(elements, function (e) { return e.getText(sourceFile); }).join(", ")
+                        ];
+                        return [
+                            createDeleteFix(ts.textChanges.ChangeTracker.with(context, function (t) {
+                                return deleteDestructuringElements(t, sourceFile, token.parent);
+                            }), diagnostic)
+                        ];
+                    }
+                    return [
+                        createDeleteFix(ts.textChanges.ChangeTracker.with(context, function (t) {
+                            return t.delete(sourceFile, token.parent.parent);
+                        }), ts.Diagnostics.Remove_unused_destructuring_declaration)
+                    ];
                 }
-                var delVar = ts.textChanges.ChangeTracker.with(context, function (t) { return tryDeleteFullVariableStatement(sourceFile, token, t); });
-                if (delVar.length) {
-                    return [createDeleteFix(delVar, ts.Diagnostics.Remove_variable_statement)];
+                if (canDeleteEntireVariableStatement(sourceFile, token)) {
+                    return [
+                        createDeleteFix(ts.textChanges.ChangeTracker.with(context, function (t) {
+                            return deleteEntireVariableStatement(t, sourceFile, token.parent);
+                        }), ts.Diagnostics.Remove_variable_statement)
+                    ];
                 }
                 var result = [];
-                if (token.kind === 132 /* InferKeyword */) {
+                if (token.kind === 135 /* InferKeyword */) {
                     var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return changeInferToUnknown(t, sourceFile, token); });
                     var name = ts.cast(token.parent, ts.isInferTypeNode).typeParameter.name.text;
                     result.push(codefix.createCodeFixAction(fixName, changes, [ts.Diagnostics.Replace_infer_0_with_unknown, name], fixIdInfer, ts.Diagnostics.Replace_all_unused_infer_with_unknown));
@@ -132115,8 +138703,9 @@ var ts;
                             tryPrefixDeclaration(changes, diag.code, sourceFile, token);
                             break;
                         case fixIdDelete: {
-                            if (token.kind === 132 /* InferKeyword */)
+                            if (token.kind === 135 /* InferKeyword */) {
                                 break; // Can't delete
+                            }
                             var importDecl = tryGetFullImport(token);
                             if (importDecl) {
                                 changes.delete(sourceFile, importDecl);
@@ -132127,14 +138716,24 @@ var ts;
                             else if (token.kind === 29 /* LessThanToken */) {
                                 deleteTypeParameters(changes, sourceFile, token);
                             }
-                            else if (!tryDeleteFullDestructure(token, changes, sourceFile, checker, sourceFiles, /*isFixAll*/ true) &&
-                                !tryDeleteFullVariableStatement(sourceFile, token, changes)) {
+                            else if (ts.isObjectBindingPattern(token.parent)) {
+                                if (ts.isParameter(token.parent.parent)) {
+                                    deleteDestructuringElements(changes, sourceFile, token.parent);
+                                }
+                                else {
+                                    changes.delete(sourceFile, token.parent.parent);
+                                }
+                            }
+                            else if (canDeleteEntireVariableStatement(sourceFile, token)) {
+                                deleteEntireVariableStatement(changes, sourceFile, token.parent);
+                            }
+                            else {
                                 tryDeleteDeclaration(sourceFile, token, changes, checker, sourceFiles, /*isFixAll*/ true);
                             }
                             break;
                         }
                         case fixIdInfer:
-                            if (token.kind === 132 /* InferKeyword */) {
+                            if (token.kind === 135 /* InferKeyword */) {
                                 changeInferToUnknown(changes, sourceFile, token);
                             }
                             break;
@@ -132145,7 +138744,7 @@ var ts;
             },
         });
         function changeInferToUnknown(changes, sourceFile, token) {
-            changes.replaceNode(sourceFile, token.parent, ts.createKeywordTypeNode(148 /* UnknownKeyword */));
+            changes.replaceNode(sourceFile, token.parent, ts.factory.createKeywordTypeNode(152 /* UnknownKeyword */));
         }
         function createDeleteFix(changes, diag) {
             return codefix.createCodeFixAction(fixName, changes, diag, fixIdDelete, ts.Diagnostics.Delete_all_unused_declarations);
@@ -132155,41 +138754,30 @@ var ts;
         }
         // Sometimes the diagnostic span is an entire ImportDeclaration, so we should remove the whole thing.
         function tryGetFullImport(token) {
-            return token.kind === 96 /* ImportKeyword */ ? ts.tryCast(token.parent, ts.isImportDeclaration) : undefined;
+            return token.kind === 99 /* ImportKeyword */ ? ts.tryCast(token.parent, ts.isImportDeclaration) : undefined;
         }
-        function tryDeleteFullDestructure(token, changes, sourceFile, checker, sourceFiles, isFixAll) {
-            if (token.kind !== 18 /* OpenBraceToken */ || !ts.isObjectBindingPattern(token.parent))
-                return false;
-            var decl = token.parent.parent;
-            if (decl.kind === 156 /* Parameter */) {
-                tryDeleteParameter(changes, sourceFile, decl, checker, sourceFiles, isFixAll);
-            }
-            else {
-                changes.delete(sourceFile, decl);
-            }
-            return true;
+        function canDeleteEntireVariableStatement(sourceFile, token) {
+            return ts.isVariableDeclarationList(token.parent) && ts.first(token.parent.getChildren(sourceFile)) === token;
         }
-        function tryDeleteFullVariableStatement(sourceFile, token, changes) {
-            var declarationList = ts.tryCast(token.parent, ts.isVariableDeclarationList);
-            if (declarationList && declarationList.getChildren(sourceFile)[0] === token) {
-                changes.delete(sourceFile, declarationList.parent.kind === 225 /* VariableStatement */ ? declarationList.parent : declarationList);
-                return true;
-            }
-            return false;
+        function deleteEntireVariableStatement(changes, sourceFile, node) {
+            changes.delete(sourceFile, node.parent.kind === 232 /* VariableStatement */ ? node.parent : node);
+        }
+        function deleteDestructuringElements(changes, sourceFile, node) {
+            ts.forEach(node.elements, function (n) { return changes.delete(sourceFile, n); });
         }
         function tryPrefixDeclaration(changes, errorCode, sourceFile, token) {
             // Don't offer to prefix a property.
             if (errorCode === ts.Diagnostics.Property_0_is_declared_but_its_value_is_never_read.code)
                 return;
-            if (token.kind === 132 /* InferKeyword */) {
+            if (token.kind === 135 /* InferKeyword */) {
                 token = ts.cast(token.parent, ts.isInferTypeNode).typeParameter.name;
             }
             if (ts.isIdentifier(token) && canPrefix(token)) {
-                changes.replaceNode(sourceFile, token, ts.createIdentifier("_" + token.text));
+                changes.replaceNode(sourceFile, token, ts.factory.createIdentifier("_" + token.text));
                 if (ts.isParameter(token.parent)) {
                     ts.getJSDocParameterTags(token.parent).forEach(function (tag) {
                         if (ts.isIdentifier(tag.name)) {
-                            changes.replaceNode(sourceFile, tag.name, ts.createIdentifier("_" + tag.name.text));
+                            changes.replaceNode(sourceFile, tag.name, ts.factory.createIdentifier("_" + tag.name.text));
                         }
                     });
                 }
@@ -132197,14 +138785,14 @@ var ts;
         }
         function canPrefix(token) {
             switch (token.parent.kind) {
-                case 156 /* Parameter */:
-                case 155 /* TypeParameter */:
+                case 160 /* Parameter */:
+                case 159 /* TypeParameter */:
                     return true;
-                case 242 /* VariableDeclaration */: {
+                case 249 /* VariableDeclaration */: {
                     var varDecl = token.parent;
                     switch (varDecl.parent.parent.kind) {
-                        case 232 /* ForOfStatement */:
-                        case 231 /* ForInStatement */:
+                        case 239 /* ForOfStatement */:
+                        case 238 /* ForInStatement */:
                             return true;
                     }
                 }
@@ -132235,12 +138823,11 @@ var ts;
             }
         }
         function tryDeleteParameter(changes, sourceFile, p, checker, sourceFiles, isFixAll) {
-            if (mayDeleteParameter(p, checker, isFixAll)) {
-                if (p.modifiers && p.modifiers.length > 0
-                    && (!ts.isIdentifier(p.name) || ts.FindAllReferences.Core.isSymbolReferencedInFile(p.name, checker, sourceFile))) {
-                    p.modifiers.forEach(function (modifier) {
-                        changes.deleteModifier(sourceFile, modifier);
-                    });
+            if (isFixAll === void 0) { isFixAll = false; }
+            if (mayDeleteParameter(checker, sourceFile, p, isFixAll)) {
+                if (p.modifiers && p.modifiers.length > 0 &&
+                    (!ts.isIdentifier(p.name) || ts.FindAllReferences.Core.isSymbolReferencedInFile(p.name, checker, sourceFile))) {
+                    p.modifiers.forEach(function (modifier) { return changes.deleteModifier(sourceFile, modifier); });
                 }
                 else {
                     changes.delete(sourceFile, p);
@@ -132248,29 +138835,28 @@ var ts;
                 }
             }
         }
-        function mayDeleteParameter(p, checker, isFixAll) {
-            var parent = p.parent;
+        function mayDeleteParameter(checker, sourceFile, parameter, isFixAll) {
+            var parent = parameter.parent;
             switch (parent.kind) {
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     // Don't remove a parameter if this overrides something.
                     var symbol = checker.getSymbolAtLocation(parent.name);
                     if (ts.isMemberSymbolInBaseType(symbol, checker))
                         return false;
                 // falls through
-                case 162 /* Constructor */:
-                case 244 /* FunctionDeclaration */:
+                case 166 /* Constructor */:
+                    return true;
+                case 251 /* FunctionDeclaration */: {
+                    if (parent.name && isCallbackLike(checker, sourceFile, parent.name)) {
+                        return isLastParameter(parent, parameter, isFixAll);
+                    }
                     return true;
-                case 201 /* FunctionExpression */:
-                case 202 /* ArrowFunction */: {
+                }
+                case 208 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
                     // Can't remove a non-last parameter in a callback. Can remove a parameter in code-fix-all if future parameters are also unused.
-                    var parameters = parent.parameters;
-                    var index = parameters.indexOf(p);
-                    ts.Debug.assert(index !== -1, "The parameter should already be in the list");
-                    return isFixAll
-                        ? parameters.slice(index + 1).every(function (p) { return p.name.kind === 75 /* Identifier */ && !p.symbol.isReferenced; })
-                        : index === parameters.length - 1;
-                }
-                case 164 /* SetAccessor */:
+                    return isLastParameter(parent, parameter, isFixAll);
+                case 168 /* SetAccessor */:
                     // Setter must have a parameter
                     return false;
                 default:
@@ -132285,6 +138871,19 @@ var ts;
                 }
             });
         }
+        function isCallbackLike(checker, sourceFile, name) {
+            return !!ts.FindAllReferences.Core.eachSymbolReferenceInFile(name, checker, sourceFile, function (reference) {
+                return ts.isIdentifier(reference) && ts.isCallExpression(reference.parent) && reference.parent.arguments.indexOf(reference) >= 0;
+            });
+        }
+        function isLastParameter(func, parameter, isFixAll) {
+            var parameters = func.parameters;
+            var index = parameters.indexOf(parameter);
+            ts.Debug.assert(index !== -1, "The parameter should already be in the list");
+            return isFixAll ?
+                parameters.slice(index + 1).every(function (p) { return ts.isIdentifier(p.name) && !p.symbol.isReferenced; }) :
+                index === parameters.length - 1;
+        }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
 /* @internal */
@@ -132319,19 +138918,19 @@ var ts;
             var container = (ts.isBlock(statement.parent) ? statement.parent : statement).parent;
             if (!ts.isBlock(statement.parent) || statement === ts.first(statement.parent.statements)) {
                 switch (container.kind) {
-                    case 227 /* IfStatement */:
+                    case 234 /* IfStatement */:
                         if (container.elseStatement) {
                             if (ts.isBlock(statement.parent)) {
                                 break;
                             }
                             else {
-                                changes.replaceNode(sourceFile, statement, ts.createBlock(ts.emptyArray));
+                                changes.replaceNode(sourceFile, statement, ts.factory.createBlock(ts.emptyArray));
                             }
                             return;
                         }
                     // falls through
-                    case 229 /* WhileStatement */:
-                    case 230 /* ForStatement */:
+                    case 236 /* WhileStatement */:
+                    case 237 /* ForStatement */:
                         changes.delete(sourceFile, container);
                         return;
                 }
@@ -132404,7 +139003,7 @@ var ts;
                 var typeNode = info.typeNode, type = info.type;
                 var original = typeNode.getText(sourceFile);
                 var actions = [fix(type, fixIdPlain, ts.Diagnostics.Change_all_jsdoc_style_types_to_TypeScript)];
-                if (typeNode.kind === 297 /* JSDocNullableType */) {
+                if (typeNode.kind === 305 /* JSDocNullableType */) {
                     // for nullable types, suggest the flow-compatible `T | null | undefined`
                     // in addition to the jsdoc/closure-compatible `T | null`
                     actions.push(fix(checker.getNullableType(type, 32768 /* Undefined */), fixIdNullable, ts.Diagnostics.Change_all_jsdoc_style_types_to_TypeScript_and_add_undefined_to_nullable_types));
@@ -132424,7 +139023,7 @@ var ts;
                     if (!info)
                         return;
                     var typeNode = info.typeNode, type = info.type;
-                    var fixedType = typeNode.kind === 297 /* JSDocNullableType */ && fixId === fixIdNullable ? checker.getNullableType(type, 32768 /* Undefined */) : type;
+                    var fixedType = typeNode.kind === 305 /* JSDocNullableType */ && fixId === fixIdNullable ? checker.getNullableType(type, 32768 /* Undefined */) : type;
                     doChange(changes, sourceFile, typeNode, fixedType, checker);
                 });
             }
@@ -132441,22 +139040,22 @@ var ts;
             // NOTE: Some locations are not handled yet:
             // MappedTypeNode.typeParameters and SignatureDeclaration.typeParameters, as well as CallExpression.typeArguments
             switch (node.kind) {
-                case 217 /* AsExpression */:
-                case 165 /* CallSignature */:
-                case 166 /* ConstructSignature */:
-                case 244 /* FunctionDeclaration */:
-                case 163 /* GetAccessor */:
-                case 167 /* IndexSignature */:
-                case 186 /* MappedType */:
-                case 161 /* MethodDeclaration */:
-                case 160 /* MethodSignature */:
-                case 156 /* Parameter */:
-                case 159 /* PropertyDeclaration */:
-                case 158 /* PropertySignature */:
-                case 164 /* SetAccessor */:
-                case 247 /* TypeAliasDeclaration */:
-                case 199 /* TypeAssertionExpression */:
-                case 242 /* VariableDeclaration */:
+                case 224 /* AsExpression */:
+                case 169 /* CallSignature */:
+                case 170 /* ConstructSignature */:
+                case 251 /* FunctionDeclaration */:
+                case 167 /* GetAccessor */:
+                case 171 /* IndexSignature */:
+                case 190 /* MappedType */:
+                case 165 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 160 /* Parameter */:
+                case 163 /* PropertyDeclaration */:
+                case 162 /* PropertySignature */:
+                case 168 /* SetAccessor */:
+                case 254 /* TypeAliasDeclaration */:
+                case 206 /* TypeAssertionExpression */:
+                case 249 /* VariableDeclaration */:
                     return true;
                 default:
                     return false;
@@ -132531,7 +139130,7 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var seen = ts.createMap();
+                var seen = new ts.Map();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
                     var nodes = getNodes(diag.file, diag.start);
                     if (!nodes || !ts.addToSeen(seen, ts.getNodeId(nodes.insertBefore)))
@@ -132558,14 +139157,14 @@ var ts;
             }
             var insertBefore;
             switch (containingFunction.kind) {
-                case 161 /* MethodDeclaration */:
+                case 165 /* MethodDeclaration */:
                     insertBefore = containingFunction.name;
                     break;
-                case 244 /* FunctionDeclaration */:
-                case 201 /* FunctionExpression */:
-                    insertBefore = ts.findChildOfKind(containingFunction, 94 /* FunctionKeyword */, sourceFile);
+                case 251 /* FunctionDeclaration */:
+                case 208 /* FunctionExpression */:
+                    insertBefore = ts.findChildOfKind(containingFunction, 97 /* FunctionKeyword */, sourceFile);
                     break;
-                case 202 /* ArrowFunction */:
+                case 209 /* ArrowFunction */:
                     insertBefore = ts.findChildOfKind(containingFunction, 20 /* OpenParenToken */, sourceFile) || ts.first(containingFunction.parameters);
                     break;
                 default:
@@ -132580,11 +139179,71 @@ var ts;
             var insertBefore = _a.insertBefore, returnType = _a.returnType;
             if (returnType) {
                 var entityName = ts.getEntityNameFromTypeNode(returnType);
-                if (!entityName || entityName.kind !== 75 /* Identifier */ || entityName.text !== "Promise") {
-                    changes.replaceNode(sourceFile, returnType, ts.createTypeReferenceNode("Promise", ts.createNodeArray([returnType])));
+                if (!entityName || entityName.kind !== 78 /* Identifier */ || entityName.text !== "Promise") {
+                    changes.replaceNode(sourceFile, returnType, ts.factory.createTypeReferenceNode("Promise", ts.factory.createNodeArray([returnType])));
                 }
             }
-            changes.insertModifierBefore(sourceFile, 126 /* AsyncKeyword */, insertBefore);
+            changes.insertModifierBefore(sourceFile, 129 /* AsyncKeyword */, insertBefore);
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var errorCodes = [
+            ts.Diagnostics._0_is_defined_as_an_accessor_in_class_1_but_is_overridden_here_in_2_as_an_instance_property.code,
+            ts.Diagnostics._0_is_defined_as_a_property_in_class_1_but_is_overridden_here_in_2_as_an_accessor.code,
+        ];
+        var fixId = "fixPropertyOverrideAccessor";
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            getCodeActions: function (context) {
+                var edits = doChange(context.sourceFile, context.span.start, context.span.length, context.errorCode, context);
+                if (edits) {
+                    return [codefix.createCodeFixAction(fixId, edits, ts.Diagnostics.Generate_get_and_set_accessors, fixId, ts.Diagnostics.Generate_get_and_set_accessors_for_all_overriding_properties)];
+                }
+            },
+            fixIds: [fixId],
+            getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
+                var edits = doChange(diag.file, diag.start, diag.length, diag.code, context);
+                if (edits) {
+                    for (var _i = 0, edits_2 = edits; _i < edits_2.length; _i++) {
+                        var edit = edits_2[_i];
+                        changes.pushRaw(context.sourceFile, edit);
+                    }
+                }
+            }); },
+        });
+        function doChange(file, start, length, code, context) {
+            var startPosition;
+            var endPosition;
+            if (code === ts.Diagnostics._0_is_defined_as_an_accessor_in_class_1_but_is_overridden_here_in_2_as_an_instance_property.code) {
+                startPosition = start;
+                endPosition = start + length;
+            }
+            else if (code === ts.Diagnostics._0_is_defined_as_a_property_in_class_1_but_is_overridden_here_in_2_as_an_accessor.code) {
+                var checker = context.program.getTypeChecker();
+                var node = ts.getTokenAtPosition(file, start).parent;
+                ts.Debug.assert(ts.isAccessor(node), "error span of fixPropertyOverrideAccessor should only be on an accessor");
+                var containingClass = node.parent;
+                ts.Debug.assert(ts.isClassLike(containingClass), "erroneous accessors should only be inside classes");
+                var base = ts.singleOrUndefined(codefix.getAllSupers(containingClass, checker));
+                if (!base)
+                    return [];
+                var name = ts.unescapeLeadingUnderscores(ts.getTextOfPropertyName(node.name));
+                var baseProp = checker.getPropertyOfType(checker.getTypeAtLocation(base), name);
+                if (!baseProp || !baseProp.valueDeclaration)
+                    return [];
+                startPosition = baseProp.valueDeclaration.pos;
+                endPosition = baseProp.valueDeclaration.end;
+                file = ts.getSourceFileOfNode(baseProp.valueDeclaration);
+            }
+            else {
+                ts.Debug.fail("fixPropertyOverrideAccessor codefix got unexpected error code " + code);
+            }
+            return codefix.generateAccessorFromProperty(file, context.program, startPosition, endPosition, context, ts.Diagnostics.Generate_get_and_set_accessors.message);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -132684,7 +139343,7 @@ var ts;
             return errorCode;
         }
         function doChange(changes, sourceFile, token, errorCode, program, cancellationToken, markSeen, host, preferences) {
-            if (!ts.isParameterPropertyModifier(token.kind) && token.kind !== 75 /* Identifier */ && token.kind !== 25 /* DotDotDotToken */ && token.kind !== 104 /* ThisKeyword */) {
+            if (!ts.isParameterPropertyModifier(token.kind) && token.kind !== 78 /* Identifier */ && token.kind !== 25 /* DotDotDotToken */ && token.kind !== 107 /* ThisKeyword */) {
                 return undefined;
             }
             var parent = token.parent;
@@ -132704,7 +139363,7 @@ var ts;
                         var typeNode = ts.getTypeNodeIfAccessible(type, parent, program, host);
                         if (typeNode) {
                             // Note that the codefix will never fire with an existing `@type` tag, so there is no need to merge tags
-                            var typeTag = ts.createJSDocTypeTag(ts.createJSDocTypeExpression(typeNode), /*comment*/ "");
+                            var typeTag = ts.factory.createJSDocTypeTag(/*tagName*/ undefined, ts.factory.createJSDocTypeExpression(typeNode), /*comment*/ "");
                             addJSDocTags(changes, sourceFile, ts.cast(parent.parent.parent, ts.isExpressionStatement), [typeTag]);
                         }
                         importAdder.writeFixes(changes);
@@ -132787,7 +139446,7 @@ var ts;
             else {
                 var needParens = ts.isArrowFunction(containingFunction) && !ts.findChildOfKind(containingFunction, 20 /* OpenParenToken */, sourceFile);
                 if (needParens)
-                    changes.insertNodeBefore(sourceFile, ts.first(containingFunction.parameters), ts.createToken(20 /* OpenParenToken */));
+                    changes.insertNodeBefore(sourceFile, ts.first(containingFunction.parameters), ts.factory.createToken(20 /* OpenParenToken */));
                 for (var _i = 0, parameterInferences_1 = parameterInferences; _i < parameterInferences_1.length; _i++) {
                     var _a = parameterInferences_1[_i], declaration = _a.declaration, type = _a.type;
                     if (declaration && !declaration.type && !declaration.initializer) {
@@ -132795,7 +139454,7 @@ var ts;
                     }
                 }
                 if (needParens)
-                    changes.insertNodeAfter(sourceFile, ts.last(containingFunction.parameters), ts.createToken(21 /* CloseParenToken */));
+                    changes.insertNodeAfter(sourceFile, ts.last(containingFunction.parameters), ts.factory.createToken(21 /* CloseParenToken */));
             }
         }
         function annotateThis(changes, sourceFile, containingFunction, program, host, cancellationToken) {
@@ -132817,7 +139476,7 @@ var ts;
         }
         function annotateJSDocThis(changes, sourceFile, containingFunction, typeNode) {
             addJSDocTags(changes, sourceFile, containingFunction, [
-                ts.createJSDocThisTag(ts.createJSDocTypeExpression(typeNode)),
+                ts.factory.createJSDocThisTag(/*tagName*/ undefined, ts.factory.createJSDocTypeExpression(typeNode)),
             ]);
         }
         function annotateSetAccessor(changes, importAdder, sourceFile, setAccessorDeclaration, program, host, cancellationToken) {
@@ -132838,23 +139497,23 @@ var ts;
         function annotate(changes, importAdder, sourceFile, declaration, type, program, host) {
             var typeNode = ts.getTypeNodeIfAccessible(type, declaration, program, host);
             if (typeNode) {
-                if (ts.isInJSFile(sourceFile) && declaration.kind !== 158 /* PropertySignature */) {
+                if (ts.isInJSFile(sourceFile) && declaration.kind !== 162 /* PropertySignature */) {
                     var parent = ts.isVariableDeclaration(declaration) ? ts.tryCast(declaration.parent.parent, ts.isVariableStatement) : declaration;
                     if (!parent) {
                         return;
                     }
-                    var typeExpression = ts.createJSDocTypeExpression(typeNode);
-                    var typeTag = ts.isGetAccessorDeclaration(declaration) ? ts.createJSDocReturnTag(typeExpression, "") : ts.createJSDocTypeTag(typeExpression, "");
+                    var typeExpression = ts.factory.createJSDocTypeExpression(typeNode);
+                    var typeTag = ts.isGetAccessorDeclaration(declaration) ? ts.factory.createJSDocReturnTag(/*tagName*/ undefined, typeExpression, "") : ts.factory.createJSDocTypeTag(/*tagName*/ undefined, typeExpression, "");
                     addJSDocTags(changes, sourceFile, parent, [typeTag]);
                 }
-                else if (!tryReplaceImportTypeNodeWithAutoImport(typeNode, declaration, type, sourceFile, changes, importAdder, ts.getEmitScriptTarget(program.getCompilerOptions()))) {
+                else if (!tryReplaceImportTypeNodeWithAutoImport(typeNode, declaration, sourceFile, changes, importAdder, ts.getEmitScriptTarget(program.getCompilerOptions()))) {
                     changes.tryInsertTypeAnnotation(sourceFile, declaration, typeNode);
                 }
             }
         }
-        function tryReplaceImportTypeNodeWithAutoImport(typeNode, declaration, type, sourceFile, changes, importAdder, scriptTarget) {
-            var importableReference = codefix.tryGetAutoImportableReferenceFromImportTypeNode(typeNode, type, scriptTarget);
-            if (importableReference && changes.tryInsertTypeAnnotation(sourceFile, declaration, importableReference.typeReference)) {
+        function tryReplaceImportTypeNodeWithAutoImport(typeNode, declaration, sourceFile, changes, importAdder, scriptTarget) {
+            var importableReference = codefix.tryGetAutoImportableReferenceFromTypeNode(typeNode, scriptTarget);
+            if (importableReference && changes.tryInsertTypeAnnotation(sourceFile, declaration, importableReference.typeNode)) {
                 ts.forEach(importableReference.symbols, function (s) { return importAdder.addImportFromExportedSymbol(s, /*usageIsTypeOnly*/ true); });
                 return true;
             }
@@ -132871,9 +139530,9 @@ var ts;
                 if (param.initializer || ts.getJSDocType(param) || !ts.isIdentifier(param.name))
                     return;
                 var typeNode = inference.type && ts.getTypeNodeIfAccessible(inference.type, param, program, host);
-                var name = ts.getSynthesizedClone(param.name);
+                var name = ts.factory.cloneNode(param.name);
                 ts.setEmitFlags(name, 1536 /* NoComments */ | 2048 /* NoNestedComments */);
-                return typeNode && ts.createJSDocParamTag(name, !!inference.isOptional, ts.createJSDocTypeExpression(typeNode), "");
+                return typeNode && ts.factory.createJSDocParameterTag(/*tagName*/ undefined, name, /*isBracketed*/ !!inference.isOptional, ts.factory.createJSDocTypeExpression(typeNode), /* isNameFirst */ false, "");
             });
             addJSDocTags(changes, sourceFile, signature, paramTags);
         }
@@ -132886,15 +139545,15 @@ var ts;
                     oldTags[i] = merged;
                 return !!merged;
             }); });
-            var tag = ts.createJSDocComment(comments.join("\n"), ts.createNodeArray(__spreadArrays((oldTags || ts.emptyArray), unmergedNewTags)));
-            var jsDocNode = parent.kind === 202 /* ArrowFunction */ ? getJsDocNodeForArrowFunction(parent) : parent;
+            var tag = ts.factory.createJSDocComment(comments.join("\n"), ts.factory.createNodeArray(__spreadArrays((oldTags || ts.emptyArray), unmergedNewTags)));
+            var jsDocNode = parent.kind === 209 /* ArrowFunction */ ? getJsDocNodeForArrowFunction(parent) : parent;
             jsDocNode.jsDoc = parent.jsDoc;
             jsDocNode.jsDocCache = parent.jsDocCache;
             changes.insertJsdocCommentBefore(sourceFile, jsDocNode, tag);
         }
         codefix.addJSDocTags = addJSDocTags;
         function getJsDocNodeForArrowFunction(signature) {
-            if (signature.parent.kind === 159 /* PropertyDeclaration */) {
+            if (signature.parent.kind === 163 /* PropertyDeclaration */) {
                 return signature.parent;
             }
             return signature.parent.parent;
@@ -132904,15 +139563,15 @@ var ts;
                 return undefined;
             }
             switch (oldTag.kind) {
-                case 317 /* JSDocParameterTag */: {
+                case 326 /* JSDocParameterTag */: {
                     var oldParam = oldTag;
                     var newParam = newTag;
                     return ts.isIdentifier(oldParam.name) && ts.isIdentifier(newParam.name) && oldParam.name.escapedText === newParam.name.escapedText
-                        ? ts.createJSDocParamTag(newParam.name, newParam.isBracketed, newParam.typeExpression, oldParam.comment)
+                        ? ts.factory.createJSDocParameterTag(/*tagName*/ undefined, newParam.name, /*isBracketed*/ false, newParam.typeExpression, newParam.isNameFirst, oldParam.comment)
                         : undefined;
                 }
-                case 318 /* JSDocReturnTag */:
-                    return ts.createJSDocReturnTag(newTag.typeExpression, oldTag.comment);
+                case 327 /* JSDocReturnTag */:
+                    return ts.factory.createJSDocReturnTag(/*tagName*/ undefined, newTag.typeExpression, oldTag.comment);
             }
         }
         function getReferences(token, program, cancellationToken) {
@@ -132936,18 +139595,18 @@ var ts;
         function getFunctionReferences(containingFunction, sourceFile, program, cancellationToken) {
             var searchToken;
             switch (containingFunction.kind) {
-                case 162 /* Constructor */:
-                    searchToken = ts.findChildOfKind(containingFunction, 129 /* ConstructorKeyword */, sourceFile);
+                case 166 /* Constructor */:
+                    searchToken = ts.findChildOfKind(containingFunction, 132 /* ConstructorKeyword */, sourceFile);
                     break;
-                case 202 /* ArrowFunction */:
-                case 201 /* FunctionExpression */:
+                case 209 /* ArrowFunction */:
+                case 208 /* FunctionExpression */:
                     var parent = containingFunction.parent;
                     searchToken = ts.isVariableDeclaration(parent) && ts.isIdentifier(parent.name) ?
                         parent.name :
                         containingFunction.name;
                     break;
-                case 244 /* FunctionDeclaration */:
-                case 161 /* MethodDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 165 /* MethodDeclaration */:
                     searchToken = containingFunction.name;
                     break;
             }
@@ -132991,7 +139650,7 @@ var ts;
                 };
             }
             function combineUsages(usages) {
-                var combinedProperties = ts.createUnderscoreEscapedMap();
+                var combinedProperties = new ts.Map();
                 for (var _i = 0, usages_1 = usages; _i < usages_1.length; _i++) {
                     var u = usages_1[_i];
                     if (u.properties) {
@@ -133003,7 +139662,7 @@ var ts;
                         });
                     }
                 }
-                var properties = ts.createUnderscoreEscapedMap();
+                var properties = new ts.Map();
                 combinedProperties.forEach(function (ps, name) {
                     properties.set(name, combineUsages(ps));
                 });
@@ -133089,24 +139748,24 @@ var ts;
                     node = node.parent;
                 }
                 switch (node.parent.kind) {
-                    case 226 /* ExpressionStatement */:
+                    case 233 /* ExpressionStatement */:
                         inferTypeFromExpressionStatement(node, usage);
                         break;
-                    case 208 /* PostfixUnaryExpression */:
+                    case 215 /* PostfixUnaryExpression */:
                         usage.isNumber = true;
                         break;
-                    case 207 /* PrefixUnaryExpression */:
+                    case 214 /* PrefixUnaryExpression */:
                         inferTypeFromPrefixUnaryExpression(node.parent, usage);
                         break;
-                    case 209 /* BinaryExpression */:
+                    case 216 /* BinaryExpression */:
                         inferTypeFromBinaryExpression(node, node.parent, usage);
                         break;
-                    case 277 /* CaseClause */:
-                    case 278 /* DefaultClause */:
+                    case 284 /* CaseClause */:
+                    case 285 /* DefaultClause */:
                         inferTypeFromSwitchStatementLabel(node.parent, usage);
                         break;
-                    case 196 /* CallExpression */:
-                    case 197 /* NewExpression */:
+                    case 203 /* CallExpression */:
+                    case 204 /* NewExpression */:
                         if (node.parent.expression === node) {
                             inferTypeFromCallExpression(node.parent, usage);
                         }
@@ -133114,20 +139773,20 @@ var ts;
                             inferTypeFromContextualType(node, usage);
                         }
                         break;
-                    case 194 /* PropertyAccessExpression */:
+                    case 201 /* PropertyAccessExpression */:
                         inferTypeFromPropertyAccessExpression(node.parent, usage);
                         break;
-                    case 195 /* ElementAccessExpression */:
+                    case 202 /* ElementAccessExpression */:
                         inferTypeFromPropertyElementExpression(node.parent, node, usage);
                         break;
-                    case 281 /* PropertyAssignment */:
-                    case 282 /* ShorthandPropertyAssignment */:
+                    case 288 /* PropertyAssignment */:
+                    case 289 /* ShorthandPropertyAssignment */:
                         inferTypeFromPropertyAssignment(node.parent, usage);
                         break;
-                    case 159 /* PropertyDeclaration */:
+                    case 163 /* PropertyDeclaration */:
                         inferTypeFromPropertyDeclaration(node.parent, usage);
                         break;
-                    case 242 /* VariableDeclaration */: {
+                    case 249 /* VariableDeclaration */: {
                         var _a = node.parent, name = _a.name, initializer = _a.initializer;
                         if (node === name) {
                             if (initializer) { // This can happen for `let x = null;` which still has an implicit-any error.
@@ -133192,7 +139851,7 @@ var ts;
                     case 68 /* PercentEqualsToken */:
                     case 72 /* AmpersandEqualsToken */:
                     case 73 /* BarEqualsToken */:
-                    case 74 /* CaretEqualsToken */:
+                    case 77 /* CaretEqualsToken */:
                     case 69 /* LessThanLessThanEqualsToken */:
                     case 71 /* GreaterThanGreaterThanGreaterThanEqualsToken */:
                     case 70 /* GreaterThanGreaterThanEqualsToken */:
@@ -133222,7 +139881,7 @@ var ts;
                         else if (otherOperandType.flags & 296 /* NumberLike */) {
                             usage.isNumber = true;
                         }
-                        else if (otherOperandType.flags & 132 /* StringLike */) {
+                        else if (otherOperandType.flags & 402653316 /* StringLike */) {
                             usage.isString = true;
                         }
                         else if (otherOperandType.flags & 1 /* Any */) {
@@ -133240,7 +139899,7 @@ var ts;
                     case 35 /* ExclamationEqualsToken */:
                         addCandidateType(usage, checker.getTypeAtLocation(parent.left === node ? parent.right : parent.left));
                         break;
-                    case 97 /* InKeyword */:
+                    case 100 /* InKeyword */:
                         if (node === parent.left) {
                             usage.isString = true;
                         }
@@ -133249,7 +139908,7 @@ var ts;
                     case 56 /* BarBarToken */:
                     case 60 /* QuestionQuestionToken */:
                         if (node === parent.left &&
-                            (node.parent.parent.kind === 242 /* VariableDeclaration */ || ts.isAssignmentExpression(node.parent.parent, /*excludeCompoundAssignment*/ true))) {
+                            (node.parent.parent.kind === 249 /* VariableDeclaration */ || ts.isAssignmentExpression(node.parent.parent, /*excludeCompoundAssignment*/ true))) {
                             // var x = x || {};
                             // TODO: use getFalsyflagsOfType
                             addCandidateType(usage, checker.getTypeAtLocation(parent.right));
@@ -133257,7 +139916,7 @@ var ts;
                         break;
                     case 55 /* AmpersandAmpersandToken */:
                     case 27 /* CommaToken */:
-                    case 98 /* InstanceOfKeyword */:
+                    case 101 /* InstanceOfKeyword */:
                         // nothing to infer here
                         break;
                 }
@@ -133277,7 +139936,7 @@ var ts;
                     }
                 }
                 calculateUsageOfNode(parent, call.return_);
-                if (parent.kind === 196 /* CallExpression */) {
+                if (parent.kind === 203 /* CallExpression */) {
                     (usage.calls || (usage.calls = [])).push(call);
                 }
                 else {
@@ -133287,7 +139946,7 @@ var ts;
             function inferTypeFromPropertyAccessExpression(parent, usage) {
                 var name = ts.escapeLeadingUnderscores(parent.name.text);
                 if (!usage.properties) {
-                    usage.properties = ts.createUnderscoreEscapedMap();
+                    usage.properties = new ts.Map();
                 }
                 var propertyUsage = usage.properties.get(name) || createEmptyUsage();
                 calculateUsageOfNode(parent, propertyUsage);
@@ -133424,7 +140083,7 @@ var ts;
                 return types;
             }
             function inferStructuralType(usage) {
-                var members = ts.createUnderscoreEscapedMap();
+                var members = new ts.Map();
                 if (usage.properties) {
                     usage.properties.forEach(function (u, name) {
                         var symbol = checker.createSymbol(4 /* Property */, name);
@@ -133570,6 +140229,59 @@ var ts;
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixId = "fixReturnTypeInAsyncFunction";
+        var errorCodes = [
+            ts.Diagnostics.The_return_type_of_an_async_function_or_method_must_be_the_global_Promise_T_type_Did_you_mean_to_write_Promise_0.code,
+        ];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            fixIds: [fixId],
+            getCodeActions: function (context) {
+                var sourceFile = context.sourceFile, program = context.program, span = context.span;
+                var checker = program.getTypeChecker();
+                var info = getInfo(sourceFile, program.getTypeChecker(), span.start);
+                if (!info) {
+                    return undefined;
+                }
+                var returnTypeNode = info.returnTypeNode, returnType = info.returnType, promisedTypeNode = info.promisedTypeNode, promisedType = info.promisedType;
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, sourceFile, returnTypeNode, promisedTypeNode); });
+                return [codefix.createCodeFixAction(fixId, changes, [ts.Diagnostics.Replace_0_with_Promise_1,
+                        checker.typeToString(returnType), checker.typeToString(promisedType)], fixId, ts.Diagnostics.Fix_all_incorrect_return_type_of_an_async_functions)];
+            },
+            getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
+                var info = getInfo(diag.file, context.program.getTypeChecker(), diag.start);
+                if (info) {
+                    doChange(changes, diag.file, info.returnTypeNode, info.promisedTypeNode);
+                }
+            }); }
+        });
+        function getInfo(sourceFile, checker, pos) {
+            if (ts.isInJSFile(sourceFile)) {
+                return undefined;
+            }
+            var token = ts.getTokenAtPosition(sourceFile, pos);
+            var func = ts.findAncestor(token, ts.isFunctionLikeDeclaration);
+            var returnTypeNode = func === null || func === void 0 ? void 0 : func.type;
+            if (!returnTypeNode) {
+                return undefined;
+            }
+            var returnType = checker.getTypeFromTypeNode(returnTypeNode);
+            var promisedType = checker.getAwaitedType(returnType) || checker.getVoidType();
+            var promisedTypeNode = checker.typeToTypeNode(promisedType, /*enclosingDeclaration*/ returnTypeNode, /*flags*/ undefined);
+            if (promisedTypeNode) {
+                return { returnTypeNode: returnTypeNode, returnType: returnType, promisedTypeNode: promisedTypeNode, promisedType: promisedType };
+            }
+        }
+        function doChange(changes, sourceFile, returnTypeNode, promisedTypeNode) {
+            changes.replaceNode(sourceFile, returnTypeNode, ts.factory.createTypeReferenceNode("Promise", [promisedTypeNode]));
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var codefix;
     (function (codefix) {
@@ -133602,7 +140314,7 @@ var ts;
             },
             fixIds: [fixId],
             getAllCodeActions: function (context) {
-                var seenLines = ts.createMap();
+                var seenLines = new ts.Set();
                 return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
                     if (ts.textChanges.isValidLocationToAddComment(diag.file, diag.start)) {
                         makeChange(changes, diag.file, diag.start, seenLines);
@@ -133613,7 +140325,7 @@ var ts;
         function makeChange(changes, sourceFile, position, seenLines) {
             var lineNumber = ts.getLineAndCharacterOfPosition(sourceFile, position).line;
             // Only need to add `// @ts-ignore` for a line once.
-            if (!seenLines || ts.addToSeen(seenLines, lineNumber)) {
+            if (!seenLines || ts.tryAddToSet(seenLines, lineNumber)) {
                 changes.insertCommentBeforeLine(sourceFile, lineNumber, position, " @ts-ignore");
             }
         }
@@ -133631,12 +140343,12 @@ var ts;
          * @param importAdder If provided, type annotations will use identifier type references instead of ImportTypeNodes, and the missing imports will be added to the importAdder.
          * @returns Empty string iff there are no member insertions.
          */
-        function createMissingMemberNodes(classDeclaration, possiblyMissingSymbols, context, preferences, importAdder, addClassElement) {
+        function createMissingMemberNodes(classDeclaration, possiblyMissingSymbols, sourceFile, context, preferences, importAdder, addClassElement) {
             var classMembers = classDeclaration.symbol.members;
             for (var _i = 0, possiblyMissingSymbols_1 = possiblyMissingSymbols; _i < possiblyMissingSymbols_1.length; _i++) {
                 var symbol = possiblyMissingSymbols_1[_i];
                 if (!classMembers.has(symbol.escapedName)) {
-                    addNewNodeForMemberSymbol(symbol, classDeclaration, context, preferences, importAdder, addClassElement);
+                    addNewNodeForMemberSymbol(symbol, classDeclaration, sourceFile, context, preferences, importAdder, addClassElement);
                 }
             }
         }
@@ -133651,7 +140363,7 @@ var ts;
         /**
          * @returns Empty string iff there we can't figure out a representation for `symbol` in `enclosingDeclaration`.
          */
-        function addNewNodeForMemberSymbol(symbol, enclosingDeclaration, context, preferences, importAdder, addClassElement) {
+        function addNewNodeForMemberSymbol(symbol, enclosingDeclaration, sourceFile, context, preferences, importAdder, addClassElement) {
             var declarations = symbol.getDeclarations();
             if (!(declarations && declarations.length)) {
                 return undefined;
@@ -133660,59 +140372,60 @@ var ts;
             var scriptTarget = ts.getEmitScriptTarget(context.program.getCompilerOptions());
             var declaration = declarations[0];
             var name = ts.getSynthesizedDeepClone(ts.getNameOfDeclaration(declaration), /*includeTrivia*/ false);
-            var visibilityModifier = createVisibilityModifier(ts.getModifierFlags(declaration));
-            var modifiers = visibilityModifier ? ts.createNodeArray([visibilityModifier]) : undefined;
+            var visibilityModifier = createVisibilityModifier(ts.getEffectiveModifierFlags(declaration));
+            var modifiers = visibilityModifier ? ts.factory.createNodeArray([visibilityModifier]) : undefined;
             var type = checker.getWidenedType(checker.getTypeOfSymbolAtLocation(symbol, enclosingDeclaration));
             var optional = !!(symbol.flags & 16777216 /* Optional */);
             var ambient = !!(enclosingDeclaration.flags & 8388608 /* Ambient */);
+            var quotePreference = ts.getQuotePreference(sourceFile, preferences);
             switch (declaration.kind) {
-                case 158 /* PropertySignature */:
-                case 159 /* PropertyDeclaration */:
-                    var flags = preferences.quotePreference === "single" ? 268435456 /* UseSingleQuotesForStringLiteralType */ : undefined;
+                case 162 /* PropertySignature */:
+                case 163 /* PropertyDeclaration */:
+                    var flags = quotePreference === 0 /* Single */ ? 268435456 /* UseSingleQuotesForStringLiteralType */ : undefined;
                     var typeNode = checker.typeToTypeNode(type, enclosingDeclaration, flags, getNoopSymbolTrackerWithResolver(context));
                     if (importAdder) {
-                        var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(typeNode, type, scriptTarget);
+                        var importableReference = tryGetAutoImportableReferenceFromTypeNode(typeNode, scriptTarget);
                         if (importableReference) {
-                            typeNode = importableReference.typeReference;
+                            typeNode = importableReference.typeNode;
                             importSymbols(importAdder, importableReference.symbols);
                         }
                     }
-                    addClassElement(ts.createProperty(
-                    /*decorators*/ undefined, modifiers, name, optional ? ts.createToken(57 /* QuestionToken */) : undefined, typeNode, 
+                    addClassElement(ts.factory.createPropertyDeclaration(
+                    /*decorators*/ undefined, modifiers, name, optional ? ts.factory.createToken(57 /* QuestionToken */) : undefined, typeNode, 
                     /*initializer*/ undefined));
                     break;
-                case 163 /* GetAccessor */:
-                case 164 /* SetAccessor */: {
+                case 167 /* GetAccessor */:
+                case 168 /* SetAccessor */: {
                     var typeNode_1 = checker.typeToTypeNode(type, enclosingDeclaration, /*flags*/ undefined, getNoopSymbolTrackerWithResolver(context));
                     var allAccessors = ts.getAllAccessorDeclarations(declarations, declaration);
                     var orderedAccessors = allAccessors.secondAccessor
                         ? [allAccessors.firstAccessor, allAccessors.secondAccessor]
                         : [allAccessors.firstAccessor];
                     if (importAdder) {
-                        var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(typeNode_1, type, scriptTarget);
+                        var importableReference = tryGetAutoImportableReferenceFromTypeNode(typeNode_1, scriptTarget);
                         if (importableReference) {
-                            typeNode_1 = importableReference.typeReference;
+                            typeNode_1 = importableReference.typeNode;
                             importSymbols(importAdder, importableReference.symbols);
                         }
                     }
                     for (var _i = 0, orderedAccessors_1 = orderedAccessors; _i < orderedAccessors_1.length; _i++) {
                         var accessor = orderedAccessors_1[_i];
                         if (ts.isGetAccessorDeclaration(accessor)) {
-                            addClassElement(ts.createGetAccessor(
-                            /*decorators*/ undefined, modifiers, name, ts.emptyArray, typeNode_1, ambient ? undefined : createStubbedMethodBody(preferences)));
+                            addClassElement(ts.factory.createGetAccessorDeclaration(
+                            /*decorators*/ undefined, modifiers, name, ts.emptyArray, typeNode_1, ambient ? undefined : createStubbedMethodBody(quotePreference)));
                         }
                         else {
                             ts.Debug.assertNode(accessor, ts.isSetAccessorDeclaration, "The counterpart to a getter should be a setter");
                             var parameter = ts.getSetAccessorValueParameter(accessor);
                             var parameterName = parameter && ts.isIdentifier(parameter.name) ? ts.idText(parameter.name) : undefined;
-                            addClassElement(ts.createSetAccessor(
-                            /*decorators*/ undefined, modifiers, name, createDummyParameters(1, [parameterName], [typeNode_1], 1, /*inJs*/ false), ambient ? undefined : createStubbedMethodBody(preferences)));
+                            addClassElement(ts.factory.createSetAccessorDeclaration(
+                            /*decorators*/ undefined, modifiers, name, createDummyParameters(1, [parameterName], [typeNode_1], 1, /*inJs*/ false), ambient ? undefined : createStubbedMethodBody(quotePreference)));
                         }
                     }
                     break;
                 }
-                case 160 /* MethodSignature */:
-                case 161 /* MethodDeclaration */:
+                case 164 /* MethodSignature */:
+                case 165 /* MethodDeclaration */:
                     // The signature for the implementation appears as an entry in `signatures` iff
                     // there is only one signature.
                     // If there are overloads and an implementation signature, it appears as an
@@ -133727,84 +140440,93 @@ var ts;
                     if (declarations.length === 1) {
                         ts.Debug.assert(signatures.length === 1, "One declaration implies one signature");
                         var signature = signatures[0];
-                        outputMethod(signature, modifiers, name, ambient ? undefined : createStubbedMethodBody(preferences));
+                        outputMethod(quotePreference, signature, modifiers, name, ambient ? undefined : createStubbedMethodBody(quotePreference));
                         break;
                     }
                     for (var _a = 0, signatures_1 = signatures; _a < signatures_1.length; _a++) {
                         var signature = signatures_1[_a];
                         // Need to ensure nodes are fresh each time so they can have different positions.
-                        outputMethod(signature, ts.getSynthesizedDeepClones(modifiers, /*includeTrivia*/ false), ts.getSynthesizedDeepClone(name, /*includeTrivia*/ false));
+                        outputMethod(quotePreference, signature, ts.getSynthesizedDeepClones(modifiers, /*includeTrivia*/ false), ts.getSynthesizedDeepClone(name, /*includeTrivia*/ false));
                     }
                     if (!ambient) {
                         if (declarations.length > signatures.length) {
                             var signature = checker.getSignatureFromDeclaration(declarations[declarations.length - 1]);
-                            outputMethod(signature, modifiers, name, createStubbedMethodBody(preferences));
+                            outputMethod(quotePreference, signature, modifiers, name, createStubbedMethodBody(quotePreference));
                         }
                         else {
                             ts.Debug.assert(declarations.length === signatures.length, "Declarations and signatures should match count");
-                            addClassElement(createMethodImplementingSignatures(signatures, name, optional, modifiers, preferences));
+                            addClassElement(createMethodImplementingSignatures(signatures, name, optional, modifiers, quotePreference));
                         }
                     }
                     break;
             }
-            function outputMethod(signature, modifiers, name, body) {
-                var method = signatureToMethodDeclaration(context, signature, enclosingDeclaration, modifiers, name, optional, body, importAdder);
+            function outputMethod(quotePreference, signature, modifiers, name, body) {
+                var method = signatureToMethodDeclaration(context, quotePreference, signature, enclosingDeclaration, modifiers, name, optional, body, importAdder);
                 if (method)
                     addClassElement(method);
             }
         }
-        function signatureToMethodDeclaration(context, signature, enclosingDeclaration, modifiers, name, optional, body, importAdder) {
+        function signatureToMethodDeclaration(context, quotePreference, signature, enclosingDeclaration, modifiers, name, optional, body, importAdder) {
             var program = context.program;
             var checker = program.getTypeChecker();
             var scriptTarget = ts.getEmitScriptTarget(program.getCompilerOptions());
-            var signatureDeclaration = checker.signatureToSignatureDeclaration(signature, 161 /* MethodDeclaration */, enclosingDeclaration, 1 /* NoTruncation */ | 256 /* SuppressAnyReturnType */, getNoopSymbolTrackerWithResolver(context));
+            var flags = 1 /* NoTruncation */ | 1073741824 /* NoUndefinedOptionalParameterType */ | 256 /* SuppressAnyReturnType */ | (quotePreference === 0 /* Single */ ? 268435456 /* UseSingleQuotesForStringLiteralType */ : 0);
+            var signatureDeclaration = checker.signatureToSignatureDeclaration(signature, 165 /* MethodDeclaration */, enclosingDeclaration, flags, getNoopSymbolTrackerWithResolver(context));
             if (!signatureDeclaration) {
                 return undefined;
             }
+            var typeParameters = signatureDeclaration.typeParameters;
+            var parameters = signatureDeclaration.parameters;
+            var type = signatureDeclaration.type;
             if (importAdder) {
-                if (signatureDeclaration.typeParameters) {
-                    ts.forEach(signatureDeclaration.typeParameters, function (typeParameterDecl, i) {
-                        var typeParameter = signature.typeParameters[i];
-                        if (typeParameterDecl.constraint) {
-                            var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(typeParameterDecl.constraint, typeParameter.constraint, scriptTarget);
+                if (typeParameters) {
+                    var newTypeParameters = ts.sameMap(typeParameters, function (typeParameterDecl) {
+                        var constraint = typeParameterDecl.constraint;
+                        var defaultType = typeParameterDecl.default;
+                        if (constraint) {
+                            var importableReference = tryGetAutoImportableReferenceFromTypeNode(constraint, scriptTarget);
                             if (importableReference) {
-                                typeParameterDecl.constraint = importableReference.typeReference;
+                                constraint = importableReference.typeNode;
                                 importSymbols(importAdder, importableReference.symbols);
                             }
                         }
-                        if (typeParameterDecl.default) {
-                            var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(typeParameterDecl.default, typeParameter.default, scriptTarget);
+                        if (defaultType) {
+                            var importableReference = tryGetAutoImportableReferenceFromTypeNode(defaultType, scriptTarget);
                             if (importableReference) {
-                                typeParameterDecl.default = importableReference.typeReference;
+                                defaultType = importableReference.typeNode;
                                 importSymbols(importAdder, importableReference.symbols);
                             }
                         }
+                        return ts.factory.updateTypeParameterDeclaration(typeParameterDecl, typeParameterDecl.name, constraint, defaultType);
                     });
+                    if (typeParameters !== newTypeParameters) {
+                        typeParameters = ts.setTextRange(ts.factory.createNodeArray(newTypeParameters, typeParameters.hasTrailingComma), typeParameters);
+                    }
                 }
-                ts.forEach(signatureDeclaration.parameters, function (parameterDecl, i) {
-                    var parameter = signature.parameters[i];
-                    var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(parameterDecl.type, checker.getTypeAtLocation(parameter.valueDeclaration), scriptTarget);
+                var newParameters = ts.sameMap(parameters, function (parameterDecl) {
+                    var importableReference = tryGetAutoImportableReferenceFromTypeNode(parameterDecl.type, scriptTarget);
+                    var type = parameterDecl.type;
                     if (importableReference) {
-                        parameterDecl.type = importableReference.typeReference;
+                        type = importableReference.typeNode;
                         importSymbols(importAdder, importableReference.symbols);
                     }
+                    return ts.factory.updateParameterDeclaration(parameterDecl, parameterDecl.decorators, parameterDecl.modifiers, parameterDecl.dotDotDotToken, parameterDecl.name, parameterDecl.questionToken, type, parameterDecl.initializer);
                 });
-                if (signatureDeclaration.type) {
-                    var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(signatureDeclaration.type, signature.resolvedReturnType, scriptTarget);
+                if (parameters !== newParameters) {
+                    parameters = ts.setTextRange(ts.factory.createNodeArray(newParameters, parameters.hasTrailingComma), parameters);
+                }
+                if (type) {
+                    var importableReference = tryGetAutoImportableReferenceFromTypeNode(type, scriptTarget);
                     if (importableReference) {
-                        signatureDeclaration.type = importableReference.typeReference;
+                        type = importableReference.typeNode;
                         importSymbols(importAdder, importableReference.symbols);
                     }
                 }
             }
-            signatureDeclaration.decorators = undefined;
-            signatureDeclaration.modifiers = modifiers;
-            signatureDeclaration.name = name;
-            signatureDeclaration.questionToken = optional ? ts.createToken(57 /* QuestionToken */) : undefined;
-            signatureDeclaration.body = body;
-            return signatureDeclaration;
+            return ts.factory.updateMethodDeclaration(signatureDeclaration, 
+            /*decorators*/ undefined, modifiers, signatureDeclaration.asteriskToken, name, optional ? ts.factory.createToken(57 /* QuestionToken */) : undefined, typeParameters, parameters, type, body);
         }
-        function createMethodFromCallExpression(context, call, methodName, inJs, makeStatic, contextNode, importAdder) {
+        function createMethodFromCallExpression(context, importAdder, call, methodName, modifierFlags, contextNode, inJs) {
             var body = !ts.isInterfaceDeclaration(contextNode);
             var typeArguments = call.typeArguments, args = call.arguments, parent = call.parent;
             var scriptTarget = ts.getEmitScriptTarget(context.program.getCompilerOptions());
@@ -133818,25 +140540,26 @@ var ts;
             });
             var contextualType = checker.getContextualType(call);
             var returnType = (inJs || !contextualType) ? undefined : checker.typeToTypeNode(contextualType, contextNode, /*flags*/ undefined, tracker);
-            return ts.createMethod(
+            var quotePreference = ts.getQuotePreference(context.sourceFile, context.preferences);
+            return ts.factory.createMethodDeclaration(
             /*decorators*/ undefined, 
-            /*modifiers*/ makeStatic ? [ts.createToken(120 /* StaticKeyword */)] : undefined, 
-            /*asteriskToken*/ ts.isYieldExpression(parent) ? ts.createToken(41 /* AsteriskToken */) : undefined, methodName, 
+            /*modifiers*/ modifierFlags ? ts.factory.createNodeArray(ts.factory.createModifiersFromModifierFlags(modifierFlags)) : undefined, 
+            /*asteriskToken*/ ts.isYieldExpression(parent) ? ts.factory.createToken(41 /* AsteriskToken */) : undefined, methodName, 
             /*questionToken*/ undefined, 
             /*typeParameters*/ inJs ? undefined : ts.map(typeArguments, function (_, i) {
-                return ts.createTypeParameterDeclaration(84 /* T */ + typeArguments.length - 1 <= 90 /* Z */ ? String.fromCharCode(84 /* T */ + i) : "T" + i);
+                return ts.factory.createTypeParameterDeclaration(84 /* T */ + typeArguments.length - 1 <= 90 /* Z */ ? String.fromCharCode(84 /* T */ + i) : "T" + i);
             }), 
             /*parameters*/ createDummyParameters(args.length, names, types, /*minArgumentCount*/ undefined, inJs), 
-            /*type*/ returnType, body ? createStubbedMethodBody(context.preferences) : undefined);
+            /*type*/ returnType, body ? createStubbedMethodBody(quotePreference) : undefined);
         }
         codefix.createMethodFromCallExpression = createMethodFromCallExpression;
         function typeToAutoImportableTypeNode(checker, importAdder, type, contextNode, scriptTarget, flags, tracker) {
             var typeNode = checker.typeToTypeNode(type, contextNode, flags, tracker);
             if (typeNode && ts.isImportTypeNode(typeNode)) {
-                var importableReference = tryGetAutoImportableReferenceFromImportTypeNode(typeNode, type, scriptTarget);
+                var importableReference = tryGetAutoImportableReferenceFromTypeNode(typeNode, scriptTarget);
                 if (importableReference) {
                     importSymbols(importAdder, importableReference.symbols);
-                    return importableReference.typeReference;
+                    return importableReference.typeNode;
                 }
             }
             return typeNode;
@@ -133845,19 +140568,19 @@ var ts;
         function createDummyParameters(argCount, names, types, minArgumentCount, inJs) {
             var parameters = [];
             for (var i = 0; i < argCount; i++) {
-                var newParameter = ts.createParameter(
+                var newParameter = ts.factory.createParameterDeclaration(
                 /*decorators*/ undefined, 
                 /*modifiers*/ undefined, 
                 /*dotDotDotToken*/ undefined, 
                 /*name*/ names && names[i] || "arg" + i, 
-                /*questionToken*/ minArgumentCount !== undefined && i >= minArgumentCount ? ts.createToken(57 /* QuestionToken */) : undefined, 
-                /*type*/ inJs ? undefined : types && types[i] || ts.createKeywordTypeNode(125 /* AnyKeyword */), 
+                /*questionToken*/ minArgumentCount !== undefined && i >= minArgumentCount ? ts.factory.createToken(57 /* QuestionToken */) : undefined, 
+                /*type*/ inJs ? undefined : types && types[i] || ts.factory.createKeywordTypeNode(128 /* AnyKeyword */), 
                 /*initializer*/ undefined);
                 parameters.push(newParameter);
             }
             return parameters;
         }
-        function createMethodImplementingSignatures(signatures, name, optional, modifiers, preferences) {
+        function createMethodImplementingSignatures(signatures, name, optional, modifiers, quotePreference) {
             /** This is *a* signature with the maximal number of arguments,
              * such that if there is a "maximal" signature without rest arguments,
              * this is one of them.
@@ -133879,36 +140602,36 @@ var ts;
             var maxArgsParameterSymbolNames = maxArgsSignature.parameters.map(function (symbol) { return symbol.name; });
             var parameters = createDummyParameters(maxNonRestArgs, maxArgsParameterSymbolNames, /* types */ undefined, minArgumentCount, /*inJs*/ false);
             if (someSigHasRestParameter) {
-                var anyArrayType = ts.createArrayTypeNode(ts.createKeywordTypeNode(125 /* AnyKeyword */));
-                var restParameter = ts.createParameter(
+                var anyArrayType = ts.factory.createArrayTypeNode(ts.factory.createKeywordTypeNode(128 /* AnyKeyword */));
+                var restParameter = ts.factory.createParameterDeclaration(
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, ts.createToken(25 /* DotDotDotToken */), maxArgsParameterSymbolNames[maxNonRestArgs] || "rest", 
-                /*questionToken*/ maxNonRestArgs >= minArgumentCount ? ts.createToken(57 /* QuestionToken */) : undefined, anyArrayType, 
+                /*modifiers*/ undefined, ts.factory.createToken(25 /* DotDotDotToken */), maxArgsParameterSymbolNames[maxNonRestArgs] || "rest", 
+                /*questionToken*/ maxNonRestArgs >= minArgumentCount ? ts.factory.createToken(57 /* QuestionToken */) : undefined, anyArrayType, 
                 /*initializer*/ undefined);
                 parameters.push(restParameter);
             }
             return createStubbedMethod(modifiers, name, optional, 
             /*typeParameters*/ undefined, parameters, 
-            /*returnType*/ undefined, preferences);
+            /*returnType*/ undefined, quotePreference);
         }
-        function createStubbedMethod(modifiers, name, optional, typeParameters, parameters, returnType, preferences) {
-            return ts.createMethod(
+        function createStubbedMethod(modifiers, name, optional, typeParameters, parameters, returnType, quotePreference) {
+            return ts.factory.createMethodDeclaration(
             /*decorators*/ undefined, modifiers, 
-            /*asteriskToken*/ undefined, name, optional ? ts.createToken(57 /* QuestionToken */) : undefined, typeParameters, parameters, returnType, createStubbedMethodBody(preferences));
+            /*asteriskToken*/ undefined, name, optional ? ts.factory.createToken(57 /* QuestionToken */) : undefined, typeParameters, parameters, returnType, createStubbedMethodBody(quotePreference));
         }
-        function createStubbedMethodBody(preferences) {
-            return ts.createBlock([ts.createThrow(ts.createNew(ts.createIdentifier("Error"), 
+        function createStubbedMethodBody(quotePreference) {
+            return ts.factory.createBlock([ts.factory.createThrowStatement(ts.factory.createNewExpression(ts.factory.createIdentifier("Error"), 
                 /*typeArguments*/ undefined, 
                 // TODO Handle auto quote preference.
-                [ts.createLiteral("Method not implemented.", /*isSingleQuote*/ preferences.quotePreference === "single")]))], 
+                [ts.factory.createStringLiteral("Method not implemented.", /*isSingleQuote*/ quotePreference === 0 /* Single */)]))], 
             /*multiline*/ true);
         }
         function createVisibilityModifier(flags) {
             if (flags & 4 /* Public */) {
-                return ts.createToken(119 /* PublicKeyword */);
+                return ts.factory.createToken(122 /* PublicKeyword */);
             }
             else if (flags & 16 /* Protected */) {
-                return ts.createToken(118 /* ProtectedKeyword */);
+                return ts.factory.createToken(121 /* ProtectedKeyword */);
             }
             return undefined;
         }
@@ -133918,7 +140641,7 @@ var ts;
                 return undefined;
             var compilerOptionsProperty = findJsonProperty(tsconfigObjectLiteral, "compilerOptions");
             if (compilerOptionsProperty === undefined) {
-                changeTracker.insertNodeAtObjectStart(configFile, tsconfigObjectLiteral, createJsonPropertyAssignment("compilerOptions", ts.createObjectLiteral(options.map(function (_a) {
+                changeTracker.insertNodeAtObjectStart(configFile, tsconfigObjectLiteral, createJsonPropertyAssignment("compilerOptions", ts.factory.createObjectLiteralExpression(options.map(function (_a) {
                     var optionName = _a[0], optionValue = _a[1];
                     return createJsonPropertyAssignment(optionName, optionValue);
                 }), /*multiLine*/ true)));
@@ -133945,7 +140668,7 @@ var ts;
         }
         codefix.setJsonCompilerOptionValue = setJsonCompilerOptionValue;
         function createJsonPropertyAssignment(name, initializer) {
-            return ts.createPropertyAssignment(ts.createStringLiteral(name), initializer);
+            return ts.factory.createPropertyAssignment(ts.factory.createStringLiteral(name), initializer);
         }
         codefix.createJsonPropertyAssignment = createJsonPropertyAssignment;
         function findJsonProperty(obj, name) {
@@ -133953,44 +140676,38 @@ var ts;
         }
         codefix.findJsonProperty = findJsonProperty;
         /**
-         * Given an ImportTypeNode 'import("./a").SomeType<import("./b").OtherType<...>>',
+         * Given a type node containing 'import("./a").SomeType<import("./b").OtherType<...>>',
          * returns an equivalent type reference node with any nested ImportTypeNodes also replaced
          * with type references, and a list of symbols that must be imported to use the type reference.
          */
-        function tryGetAutoImportableReferenceFromImportTypeNode(importTypeNode, type, scriptTarget) {
-            if (importTypeNode && ts.isLiteralImportTypeNode(importTypeNode) && importTypeNode.qualifier && (!type || type.symbol)) {
-                // Symbol for the left-most thing after the dot
-                var firstIdentifier = ts.getFirstIdentifier(importTypeNode.qualifier);
-                var name = ts.getNameForExportedSymbol(firstIdentifier.symbol, scriptTarget);
-                var qualifier = name !== firstIdentifier.text
-                    ? replaceFirstIdentifierOfEntityName(importTypeNode.qualifier, ts.createIdentifier(name))
-                    : importTypeNode.qualifier;
-                var symbols_4 = [firstIdentifier.symbol];
-                var typeArguments_1 = [];
-                if (importTypeNode.typeArguments) {
-                    importTypeNode.typeArguments.forEach(function (arg) {
-                        var ref = tryGetAutoImportableReferenceFromImportTypeNode(arg, /*undefined*/ type, scriptTarget);
-                        if (ref) {
-                            symbols_4.push.apply(symbols_4, ref.symbols);
-                            typeArguments_1.push(ref.typeReference);
-                        }
-                        else {
-                            typeArguments_1.push(arg);
-                        }
-                    });
+        function tryGetAutoImportableReferenceFromTypeNode(importTypeNode, scriptTarget) {
+            var symbols;
+            var typeNode = ts.visitNode(importTypeNode, visit);
+            if (symbols && typeNode) {
+                return { typeNode: typeNode, symbols: symbols };
+            }
+            function visit(node) {
+                var _a;
+                if (ts.isLiteralImportTypeNode(node) && node.qualifier) {
+                    // Symbol for the left-most thing after the dot
+                    var firstIdentifier = ts.getFirstIdentifier(node.qualifier);
+                    var name = ts.getNameForExportedSymbol(firstIdentifier.symbol, scriptTarget);
+                    var qualifier = name !== firstIdentifier.text
+                        ? replaceFirstIdentifierOfEntityName(node.qualifier, ts.factory.createIdentifier(name))
+                        : node.qualifier;
+                    symbols = ts.append(symbols, firstIdentifier.symbol);
+                    var typeArguments = (_a = node.typeArguments) === null || _a === void 0 ? void 0 : _a.map(visit);
+                    return ts.factory.createTypeReferenceNode(qualifier, typeArguments);
                 }
-                return {
-                    symbols: symbols_4,
-                    typeReference: ts.createTypeReferenceNode(qualifier, typeArguments_1)
-                };
+                return ts.visitEachChild(node, visit, ts.nullTransformationContext);
             }
         }
-        codefix.tryGetAutoImportableReferenceFromImportTypeNode = tryGetAutoImportableReferenceFromImportTypeNode;
+        codefix.tryGetAutoImportableReferenceFromTypeNode = tryGetAutoImportableReferenceFromTypeNode;
         function replaceFirstIdentifierOfEntityName(name, newIdentifier) {
-            if (name.kind === 75 /* Identifier */) {
+            if (name.kind === 78 /* Identifier */) {
                 return newIdentifier;
             }
-            return ts.createQualifiedName(replaceFirstIdentifierOfEntityName(name.left, newIdentifier), name.right);
+            return ts.factory.createQualifiedName(replaceFirstIdentifierOfEntityName(name.left, newIdentifier), name.right);
         }
         function importSymbols(importAdder, symbols) {
             symbols.forEach(function (s) { return importAdder.addImportFromExportedSymbol(s, /*usageIsTypeOnly*/ true); });
@@ -134000,6 +140717,217 @@ var ts;
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        function generateAccessorFromProperty(file, program, start, end, context, _actionName) {
+            var fieldInfo = getAccessorConvertiblePropertyAtPosition(file, program, start, end);
+            if (!fieldInfo || !fieldInfo.info)
+                return undefined;
+            var changeTracker = ts.textChanges.ChangeTracker.fromContext(context);
+            var _a = fieldInfo.info, isStatic = _a.isStatic, isReadonly = _a.isReadonly, fieldName = _a.fieldName, accessorName = _a.accessorName, originalName = _a.originalName, type = _a.type, container = _a.container, declaration = _a.declaration;
+            ts.suppressLeadingAndTrailingTrivia(fieldName);
+            ts.suppressLeadingAndTrailingTrivia(accessorName);
+            ts.suppressLeadingAndTrailingTrivia(declaration);
+            ts.suppressLeadingAndTrailingTrivia(container);
+            var accessorModifiers;
+            var fieldModifiers;
+            if (ts.isClassLike(container)) {
+                var modifierFlags = ts.getEffectiveModifierFlags(declaration);
+                if (ts.isSourceFileJS(file)) {
+                    var modifiers = createModifiers(modifierFlags);
+                    accessorModifiers = modifiers;
+                    fieldModifiers = modifiers;
+                }
+                else {
+                    accessorModifiers = createModifiers(prepareModifierFlagsForAccessor(modifierFlags));
+                    fieldModifiers = createModifiers(prepareModifierFlagsForField(modifierFlags));
+                }
+            }
+            updateFieldDeclaration(changeTracker, file, declaration, type, fieldName, fieldModifiers);
+            var getAccessor = generateGetAccessor(fieldName, accessorName, type, accessorModifiers, isStatic, container);
+            ts.suppressLeadingAndTrailingTrivia(getAccessor);
+            insertAccessor(changeTracker, file, getAccessor, declaration, container);
+            if (isReadonly) {
+                // readonly modifier only existed in classLikeDeclaration
+                var constructor = ts.getFirstConstructorWithBody(container);
+                if (constructor) {
+                    updateReadonlyPropertyInitializerStatementConstructor(changeTracker, file, constructor, fieldName.text, originalName);
+                }
+            }
+            else {
+                var setAccessor = generateSetAccessor(fieldName, accessorName, type, accessorModifiers, isStatic, container);
+                ts.suppressLeadingAndTrailingTrivia(setAccessor);
+                insertAccessor(changeTracker, file, setAccessor, declaration, container);
+            }
+            return changeTracker.getChanges();
+        }
+        codefix.generateAccessorFromProperty = generateAccessorFromProperty;
+        function isConvertibleName(name) {
+            return ts.isIdentifier(name) || ts.isStringLiteral(name);
+        }
+        function isAcceptedDeclaration(node) {
+            return ts.isParameterPropertyDeclaration(node, node.parent) || ts.isPropertyDeclaration(node) || ts.isPropertyAssignment(node);
+        }
+        function createPropertyName(name, originalName) {
+            return ts.isIdentifier(originalName) ? ts.factory.createIdentifier(name) : ts.factory.createStringLiteral(name);
+        }
+        function createAccessorAccessExpression(fieldName, isStatic, container) {
+            var leftHead = isStatic ? container.name : ts.factory.createThis(); // TODO: GH#18217
+            return ts.isIdentifier(fieldName) ? ts.factory.createPropertyAccessExpression(leftHead, fieldName) : ts.factory.createElementAccessExpression(leftHead, ts.factory.createStringLiteralFromNode(fieldName));
+        }
+        function createModifiers(modifierFlags) {
+            return modifierFlags ? ts.factory.createNodeArray(ts.factory.createModifiersFromModifierFlags(modifierFlags)) : undefined;
+        }
+        function prepareModifierFlagsForAccessor(modifierFlags) {
+            modifierFlags &= ~64 /* Readonly */; // avoid Readonly modifier because it will convert to get accessor
+            modifierFlags &= ~8 /* Private */;
+            if (!(modifierFlags & 16 /* Protected */)) {
+                modifierFlags |= 4 /* Public */;
+            }
+            return modifierFlags;
+        }
+        function prepareModifierFlagsForField(modifierFlags) {
+            modifierFlags &= ~4 /* Public */;
+            modifierFlags &= ~16 /* Protected */;
+            modifierFlags |= 8 /* Private */;
+            return modifierFlags;
+        }
+        function getAccessorConvertiblePropertyAtPosition(file, program, start, end, considerEmptySpans) {
+            if (considerEmptySpans === void 0) { considerEmptySpans = true; }
+            var node = ts.getTokenAtPosition(file, start);
+            var cursorRequest = start === end && considerEmptySpans;
+            var declaration = ts.findAncestor(node.parent, isAcceptedDeclaration);
+            // make sure declaration have AccessibilityModifier or Static Modifier or Readonly Modifier
+            var meaning = 28 /* AccessibilityModifier */ | 32 /* Static */ | 64 /* Readonly */;
+            if (!declaration || (!(ts.nodeOverlapsWithStartEnd(declaration.name, file, start, end) || cursorRequest))) {
+                return {
+                    error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_property_for_which_to_generate_accessor)
+                };
+            }
+            if (!isConvertibleName(declaration.name)) {
+                return {
+                    error: ts.getLocaleSpecificMessage(ts.Diagnostics.Name_is_not_valid)
+                };
+            }
+            if ((ts.getEffectiveModifierFlags(declaration) | meaning) !== meaning) {
+                return {
+                    error: ts.getLocaleSpecificMessage(ts.Diagnostics.Can_only_convert_property_with_modifier)
+                };
+            }
+            var name = declaration.name.text;
+            var startWithUnderscore = ts.startsWithUnderscore(name);
+            var fieldName = createPropertyName(startWithUnderscore ? name : ts.getUniqueName("_" + name, file), declaration.name);
+            var accessorName = createPropertyName(startWithUnderscore ? ts.getUniqueName(name.substring(1), file) : name, declaration.name);
+            return {
+                info: {
+                    isStatic: ts.hasStaticModifier(declaration),
+                    isReadonly: ts.hasEffectiveReadonlyModifier(declaration),
+                    type: getDeclarationType(declaration, program),
+                    container: declaration.kind === 160 /* Parameter */ ? declaration.parent.parent : declaration.parent,
+                    originalName: declaration.name.text,
+                    declaration: declaration,
+                    fieldName: fieldName,
+                    accessorName: accessorName,
+                    renameAccessor: startWithUnderscore
+                }
+            };
+        }
+        codefix.getAccessorConvertiblePropertyAtPosition = getAccessorConvertiblePropertyAtPosition;
+        function generateGetAccessor(fieldName, accessorName, type, modifiers, isStatic, container) {
+            return ts.factory.createGetAccessorDeclaration(
+            /*decorators*/ undefined, modifiers, accessorName, 
+            /*parameters*/ undefined, // TODO: GH#18217
+            type, ts.factory.createBlock([
+                ts.factory.createReturnStatement(createAccessorAccessExpression(fieldName, isStatic, container))
+            ], /*multiLine*/ true));
+        }
+        function generateSetAccessor(fieldName, accessorName, type, modifiers, isStatic, container) {
+            return ts.factory.createSetAccessorDeclaration(
+            /*decorators*/ undefined, modifiers, accessorName, [ts.factory.createParameterDeclaration(
+                /*decorators*/ undefined, 
+                /*modifiers*/ undefined, 
+                /*dotDotDotToken*/ undefined, ts.factory.createIdentifier("value"), 
+                /*questionToken*/ undefined, type)], ts.factory.createBlock([
+                ts.factory.createExpressionStatement(ts.factory.createAssignment(createAccessorAccessExpression(fieldName, isStatic, container), ts.factory.createIdentifier("value")))
+            ], /*multiLine*/ true));
+        }
+        function updatePropertyDeclaration(changeTracker, file, declaration, type, fieldName, modifiers) {
+            var property = ts.factory.updatePropertyDeclaration(declaration, declaration.decorators, modifiers, fieldName, declaration.questionToken || declaration.exclamationToken, type, declaration.initializer);
+            changeTracker.replaceNode(file, declaration, property);
+        }
+        function updatePropertyAssignmentDeclaration(changeTracker, file, declaration, fieldName) {
+            var assignment = ts.factory.updatePropertyAssignment(declaration, fieldName, declaration.initializer);
+            changeTracker.replacePropertyAssignment(file, declaration, assignment);
+        }
+        function updateFieldDeclaration(changeTracker, file, declaration, type, fieldName, modifiers) {
+            if (ts.isPropertyDeclaration(declaration)) {
+                updatePropertyDeclaration(changeTracker, file, declaration, type, fieldName, modifiers);
+            }
+            else if (ts.isPropertyAssignment(declaration)) {
+                updatePropertyAssignmentDeclaration(changeTracker, file, declaration, fieldName);
+            }
+            else {
+                changeTracker.replaceNode(file, declaration, ts.factory.updateParameterDeclaration(declaration, declaration.decorators, modifiers, declaration.dotDotDotToken, ts.cast(fieldName, ts.isIdentifier), declaration.questionToken, declaration.type, declaration.initializer));
+            }
+        }
+        function insertAccessor(changeTracker, file, accessor, declaration, container) {
+            ts.isParameterPropertyDeclaration(declaration, declaration.parent) ? changeTracker.insertNodeAtClassStart(file, container, accessor) :
+                ts.isPropertyAssignment(declaration) ? changeTracker.insertNodeAfterComma(file, declaration, accessor) :
+                    changeTracker.insertNodeAfter(file, declaration, accessor);
+        }
+        function updateReadonlyPropertyInitializerStatementConstructor(changeTracker, file, constructor, fieldName, originalName) {
+            if (!constructor.body)
+                return;
+            constructor.body.forEachChild(function recur(node) {
+                if (ts.isElementAccessExpression(node) &&
+                    node.expression.kind === 107 /* ThisKeyword */ &&
+                    ts.isStringLiteral(node.argumentExpression) &&
+                    node.argumentExpression.text === originalName &&
+                    ts.isWriteAccess(node)) {
+                    changeTracker.replaceNode(file, node.argumentExpression, ts.factory.createStringLiteral(fieldName));
+                }
+                if (ts.isPropertyAccessExpression(node) && node.expression.kind === 107 /* ThisKeyword */ && node.name.text === originalName && ts.isWriteAccess(node)) {
+                    changeTracker.replaceNode(file, node.name, ts.factory.createIdentifier(fieldName));
+                }
+                if (!ts.isFunctionLike(node) && !ts.isClassLike(node)) {
+                    node.forEachChild(recur);
+                }
+            });
+        }
+        function getDeclarationType(declaration, program) {
+            var typeNode = ts.getTypeAnnotationNode(declaration);
+            if (ts.isPropertyDeclaration(declaration) && typeNode && declaration.questionToken) {
+                var typeChecker = program.getTypeChecker();
+                var type = typeChecker.getTypeFromTypeNode(typeNode);
+                if (!typeChecker.isTypeAssignableTo(typeChecker.getUndefinedType(), type)) {
+                    var types = ts.isUnionTypeNode(typeNode) ? typeNode.types : [typeNode];
+                    return ts.factory.createUnionTypeNode(__spreadArrays(types, [ts.factory.createKeywordTypeNode(150 /* UndefinedKeyword */)]));
+                }
+            }
+            return typeNode;
+        }
+        function getAllSupers(decl, checker) {
+            var res = [];
+            while (decl) {
+                var superElement = ts.getClassExtendsHeritageElement(decl);
+                var superSymbol = superElement && checker.getSymbolAtLocation(superElement.expression);
+                if (!superSymbol)
+                    break;
+                var symbol = superSymbol.flags & 2097152 /* Alias */ ? checker.getAliasedSymbol(superSymbol) : superSymbol;
+                var superDecl = ts.find(symbol.declarations, ts.isClassLike);
+                if (!superDecl)
+                    break;
+                res.push(superDecl);
+                decl = superDecl;
+            }
+            return res;
+        }
+        codefix.getAllSupers = getAllSupers;
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var codefix;
     (function (codefix) {
@@ -134013,9 +140941,9 @@ var ts;
             variations.push(createAction(context, sourceFile, node, ts.makeImport(namespace.name, /*namedImports*/ undefined, node.moduleSpecifier, ts.getQuotePreference(sourceFile, context.preferences))));
             if (ts.getEmitModuleKind(opts) === ts.ModuleKind.CommonJS) {
                 // import Bluebird = require("bluebird");
-                variations.push(createAction(context, sourceFile, node, ts.createImportEqualsDeclaration(
+                variations.push(createAction(context, sourceFile, node, ts.factory.createImportEqualsDeclaration(
                 /*decorators*/ undefined, 
-                /*modifiers*/ undefined, namespace.name, ts.createExternalModuleReference(node.moduleSpecifier))));
+                /*modifiers*/ undefined, namespace.name, ts.factory.createExternalModuleReference(node.moduleSpecifier))));
             }
             return variations;
         }
@@ -134032,7 +140960,7 @@ var ts;
         });
         function getActionsForUsageOfInvalidImport(context) {
             var sourceFile = context.sourceFile;
-            var targetKind = ts.Diagnostics.This_expression_is_not_callable.code === context.errorCode ? 196 /* CallExpression */ : 197 /* NewExpression */;
+            var targetKind = ts.Diagnostics.This_expression_is_not_callable.code === context.errorCode ? 203 /* CallExpression */ : 204 /* NewExpression */;
             var node = ts.findAncestor(ts.getTokenAtPosition(sourceFile, context.span.start), function (a) { return a.kind === targetKind; });
             if (!node) {
                 return [];
@@ -134078,7 +141006,7 @@ var ts;
             }
             if (ts.isExpression(expr) && !(ts.isNamedDeclaration(expr.parent) && expr.parent.name === expr)) {
                 var sourceFile_1 = context.sourceFile;
-                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return t.replaceNode(sourceFile_1, expr, ts.createPropertyAccess(expr, "default"), {}); });
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return t.replaceNode(sourceFile_1, expr, ts.factory.createPropertyAccessExpression(expr, "default"), {}); });
                 fixes.push(codefix.createCodeFixActionWithoutFixAll(fixName, changes, ts.Diagnostics.Use_synthetic_default_member));
             }
             return fixes;
@@ -134143,7 +141071,7 @@ var ts;
             return codefix.createCodeFixAction(fixName, changes, [ts.Diagnostics.Add_definite_assignment_assertion_to_property_0, propertyDeclaration.getText()], fixIdAddDefiniteAssignmentAssertions, ts.Diagnostics.Add_definite_assignment_assertions_to_all_uninitialized_properties);
         }
         function addDefiniteAssignmentAssertion(changeTracker, propertyDeclarationSourceFile, propertyDeclaration) {
-            var property = ts.updateProperty(propertyDeclaration, propertyDeclaration.decorators, propertyDeclaration.modifiers, propertyDeclaration.name, ts.createToken(53 /* ExclamationToken */), propertyDeclaration.type, propertyDeclaration.initializer);
+            var property = ts.factory.updatePropertyDeclaration(propertyDeclaration, propertyDeclaration.decorators, propertyDeclaration.modifiers, propertyDeclaration.name, ts.factory.createToken(53 /* ExclamationToken */), propertyDeclaration.type, propertyDeclaration.initializer);
             changeTracker.replaceNode(propertyDeclarationSourceFile, propertyDeclaration, property);
         }
         function getActionForAddMissingUndefinedType(context, propertyDeclaration) {
@@ -134151,10 +141079,10 @@ var ts;
             return codefix.createCodeFixAction(fixName, changes, [ts.Diagnostics.Add_undefined_type_to_property_0, propertyDeclaration.name.getText()], fixIdAddUndefinedType, ts.Diagnostics.Add_undefined_type_to_all_uninitialized_properties);
         }
         function addUndefinedType(changeTracker, propertyDeclarationSourceFile, propertyDeclaration) {
-            var undefinedTypeNode = ts.createKeywordTypeNode(146 /* UndefinedKeyword */);
+            var undefinedTypeNode = ts.factory.createKeywordTypeNode(150 /* UndefinedKeyword */);
             var type = propertyDeclaration.type; // TODO: GH#18217
             var types = ts.isUnionTypeNode(type) ? type.types.concat(undefinedTypeNode) : [type, undefinedTypeNode];
-            changeTracker.replaceNode(propertyDeclarationSourceFile, type, ts.createUnionTypeNode(types));
+            changeTracker.replaceNode(propertyDeclarationSourceFile, type, ts.factory.createUnionTypeNode(types));
         }
         function getActionForAddMissingInitializer(context, propertyDeclaration) {
             var checker = context.program.getTypeChecker();
@@ -134165,7 +141093,7 @@ var ts;
             return codefix.createCodeFixAction(fixName, changes, [ts.Diagnostics.Add_initializer_to_property_0, propertyDeclaration.name.getText()], fixIdAddInitializer, ts.Diagnostics.Add_initializers_to_all_uninitialized_properties);
         }
         function addInitializer(changeTracker, propertyDeclarationSourceFile, propertyDeclaration, initializer) {
-            var property = ts.updateProperty(propertyDeclaration, propertyDeclaration.decorators, propertyDeclaration.modifiers, propertyDeclaration.name, propertyDeclaration.questionToken, propertyDeclaration.type, initializer);
+            var property = ts.factory.updatePropertyDeclaration(propertyDeclaration, propertyDeclaration.decorators, propertyDeclaration.modifiers, propertyDeclaration.name, propertyDeclaration.questionToken, propertyDeclaration.type, initializer);
             changeTracker.replaceNode(propertyDeclarationSourceFile, propertyDeclaration, property);
         }
         function getInitializer(checker, propertyDeclaration) {
@@ -134173,25 +141101,31 @@ var ts;
         }
         function getDefaultValueFromType(checker, type) {
             if (type.flags & 512 /* BooleanLiteral */) {
-                return (type === checker.getFalseType() || type === checker.getFalseType(/*fresh*/ true)) ? ts.createFalse() : ts.createTrue();
+                return (type === checker.getFalseType() || type === checker.getFalseType(/*fresh*/ true)) ? ts.factory.createFalse() : ts.factory.createTrue();
+            }
+            else if (type.isStringLiteral()) {
+                return ts.factory.createStringLiteral(type.value);
             }
-            else if (type.isLiteral()) {
-                return ts.createLiteral(type.value);
+            else if (type.isNumberLiteral()) {
+                return ts.factory.createNumericLiteral(type.value);
+            }
+            else if (type.flags & 2048 /* BigIntLiteral */) {
+                return ts.factory.createBigIntLiteral(type.value);
             }
             else if (type.isUnion()) {
                 return ts.firstDefined(type.types, function (t) { return getDefaultValueFromType(checker, t); });
             }
             else if (type.isClass()) {
                 var classDeclaration = ts.getClassLikeDeclarationOfSymbol(type.symbol);
-                if (!classDeclaration || ts.hasModifier(classDeclaration, 128 /* Abstract */))
+                if (!classDeclaration || ts.hasSyntacticModifier(classDeclaration, 128 /* Abstract */))
                     return undefined;
                 var constructorDeclaration = ts.getFirstConstructorWithBody(classDeclaration);
                 if (constructorDeclaration && constructorDeclaration.parameters.length)
                     return undefined;
-                return ts.createNew(ts.createIdentifier(type.symbol.name), /*typeArguments*/ undefined, /*argumentsArray*/ undefined);
+                return ts.factory.createNewExpression(ts.factory.createIdentifier(type.symbol.name), /*typeArguments*/ undefined, /*argumentsArray*/ undefined);
             }
             else if (checker.isArrayLikeType(type)) {
-                return ts.createArrayLiteral();
+                return ts.factory.createArrayLiteralExpression();
             }
             return undefined;
         }
@@ -134207,24 +141141,57 @@ var ts;
         codefix.registerCodeFix({
             errorCodes: errorCodes,
             getCodeActions: function (context) {
-                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, context.sourceFile, context.span.start, context.program); });
+                var info = getInfo(context.sourceFile, context.program, context.span.start);
+                if (!info) {
+                    return undefined;
+                }
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, context.sourceFile, info); });
                 return [codefix.createCodeFixAction(fixId, changes, ts.Diagnostics.Convert_require_to_import, fixId, ts.Diagnostics.Convert_all_require_to_import)];
             },
             fixIds: [fixId],
-            getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) { return doChange(changes, diag.file, diag.start, context.program); }); },
+            getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
+                var info = getInfo(diag.file, context.program, diag.start);
+                if (info) {
+                    doChange(changes, context.sourceFile, info);
+                }
+            }); },
         });
-        function doChange(changes, sourceFile, pos, program) {
-            var _a = getInfo(sourceFile, pos), statement = _a.statement, name = _a.name, required = _a.required;
-            changes.replaceNode(sourceFile, statement, ts.getAllowSyntheticDefaultImports(program.getCompilerOptions())
-                ? ts.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.createImportClause(name, /*namedBindings*/ undefined), required)
-                : ts.createImportEqualsDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, name, ts.createExternalModuleReference(required)));
+        function doChange(changes, sourceFile, info) {
+            var allowSyntheticDefaults = info.allowSyntheticDefaults, defaultImportName = info.defaultImportName, namedImports = info.namedImports, statement = info.statement, required = info.required;
+            changes.replaceNode(sourceFile, statement, defaultImportName && !allowSyntheticDefaults
+                ? ts.factory.createImportEqualsDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, defaultImportName, ts.factory.createExternalModuleReference(required))
+                : ts.factory.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, defaultImportName, namedImports), required));
         }
-        function getInfo(sourceFile, pos) {
+        function getInfo(sourceFile, program, pos) {
             var parent = ts.getTokenAtPosition(sourceFile, pos).parent;
-            if (!ts.isRequireCall(parent, /*checkArgumentIsStringLiteralLike*/ true))
+            if (!ts.isRequireCall(parent, /*checkArgumentIsStringLiteralLike*/ true)) {
                 throw ts.Debug.failBadSyntaxKind(parent);
+            }
             var decl = ts.cast(parent.parent, ts.isVariableDeclaration);
-            return { statement: ts.cast(decl.parent.parent, ts.isVariableStatement), name: ts.cast(decl.name, ts.isIdentifier), required: parent.arguments[0] };
+            var defaultImportName = ts.tryCast(decl.name, ts.isIdentifier);
+            var namedImports = ts.isObjectBindingPattern(decl.name) ? tryCreateNamedImportsFromObjectBindingPattern(decl.name) : undefined;
+            if (defaultImportName || namedImports) {
+                return {
+                    allowSyntheticDefaults: ts.getAllowSyntheticDefaultImports(program.getCompilerOptions()),
+                    defaultImportName: defaultImportName,
+                    namedImports: namedImports,
+                    statement: ts.cast(decl.parent.parent, ts.isVariableStatement),
+                    required: ts.first(parent.arguments)
+                };
+            }
+        }
+        function tryCreateNamedImportsFromObjectBindingPattern(node) {
+            var importSpecifiers = [];
+            for (var _i = 0, _a = node.elements; _i < _a.length; _i++) {
+                var element = _a[_i];
+                if (!ts.isIdentifier(element.name) || element.initializer) {
+                    return undefined;
+                }
+                importSpecifiers.push(ts.factory.createImportSpecifier(ts.tryCast(element.propertyName, ts.isIdentifier), element.name));
+            }
+            if (importSpecifiers.length) {
+                return ts.factory.createNamedImports(importSpecifiers);
+            }
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -134299,7 +141266,7 @@ var ts;
             }
             // We use .getText to overcome parser inaccuracies: https://github.com/microsoft/TypeScript/issues/33298
             var newText = numericLiteral.getText(sourceFile) + "n";
-            changeTracker.replaceNode(sourceFile, numericLiteral, ts.createBigIntLiteral(newText));
+            changeTracker.replaceNode(sourceFile, numericLiteral, ts.factory.createBigIntLiteral(newText));
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -134326,18 +141293,98 @@ var ts;
         });
         function getImportTypeNode(sourceFile, pos) {
             var token = ts.getTokenAtPosition(sourceFile, pos);
-            ts.Debug.assert(token.kind === 96 /* ImportKeyword */, "This token should be an ImportKeyword");
-            ts.Debug.assert(token.parent.kind === 188 /* ImportType */, "Token parent should be an ImportType");
+            ts.Debug.assert(token.kind === 99 /* ImportKeyword */, "This token should be an ImportKeyword");
+            ts.Debug.assert(token.parent.kind === 195 /* ImportType */, "Token parent should be an ImportType");
             return token.parent;
         }
         function doChange(changes, sourceFile, importType) {
-            var newTypeNode = ts.updateImportTypeNode(importType, importType.argument, importType.qualifier, importType.typeArguments, /* isTypeOf */ true);
+            var newTypeNode = ts.factory.updateImportTypeNode(importType, importType.argument, importType.qualifier, importType.typeArguments, /* isTypeOf */ true);
             changes.replaceNode(sourceFile, importType, newTypeNode);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixID = "wrapJsxInFragment";
+        var errorCodes = [ts.Diagnostics.JSX_expressions_must_have_one_parent_element.code];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            getCodeActions: function (context) {
+                var jsx = context.program.getCompilerOptions().jsx;
+                if (jsx !== 2 /* React */ && jsx !== 3 /* ReactNative */) {
+                    return undefined;
+                }
+                var sourceFile = context.sourceFile, span = context.span;
+                var node = findNodeToFix(sourceFile, span.start);
+                if (!node)
+                    return undefined;
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(t, sourceFile, node); });
+                return [codefix.createCodeFixAction(fixID, changes, ts.Diagnostics.Wrap_in_JSX_fragment, fixID, ts.Diagnostics.Wrap_all_unparented_JSX_in_JSX_fragment)];
+            },
+            fixIds: [fixID],
+            getAllCodeActions: function (context) { return codefix.codeFixAll(context, errorCodes, function (changes, diag) {
+                var node = findNodeToFix(context.sourceFile, diag.start);
+                if (!node)
+                    return undefined;
+                doChange(changes, context.sourceFile, node);
+            }); },
+        });
+        function findNodeToFix(sourceFile, pos) {
+            // The error always at 1st token that is "<" in "<a /><a />"
+            var lessThanToken = ts.getTokenAtPosition(sourceFile, pos);
+            var firstJsxElementOrOpenElement = lessThanToken.parent;
+            var binaryExpr = firstJsxElementOrOpenElement.parent;
+            if (!ts.isBinaryExpression(binaryExpr)) {
+                // In case the start element is a JsxSelfClosingElement, it the end.
+                // For JsxOpenElement, find one more parent
+                binaryExpr = binaryExpr.parent;
+                if (!ts.isBinaryExpression(binaryExpr))
+                    return undefined;
+            }
+            if (!ts.nodeIsMissing(binaryExpr.operatorToken))
+                return undefined;
+            return binaryExpr;
+        }
+        function doChange(changeTracker, sf, node) {
+            var jsx = flattenInvalidBinaryExpr(node);
+            if (jsx)
+                changeTracker.replaceNode(sf, node, ts.factory.createJsxFragment(ts.factory.createJsxOpeningFragment(), jsx, ts.factory.createJsxJsxClosingFragment()));
+        }
+        // The invalid syntax is constructed as
+        // InvalidJsxTree :: One of
+        //     JsxElement CommaToken InvalidJsxTree
+        //     JsxElement CommaToken JsxElement
+        function flattenInvalidBinaryExpr(node) {
+            var children = [];
+            var current = node;
+            while (true) {
+                if (ts.isBinaryExpression(current) && ts.nodeIsMissing(current.operatorToken) && current.operatorToken.kind === 27 /* CommaToken */) {
+                    children.push(current.left);
+                    if (ts.isJsxChild(current.right)) {
+                        children.push(current.right);
+                        // Indicates the tree has go to the bottom
+                        return children;
+                    }
+                    else if (ts.isBinaryExpression(current.right)) {
+                        current = current.right;
+                        continue;
+                    }
+                    // Unreachable case
+                    else
+                        return undefined;
+                }
+                // Unreachable case
+                else
+                    return undefined;
+            }
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var codefix;
     (function (codefix) {
@@ -134371,24 +141418,50 @@ var ts;
             return { indexSignature: indexSignature, container: container };
         }
         function createTypeAliasFromInterface(declaration, type) {
-            return ts.createTypeAliasDeclaration(declaration.decorators, declaration.modifiers, declaration.name, declaration.typeParameters, type);
+            return ts.factory.createTypeAliasDeclaration(declaration.decorators, declaration.modifiers, declaration.name, declaration.typeParameters, type);
         }
         function doChange(changes, sourceFile, _a) {
             var indexSignature = _a.indexSignature, container = _a.container;
             var members = ts.isInterfaceDeclaration(container) ? container.members : container.type.members;
             var otherMembers = members.filter(function (member) { return !ts.isIndexSignatureDeclaration(member); });
             var parameter = ts.first(indexSignature.parameters);
-            var mappedTypeParameter = ts.createTypeParameterDeclaration(ts.cast(parameter.name, ts.isIdentifier), parameter.type);
-            var mappedIntersectionType = ts.createMappedTypeNode(ts.hasReadonlyModifier(indexSignature) ? ts.createModifier(138 /* ReadonlyKeyword */) : undefined, mappedTypeParameter, indexSignature.questionToken, indexSignature.type);
-            var intersectionType = ts.createIntersectionTypeNode(__spreadArrays(ts.getAllSuperTypeNodes(container), [
+            var mappedTypeParameter = ts.factory.createTypeParameterDeclaration(ts.cast(parameter.name, ts.isIdentifier), parameter.type);
+            var mappedIntersectionType = ts.factory.createMappedTypeNode(ts.hasEffectiveReadonlyModifier(indexSignature) ? ts.factory.createModifier(142 /* ReadonlyKeyword */) : undefined, mappedTypeParameter, 
+            /*nameType*/ undefined, indexSignature.questionToken, indexSignature.type);
+            var intersectionType = ts.factory.createIntersectionTypeNode(__spreadArrays(ts.getAllSuperTypeNodes(container), [
                 mappedIntersectionType
-            ], (otherMembers.length ? [ts.createTypeLiteralNode(otherMembers)] : ts.emptyArray)));
+            ], (otherMembers.length ? [ts.factory.createTypeLiteralNode(otherMembers)] : ts.emptyArray)));
             changes.replaceNode(sourceFile, container, createTypeAliasFromInterface(container, intersectionType));
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixId = "removeAccidentalCallParentheses";
+        var errorCodes = [
+            ts.Diagnostics.This_expression_is_not_callable_because_it_is_a_get_accessor_Did_you_mean_to_use_it_without.code,
+        ];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            getCodeActions: function (context) {
+                var callExpression = ts.findAncestor(ts.getTokenAtPosition(context.sourceFile, context.span.start), ts.isCallExpression);
+                if (!callExpression) {
+                    return undefined;
+                }
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) {
+                    t.deleteRange(context.sourceFile, { pos: callExpression.expression.end, end: callExpression.end });
+                });
+                return [codefix.createCodeFixActionWithoutFixAll(fixId, changes, ts.Diagnostics.Remove_parentheses)];
+            },
+            fixIds: [fixId],
+        });
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var codefix;
     (function (codefix) {
@@ -134410,7 +141483,7 @@ var ts;
             },
         });
         function makeChange(changeTracker, sourceFile, span) {
-            var awaitKeyword = ts.tryCast(ts.getTokenAtPosition(sourceFile, span.start), function (node) { return node.kind === 127 /* AwaitKeyword */; });
+            var awaitKeyword = ts.tryCast(ts.getTokenAtPosition(sourceFile, span.start), function (node) { return node.kind === 130 /* AwaitKeyword */; });
             var awaitExpression = awaitKeyword && ts.tryCast(awaitKeyword.parent, ts.isAwaitExpression);
             if (!awaitExpression) {
                 return;
@@ -134421,7 +141494,7 @@ var ts;
                 var leftMostExpression = ts.getLeftmostExpression(awaitExpression.expression, /*stopAtCallExpressions*/ false);
                 if (ts.isIdentifier(leftMostExpression)) {
                     var precedingToken = ts.findPrecedingToken(awaitExpression.parent.pos, sourceFile);
-                    if (precedingToken && precedingToken.kind !== 99 /* NewKeyword */) {
+                    if (precedingToken && precedingToken.kind !== 102 /* NewKeyword */) {
                         expressionToReplace = awaitExpression.parent;
                     }
                 }
@@ -134460,10 +141533,10 @@ var ts;
                 return;
             }
             var importClause = ts.Debug.checkDefined(importDeclaration.importClause);
-            changes.replaceNode(context.sourceFile, importDeclaration, ts.updateImportDeclaration(importDeclaration, importDeclaration.decorators, importDeclaration.modifiers, ts.updateImportClause(importClause, importClause.name, /*namedBindings*/ undefined, importClause.isTypeOnly), importDeclaration.moduleSpecifier));
-            changes.insertNodeAfter(context.sourceFile, importDeclaration, ts.createImportDeclaration(
+            changes.replaceNode(context.sourceFile, importDeclaration, ts.factory.updateImportDeclaration(importDeclaration, importDeclaration.decorators, importDeclaration.modifiers, ts.factory.updateImportClause(importClause, importClause.isTypeOnly, importClause.name, /*namedBindings*/ undefined), importDeclaration.moduleSpecifier));
+            changes.insertNodeAfter(context.sourceFile, importDeclaration, ts.factory.createImportDeclaration(
             /*decorators*/ undefined, 
-            /*modifiers*/ undefined, ts.updateImportClause(importClause, /*name*/ undefined, importClause.namedBindings, importClause.isTypeOnly), importDeclaration.moduleSpecifier));
+            /*modifiers*/ undefined, ts.factory.updateImportClause(importClause, importClause.isTypeOnly, /*name*/ undefined, importClause.namedBindings), importDeclaration.moduleSpecifier));
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
@@ -134536,13 +141609,97 @@ var ts;
         }
         function doChange(changes, sourceFile, _a) {
             var node = _a.node;
-            var newNode = ts.createNode(27 /* CommaToken */);
+            var newNode = ts.factory.createToken(27 /* CommaToken */);
             changes.replaceNode(sourceFile, node, newNode);
         }
     })(codefix = ts.codefix || (ts.codefix = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var codefix;
+    (function (codefix) {
+        var fixName = "addVoidToPromise";
+        var fixId = "addVoidToPromise";
+        var errorCodes = [
+            ts.Diagnostics.Expected_0_arguments_but_got_1_Did_you_forget_to_include_void_in_your_type_argument_to_Promise.code
+        ];
+        codefix.registerCodeFix({
+            errorCodes: errorCodes,
+            fixIds: [fixId],
+            getCodeActions: function (context) {
+                var changes = ts.textChanges.ChangeTracker.with(context, function (t) { return makeChange(t, context.sourceFile, context.span, context.program); });
+                if (changes.length > 0) {
+                    return [codefix.createCodeFixAction(fixName, changes, ts.Diagnostics.Add_void_to_Promise_resolved_without_a_value, fixId, ts.Diagnostics.Add_void_to_all_Promises_resolved_without_a_value)];
+                }
+            },
+            getAllCodeActions: function (context) {
+                return codefix.codeFixAll(context, errorCodes, function (changes, diag) { return makeChange(changes, diag.file, diag, context.program, new ts.Set()); });
+            }
+        });
+        function makeChange(changes, sourceFile, span, program, seen) {
+            var node = ts.getTokenAtPosition(sourceFile, span.start);
+            if (!ts.isIdentifier(node) || !ts.isCallExpression(node.parent) || node.parent.expression !== node || node.parent.arguments.length !== 0)
+                return;
+            var checker = program.getTypeChecker();
+            var symbol = checker.getSymbolAtLocation(node);
+            // decl should be `new Promise((<decl>) => {})`
+            var decl = symbol === null || symbol === void 0 ? void 0 : symbol.valueDeclaration;
+            if (!decl || !ts.isParameter(decl) || !ts.isNewExpression(decl.parent.parent))
+                return;
+            // no need to make this change if we have already seen this parameter.
+            if (seen === null || seen === void 0 ? void 0 : seen.has(decl))
+                return;
+            seen === null || seen === void 0 ? void 0 : seen.add(decl);
+            var typeArguments = getEffectiveTypeArguments(decl.parent.parent);
+            if (ts.some(typeArguments)) {
+                // append ` | void` to type argument
+                var typeArgument = typeArguments[0];
+                var needsParens = !ts.isUnionTypeNode(typeArgument) && !ts.isParenthesizedTypeNode(typeArgument) &&
+                    ts.isParenthesizedTypeNode(ts.factory.createUnionTypeNode([typeArgument, ts.factory.createKeywordTypeNode(113 /* VoidKeyword */)]).types[0]);
+                if (needsParens) {
+                    changes.insertText(sourceFile, typeArgument.pos, "(");
+                }
+                changes.insertText(sourceFile, typeArgument.end, needsParens ? ") | void" : " | void");
+            }
+            else {
+                // make sure the Promise is type is untyped (i.e., `unknown`)
+                var signature = checker.getResolvedSignature(node.parent);
+                var parameter = signature === null || signature === void 0 ? void 0 : signature.parameters[0];
+                var parameterType = parameter && checker.getTypeOfSymbolAtLocation(parameter, decl.parent.parent);
+                if (ts.isInJSFile(decl)) {
+                    if (!parameterType || parameterType.flags & 3 /* AnyOrUnknown */) {
+                        // give the expression a type
+                        changes.insertText(sourceFile, decl.parent.parent.end, ")");
+                        changes.insertText(sourceFile, ts.skipTrivia(sourceFile.text, decl.parent.parent.pos), "/** @type {Promise<void>} */(");
+                    }
+                }
+                else {
+                    if (!parameterType || parameterType.flags & 2 /* Unknown */) {
+                        // add `void` type argument
+                        changes.insertText(sourceFile, decl.parent.parent.expression.end, "<void>");
+                    }
+                }
+            }
+        }
+        function getEffectiveTypeArguments(node) {
+            var _a;
+            if (ts.isInJSFile(node)) {
+                if (ts.isParenthesizedExpression(node.parent)) {
+                    var jsDocType = (_a = ts.getJSDocTypeTag(node.parent)) === null || _a === void 0 ? void 0 : _a.typeExpression.type;
+                    if (jsDocType && ts.isTypeReferenceNode(jsDocType) && ts.isIdentifier(jsDocType.typeName) && ts.idText(jsDocType.typeName) === "Promise") {
+                        return jsDocType.typeArguments;
+                    }
+                }
+            }
+            else {
+                return node.typeArguments;
+            }
+        }
+    })(codefix = ts.codefix || (ts.codefix = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var refactor;
     (function (refactor) {
@@ -134551,45 +141708,55 @@ var ts;
         var actionNameNamedToDefault = "Convert named export to default export";
         refactor.registerRefactor(refactorName, {
             getAvailableActions: function (context) {
-                var info = getInfo(context);
+                var info = getInfo(context, context.triggerReason === "invoked");
                 if (!info)
                     return ts.emptyArray;
-                var description = info.wasDefault ? ts.Diagnostics.Convert_default_export_to_named_export.message : ts.Diagnostics.Convert_named_export_to_default_export.message;
-                var actionName = info.wasDefault ? actionNameDefaultToNamed : actionNameNamedToDefault;
-                return [{ name: refactorName, description: description, actions: [{ name: actionName, description: description }] }];
+                if (info.error === undefined) {
+                    var description = info.info.wasDefault ? ts.Diagnostics.Convert_default_export_to_named_export.message : ts.Diagnostics.Convert_named_export_to_default_export.message;
+                    var actionName = info.info.wasDefault ? actionNameDefaultToNamed : actionNameNamedToDefault;
+                    return [{ name: refactorName, description: description, actions: [{ name: actionName, description: description }] }];
+                }
+                if (context.preferences.provideRefactorNotApplicableReason) {
+                    return [
+                        { name: refactorName, description: ts.Diagnostics.Convert_default_export_to_named_export.message, actions: [{ name: actionNameDefaultToNamed, description: ts.Diagnostics.Convert_default_export_to_named_export.message, notApplicableReason: info.error }] },
+                        { name: refactorName, description: ts.Diagnostics.Convert_named_export_to_default_export.message, actions: [{ name: actionNameNamedToDefault, description: ts.Diagnostics.Convert_named_export_to_default_export.message, notApplicableReason: info.error }] },
+                    ];
+                }
+                return ts.emptyArray;
             },
             getEditsForAction: function (context, actionName) {
                 ts.Debug.assert(actionName === actionNameDefaultToNamed || actionName === actionNameNamedToDefault, "Unexpected action name");
-                var edits = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(context.file, context.program, ts.Debug.checkDefined(getInfo(context), "context must have info"), t, context.cancellationToken); });
+                var edits = ts.textChanges.ChangeTracker.with(context, function (t) { var _a; return doChange(context.file, context.program, ts.Debug.checkDefined((_a = getInfo(context)) === null || _a === void 0 ? void 0 : _a.info, "context must have info"), t, context.cancellationToken); });
                 return { edits: edits, renameFilename: undefined, renameLocation: undefined };
             },
         });
-        function getInfo(context) {
+        function getInfo(context, considerPartialSpans) {
+            if (considerPartialSpans === void 0) { considerPartialSpans = true; }
             var file = context.file;
             var span = ts.getRefactorContextSpan(context);
             var token = ts.getTokenAtPosition(file, span.start);
-            var exportNode = ts.getParentNodeInSpan(token, file, span);
+            var exportNode = !!(token.parent && ts.getSyntacticModifierFlags(token.parent) & 1 /* Export */) && considerPartialSpans ? token.parent : ts.getParentNodeInSpan(token, file, span);
             if (!exportNode || (!ts.isSourceFile(exportNode.parent) && !(ts.isModuleBlock(exportNode.parent) && ts.isAmbientModule(exportNode.parent.parent)))) {
-                return undefined;
+                return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_export_statement) };
             }
             var exportingModuleSymbol = ts.isSourceFile(exportNode.parent) ? exportNode.parent.symbol : exportNode.parent.parent.symbol;
-            var flags = ts.getModifierFlags(exportNode);
+            var flags = ts.getSyntacticModifierFlags(exportNode);
             var wasDefault = !!(flags & 512 /* Default */);
             // If source file already has a default export, don't offer refactor.
             if (!(flags & 1 /* Export */) || !wasDefault && exportingModuleSymbol.exports.has("default" /* Default */)) {
-                return undefined;
+                return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.This_file_already_has_a_default_export) };
             }
             switch (exportNode.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 249 /* ModuleDeclaration */: {
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 256 /* ModuleDeclaration */: {
                     var node = exportNode;
-                    return node.name && ts.isIdentifier(node.name) ? { exportNode: node, exportName: node.name, wasDefault: wasDefault, exportingModuleSymbol: exportingModuleSymbol } : undefined;
+                    return node.name && ts.isIdentifier(node.name) ? { info: { exportNode: node, exportName: node.name, wasDefault: wasDefault, exportingModuleSymbol: exportingModuleSymbol } } : undefined;
                 }
-                case 225 /* VariableStatement */: {
+                case 232 /* VariableStatement */: {
                     var vs = exportNode;
                     // Must be `export const x = something;`.
                     if (!(vs.declarationList.flags & 2 /* Const */) || vs.declarationList.declarations.length !== 1) {
@@ -134599,7 +141766,7 @@ var ts;
                     if (!decl.initializer)
                         return undefined;
                     ts.Debug.assert(!wasDefault, "Can't have a default flag here");
-                    return ts.isIdentifier(decl.name) ? { exportNode: vs, exportName: decl.name, wasDefault: wasDefault, exportingModuleSymbol: exportingModuleSymbol } : undefined;
+                    return ts.isIdentifier(decl.name) ? { info: { exportNode: vs, exportName: decl.name, wasDefault: wasDefault, exportingModuleSymbol: exportingModuleSymbol } } : undefined;
                 }
                 default:
                     return undefined;
@@ -134612,30 +141779,31 @@ var ts;
         function changeExport(exportingSourceFile, _a, changes, checker) {
             var wasDefault = _a.wasDefault, exportNode = _a.exportNode, exportName = _a.exportName;
             if (wasDefault) {
-                changes.delete(exportingSourceFile, ts.Debug.checkDefined(ts.findModifier(exportNode, 84 /* DefaultKeyword */), "Should find a default keyword in modifier list"));
+                changes.delete(exportingSourceFile, ts.Debug.checkDefined(ts.findModifier(exportNode, 87 /* DefaultKeyword */), "Should find a default keyword in modifier list"));
             }
             else {
-                var exportKeyword = ts.Debug.checkDefined(ts.findModifier(exportNode, 89 /* ExportKeyword */), "Should find an export keyword in modifier list");
+                var exportKeyword = ts.Debug.checkDefined(ts.findModifier(exportNode, 92 /* ExportKeyword */), "Should find an export keyword in modifier list");
                 switch (exportNode.kind) {
-                    case 244 /* FunctionDeclaration */:
-                    case 245 /* ClassDeclaration */:
-                    case 246 /* InterfaceDeclaration */:
-                        changes.insertNodeAfter(exportingSourceFile, exportKeyword, ts.createToken(84 /* DefaultKeyword */));
+                    case 251 /* FunctionDeclaration */:
+                    case 252 /* ClassDeclaration */:
+                    case 253 /* InterfaceDeclaration */:
+                        changes.insertNodeAfter(exportingSourceFile, exportKeyword, ts.factory.createToken(87 /* DefaultKeyword */));
                         break;
-                    case 225 /* VariableStatement */:
-                        // If 'x' isn't used in this file, `export const x = 0;` --> `export default 0;`
-                        if (!ts.FindAllReferences.Core.isSymbolReferencedInFile(exportName, checker, exportingSourceFile)) {
+                    case 232 /* VariableStatement */:
+                        // If 'x' isn't used in this file and doesn't have type definition, `export const x = 0;` --> `export default 0;`
+                        var decl = ts.first(exportNode.declarationList.declarations);
+                        if (!ts.FindAllReferences.Core.isSymbolReferencedInFile(exportName, checker, exportingSourceFile) && !decl.type) {
                             // We checked in `getInfo` that an initializer exists.
-                            changes.replaceNode(exportingSourceFile, exportNode, ts.createExportDefault(ts.Debug.checkDefined(ts.first(exportNode.declarationList.declarations).initializer, "Initializer was previously known to be present")));
+                            changes.replaceNode(exportingSourceFile, exportNode, ts.factory.createExportDefault(ts.Debug.checkDefined(decl.initializer, "Initializer was previously known to be present")));
                             break;
                         }
                     // falls through
-                    case 248 /* EnumDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 249 /* ModuleDeclaration */:
+                    case 255 /* EnumDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 256 /* ModuleDeclaration */:
                         // `export type T = number;` -> `type T = number; export default T;`
                         changes.deleteModifier(exportingSourceFile, exportKeyword);
-                        changes.insertNodeAfter(exportingSourceFile, exportNode, ts.createExportDefault(ts.createIdentifier(exportName.text)));
+                        changes.insertNodeAfter(exportingSourceFile, exportNode, ts.factory.createExportDefault(ts.factory.createIdentifier(exportName.text)));
                         break;
                     default:
                         ts.Debug.assertNever(exportNode, "Unexpected exportNode kind " + exportNode.kind);
@@ -134659,27 +141827,27 @@ var ts;
         function changeDefaultToNamedImport(importingSourceFile, ref, changes, exportName) {
             var parent = ref.parent;
             switch (parent.kind) {
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     // `a.default` --> `a.foo`
-                    changes.replaceNode(importingSourceFile, ref, ts.createIdentifier(exportName));
+                    changes.replaceNode(importingSourceFile, ref, ts.factory.createIdentifier(exportName));
                     break;
-                case 258 /* ImportSpecifier */:
-                case 263 /* ExportSpecifier */: {
+                case 265 /* ImportSpecifier */:
+                case 270 /* ExportSpecifier */: {
                     var spec = parent;
                     // `default as foo` --> `foo`, `default as bar` --> `foo as bar`
                     changes.replaceNode(importingSourceFile, spec, makeImportSpecifier(exportName, spec.name.text));
                     break;
                 }
-                case 255 /* ImportClause */: {
+                case 262 /* ImportClause */: {
                     var clause = parent;
                     ts.Debug.assert(clause.name === ref, "Import clause name should match provided ref");
                     var spec = makeImportSpecifier(exportName, ref.text);
                     var namedBindings = clause.namedBindings;
                     if (!namedBindings) {
                         // `import foo from "./a";` --> `import { foo } from "./a";`
-                        changes.replaceNode(importingSourceFile, ref, ts.createNamedImports([spec]));
+                        changes.replaceNode(importingSourceFile, ref, ts.factory.createNamedImports([spec]));
                     }
-                    else if (namedBindings.kind === 256 /* NamespaceImport */) {
+                    else if (namedBindings.kind === 263 /* NamespaceImport */) {
                         // `import foo, * as a from "./a";` --> `import * as a from ".a/"; import { foo } from "./a";`
                         changes.deleteRange(importingSourceFile, { pos: ref.getStart(importingSourceFile), end: namedBindings.getStart(importingSourceFile) });
                         var quotePreference = ts.isStringLiteral(clause.parent.moduleSpecifier) ? ts.quotePreferenceFromString(clause.parent.moduleSpecifier, importingSourceFile) : 1 /* Double */;
@@ -134700,14 +141868,14 @@ var ts;
         function changeNamedToDefaultImport(importingSourceFile, ref, changes) {
             var parent = ref.parent;
             switch (parent.kind) {
-                case 194 /* PropertyAccessExpression */:
+                case 201 /* PropertyAccessExpression */:
                     // `a.foo` --> `a.default`
-                    changes.replaceNode(importingSourceFile, ref, ts.createIdentifier("default"));
+                    changes.replaceNode(importingSourceFile, ref, ts.factory.createIdentifier("default"));
                     break;
-                case 258 /* ImportSpecifier */: {
+                case 265 /* ImportSpecifier */: {
                     // `import { foo } from "./a";` --> `import foo from "./a";`
                     // `import { foo as bar } from "./a";` --> `import bar from "./a";`
-                    var defaultImport = ts.createIdentifier(parent.name.text);
+                    var defaultImport = ts.factory.createIdentifier(parent.name.text);
                     if (parent.parent.elements.length === 1) {
                         changes.replaceNode(importingSourceFile, parent.parent, defaultImport);
                     }
@@ -134717,7 +141885,7 @@ var ts;
                     }
                     break;
                 }
-                case 263 /* ExportSpecifier */: {
+                case 270 /* ExportSpecifier */: {
                     // `export { foo } from "./a";` --> `export { default as foo } from "./a";`
                     // `export { foo as bar } from "./a";` --> `export { default as bar } from "./a";`
                     // `export { foo as default } from "./a";` --> `export { default } from "./a";`
@@ -134730,10 +141898,10 @@ var ts;
             }
         }
         function makeImportSpecifier(propertyName, name) {
-            return ts.createImportSpecifier(propertyName === name ? undefined : ts.createIdentifier(propertyName), ts.createIdentifier(name));
+            return ts.factory.createImportSpecifier(propertyName === name ? undefined : ts.factory.createIdentifier(propertyName), ts.factory.createIdentifier(name));
         }
         function makeExportSpecifier(propertyName, name) {
-            return ts.createExportSpecifier(propertyName === name ? undefined : ts.createIdentifier(propertyName), ts.createIdentifier(name));
+            return ts.factory.createExportSpecifier(propertyName === name ? undefined : ts.factory.createIdentifier(propertyName), ts.factory.createIdentifier(name));
         }
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
@@ -134747,33 +141915,51 @@ var ts;
         var actionNameNamedToNamespace = "Convert named imports to namespace import";
         refactor.registerRefactor(refactorName, {
             getAvailableActions: function (context) {
-                var i = getImportToConvert(context);
+                var i = getImportToConvert(context, context.triggerReason === "invoked");
                 if (!i)
                     return ts.emptyArray;
-                var description = i.kind === 256 /* NamespaceImport */ ? ts.Diagnostics.Convert_namespace_import_to_named_imports.message : ts.Diagnostics.Convert_named_imports_to_namespace_import.message;
-                var actionName = i.kind === 256 /* NamespaceImport */ ? actionNameNamespaceToNamed : actionNameNamedToNamespace;
-                return [{ name: refactorName, description: description, actions: [{ name: actionName, description: description }] }];
+                if (i.error === undefined) {
+                    var description = i.info.kind === 263 /* NamespaceImport */ ? ts.Diagnostics.Convert_namespace_import_to_named_imports.message : ts.Diagnostics.Convert_named_imports_to_namespace_import.message;
+                    var actionName = i.info.kind === 263 /* NamespaceImport */ ? actionNameNamespaceToNamed : actionNameNamedToNamespace;
+                    return [{ name: refactorName, description: description, actions: [{ name: actionName, description: description }] }];
+                }
+                if (context.preferences.provideRefactorNotApplicableReason) {
+                    return [
+                        { name: refactorName, description: ts.Diagnostics.Convert_namespace_import_to_named_imports.message, actions: [{ name: actionNameNamespaceToNamed, description: ts.Diagnostics.Convert_namespace_import_to_named_imports.message, notApplicableReason: i.error }] },
+                        { name: refactorName, description: ts.Diagnostics.Convert_named_imports_to_namespace_import.message, actions: [{ name: actionNameNamedToNamespace, description: ts.Diagnostics.Convert_named_imports_to_namespace_import.message, notApplicableReason: i.error }] }
+                    ];
+                }
+                return ts.emptyArray;
             },
             getEditsForAction: function (context, actionName) {
                 ts.Debug.assert(actionName === actionNameNamespaceToNamed || actionName === actionNameNamedToNamespace, "Unexpected action name");
-                var edits = ts.textChanges.ChangeTracker.with(context, function (t) { return doChange(context.file, context.program, t, ts.Debug.checkDefined(getImportToConvert(context), "Context must provide an import to convert")); });
+                var edits = ts.textChanges.ChangeTracker.with(context, function (t) { var _a; return doChange(context.file, context.program, t, ts.Debug.checkDefined((_a = getImportToConvert(context)) === null || _a === void 0 ? void 0 : _a.info, "Context must provide an import to convert")); });
                 return { edits: edits, renameFilename: undefined, renameLocation: undefined };
             }
         });
         // Can convert imports of the form `import * as m from "m";` or `import d, { x, y } from "m";`.
-        function getImportToConvert(context) {
+        function getImportToConvert(context, considerPartialSpans) {
+            if (considerPartialSpans === void 0) { considerPartialSpans = true; }
             var file = context.file;
             var span = ts.getRefactorContextSpan(context);
             var token = ts.getTokenAtPosition(file, span.start);
-            var importDecl = ts.getParentNodeInSpan(token, file, span);
+            var importDecl = considerPartialSpans ? ts.findAncestor(token, ts.isImportDeclaration) : ts.getParentNodeInSpan(token, file, span);
             if (!importDecl || !ts.isImportDeclaration(importDecl))
+                return { error: "Selection is not an import declaration." };
+            if (importDecl.getEnd() < span.start + span.length)
                 return undefined;
             var importClause = importDecl.importClause;
-            return importClause && importClause.namedBindings;
+            if (!importClause) {
+                return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_import_clause) };
+            }
+            if (!importClause.namedBindings) {
+                return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_namespace_import_or_named_imports) };
+            }
+            return { info: importClause.namedBindings };
         }
         function doChange(sourceFile, program, changes, toConvert) {
             var checker = program.getTypeChecker();
-            if (toConvert.kind === 256 /* NamespaceImport */) {
+            if (toConvert.kind === 263 /* NamespaceImport */) {
                 doChangeNamespaceToNamed(sourceFile, checker, changes, toConvert, ts.getAllowSyntheticDefaultImports(program.getCompilerOptions()));
             }
             else {
@@ -134783,7 +141969,7 @@ var ts;
         function doChangeNamespaceToNamed(sourceFile, checker, changes, toConvert, allowSyntheticDefaultImports) {
             var usedAsNamespaceOrDefault = false;
             var nodesToReplace = [];
-            var conflictingNames = ts.createMap();
+            var conflictingNames = new ts.Map();
             ts.FindAllReferences.Core.eachSymbolReferenceInFile(toConvert.name, checker, sourceFile, function (id) {
                 if (!ts.isPropertyAccessExpression(id.parent)) {
                     usedAsNamespaceOrDefault = true;
@@ -134799,7 +141985,7 @@ var ts;
                 }
             });
             // We may need to change `mod.x` to `_x` to avoid a name conflict.
-            var exportNameToImportName = ts.createMap();
+            var exportNameToImportName = new ts.Map();
             for (var _i = 0, nodesToReplace_1 = nodesToReplace; _i < nodesToReplace_1.length; _i++) {
                 var propertyAccess = nodesToReplace_1[_i];
                 var exportName = propertyAccess.name.text;
@@ -134807,11 +141993,11 @@ var ts;
                 if (importName === undefined) {
                     exportNameToImportName.set(exportName, importName = conflictingNames.has(exportName) ? ts.getUniqueName(exportName, sourceFile) : exportName);
                 }
-                changes.replaceNode(sourceFile, propertyAccess, ts.createIdentifier(importName));
+                changes.replaceNode(sourceFile, propertyAccess, ts.factory.createIdentifier(importName));
             }
             var importSpecifiers = [];
             exportNameToImportName.forEach(function (name, propertyName) {
-                importSpecifiers.push(ts.createImportSpecifier(name === propertyName ? undefined : ts.createIdentifier(propertyName), ts.createIdentifier(name)));
+                importSpecifiers.push(ts.factory.createImportSpecifier(name === propertyName ? undefined : ts.factory.createIdentifier(propertyName), ts.factory.createIdentifier(name)));
             });
             var importDecl = toConvert.parent.parent;
             if (usedAsNamespaceOrDefault && !allowSyntheticDefaultImports) {
@@ -134819,7 +142005,7 @@ var ts;
                 changes.insertNodeAfter(sourceFile, importDecl, updateImport(importDecl, /*defaultImportName*/ undefined, importSpecifiers));
             }
             else {
-                changes.replaceNode(sourceFile, importDecl, updateImport(importDecl, usedAsNamespaceOrDefault ? ts.createIdentifier(toConvert.name.text) : undefined, importSpecifiers));
+                changes.replaceNode(sourceFile, importDecl, updateImport(importDecl, usedAsNamespaceOrDefault ? ts.factory.createIdentifier(toConvert.name.text) : undefined, importSpecifiers));
             }
         }
         function doChangeNamedToNamespace(sourceFile, checker, changes, toConvert) {
@@ -134836,13 +142022,13 @@ var ts;
             var _loop_15 = function (element) {
                 var propertyName = (element.propertyName || element.name).text;
                 ts.FindAllReferences.Core.eachSymbolReferenceInFile(element.name, checker, sourceFile, function (id) {
-                    var access = ts.createPropertyAccess(ts.createIdentifier(namespaceImportName), propertyName);
+                    var access = ts.factory.createPropertyAccessExpression(ts.factory.createIdentifier(namespaceImportName), propertyName);
                     if (ts.isShorthandPropertyAssignment(id.parent)) {
-                        changes.replaceNode(sourceFile, id.parent, ts.createPropertyAssignment(id.text, access));
+                        changes.replaceNode(sourceFile, id.parent, ts.factory.createPropertyAssignment(id.text, access));
                     }
                     else if (ts.isExportSpecifier(id.parent) && !id.parent.propertyName) {
                         if (!neededNamedImports.some(function (n) { return n.name === element.name; })) {
-                            neededNamedImports.push(ts.createImportSpecifier(element.propertyName && ts.createIdentifier(element.propertyName.text), ts.createIdentifier(element.name.text)));
+                            neededNamedImports.push(ts.factory.createImportSpecifier(element.propertyName && ts.factory.createIdentifier(element.propertyName.text), ts.factory.createIdentifier(element.name.text)));
                         }
                     }
                     else {
@@ -134854,18 +142040,440 @@ var ts;
                 var element = _a[_i];
                 _loop_15(element);
             }
-            changes.replaceNode(sourceFile, toConvert, ts.createNamespaceImport(ts.createIdentifier(namespaceImportName)));
+            changes.replaceNode(sourceFile, toConvert, ts.factory.createNamespaceImport(ts.factory.createIdentifier(namespaceImportName)));
             if (neededNamedImports.length) {
                 changes.insertNodeAfter(sourceFile, toConvert.parent.parent, updateImport(importDecl, /*defaultImportName*/ undefined, neededNamedImports));
             }
         }
         function updateImport(old, defaultImportName, elements) {
-            return ts.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.createImportClause(defaultImportName, elements && elements.length ? ts.createNamedImports(elements) : undefined), old.moduleSpecifier);
+            return ts.factory.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, defaultImportName, elements && elements.length ? ts.factory.createNamedImports(elements) : undefined), old.moduleSpecifier);
         }
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
 /* @internal */
 var ts;
+(function (ts) {
+    var refactor;
+    (function (refactor) {
+        var convertToOptionalChainExpression;
+        (function (convertToOptionalChainExpression) {
+            var refactorName = "Convert to optional chain expression";
+            var convertToOptionalChainExpressionMessage = ts.getLocaleSpecificMessage(ts.Diagnostics.Convert_to_optional_chain_expression);
+            refactor.registerRefactor(refactorName, { getAvailableActions: getAvailableActions, getEditsForAction: getEditsForAction });
+            function getAvailableActions(context) {
+                var info = getInfo(context, context.triggerReason === "invoked");
+                if (!info)
+                    return ts.emptyArray;
+                if (!info.error) {
+                    return [{
+                            name: refactorName,
+                            description: convertToOptionalChainExpressionMessage,
+                            actions: [{
+                                    name: refactorName,
+                                    description: convertToOptionalChainExpressionMessage
+                                }]
+                        }];
+                }
+                if (context.preferences.provideRefactorNotApplicableReason) {
+                    return [{
+                            name: refactorName,
+                            description: convertToOptionalChainExpressionMessage,
+                            actions: [{
+                                    name: refactorName,
+                                    description: convertToOptionalChainExpressionMessage,
+                                    notApplicableReason: info.error
+                                }]
+                        }];
+                }
+                return ts.emptyArray;
+            }
+            function getEditsForAction(context, actionName) {
+                var info = getInfo(context);
+                if (!info || !info.info)
+                    return undefined;
+                var edits = ts.textChanges.ChangeTracker.with(context, function (t) {
+                    return doChange(context.file, context.program.getTypeChecker(), t, ts.Debug.checkDefined(info.info, "context must have info"), actionName);
+                });
+                return { edits: edits, renameFilename: undefined, renameLocation: undefined };
+            }
+            ;
+            function isValidExpression(node) {
+                return ts.isBinaryExpression(node) || ts.isConditionalExpression(node);
+            }
+            function isValidStatement(node) {
+                return ts.isExpressionStatement(node) || ts.isReturnStatement(node) || ts.isVariableStatement(node);
+            }
+            function isValidExpressionOrStatement(node) {
+                return isValidExpression(node) || isValidStatement(node);
+            }
+            function getInfo(context, considerEmptySpans) {
+                if (considerEmptySpans === void 0) { considerEmptySpans = true; }
+                var file = context.file, program = context.program;
+                var span = ts.getRefactorContextSpan(context);
+                var forEmptySpan = span.length === 0;
+                if (forEmptySpan && !considerEmptySpans)
+                    return undefined;
+                // selecting fo[|o && foo.ba|]r should be valid, so adjust span to fit start and end tokens
+                var startToken = ts.getTokenAtPosition(file, span.start);
+                var endToken = ts.findTokenOnLeftOfPosition(file, span.start + span.length);
+                var adjustedSpan = ts.createTextSpanFromBounds(startToken.pos, endToken && endToken.end >= startToken.pos ? endToken.getEnd() : startToken.getEnd());
+                var parent = forEmptySpan ? getValidParentNodeOfEmptySpan(startToken) : getValidParentNodeContainingSpan(startToken, adjustedSpan);
+                var expression = parent && isValidExpressionOrStatement(parent) ? getExpression(parent) : undefined;
+                if (!expression)
+                    return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_convertible_access_expression) };
+                var checker = program.getTypeChecker();
+                return ts.isConditionalExpression(expression) ? getConditionalInfo(expression, checker) : getBinaryInfo(expression);
+            }
+            function getConditionalInfo(expression, checker) {
+                var condition = expression.condition;
+                var finalExpression = getFinalExpressionInChain(expression.whenTrue);
+                if (!finalExpression || checker.isNullableType(checker.getTypeAtLocation(finalExpression))) {
+                    return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_convertible_access_expression) };
+                }
+                if ((ts.isPropertyAccessExpression(condition) || ts.isIdentifier(condition))
+                    && getMatchingStart(condition, finalExpression.expression)) {
+                    return { info: { finalExpression: finalExpression, occurrences: [condition], expression: expression } };
+                }
+                else if (ts.isBinaryExpression(condition)) {
+                    var occurrences = getOccurrencesInExpression(finalExpression.expression, condition);
+                    return occurrences ? { info: { finalExpression: finalExpression, occurrences: occurrences, expression: expression } } :
+                        { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_matching_access_expressions) };
+                }
+            }
+            function getBinaryInfo(expression) {
+                if (expression.operatorToken.kind !== 55 /* AmpersandAmpersandToken */) {
+                    return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Can_only_convert_logical_AND_access_chains) };
+                }
+                ;
+                var finalExpression = getFinalExpressionInChain(expression.right);
+                if (!finalExpression)
+                    return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_convertible_access_expression) };
+                var occurrences = getOccurrencesInExpression(finalExpression.expression, expression.left);
+                return occurrences ? { info: { finalExpression: finalExpression, occurrences: occurrences, expression: expression } } :
+                    { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_matching_access_expressions) };
+            }
+            /**
+             * Gets a list of property accesses that appear in matchTo and occur in sequence in expression.
+             */
+            function getOccurrencesInExpression(matchTo, expression) {
+                var occurrences = [];
+                while (ts.isBinaryExpression(expression) && expression.operatorToken.kind === 55 /* AmpersandAmpersandToken */) {
+                    var match = getMatchingStart(ts.skipParentheses(matchTo), ts.skipParentheses(expression.right));
+                    if (!match) {
+                        break;
+                    }
+                    occurrences.push(match);
+                    matchTo = match;
+                    expression = expression.left;
+                }
+                var finalMatch = getMatchingStart(matchTo, expression);
+                if (finalMatch) {
+                    occurrences.push(finalMatch);
+                }
+                return occurrences.length > 0 ? occurrences : undefined;
+            }
+            /**
+             * Returns subchain if chain begins with subchain syntactically.
+             */
+            function getMatchingStart(chain, subchain) {
+                if (!ts.isIdentifier(subchain) && !ts.isPropertyAccessExpression(subchain) && !ts.isElementAccessExpression(subchain)) {
+                    return undefined;
+                }
+                return chainStartsWith(chain, subchain) ? subchain : undefined;
+            }
+            /**
+             * Returns true if chain begins with subchain syntactically.
+             */
+            function chainStartsWith(chain, subchain) {
+                // skip until we find a matching identifier.
+                while (ts.isCallExpression(chain) || ts.isPropertyAccessExpression(chain) || ts.isElementAccessExpression(chain)) {
+                    if (getTextOfChainNode(chain) === getTextOfChainNode(subchain))
+                        break;
+                    chain = chain.expression;
+                }
+                // check that the chains match at each access. Call chains in subchain are not valid.
+                while ((ts.isPropertyAccessExpression(chain) && ts.isPropertyAccessExpression(subchain)) ||
+                    (ts.isElementAccessExpression(chain) && ts.isElementAccessExpression(subchain))) {
+                    if (getTextOfChainNode(chain) !== getTextOfChainNode(subchain))
+                        return false;
+                    chain = chain.expression;
+                    subchain = subchain.expression;
+                }
+                // check if we have reached a final identifier.
+                return ts.isIdentifier(chain) && ts.isIdentifier(subchain) && chain.getText() === subchain.getText();
+            }
+            function getTextOfChainNode(node) {
+                if (ts.isIdentifier(node) || ts.isStringOrNumericLiteralLike(node)) {
+                    return node.getText();
+                }
+                if (ts.isPropertyAccessExpression(node)) {
+                    return getTextOfChainNode(node.name);
+                }
+                if (ts.isElementAccessExpression(node)) {
+                    return getTextOfChainNode(node.argumentExpression);
+                }
+                return undefined;
+            }
+            /**
+             * Find the least ancestor of the input node that is a valid type for extraction and contains the input span.
+             */
+            function getValidParentNodeContainingSpan(node, span) {
+                while (node.parent) {
+                    if (isValidExpressionOrStatement(node) && span.length !== 0 && node.end >= span.start + span.length) {
+                        return node;
+                    }
+                    node = node.parent;
+                }
+                return undefined;
+            }
+            /**
+             * Finds an ancestor of the input node that is a valid type for extraction, skipping subexpressions.
+             */
+            function getValidParentNodeOfEmptySpan(node) {
+                while (node.parent) {
+                    if (isValidExpressionOrStatement(node) && !isValidExpressionOrStatement(node.parent)) {
+                        return node;
+                    }
+                    node = node.parent;
+                }
+                return undefined;
+            }
+            /**
+             * Gets an expression of valid extraction type from a valid statement or expression.
+             */
+            function getExpression(node) {
+                if (isValidExpression(node)) {
+                    return node;
+                }
+                if (ts.isVariableStatement(node)) {
+                    var variable = ts.getSingleVariableOfVariableStatement(node);
+                    var initializer = variable === null || variable === void 0 ? void 0 : variable.initializer;
+                    return initializer && isValidExpression(initializer) ? initializer : undefined;
+                }
+                return node.expression && isValidExpression(node.expression) ? node.expression : undefined;
+            }
+            /**
+             * Gets a property access expression which may be nested inside of a binary expression. The final
+             * expression in an && chain will occur as the right child of the parent binary expression, unless
+             * it is followed by a different binary operator.
+             * @param node the right child of a binary expression or a call expression.
+             */
+            function getFinalExpressionInChain(node) {
+                // foo && |foo.bar === 1|; - here the right child of the && binary expression is another binary expression.
+                // the rightmost member of the && chain should be the leftmost child of that expression.
+                node = ts.skipParentheses(node);
+                if (ts.isBinaryExpression(node)) {
+                    return getFinalExpressionInChain(node.left);
+                }
+                // foo && |foo.bar()()| - nested calls are treated like further accesses.
+                else if ((ts.isPropertyAccessExpression(node) || ts.isElementAccessExpression(node) || ts.isCallExpression(node)) && !ts.isOptionalChain(node)) {
+                    return node;
+                }
+                return undefined;
+            }
+            /**
+             * Creates an access chain from toConvert with '?.' accesses at expressions appearing in occurrences.
+             */
+            function convertOccurrences(checker, toConvert, occurrences) {
+                if (ts.isPropertyAccessExpression(toConvert) || ts.isElementAccessExpression(toConvert) || ts.isCallExpression(toConvert)) {
+                    var chain = convertOccurrences(checker, toConvert.expression, occurrences);
+                    var lastOccurrence = occurrences.length > 0 ? occurrences[occurrences.length - 1] : undefined;
+                    var isOccurrence = (lastOccurrence === null || lastOccurrence === void 0 ? void 0 : lastOccurrence.getText()) === toConvert.expression.getText();
+                    if (isOccurrence)
+                        occurrences.pop();
+                    if (ts.isCallExpression(toConvert)) {
+                        return isOccurrence ?
+                            ts.factory.createCallChain(chain, ts.factory.createToken(28 /* QuestionDotToken */), toConvert.typeArguments, toConvert.arguments) :
+                            ts.factory.createCallChain(chain, toConvert.questionDotToken, toConvert.typeArguments, toConvert.arguments);
+                    }
+                    else if (ts.isPropertyAccessExpression(toConvert)) {
+                        return isOccurrence ?
+                            ts.factory.createPropertyAccessChain(chain, ts.factory.createToken(28 /* QuestionDotToken */), toConvert.name) :
+                            ts.factory.createPropertyAccessChain(chain, toConvert.questionDotToken, toConvert.name);
+                    }
+                    else if (ts.isElementAccessExpression(toConvert)) {
+                        return isOccurrence ?
+                            ts.factory.createElementAccessChain(chain, ts.factory.createToken(28 /* QuestionDotToken */), toConvert.argumentExpression) :
+                            ts.factory.createElementAccessChain(chain, toConvert.questionDotToken, toConvert.argumentExpression);
+                    }
+                }
+                return toConvert;
+            }
+            function doChange(sourceFile, checker, changes, info, _actionName) {
+                var finalExpression = info.finalExpression, occurrences = info.occurrences, expression = info.expression;
+                var firstOccurrence = occurrences[occurrences.length - 1];
+                var convertedChain = convertOccurrences(checker, finalExpression, occurrences);
+                if (convertedChain && (ts.isPropertyAccessExpression(convertedChain) || ts.isElementAccessExpression(convertedChain) || ts.isCallExpression(convertedChain))) {
+                    if (ts.isBinaryExpression(expression)) {
+                        changes.replaceNodeRange(sourceFile, firstOccurrence, finalExpression, convertedChain);
+                    }
+                    else if (ts.isConditionalExpression(expression)) {
+                        changes.replaceNode(sourceFile, expression, ts.factory.createBinaryExpression(convertedChain, ts.factory.createToken(60 /* QuestionQuestionToken */), expression.whenFalse));
+                    }
+                }
+            }
+        })(convertToOptionalChainExpression = refactor.convertToOptionalChainExpression || (refactor.convertToOptionalChainExpression = {}));
+    })(refactor = ts.refactor || (ts.refactor = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var refactor;
+    (function (refactor) {
+        var addOrRemoveBracesToArrowFunction;
+        (function (addOrRemoveBracesToArrowFunction) {
+            var refactorName = "Convert overload list to single signature";
+            var refactorDescription = ts.Diagnostics.Convert_overload_list_to_single_signature.message;
+            refactor.registerRefactor(refactorName, { getEditsForAction: getEditsForAction, getAvailableActions: getAvailableActions });
+            function getAvailableActions(context) {
+                var file = context.file, startPosition = context.startPosition, program = context.program;
+                var info = getConvertableOverloadListAtPosition(file, startPosition, program);
+                if (!info)
+                    return ts.emptyArray;
+                return [{
+                        name: refactorName,
+                        description: refactorDescription,
+                        actions: [{
+                                name: refactorName,
+                                description: refactorDescription
+                            }]
+                    }];
+            }
+            function getEditsForAction(context) {
+                var file = context.file, startPosition = context.startPosition, program = context.program;
+                var signatureDecls = getConvertableOverloadListAtPosition(file, startPosition, program);
+                if (!signatureDecls)
+                    return undefined;
+                var checker = program.getTypeChecker();
+                var lastDeclaration = signatureDecls[signatureDecls.length - 1];
+                var updated = lastDeclaration;
+                switch (lastDeclaration.kind) {
+                    case 164 /* MethodSignature */: {
+                        updated = ts.factory.updateMethodSignature(lastDeclaration, lastDeclaration.modifiers, lastDeclaration.name, lastDeclaration.questionToken, lastDeclaration.typeParameters, getNewParametersForCombinedSignature(signatureDecls), lastDeclaration.type);
+                        break;
+                    }
+                    case 165 /* MethodDeclaration */: {
+                        updated = ts.factory.updateMethodDeclaration(lastDeclaration, lastDeclaration.decorators, lastDeclaration.modifiers, lastDeclaration.asteriskToken, lastDeclaration.name, lastDeclaration.questionToken, lastDeclaration.typeParameters, getNewParametersForCombinedSignature(signatureDecls), lastDeclaration.type, lastDeclaration.body);
+                        break;
+                    }
+                    case 169 /* CallSignature */: {
+                        updated = ts.factory.updateCallSignature(lastDeclaration, lastDeclaration.typeParameters, getNewParametersForCombinedSignature(signatureDecls), lastDeclaration.type);
+                        break;
+                    }
+                    case 166 /* Constructor */: {
+                        updated = ts.factory.updateConstructorDeclaration(lastDeclaration, lastDeclaration.decorators, lastDeclaration.modifiers, getNewParametersForCombinedSignature(signatureDecls), lastDeclaration.body);
+                        break;
+                    }
+                    case 170 /* ConstructSignature */: {
+                        updated = ts.factory.updateConstructSignature(lastDeclaration, lastDeclaration.typeParameters, getNewParametersForCombinedSignature(signatureDecls), lastDeclaration.type);
+                        break;
+                    }
+                    case 251 /* FunctionDeclaration */: {
+                        updated = ts.factory.updateFunctionDeclaration(lastDeclaration, lastDeclaration.decorators, lastDeclaration.modifiers, lastDeclaration.asteriskToken, lastDeclaration.name, lastDeclaration.typeParameters, getNewParametersForCombinedSignature(signatureDecls), lastDeclaration.type, lastDeclaration.body);
+                        break;
+                    }
+                    default: return ts.Debug.failBadSyntaxKind(lastDeclaration, "Unhandled signature kind in overload list conversion refactoring");
+                }
+                if (updated === lastDeclaration) {
+                    return; // No edits to apply, do nothing
+                }
+                var edits = ts.textChanges.ChangeTracker.with(context, function (t) {
+                    t.replaceNodeRange(file, signatureDecls[0], signatureDecls[signatureDecls.length - 1], updated);
+                });
+                return { renameFilename: undefined, renameLocation: undefined, edits: edits };
+                function getNewParametersForCombinedSignature(signatureDeclarations) {
+                    var lastSig = signatureDeclarations[signatureDeclarations.length - 1];
+                    if (ts.isFunctionLikeDeclaration(lastSig) && lastSig.body) {
+                        // Trim away implementation signature arguments (they should already be compatible with overloads, but are likely less precise to guarantee compatability with the overloads)
+                        signatureDeclarations = signatureDeclarations.slice(0, signatureDeclarations.length - 1);
+                    }
+                    return ts.factory.createNodeArray([
+                        ts.factory.createParameterDeclaration(
+                        /*decorators*/ undefined, 
+                        /*modifiers*/ undefined, ts.factory.createToken(25 /* DotDotDotToken */), "args", 
+                        /*questionToken*/ undefined, ts.factory.createUnionTypeNode(ts.map(signatureDeclarations, convertSignatureParametersToTuple)))
+                    ]);
+                }
+                function convertSignatureParametersToTuple(decl) {
+                    var members = ts.map(decl.parameters, convertParameterToNamedTupleMember);
+                    return ts.setEmitFlags(ts.factory.createTupleTypeNode(members), ts.some(members, function (m) { return !!ts.length(ts.getSyntheticLeadingComments(m)); }) ? 0 /* None */ : 1 /* SingleLine */);
+                }
+                function convertParameterToNamedTupleMember(p) {
+                    ts.Debug.assert(ts.isIdentifier(p.name)); // This is checked during refactoring applicability checking
+                    var result = ts.setTextRange(ts.factory.createNamedTupleMember(p.dotDotDotToken, p.name, p.questionToken, p.type || ts.factory.createKeywordTypeNode(128 /* AnyKeyword */)), p);
+                    var parameterDocComment = p.symbol && p.symbol.getDocumentationComment(checker);
+                    if (parameterDocComment) {
+                        var newComment = ts.displayPartsToString(parameterDocComment);
+                        if (newComment.length) {
+                            ts.setSyntheticLeadingComments(result, [{
+                                    text: "*\n" + newComment.split("\n").map(function (c) { return " * " + c; }).join("\n") + "\n ",
+                                    kind: 3 /* MultiLineCommentTrivia */,
+                                    pos: -1,
+                                    end: -1,
+                                    hasTrailingNewLine: true,
+                                    hasLeadingNewline: true,
+                                }]);
+                        }
+                    }
+                    return result;
+                }
+            }
+            function isConvertableSignatureDeclaration(d) {
+                switch (d.kind) {
+                    case 164 /* MethodSignature */:
+                    case 165 /* MethodDeclaration */:
+                    case 169 /* CallSignature */:
+                    case 166 /* Constructor */:
+                    case 170 /* ConstructSignature */:
+                    case 251 /* FunctionDeclaration */:
+                        return true;
+                }
+                return false;
+            }
+            function getConvertableOverloadListAtPosition(file, startPosition, program) {
+                var node = ts.getTokenAtPosition(file, startPosition);
+                var containingDecl = ts.findAncestor(node, isConvertableSignatureDeclaration);
+                if (!containingDecl) {
+                    return;
+                }
+                var checker = program.getTypeChecker();
+                var signatureSymbol = containingDecl.symbol;
+                if (!signatureSymbol) {
+                    return;
+                }
+                var decls = signatureSymbol.declarations;
+                if (ts.length(decls) <= 1) {
+                    return;
+                }
+                if (!ts.every(decls, function (d) { return ts.getSourceFileOfNode(d) === file; })) {
+                    return;
+                }
+                if (!isConvertableSignatureDeclaration(decls[0])) {
+                    return;
+                }
+                var kindOne = decls[0].kind;
+                if (!ts.every(decls, function (d) { return d.kind === kindOne; })) {
+                    return;
+                }
+                var signatureDecls = decls;
+                if (ts.some(signatureDecls, function (d) { return !!d.typeParameters || ts.some(d.parameters, function (p) { return !!p.decorators || !!p.modifiers || !ts.isIdentifier(p.name); }); })) {
+                    return;
+                }
+                var signatures = ts.mapDefined(signatureDecls, function (d) { return checker.getSignatureFromDeclaration(d); });
+                if (ts.length(signatures) !== ts.length(decls)) {
+                    return;
+                }
+                var returnOne = checker.getReturnTypeOfSignature(signatures[0]);
+                if (!ts.every(signatures, function (s) { return checker.getReturnTypeOfSignature(s) === returnOne; })) {
+                    return;
+                }
+                return signatureDecls;
+            }
+        })(addOrRemoveBracesToArrowFunction = refactor.addOrRemoveBracesToArrowFunction || (refactor.addOrRemoveBracesToArrowFunction = {}));
+    })(refactor = ts.refactor || (ts.refactor = {}));
+})(ts || (ts = {}));
+/* @internal */
+var ts;
 (function (ts) {
     var refactor;
     (function (refactor) {
@@ -134878,10 +142486,30 @@ var ts;
              * Exported for tests.
              */
             function getAvailableActions(context) {
-                var rangeToExtract = getRangeToExtract(context.file, ts.getRefactorContextSpan(context));
+                var rangeToExtract = getRangeToExtract(context.file, ts.getRefactorContextSpan(context), context.triggerReason === "invoked");
                 var targetRange = rangeToExtract.targetRange;
                 if (targetRange === undefined) {
-                    return ts.emptyArray;
+                    if (!rangeToExtract.errors || rangeToExtract.errors.length === 0 || !context.preferences.provideRefactorNotApplicableReason) {
+                        return ts.emptyArray;
+                    }
+                    return [{
+                            name: refactorName,
+                            description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_function),
+                            actions: [{
+                                    description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_function),
+                                    name: "function_extract_error",
+                                    notApplicableReason: getStringError(rangeToExtract.errors)
+                                }]
+                        },
+                        {
+                            name: refactorName,
+                            description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_constant),
+                            actions: [{
+                                    description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_constant),
+                                    name: "constant_extract_error",
+                                    notApplicableReason: getStringError(rangeToExtract.errors)
+                                }]
+                        }];
                 }
                 var extractions = getPossibleExtractions(targetRange, context);
                 if (extractions === undefined) {
@@ -134889,18 +142517,19 @@ var ts;
                     return ts.emptyArray;
                 }
                 var functionActions = [];
-                var usedFunctionNames = ts.createMap();
+                var usedFunctionNames = new ts.Map();
+                var innermostErrorFunctionAction;
                 var constantActions = [];
-                var usedConstantNames = ts.createMap();
+                var usedConstantNames = new ts.Map();
+                var innermostErrorConstantAction;
                 var i = 0;
                 for (var _i = 0, extractions_1 = extractions; _i < extractions_1.length; _i++) {
                     var _a = extractions_1[_i], functionExtraction = _a.functionExtraction, constantExtraction = _a.constantExtraction;
-                    // Skip these since we don't have a way to report errors yet
+                    var description = functionExtraction.description;
                     if (functionExtraction.errors.length === 0) {
                         // Don't issue refactorings with duplicated names.
                         // Scopes come back in "innermost first" order, so extractions will
                         // preferentially go into nearer scopes
-                        var description = functionExtraction.description;
                         if (!usedFunctionNames.has(description)) {
                             usedFunctionNames.set(description, true);
                             functionActions.push({
@@ -134909,25 +142538,53 @@ var ts;
                             });
                         }
                     }
+                    else if (!innermostErrorFunctionAction) {
+                        innermostErrorFunctionAction = {
+                            description: description,
+                            name: "function_scope_" + i,
+                            notApplicableReason: getStringError(functionExtraction.errors)
+                        };
+                    }
                     // Skip these since we don't have a way to report errors yet
                     if (constantExtraction.errors.length === 0) {
                         // Don't issue refactorings with duplicated names.
                         // Scopes come back in "innermost first" order, so extractions will
                         // preferentially go into nearer scopes
-                        var description = constantExtraction.description;
-                        if (!usedConstantNames.has(description)) {
-                            usedConstantNames.set(description, true);
+                        var description_1 = constantExtraction.description;
+                        if (!usedConstantNames.has(description_1)) {
+                            usedConstantNames.set(description_1, true);
                             constantActions.push({
-                                description: description,
+                                description: description_1,
                                 name: "constant_scope_" + i
                             });
                         }
                     }
+                    else if (!innermostErrorConstantAction) {
+                        innermostErrorConstantAction = {
+                            description: description,
+                            name: "constant_scope_" + i,
+                            notApplicableReason: getStringError(constantExtraction.errors)
+                        };
+                    }
                     // *do* increment i anyway because we'll look for the i-th scope
                     // later when actually doing the refactoring if the user requests it
                     i++;
                 }
                 var infos = [];
+                if (functionActions.length) {
+                    infos.push({
+                        name: refactorName,
+                        description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_function),
+                        actions: functionActions
+                    });
+                }
+                else if (context.preferences.provideRefactorNotApplicableReason && innermostErrorFunctionAction) {
+                    infos.push({
+                        name: refactorName,
+                        description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_function),
+                        actions: [innermostErrorFunctionAction]
+                    });
+                }
                 if (constantActions.length) {
                     infos.push({
                         name: refactorName,
@@ -134935,14 +142592,21 @@ var ts;
                         actions: constantActions
                     });
                 }
-                if (functionActions.length) {
+                else if (context.preferences.provideRefactorNotApplicableReason && innermostErrorConstantAction) {
                     infos.push({
                         name: refactorName,
-                        description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_function),
-                        actions: functionActions
+                        description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_constant),
+                        actions: [innermostErrorConstantAction]
                     });
                 }
                 return infos.length ? infos : ts.emptyArray;
+                function getStringError(errors) {
+                    var error = errors[0].messageText;
+                    if (typeof error !== "string") {
+                        error = error.messageText;
+                    }
+                    return error;
+                }
             }
             extractSymbol.getAvailableActions = getAvailableActions;
             /* Exported for tests */
@@ -135013,16 +142677,20 @@ var ts;
              * not shown to the user, but can be used by us diagnostically)
              */
             // exported only for tests
-            function getRangeToExtract(sourceFile, span) {
+            function getRangeToExtract(sourceFile, span, considerEmptySpans) {
+                if (considerEmptySpans === void 0) { considerEmptySpans = true; }
                 var length = span.length;
-                if (length === 0) {
+                if (length === 0 && !considerEmptySpans) {
                     return { errors: [ts.createFileDiagnostic(sourceFile, span.start, length, Messages.cannotExtractEmpty)] };
                 }
+                var cursorRequest = length === 0 && considerEmptySpans;
                 // Walk up starting from the the start position until we find a non-SourceFile node that subsumes the selected span.
                 // This may fail (e.g. you select two statements in the root of a source file)
-                var start = ts.getParentNodeInSpan(ts.getTokenAtPosition(sourceFile, span.start), sourceFile, span);
+                var startToken = ts.getTokenAtPosition(sourceFile, span.start);
+                var start = cursorRequest ? getExtractableParent(startToken) : ts.getParentNodeInSpan(startToken, sourceFile, span);
                 // Do the same for the ending position
-                var end = ts.getParentNodeInSpan(ts.findTokenOnLeftOfPosition(sourceFile, ts.textSpanEnd(span)), sourceFile, span);
+                var endToken = ts.findTokenOnLeftOfPosition(sourceFile, ts.textSpanEnd(span));
+                var end = cursorRequest ? start : ts.getParentNodeInSpan(endToken, sourceFile, span);
                 var declarations = [];
                 // We'll modify these flags as we walk the tree to collect data
                 // about what things need to be done as part of the extraction.
@@ -135120,21 +142788,21 @@ var ts;
                 function checkForStaticContext(nodeToCheck, containingClass) {
                     var current = nodeToCheck;
                     while (current !== containingClass) {
-                        if (current.kind === 159 /* PropertyDeclaration */) {
-                            if (ts.hasModifier(current, 32 /* Static */)) {
+                        if (current.kind === 163 /* PropertyDeclaration */) {
+                            if (ts.hasSyntacticModifier(current, 32 /* Static */)) {
                                 rangeFacts |= RangeFacts.InStaticRegion;
                             }
                             break;
                         }
-                        else if (current.kind === 156 /* Parameter */) {
+                        else if (current.kind === 160 /* Parameter */) {
                             var ctorOrMethod = ts.getContainingFunction(current);
-                            if (ctorOrMethod.kind === 162 /* Constructor */) {
+                            if (ctorOrMethod.kind === 166 /* Constructor */) {
                                 rangeFacts |= RangeFacts.InStaticRegion;
                             }
                             break;
                         }
-                        else if (current.kind === 161 /* MethodDeclaration */) {
-                            if (ts.hasModifier(current, 32 /* Static */)) {
+                        else if (current.kind === 165 /* MethodDeclaration */) {
+                            if (ts.hasSyntacticModifier(current, 32 /* Static */)) {
                                 rangeFacts |= RangeFacts.InStaticRegion;
                             }
                         }
@@ -135176,8 +142844,8 @@ var ts;
                             return true;
                         }
                         if (ts.isDeclaration(node)) {
-                            var declaringNode = (node.kind === 242 /* VariableDeclaration */) ? node.parent.parent : node;
-                            if (ts.hasModifier(declaringNode, 1 /* Export */)) {
+                            var declaringNode = (node.kind === 249 /* VariableDeclaration */) ? node.parent.parent : node;
+                            if (ts.hasSyntacticModifier(declaringNode, 1 /* Export */)) {
                                 // TODO: GH#18217 Silly to use `errors ||` since it's definitely not defined (see top of `visit`)
                                 // Also, if we're only pushing one error, just use `let error: Diagnostic | undefined`!
                                 // Also TODO: GH#19956
@@ -135188,13 +142856,16 @@ var ts;
                         }
                         // Some things can't be extracted in certain situations
                         switch (node.kind) {
-                            case 254 /* ImportDeclaration */:
+                            case 261 /* ImportDeclaration */:
                                 (errors || (errors = [])).push(ts.createDiagnosticForNode(node, Messages.cannotExtractImport));
                                 return true;
-                            case 102 /* SuperKeyword */:
+                            case 266 /* ExportAssignment */:
+                                (errors || (errors = [])).push(ts.createDiagnosticForNode(node, Messages.cannotExtractExportedEntity));
+                                return true;
+                            case 105 /* SuperKeyword */:
                                 // For a super *constructor call*, we have to be extracting the entire class,
                                 // but a super *method call* simply implies a 'this' reference
-                                if (node.parent.kind === 196 /* CallExpression */) {
+                                if (node.parent.kind === 203 /* CallExpression */) {
                                     // Super constructor call
                                     var containingClass_1 = ts.getContainingClass(node); // TODO:GH#18217
                                     if (containingClass_1.pos < span.start || containingClass_1.end >= (span.start + span.length)) {
@@ -135206,37 +142877,53 @@ var ts;
                                     rangeFacts |= RangeFacts.UsesThis;
                                 }
                                 break;
-                        }
-                        if (ts.isFunctionLikeDeclaration(node) || ts.isClassLike(node)) {
-                            switch (node.kind) {
-                                case 244 /* FunctionDeclaration */:
-                                case 245 /* ClassDeclaration */:
-                                    if (ts.isSourceFile(node.parent) && node.parent.externalModuleIndicator === undefined) {
-                                        // You cannot extract global declarations
-                                        (errors || (errors = [])).push(ts.createDiagnosticForNode(node, Messages.functionWillNotBeVisibleInTheNewScope));
+                            case 209 /* ArrowFunction */:
+                                // check if arrow function uses this
+                                ts.forEachChild(node, function check(n) {
+                                    if (ts.isThis(n)) {
+                                        rangeFacts |= RangeFacts.UsesThis;
                                     }
-                                    break;
-                            }
-                            // do not dive into functions or classes
-                            return false;
+                                    else if (ts.isClassLike(n) || (ts.isFunctionLike(n) && !ts.isArrowFunction(n))) {
+                                        return false;
+                                    }
+                                    else {
+                                        ts.forEachChild(n, check);
+                                    }
+                                });
+                            // falls through
+                            case 252 /* ClassDeclaration */:
+                            case 251 /* FunctionDeclaration */:
+                                if (ts.isSourceFile(node.parent) && node.parent.externalModuleIndicator === undefined) {
+                                    // You cannot extract global declarations
+                                    (errors || (errors = [])).push(ts.createDiagnosticForNode(node, Messages.functionWillNotBeVisibleInTheNewScope));
+                                }
+                            // falls through
+                            case 221 /* ClassExpression */:
+                            case 208 /* FunctionExpression */:
+                            case 165 /* MethodDeclaration */:
+                            case 166 /* Constructor */:
+                            case 167 /* GetAccessor */:
+                            case 168 /* SetAccessor */:
+                                // do not dive into functions or classes
+                                return false;
                         }
                         var savedPermittedJumps = permittedJumps;
                         switch (node.kind) {
-                            case 227 /* IfStatement */:
+                            case 234 /* IfStatement */:
                                 permittedJumps = 0 /* None */;
                                 break;
-                            case 240 /* TryStatement */:
+                            case 247 /* TryStatement */:
                                 // forbid all jumps inside try blocks
                                 permittedJumps = 0 /* None */;
                                 break;
-                            case 223 /* Block */:
-                                if (node.parent && node.parent.kind === 240 /* TryStatement */ && node.parent.finallyBlock === node) {
+                            case 230 /* Block */:
+                                if (node.parent && node.parent.kind === 247 /* TryStatement */ && node.parent.finallyBlock === node) {
                                     // allow unconditional returns from finally blocks
                                     permittedJumps = 4 /* Return */;
                                 }
                                 break;
-                            case 278 /* DefaultClause */:
-                            case 277 /* CaseClause */:
+                            case 285 /* DefaultClause */:
+                            case 284 /* CaseClause */:
                                 // allow unlabeled break inside case clauses
                                 permittedJumps |= 1 /* Break */;
                                 break;
@@ -135248,19 +142935,19 @@ var ts;
                                 break;
                         }
                         switch (node.kind) {
-                            case 183 /* ThisType */:
-                            case 104 /* ThisKeyword */:
+                            case 187 /* ThisType */:
+                            case 107 /* ThisKeyword */:
                                 rangeFacts |= RangeFacts.UsesThis;
                                 break;
-                            case 238 /* LabeledStatement */: {
+                            case 245 /* LabeledStatement */: {
                                 var label = node.label;
                                 (seenLabels || (seenLabels = [])).push(label.escapedText);
                                 ts.forEachChild(node, visit);
                                 seenLabels.pop();
                                 break;
                             }
-                            case 234 /* BreakStatement */:
-                            case 233 /* ContinueStatement */: {
+                            case 241 /* BreakStatement */:
+                            case 240 /* ContinueStatement */: {
                                 var label = node.label;
                                 if (label) {
                                     if (!ts.contains(seenLabels, label.escapedText)) {
@@ -135269,20 +142956,20 @@ var ts;
                                     }
                                 }
                                 else {
-                                    if (!(permittedJumps & (node.kind === 234 /* BreakStatement */ ? 1 /* Break */ : 2 /* Continue */))) {
+                                    if (!(permittedJumps & (node.kind === 241 /* BreakStatement */ ? 1 /* Break */ : 2 /* Continue */))) {
                                         // attempt to break or continue in a forbidden context
                                         (errors || (errors = [])).push(ts.createDiagnosticForNode(node, Messages.cannotExtractRangeContainingConditionalBreakOrContinueStatements));
                                     }
                                 }
                                 break;
                             }
-                            case 206 /* AwaitExpression */:
+                            case 213 /* AwaitExpression */:
                                 rangeFacts |= RangeFacts.IsAsyncFunction;
                                 break;
-                            case 212 /* YieldExpression */:
+                            case 219 /* YieldExpression */:
                                 rangeFacts |= RangeFacts.IsGenerator;
                                 break;
-                            case 235 /* ReturnStatement */:
+                            case 242 /* ReturnStatement */:
                                 if (permittedJumps & 4 /* Return */) {
                                     rangeFacts |= RangeFacts.HasReturn;
                                 }
@@ -135336,7 +143023,7 @@ var ts;
                 while (true) {
                     current = current.parent;
                     // A function parameter's initializer is actually in the outer scope, not the function declaration
-                    if (current.kind === 156 /* Parameter */) {
+                    if (current.kind === 160 /* Parameter */) {
                         // Skip all the way to the outer scope of the function that declared this parameter
                         current = ts.findAncestor(current, function (parent) { return ts.isFunctionLikeDeclaration(parent); }).parent;
                     }
@@ -135347,7 +143034,7 @@ var ts;
                     //  * Module/namespace or source file
                     if (isScope(current)) {
                         scopes.push(current);
-                        if (current.kind === 290 /* SourceFile */) {
+                        if (current.kind === 297 /* SourceFile */) {
                             return scopes;
                         }
                     }
@@ -135437,32 +143124,32 @@ var ts;
             }
             function getDescriptionForFunctionLikeDeclaration(scope) {
                 switch (scope.kind) {
-                    case 162 /* Constructor */:
+                    case 166 /* Constructor */:
                         return "constructor";
-                    case 201 /* FunctionExpression */:
-                    case 244 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
+                    case 251 /* FunctionDeclaration */:
                         return scope.name
                             ? "function '" + scope.name.text + "'"
                             : ts.ANONYMOUS;
-                    case 202 /* ArrowFunction */:
+                    case 209 /* ArrowFunction */:
                         return "arrow function";
-                    case 161 /* MethodDeclaration */:
+                    case 165 /* MethodDeclaration */:
                         return "method '" + scope.name.getText() + "'";
-                    case 163 /* GetAccessor */:
+                    case 167 /* GetAccessor */:
                         return "'get " + scope.name.getText() + "'";
-                    case 164 /* SetAccessor */:
+                    case 168 /* SetAccessor */:
                         return "'set " + scope.name.getText() + "'";
                     default:
                         throw ts.Debug.assertNever(scope, "Unexpected scope kind " + scope.kind);
                 }
             }
             function getDescriptionForClassLikeDeclaration(scope) {
-                return scope.kind === 245 /* ClassDeclaration */
+                return scope.kind === 252 /* ClassDeclaration */
                     ? scope.name ? "class '" + scope.name.text + "'" : "anonymous class declaration"
                     : scope.name ? "class expression '" + scope.name.text + "'" : "anonymous class expression";
             }
             function getDescriptionForModuleLikeDeclaration(scope) {
-                return scope.kind === 250 /* ModuleBlock */
+                return scope.kind === 257 /* ModuleBlock */
                     ? "namespace '" + scope.parent.name.getText() + "'"
                     : scope.externalModuleIndicator ? 0 /* Module */ : 1 /* Global */;
             }
@@ -135484,7 +143171,7 @@ var ts;
                 var file = scope.getSourceFile();
                 var functionNameText = ts.getUniqueName(ts.isClassLike(scope) ? "newMethod" : "newFunction", file);
                 var isJS = ts.isInJSFile(scope);
-                var functionName = ts.createIdentifier(functionNameText);
+                var functionName = ts.factory.createIdentifier(functionNameText);
                 var returnType;
                 var parameters = [];
                 var callArguments = [];
@@ -135497,7 +143184,7 @@ var ts;
                         type = checker.getBaseTypeOfLiteralType(type);
                         typeNode = ts.codefix.typeToAutoImportableTypeNode(checker, importAdder, type, scope, scriptTarget, 1 /* NoTruncation */);
                     }
-                    var paramDecl = ts.createParameter(
+                    var paramDecl = ts.factory.createParameterDeclaration(
                     /*decorators*/ undefined, 
                     /*modifiers*/ undefined, 
                     /*dotDotDotToken*/ undefined, 
@@ -135507,7 +143194,7 @@ var ts;
                     if (usage.usage === 2 /* Write */) {
                         (writes || (writes = [])).push(usage);
                     }
-                    callArguments.push(ts.createIdentifier(name));
+                    callArguments.push(ts.factory.createIdentifier(name));
                 });
                 var typeParametersAndDeclarations = ts.arrayFrom(typeParameterUsages.values()).map(function (type) { return ({ type: type, declaration: getFirstDeclaration(type) }); });
                 var sortedTypeParametersAndDeclarations = typeParametersAndDeclarations.sort(compareTypesByDeclarationOrder);
@@ -135517,7 +143204,7 @@ var ts;
                 // Strictly speaking, we should check whether each name actually binds to the appropriate type
                 // parameter.  In cases of shadowing, they may not.
                 var callTypeArguments = typeParameters !== undefined
-                    ? typeParameters.map(function (decl) { return ts.createTypeReferenceNode(decl.name, /*typeArguments*/ undefined); })
+                    ? typeParameters.map(function (decl) { return ts.factory.createTypeReferenceNode(decl.name, /*typeArguments*/ undefined); })
                     : undefined;
                 // Provide explicit return types for contextually-typed functions
                 // to avoid problems when there are literal types present
@@ -135530,20 +143217,20 @@ var ts;
                 var newFunction;
                 if (ts.isClassLike(scope)) {
                     // always create private method in TypeScript files
-                    var modifiers = isJS ? [] : [ts.createToken(117 /* PrivateKeyword */)];
+                    var modifiers = isJS ? [] : [ts.factory.createModifier(120 /* PrivateKeyword */)];
                     if (range.facts & RangeFacts.InStaticRegion) {
-                        modifiers.push(ts.createToken(120 /* StaticKeyword */));
+                        modifiers.push(ts.factory.createModifier(123 /* StaticKeyword */));
                     }
                     if (range.facts & RangeFacts.IsAsyncFunction) {
-                        modifiers.push(ts.createToken(126 /* AsyncKeyword */));
+                        modifiers.push(ts.factory.createModifier(129 /* AsyncKeyword */));
                     }
-                    newFunction = ts.createMethod(
-                    /*decorators*/ undefined, modifiers.length ? modifiers : undefined, range.facts & RangeFacts.IsGenerator ? ts.createToken(41 /* AsteriskToken */) : undefined, functionName, 
+                    newFunction = ts.factory.createMethodDeclaration(
+                    /*decorators*/ undefined, modifiers.length ? modifiers : undefined, range.facts & RangeFacts.IsGenerator ? ts.factory.createToken(41 /* AsteriskToken */) : undefined, functionName, 
                     /*questionToken*/ undefined, typeParameters, parameters, returnType, body);
                 }
                 else {
-                    newFunction = ts.createFunctionDeclaration(
-                    /*decorators*/ undefined, range.facts & RangeFacts.IsAsyncFunction ? [ts.createToken(126 /* AsyncKeyword */)] : undefined, range.facts & RangeFacts.IsGenerator ? ts.createToken(41 /* AsteriskToken */) : undefined, functionName, typeParameters, parameters, returnType, body);
+                    newFunction = ts.factory.createFunctionDeclaration(
+                    /*decorators*/ undefined, range.facts & RangeFacts.IsAsyncFunction ? [ts.factory.createToken(129 /* AsyncKeyword */)] : undefined, range.facts & RangeFacts.IsGenerator ? ts.factory.createToken(41 /* AsteriskToken */) : undefined, functionName, typeParameters, parameters, returnType, body);
                 }
                 var changeTracker = ts.textChanges.ChangeTracker.fromContext(context);
                 var minInsertionPos = (isReadonlyArray(range.range) ? ts.last(range.range) : range.range).end;
@@ -135558,13 +143245,16 @@ var ts;
                 var newNodes = [];
                 // replace range with function call
                 var called = getCalledExpression(scope, range, functionNameText);
-                var call = ts.createCall(called, callTypeArguments, // Note that no attempt is made to take advantage of type argument inference
+                var call = ts.factory.createCallExpression(called, callTypeArguments, // Note that no attempt is made to take advantage of type argument inference
                 callArguments);
                 if (range.facts & RangeFacts.IsGenerator) {
-                    call = ts.createYield(ts.createToken(41 /* AsteriskToken */), call);
+                    call = ts.factory.createYieldExpression(ts.factory.createToken(41 /* AsteriskToken */), call);
                 }
                 if (range.facts & RangeFacts.IsAsyncFunction) {
-                    call = ts.createAwait(call);
+                    call = ts.factory.createAwaitExpression(call);
+                }
+                if (isInJSXContent(node)) {
+                    call = ts.factory.createJsxExpression(/*dotDotDotToken*/ undefined, call);
                 }
                 if (exposedVariableDeclarations.length && !writes) {
                     // No need to mix declarations and writes.
@@ -135574,8 +143264,8 @@ var ts;
                     if (exposedVariableDeclarations.length === 1) {
                         // Declaring exactly one variable: let x = newFunction();
                         var variableDeclaration = exposedVariableDeclarations[0];
-                        newNodes.push(ts.createVariableStatement(
-                        /*modifiers*/ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(ts.getSynthesizedDeepClone(variableDeclaration.name), /*type*/ ts.getSynthesizedDeepClone(variableDeclaration.type), /*initializer*/ call)], // TODO (acasey): test binding patterns
+                        newNodes.push(ts.factory.createVariableStatement(
+                        /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(ts.getSynthesizedDeepClone(variableDeclaration.name), /*exclamationToken*/ undefined, /*type*/ ts.getSynthesizedDeepClone(variableDeclaration.type), /*initializer*/ call)], // TODO (acasey): test binding patterns
                         variableDeclaration.parent.flags)));
                     }
                     else {
@@ -135587,27 +143277,27 @@ var ts;
                         var sawExplicitType = false;
                         for (var _i = 0, exposedVariableDeclarations_1 = exposedVariableDeclarations; _i < exposedVariableDeclarations_1.length; _i++) {
                             var variableDeclaration = exposedVariableDeclarations_1[_i];
-                            bindingElements.push(ts.createBindingElement(
+                            bindingElements.push(ts.factory.createBindingElement(
                             /*dotDotDotToken*/ undefined, 
                             /*propertyName*/ undefined, 
                             /*name*/ ts.getSynthesizedDeepClone(variableDeclaration.name)));
                             // Being returned through an object literal will have widened the type.
                             var variableType = checker.typeToTypeNode(checker.getBaseTypeOfLiteralType(checker.getTypeAtLocation(variableDeclaration)), scope, 1 /* NoTruncation */);
-                            typeElements.push(ts.createPropertySignature(
+                            typeElements.push(ts.factory.createPropertySignature(
                             /*modifiers*/ undefined, 
                             /*name*/ variableDeclaration.symbol.name, 
                             /*questionToken*/ undefined, 
-                            /*type*/ variableType, 
-                            /*initializer*/ undefined));
+                            /*type*/ variableType));
                             sawExplicitType = sawExplicitType || variableDeclaration.type !== undefined;
                             commonNodeFlags = commonNodeFlags & variableDeclaration.parent.flags;
                         }
-                        var typeLiteral = sawExplicitType ? ts.createTypeLiteralNode(typeElements) : undefined;
+                        var typeLiteral = sawExplicitType ? ts.factory.createTypeLiteralNode(typeElements) : undefined;
                         if (typeLiteral) {
                             ts.setEmitFlags(typeLiteral, 1 /* SingleLine */);
                         }
-                        newNodes.push(ts.createVariableStatement(
-                        /*modifiers*/ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(ts.createObjectBindingPattern(bindingElements), 
+                        newNodes.push(ts.factory.createVariableStatement(
+                        /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(ts.factory.createObjectBindingPattern(bindingElements), 
+                            /*exclamationToken*/ undefined, 
                             /*type*/ typeLiteral, 
                             /*initializer*/ call)], commonNodeFlags)));
                     }
@@ -135621,45 +143311,45 @@ var ts;
                             if (flags & 2 /* Const */) {
                                 flags = (flags & ~2 /* Const */) | 1 /* Let */;
                             }
-                            newNodes.push(ts.createVariableStatement(
-                            /*modifiers*/ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(variableDeclaration.symbol.name, getTypeDeepCloneUnionUndefined(variableDeclaration.type))], flags)));
+                            newNodes.push(ts.factory.createVariableStatement(
+                            /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(variableDeclaration.symbol.name, /*exclamationToken*/ undefined, getTypeDeepCloneUnionUndefined(variableDeclaration.type))], flags)));
                         }
                     }
                     if (returnValueProperty) {
                         // has both writes and return, need to create variable declaration to hold return value;
-                        newNodes.push(ts.createVariableStatement(
-                        /*modifiers*/ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(returnValueProperty, getTypeDeepCloneUnionUndefined(returnType))], 1 /* Let */)));
+                        newNodes.push(ts.factory.createVariableStatement(
+                        /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(returnValueProperty, /*exclamationToken*/ undefined, getTypeDeepCloneUnionUndefined(returnType))], 1 /* Let */)));
                     }
                     var assignments = getPropertyAssignmentsForWritesAndVariableDeclarations(exposedVariableDeclarations, writes);
                     if (returnValueProperty) {
-                        assignments.unshift(ts.createShorthandPropertyAssignment(returnValueProperty));
+                        assignments.unshift(ts.factory.createShorthandPropertyAssignment(returnValueProperty));
                     }
                     // propagate writes back
                     if (assignments.length === 1) {
                         // We would only have introduced a return value property if there had been
                         // other assignments to make.
                         ts.Debug.assert(!returnValueProperty, "Shouldn't have returnValueProperty here");
-                        newNodes.push(ts.createStatement(ts.createAssignment(assignments[0].name, call)));
+                        newNodes.push(ts.factory.createExpressionStatement(ts.factory.createAssignment(assignments[0].name, call)));
                         if (range.facts & RangeFacts.HasReturn) {
-                            newNodes.push(ts.createReturn());
+                            newNodes.push(ts.factory.createReturnStatement());
                         }
                     }
                     else {
                         // emit e.g.
                         //   { a, b, __return } = newFunction(a, b);
                         //   return __return;
-                        newNodes.push(ts.createStatement(ts.createAssignment(ts.createObjectLiteral(assignments), call)));
+                        newNodes.push(ts.factory.createExpressionStatement(ts.factory.createAssignment(ts.factory.createObjectLiteralExpression(assignments), call)));
                         if (returnValueProperty) {
-                            newNodes.push(ts.createReturn(ts.createIdentifier(returnValueProperty)));
+                            newNodes.push(ts.factory.createReturnStatement(ts.factory.createIdentifier(returnValueProperty)));
                         }
                     }
                 }
                 else {
                     if (range.facts & RangeFacts.HasReturn) {
-                        newNodes.push(ts.createReturn(call));
+                        newNodes.push(ts.factory.createReturnStatement(call));
                     }
                     else if (isReadonlyArray(range.range)) {
-                        newNodes.push(ts.createStatement(call));
+                        newNodes.push(ts.factory.createExpressionStatement(call));
                     }
                     else {
                         newNodes.push(call);
@@ -135685,9 +143375,9 @@ var ts;
                     while (ts.isParenthesizedTypeNode(withoutParens)) {
                         withoutParens = withoutParens.type;
                     }
-                    return ts.isUnionTypeNode(withoutParens) && ts.find(withoutParens.types, function (t) { return t.kind === 146 /* UndefinedKeyword */; })
+                    return ts.isUnionTypeNode(withoutParens) && ts.find(withoutParens.types, function (t) { return t.kind === 150 /* UndefinedKeyword */; })
                         ? clone
-                        : ts.createUnionTypeNode([clone, ts.createKeywordTypeNode(146 /* UndefinedKeyword */)]);
+                        : ts.factory.createUnionTypeNode([clone, ts.factory.createKeywordTypeNode(150 /* UndefinedKeyword */)]);
                 }
             }
             /**
@@ -135712,17 +143402,20 @@ var ts;
                 if (ts.isClassLike(scope)) {
                     ts.Debug.assert(!isJS, "Cannot extract to a JS class"); // See CannotExtractToJSClass
                     var modifiers = [];
-                    modifiers.push(ts.createToken(117 /* PrivateKeyword */));
+                    modifiers.push(ts.factory.createModifier(120 /* PrivateKeyword */));
                     if (rangeFacts & RangeFacts.InStaticRegion) {
-                        modifiers.push(ts.createToken(120 /* StaticKeyword */));
+                        modifiers.push(ts.factory.createModifier(123 /* StaticKeyword */));
                     }
-                    modifiers.push(ts.createToken(138 /* ReadonlyKeyword */));
-                    var newVariable = ts.createProperty(
+                    modifiers.push(ts.factory.createModifier(142 /* ReadonlyKeyword */));
+                    var newVariable = ts.factory.createPropertyDeclaration(
                     /*decorators*/ undefined, modifiers, localNameText, 
                     /*questionToken*/ undefined, variableType, initializer);
-                    var localReference = ts.createPropertyAccess(rangeFacts & RangeFacts.InStaticRegion
-                        ? ts.createIdentifier(scope.name.getText()) // TODO: GH#18217
-                        : ts.createThis(), ts.createIdentifier(localNameText));
+                    var localReference = ts.factory.createPropertyAccessExpression(rangeFacts & RangeFacts.InStaticRegion
+                        ? ts.factory.createIdentifier(scope.name.getText()) // TODO: GH#18217
+                        : ts.factory.createThis(), ts.factory.createIdentifier(localNameText));
+                    if (isInJSXContent(node)) {
+                        localReference = ts.factory.createJsxExpression(/*dotDotDotToken*/ undefined, localReference);
+                    }
                     // Declare
                     var maxInsertionPos = node.pos;
                     var nodeToInsertBefore = getNodeToInsertPropertyBefore(maxInsertionPos, scope);
@@ -135731,7 +143424,7 @@ var ts;
                     changeTracker.replaceNode(context.file, node, localReference);
                 }
                 else {
-                    var newVariableDeclaration = ts.createVariableDeclaration(localNameText, variableType, initializer);
+                    var newVariableDeclaration = ts.factory.createVariableDeclaration(localNameText, /*exclamationToken*/ undefined, variableType, initializer);
                     // If the node is part of an initializer in a list of variable declarations, insert a new
                     // variable declaration into the list (in case it depends on earlier ones).
                     // CONSIDER: If the declaration list isn't const, we might want to split it into multiple
@@ -135742,19 +143435,19 @@ var ts;
                         // CONSIDER: could detect that each is on a separate line (See `extractConstant_VariableList_MultipleLines` in `extractConstants.ts`)
                         changeTracker.insertNodeBefore(context.file, oldVariableDeclaration, newVariableDeclaration);
                         // Consume
-                        var localReference = ts.createIdentifier(localNameText);
+                        var localReference = ts.factory.createIdentifier(localNameText);
                         changeTracker.replaceNode(context.file, node, localReference);
                     }
-                    else if (node.parent.kind === 226 /* ExpressionStatement */ && scope === ts.findAncestor(node, isScope)) {
+                    else if (node.parent.kind === 233 /* ExpressionStatement */ && scope === ts.findAncestor(node, isScope)) {
                         // If the parent is an expression statement and the target scope is the immediately enclosing one,
                         // replace the statement with the declaration.
-                        var newVariableStatement = ts.createVariableStatement(
-                        /*modifiers*/ undefined, ts.createVariableDeclarationList([newVariableDeclaration], 2 /* Const */));
+                        var newVariableStatement = ts.factory.createVariableStatement(
+                        /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([newVariableDeclaration], 2 /* Const */));
                         changeTracker.replaceNode(context.file, node.parent, newVariableStatement);
                     }
                     else {
-                        var newVariableStatement = ts.createVariableStatement(
-                        /*modifiers*/ undefined, ts.createVariableDeclarationList([newVariableDeclaration], 2 /* Const */));
+                        var newVariableStatement = ts.factory.createVariableStatement(
+                        /*modifiers*/ undefined, ts.factory.createVariableDeclarationList([newVariableDeclaration], 2 /* Const */));
                         // Declare
                         var nodeToInsertBefore = getNodeToInsertConstantBefore(node, scope);
                         if (nodeToInsertBefore.pos === 0) {
@@ -135764,12 +143457,17 @@ var ts;
                             changeTracker.insertNodeBefore(context.file, nodeToInsertBefore, newVariableStatement, /*blankLineBetween*/ false);
                         }
                         // Consume
-                        if (node.parent.kind === 226 /* ExpressionStatement */) {
+                        if (node.parent.kind === 233 /* ExpressionStatement */) {
                             // If the parent is an expression statement, delete it.
                             changeTracker.delete(context.file, node.parent);
                         }
                         else {
-                            var localReference = ts.createIdentifier(localNameText);
+                            var localReference = ts.factory.createIdentifier(localNameText);
+                            // When extract to a new variable in JSX content, need to wrap a {} out of the new variable
+                            // or it will become a plain text
+                            if (isInJSXContent(node)) {
+                                localReference = ts.factory.createJsxExpression(/*dotDotDotToken*/ undefined, localReference);
+                            }
                             changeTracker.replaceNode(context.file, node, localReference);
                         }
                     }
@@ -135805,7 +143503,7 @@ var ts;
                             var paramType = checker.getTypeAtLocation(p);
                             if (paramType === checker.getAnyType())
                                 hasAny = true;
-                            parameters.push(ts.updateParameter(p, p.decorators, p.modifiers, p.dotDotDotToken, p.name, p.questionToken, p.type || checker.typeToTypeNode(paramType, scope, 1 /* NoTruncation */), p.initializer));
+                            parameters.push(ts.factory.updateParameterDeclaration(p, p.decorators, p.modifiers, p.dotDotDotToken, p.name, p.questionToken, p.type || checker.typeToTypeNode(paramType, scope, 1 /* NoTruncation */), p.initializer));
                         }
                     }
                     // If a parameter was inferred as any we skip adding function parameters at all.
@@ -135815,7 +143513,7 @@ var ts;
                         return { variableType: variableType, initializer: initializer };
                     variableType = undefined;
                     if (ts.isArrowFunction(initializer)) {
-                        initializer = ts.updateArrowFunction(initializer, node.modifiers, initializer.typeParameters, parameters, initializer.type || checker.typeToTypeNode(functionSignature.getReturnType(), scope, 1 /* NoTruncation */), initializer.equalsGreaterThanToken, initializer.body);
+                        initializer = ts.factory.updateArrowFunction(initializer, node.modifiers, initializer.typeParameters, parameters, initializer.type || checker.typeToTypeNode(functionSignature.getReturnType(), scope, 1 /* NoTruncation */), initializer.equalsGreaterThanToken, initializer.body);
                     }
                     else {
                         if (functionSignature && !!functionSignature.thisParameter) {
@@ -135824,14 +143522,14 @@ var ts;
                             // Note: If this parameter was already there, it would have been previously updated with the type if not type was present
                             if ((!firstParameter || (ts.isIdentifier(firstParameter.name) && firstParameter.name.escapedText !== "this"))) {
                                 var thisType = checker.getTypeOfSymbolAtLocation(functionSignature.thisParameter, node);
-                                parameters.splice(0, 0, ts.createParameter(
+                                parameters.splice(0, 0, ts.factory.createParameterDeclaration(
                                 /* decorators */ undefined, 
                                 /* modifiers */ undefined, 
                                 /* dotDotDotToken */ undefined, "this", 
                                 /* questionToken */ undefined, checker.typeToTypeNode(thisType, scope, 1 /* NoTruncation */)));
                             }
                         }
-                        initializer = ts.updateFunctionExpression(initializer, node.modifiers, initializer.asteriskToken, initializer.name, initializer.typeParameters, parameters, initializer.type || checker.typeToTypeNode(functionSignature.getReturnType(), scope, 1 /* NoTruncation */), initializer.body);
+                        initializer = ts.factory.updateFunctionExpression(initializer, node.modifiers, initializer.asteriskToken, initializer.name, initializer.typeParameters, parameters, initializer.type || checker.typeToTypeNode(functionSignature.getReturnType(), scope, 1 /* NoTruncation */), initializer.body);
                     }
                     return { variableType: variableType, initializer: initializer };
                 }
@@ -135870,10 +143568,10 @@ var ts;
                     || ts.compareValues(type1.id, type2.id);
             }
             function getCalledExpression(scope, range, functionNameText) {
-                var functionReference = ts.createIdentifier(functionNameText);
+                var functionReference = ts.factory.createIdentifier(functionNameText);
                 if (ts.isClassLike(scope)) {
-                    var lhs = range.facts & RangeFacts.InStaticRegion ? ts.createIdentifier(scope.name.text) : ts.createThis(); // TODO: GH#18217
-                    return ts.createPropertyAccess(lhs, functionReference);
+                    var lhs = range.facts & RangeFacts.InStaticRegion ? ts.factory.createIdentifier(scope.name.text) : ts.factory.createThis(); // TODO: GH#18217
+                    return ts.factory.createPropertyAccessExpression(lhs, functionReference);
                 }
                 else {
                     return functionReference;
@@ -135883,11 +143581,11 @@ var ts;
                 var hasWritesOrVariableDeclarations = writes !== undefined || exposedVariableDeclarations.length > 0;
                 if (ts.isBlock(body) && !hasWritesOrVariableDeclarations && substitutions.size === 0) {
                     // already block, no declarations or writes to propagate back, no substitutions - can use node as is
-                    return { body: ts.createBlock(body.statements, /*multLine*/ true), returnValueProperty: undefined };
+                    return { body: ts.factory.createBlock(body.statements, /*multLine*/ true), returnValueProperty: undefined };
                 }
                 var returnValueProperty;
                 var ignoreReturns = false;
-                var statements = ts.createNodeArray(ts.isBlock(body) ? body.statements.slice(0) : [ts.isStatement(body) ? body : ts.createReturn(body)]);
+                var statements = ts.factory.createNodeArray(ts.isBlock(body) ? body.statements.slice(0) : [ts.isStatement(body) ? body : ts.factory.createReturnStatement(body)]);
                 // rewrite body if either there are writes that should be propagated back via return statements or there are substitutions
                 if (hasWritesOrVariableDeclarations || substitutions.size) {
                     var rewrittenStatements = ts.visitNodes(statements, visitor).slice();
@@ -135896,31 +143594,31 @@ var ts;
                         // it is ok to know that range has at least one return since it we only allow unconditional returns
                         var assignments = getPropertyAssignmentsForWritesAndVariableDeclarations(exposedVariableDeclarations, writes);
                         if (assignments.length === 1) {
-                            rewrittenStatements.push(ts.createReturn(assignments[0].name));
+                            rewrittenStatements.push(ts.factory.createReturnStatement(assignments[0].name));
                         }
                         else {
-                            rewrittenStatements.push(ts.createReturn(ts.createObjectLiteral(assignments)));
+                            rewrittenStatements.push(ts.factory.createReturnStatement(ts.factory.createObjectLiteralExpression(assignments)));
                         }
                     }
-                    return { body: ts.createBlock(rewrittenStatements, /*multiLine*/ true), returnValueProperty: returnValueProperty };
+                    return { body: ts.factory.createBlock(rewrittenStatements, /*multiLine*/ true), returnValueProperty: returnValueProperty };
                 }
                 else {
-                    return { body: ts.createBlock(statements, /*multiLine*/ true), returnValueProperty: undefined };
+                    return { body: ts.factory.createBlock(statements, /*multiLine*/ true), returnValueProperty: undefined };
                 }
                 function visitor(node) {
-                    if (!ignoreReturns && node.kind === 235 /* ReturnStatement */ && hasWritesOrVariableDeclarations) {
+                    if (!ignoreReturns && ts.isReturnStatement(node) && hasWritesOrVariableDeclarations) {
                         var assignments = getPropertyAssignmentsForWritesAndVariableDeclarations(exposedVariableDeclarations, writes);
                         if (node.expression) {
                             if (!returnValueProperty) {
                                 returnValueProperty = "__return";
                             }
-                            assignments.unshift(ts.createPropertyAssignment(returnValueProperty, ts.visitNode(node.expression, visitor)));
+                            assignments.unshift(ts.factory.createPropertyAssignment(returnValueProperty, ts.visitNode(node.expression, visitor)));
                         }
                         if (assignments.length === 1) {
-                            return ts.createReturn(assignments[0].name);
+                            return ts.factory.createReturnStatement(assignments[0].name);
                         }
                         else {
-                            return ts.createReturn(ts.createObjectLiteral(assignments));
+                            return ts.factory.createReturnStatement(ts.factory.createObjectLiteralExpression(assignments));
                         }
                     }
                     else {
@@ -136023,8 +143721,8 @@ var ts;
                 }
             }
             function getPropertyAssignmentsForWritesAndVariableDeclarations(exposedVariableDeclarations, writes) {
-                var variableAssignments = ts.map(exposedVariableDeclarations, function (v) { return ts.createShorthandPropertyAssignment(v.symbol.name); });
-                var writeAssignments = ts.map(writes, function (w) { return ts.createShorthandPropertyAssignment(w.symbol.name); });
+                var variableAssignments = ts.map(exposedVariableDeclarations, function (v) { return ts.factory.createShorthandPropertyAssignment(v.symbol.name); });
+                var writeAssignments = ts.map(writes, function (w) { return ts.factory.createShorthandPropertyAssignment(w.symbol.name); });
                 // TODO: GH#18217 `variableAssignments` not possibly undefined!
                 return variableAssignments === undefined
                     ? writeAssignments
@@ -136057,13 +143755,13 @@ var ts;
                 Usage[Usage["Write"] = 2] = "Write";
             })(Usage || (Usage = {}));
             function collectReadsAndWrites(targetRange, scopes, enclosingTextRange, sourceFile, checker, cancellationToken) {
-                var allTypeParameterUsages = ts.createMap(); // Key is type ID
+                var allTypeParameterUsages = new ts.Map(); // Key is type ID
                 var usagesPerScope = [];
                 var substitutionsPerScope = [];
                 var functionErrorsPerScope = [];
                 var constantErrorsPerScope = [];
                 var visibleDeclarationsInExtractedRange = [];
-                var exposedVariableSymbolSet = ts.createMap(); // Key is symbol ID
+                var exposedVariableSymbolSet = new ts.Map(); // Key is symbol ID
                 var exposedVariableDeclarations = [];
                 var firstExposedNonVariableDeclaration;
                 var expression = !isReadonlyArray(targetRange.range)
@@ -136084,9 +143782,9 @@ var ts;
                 // initialize results
                 for (var _i = 0, scopes_1 = scopes; _i < scopes_1.length; _i++) {
                     var scope = scopes_1[_i];
-                    usagesPerScope.push({ usages: ts.createMap(), typeParameterUsages: ts.createMap(), substitutions: ts.createMap() });
-                    substitutionsPerScope.push(ts.createMap());
-                    functionErrorsPerScope.push(ts.isFunctionLikeDeclaration(scope) && scope.kind !== 244 /* FunctionDeclaration */
+                    usagesPerScope.push({ usages: new ts.Map(), typeParameterUsages: new ts.Map(), substitutions: new ts.Map() });
+                    substitutionsPerScope.push(new ts.Map());
+                    functionErrorsPerScope.push(ts.isFunctionLikeDeclaration(scope) && scope.kind !== 251 /* FunctionDeclaration */
                         ? [ts.createDiagnosticForNode(scope, Messages.cannotExtractToOtherFunctionLike)]
                         : []);
                     var constantErrors = [];
@@ -136102,8 +143800,8 @@ var ts;
                     }
                     constantErrorsPerScope.push(constantErrors);
                 }
-                var seenUsages = ts.createMap();
-                var target = isReadonlyArray(targetRange.range) ? ts.createBlock(targetRange.range) : targetRange.range;
+                var seenUsages = new ts.Map();
+                var target = isReadonlyArray(targetRange.range) ? ts.factory.createBlock(targetRange.range) : targetRange.range;
                 var unmodifiedNode = isReadonlyArray(targetRange.range) ? ts.first(targetRange.range) : targetRange.range;
                 var inGenericContext = isInGenericContext(unmodifiedNode);
                 collectUsages(target);
@@ -136115,15 +143813,15 @@ var ts;
                     recordTypeParameterUsages(contextualType);
                 }
                 if (allTypeParameterUsages.size > 0) {
-                    var seenTypeParameterUsages = ts.createMap(); // Key is type ID
-                    var i_1 = 0;
-                    for (var curr = unmodifiedNode; curr !== undefined && i_1 < scopes.length; curr = curr.parent) {
-                        if (curr === scopes[i_1]) {
+                    var seenTypeParameterUsages = new ts.Map(); // Key is type ID
+                    var i_2 = 0;
+                    for (var curr = unmodifiedNode; curr !== undefined && i_2 < scopes.length; curr = curr.parent) {
+                        if (curr === scopes[i_2]) {
                             // Copy current contents of seenTypeParameterUsages into scope.
                             seenTypeParameterUsages.forEach(function (typeParameter, id) {
-                                usagesPerScope[i_1].typeParameterUsages.set(id, typeParameter);
+                                usagesPerScope[i_2].typeParameterUsages.set(id, typeParameter);
                             });
-                            i_1++;
+                            i_2++;
                         }
                         // Note that we add the current node's type parameters *after* updating the corresponding scope.
                         if (ts.isDeclarationWithTypeParameters(curr)) {
@@ -136139,7 +143837,7 @@ var ts;
                     // If we didn't get through all the scopes, then there were some that weren't in our
                     // parent chain (impossible at time of writing).  A conservative solution would be to
                     // copy allTypeParameterUsages into all remaining scopes.
-                    ts.Debug.assert(i_1 === scopes.length, "Should have iterated all scopes");
+                    ts.Debug.assert(i_2 === scopes.length, "Should have iterated all scopes");
                 }
                 // If there are any declarations in the extracted block that are used in the same enclosing
                 // lexical scope, we can't move the extraction "up" as those declarations will become unreachable
@@ -136165,7 +143863,7 @@ var ts;
                             hasWrite = true;
                             if (value.symbol.flags & 106500 /* ClassMember */ &&
                                 value.symbol.valueDeclaration &&
-                                ts.hasModifier(value.symbol.valueDeclaration, 64 /* Readonly */)) {
+                                ts.hasEffectiveModifier(value.symbol.valueDeclaration, 64 /* Readonly */)) {
                                 readonlyClassPropertyWrite = value.symbol.valueDeclaration;
                             }
                         }
@@ -136381,17 +144079,20 @@ var ts;
                     }
                     var decls = symbol.getDeclarations();
                     if (decls && decls.some(function (d) { return d.parent === scopeDecl; })) {
-                        return ts.createIdentifier(symbol.name);
+                        return ts.factory.createIdentifier(symbol.name);
                     }
                     var prefix = tryReplaceWithQualifiedNameOrPropertyAccess(symbol.parent, scopeDecl, isTypeNode);
                     if (prefix === undefined) {
                         return undefined;
                     }
                     return isTypeNode
-                        ? ts.createQualifiedName(prefix, ts.createIdentifier(symbol.name))
-                        : ts.createPropertyAccess(prefix, symbol.name);
+                        ? ts.factory.createQualifiedName(prefix, ts.factory.createIdentifier(symbol.name))
+                        : ts.factory.createPropertyAccessExpression(prefix, symbol.name);
                 }
             }
+            function getExtractableParent(node) {
+                return ts.findAncestor(node, function (node) { return node.parent && isExtractableExpression(node) && !ts.isBinaryExpression(node.parent); });
+            }
             /**
              * Computes whether or not a node represents an expression in a position where it could
              * be extracted.
@@ -136402,35 +144103,38 @@ var ts;
             function isExtractableExpression(node) {
                 var parent = node.parent;
                 switch (parent.kind) {
-                    case 284 /* EnumMember */:
+                    case 291 /* EnumMember */:
                         return false;
                 }
                 switch (node.kind) {
                     case 10 /* StringLiteral */:
-                        return parent.kind !== 254 /* ImportDeclaration */ &&
-                            parent.kind !== 258 /* ImportSpecifier */;
-                    case 213 /* SpreadElement */:
-                    case 189 /* ObjectBindingPattern */:
-                    case 191 /* BindingElement */:
+                        return parent.kind !== 261 /* ImportDeclaration */ &&
+                            parent.kind !== 265 /* ImportSpecifier */;
+                    case 220 /* SpreadElement */:
+                    case 196 /* ObjectBindingPattern */:
+                    case 198 /* BindingElement */:
                         return false;
-                    case 75 /* Identifier */:
-                        return parent.kind !== 191 /* BindingElement */ &&
-                            parent.kind !== 258 /* ImportSpecifier */ &&
-                            parent.kind !== 263 /* ExportSpecifier */;
+                    case 78 /* Identifier */:
+                        return parent.kind !== 198 /* BindingElement */ &&
+                            parent.kind !== 265 /* ImportSpecifier */ &&
+                            parent.kind !== 270 /* ExportSpecifier */;
                 }
                 return true;
             }
             function isBlockLike(node) {
                 switch (node.kind) {
-                    case 223 /* Block */:
-                    case 290 /* SourceFile */:
-                    case 250 /* ModuleBlock */:
-                    case 277 /* CaseClause */:
+                    case 230 /* Block */:
+                    case 297 /* SourceFile */:
+                    case 257 /* ModuleBlock */:
+                    case 284 /* CaseClause */:
                         return true;
                     default:
                         return false;
                 }
             }
+            function isInJSXContent(node) {
+                return (ts.isJsxElement(node) || ts.isJsxSelfClosingElement(node) || ts.isJsxFragment(node)) && ts.isJsxElement(node.parent);
+            }
         })(extractSymbol = refactor.extractSymbol || (refactor.extractSymbol = {}));
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
@@ -136445,24 +144149,39 @@ var ts;
         var extractToTypeDef = "Extract to typedef";
         refactor.registerRefactor(refactorName, {
             getAvailableActions: function (context) {
-                var info = getRangeToExtract(context);
+                var info = getRangeToExtract(context, context.triggerReason === "invoked");
                 if (!info)
                     return ts.emptyArray;
-                return [{
-                        name: refactorName,
-                        description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_type),
-                        actions: info.isJS ? [{
-                                name: extractToTypeDef, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_typedef)
-                            }] : ts.append([{
-                                name: extractToTypeAlias, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_type_alias)
-                            }], info.typeElements && {
-                            name: extractToInterface, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_interface)
-                        })
-                    }];
+                if (info.error === undefined) {
+                    return [{
+                            name: refactorName,
+                            description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_type),
+                            actions: info.info.isJS ? [{
+                                    name: extractToTypeDef, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_typedef)
+                                }] : ts.append([{
+                                    name: extractToTypeAlias, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_type_alias)
+                                }], info.info.typeElements && {
+                                name: extractToInterface, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_interface)
+                            })
+                        }];
+                }
+                if (context.preferences.provideRefactorNotApplicableReason) {
+                    return [{
+                            name: refactorName,
+                            description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_type),
+                            actions: [
+                                { name: extractToTypeDef, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_typedef), notApplicableReason: info.error },
+                                { name: extractToTypeAlias, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_type_alias), notApplicableReason: info.error },
+                                { name: extractToInterface, description: ts.getLocaleSpecificMessage(ts.Diagnostics.Extract_to_interface), notApplicableReason: info.error },
+                            ]
+                        }];
+                }
+                return ts.emptyArray;
             },
             getEditsForAction: function (context, actionName) {
+                var _a;
                 var file = context.file;
-                var info = ts.Debug.checkDefined(getRangeToExtract(context), "Expected to find a range to extract");
+                var info = ts.Debug.checkDefined((_a = getRangeToExtract(context)) === null || _a === void 0 ? void 0 : _a.info, "Expected to find a range to extract");
                 var name = ts.getUniqueName("NewType", file);
                 var edits = ts.textChanges.ChangeTracker.with(context, function (changes) {
                     switch (actionName) {
@@ -136484,28 +144203,31 @@ var ts;
                 return { edits: edits, renameFilename: renameFilename, renameLocation: renameLocation };
             }
         });
-        function getRangeToExtract(context) {
+        function getRangeToExtract(context, considerEmptySpans) {
+            if (considerEmptySpans === void 0) { considerEmptySpans = true; }
             var file = context.file, startPosition = context.startPosition;
             var isJS = ts.isSourceFileJS(file);
             var current = ts.getTokenAtPosition(file, startPosition);
             var range = ts.createTextRangeFromSpan(ts.getRefactorContextSpan(context));
-            var selection = ts.findAncestor(current, (function (node) { return node.parent && rangeContainsSkipTrivia(range, node, file) && !rangeContainsSkipTrivia(range, node.parent, file); }));
+            var cursorRequest = range.pos === range.end && considerEmptySpans;
+            var selection = ts.findAncestor(current, (function (node) { return node.parent && ts.isTypeNode(node) && !rangeContainsSkipTrivia(range, node.parent, file) &&
+                (cursorRequest || ts.nodeOverlapsWithStartEnd(current, file, range.pos, range.end)); }));
             if (!selection || !ts.isTypeNode(selection))
-                return undefined;
+                return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.Selection_is_not_a_valid_type_node) };
             var checker = context.program.getTypeChecker();
             var firstStatement = ts.Debug.checkDefined(ts.findAncestor(selection, ts.isStatement), "Should find a statement");
             var typeParameters = collectTypeParameters(checker, selection, firstStatement, file);
             if (!typeParameters)
-                return undefined;
+                return { error: ts.getLocaleSpecificMessage(ts.Diagnostics.No_type_could_be_extracted_from_this_type_node) };
             var typeElements = flattenTypeLiteralNodeReference(checker, selection);
-            return { isJS: isJS, selection: selection, firstStatement: firstStatement, typeParameters: typeParameters, typeElements: typeElements };
+            return { info: { isJS: isJS, selection: selection, firstStatement: firstStatement, typeParameters: typeParameters, typeElements: typeElements } };
         }
         function flattenTypeLiteralNodeReference(checker, node) {
             if (!node)
                 return undefined;
             if (ts.isIntersectionTypeNode(node)) {
                 var result = [];
-                var seen_1 = ts.createMap();
+                var seen_1 = new ts.Map();
                 for (var _i = 0, _a = node.types; _i < _a.length; _i++) {
                     var type = _a[_i];
                     var flattenedTypeMembers = flattenTypeLiteralNodeReference(checker, type);
@@ -136537,7 +144259,7 @@ var ts;
                         if (symbol) {
                             var declaration = ts.cast(ts.first(symbol.declarations), ts.isTypeParameterDeclaration);
                             if (rangeContainsSkipTrivia(statement, declaration, file) && !rangeContainsSkipTrivia(selection, declaration, file)) {
-                                result.push(declaration);
+                                ts.pushIfUnique(result, declaration);
                             }
                         }
                     }
@@ -136567,46 +144289,43 @@ var ts;
                         }
                     }
                 }
+                if (file && ts.isTupleTypeNode(node) && (ts.getLineAndCharacterOfPosition(file, node.pos).line === ts.getLineAndCharacterOfPosition(file, node.end).line)) {
+                    ts.setEmitFlags(node, 1 /* SingleLine */);
+                }
                 return ts.forEachChild(node, visitor);
             }
         }
         function doTypeAliasChange(changes, file, name, info) {
             var firstStatement = info.firstStatement, selection = info.selection, typeParameters = info.typeParameters;
-            var newTypeNode = ts.createTypeAliasDeclaration(
+            var newTypeNode = ts.factory.createTypeAliasDeclaration(
             /* decorators */ undefined, 
-            /* modifiers */ undefined, name, typeParameters.map(function (id) { return ts.updateTypeParameterDeclaration(id, id.name, id.constraint, /* defaultType */ undefined); }), selection);
+            /* modifiers */ undefined, name, typeParameters.map(function (id) { return ts.factory.updateTypeParameterDeclaration(id, id.name, id.constraint, /* defaultType */ undefined); }), selection);
             changes.insertNodeBefore(file, firstStatement, ts.ignoreSourceNewlines(newTypeNode), /* blankLineBetween */ true);
-            changes.replaceNode(file, selection, ts.createTypeReferenceNode(name, typeParameters.map(function (id) { return ts.createTypeReferenceNode(id.name, /* typeArguments */ undefined); })));
+            changes.replaceNode(file, selection, ts.factory.createTypeReferenceNode(name, typeParameters.map(function (id) { return ts.factory.createTypeReferenceNode(id.name, /* typeArguments */ undefined); })), { leadingTriviaOption: ts.textChanges.LeadingTriviaOption.Exclude, trailingTriviaOption: ts.textChanges.TrailingTriviaOption.ExcludeWhitespace });
         }
         function doInterfaceChange(changes, file, name, info) {
+            var _a;
             var firstStatement = info.firstStatement, selection = info.selection, typeParameters = info.typeParameters, typeElements = info.typeElements;
-            var newTypeNode = ts.createInterfaceDeclaration(
+            var newTypeNode = ts.factory.createInterfaceDeclaration(
             /* decorators */ undefined, 
             /* modifiers */ undefined, name, typeParameters, 
             /* heritageClauses */ undefined, typeElements);
+            ts.setTextRange(newTypeNode, (_a = typeElements[0]) === null || _a === void 0 ? void 0 : _a.parent);
             changes.insertNodeBefore(file, firstStatement, ts.ignoreSourceNewlines(newTypeNode), /* blankLineBetween */ true);
-            changes.replaceNode(file, selection, ts.createTypeReferenceNode(name, typeParameters.map(function (id) { return ts.createTypeReferenceNode(id.name, /* typeArguments */ undefined); })));
+            changes.replaceNode(file, selection, ts.factory.createTypeReferenceNode(name, typeParameters.map(function (id) { return ts.factory.createTypeReferenceNode(id.name, /* typeArguments */ undefined); })), { leadingTriviaOption: ts.textChanges.LeadingTriviaOption.Exclude, trailingTriviaOption: ts.textChanges.TrailingTriviaOption.ExcludeWhitespace });
         }
         function doTypedefChange(changes, file, name, info) {
             var firstStatement = info.firstStatement, selection = info.selection, typeParameters = info.typeParameters;
-            var node = ts.createNode(322 /* JSDocTypedefTag */);
-            node.tagName = ts.createIdentifier("typedef"); // TODO: jsdoc factory https://github.com/Microsoft/TypeScript/pull/29539
-            node.fullName = ts.createIdentifier(name);
-            node.name = node.fullName;
-            node.typeExpression = ts.createJSDocTypeExpression(selection);
+            var node = ts.factory.createJSDocTypedefTag(ts.factory.createIdentifier("typedef"), ts.factory.createJSDocTypeExpression(selection), ts.factory.createIdentifier(name));
             var templates = [];
             ts.forEach(typeParameters, function (typeParameter) {
                 var constraint = ts.getEffectiveConstraintOfTypeParameter(typeParameter);
-                var template = ts.createNode(321 /* JSDocTemplateTag */);
-                template.tagName = ts.createIdentifier("template");
-                template.constraint = constraint && ts.cast(constraint, ts.isJSDocTypeExpression);
-                var parameter = ts.createNode(155 /* TypeParameter */);
-                parameter.name = typeParameter.name;
-                template.typeParameters = ts.createNodeArray([parameter]);
+                var parameter = ts.factory.createTypeParameterDeclaration(typeParameter.name);
+                var template = ts.factory.createJSDocTemplateTag(ts.factory.createIdentifier("template"), constraint && ts.cast(constraint, ts.isJSDocTypeExpression), [parameter]);
                 templates.push(template);
             });
-            changes.insertNodeBefore(file, firstStatement, ts.createJSDocComment(/* comment */ undefined, ts.createNodeArray(ts.concatenate(templates, [node]))), /* blankLineBetween */ true);
-            changes.replaceNode(file, selection, ts.createTypeReferenceNode(name, typeParameters.map(function (id) { return ts.createTypeReferenceNode(id.name, /* typeArguments */ undefined); })));
+            changes.insertNodeBefore(file, firstStatement, ts.factory.createJSDocComment(/* comment */ undefined, ts.factory.createNodeArray(ts.concatenate(templates, [node]))), /* blankLineBetween */ true);
+            changes.replaceNode(file, selection, ts.factory.createTypeReferenceNode(name, typeParameters.map(function (id) { return ts.factory.createTypeReferenceNode(id.name, /* typeArguments */ undefined); })));
         }
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
@@ -136619,168 +144338,54 @@ var ts;
         (function (generateGetAccessorAndSetAccessor) {
             var actionName = "Generate 'get' and 'set' accessors";
             var actionDescription = ts.Diagnostics.Generate_get_and_set_accessors.message;
-            refactor.registerRefactor(actionName, { getEditsForAction: getEditsForAction, getAvailableActions: getAvailableActions });
-            function getAvailableActions(context) {
-                if (!getConvertibleFieldAtPosition(context))
-                    return ts.emptyArray;
-                return [{
-                        name: actionName,
-                        description: actionDescription,
-                        actions: [
-                            {
+            refactor.registerRefactor(actionName, {
+                getEditsForAction: function (context, actionName) {
+                    if (!context.endPosition)
+                        return undefined;
+                    var info = ts.codefix.getAccessorConvertiblePropertyAtPosition(context.file, context.program, context.startPosition, context.endPosition);
+                    if (!info || !info.info)
+                        return undefined;
+                    var edits = ts.codefix.generateAccessorFromProperty(context.file, context.program, context.startPosition, context.endPosition, context, actionName);
+                    if (!edits)
+                        return undefined;
+                    var renameFilename = context.file.fileName;
+                    var nameNeedRename = info.info.renameAccessor ? info.info.accessorName : info.info.fieldName;
+                    var renameLocationOffset = ts.isIdentifier(nameNeedRename) ? 0 : -1;
+                    var renameLocation = renameLocationOffset + ts.getRenameLocation(edits, renameFilename, nameNeedRename.text, /*preferLastLocation*/ ts.isParameter(info.info.declaration));
+                    return { renameFilename: renameFilename, renameLocation: renameLocation, edits: edits };
+                },
+                getAvailableActions: function (context) {
+                    if (!context.endPosition)
+                        return ts.emptyArray;
+                    var info = ts.codefix.getAccessorConvertiblePropertyAtPosition(context.file, context.program, context.startPosition, context.endPosition, context.triggerReason === "invoked");
+                    if (!info)
+                        return ts.emptyArray;
+                    if (!info.error) {
+                        return [{
                                 name: actionName,
-                                description: actionDescription
-                            }
-                        ]
-                    }];
-            }
-            function getEditsForAction(context, _actionName) {
-                var file = context.file;
-                var fieldInfo = getConvertibleFieldAtPosition(context);
-                if (!fieldInfo)
-                    return undefined;
-                var isJS = ts.isSourceFileJS(file);
-                var changeTracker = ts.textChanges.ChangeTracker.fromContext(context);
-                var isStatic = fieldInfo.isStatic, isReadonly = fieldInfo.isReadonly, fieldName = fieldInfo.fieldName, accessorName = fieldInfo.accessorName, originalName = fieldInfo.originalName, type = fieldInfo.type, container = fieldInfo.container, declaration = fieldInfo.declaration, renameAccessor = fieldInfo.renameAccessor;
-                ts.suppressLeadingAndTrailingTrivia(fieldName);
-                ts.suppressLeadingAndTrailingTrivia(accessorName);
-                ts.suppressLeadingAndTrailingTrivia(declaration);
-                ts.suppressLeadingAndTrailingTrivia(container);
-                var isInClassLike = ts.isClassLike(container);
-                // avoid Readonly modifier because it will convert to get accessor
-                var modifierFlags = ts.getModifierFlags(declaration) & ~64 /* Readonly */;
-                var accessorModifiers = isInClassLike
-                    ? !modifierFlags || modifierFlags & 8 /* Private */
-                        ? getModifiers(isJS, isStatic, 119 /* PublicKeyword */)
-                        : ts.createNodeArray(ts.createModifiersFromModifierFlags(modifierFlags))
-                    : undefined;
-                var fieldModifiers = isInClassLike ? getModifiers(isJS, isStatic, 117 /* PrivateKeyword */) : undefined;
-                updateFieldDeclaration(changeTracker, file, declaration, fieldName, fieldModifiers);
-                var getAccessor = generateGetAccessor(fieldName, accessorName, type, accessorModifiers, isStatic, container);
-                ts.suppressLeadingAndTrailingTrivia(getAccessor);
-                insertAccessor(changeTracker, file, getAccessor, declaration, container);
-                if (isReadonly) {
-                    // readonly modifier only existed in classLikeDeclaration
-                    var constructor = ts.getFirstConstructorWithBody(container);
-                    if (constructor) {
-                        updateReadonlyPropertyInitializerStatementConstructor(changeTracker, file, constructor, fieldName.text, originalName);
-                    }
-                }
-                else {
-                    var setAccessor = generateSetAccessor(fieldName, accessorName, type, accessorModifiers, isStatic, container);
-                    ts.suppressLeadingAndTrailingTrivia(setAccessor);
-                    insertAccessor(changeTracker, file, setAccessor, declaration, container);
-                }
-                var edits = changeTracker.getChanges();
-                var renameFilename = file.fileName;
-                var nameNeedRename = renameAccessor ? accessorName : fieldName;
-                var renameLocationOffset = ts.isIdentifier(nameNeedRename) ? 0 : -1;
-                var renameLocation = renameLocationOffset + ts.getRenameLocation(edits, renameFilename, nameNeedRename.text, /*preferLastLocation*/ ts.isParameter(declaration));
-                return { renameFilename: renameFilename, renameLocation: renameLocation, edits: edits };
-            }
-            function isConvertibleName(name) {
-                return ts.isIdentifier(name) || ts.isStringLiteral(name);
-            }
-            function isAcceptedDeclaration(node) {
-                return ts.isParameterPropertyDeclaration(node, node.parent) || ts.isPropertyDeclaration(node) || ts.isPropertyAssignment(node);
-            }
-            function createPropertyName(name, originalName) {
-                return ts.isIdentifier(originalName) ? ts.createIdentifier(name) : ts.createLiteral(name);
-            }
-            function createAccessorAccessExpression(fieldName, isStatic, container) {
-                var leftHead = isStatic ? container.name : ts.createThis(); // TODO: GH#18217
-                return ts.isIdentifier(fieldName) ? ts.createPropertyAccess(leftHead, fieldName) : ts.createElementAccess(leftHead, ts.createLiteral(fieldName));
-            }
-            function getModifiers(isJS, isStatic, accessModifier) {
-                var modifiers = ts.append(!isJS ? [ts.createToken(accessModifier)] : undefined, isStatic ? ts.createToken(120 /* StaticKeyword */) : undefined);
-                return modifiers && ts.createNodeArray(modifiers);
-            }
-            function getConvertibleFieldAtPosition(context) {
-                var file = context.file, startPosition = context.startPosition, endPosition = context.endPosition;
-                var node = ts.getTokenAtPosition(file, startPosition);
-                var declaration = ts.findAncestor(node.parent, isAcceptedDeclaration);
-                // make sure declaration have AccessibilityModifier or Static Modifier or Readonly Modifier
-                var meaning = 28 /* AccessibilityModifier */ | 32 /* Static */ | 64 /* Readonly */;
-                if (!declaration || !ts.nodeOverlapsWithStartEnd(declaration.name, file, startPosition, endPosition) // TODO: GH#18217
-                    || !isConvertibleName(declaration.name) || (ts.getModifierFlags(declaration) | meaning) !== meaning)
-                    return undefined;
-                var name = declaration.name.text;
-                var startWithUnderscore = ts.startsWithUnderscore(name);
-                var fieldName = createPropertyName(startWithUnderscore ? name : ts.getUniqueName("_" + name, file), declaration.name);
-                var accessorName = createPropertyName(startWithUnderscore ? ts.getUniqueName(name.substring(1), file) : name, declaration.name);
-                return {
-                    isStatic: ts.hasStaticModifier(declaration),
-                    isReadonly: ts.hasReadonlyModifier(declaration),
-                    type: ts.getTypeAnnotationNode(declaration),
-                    container: declaration.kind === 156 /* Parameter */ ? declaration.parent.parent : declaration.parent,
-                    originalName: declaration.name.text,
-                    declaration: declaration,
-                    fieldName: fieldName,
-                    accessorName: accessorName,
-                    renameAccessor: startWithUnderscore
-                };
-            }
-            function generateGetAccessor(fieldName, accessorName, type, modifiers, isStatic, container) {
-                return ts.createGetAccessor(
-                /*decorators*/ undefined, modifiers, accessorName, 
-                /*parameters*/ undefined, // TODO: GH#18217
-                type, ts.createBlock([
-                    ts.createReturn(createAccessorAccessExpression(fieldName, isStatic, container))
-                ], /*multiLine*/ true));
-            }
-            function generateSetAccessor(fieldName, accessorName, type, modifiers, isStatic, container) {
-                return ts.createSetAccessor(
-                /*decorators*/ undefined, modifiers, accessorName, [ts.createParameter(
-                    /*decorators*/ undefined, 
-                    /*modifiers*/ undefined, 
-                    /*dotDotDotToken*/ undefined, ts.createIdentifier("value"), 
-                    /*questionToken*/ undefined, type)], ts.createBlock([
-                    ts.createStatement(ts.createAssignment(createAccessorAccessExpression(fieldName, isStatic, container), ts.createIdentifier("value")))
-                ], /*multiLine*/ true));
-            }
-            function updatePropertyDeclaration(changeTracker, file, declaration, fieldName, modifiers) {
-                var property = ts.updateProperty(declaration, declaration.decorators, modifiers, fieldName, declaration.questionToken || declaration.exclamationToken, declaration.type, declaration.initializer);
-                changeTracker.replaceNode(file, declaration, property);
-            }
-            function updatePropertyAssignmentDeclaration(changeTracker, file, declaration, fieldName) {
-                var assignment = ts.updatePropertyAssignment(declaration, fieldName, declaration.initializer);
-                changeTracker.replacePropertyAssignment(file, declaration, assignment);
-            }
-            function updateFieldDeclaration(changeTracker, file, declaration, fieldName, modifiers) {
-                if (ts.isPropertyDeclaration(declaration)) {
-                    updatePropertyDeclaration(changeTracker, file, declaration, fieldName, modifiers);
-                }
-                else if (ts.isPropertyAssignment(declaration)) {
-                    updatePropertyAssignmentDeclaration(changeTracker, file, declaration, fieldName);
-                }
-                else {
-                    changeTracker.replaceNode(file, declaration, ts.updateParameter(declaration, declaration.decorators, modifiers, declaration.dotDotDotToken, ts.cast(fieldName, ts.isIdentifier), declaration.questionToken, declaration.type, declaration.initializer));
-                }
-            }
-            function insertAccessor(changeTracker, file, accessor, declaration, container) {
-                ts.isParameterPropertyDeclaration(declaration, declaration.parent) ? changeTracker.insertNodeAtClassStart(file, container, accessor) :
-                    ts.isPropertyAssignment(declaration) ? changeTracker.insertNodeAfterComma(file, declaration, accessor) :
-                        changeTracker.insertNodeAfter(file, declaration, accessor);
-            }
-            function updateReadonlyPropertyInitializerStatementConstructor(changeTracker, file, constructor, fieldName, originalName) {
-                if (!constructor.body)
-                    return;
-                constructor.body.forEachChild(function recur(node) {
-                    if (ts.isElementAccessExpression(node) &&
-                        node.expression.kind === 104 /* ThisKeyword */ &&
-                        ts.isStringLiteral(node.argumentExpression) &&
-                        node.argumentExpression.text === originalName &&
-                        ts.isWriteAccess(node)) {
-                        changeTracker.replaceNode(file, node.argumentExpression, ts.createStringLiteral(fieldName));
-                    }
-                    if (ts.isPropertyAccessExpression(node) && node.expression.kind === 104 /* ThisKeyword */ && node.name.text === originalName && ts.isWriteAccess(node)) {
-                        changeTracker.replaceNode(file, node.name, ts.createIdentifier(fieldName));
+                                description: actionDescription,
+                                actions: [
+                                    {
+                                        name: actionName,
+                                        description: actionDescription
+                                    }
+                                ]
+                            }];
                     }
-                    if (!ts.isFunctionLike(node) && !ts.isClassLike(node)) {
-                        node.forEachChild(recur);
+                    if (context.preferences.provideRefactorNotApplicableReason) {
+                        return [{
+                                name: actionName,
+                                description: actionDescription,
+                                actions: [{
+                                        name: actionName,
+                                        description: actionDescription,
+                                        notApplicableReason: info.error
+                                    }]
+                            }];
                     }
-                });
-            }
+                    return ts.emptyArray;
+                }
+            });
         })(generateGetAccessorAndSetAccessor = refactor.generateGetAccessorAndSetAccessor || (refactor.generateGetAccessorAndSetAccessor = {}));
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
@@ -136838,7 +144443,6 @@ var ts;
             changes.createNewFile(oldFile, ts.combinePaths(currentDirectory, newFileNameWithExtension), getNewStatementsAndRemoveFromOldFile(oldFile, usage, changes, toMove, program, newModuleName, preferences));
             addNewFileToTsconfig(program, changes, oldFile.fileName, newFileNameWithExtension, ts.hostGetCanonicalFileName(host));
         }
-        // Filters imports out of the range of statements to move. Imports will be copied to the new file anyway, and may still be needed in the old file.
         function getStatementsToMove(context) {
             var rangeToMove = getRangeToMove(context);
             if (rangeToMove === undefined)
@@ -136846,20 +144450,27 @@ var ts;
             var all = [];
             var ranges = [];
             var toMove = rangeToMove.toMove, afterLast = rangeToMove.afterLast;
-            ts.getRangesWhere(toMove, function (s) { return !isPureImport(s); }, function (start, afterEndIndex) {
+            ts.getRangesWhere(toMove, isAllowedStatementToMove, function (start, afterEndIndex) {
                 for (var i = start; i < afterEndIndex; i++)
                     all.push(toMove[i]);
                 ranges.push({ first: toMove[start], afterLast: afterLast });
             });
             return all.length === 0 ? undefined : { all: all, ranges: ranges };
         }
+        function isAllowedStatementToMove(statement) {
+            // Filters imports and prologue directives out of the range of statements to move.
+            // Imports will be copied to the new file anyway, and may still be needed in the old file.
+            // Prologue directives will be copied to the new file and should be left in the old file.
+            return !isPureImport(statement) && !ts.isPrologueDirective(statement);
+            ;
+        }
         function isPureImport(node) {
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     return true;
-                case 253 /* ImportEqualsDeclaration */:
-                    return !ts.hasModifier(node, 1 /* Export */);
-                case 225 /* VariableStatement */:
+                case 260 /* ImportEqualsDeclaration */:
+                    return !ts.hasSyntacticModifier(node, 1 /* Export */);
+                case 232 /* VariableStatement */:
                     return node.declarationList.declarations.every(function (d) { return !!d.initializer && ts.isRequireCall(d.initializer, /*checkArgumentIsStringLiteralLike*/ true); });
                 default:
                     return false;
@@ -136876,14 +144487,15 @@ var ts;
                 return ts.isPropertyAssignment(prop) && ts.isStringLiteral(prop.name) && prop.name.text === "files";
             });
             if (filesProp && ts.isArrayLiteralExpression(filesProp.initializer)) {
-                changes.insertNodeInListAfter(cfg, ts.last(filesProp.initializer.elements), ts.createLiteral(newFilePath), filesProp.initializer.elements);
+                changes.insertNodeInListAfter(cfg, ts.last(filesProp.initializer.elements), ts.factory.createStringLiteral(newFilePath), filesProp.initializer.elements);
             }
         }
         function getNewStatementsAndRemoveFromOldFile(oldFile, usage, changes, toMove, program, newModuleName, preferences) {
             var checker = program.getTypeChecker();
+            var prologueDirectives = ts.takeWhile(oldFile.statements, ts.isPrologueDirective);
             if (!oldFile.externalModuleIndicator && !oldFile.commonJsModuleIndicator) {
                 deleteMovedStatements(oldFile, toMove.ranges, changes);
-                return toMove.all;
+                return __spreadArrays(prologueDirectives, toMove.all);
             }
             var useEs6ModuleSyntax = !!oldFile.externalModuleIndicator;
             var quotePreference = ts.getQuotePreference(oldFile, preferences);
@@ -136894,7 +144506,14 @@ var ts;
             deleteUnusedOldImports(oldFile, toMove.all, changes, usage.unusedImportsFromOldFile, checker);
             deleteMovedStatements(oldFile, toMove.ranges, changes);
             updateImportsInOtherFiles(changes, program, oldFile, usage.movedSymbols, newModuleName);
-            return __spreadArrays(getNewFileImportsAndAddExportInOldFile(oldFile, usage.oldImportsNeededByNewFile, usage.newFileImportsFromOldFile, changes, checker, useEs6ModuleSyntax, quotePreference), addExports(oldFile, toMove.all, usage.oldFileImportsFromNewFile, useEs6ModuleSyntax));
+            var imports = getNewFileImportsAndAddExportInOldFile(oldFile, usage.oldImportsNeededByNewFile, usage.newFileImportsFromOldFile, changes, checker, useEs6ModuleSyntax, quotePreference);
+            var body = addExports(oldFile, toMove.all, usage.oldFileImportsFromNewFile, useEs6ModuleSyntax);
+            if (imports.length && body.length) {
+                return __spreadArrays(prologueDirectives, imports, [
+                    4 /* NewLineTrivia */
+                ], body);
+            }
+            return __spreadArrays(prologueDirectives, imports, body);
         }
         function deleteMovedStatements(sourceFile, moved, changes) {
             for (var _i = 0, moved_1 = moved; _i < moved_1.length; _i++) {
@@ -136927,7 +144546,7 @@ var ts;
                         };
                         deleteUnusedImports(sourceFile, importNode, changes, shouldMove); // These will be changed to imports from the new file
                         var newModuleSpecifier = ts.combinePaths(ts.getDirectoryPath(moduleSpecifierFromImport(importNode).text), newModuleName);
-                        var newImportDeclaration = filterImport(importNode, ts.createLiteral(newModuleSpecifier), shouldMove);
+                        var newImportDeclaration = filterImport(importNode, ts.factory.createStringLiteral(newModuleSpecifier), shouldMove);
                         if (newImportDeclaration)
                             changes.insertNodeAfter(sourceFile, statement, newImportDeclaration);
                         var ns = getNamespaceLikeImport(importNode);
@@ -136947,12 +144566,12 @@ var ts;
         }
         function getNamespaceLikeImport(node) {
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
-                    return node.importClause && node.importClause.namedBindings && node.importClause.namedBindings.kind === 256 /* NamespaceImport */ ?
+                case 261 /* ImportDeclaration */:
+                    return node.importClause && node.importClause.namedBindings && node.importClause.namedBindings.kind === 263 /* NamespaceImport */ ?
                         node.importClause.namedBindings.name : undefined;
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return node.name;
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return ts.tryCast(node.name, ts.isIdentifier);
                 default:
                     return ts.Debug.assertNever(node, "Unexpected node kind " + node.kind);
@@ -136974,29 +144593,29 @@ var ts;
                 var newNamespaceName = needUniqueName ? ts.getUniqueName(preferredNewNamespaceName, sourceFile) : preferredNewNamespaceName;
                 for (var _i = 0, toChange_1 = toChange; _i < toChange_1.length; _i++) {
                     var ref = toChange_1[_i];
-                    changes.replaceNode(sourceFile, ref, ts.createIdentifier(newNamespaceName));
+                    changes.replaceNode(sourceFile, ref, ts.factory.createIdentifier(newNamespaceName));
                 }
                 changes.insertNodeAfter(sourceFile, oldImportNode, updateNamespaceLikeImportNode(oldImportNode, newModuleName, newModuleSpecifier));
             }
         }
         function updateNamespaceLikeImportNode(node, newNamespaceName, newModuleSpecifier) {
-            var newNamespaceId = ts.createIdentifier(newNamespaceName);
-            var newModuleString = ts.createLiteral(newModuleSpecifier);
+            var newNamespaceId = ts.factory.createIdentifier(newNamespaceName);
+            var newModuleString = ts.factory.createStringLiteral(newModuleSpecifier);
             switch (node.kind) {
-                case 254 /* ImportDeclaration */:
-                    return ts.createImportDeclaration(
-                    /*decorators*/ undefined, /*modifiers*/ undefined, ts.createImportClause(/*name*/ undefined, ts.createNamespaceImport(newNamespaceId)), newModuleString);
-                case 253 /* ImportEqualsDeclaration */:
-                    return ts.createImportEqualsDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, newNamespaceId, ts.createExternalModuleReference(newModuleString));
-                case 242 /* VariableDeclaration */:
-                    return ts.createVariableDeclaration(newNamespaceId, /*type*/ undefined, createRequireCall(newModuleString));
+                case 261 /* ImportDeclaration */:
+                    return ts.factory.createImportDeclaration(
+                    /*decorators*/ undefined, /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, /*name*/ undefined, ts.factory.createNamespaceImport(newNamespaceId)), newModuleString);
+                case 260 /* ImportEqualsDeclaration */:
+                    return ts.factory.createImportEqualsDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, newNamespaceId, ts.factory.createExternalModuleReference(newModuleString));
+                case 249 /* VariableDeclaration */:
+                    return ts.factory.createVariableDeclaration(newNamespaceId, /*exclamationToken*/ undefined, /*type*/ undefined, createRequireCall(newModuleString));
                 default:
                     return ts.Debug.assertNever(node, "Unexpected node kind " + node.kind);
             }
         }
         function moduleSpecifierFromImport(i) {
-            return (i.kind === 254 /* ImportDeclaration */ ? i.moduleSpecifier
-                : i.kind === 253 /* ImportEqualsDeclaration */ ? i.moduleReference.expression
+            return (i.kind === 261 /* ImportDeclaration */ ? i.moduleSpecifier
+                : i.kind === 260 /* ImportEqualsDeclaration */ ? i.moduleReference.expression
                     : i.initializer.arguments[0]);
         }
         function forEachImportInStatement(statement, cb) {
@@ -137023,7 +144642,7 @@ var ts;
             var imports = [];
             newFileNeedExport.forEach(function (symbol) {
                 if (symbol.escapedName === "default" /* Default */) {
-                    defaultImport = ts.createIdentifier(ts.symbolNameNoDefault(symbol)); // TODO: GH#18217
+                    defaultImport = ts.factory.createIdentifier(ts.symbolNameNoDefault(symbol)); // TODO: GH#18217
                 }
                 else {
                     imports.push(symbol.name);
@@ -137034,23 +144653,23 @@ var ts;
         function makeImportOrRequire(defaultImport, imports, path, useEs6Imports, quotePreference) {
             path = ts.ensurePathIsNonModuleName(path);
             if (useEs6Imports) {
-                var specifiers = imports.map(function (i) { return ts.createImportSpecifier(/*propertyName*/ undefined, ts.createIdentifier(i)); });
+                var specifiers = imports.map(function (i) { return ts.factory.createImportSpecifier(/*propertyName*/ undefined, ts.factory.createIdentifier(i)); });
                 return ts.makeImportIfNecessary(defaultImport, specifiers, path, quotePreference);
             }
             else {
                 ts.Debug.assert(!defaultImport, "No default import should exist"); // If there's a default export, it should have been an es6 module.
-                var bindingElements = imports.map(function (i) { return ts.createBindingElement(/*dotDotDotToken*/ undefined, /*propertyName*/ undefined, i); });
+                var bindingElements = imports.map(function (i) { return ts.factory.createBindingElement(/*dotDotDotToken*/ undefined, /*propertyName*/ undefined, i); });
                 return bindingElements.length
-                    ? makeVariableStatement(ts.createObjectBindingPattern(bindingElements), /*type*/ undefined, createRequireCall(ts.createLiteral(path)))
+                    ? makeVariableStatement(ts.factory.createObjectBindingPattern(bindingElements), /*type*/ undefined, createRequireCall(ts.factory.createStringLiteral(path)))
                     : undefined;
             }
         }
         function makeVariableStatement(name, type, initializer, flags) {
             if (flags === void 0) { flags = 2 /* Const */; }
-            return ts.createVariableStatement(/*modifiers*/ undefined, ts.createVariableDeclarationList([ts.createVariableDeclaration(name, type, initializer)], flags));
+            return ts.factory.createVariableStatement(/*modifiers*/ undefined, ts.factory.createVariableDeclarationList([ts.factory.createVariableDeclaration(name, /*exclamationToken*/ undefined, type, initializer)], flags));
         }
         function createRequireCall(moduleSpecifier) {
-            return ts.createCall(ts.createIdentifier("require"), /*typeArguments*/ undefined, [moduleSpecifier]);
+            return ts.factory.createCallExpression(ts.factory.createIdentifier("require"), /*typeArguments*/ undefined, [moduleSpecifier]);
         }
         function addExports(sourceFile, toMove, needExport, useEs6Exports) {
             return ts.flatMap(toMove, function (statement) {
@@ -137066,15 +144685,15 @@ var ts;
         }
         function deleteUnusedImports(sourceFile, importDecl, changes, isUnused) {
             switch (importDecl.kind) {
-                case 254 /* ImportDeclaration */:
+                case 261 /* ImportDeclaration */:
                     deleteUnusedImportsInDeclaration(sourceFile, importDecl, changes, isUnused);
                     break;
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     if (isUnused(importDecl.name)) {
                         changes.delete(sourceFile, importDecl);
                     }
                     break;
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     deleteUnusedImportsInVariableDeclaration(sourceFile, importDecl, changes, isUnused);
                     break;
                 default:
@@ -137087,7 +144706,7 @@ var ts;
             var _a = importDecl.importClause, name = _a.name, namedBindings = _a.namedBindings;
             var defaultUnused = !name || isUnused(name);
             var namedBindingsUnused = !namedBindings ||
-                (namedBindings.kind === 256 /* NamespaceImport */ ? isUnused(namedBindings.name) : namedBindings.elements.length !== 0 && namedBindings.elements.every(function (e) { return isUnused(e.name); }));
+                (namedBindings.kind === 263 /* NamespaceImport */ ? isUnused(namedBindings.name) : namedBindings.elements.length !== 0 && namedBindings.elements.every(function (e) { return isUnused(e.name); }));
             if (defaultUnused && namedBindingsUnused) {
                 changes.delete(sourceFile, importDecl);
             }
@@ -137097,9 +144716,9 @@ var ts;
                 }
                 if (namedBindings) {
                     if (namedBindingsUnused) {
-                        changes.replaceNode(sourceFile, importDecl.importClause, ts.updateImportClause(importDecl.importClause, name, /*namedBindings*/ undefined, importDecl.importClause.isTypeOnly));
+                        changes.replaceNode(sourceFile, importDecl.importClause, ts.factory.updateImportClause(importDecl.importClause, importDecl.importClause.isTypeOnly, name, /*namedBindings*/ undefined));
                     }
-                    else if (namedBindings.kind === 257 /* NamedImports */) {
+                    else if (namedBindings.kind === 264 /* NamedImports */) {
                         for (var _i = 0, _b = namedBindings.elements; _i < _b.length; _i++) {
                             var element = _b[_i];
                             if (isUnused(element.name))
@@ -137112,14 +144731,14 @@ var ts;
         function deleteUnusedImportsInVariableDeclaration(sourceFile, varDecl, changes, isUnused) {
             var name = varDecl.name;
             switch (name.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     if (isUnused(name)) {
                         changes.delete(sourceFile, name);
                     }
                     break;
-                case 190 /* ArrayBindingPattern */:
+                case 197 /* ArrayBindingPattern */:
                     break;
-                case 189 /* ObjectBindingPattern */:
+                case 196 /* ObjectBindingPattern */:
                     if (name.elements.every(function (e) { return ts.isIdentifier(e.name) && isUnused(e.name); })) {
                         changes.delete(sourceFile, ts.isVariableDeclarationList(varDecl.parent) && varDecl.parent.declarations.length === 1 ? varDecl.parent.parent : varDecl);
                     }
@@ -137158,7 +144777,7 @@ var ts;
                     if (markSeenTop(top)) {
                         addExportToChanges(oldFile, top, changes, useEs6ModuleSyntax);
                     }
-                    if (ts.hasModifier(decl, 512 /* Default */)) {
+                    if (ts.hasSyntacticModifier(decl, 512 /* Default */)) {
                         oldFileDefault = name;
                     }
                     else {
@@ -137246,14 +144865,14 @@ var ts;
         // Below should all be utilities
         function isInImport(decl) {
             switch (decl.kind) {
-                case 253 /* ImportEqualsDeclaration */:
-                case 258 /* ImportSpecifier */:
-                case 255 /* ImportClause */:
-                case 256 /* NamespaceImport */:
+                case 260 /* ImportEqualsDeclaration */:
+                case 265 /* ImportSpecifier */:
+                case 262 /* ImportClause */:
+                case 263 /* NamespaceImport */:
                     return true;
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return isVariableDeclarationInImport(decl);
-                case 191 /* BindingElement */:
+                case 198 /* BindingElement */:
                     return ts.isVariableDeclaration(decl.parent.parent) && isVariableDeclarationInImport(decl.parent.parent);
                 default:
                     return false;
@@ -137265,19 +144884,19 @@ var ts;
         }
         function filterImport(i, moduleSpecifier, keep) {
             switch (i.kind) {
-                case 254 /* ImportDeclaration */: {
+                case 261 /* ImportDeclaration */: {
                     var clause = i.importClause;
                     if (!clause)
                         return undefined;
                     var defaultImport = clause.name && keep(clause.name) ? clause.name : undefined;
                     var namedBindings = clause.namedBindings && filterNamedBindings(clause.namedBindings, keep);
                     return defaultImport || namedBindings
-                        ? ts.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.createImportClause(defaultImport, namedBindings), moduleSpecifier)
+                        ? ts.factory.createImportDeclaration(/*decorators*/ undefined, /*modifiers*/ undefined, ts.factory.createImportClause(/*isTypeOnly*/ false, defaultImport, namedBindings), moduleSpecifier)
                         : undefined;
                 }
-                case 253 /* ImportEqualsDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return keep(i.name) ? i : undefined;
-                case 242 /* VariableDeclaration */: {
+                case 249 /* VariableDeclaration */: {
                     var name = filterBindingName(i.name, keep);
                     return name ? makeVariableStatement(name, i.type, createRequireCall(moduleSpecifier), i.parent.flags) : undefined;
                 }
@@ -137286,24 +144905,24 @@ var ts;
             }
         }
         function filterNamedBindings(namedBindings, keep) {
-            if (namedBindings.kind === 256 /* NamespaceImport */) {
+            if (namedBindings.kind === 263 /* NamespaceImport */) {
                 return keep(namedBindings.name) ? namedBindings : undefined;
             }
             else {
                 var newElements = namedBindings.elements.filter(function (e) { return keep(e.name); });
-                return newElements.length ? ts.createNamedImports(newElements) : undefined;
+                return newElements.length ? ts.factory.createNamedImports(newElements) : undefined;
             }
         }
         function filterBindingName(name, keep) {
             switch (name.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return keep(name) ? name : undefined;
-                case 190 /* ArrayBindingPattern */:
+                case 197 /* ArrayBindingPattern */:
                     return name;
-                case 189 /* ObjectBindingPattern */: {
+                case 196 /* ObjectBindingPattern */: {
                     // We can't handle nested destructurings or property names well here, so just copy them all.
                     var newElements = name.elements.filter(function (prop) { return prop.propertyName || !ts.isIdentifier(prop.name) || keep(prop.name); });
-                    return newElements.length ? ts.createObjectBindingPattern(newElements) : undefined;
+                    return newElements.length ? ts.factory.createObjectBindingPattern(newElements) : undefined;
                 }
             }
         }
@@ -137321,7 +144940,7 @@ var ts;
         }
         var SymbolSet = /** @class */ (function () {
             function SymbolSet() {
-                this.map = ts.createMap();
+                this.map = new ts.Map();
             }
             SymbolSet.prototype.add = function (symbol) {
                 this.map.set(String(ts.getSymbolId(symbol)), symbol);
@@ -137357,13 +144976,13 @@ var ts;
         }
         function isNonVariableTopLevelDeclaration(node) {
             switch (node.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 249 /* ModuleDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 253 /* ImportEqualsDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return true;
                 default:
                     return false;
@@ -137371,17 +144990,17 @@ var ts;
         }
         function forEachTopLevelDeclaration(statement, cb) {
             switch (statement.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 245 /* ClassDeclaration */:
-                case 249 /* ModuleDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 253 /* ImportEqualsDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return cb(statement);
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return ts.firstDefined(statement.declarationList.declarations, function (decl) { return forEachTopLevelDeclarationInBindingName(decl.name, cb); });
-                case 226 /* ExpressionStatement */: {
+                case 233 /* ExpressionStatement */: {
                     var expression = statement.expression;
                     return ts.isBinaryExpression(expression) && ts.getAssignmentDeclarationKind(expression) === 1 /* ExportsProperty */
                         ? cb(statement)
@@ -137391,10 +145010,10 @@ var ts;
         }
         function forEachTopLevelDeclarationInBindingName(name, cb) {
             switch (name.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return cb(ts.cast(name.parent, function (x) { return ts.isVariableDeclaration(x) || ts.isBindingElement(x); }));
-                case 190 /* ArrayBindingPattern */:
-                case 189 /* ObjectBindingPattern */:
+                case 197 /* ArrayBindingPattern */:
+                case 196 /* ObjectBindingPattern */:
                     return ts.firstDefined(name.elements, function (em) { return ts.isOmittedExpression(em) ? undefined : forEachTopLevelDeclarationInBindingName(em.name, cb); });
                 default:
                     return ts.Debug.assertNever(name, "Unexpected name kind " + name.kind);
@@ -137405,9 +145024,9 @@ var ts;
         }
         function getTopLevelDeclarationStatement(d) {
             switch (d.kind) {
-                case 242 /* VariableDeclaration */:
+                case 249 /* VariableDeclaration */:
                     return d.parent.parent;
-                case 191 /* BindingElement */:
+                case 198 /* BindingElement */:
                     return getTopLevelDeclarationStatement(ts.cast(d.parent.parent, function (p) { return ts.isVariableDeclaration(p) || ts.isBindingElement(p); }));
                 default:
                     return d;
@@ -137428,7 +145047,7 @@ var ts;
         }
         function isExported(sourceFile, decl, useEs6Exports) {
             if (useEs6Exports) {
-                return !ts.isExpressionStatement(decl) && ts.hasModifier(decl, 1 /* Export */);
+                return !ts.isExpressionStatement(decl) && ts.hasSyntacticModifier(decl, 1 /* Export */);
             }
             else {
                 return getNamesToExportInCommonJS(decl).some(function (name) { return sourceFile.symbol.exports.has(ts.escapeLeadingUnderscores(name)); });
@@ -137438,25 +145057,25 @@ var ts;
             return useEs6Exports ? [addEs6Export(decl)] : addCommonjsExport(decl);
         }
         function addEs6Export(d) {
-            var modifiers = ts.concatenate([ts.createModifier(89 /* ExportKeyword */)], d.modifiers);
+            var modifiers = ts.concatenate([ts.factory.createModifier(92 /* ExportKeyword */)], d.modifiers);
             switch (d.kind) {
-                case 244 /* FunctionDeclaration */:
-                    return ts.updateFunctionDeclaration(d, d.decorators, modifiers, d.asteriskToken, d.name, d.typeParameters, d.parameters, d.type, d.body);
-                case 245 /* ClassDeclaration */:
-                    return ts.updateClassDeclaration(d, d.decorators, modifiers, d.name, d.typeParameters, d.heritageClauses, d.members);
-                case 225 /* VariableStatement */:
-                    return ts.updateVariableStatement(d, modifiers, d.declarationList);
-                case 249 /* ModuleDeclaration */:
-                    return ts.updateModuleDeclaration(d, d.decorators, modifiers, d.name, d.body);
-                case 248 /* EnumDeclaration */:
-                    return ts.updateEnumDeclaration(d, d.decorators, modifiers, d.name, d.members);
-                case 247 /* TypeAliasDeclaration */:
-                    return ts.updateTypeAliasDeclaration(d, d.decorators, modifiers, d.name, d.typeParameters, d.type);
-                case 246 /* InterfaceDeclaration */:
-                    return ts.updateInterfaceDeclaration(d, d.decorators, modifiers, d.name, d.typeParameters, d.heritageClauses, d.members);
-                case 253 /* ImportEqualsDeclaration */:
-                    return ts.updateImportEqualsDeclaration(d, d.decorators, modifiers, d.name, d.moduleReference);
-                case 226 /* ExpressionStatement */:
+                case 251 /* FunctionDeclaration */:
+                    return ts.factory.updateFunctionDeclaration(d, d.decorators, modifiers, d.asteriskToken, d.name, d.typeParameters, d.parameters, d.type, d.body);
+                case 252 /* ClassDeclaration */:
+                    return ts.factory.updateClassDeclaration(d, d.decorators, modifiers, d.name, d.typeParameters, d.heritageClauses, d.members);
+                case 232 /* VariableStatement */:
+                    return ts.factory.updateVariableStatement(d, modifiers, d.declarationList);
+                case 256 /* ModuleDeclaration */:
+                    return ts.factory.updateModuleDeclaration(d, d.decorators, modifiers, d.name, d.body);
+                case 255 /* EnumDeclaration */:
+                    return ts.factory.updateEnumDeclaration(d, d.decorators, modifiers, d.name, d.members);
+                case 254 /* TypeAliasDeclaration */:
+                    return ts.factory.updateTypeAliasDeclaration(d, d.decorators, modifiers, d.name, d.typeParameters, d.type);
+                case 253 /* InterfaceDeclaration */:
+                    return ts.factory.updateInterfaceDeclaration(d, d.decorators, modifiers, d.name, d.typeParameters, d.heritageClauses, d.members);
+                case 260 /* ImportEqualsDeclaration */:
+                    return ts.factory.updateImportEqualsDeclaration(d, d.decorators, modifiers, d.name, d.moduleReference);
+                case 233 /* ExpressionStatement */:
                     return ts.Debug.fail(); // Shouldn't try to add 'export' keyword to `exports.x = ...`
                 default:
                     return ts.Debug.assertNever(d, "Unexpected declaration kind " + d.kind);
@@ -137467,18 +145086,18 @@ var ts;
         }
         function getNamesToExportInCommonJS(decl) {
             switch (decl.kind) {
-                case 244 /* FunctionDeclaration */:
-                case 245 /* ClassDeclaration */:
+                case 251 /* FunctionDeclaration */:
+                case 252 /* ClassDeclaration */:
                     return [decl.name.text]; // TODO: GH#18217
-                case 225 /* VariableStatement */:
+                case 232 /* VariableStatement */:
                     return ts.mapDefined(decl.declarationList.declarations, function (d) { return ts.isIdentifier(d.name) ? d.name.text : undefined; });
-                case 249 /* ModuleDeclaration */:
-                case 248 /* EnumDeclaration */:
-                case 247 /* TypeAliasDeclaration */:
-                case 246 /* InterfaceDeclaration */:
-                case 253 /* ImportEqualsDeclaration */:
+                case 256 /* ModuleDeclaration */:
+                case 255 /* EnumDeclaration */:
+                case 254 /* TypeAliasDeclaration */:
+                case 253 /* InterfaceDeclaration */:
+                case 260 /* ImportEqualsDeclaration */:
                     return ts.emptyArray;
-                case 226 /* ExpressionStatement */:
+                case 233 /* ExpressionStatement */:
                     return ts.Debug.fail("Can't export an ExpressionStatement"); // Shouldn't try to add 'export' keyword to `exports.x = ...`
                 default:
                     return ts.Debug.assertNever(decl, "Unexpected decl kind " + decl.kind);
@@ -137486,7 +145105,7 @@ var ts;
         }
         /** Creates `exports.x = x;` */
         function createExportAssignment(name) {
-            return ts.createExpressionStatement(ts.createBinary(ts.createPropertyAccess(ts.createIdentifier("exports"), ts.createIdentifier(name)), 62 /* EqualsToken */, ts.createIdentifier(name)));
+            return ts.factory.createExpressionStatement(ts.factory.createBinaryExpression(ts.factory.createPropertyAccessExpression(ts.factory.createIdentifier("exports"), ts.factory.createIdentifier(name)), 62 /* EqualsToken */, ts.factory.createIdentifier(name)));
         }
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
@@ -137505,41 +145124,59 @@ var ts;
             var removeBracesActionDescription = ts.Diagnostics.Remove_braces_from_arrow_function.message;
             refactor.registerRefactor(refactorName, { getEditsForAction: getEditsForAction, getAvailableActions: getAvailableActions });
             function getAvailableActions(context) {
-                var file = context.file, startPosition = context.startPosition;
-                var info = getConvertibleArrowFunctionAtPosition(file, startPosition);
+                var file = context.file, startPosition = context.startPosition, triggerReason = context.triggerReason;
+                var info = getConvertibleArrowFunctionAtPosition(file, startPosition, triggerReason === "invoked");
                 if (!info)
                     return ts.emptyArray;
-                return [{
-                        name: refactorName,
-                        description: refactorDescription,
-                        actions: [
-                            info.addBraces ?
-                                {
+                if (info.error === undefined) {
+                    return [{
+                            name: refactorName,
+                            description: refactorDescription,
+                            actions: [
+                                info.info.addBraces ?
+                                    {
+                                        name: addBracesActionName,
+                                        description: addBracesActionDescription
+                                    } : {
+                                    name: removeBracesActionName,
+                                    description: removeBracesActionDescription
+                                }
+                            ]
+                        }];
+                }
+                if (context.preferences.provideRefactorNotApplicableReason) {
+                    return [{
+                            name: refactorName,
+                            description: refactorDescription,
+                            actions: [{
                                     name: addBracesActionName,
-                                    description: addBracesActionDescription
-                                } : {
-                                name: removeBracesActionName,
-                                description: removeBracesActionDescription
-                            }
-                        ]
-                    }];
+                                    description: addBracesActionDescription,
+                                    notApplicableReason: info.error
+                                }, {
+                                    name: removeBracesActionName,
+                                    description: removeBracesActionDescription,
+                                    notApplicableReason: info.error
+                                }]
+                        }];
+                }
+                return ts.emptyArray;
             }
             function getEditsForAction(context, actionName) {
                 var file = context.file, startPosition = context.startPosition;
                 var info = getConvertibleArrowFunctionAtPosition(file, startPosition);
-                if (!info)
+                if (!info || !info.info)
                     return undefined;
-                var expression = info.expression, returnStatement = info.returnStatement, func = info.func;
+                var _a = info.info, expression = _a.expression, returnStatement = _a.returnStatement, func = _a.func;
                 var body;
                 if (actionName === addBracesActionName) {
-                    var returnStatement_1 = ts.createReturn(expression);
-                    body = ts.createBlock([returnStatement_1], /* multiLine */ true);
+                    var returnStatement_1 = ts.factory.createReturnStatement(expression);
+                    body = ts.factory.createBlock([returnStatement_1], /* multiLine */ true);
                     ts.suppressLeadingAndTrailingTrivia(body);
                     ts.copyLeadingComments(expression, returnStatement_1, file, 3 /* MultiLineCommentTrivia */, /* hasTrailingNewLine */ true);
                 }
                 else if (actionName === removeBracesActionName && returnStatement) {
-                    var actualExpression = expression || ts.createVoidZero();
-                    body = ts.needsParentheses(actualExpression) ? ts.createParen(actualExpression) : actualExpression;
+                    var actualExpression = expression || ts.factory.createVoidZero();
+                    body = ts.needsParentheses(actualExpression) ? ts.factory.createParenthesizedExpression(actualExpression) : actualExpression;
                     ts.suppressLeadingAndTrailingTrivia(body);
                     ts.copyTrailingAsLeadingComments(returnStatement, body, file, 3 /* MultiLineCommentTrivia */, /* hasTrailingNewLine */ false);
                     ts.copyLeadingComments(returnStatement, body, file, 3 /* MultiLineCommentTrivia */, /* hasTrailingNewLine */ false);
@@ -137553,26 +145190,42 @@ var ts;
                 });
                 return { renameFilename: undefined, renameLocation: undefined, edits: edits };
             }
-            function getConvertibleArrowFunctionAtPosition(file, startPosition) {
+            function getConvertibleArrowFunctionAtPosition(file, startPosition, considerFunctionBodies) {
+                if (considerFunctionBodies === void 0) { considerFunctionBodies = true; }
                 var node = ts.getTokenAtPosition(file, startPosition);
                 var func = ts.getContainingFunction(node);
-                if (!func || !ts.isArrowFunction(func) || (!ts.rangeContainsRange(func, node) || ts.rangeContainsRange(func.body, node)))
+                if (!func) {
+                    return {
+                        error: ts.getLocaleSpecificMessage(ts.Diagnostics.Could_not_find_a_containing_arrow_function)
+                    };
+                }
+                if (!ts.isArrowFunction(func)) {
+                    return {
+                        error: ts.getLocaleSpecificMessage(ts.Diagnostics.Containing_function_is_not_an_arrow_function)
+                    };
+                }
+                if ((!ts.rangeContainsRange(func, node) || ts.rangeContainsRange(func.body, node) && !considerFunctionBodies)) {
                     return undefined;
+                }
                 if (ts.isExpression(func.body)) {
                     return {
-                        func: func,
-                        addBraces: true,
-                        expression: func.body
+                        info: {
+                            func: func,
+                            addBraces: true,
+                            expression: func.body
+                        }
                     };
                 }
                 else if (func.body.statements.length === 1) {
                     var firstStatement = ts.first(func.body.statements);
                     if (ts.isReturnStatement(firstStatement)) {
                         return {
-                            func: func,
-                            addBraces: false,
-                            expression: firstStatement.expression,
-                            returnStatement: firstStatement
+                            info: {
+                                func: func,
+                                addBraces: false,
+                                expression: firstStatement.expression,
+                                returnStatement: firstStatement
+                            }
                         };
                     }
                 }
@@ -137749,15 +145402,15 @@ var ts;
                     var parent = functionReference.parent;
                     switch (parent.kind) {
                         // foo(...) or super(...) or new Foo(...)
-                        case 196 /* CallExpression */:
-                        case 197 /* NewExpression */:
+                        case 203 /* CallExpression */:
+                        case 204 /* NewExpression */:
                             var callOrNewExpression = ts.tryCast(parent, ts.isCallOrNewExpression);
                             if (callOrNewExpression && callOrNewExpression.expression === functionReference) {
                                 return callOrNewExpression;
                             }
                             break;
                         // x.foo(...)
-                        case 194 /* PropertyAccessExpression */:
+                        case 201 /* PropertyAccessExpression */:
                             var propertyAccessExpression = ts.tryCast(parent, ts.isPropertyAccessExpression);
                             if (propertyAccessExpression && propertyAccessExpression.parent && propertyAccessExpression.name === functionReference) {
                                 var callOrNewExpression_1 = ts.tryCast(propertyAccessExpression.parent, ts.isCallOrNewExpression);
@@ -137767,7 +145420,7 @@ var ts;
                             }
                             break;
                         // x["foo"](...)
-                        case 195 /* ElementAccessExpression */:
+                        case 202 /* ElementAccessExpression */:
                             var elementAccessExpression = ts.tryCast(parent, ts.isElementAccessExpression);
                             if (elementAccessExpression && elementAccessExpression.parent && elementAccessExpression.argumentExpression === functionReference) {
                                 var callOrNewExpression_2 = ts.tryCast(elementAccessExpression.parent, ts.isCallOrNewExpression);
@@ -137786,14 +145439,14 @@ var ts;
                     var parent = reference.parent;
                     switch (parent.kind) {
                         // `C.foo`
-                        case 194 /* PropertyAccessExpression */:
+                        case 201 /* PropertyAccessExpression */:
                             var propertyAccessExpression = ts.tryCast(parent, ts.isPropertyAccessExpression);
                             if (propertyAccessExpression && propertyAccessExpression.expression === reference) {
                                 return propertyAccessExpression;
                             }
                             break;
                         // `C["foo"]`
-                        case 195 /* ElementAccessExpression */:
+                        case 202 /* ElementAccessExpression */:
                             var elementAccessExpression = ts.tryCast(parent, ts.isElementAccessExpression);
                             if (elementAccessExpression && elementAccessExpression.expression === reference) {
                                 return elementAccessExpression;
@@ -137835,11 +145488,11 @@ var ts;
                 if (!isValidParameterNodeArray(functionDeclaration.parameters, checker))
                     return false;
                 switch (functionDeclaration.kind) {
-                    case 244 /* FunctionDeclaration */:
+                    case 251 /* FunctionDeclaration */:
                         return hasNameOrDefault(functionDeclaration) && isSingleImplementation(functionDeclaration, checker);
-                    case 161 /* MethodDeclaration */:
+                    case 165 /* MethodDeclaration */:
                         return isSingleImplementation(functionDeclaration, checker);
-                    case 162 /* Constructor */:
+                    case 166 /* Constructor */:
                         if (ts.isClassDeclaration(functionDeclaration.parent)) {
                             return hasNameOrDefault(functionDeclaration.parent) && isSingleImplementation(functionDeclaration, checker);
                         }
@@ -137847,8 +145500,8 @@ var ts;
                             return isValidVariableDeclaration(functionDeclaration.parent.parent)
                                 && isSingleImplementation(functionDeclaration, checker);
                         }
-                    case 201 /* FunctionExpression */:
-                    case 202 /* ArrowFunction */:
+                    case 208 /* FunctionExpression */:
+                    case 209 /* ArrowFunction */:
                         return isValidVariableDeclaration(functionDeclaration.parent);
                 }
                 return false;
@@ -137858,7 +145511,7 @@ var ts;
             }
             function hasNameOrDefault(functionOrClassDeclaration) {
                 if (!functionOrClassDeclaration.name) {
-                    var defaultKeyword = ts.findModifier(functionOrClassDeclaration, 84 /* DefaultKeyword */);
+                    var defaultKeyword = ts.findModifier(functionOrClassDeclaration, 87 /* DefaultKeyword */);
                     return !!defaultKeyword;
                 }
                 return true;
@@ -137889,15 +145542,15 @@ var ts;
             }
             function getRefactorableParameters(parameters) {
                 if (hasThisParameter(parameters)) {
-                    parameters = ts.createNodeArray(parameters.slice(1), parameters.hasTrailingComma);
+                    parameters = ts.factory.createNodeArray(parameters.slice(1), parameters.hasTrailingComma);
                 }
                 return parameters;
             }
             function createPropertyOrShorthandAssignment(name, initializer) {
                 if (ts.isIdentifier(initializer) && ts.getTextOfIdentifierOrLiteral(initializer) === name) {
-                    return ts.createShorthandPropertyAssignment(name);
+                    return ts.factory.createShorthandPropertyAssignment(name);
                 }
-                return ts.createPropertyAssignment(name, initializer);
+                return ts.factory.createPropertyAssignment(name, initializer);
             }
             function createNewArgument(functionDeclaration, functionArguments) {
                 var parameters = getRefactorableParameters(functionDeclaration.parameters);
@@ -137914,31 +145567,31 @@ var ts;
                 });
                 if (hasRestParameter && functionArguments.length >= parameters.length) {
                     var restArguments = functionArguments.slice(parameters.length - 1);
-                    var restProperty = ts.createPropertyAssignment(getParameterName(ts.last(parameters)), ts.createArrayLiteral(restArguments));
+                    var restProperty = ts.factory.createPropertyAssignment(getParameterName(ts.last(parameters)), ts.factory.createArrayLiteralExpression(restArguments));
                     properties.push(restProperty);
                 }
-                var objectLiteral = ts.createObjectLiteral(properties, /*multiLine*/ false);
+                var objectLiteral = ts.factory.createObjectLiteralExpression(properties, /*multiLine*/ false);
                 return objectLiteral;
             }
             function createNewParameters(functionDeclaration, program, host) {
                 var checker = program.getTypeChecker();
                 var refactorableParameters = getRefactorableParameters(functionDeclaration.parameters);
                 var bindingElements = ts.map(refactorableParameters, createBindingElementFromParameterDeclaration);
-                var objectParameterName = ts.createObjectBindingPattern(bindingElements);
+                var objectParameterName = ts.factory.createObjectBindingPattern(bindingElements);
                 var objectParameterType = createParameterTypeNode(refactorableParameters);
                 var objectInitializer;
                 // If every parameter in the original function was optional, add an empty object initializer to the new object parameter
                 if (ts.every(refactorableParameters, isOptionalParameter)) {
-                    objectInitializer = ts.createObjectLiteral();
+                    objectInitializer = ts.factory.createObjectLiteralExpression();
                 }
-                var objectParameter = ts.createParameter(
+                var objectParameter = ts.factory.createParameterDeclaration(
                 /*decorators*/ undefined, 
                 /*modifiers*/ undefined, 
                 /*dotDotDotToken*/ undefined, objectParameterName, 
                 /*questionToken*/ undefined, objectParameterType, objectInitializer);
                 if (hasThisParameter(functionDeclaration.parameters)) {
                     var thisParameter = functionDeclaration.parameters[0];
-                    var newThisParameter = ts.createParameter(
+                    var newThisParameter = ts.factory.createParameterDeclaration(
                     /*decorators*/ undefined, 
                     /*modifiers*/ undefined, 
                     /*dotDotDotToken*/ undefined, thisParameter.name, 
@@ -137949,13 +145602,13 @@ var ts;
                         ts.suppressLeadingAndTrailingTrivia(newThisParameter.type);
                         ts.copyComments(thisParameter.type, newThisParameter.type);
                     }
-                    return ts.createNodeArray([newThisParameter, objectParameter]);
+                    return ts.factory.createNodeArray([newThisParameter, objectParameter]);
                 }
-                return ts.createNodeArray([objectParameter]);
+                return ts.factory.createNodeArray([objectParameter]);
                 function createBindingElementFromParameterDeclaration(parameterDeclaration) {
-                    var element = ts.createBindingElement(
+                    var element = ts.factory.createBindingElement(
                     /*dotDotDotToken*/ undefined, 
-                    /*propertyName*/ undefined, getParameterName(parameterDeclaration), ts.isRestParameter(parameterDeclaration) && isOptionalParameter(parameterDeclaration) ? ts.createArrayLiteral() : parameterDeclaration.initializer);
+                    /*propertyName*/ undefined, getParameterName(parameterDeclaration), ts.isRestParameter(parameterDeclaration) && isOptionalParameter(parameterDeclaration) ? ts.factory.createArrayLiteralExpression() : parameterDeclaration.initializer);
                     ts.suppressLeadingAndTrailingTrivia(element);
                     if (parameterDeclaration.initializer && element.initializer) {
                         ts.copyComments(parameterDeclaration.initializer, element.initializer);
@@ -137964,7 +145617,7 @@ var ts;
                 }
                 function createParameterTypeNode(parameters) {
                     var members = ts.map(parameters, createPropertySignatureFromParameterDeclaration);
-                    var typeNode = ts.addEmitFlags(ts.createTypeLiteralNode(members), 1 /* SingleLine */);
+                    var typeNode = ts.addEmitFlags(ts.factory.createTypeLiteralNode(members), 1 /* SingleLine */);
                     return typeNode;
                 }
                 function createPropertySignatureFromParameterDeclaration(parameterDeclaration) {
@@ -137972,9 +145625,8 @@ var ts;
                     if (!parameterType && (parameterDeclaration.initializer || ts.isRestParameter(parameterDeclaration))) {
                         parameterType = getTypeNode(parameterDeclaration);
                     }
-                    var propertySignature = ts.createPropertySignature(
-                    /*modifiers*/ undefined, getParameterName(parameterDeclaration), isOptionalParameter(parameterDeclaration) ? ts.createToken(57 /* QuestionToken */) : parameterDeclaration.questionToken, parameterType, 
-                    /*initializer*/ undefined);
+                    var propertySignature = ts.factory.createPropertySignature(
+                    /*modifiers*/ undefined, getParameterName(parameterDeclaration), isOptionalParameter(parameterDeclaration) ? ts.factory.createToken(57 /* QuestionToken */) : parameterDeclaration.questionToken, parameterType);
                     ts.suppressLeadingAndTrailingTrivia(propertySignature);
                     ts.copyComments(parameterDeclaration.name, propertySignature.name);
                     if (parameterDeclaration.type && propertySignature.type) {
@@ -137999,15 +145651,15 @@ var ts;
             }
             function getClassNames(constructorDeclaration) {
                 switch (constructorDeclaration.parent.kind) {
-                    case 245 /* ClassDeclaration */:
+                    case 252 /* ClassDeclaration */:
                         var classDeclaration = constructorDeclaration.parent;
                         if (classDeclaration.name)
                             return [classDeclaration.name];
                         // If the class declaration doesn't have a name, it should have a default modifier.
                         // We validated this in `isValidFunctionDeclaration` through `hasNameOrDefault`
-                        var defaultModifier = ts.Debug.checkDefined(ts.findModifier(classDeclaration, 84 /* DefaultKeyword */), "Nameless class declaration should be a default export");
+                        var defaultModifier = ts.Debug.checkDefined(ts.findModifier(classDeclaration, 87 /* DefaultKeyword */), "Nameless class declaration should be a default export");
                         return [defaultModifier];
-                    case 214 /* ClassExpression */:
+                    case 221 /* ClassExpression */:
                         var classExpression = constructorDeclaration.parent;
                         var variableDeclaration = constructorDeclaration.parent.parent;
                         var className = classExpression.name;
@@ -138018,25 +145670,25 @@ var ts;
             }
             function getFunctionNames(functionDeclaration) {
                 switch (functionDeclaration.kind) {
-                    case 244 /* FunctionDeclaration */:
+                    case 251 /* FunctionDeclaration */:
                         if (functionDeclaration.name)
                             return [functionDeclaration.name];
                         // If the function declaration doesn't have a name, it should have a default modifier.
                         // We validated this in `isValidFunctionDeclaration` through `hasNameOrDefault`
-                        var defaultModifier = ts.Debug.checkDefined(ts.findModifier(functionDeclaration, 84 /* DefaultKeyword */), "Nameless function declaration should be a default export");
+                        var defaultModifier = ts.Debug.checkDefined(ts.findModifier(functionDeclaration, 87 /* DefaultKeyword */), "Nameless function declaration should be a default export");
                         return [defaultModifier];
-                    case 161 /* MethodDeclaration */:
+                    case 165 /* MethodDeclaration */:
                         return [functionDeclaration.name];
-                    case 162 /* Constructor */:
-                        var ctrKeyword = ts.Debug.checkDefined(ts.findChildOfKind(functionDeclaration, 129 /* ConstructorKeyword */, functionDeclaration.getSourceFile()), "Constructor declaration should have constructor keyword");
-                        if (functionDeclaration.parent.kind === 214 /* ClassExpression */) {
+                    case 166 /* Constructor */:
+                        var ctrKeyword = ts.Debug.checkDefined(ts.findChildOfKind(functionDeclaration, 132 /* ConstructorKeyword */, functionDeclaration.getSourceFile()), "Constructor declaration should have constructor keyword");
+                        if (functionDeclaration.parent.kind === 221 /* ClassExpression */) {
                             var variableDeclaration = functionDeclaration.parent.parent;
                             return [variableDeclaration.name, ctrKeyword];
                         }
                         return [ctrKeyword];
-                    case 202 /* ArrowFunction */:
+                    case 209 /* ArrowFunction */:
                         return [functionDeclaration.parent.name];
-                    case 201 /* FunctionExpression */:
+                    case 208 /* FunctionExpression */:
                         if (functionDeclaration.name)
                             return [functionDeclaration.name, functionDeclaration.parent.name];
                         return [functionDeclaration.parent.name];
@@ -138112,10 +145764,18 @@ var ts;
                 return node.operatorToken.kind !== 62 /* EqualsToken */;
             }
             function getParentBinaryExpression(expr) {
-                while (ts.isBinaryExpression(expr.parent) && isNotEqualsOperator(expr.parent)) {
-                    expr = expr.parent;
-                }
-                return expr;
+                var container = ts.findAncestor(expr.parent, function (n) {
+                    switch (n.kind) {
+                        case 201 /* PropertyAccessExpression */:
+                        case 202 /* ElementAccessExpression */:
+                            return false;
+                        case 216 /* BinaryExpression */:
+                            return !(ts.isBinaryExpression(n.parent) && isNotEqualsOperator(n.parent));
+                        default:
+                            return "quit";
+                    }
+                });
+                return container || expr;
             }
             function isStringConcatenationValid(node) {
                 var _a = treeToArray(node), containsString = _a.containsString, areOperatorsValid = _a.areOperatorsValid;
@@ -138170,23 +145830,23 @@ var ts;
                 var copyCommentFromStringLiterals = copyCommentFromMultiNode(nodes, file, copyOperatorComments);
                 var _b = concatConsecutiveString(0, nodes), begin = _b[0], headText = _b[1], headIndexes = _b[2];
                 if (begin === nodes.length) {
-                    var noSubstitutionTemplateLiteral = ts.createNoSubstitutionTemplateLiteral(headText);
+                    var noSubstitutionTemplateLiteral = ts.factory.createNoSubstitutionTemplateLiteral(headText);
                     copyCommentFromStringLiterals(headIndexes, noSubstitutionTemplateLiteral);
                     return noSubstitutionTemplateLiteral;
                 }
                 var templateSpans = [];
-                var templateHead = ts.createTemplateHead(headText);
+                var templateHead = ts.factory.createTemplateHead(headText);
                 copyCommentFromStringLiterals(headIndexes, templateHead);
                 for (var i = begin; i < nodes.length; i++) {
                     var currentNode = getExpressionFromParenthesesOrExpression(nodes[i]);
                     copyOperatorComments(i, currentNode);
                     var _c = concatConsecutiveString(i + 1, nodes), newIndex = _c[0], subsequentText = _c[1], stringIndexes = _c[2];
                     i = newIndex - 1;
-                    var templatePart = i === nodes.length - 1 ? ts.createTemplateTail(subsequentText) : ts.createTemplateMiddle(subsequentText);
+                    var templatePart = i === nodes.length - 1 ? ts.factory.createTemplateTail(subsequentText) : ts.factory.createTemplateMiddle(subsequentText);
                     copyCommentFromStringLiterals(stringIndexes, templatePart);
-                    templateSpans.push(ts.createTemplateSpan(currentNode, templatePart));
+                    templateSpans.push(ts.factory.createTemplateSpan(currentNode, templatePart));
                 }
-                return ts.createTemplateExpression(templateHead, templateSpans);
+                return ts.factory.createTemplateExpression(templateHead, templateSpans);
             }
             // to copy comments following the opening & closing parentheses
             // "foo" + ( /* comment */ 5 + 5 ) /* comment */ + "bar"
@@ -138205,14 +145865,192 @@ var ts;
         })(convertStringOrTemplateLiteral = refactor.convertStringOrTemplateLiteral || (refactor.convertStringOrTemplateLiteral = {}));
     })(refactor = ts.refactor || (ts.refactor = {}));
 })(ts || (ts = {}));
+/* @internal */
+var ts;
+(function (ts) {
+    var refactor;
+    (function (refactor) {
+        var convertArrowFunctionOrFunctionExpression;
+        (function (convertArrowFunctionOrFunctionExpression) {
+            var refactorName = "Convert arrow function or function expression";
+            var refactorDescription = ts.getLocaleSpecificMessage(ts.Diagnostics.Convert_arrow_function_or_function_expression);
+            var toAnonymousFunctionActionName = "Convert to anonymous function";
+            var toNamedFunctionActionName = "Convert to named function";
+            var toArrowFunctionActionName = "Convert to arrow function";
+            var toAnonymousFunctionActionDescription = ts.getLocaleSpecificMessage(ts.Diagnostics.Convert_to_anonymous_function);
+            var toNamedFunctionActionDescription = ts.getLocaleSpecificMessage(ts.Diagnostics.Convert_to_named_function);
+            var toArrowFunctionActionDescription = ts.getLocaleSpecificMessage(ts.Diagnostics.Convert_to_arrow_function);
+            refactor.registerRefactor(refactorName, { getEditsForAction: getEditsForAction, getAvailableActions: getAvailableActions });
+            function getAvailableActions(context) {
+                var file = context.file, startPosition = context.startPosition, program = context.program;
+                var info = getFunctionInfo(file, startPosition, program);
+                if (!info)
+                    return ts.emptyArray;
+                var selectedVariableDeclaration = info.selectedVariableDeclaration, func = info.func;
+                var possibleActions = [];
+                if (selectedVariableDeclaration || (ts.isArrowFunction(func) && ts.isVariableDeclaration(func.parent))) {
+                    possibleActions.push({
+                        name: toNamedFunctionActionName,
+                        description: toNamedFunctionActionDescription
+                    });
+                }
+                if (!selectedVariableDeclaration && ts.isArrowFunction(func)) {
+                    possibleActions.push({
+                        name: toAnonymousFunctionActionName,
+                        description: toAnonymousFunctionActionDescription
+                    });
+                }
+                if (ts.isFunctionExpression(func)) {
+                    possibleActions.push({
+                        name: toArrowFunctionActionName,
+                        description: toArrowFunctionActionDescription
+                    });
+                }
+                return [{
+                        name: refactorName,
+                        description: refactorDescription,
+                        actions: possibleActions
+                    }];
+            }
+            function getEditsForAction(context, actionName) {
+                var file = context.file, startPosition = context.startPosition, program = context.program;
+                var info = getFunctionInfo(file, startPosition, program);
+                if (!info)
+                    return undefined;
+                var func = info.func;
+                var edits = [];
+                switch (actionName) {
+                    case toAnonymousFunctionActionName:
+                        edits.push.apply(edits, getEditInfoForConvertToAnonymousFunction(context, func));
+                        break;
+                    case toNamedFunctionActionName:
+                        var variableInfo = getVariableInfo(func);
+                        if (!variableInfo)
+                            return undefined;
+                        edits.push.apply(edits, getEditInfoForConvertToNamedFunction(context, func, variableInfo));
+                        break;
+                    case toArrowFunctionActionName:
+                        if (!ts.isFunctionExpression(func))
+                            return undefined;
+                        edits.push.apply(edits, getEditInfoForConvertToArrowFunction(context, func));
+                        break;
+                    default:
+                        return ts.Debug.fail("invalid action");
+                }
+                return { renameFilename: undefined, renameLocation: undefined, edits: edits };
+            }
+            function containingThis(node) {
+                var containsThis = false;
+                node.forEachChild(function checkThis(child) {
+                    if (ts.isThis(child)) {
+                        containsThis = true;
+                        return;
+                    }
+                    if (!ts.isClassLike(child) && !ts.isFunctionDeclaration(child) && !ts.isFunctionExpression(child)) {
+                        ts.forEachChild(child, checkThis);
+                    }
+                });
+                return containsThis;
+            }
+            function getFunctionInfo(file, startPosition, program) {
+                var token = ts.getTokenAtPosition(file, startPosition);
+                var arrowFunc = getArrowFunctionFromVariableDeclaration(token.parent);
+                if (arrowFunc && !containingThis(arrowFunc.body))
+                    return { selectedVariableDeclaration: true, func: arrowFunc };
+                var maybeFunc = ts.getContainingFunction(token);
+                var typeChecker = program.getTypeChecker();
+                if (maybeFunc &&
+                    (ts.isFunctionExpression(maybeFunc) || ts.isArrowFunction(maybeFunc)) &&
+                    !ts.rangeContainsRange(maybeFunc.body, token) &&
+                    !containingThis(maybeFunc.body)) {
+                    if ((ts.isFunctionExpression(maybeFunc) && maybeFunc.name && ts.FindAllReferences.Core.isSymbolReferencedInFile(maybeFunc.name, typeChecker, file)))
+                        return undefined;
+                    return { selectedVariableDeclaration: false, func: maybeFunc };
+                }
+                return undefined;
+            }
+            function isSingleVariableDeclaration(parent) {
+                return ts.isVariableDeclaration(parent) || (ts.isVariableDeclarationList(parent) && parent.declarations.length === 1);
+            }
+            function getArrowFunctionFromVariableDeclaration(parent) {
+                if (!isSingleVariableDeclaration(parent))
+                    return undefined;
+                var variableDeclaration = ts.isVariableDeclaration(parent) ? parent : parent.declarations[0];
+                var initializer = variableDeclaration.initializer;
+                if (!initializer || !ts.isArrowFunction(initializer))
+                    return undefined;
+                return initializer;
+            }
+            function convertToBlock(body) {
+                if (ts.isExpression(body)) {
+                    return ts.factory.createBlock([ts.factory.createReturnStatement(body)], /* multiLine */ true);
+                }
+                else {
+                    return body;
+                }
+            }
+            function getVariableInfo(func) {
+                var variableDeclaration = func.parent;
+                if (!ts.isVariableDeclaration(variableDeclaration) || !ts.isVariableDeclarationInVariableStatement(variableDeclaration))
+                    return undefined;
+                var variableDeclarationList = variableDeclaration.parent;
+                var statement = variableDeclarationList.parent;
+                if (!ts.isVariableDeclarationList(variableDeclarationList) || !ts.isVariableStatement(statement) || !ts.isIdentifier(variableDeclaration.name))
+                    return undefined;
+                return { variableDeclaration: variableDeclaration, variableDeclarationList: variableDeclarationList, statement: statement, name: variableDeclaration.name };
+            }
+            function getEditInfoForConvertToAnonymousFunction(context, func) {
+                var file = context.file;
+                var body = convertToBlock(func.body);
+                var newNode = ts.factory.createFunctionExpression(func.modifiers, func.asteriskToken, /* name */ undefined, func.typeParameters, func.parameters, func.type, body);
+                return ts.textChanges.ChangeTracker.with(context, function (t) { return t.replaceNode(file, func, newNode); });
+            }
+            function getEditInfoForConvertToNamedFunction(context, func, variableInfo) {
+                var file = context.file;
+                var body = convertToBlock(func.body);
+                var variableDeclaration = variableInfo.variableDeclaration, variableDeclarationList = variableInfo.variableDeclarationList, statement = variableInfo.statement, name = variableInfo.name;
+                ts.suppressLeadingTrivia(statement);
+                var newNode = ts.factory.createFunctionDeclaration(func.decorators, statement.modifiers, func.asteriskToken, name, func.typeParameters, func.parameters, func.type, body);
+                if (variableDeclarationList.declarations.length === 1) {
+                    return ts.textChanges.ChangeTracker.with(context, function (t) { return t.replaceNode(file, statement, newNode); });
+                }
+                else {
+                    return ts.textChanges.ChangeTracker.with(context, function (t) {
+                        t.delete(file, variableDeclaration);
+                        t.insertNodeAfter(file, statement, newNode);
+                    });
+                }
+            }
+            function getEditInfoForConvertToArrowFunction(context, func) {
+                var file = context.file;
+                var statements = func.body.statements;
+                var head = statements[0];
+                var body;
+                if (canBeConvertedToExpression(func.body, head)) {
+                    body = head.expression;
+                    ts.suppressLeadingAndTrailingTrivia(body);
+                    ts.copyComments(head, body);
+                }
+                else {
+                    body = func.body;
+                }
+                var newNode = ts.factory.createArrowFunction(func.modifiers, func.typeParameters, func.parameters, func.type, ts.factory.createToken(38 /* EqualsGreaterThanToken */), body);
+                return ts.textChanges.ChangeTracker.with(context, function (t) { return t.replaceNode(file, func, newNode); });
+            }
+            function canBeConvertedToExpression(body, head) {
+                return body.statements.length === 1 && ((ts.isReturnStatement(head) && !!head.expression));
+            }
+        })(convertArrowFunctionOrFunctionExpression = refactor.convertArrowFunctionOrFunctionExpression || (refactor.convertArrowFunctionOrFunctionExpression = {}));
+    })(refactor = ts.refactor || (ts.refactor = {}));
+})(ts || (ts = {}));
 var ts;
 (function (ts) {
     /** The version of the language service API */
     ts.servicesVersion = "0.8";
     function createNode(kind, pos, end, parent) {
         var node = ts.isNodeKind(kind) ? new NodeObject(kind, pos, end) :
-            kind === 75 /* Identifier */ ? new IdentifierObject(75 /* Identifier */, pos, end) :
-                kind === 76 /* PrivateIdentifier */ ? new PrivateIdentifierObject(76 /* PrivateIdentifier */, pos, end) :
+            kind === 78 /* Identifier */ ? new IdentifierObject(78 /* Identifier */, pos, end) :
+                kind === 79 /* PrivateIdentifier */ ? new PrivateIdentifierObject(79 /* PrivateIdentifier */, pos, end) :
                     new TokenObject(kind, pos, end);
         node.parent = parent;
         node.flags = parent.flags & 25358336 /* ContextFlags */;
@@ -138286,8 +146124,8 @@ var ts;
             if (!children.length) {
                 return undefined;
             }
-            var child = ts.find(children, function (kid) { return kid.kind < 294 /* FirstJSDocNode */ || kid.kind > 323 /* LastJSDocNode */; });
-            return child.kind < 153 /* FirstNode */ ?
+            var child = ts.find(children, function (kid) { return kid.kind < 301 /* FirstJSDocNode */ || kid.kind > 333 /* LastJSDocNode */; });
+            return child.kind < 157 /* FirstNode */ ?
                 child :
                 child.getFirstToken(sourceFile);
         };
@@ -138298,7 +146136,7 @@ var ts;
             if (!child) {
                 return undefined;
             }
-            return child.kind < 153 /* FirstNode */ ? child : child.getLastToken(sourceFile);
+            return child.kind < 157 /* FirstNode */ ? child : child.getLastToken(sourceFile);
         };
         NodeObject.prototype.forEachChild = function (cbNode, cbNodeArray) {
             return ts.forEachChild(this, cbNode, cbNodeArray);
@@ -138344,7 +146182,7 @@ var ts;
             var token = ts.scanner.scan();
             var textPos = ts.scanner.getTextPos();
             if (textPos <= end) {
-                if (token === 75 /* Identifier */) {
+                if (token === 78 /* Identifier */) {
                     ts.Debug.fail("Did not expect " + ts.Debug.formatSyntaxKind(parent.kind) + " to have an Identifier in its trivia");
                 }
                 nodes.push(createNode(token, pos, textPos, parent));
@@ -138356,7 +146194,7 @@ var ts;
         }
     }
     function createSyntaxList(nodes, parent) {
-        var list = createNode(324 /* SyntaxList */, nodes.pos, nodes.end, parent);
+        var list = createNode(334 /* SyntaxList */, nodes.pos, nodes.end, parent);
         list._children = [];
         var pos = nodes.pos;
         for (var _i = 0, nodes_1 = nodes; _i < nodes_1.length; _i++) {
@@ -138455,10 +146293,34 @@ var ts;
         SymbolObject.prototype.getDocumentationComment = function (checker) {
             if (!this.documentationComment) {
                 this.documentationComment = ts.emptyArray; // Set temporarily to avoid an infinite loop finding inherited docs
-                this.documentationComment = getDocumentationComment(this.declarations, checker);
+                if (!this.declarations && this.target && this.target.tupleLabelDeclaration) {
+                    var labelDecl = this.target.tupleLabelDeclaration;
+                    this.documentationComment = getDocumentationComment([labelDecl], checker);
+                }
+                else {
+                    this.documentationComment = getDocumentationComment(this.declarations, checker);
+                }
             }
             return this.documentationComment;
         };
+        SymbolObject.prototype.getContextualDocumentationComment = function (context, checker) {
+            switch (context === null || context === void 0 ? void 0 : context.kind) {
+                case 167 /* GetAccessor */:
+                    if (!this.contextualGetAccessorDocumentationComment) {
+                        this.contextualGetAccessorDocumentationComment = ts.emptyArray;
+                        this.contextualGetAccessorDocumentationComment = getDocumentationComment(ts.filter(this.declarations, ts.isGetAccessor), checker);
+                    }
+                    return this.contextualGetAccessorDocumentationComment;
+                case 168 /* SetAccessor */:
+                    if (!this.contextualSetAccessorDocumentationComment) {
+                        this.contextualSetAccessorDocumentationComment = ts.emptyArray;
+                        this.contextualSetAccessorDocumentationComment = getDocumentationComment(ts.filter(this.declarations, ts.isSetAccessor), checker);
+                    }
+                    return this.contextualSetAccessorDocumentationComment;
+                default:
+                    return this.getDocumentationComment(checker);
+            }
+        };
         SymbolObject.prototype.getJsDocTags = function () {
             if (this.tags === undefined) {
                 this.tags = ts.JsDoc.getJsDocTagsFromDeclarations(this.declarations);
@@ -138480,7 +146342,7 @@ var ts;
         __extends(IdentifierObject, _super);
         function IdentifierObject(_kind, pos, end) {
             var _this = _super.call(this, pos, end) || this;
-            _this.kind = 75 /* Identifier */;
+            _this.kind = 78 /* Identifier */;
             return _this;
         }
         Object.defineProperty(IdentifierObject.prototype, "text", {
@@ -138492,7 +146354,7 @@ var ts;
         });
         return IdentifierObject;
     }(TokenOrIdentifierObject));
-    IdentifierObject.prototype.kind = 75 /* Identifier */;
+    IdentifierObject.prototype.kind = 78 /* Identifier */;
     var PrivateIdentifierObject = /** @class */ (function (_super) {
         __extends(PrivateIdentifierObject, _super);
         function PrivateIdentifierObject(_kind, pos, end) {
@@ -138507,7 +146369,7 @@ var ts;
         });
         return PrivateIdentifierObject;
     }(TokenOrIdentifierObject));
-    PrivateIdentifierObject.prototype.kind = 76 /* PrivateIdentifier */;
+    PrivateIdentifierObject.prototype.kind = 79 /* PrivateIdentifier */;
     var TypeObject = /** @class */ (function () {
         function TypeObject(checker, flags) {
             this.checker = checker;
@@ -138670,7 +146532,7 @@ var ts;
         __extends(SourceFileObject, _super);
         function SourceFileObject(kind, pos, end) {
             var _this = _super.call(this, kind, pos, end) || this;
-            _this.kind = 290 /* SourceFile */;
+            _this.kind = 297 /* SourceFile */;
             return _this;
         }
         SourceFileObject.prototype.update = function (newText, textChangeRange) {
@@ -138729,10 +146591,10 @@ var ts;
             }
             function visit(node) {
                 switch (node.kind) {
-                    case 244 /* FunctionDeclaration */:
-                    case 201 /* FunctionExpression */:
-                    case 161 /* MethodDeclaration */:
-                    case 160 /* MethodSignature */:
+                    case 251 /* FunctionDeclaration */:
+                    case 208 /* FunctionExpression */:
+                    case 165 /* MethodDeclaration */:
+                    case 164 /* MethodSignature */:
                         var functionDeclaration = node;
                         var declarationName = getDeclarationName(functionDeclaration);
                         if (declarationName) {
@@ -138752,31 +146614,31 @@ var ts;
                         }
                         ts.forEachChild(node, visit);
                         break;
-                    case 245 /* ClassDeclaration */:
-                    case 214 /* ClassExpression */:
-                    case 246 /* InterfaceDeclaration */:
-                    case 247 /* TypeAliasDeclaration */:
-                    case 248 /* EnumDeclaration */:
-                    case 249 /* ModuleDeclaration */:
-                    case 253 /* ImportEqualsDeclaration */:
-                    case 263 /* ExportSpecifier */:
-                    case 258 /* ImportSpecifier */:
-                    case 255 /* ImportClause */:
-                    case 256 /* NamespaceImport */:
-                    case 163 /* GetAccessor */:
-                    case 164 /* SetAccessor */:
-                    case 173 /* TypeLiteral */:
+                    case 252 /* ClassDeclaration */:
+                    case 221 /* ClassExpression */:
+                    case 253 /* InterfaceDeclaration */:
+                    case 254 /* TypeAliasDeclaration */:
+                    case 255 /* EnumDeclaration */:
+                    case 256 /* ModuleDeclaration */:
+                    case 260 /* ImportEqualsDeclaration */:
+                    case 270 /* ExportSpecifier */:
+                    case 265 /* ImportSpecifier */:
+                    case 262 /* ImportClause */:
+                    case 263 /* NamespaceImport */:
+                    case 167 /* GetAccessor */:
+                    case 168 /* SetAccessor */:
+                    case 177 /* TypeLiteral */:
                         addDeclaration(node);
                         ts.forEachChild(node, visit);
                         break;
-                    case 156 /* Parameter */:
+                    case 160 /* Parameter */:
                         // Only consider parameter properties
-                        if (!ts.hasModifier(node, 92 /* ParameterPropertyModifier */)) {
+                        if (!ts.hasSyntacticModifier(node, 92 /* ParameterPropertyModifier */)) {
                             break;
                         }
                     // falls through
-                    case 242 /* VariableDeclaration */:
-                    case 191 /* BindingElement */: {
+                    case 249 /* VariableDeclaration */:
+                    case 198 /* BindingElement */: {
                         var decl = node;
                         if (ts.isBindingPattern(decl.name)) {
                             ts.forEachChild(decl.name, visit);
@@ -138787,12 +146649,12 @@ var ts;
                         }
                     }
                     // falls through
-                    case 284 /* EnumMember */:
-                    case 159 /* PropertyDeclaration */:
-                    case 158 /* PropertySignature */:
+                    case 291 /* EnumMember */:
+                    case 163 /* PropertyDeclaration */:
+                    case 162 /* PropertySignature */:
                         addDeclaration(node);
                         break;
-                    case 260 /* ExportDeclaration */:
+                    case 267 /* ExportDeclaration */:
                         // Handle named exports case e.g.:
                         //    export {a, b as B} from "mod";
                         var exportDeclaration = node;
@@ -138805,7 +146667,7 @@ var ts;
                             }
                         }
                         break;
-                    case 254 /* ImportDeclaration */:
+                    case 261 /* ImportDeclaration */:
                         var importClause = node.importClause;
                         if (importClause) {
                             // Handle default import case e.g.:
@@ -138817,7 +146679,7 @@ var ts;
                             //    import * as NS from "mod";
                             //    import {a, b as B} from "mod";
                             if (importClause.namedBindings) {
-                                if (importClause.namedBindings.kind === 256 /* NamespaceImport */) {
+                                if (importClause.namedBindings.kind === 263 /* NamespaceImport */) {
                                     addDeclaration(importClause.namedBindings);
                                 }
                                 else {
@@ -138826,7 +146688,7 @@ var ts;
                             }
                         }
                         break;
-                    case 209 /* BinaryExpression */:
+                    case 216 /* BinaryExpression */:
                         if (ts.getAssignmentDeclarationKind(node) !== 0 /* None */) {
                             addDeclaration(node);
                         }
@@ -138913,7 +146775,7 @@ var ts;
             this.host = host;
             // script id => script index
             this.currentDirectory = host.getCurrentDirectory();
-            this.fileNameToEntry = ts.createMap();
+            this.fileNameToEntry = new ts.Map();
             // Initialize the list with the root file names
             var rootFileNames = host.getScriptFileNames();
             for (var _i = 0, rootFileNames_1 = rootFileNames; _i < rootFileNames_1.length; _i++) {
@@ -139109,10 +146971,58 @@ var ts;
         return ThrottledCancellationToken;
     }());
     ts.ThrottledCancellationToken = ThrottledCancellationToken;
-    function createLanguageService(host, documentRegistry, syntaxOnly) {
+    var invalidOperationsInPartialSemanticMode = [
+        "getSyntacticDiagnostics",
+        "getSemanticDiagnostics",
+        "getSuggestionDiagnostics",
+        "getCompilerOptionsDiagnostics",
+        "getSemanticClassifications",
+        "getEncodedSemanticClassifications",
+        "getCodeFixesAtPosition",
+        "getCombinedCodeFix",
+        "applyCodeActionCommand",
+        "organizeImports",
+        "getEditsForFileRename",
+        "getEmitOutput",
+        "getApplicableRefactors",
+        "getEditsForRefactor",
+        "prepareCallHierarchy",
+        "provideCallHierarchyIncomingCalls",
+        "provideCallHierarchyOutgoingCalls",
+    ];
+    var invalidOperationsInSyntacticMode = __spreadArrays(invalidOperationsInPartialSemanticMode, [
+        "getCompletionsAtPosition",
+        "getCompletionEntryDetails",
+        "getCompletionEntrySymbol",
+        "getSignatureHelpItems",
+        "getQuickInfoAtPosition",
+        "getDefinitionAtPosition",
+        "getDefinitionAndBoundSpan",
+        "getImplementationAtPosition",
+        "getTypeDefinitionAtPosition",
+        "getReferencesAtPosition",
+        "findReferences",
+        "getOccurrencesAtPosition",
+        "getDocumentHighlights",
+        "getNavigateToItems",
+        "getRenameInfo",
+        "findRenameLocations",
+        "getApplicableRefactors",
+    ]);
+    function createLanguageService(host, documentRegistry, syntaxOnlyOrLanguageServiceMode) {
         var _a;
         if (documentRegistry === void 0) { documentRegistry = ts.createDocumentRegistry(host.useCaseSensitiveFileNames && host.useCaseSensitiveFileNames(), host.getCurrentDirectory()); }
-        if (syntaxOnly === void 0) { syntaxOnly = false; }
+        var languageServiceMode;
+        if (syntaxOnlyOrLanguageServiceMode === undefined) {
+            languageServiceMode = ts.LanguageServiceMode.Semantic;
+        }
+        else if (typeof syntaxOnlyOrLanguageServiceMode === "boolean") {
+            // languageServiceMode = SyntaxOnly
+            languageServiceMode = syntaxOnlyOrLanguageServiceMode ? ts.LanguageServiceMode.Syntactic : ts.LanguageServiceMode.Semantic;
+        }
+        else {
+            languageServiceMode = syntaxOnlyOrLanguageServiceMode;
+        }
         var syntaxTreeCache = new SyntaxTreeCache(host);
         var program;
         var lastProjectVersion;
@@ -139152,13 +147062,13 @@ var ts;
             return sourceFile;
         }
         function synchronizeHostData() {
-            var _a;
-            ts.Debug.assert(!syntaxOnly);
+            var _a, _b;
+            ts.Debug.assert(languageServiceMode !== ts.LanguageServiceMode.Syntactic);
             // perform fast check if host supports it
             if (host.getProjectVersion) {
                 var hostProjectVersion = host.getProjectVersion();
                 if (hostProjectVersion) {
-                    if (lastProjectVersion === hostProjectVersion && !host.hasChangedAutomaticTypeDirectiveNames) {
+                    if (lastProjectVersion === hostProjectVersion && !((_a = host.hasChangedAutomaticTypeDirectiveNames) === null || _a === void 0 ? void 0 : _a.call(host))) {
                         return;
                     }
                     lastProjectVersion = hostProjectVersion;
@@ -139174,9 +147084,10 @@ var ts;
             var hostCache = new HostCache(host, getCanonicalFileName);
             var rootFileNames = hostCache.getRootFileNames();
             var hasInvalidatedResolution = host.hasInvalidatedResolution || ts.returnFalse;
+            var hasChangedAutomaticTypeDirectiveNames = ts.maybeBind(host, host.hasChangedAutomaticTypeDirectiveNames);
             var projectReferences = hostCache.getProjectReferences();
             // If the program is already up-to-date, we can reuse it
-            if (ts.isProgramUptoDate(program, rootFileNames, hostCache.compilationSettings(), function (_path, fileName) { return host.getScriptVersion(fileName); }, fileExists, hasInvalidatedResolution, !!host.hasChangedAutomaticTypeDirectiveNames, projectReferences)) {
+            if (ts.isProgramUptoDate(program, rootFileNames, hostCache.compilationSettings(), function (_path, fileName) { return host.getScriptVersion(fileName); }, fileExists, hasInvalidatedResolution, hasChangedAutomaticTypeDirectiveNames, projectReferences)) {
                 return;
             }
             // IMPORTANT - It is critical from this moment onward that we do not check
@@ -139198,7 +147109,8 @@ var ts;
                 getCurrentDirectory: function () { return currentDirectory; },
                 fileExists: fileExists,
                 readFile: readFile,
-                realpath: host.realpath && (function (path) { return host.realpath(path); }),
+                getSymlinkCache: ts.maybeBind(host, host.getSymlinkCache),
+                realpath: ts.maybeBind(host, host.realpath),
                 directoryExists: function (directoryName) {
                     return ts.directoryProbablyExists(directoryName, host);
                 },
@@ -139211,33 +147123,13 @@ var ts;
                 },
                 onReleaseOldSourceFile: onReleaseOldSourceFile,
                 hasInvalidatedResolution: hasInvalidatedResolution,
-                hasChangedAutomaticTypeDirectiveNames: host.hasChangedAutomaticTypeDirectiveNames
+                hasChangedAutomaticTypeDirectiveNames: hasChangedAutomaticTypeDirectiveNames,
+                trace: ts.maybeBind(host, host.trace),
+                resolveModuleNames: ts.maybeBind(host, host.resolveModuleNames),
+                resolveTypeReferenceDirectives: ts.maybeBind(host, host.resolveTypeReferenceDirectives),
+                useSourceOfProjectReferenceRedirect: ts.maybeBind(host, host.useSourceOfProjectReferenceRedirect),
             };
-            if (host.trace) {
-                compilerHost.trace = function (message) { return host.trace(message); };
-            }
-            if (host.resolveModuleNames) {
-                compilerHost.resolveModuleNames = function () {
-                    var args = [];
-                    for (var _i = 0; _i < arguments.length; _i++) {
-                        args[_i] = arguments[_i];
-                    }
-                    return host.resolveModuleNames.apply(host, args);
-                };
-            }
-            if (host.resolveTypeReferenceDirectives) {
-                compilerHost.resolveTypeReferenceDirectives = function () {
-                    var args = [];
-                    for (var _i = 0; _i < arguments.length; _i++) {
-                        args[_i] = arguments[_i];
-                    }
-                    return host.resolveTypeReferenceDirectives.apply(host, args);
-                };
-            }
-            if (host.useSourceOfProjectReferenceRedirect) {
-                compilerHost.useSourceOfProjectReferenceRedirect = function () { return host.useSourceOfProjectReferenceRedirect(); };
-            }
-            (_a = host.setCompilerHost) === null || _a === void 0 ? void 0 : _a.call(host, compilerHost);
+            (_b = host.setCompilerHost) === null || _b === void 0 ? void 0 : _b.call(host, compilerHost);
             var documentRegistryBucketKey = documentRegistry.getKeyForCompilationSettings(newSettings);
             var options = {
                 rootNames: rootFileNames,
@@ -139311,7 +147203,7 @@ var ts;
                         //
                         // Each LS has a reference to file 'foo.ts' at version 1.  LS2 then updates
                         // it's version of 'foo.ts' to version 2.  This will cause LS2 and the
-                        // DocumentRegistry to have version 2 of the document.  HOwever, LS1 will
+                        // DocumentRegistry to have version 2 of the document.  However, LS1 will
                         // have version 1.  And *importantly* this source file will be *corrupt*.
                         // The act of creating version 2 of the file irrevocably damages the version
                         // 1 file.
@@ -139334,19 +147226,23 @@ var ts;
         }
         // TODO: GH#18217 frequently asserted as defined
         function getProgram() {
-            if (syntaxOnly) {
+            if (languageServiceMode === ts.LanguageServiceMode.Syntactic) {
                 ts.Debug.assert(program === undefined);
                 return undefined;
             }
             synchronizeHostData();
             return program;
         }
+        function getAutoImportProvider() {
+            var _a;
+            return (_a = host.getPackageJsonAutoImportProvider) === null || _a === void 0 ? void 0 : _a.call(host);
+        }
         function cleanupSemanticCache() {
             program = undefined; // TODO: GH#18217
         }
         function dispose() {
             if (program) {
-                // Use paths to ensure we are using correct key and paths as document registry could bre created with different current directory than host
+                // Use paths to ensure we are using correct key and paths as document registry could be created with different current directory than host
                 var key_1 = documentRegistry.getKeyForCompilationSettings(program.getCompilerOptions());
                 ts.forEach(program.getSourceFiles(), function (f) {
                     return documentRegistry.releaseDocumentWithKey(f.resolvedPath, key_1);
@@ -139445,15 +147341,15 @@ var ts;
         }
         function shouldGetType(sourceFile, node, position) {
             switch (node.kind) {
-                case 75 /* Identifier */:
+                case 78 /* Identifier */:
                     return !ts.isLabelName(node) && !ts.isTagName(node);
-                case 194 /* PropertyAccessExpression */:
-                case 153 /* QualifiedName */:
+                case 201 /* PropertyAccessExpression */:
+                case 157 /* QualifiedName */:
                     // Don't return quickInfo if inside the comment in `a/**/.b`
                     return !ts.isInComment(sourceFile, position);
-                case 104 /* ThisKeyword */:
-                case 183 /* ThisType */:
-                case 102 /* SuperKeyword */:
+                case 107 /* ThisKeyword */:
+                case 187 /* ThisType */:
+                case 105 /* SuperKeyword */:
                     return true;
                 default:
                     return false;
@@ -139537,7 +147433,7 @@ var ts;
          * This is a semantic operation.
          */
         function getSignatureHelpItems(fileName, position, _a) {
-            var triggerReason = (_a === void 0 ? ts.emptyOptions : _a).triggerReason;
+            var _b = _a === void 0 ? ts.emptyOptions : _a, triggerReason = _b.triggerReason;
             synchronizeHostData();
             var sourceFile = getValidSourceFile(fileName);
             return ts.SignatureHelp.getSignatureHelpItems(program, sourceFile, position, triggerReason, cancellationToken);
@@ -139554,16 +147450,16 @@ var ts;
                 return undefined;
             }
             switch (node.kind) {
-                case 194 /* PropertyAccessExpression */:
-                case 153 /* QualifiedName */:
+                case 201 /* PropertyAccessExpression */:
+                case 157 /* QualifiedName */:
                 case 10 /* StringLiteral */:
-                case 91 /* FalseKeyword */:
-                case 106 /* TrueKeyword */:
-                case 100 /* NullKeyword */:
-                case 102 /* SuperKeyword */:
-                case 104 /* ThisKeyword */:
-                case 183 /* ThisType */:
-                case 75 /* Identifier */:
+                case 94 /* FalseKeyword */:
+                case 109 /* TrueKeyword */:
+                case 103 /* NullKeyword */:
+                case 105 /* SuperKeyword */:
+                case 107 /* ThisKeyword */:
+                case 187 /* ThisType */:
+                case 78 /* Identifier */:
                     break;
                 // Cant create the text span
                 default:
@@ -139579,7 +147475,7 @@ var ts;
                     // If this is name of a module declarations, check if this is right side of dotted module name
                     // If parent of the module declaration which is parent of this node is module declaration and its body is the module declaration that this node is name of
                     // Then this name is name from dotted module
-                    if (nodeForStartPos.parent.parent.kind === 249 /* ModuleDeclaration */ &&
+                    if (nodeForStartPos.parent.parent.kind === 256 /* ModuleDeclaration */ &&
                         nodeForStartPos.parent.parent.body === nodeForStartPos.parent) {
                         // Use parent module declarations name for start pos
                         nodeForStartPos = nodeForStartPos.parent.parent.name;
@@ -139611,21 +147507,33 @@ var ts;
             var kind = ts.getScriptKind(fileName, host);
             return kind === 3 /* TS */ || kind === 4 /* TSX */;
         }
-        function getSemanticClassifications(fileName, span) {
+        function getSemanticClassifications(fileName, span, format) {
             if (!isTsOrTsxFile(fileName)) {
                 // do not run semantic classification on non-ts-or-tsx files
                 return [];
             }
             synchronizeHostData();
-            return ts.getSemanticClassifications(program.getTypeChecker(), cancellationToken, getValidSourceFile(fileName), program.getClassifiableNames(), span);
+            var responseFormat = format || "original" /* Original */;
+            if (responseFormat === "2020" /* TwentyTwenty */) {
+                return ts.classifier.v2020.getSemanticClassifications(program, cancellationToken, getValidSourceFile(fileName), span);
+            }
+            else {
+                return ts.getSemanticClassifications(program.getTypeChecker(), cancellationToken, getValidSourceFile(fileName), program.getClassifiableNames(), span);
+            }
         }
-        function getEncodedSemanticClassifications(fileName, span) {
+        function getEncodedSemanticClassifications(fileName, span, format) {
             if (!isTsOrTsxFile(fileName)) {
                 // do not run semantic classification on non-ts-or-tsx files
                 return { spans: [], endOfLineState: 0 /* None */ };
             }
             synchronizeHostData();
-            return ts.getEncodedSemanticClassifications(program.getTypeChecker(), cancellationToken, getValidSourceFile(fileName), program.getClassifiableNames(), span);
+            var responseFormat = format || "original" /* Original */;
+            if (responseFormat === "original" /* Original */) {
+                return ts.getEncodedSemanticClassifications(program.getTypeChecker(), cancellationToken, getValidSourceFile(fileName), program.getClassifiableNames(), span);
+            }
+            else {
+                return ts.classifier.v2020.getEncodedSemanticClassifications(program, cancellationToken, getValidSourceFile(fileName), span);
+            }
         }
         function getSyntacticClassifications(fileName, span) {
             // doesn't use compiler - no need to synchronize with host
@@ -139640,12 +147548,12 @@ var ts;
             var sourceFile = syntaxTreeCache.getCurrentSourceFile(fileName);
             return ts.OutliningElementsCollector.collectElements(sourceFile, cancellationToken);
         }
-        var braceMatching = ts.createMapFromTemplate((_a = {},
+        var braceMatching = new ts.Map(ts.getEntries((_a = {},
             _a[18 /* OpenBraceToken */] = 19 /* CloseBraceToken */,
             _a[20 /* OpenParenToken */] = 21 /* CloseParenToken */,
             _a[22 /* OpenBracketToken */] = 23 /* CloseBracketToken */,
             _a[31 /* GreaterThanToken */] = 29 /* LessThanToken */,
-            _a));
+            _a)));
         braceMatching.forEach(function (value, key) { return braceMatching.set(value.toString(), Number(key)); });
         function getBraceMatchingAtPosition(fileName, position) {
             var sourceFile = syntaxTreeCache.getCurrentSourceFile(fileName);
@@ -139774,6 +147682,211 @@ var ts;
                 return { newText: "</" + element.openingElement.tagName.getText(sourceFile) + ">" };
             }
         }
+        function getLinesForRange(sourceFile, textRange) {
+            return {
+                lineStarts: sourceFile.getLineStarts(),
+                firstLine: sourceFile.getLineAndCharacterOfPosition(textRange.pos).line,
+                lastLine: sourceFile.getLineAndCharacterOfPosition(textRange.end).line
+            };
+        }
+        function toggleLineComment(fileName, textRange, insertComment) {
+            var sourceFile = syntaxTreeCache.getCurrentSourceFile(fileName);
+            var textChanges = [];
+            var _a = getLinesForRange(sourceFile, textRange), lineStarts = _a.lineStarts, firstLine = _a.firstLine, lastLine = _a.lastLine;
+            var isCommenting = insertComment || false;
+            var leftMostPosition = Number.MAX_VALUE;
+            var lineTextStarts = new ts.Map();
+            var firstNonWhitespaceCharacterRegex = new RegExp(/\S/);
+            var isJsx = ts.isInsideJsxElement(sourceFile, lineStarts[firstLine]);
+            var openComment = isJsx ? "{/*" : "//";
+            // Check each line before any text changes.
+            for (var i = firstLine; i <= lastLine; i++) {
+                var lineText = sourceFile.text.substring(lineStarts[i], sourceFile.getLineEndOfPosition(lineStarts[i]));
+                // Find the start of text and the left-most character. No-op on empty lines.
+                var regExec = firstNonWhitespaceCharacterRegex.exec(lineText);
+                if (regExec) {
+                    leftMostPosition = Math.min(leftMostPosition, regExec.index);
+                    lineTextStarts.set(i.toString(), regExec.index);
+                    if (lineText.substr(regExec.index, openComment.length) !== openComment) {
+                        isCommenting = insertComment === undefined || insertComment;
+                    }
+                }
+            }
+            // Push all text changes.
+            for (var i = firstLine; i <= lastLine; i++) {
+                // If the range is multiline and ends on a beginning of a line, don't comment/uncomment.
+                if (firstLine !== lastLine && lineStarts[i] === textRange.end) {
+                    continue;
+                }
+                var lineTextStart = lineTextStarts.get(i.toString());
+                // If the line is not an empty line; otherwise no-op.
+                if (lineTextStart !== undefined) {
+                    if (isJsx) {
+                        textChanges.push.apply(textChanges, toggleMultilineComment(fileName, { pos: lineStarts[i] + leftMostPosition, end: sourceFile.getLineEndOfPosition(lineStarts[i]) }, isCommenting, isJsx));
+                    }
+                    else if (isCommenting) {
+                        textChanges.push({
+                            newText: openComment,
+                            span: {
+                                length: 0,
+                                start: lineStarts[i] + leftMostPosition
+                            }
+                        });
+                    }
+                    else if (sourceFile.text.substr(lineStarts[i] + lineTextStart, openComment.length) === openComment) {
+                        textChanges.push({
+                            newText: "",
+                            span: {
+                                length: openComment.length,
+                                start: lineStarts[i] + lineTextStart
+                            }
+                        });
+                    }
+                }
+            }
+            return textChanges;
+        }
+        function toggleMultilineComment(fileName, textRange, insertComment, isInsideJsx) {
+            var _a;
+            var sourceFile = syntaxTreeCache.getCurrentSourceFile(fileName);
+            var textChanges = [];
+            var text = sourceFile.text;
+            var hasComment = false;
+            var isCommenting = insertComment || false;
+            var positions = [];
+            var pos = textRange.pos;
+            var isJsx = isInsideJsx !== undefined ? isInsideJsx : ts.isInsideJsxElement(sourceFile, pos);
+            var openMultiline = isJsx ? "{/*" : "/*";
+            var closeMultiline = isJsx ? "*/}" : "*/";
+            var openMultilineRegex = isJsx ? "\\{\\/\\*" : "\\/\\*";
+            var closeMultilineRegex = isJsx ? "\\*\\/\\}" : "\\*\\/";
+            // Get all comment positions
+            while (pos <= textRange.end) {
+                // Start of comment is considered inside comment.
+                var offset = text.substr(pos, openMultiline.length) === openMultiline ? openMultiline.length : 0;
+                var commentRange = ts.isInComment(sourceFile, pos + offset);
+                // If position is in a comment add it to the positions array.
+                if (commentRange) {
+                    // Comment range doesn't include the brace character. Increase it to include them.
+                    if (isJsx) {
+                        commentRange.pos--;
+                        commentRange.end++;
+                    }
+                    positions.push(commentRange.pos);
+                    if (commentRange.kind === 3 /* MultiLineCommentTrivia */) {
+                        positions.push(commentRange.end);
+                    }
+                    hasComment = true;
+                    pos = commentRange.end + 1;
+                }
+                else { // If it's not in a comment range, then we need to comment the uncommented portions.
+                    var newPos = text.substring(pos, textRange.end).search("(" + openMultilineRegex + ")|(" + closeMultilineRegex + ")");
+                    isCommenting = insertComment !== undefined
+                        ? insertComment
+                        : isCommenting || !ts.isTextWhiteSpaceLike(text, pos, newPos === -1 ? textRange.end : pos + newPos); // If isCommenting is already true we don't need to check whitespace again.
+                    pos = newPos === -1 ? textRange.end + 1 : pos + newPos + closeMultiline.length;
+                }
+            }
+            // If it didn't found a comment and isCommenting is false means is only empty space.
+            // We want to insert comment in this scenario.
+            if (isCommenting || !hasComment) {
+                if (((_a = ts.isInComment(sourceFile, textRange.pos)) === null || _a === void 0 ? void 0 : _a.kind) !== 2 /* SingleLineCommentTrivia */) {
+                    ts.insertSorted(positions, textRange.pos, ts.compareValues);
+                }
+                ts.insertSorted(positions, textRange.end, ts.compareValues);
+                // Insert open comment if the first position is not a comment already.
+                var firstPos = positions[0];
+                if (text.substr(firstPos, openMultiline.length) !== openMultiline) {
+                    textChanges.push({
+                        newText: openMultiline,
+                        span: {
+                            length: 0,
+                            start: firstPos
+                        }
+                    });
+                }
+                // Insert open and close comment to all positions between first and last. Exclusive.
+                for (var i = 1; i < positions.length - 1; i++) {
+                    if (text.substr(positions[i] - closeMultiline.length, closeMultiline.length) !== closeMultiline) {
+                        textChanges.push({
+                            newText: closeMultiline,
+                            span: {
+                                length: 0,
+                                start: positions[i]
+                            }
+                        });
+                    }
+                    if (text.substr(positions[i], openMultiline.length) !== openMultiline) {
+                        textChanges.push({
+                            newText: openMultiline,
+                            span: {
+                                length: 0,
+                                start: positions[i]
+                            }
+                        });
+                    }
+                }
+                // Insert open comment if the last position is not a comment already.
+                if (textChanges.length % 2 !== 0) {
+                    textChanges.push({
+                        newText: closeMultiline,
+                        span: {
+                            length: 0,
+                            start: positions[positions.length - 1]
+                        }
+                    });
+                }
+            }
+            else {
+                // If is not commenting then remove all comments found.
+                for (var _i = 0, positions_1 = positions; _i < positions_1.length; _i++) {
+                    var pos_1 = positions_1[_i];
+                    var from = pos_1 - closeMultiline.length > 0 ? pos_1 - closeMultiline.length : 0;
+                    var offset = text.substr(from, closeMultiline.length) === closeMultiline ? closeMultiline.length : 0;
+                    textChanges.push({
+                        newText: "",
+                        span: {
+                            length: openMultiline.length,
+                            start: pos_1 - offset
+                        }
+                    });
+                }
+            }
+            return textChanges;
+        }
+        function commentSelection(fileName, textRange) {
+            var sourceFile = syntaxTreeCache.getCurrentSourceFile(fileName);
+            var _a = getLinesForRange(sourceFile, textRange), firstLine = _a.firstLine, lastLine = _a.lastLine;
+            // If there is a selection that is on the same line, add multiline.
+            return firstLine === lastLine && textRange.pos !== textRange.end
+                ? toggleMultilineComment(fileName, textRange, /*insertComment*/ true)
+                : toggleLineComment(fileName, textRange, /*insertComment*/ true);
+        }
+        function uncommentSelection(fileName, textRange) {
+            var sourceFile = syntaxTreeCache.getCurrentSourceFile(fileName);
+            var textChanges = [];
+            var pos = textRange.pos;
+            var end = textRange.end;
+            // If cursor is not a selection we need to increase the end position
+            // to include the start of the comment.
+            if (pos === end) {
+                end += ts.isInsideJsxElement(sourceFile, pos) ? 2 : 1;
+            }
+            for (var i = pos; i <= end; i++) {
+                var commentRange = ts.isInComment(sourceFile, i);
+                if (commentRange) {
+                    switch (commentRange.kind) {
+                        case 2 /* SingleLineCommentTrivia */:
+                            textChanges.push.apply(textChanges, toggleLineComment(fileName, { end: commentRange.end, pos: commentRange.pos + 1 }, /*insertComment*/ false));
+                            break;
+                        case 3 /* MultiLineCommentTrivia */:
+                            textChanges.push.apply(textChanges, toggleMultilineComment(fileName, { end: commentRange.end, pos: commentRange.pos + 1 }, /*insertComment*/ false));
+                    }
+                    i = commentRange.end + 1;
+                }
+            }
+            return textChanges;
+        }
         function isUnclosedTag(_a) {
             var openingElement = _a.openingElement, closingElement = _a.closingElement, parent = _a.parent;
             return !ts.tagNamesAreEquivalent(openingElement.tagName, closingElement.tagName) ||
@@ -139912,7 +148025,7 @@ var ts;
             synchronizeHostData();
             return ts.Rename.getRenameInfo(program, getValidSourceFile(fileName), position, options);
         }
-        function getRefactorContext(file, positionOrRange, preferences, formatOptions) {
+        function getRefactorContext(file, positionOrRange, preferences, formatOptions, triggerReason) {
             var _a = typeof positionOrRange === "number" ? [positionOrRange, undefined] : [positionOrRange.pos, positionOrRange.end], startPosition = _a[0], endPosition = _a[1];
             return {
                 file: file,
@@ -139923,16 +148036,17 @@ var ts;
                 formatContext: ts.formatting.getFormatContext(formatOptions, host),
                 cancellationToken: cancellationToken,
                 preferences: preferences,
+                triggerReason: triggerReason,
             };
         }
         function getSmartSelectionRange(fileName, position) {
             return ts.SmartSelectionRange.getSmartSelectionRange(position, syntaxTreeCache.getCurrentSourceFile(fileName));
         }
-        function getApplicableRefactors(fileName, positionOrRange, preferences) {
+        function getApplicableRefactors(fileName, positionOrRange, preferences, triggerReason) {
             if (preferences === void 0) { preferences = ts.emptyOptions; }
             synchronizeHostData();
             var file = getValidSourceFile(fileName);
-            return ts.refactor.getApplicableRefactors(getRefactorContext(file, positionOrRange, preferences));
+            return ts.refactor.getApplicableRefactors(getRefactorContext(file, positionOrRange, preferences, ts.emptyOptions, triggerReason));
         }
         function getEditsForRefactor(fileName, formatOptions, positionOrRange, refactorName, actionName, preferences) {
             if (preferences === void 0) { preferences = ts.emptyOptions; }
@@ -139957,7 +148071,7 @@ var ts;
             var declaration = ts.firstOrOnly(ts.CallHierarchy.resolveCallHierarchyDeclaration(program, position === 0 ? sourceFile : ts.getTouchingPropertyName(sourceFile, position)));
             return declaration ? ts.CallHierarchy.getOutgoingCalls(program, declaration) : [];
         }
-        return {
+        var ls = {
             dispose: dispose,
             cleanupSemanticCache: cleanupSemanticCache,
             getSyntacticDiagnostics: getSyntacticDiagnostics,
@@ -140008,6 +148122,7 @@ var ts;
             getEmitOutput: getEmitOutput,
             getNonBoundSourceFile: getNonBoundSourceFile,
             getProgram: getProgram,
+            getAutoImportProvider: getAutoImportProvider,
             getApplicableRefactors: getApplicableRefactors,
             getEditsForRefactor: getEditsForRefactor,
             toLineColumnOffset: sourceMapper.toLineColumnOffset,
@@ -140015,8 +148130,33 @@ var ts;
             clearSourceMapperCache: function () { return sourceMapper.clearCache(); },
             prepareCallHierarchy: prepareCallHierarchy,
             provideCallHierarchyIncomingCalls: provideCallHierarchyIncomingCalls,
-            provideCallHierarchyOutgoingCalls: provideCallHierarchyOutgoingCalls
+            provideCallHierarchyOutgoingCalls: provideCallHierarchyOutgoingCalls,
+            toggleLineComment: toggleLineComment,
+            toggleMultilineComment: toggleMultilineComment,
+            commentSelection: commentSelection,
+            uncommentSelection: uncommentSelection,
         };
+        switch (languageServiceMode) {
+            case ts.LanguageServiceMode.Semantic:
+                break;
+            case ts.LanguageServiceMode.PartialSemantic:
+                invalidOperationsInPartialSemanticMode.forEach(function (key) {
+                    return ls[key] = function () {
+                        throw new Error("LanguageService Operation: " + key + " not allowed in LanguageServiceMode.PartialSemantic");
+                    };
+                });
+                break;
+            case ts.LanguageServiceMode.Syntactic:
+                invalidOperationsInSyntacticMode.forEach(function (key) {
+                    return ls[key] = function () {
+                        throw new Error("LanguageService Operation: " + key + " not allowed in LanguageServiceMode.Syntactic");
+                    };
+                });
+                break;
+            default:
+                ts.Debug.assertNever(languageServiceMode);
+        }
+        return ls;
     }
     ts.createLanguageService = createLanguageService;
     /* @internal */
@@ -140029,7 +148169,7 @@ var ts;
     }
     ts.getNameTable = getNameTable;
     function initializeNameTable(sourceFile) {
-        var nameTable = sourceFile.nameTable = ts.createUnderscoreEscapedMap();
+        var nameTable = sourceFile.nameTable = new ts.Map();
         sourceFile.forEachChild(function walk(node) {
             if (ts.isIdentifier(node) && !ts.isTagName(node) && node.escapedText || ts.isStringOrNumericLiteralLike(node) && literalIsName(node)) {
                 var text = ts.getEscapedTextOfIdentifierOrLiteral(node);
@@ -140056,7 +148196,7 @@ var ts;
      */
     function literalIsName(node) {
         return ts.isDeclarationName(node) ||
-            node.parent.kind === 265 /* ExternalModuleReference */ ||
+            node.parent.kind === 272 /* ExternalModuleReference */ ||
             isArgumentOfElementAccessExpression(node) ||
             ts.isLiteralComputedPropertyDeclarationName(node);
     }
@@ -140074,13 +148214,13 @@ var ts;
             case 10 /* StringLiteral */:
             case 14 /* NoSubstitutionTemplateLiteral */:
             case 8 /* NumericLiteral */:
-                if (node.parent.kind === 154 /* ComputedPropertyName */) {
+                if (node.parent.kind === 158 /* ComputedPropertyName */) {
                     return ts.isObjectLiteralElement(node.parent.parent) ? node.parent.parent : undefined;
                 }
             // falls through
-            case 75 /* Identifier */:
+            case 78 /* Identifier */:
                 return ts.isObjectLiteralElement(node.parent) &&
-                    (node.parent.parent.kind === 193 /* ObjectLiteralExpression */ || node.parent.parent.kind === 274 /* JsxAttributes */) &&
+                    (node.parent.parent.kind === 200 /* ObjectLiteralExpression */ || node.parent.parent.kind === 281 /* JsxAttributes */) &&
                     node.parent.name === node ? node.parent : undefined;
         }
         return undefined;
@@ -140122,7 +148262,7 @@ var ts;
     function isArgumentOfElementAccessExpression(node) {
         return node &&
             node.parent &&
-            node.parent.kind === 195 /* ElementAccessExpression */ &&
+            node.parent.kind === 202 /* ElementAccessExpression */ &&
             node.parent.argumentExpression === node;
     }
     /**
@@ -140202,114 +148342,114 @@ var ts;
                 if (node) {
                     var parent = node.parent;
                     switch (node.kind) {
-                        case 225 /* VariableStatement */:
+                        case 232 /* VariableStatement */:
                             // Span on first variable declaration
                             return spanInVariableDeclaration(node.declarationList.declarations[0]);
-                        case 242 /* VariableDeclaration */:
-                        case 159 /* PropertyDeclaration */:
-                        case 158 /* PropertySignature */:
+                        case 249 /* VariableDeclaration */:
+                        case 163 /* PropertyDeclaration */:
+                        case 162 /* PropertySignature */:
                             return spanInVariableDeclaration(node);
-                        case 156 /* Parameter */:
+                        case 160 /* Parameter */:
                             return spanInParameterDeclaration(node);
-                        case 244 /* FunctionDeclaration */:
-                        case 161 /* MethodDeclaration */:
-                        case 160 /* MethodSignature */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
-                        case 162 /* Constructor */:
-                        case 201 /* FunctionExpression */:
-                        case 202 /* ArrowFunction */:
+                        case 251 /* FunctionDeclaration */:
+                        case 165 /* MethodDeclaration */:
+                        case 164 /* MethodSignature */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
+                        case 166 /* Constructor */:
+                        case 208 /* FunctionExpression */:
+                        case 209 /* ArrowFunction */:
                             return spanInFunctionDeclaration(node);
-                        case 223 /* Block */:
+                        case 230 /* Block */:
                             if (ts.isFunctionBlock(node)) {
                                 return spanInFunctionBlock(node);
                             }
                         // falls through
-                        case 250 /* ModuleBlock */:
+                        case 257 /* ModuleBlock */:
                             return spanInBlock(node);
-                        case 280 /* CatchClause */:
+                        case 287 /* CatchClause */:
                             return spanInBlock(node.block);
-                        case 226 /* ExpressionStatement */:
+                        case 233 /* ExpressionStatement */:
                             // span on the expression
                             return textSpan(node.expression);
-                        case 235 /* ReturnStatement */:
+                        case 242 /* ReturnStatement */:
                             // span on return keyword and expression if present
                             return textSpan(node.getChildAt(0), node.expression);
-                        case 229 /* WhileStatement */:
+                        case 236 /* WhileStatement */:
                             // Span on while(...)
                             return textSpanEndingAtNextToken(node, node.expression);
-                        case 228 /* DoStatement */:
+                        case 235 /* DoStatement */:
                             // span in statement of the do statement
                             return spanInNode(node.statement);
-                        case 241 /* DebuggerStatement */:
+                        case 248 /* DebuggerStatement */:
                             // span on debugger keyword
                             return textSpan(node.getChildAt(0));
-                        case 227 /* IfStatement */:
+                        case 234 /* IfStatement */:
                             // set on if(..) span
                             return textSpanEndingAtNextToken(node, node.expression);
-                        case 238 /* LabeledStatement */:
+                        case 245 /* LabeledStatement */:
                             // span in statement
                             return spanInNode(node.statement);
-                        case 234 /* BreakStatement */:
-                        case 233 /* ContinueStatement */:
+                        case 241 /* BreakStatement */:
+                        case 240 /* ContinueStatement */:
                             // On break or continue keyword and label if present
                             return textSpan(node.getChildAt(0), node.label);
-                        case 230 /* ForStatement */:
+                        case 237 /* ForStatement */:
                             return spanInForStatement(node);
-                        case 231 /* ForInStatement */:
+                        case 238 /* ForInStatement */:
                             // span of for (a in ...)
                             return textSpanEndingAtNextToken(node, node.expression);
-                        case 232 /* ForOfStatement */:
+                        case 239 /* ForOfStatement */:
                             // span in initializer
                             return spanInInitializerOfForLike(node);
-                        case 237 /* SwitchStatement */:
+                        case 244 /* SwitchStatement */:
                             // span on switch(...)
                             return textSpanEndingAtNextToken(node, node.expression);
-                        case 277 /* CaseClause */:
-                        case 278 /* DefaultClause */:
+                        case 284 /* CaseClause */:
+                        case 285 /* DefaultClause */:
                             // span in first statement of the clause
                             return spanInNode(node.statements[0]);
-                        case 240 /* TryStatement */:
+                        case 247 /* TryStatement */:
                             // span in try block
                             return spanInBlock(node.tryBlock);
-                        case 239 /* ThrowStatement */:
+                        case 246 /* ThrowStatement */:
                             // span in throw ...
                             return textSpan(node, node.expression);
-                        case 259 /* ExportAssignment */:
+                        case 266 /* ExportAssignment */:
                             // span on export = id
                             return textSpan(node, node.expression);
-                        case 253 /* ImportEqualsDeclaration */:
+                        case 260 /* ImportEqualsDeclaration */:
                             // import statement without including semicolon
                             return textSpan(node, node.moduleReference);
-                        case 254 /* ImportDeclaration */:
+                        case 261 /* ImportDeclaration */:
                             // import statement without including semicolon
                             return textSpan(node, node.moduleSpecifier);
-                        case 260 /* ExportDeclaration */:
+                        case 267 /* ExportDeclaration */:
                             // import statement without including semicolon
                             return textSpan(node, node.moduleSpecifier);
-                        case 249 /* ModuleDeclaration */:
+                        case 256 /* ModuleDeclaration */:
                             // span on complete module if it is instantiated
                             if (ts.getModuleInstanceState(node) !== 1 /* Instantiated */) {
                                 return undefined;
                             }
                         // falls through
-                        case 245 /* ClassDeclaration */:
-                        case 248 /* EnumDeclaration */:
-                        case 284 /* EnumMember */:
-                        case 191 /* BindingElement */:
+                        case 252 /* ClassDeclaration */:
+                        case 255 /* EnumDeclaration */:
+                        case 291 /* EnumMember */:
+                        case 198 /* BindingElement */:
                             // span on complete node
                             return textSpan(node);
-                        case 236 /* WithStatement */:
+                        case 243 /* WithStatement */:
                             // span in statement
                             return spanInNode(node.statement);
-                        case 157 /* Decorator */:
+                        case 161 /* Decorator */:
                             return spanInNodeArray(parent.decorators);
-                        case 189 /* ObjectBindingPattern */:
-                        case 190 /* ArrayBindingPattern */:
+                        case 196 /* ObjectBindingPattern */:
+                        case 197 /* ArrayBindingPattern */:
                             return spanInBindingPattern(node);
                         // No breakpoint in interface, type alias
-                        case 246 /* InterfaceDeclaration */:
-                        case 247 /* TypeAliasDeclaration */:
+                        case 253 /* InterfaceDeclaration */:
+                        case 254 /* TypeAliasDeclaration */:
                             return undefined;
                         // Tokens:
                         case 26 /* SemicolonToken */:
@@ -140333,13 +148473,13 @@ var ts;
                         case 29 /* LessThanToken */:
                             return spanInGreaterThanOrLessThanToken(node);
                         // Keywords:
-                        case 111 /* WhileKeyword */:
+                        case 114 /* WhileKeyword */:
                             return spanInWhileKeyword(node);
-                        case 87 /* ElseKeyword */:
-                        case 79 /* CatchKeyword */:
-                        case 92 /* FinallyKeyword */:
+                        case 90 /* ElseKeyword */:
+                        case 82 /* CatchKeyword */:
+                        case 95 /* FinallyKeyword */:
                             return spanInNextNode(node);
-                        case 152 /* OfKeyword */:
+                        case 156 /* OfKeyword */:
                             return spanInOfKeyword(node);
                         default:
                             // Destructuring pattern in destructuring assignment
@@ -140351,14 +148491,14 @@ var ts;
                             // Set breakpoint on identifier element of destructuring pattern
                             // `a` or `...c` or `d: x` from
                             // `[a, b, ...c]` or `{ a, b }` or `{ d: x }` from destructuring pattern
-                            if ((node.kind === 75 /* Identifier */ ||
-                                node.kind === 213 /* SpreadElement */ ||
-                                node.kind === 281 /* PropertyAssignment */ ||
-                                node.kind === 282 /* ShorthandPropertyAssignment */) &&
+                            if ((node.kind === 78 /* Identifier */ ||
+                                node.kind === 220 /* SpreadElement */ ||
+                                node.kind === 288 /* PropertyAssignment */ ||
+                                node.kind === 289 /* ShorthandPropertyAssignment */) &&
                                 ts.isArrayLiteralOrObjectLiteralDestructuringPattern(parent)) {
                                 return textSpan(node);
                             }
-                            if (node.kind === 209 /* BinaryExpression */) {
+                            if (node.kind === 216 /* BinaryExpression */) {
                                 var _a = node, left = _a.left, operatorToken = _a.operatorToken;
                                 // Set breakpoint in destructuring pattern if its destructuring assignment
                                 // [a, b, c] or {a, b, c} of
@@ -140380,22 +148520,22 @@ var ts;
                             }
                             if (ts.isExpressionNode(node)) {
                                 switch (parent.kind) {
-                                    case 228 /* DoStatement */:
+                                    case 235 /* DoStatement */:
                                         // Set span as if on while keyword
                                         return spanInPreviousNode(node);
-                                    case 157 /* Decorator */:
+                                    case 161 /* Decorator */:
                                         // Set breakpoint on the decorator emit
                                         return spanInNode(node.parent);
-                                    case 230 /* ForStatement */:
-                                    case 232 /* ForOfStatement */:
+                                    case 237 /* ForStatement */:
+                                    case 239 /* ForOfStatement */:
                                         return textSpan(node);
-                                    case 209 /* BinaryExpression */:
+                                    case 216 /* BinaryExpression */:
                                         if (node.parent.operatorToken.kind === 27 /* CommaToken */) {
                                             // If this is a comma expression, the breakpoint is possible in this expression
                                             return textSpan(node);
                                         }
                                         break;
-                                    case 202 /* ArrowFunction */:
+                                    case 209 /* ArrowFunction */:
                                         if (node.parent.body === node) {
                                             // If this is body of arrow function, it is allowed to have the breakpoint
                                             return textSpan(node);
@@ -140404,21 +148544,21 @@ var ts;
                                 }
                             }
                             switch (node.parent.kind) {
-                                case 281 /* PropertyAssignment */:
+                                case 288 /* PropertyAssignment */:
                                     // If this is name of property assignment, set breakpoint in the initializer
                                     if (node.parent.name === node &&
                                         !ts.isArrayLiteralOrObjectLiteralDestructuringPattern(node.parent.parent)) {
                                         return spanInNode(node.parent.initializer);
                                     }
                                     break;
-                                case 199 /* TypeAssertionExpression */:
+                                case 206 /* TypeAssertionExpression */:
                                     // Breakpoint in type assertion goes to its operand
                                     if (node.parent.type === node) {
                                         return spanInNextNode(node.parent.type);
                                     }
                                     break;
-                                case 242 /* VariableDeclaration */:
-                                case 156 /* Parameter */: {
+                                case 249 /* VariableDeclaration */:
+                                case 160 /* Parameter */: {
                                     // initializer of variable/parameter declaration go to previous node
                                     var _b = node.parent, initializer = _b.initializer, type = _b.type;
                                     if (initializer === node || type === node || ts.isAssignmentOperator(node.kind)) {
@@ -140426,7 +148566,7 @@ var ts;
                                     }
                                     break;
                                 }
-                                case 209 /* BinaryExpression */: {
+                                case 216 /* BinaryExpression */: {
                                     var left = node.parent.left;
                                     if (ts.isArrayLiteralOrObjectLiteralDestructuringPattern(left) && node !== left) {
                                         // If initializer of destructuring assignment move to previous token
@@ -140456,7 +148596,7 @@ var ts;
                 }
                 function spanInVariableDeclaration(variableDeclaration) {
                     // If declaration of for in statement, just set the span in parent
-                    if (variableDeclaration.parent.parent.kind === 231 /* ForInStatement */) {
+                    if (variableDeclaration.parent.parent.kind === 238 /* ForInStatement */) {
                         return spanInNode(variableDeclaration.parent.parent);
                     }
                     var parent = variableDeclaration.parent;
@@ -140467,8 +148607,8 @@ var ts;
                     // Breakpoint is possible in variableDeclaration only if there is initialization
                     // or its declaration from 'for of'
                     if (variableDeclaration.initializer ||
-                        ts.hasModifier(variableDeclaration, 1 /* Export */) ||
-                        parent.parent.kind === 232 /* ForOfStatement */) {
+                        ts.hasSyntacticModifier(variableDeclaration, 1 /* Export */) ||
+                        parent.parent.kind === 239 /* ForOfStatement */) {
                         return textSpanFromVariableDeclaration(variableDeclaration);
                     }
                     if (ts.isVariableDeclarationList(variableDeclaration.parent) &&
@@ -140483,7 +148623,7 @@ var ts;
                 function canHaveSpanInParameterDeclaration(parameter) {
                     // Breakpoint is possible on parameter only if it has initializer, is a rest parameter, or has public or private modifier
                     return !!parameter.initializer || parameter.dotDotDotToken !== undefined ||
-                        ts.hasModifier(parameter, 4 /* Public */ | 8 /* Private */);
+                        ts.hasSyntacticModifier(parameter, 4 /* Public */ | 8 /* Private */);
                 }
                 function spanInParameterDeclaration(parameter) {
                     if (ts.isBindingPattern(parameter.name)) {
@@ -140508,8 +148648,8 @@ var ts;
                     }
                 }
                 function canFunctionHaveSpanInWholeDeclaration(functionDeclaration) {
-                    return ts.hasModifier(functionDeclaration, 1 /* Export */) ||
-                        (functionDeclaration.parent.kind === 245 /* ClassDeclaration */ && functionDeclaration.kind !== 162 /* Constructor */);
+                    return ts.hasSyntacticModifier(functionDeclaration, 1 /* Export */) ||
+                        (functionDeclaration.parent.kind === 252 /* ClassDeclaration */ && functionDeclaration.kind !== 166 /* Constructor */);
                 }
                 function spanInFunctionDeclaration(functionDeclaration) {
                     // No breakpoints in the function signature
@@ -140532,26 +148672,26 @@ var ts;
                 }
                 function spanInBlock(block) {
                     switch (block.parent.kind) {
-                        case 249 /* ModuleDeclaration */:
+                        case 256 /* ModuleDeclaration */:
                             if (ts.getModuleInstanceState(block.parent) !== 1 /* Instantiated */) {
                                 return undefined;
                             }
                         // Set on parent if on same line otherwise on first statement
                         // falls through
-                        case 229 /* WhileStatement */:
-                        case 227 /* IfStatement */:
-                        case 231 /* ForInStatement */:
+                        case 236 /* WhileStatement */:
+                        case 234 /* IfStatement */:
+                        case 238 /* ForInStatement */:
                             return spanInNodeIfStartsOnSameLine(block.parent, block.statements[0]);
                         // Set span on previous token if it starts on same line otherwise on the first statement of the block
-                        case 230 /* ForStatement */:
-                        case 232 /* ForOfStatement */:
+                        case 237 /* ForStatement */:
+                        case 239 /* ForOfStatement */:
                             return spanInNodeIfStartsOnSameLine(ts.findPrecedingToken(block.pos, sourceFile, block.parent), block.statements[0]);
                     }
                     // Default action is to set on first statement
                     return spanInNode(block.statements[0]);
                 }
                 function spanInInitializerOfForLike(forLikeStatement) {
-                    if (forLikeStatement.initializer.kind === 243 /* VariableDeclarationList */) {
+                    if (forLikeStatement.initializer.kind === 250 /* VariableDeclarationList */) {
                         // Declaration list - set breakpoint in first declaration
                         var variableDeclarationList = forLikeStatement.initializer;
                         if (variableDeclarationList.declarations.length > 0) {
@@ -140576,21 +148716,21 @@ var ts;
                 }
                 function spanInBindingPattern(bindingPattern) {
                     // Set breakpoint in first binding element
-                    var firstBindingElement = ts.forEach(bindingPattern.elements, function (element) { return element.kind !== 215 /* OmittedExpression */ ? element : undefined; });
+                    var firstBindingElement = ts.forEach(bindingPattern.elements, function (element) { return element.kind !== 222 /* OmittedExpression */ ? element : undefined; });
                     if (firstBindingElement) {
                         return spanInNode(firstBindingElement);
                     }
                     // Empty binding pattern of binding element, set breakpoint on binding element
-                    if (bindingPattern.parent.kind === 191 /* BindingElement */) {
+                    if (bindingPattern.parent.kind === 198 /* BindingElement */) {
                         return textSpan(bindingPattern.parent);
                     }
                     // Variable declaration is used as the span
                     return textSpanFromVariableDeclaration(bindingPattern.parent);
                 }
                 function spanInArrayLiteralOrObjectLiteralDestructuringPattern(node) {
-                    ts.Debug.assert(node.kind !== 190 /* ArrayBindingPattern */ && node.kind !== 189 /* ObjectBindingPattern */);
-                    var elements = node.kind === 192 /* ArrayLiteralExpression */ ? node.elements : node.properties;
-                    var firstBindingElement = ts.forEach(elements, function (element) { return element.kind !== 215 /* OmittedExpression */ ? element : undefined; });
+                    ts.Debug.assert(node.kind !== 197 /* ArrayBindingPattern */ && node.kind !== 196 /* ObjectBindingPattern */);
+                    var elements = node.kind === 199 /* ArrayLiteralExpression */ ? node.elements : node.properties;
+                    var firstBindingElement = ts.forEach(elements, function (element) { return element.kind !== 222 /* OmittedExpression */ ? element : undefined; });
                     if (firstBindingElement) {
                         return spanInNode(firstBindingElement);
                     }
@@ -140598,18 +148738,18 @@ var ts;
                     // just nested element in another destructuring assignment
                     // set breakpoint on assignment when parent is destructuring assignment
                     // Otherwise set breakpoint for this element
-                    return textSpan(node.parent.kind === 209 /* BinaryExpression */ ? node.parent : node);
+                    return textSpan(node.parent.kind === 216 /* BinaryExpression */ ? node.parent : node);
                 }
                 // Tokens:
                 function spanInOpenBraceToken(node) {
                     switch (node.parent.kind) {
-                        case 248 /* EnumDeclaration */:
+                        case 255 /* EnumDeclaration */:
                             var enumDeclaration = node.parent;
                             return spanInNodeIfStartsOnSameLine(ts.findPrecedingToken(node.pos, sourceFile, node.parent), enumDeclaration.members.length ? enumDeclaration.members[0] : enumDeclaration.getLastToken(sourceFile));
-                        case 245 /* ClassDeclaration */:
+                        case 252 /* ClassDeclaration */:
                             var classDeclaration = node.parent;
                             return spanInNodeIfStartsOnSameLine(ts.findPrecedingToken(node.pos, sourceFile, node.parent), classDeclaration.members.length ? classDeclaration.members[0] : classDeclaration.getLastToken(sourceFile));
-                        case 251 /* CaseBlock */:
+                        case 258 /* CaseBlock */:
                             return spanInNodeIfStartsOnSameLine(node.parent.parent, node.parent.clauses[0]);
                     }
                     // Default to parent node
@@ -140617,25 +148757,25 @@ var ts;
                 }
                 function spanInCloseBraceToken(node) {
                     switch (node.parent.kind) {
-                        case 250 /* ModuleBlock */:
+                        case 257 /* ModuleBlock */:
                             // If this is not an instantiated module block, no bp span
                             if (ts.getModuleInstanceState(node.parent.parent) !== 1 /* Instantiated */) {
                                 return undefined;
                             }
                         // falls through
-                        case 248 /* EnumDeclaration */:
-                        case 245 /* ClassDeclaration */:
+                        case 255 /* EnumDeclaration */:
+                        case 252 /* ClassDeclaration */:
                             // Span on close brace token
                             return textSpan(node);
-                        case 223 /* Block */:
+                        case 230 /* Block */:
                             if (ts.isFunctionBlock(node.parent)) {
                                 // Span on close brace token
                                 return textSpan(node);
                             }
                         // falls through
-                        case 280 /* CatchClause */:
+                        case 287 /* CatchClause */:
                             return spanInNode(ts.lastOrUndefined(node.parent.statements));
-                        case 251 /* CaseBlock */:
+                        case 258 /* CaseBlock */:
                             // breakpoint in last statement of the last clause
                             var caseBlock = node.parent;
                             var lastClause = ts.lastOrUndefined(caseBlock.clauses);
@@ -140643,7 +148783,7 @@ var ts;
                                 return spanInNode(ts.lastOrUndefined(lastClause.statements));
                             }
                             return undefined;
-                        case 189 /* ObjectBindingPattern */:
+                        case 196 /* ObjectBindingPattern */:
                             // Breakpoint in last binding element or binding pattern if it contains no elements
                             var bindingPattern = node.parent;
                             return spanInNode(ts.lastOrUndefined(bindingPattern.elements) || bindingPattern);
@@ -140659,7 +148799,7 @@ var ts;
                 }
                 function spanInCloseBracketToken(node) {
                     switch (node.parent.kind) {
-                        case 190 /* ArrayBindingPattern */:
+                        case 197 /* ArrayBindingPattern */:
                             // Breakpoint in last binding element or binding pattern if it contains no elements
                             var bindingPattern = node.parent;
                             return textSpan(ts.lastOrUndefined(bindingPattern.elements) || bindingPattern);
@@ -140674,12 +148814,12 @@ var ts;
                     }
                 }
                 function spanInOpenParenToken(node) {
-                    if (node.parent.kind === 228 /* DoStatement */ || // Go to while keyword and do action instead
-                        node.parent.kind === 196 /* CallExpression */ ||
-                        node.parent.kind === 197 /* NewExpression */) {
+                    if (node.parent.kind === 235 /* DoStatement */ || // Go to while keyword and do action instead
+                        node.parent.kind === 203 /* CallExpression */ ||
+                        node.parent.kind === 204 /* NewExpression */) {
                         return spanInPreviousNode(node);
                     }
-                    if (node.parent.kind === 200 /* ParenthesizedExpression */) {
+                    if (node.parent.kind === 207 /* ParenthesizedExpression */) {
                         return spanInNextNode(node);
                     }
                     // Default to parent node
@@ -140688,21 +148828,21 @@ var ts;
                 function spanInCloseParenToken(node) {
                     // Is this close paren token of parameter list, set span in previous token
                     switch (node.parent.kind) {
-                        case 201 /* FunctionExpression */:
-                        case 244 /* FunctionDeclaration */:
-                        case 202 /* ArrowFunction */:
-                        case 161 /* MethodDeclaration */:
-                        case 160 /* MethodSignature */:
-                        case 163 /* GetAccessor */:
-                        case 164 /* SetAccessor */:
-                        case 162 /* Constructor */:
-                        case 229 /* WhileStatement */:
-                        case 228 /* DoStatement */:
-                        case 230 /* ForStatement */:
-                        case 232 /* ForOfStatement */:
-                        case 196 /* CallExpression */:
-                        case 197 /* NewExpression */:
-                        case 200 /* ParenthesizedExpression */:
+                        case 208 /* FunctionExpression */:
+                        case 251 /* FunctionDeclaration */:
+                        case 209 /* ArrowFunction */:
+                        case 165 /* MethodDeclaration */:
+                        case 164 /* MethodSignature */:
+                        case 167 /* GetAccessor */:
+                        case 168 /* SetAccessor */:
+                        case 166 /* Constructor */:
+                        case 236 /* WhileStatement */:
+                        case 235 /* DoStatement */:
+                        case 237 /* ForStatement */:
+                        case 239 /* ForOfStatement */:
+                        case 203 /* CallExpression */:
+                        case 204 /* NewExpression */:
+                        case 207 /* ParenthesizedExpression */:
                             return spanInPreviousNode(node);
                         // Default to parent node
                         default:
@@ -140712,20 +148852,20 @@ var ts;
                 function spanInColonToken(node) {
                     // Is this : specifying return annotation of the function declaration
                     if (ts.isFunctionLike(node.parent) ||
-                        node.parent.kind === 281 /* PropertyAssignment */ ||
-                        node.parent.kind === 156 /* Parameter */) {
+                        node.parent.kind === 288 /* PropertyAssignment */ ||
+                        node.parent.kind === 160 /* Parameter */) {
                         return spanInPreviousNode(node);
                     }
                     return spanInNode(node.parent);
                 }
                 function spanInGreaterThanOrLessThanToken(node) {
-                    if (node.parent.kind === 199 /* TypeAssertionExpression */) {
+                    if (node.parent.kind === 206 /* TypeAssertionExpression */) {
                         return spanInNextNode(node);
                     }
                     return spanInNode(node.parent);
                 }
                 function spanInWhileKeyword(node) {
-                    if (node.parent.kind === 228 /* DoStatement */) {
+                    if (node.parent.kind === 235 /* DoStatement */) {
                         // Set span on while expression
                         return textSpanEndingAtNextToken(node, node.parent.expression);
                     }
@@ -140733,7 +148873,7 @@ var ts;
                     return spanInNode(node.parent);
                 }
                 function spanInOfKeyword(node) {
-                    if (node.parent.kind === 232 /* ForOfStatement */) {
+                    if (node.parent.kind === 239 /* ForOfStatement */) {
                         // Set using next token
                         return spanInNextNode(node);
                     }
@@ -140757,7 +148897,7 @@ var ts;
         var diagnostics = [];
         compilerOptions = ts.fixupCompilerOptions(compilerOptions, diagnostics); // TODO: GH#18217
         var nodes = ts.isArray(source) ? source : [source];
-        var result = ts.transformNodes(/*resolver*/ undefined, /*emitHost*/ undefined, compilerOptions, nodes, transformers, /*allowDtsFiles*/ true);
+        var result = ts.transformNodes(/*resolver*/ undefined, /*emitHost*/ undefined, ts.factory, compilerOptions, nodes, transformers, /*allowDtsFiles*/ true);
         result.diagnostics = ts.concatenate(result.diagnostics, diagnostics);
         return result;
     }
@@ -141039,6 +149179,7 @@ var ts;
             category: ts.diagnosticCategoryName(diagnostic),
             code: diagnostic.code,
             reportsUnnecessary: diagnostic.reportsUnnecessary,
+            reportsDeprecated: diagnostic.reportsDeprecated
         };
     }
     var LanguageServiceShimObject = /** @class */ (function (_super) {
@@ -141349,6 +149490,22 @@ var ts;
             return forwardCall(this.logger, "getEmitOutput('" + fileName + "')", 
             /*returnJson*/ false, function () { return _this.languageService.getEmitOutput(fileName); }, this.logPerformance);
         };
+        LanguageServiceShimObject.prototype.toggleLineComment = function (fileName, textRange) {
+            var _this = this;
+            return this.forwardJSONCall("toggleLineComment('" + fileName + "', '" + JSON.stringify(textRange) + "')", function () { return _this.languageService.toggleLineComment(fileName, textRange); });
+        };
+        LanguageServiceShimObject.prototype.toggleMultilineComment = function (fileName, textRange) {
+            var _this = this;
+            return this.forwardJSONCall("toggleMultilineComment('" + fileName + "', '" + JSON.stringify(textRange) + "')", function () { return _this.languageService.toggleMultilineComment(fileName, textRange); });
+        };
+        LanguageServiceShimObject.prototype.commentSelection = function (fileName, textRange) {
+            var _this = this;
+            return this.forwardJSONCall("commentSelection('" + fileName + "', '" + JSON.stringify(textRange) + "')", function () { return _this.languageService.commentSelection(fileName, textRange); });
+        };
+        LanguageServiceShimObject.prototype.uncommentSelection = function (fileName, textRange) {
+            var _this = this;
+            return this.forwardJSONCall("uncommentSelection('" + fileName + "', '" + JSON.stringify(textRange) + "')", function () { return _this.languageService.uncommentSelection(fileName, textRange); });
+        };
         return LanguageServiceShimObject;
     }(ShimBase));
     function convertClassifications(classifications) {
@@ -141669,7 +149826,7 @@ var ts;
         }
         server.asNormalizedPath = asNormalizedPath;
         function createNormalizedPathMap() {
-            var map = ts.createMap();
+            var map = new ts.Map();
             return {
                 get: function (path) {
                     return map.get(path);
@@ -141695,6 +149852,11 @@ var ts;
             return "/dev/null/inferredProject" + counter + "*";
         }
         server.makeInferredProjectName = makeInferredProjectName;
+        /*@internal*/
+        function makeAutoImportProviderProjectName(counter) {
+            return "/dev/null/autoImportProviderProject" + counter + "*";
+        }
+        server.makeAutoImportProviderProjectName = makeAutoImportProviderProjectName;
         function createSortedArray() {
             return []; // TODO: GH#19873
         }
@@ -141709,7 +149871,7 @@ var ts;
         var ThrottledOperations = /** @class */ (function () {
             function ThrottledOperations(host, logger) {
                 this.host = host;
-                this.pendingTimeouts = ts.createMap();
+                this.pendingTimeouts = new ts.Map();
                 this.logger = logger.hasLevel(server.LogLevel.verbose) ? logger : undefined;
             }
             /**
@@ -141730,6 +149892,13 @@ var ts;
                     this.logger.info("Scheduled: " + operationId + (pendingTimeout ? ", Cancelled earlier one" : ""));
                 }
             };
+            ThrottledOperations.prototype.cancel = function (operationId) {
+                var pendingTimeout = this.pendingTimeouts.get(operationId);
+                if (!pendingTimeout)
+                    return false;
+                this.host.clearTimeout(pendingTimeout);
+                return this.pendingTimeouts.delete(operationId);
+            };
             ThrottledOperations.run = function (self, operationId, cb) {
                 ts.perfLogger.logStartScheduledOperation(operationId);
                 self.pendingTimeouts.delete(operationId);
@@ -141953,6 +150122,18 @@ var ts;
                 CommandTypes["SelectionRange"] = "selectionRange";
                 /* @internal */
                 CommandTypes["SelectionRangeFull"] = "selectionRange-full";
+                CommandTypes["ToggleLineComment"] = "toggleLineComment";
+                /* @internal */
+                CommandTypes["ToggleLineCommentFull"] = "toggleLineComment-full";
+                CommandTypes["ToggleMultilineComment"] = "toggleMultilineComment";
+                /* @internal */
+                CommandTypes["ToggleMultilineCommentFull"] = "toggleMultilineComment-full";
+                CommandTypes["CommentSelection"] = "commentSelection";
+                /* @internal */
+                CommandTypes["CommentSelectionFull"] = "commentSelection-full";
+                CommandTypes["UncommentSelection"] = "uncommentSelection";
+                /* @internal */
+                CommandTypes["UncommentSelectionFull"] = "uncommentSelection-full";
                 CommandTypes["PrepareCallHierarchy"] = "prepareCallHierarchy";
                 CommandTypes["ProvideCallHierarchyIncomingCalls"] = "provideCallHierarchyIncomingCalls";
                 CommandTypes["ProvideCallHierarchyOutgoingCalls"] = "provideCallHierarchyOutgoingCalls";
@@ -141996,6 +150177,8 @@ var ts;
                 JsxEmit["Preserve"] = "Preserve";
                 JsxEmit["ReactNative"] = "ReactNative";
                 JsxEmit["React"] = "React";
+                JsxEmit["ReactJSX"] = "ReactJSX";
+                JsxEmit["ReactJSXDev"] = "ReactJSXDev";
             })(JsxEmit = protocol.JsxEmit || (protocol.JsxEmit = {}));
             var ModuleKind;
             (function (ModuleKind) {
@@ -142006,6 +150189,7 @@ var ts;
                 ModuleKind["System"] = "System";
                 ModuleKind["ES6"] = "ES6";
                 ModuleKind["ES2015"] = "ES2015";
+                ModuleKind["ES2020"] = "ES2020";
                 ModuleKind["ESNext"] = "ESNext";
             })(ModuleKind = protocol.ModuleKind || (protocol.ModuleKind = {}));
             var ModuleResolutionKind;
@@ -142029,6 +150213,7 @@ var ts;
                 ScriptTarget["ES2018"] = "ES2018";
                 ScriptTarget["ES2019"] = "ES2019";
                 ScriptTarget["ES2020"] = "ES2020";
+                ScriptTarget["JSON"] = "JSON";
                 ScriptTarget["ESNext"] = "ESNext";
             })(ScriptTarget = protocol.ScriptTarget || (protocol.ScriptTarget = {}));
         })(protocol = server.protocol || (server.protocol = {}));
@@ -142254,7 +150439,6 @@ var ts;
             return TextStorage;
         }());
         server.TextStorage = TextStorage;
-        /*@internal*/
         function isDynamicFileName(fileName) {
             return fileName[0] === "^" ||
                 ((ts.stringContains(fileName, "walkThroughSnippet:/") || ts.stringContains(fileName, "untitled:/")) &&
@@ -142415,7 +150599,7 @@ var ts;
                     case 0:
                         return server.Errors.ThrowNoProject();
                     case 1:
-                        return this.containingProjects[0];
+                        return ensureNotAutoImportProvider(this.containingProjects[0]);
                     default:
                         // If this file belongs to multiple projects, below is the order in which default project is used
                         // - for open script info, its default configured project during opening is default if info is part of it
@@ -142425,6 +150609,7 @@ var ts;
                         // - first inferred project
                         var firstExternalProject = void 0;
                         var firstConfiguredProject = void 0;
+                        var firstInferredProject = void 0;
                         var firstNonSourceOfProjectReferenceRedirect = void 0;
                         var defaultConfiguredProject = void 0;
                         for (var index = 0; index < this.containingProjects.length; index++) {
@@ -142448,12 +150633,15 @@ var ts;
                             else if (!firstExternalProject && server.isExternalProject(project)) {
                                 firstExternalProject = project;
                             }
+                            else if (!firstInferredProject && server.isInferredProject(project)) {
+                                firstInferredProject = project;
+                            }
                         }
-                        return defaultConfiguredProject ||
+                        return ensureNotAutoImportProvider(defaultConfiguredProject ||
                             firstNonSourceOfProjectReferenceRedirect ||
                             firstConfiguredProject ||
                             firstExternalProject ||
-                            this.containingProjects[0];
+                            firstInferredProject);
                 }
             };
             ScriptInfo.prototype.registerFileUpdate = function () {
@@ -142524,6 +150712,10 @@ var ts;
             ScriptInfo.prototype.isOrphan = function () {
                 return !ts.forEach(this.containingProjects, function (p) { return !p.isOrphan(); });
             };
+            /*@internal*/
+            ScriptInfo.prototype.isContainedByAutoImportProvider = function () {
+                return ts.some(this.containingProjects, function (p) { return p.projectKind === server.ProjectKind.AutoImportProvider; });
+            };
             /**
              *  @param line 1 based index
              */
@@ -142556,6 +150748,12 @@ var ts;
             return ScriptInfo;
         }());
         server.ScriptInfo = ScriptInfo;
+        function ensureNotAutoImportProvider(project) {
+            if (!project || project.projectKind === server.ProjectKind.AutoImportProvider) {
+                return server.Errors.ThrowNoProject();
+            }
+            return project;
+        }
         function failIfInvalidPosition(position) {
             ts.Debug.assert(typeof position === "number", "Expected position " + position + " to be a number.");
             ts.Debug.assert(position >= 0, "Expected position to be non-negative.");
@@ -142588,7 +150786,7 @@ var ts;
             if ((arr1 || server.emptyArray).length === 0 && (arr2 || server.emptyArray).length === 0) {
                 return true;
             }
-            var set = ts.createMap();
+            var set = new ts.Map();
             var unique = 0;
             for (var _i = 0, _a = arr1; _i < _a.length; _i++) {
                 var v = _a[_i];
@@ -142617,7 +150815,7 @@ var ts;
         }
         function compilerOptionsChanged(opt1, opt2) {
             // TODO: add more relevant properties
-            return opt1.allowJs !== opt2.allowJs;
+            return ts.getAllowJSCompilerOption(opt1) !== ts.getAllowJSCompilerOption(opt2);
         }
         function unresolvedImportsChanged(imports1, imports2) {
             if (imports1 === imports2) {
@@ -142629,7 +150827,7 @@ var ts;
         var TypingsCache = /** @class */ (function () {
             function TypingsCache(installer) {
                 this.installer = installer;
-                this.perProjectCache = ts.createMap();
+                this.perProjectCache = new ts.Map();
             }
             TypingsCache.prototype.isKnownTypesPackageName = function (name) {
                 return this.installer.isKnownTypesPackageName(name);
@@ -142690,6 +150888,7 @@ var ts;
             ProjectKind[ProjectKind["Inferred"] = 0] = "Inferred";
             ProjectKind[ProjectKind["Configured"] = 1] = "Configured";
             ProjectKind[ProjectKind["External"] = 2] = "External";
+            ProjectKind[ProjectKind["AutoImportProvider"] = 3] = "AutoImportProvider";
         })(ProjectKind = server.ProjectKind || (server.ProjectKind = {}));
         /* @internal */
         function countEachFileTypes(infos, includeSizes) {
@@ -142772,7 +150971,7 @@ var ts;
                 this.compileOnSaveEnabled = compileOnSaveEnabled;
                 this.watchOptions = watchOptions;
                 this.rootFiles = [];
-                this.rootFilesMap = ts.createMap();
+                this.rootFilesMap = new ts.Map();
                 this.plugins = [];
                 /*@internal*/
                 /**
@@ -142780,7 +150979,7 @@ var ts;
                  * Maop does not contain entries for files that do not have unresolved imports
                  * This helps in containing the set of files to invalidate
                  */
-                this.cachedUnresolvedImportsPerFile = ts.createMap();
+                this.cachedUnresolvedImportsPerFile = new ts.Map();
                 /*@internal*/
                 this.hasAddedorRemovedFiles = false;
                 /**
@@ -142803,8 +151002,6 @@ var ts;
                 /*@internal*/
                 this.dirty = false;
                 /*@internal*/
-                this.hasChangedAutomaticTypeDirectiveNames = false;
-                /*@internal*/
                 this.typingFiles = server.emptyArray;
                 /*@internal*/
                 this.importSuggestionsCache = ts.Completions.createImportSuggestionsForFileCache();
@@ -142819,11 +151016,27 @@ var ts;
                     this.compilerOptions.allowNonTsExtensions = true;
                     this.compilerOptions.allowJs = true;
                 }
-                else if (hasExplicitListOfFiles || this.compilerOptions.allowJs || this.projectService.hasDeferredExtension()) {
+                else if (hasExplicitListOfFiles || ts.getAllowJSCompilerOption(this.compilerOptions) || this.projectService.hasDeferredExtension()) {
                     // If files are listed explicitly or allowJs is specified, allow all extensions
                     this.compilerOptions.allowNonTsExtensions = true;
                 }
-                this.languageServiceEnabled = !projectService.syntaxOnly;
+                switch (projectService.serverMode) {
+                    case ts.LanguageServiceMode.Semantic:
+                        this.languageServiceEnabled = true;
+                        break;
+                    case ts.LanguageServiceMode.PartialSemantic:
+                        this.languageServiceEnabled = true;
+                        this.compilerOptions.noResolve = true;
+                        this.compilerOptions.types = [];
+                        break;
+                    case ts.LanguageServiceMode.Syntactic:
+                        this.languageServiceEnabled = false;
+                        this.compilerOptions.noResolve = true;
+                        this.compilerOptions.types = [];
+                        break;
+                    default:
+                        ts.Debug.assertNever(projectService.serverMode);
+                }
                 this.setInternalCompilerOptionsForEmittingJsFiles();
                 var host = this.projectService.host;
                 if (this.projectService.logger.loggingEnabled()) {
@@ -142834,14 +151047,16 @@ var ts;
                 }
                 this.realpath = ts.maybeBind(host, host.realpath);
                 // Use the current directory as resolution root only if the project created using current directory string
-                this.resolutionCache = ts.createResolutionCache(this, currentDirectory && this.currentDirectory, /*logChangesWhenResolvingModule*/ true);
-                this.languageService = ts.createLanguageService(this, this.documentRegistry, projectService.syntaxOnly);
+                this.resolutionCache = ts.createResolutionCache(this, currentDirectory && this.currentDirectory, 
+                /*logChangesWhenResolvingModule*/ true);
+                this.languageService = ts.createLanguageService(this, this.documentRegistry, this.projectService.serverMode);
                 if (lastFileExceededProgramSize) {
                     this.disableLanguageService(lastFileExceededProgramSize);
                 }
                 this.markAsDirty();
-                this.projectService.pendingEnsureProjectForOpenFiles = true;
-                this.packageJsonCache = server.createPackageJsonCache(this);
+                if (projectKind !== ProjectKind.AutoImportProvider) {
+                    this.projectService.pendingEnsureProjectForOpenFiles = true;
+                }
             }
             /*@internal*/
             Project.prototype.getResolvedProjectReferenceToRedirect = function (_fileName) {
@@ -142884,8 +151099,9 @@ var ts;
                 configurable: true
             });
             /*@internal*/
-            Project.prototype.getProbableSymlinks = function (files) {
-                return this.symlinks || (this.symlinks = ts.discoverProbableSymlinks(files, this.getCanonicalFileName, this.getCurrentDirectory()));
+            Project.prototype.getSymlinkCache = function () {
+                var _a;
+                return this.symlinks || (this.symlinks = ts.discoverProbableSymlinks(((_a = this.program) === null || _a === void 0 ? void 0 : _a.getSourceFiles()) || server.emptyArray, this.getCanonicalFileName, this.getCurrentDirectory()));
             };
             // Method of LanguageServiceHost
             Project.prototype.getCompilationSettings = function () {
@@ -143002,6 +151218,27 @@ var ts;
                 return this.projectService.watchFactory.watchDirectory(this.projectService.host, directory, cb, flags, this.projectService.getWatchOptions(this), ts.WatchType.FailedLookupLocations, this);
             };
             /*@internal*/
+            Project.prototype.clearInvalidateResolutionOfFailedLookupTimer = function () {
+                return this.projectService.throttledOperations.cancel(this.getProjectName() + "FailedLookupInvalidation");
+            };
+            /*@internal*/
+            Project.prototype.scheduleInvalidateResolutionsOfFailedLookupLocations = function () {
+                var _this = this;
+                this.projectService.throttledOperations.schedule(this.getProjectName() + "FailedLookupInvalidation", /*delay*/ 1000, function () {
+                    if (_this.resolutionCache.invalidateResolutionsOfFailedLookupLocations()) {
+                        _this.projectService.delayUpdateProjectGraphAndEnsureProjectStructureForOpenFiles(_this);
+                    }
+                });
+            };
+            /*@internal*/
+            Project.prototype.invalidateResolutionsOfFailedLookupLocations = function () {
+                if (this.clearInvalidateResolutionOfFailedLookupTimer() &&
+                    this.resolutionCache.invalidateResolutionsOfFailedLookupLocations()) {
+                    this.markAsDirty();
+                    this.projectService.delayEnsureProjectForOpenFiles();
+                }
+            };
+            /*@internal*/
             Project.prototype.onInvalidatedResolution = function () {
                 this.projectService.delayUpdateProjectGraphAndEnsureProjectStructureForOpenFiles(this);
             };
@@ -143010,8 +151247,11 @@ var ts;
                 return this.projectService.watchFactory.watchDirectory(this.projectService.host, directory, cb, flags, this.projectService.getWatchOptions(this), ts.WatchType.TypeRoots, this);
             };
             /*@internal*/
+            Project.prototype.hasChangedAutomaticTypeDirectiveNames = function () {
+                return this.resolutionCache.hasChangedAutomaticTypeDirectiveNames();
+            };
+            /*@internal*/
             Project.prototype.onChangedAutomaticTypeDirectiveNames = function () {
-                this.hasChangedAutomaticTypeDirectiveNames = true;
                 this.projectService.delayUpdateProjectGraphAndEnsureProjectStructureForOpenFiles(this);
             };
             /*@internal*/
@@ -143082,9 +151322,7 @@ var ts;
                 }
                 server.updateProjectIfDirty(this);
                 this.builderState = ts.BuilderState.create(this.program, this.projectService.toCanonicalFileName, this.builderState);
-                return ts.mapDefined(ts.BuilderState.getFilesAffectedBy(this.builderState, this.program, scriptInfo.path, this.cancellationToken, function (data) { return _this.projectService.host.createHash(data); }), // TODO: GH#18217
-                function (// TODO: GH#18217
-                sourceFile) { return _this.shouldEmitFile(_this.projectService.getScriptInfoForPath(sourceFile.path)) ? sourceFile.fileName : undefined; });
+                return ts.mapDefined(ts.BuilderState.getFilesAffectedBy(this.builderState, this.program, scriptInfo.path, this.cancellationToken, ts.maybeBind(this.projectService.host, this.projectService.host.createHash)), function (sourceFile) { return _this.shouldEmitFile(_this.projectService.getScriptInfoForPath(sourceFile.path)) ? sourceFile.fileName : undefined; });
             };
             /**
              * Returns true if emit was conducted
@@ -143105,14 +151343,17 @@ var ts;
                         var dtsFiles = outputFiles.filter(function (f) { return ts.fileExtensionIs(f.name, ".d.ts" /* Dts */); });
                         if (dtsFiles.length === 1) {
                             var sourceFile = this.program.getSourceFile(scriptInfo.fileName);
-                            ts.BuilderState.updateSignatureOfFile(this.builderState, this.projectService.host.createHash(dtsFiles[0].text), sourceFile.resolvedPath);
+                            var signature = this.projectService.host.createHash ?
+                                this.projectService.host.createHash(dtsFiles[0].text) :
+                                ts.generateDjb2Hash(dtsFiles[0].text);
+                            ts.BuilderState.updateSignatureOfFile(this.builderState, signature, sourceFile.resolvedPath);
                         }
                     }
                 }
                 return { emitSkipped: emitSkipped, diagnostics: diagnostics };
             };
             Project.prototype.enableLanguageService = function () {
-                if (this.languageServiceEnabled || this.projectService.syntaxOnly) {
+                if (this.languageServiceEnabled || this.projectService.serverMode === ts.LanguageServiceMode.Syntactic) {
                     return;
                 }
                 this.languageServiceEnabled = true;
@@ -143123,11 +151364,15 @@ var ts;
                 if (!this.languageServiceEnabled) {
                     return;
                 }
-                ts.Debug.assert(!this.projectService.syntaxOnly);
+                ts.Debug.assert(this.projectService.serverMode !== ts.LanguageServiceMode.Syntactic);
                 this.languageService.cleanupSemanticCache();
                 this.languageServiceEnabled = false;
                 this.lastFileExceededProgramSize = lastFileExceededProgramSize;
                 this.builderState = undefined;
+                if (this.autoImportProviderHost) {
+                    this.autoImportProviderHost.close();
+                }
+                this.autoImportProviderHost = undefined;
                 this.resolutionCache.closeTypeRootsWatch();
                 this.clearGeneratedFileWatch();
                 this.projectService.onUpdateLanguageServiceStateForProject(this, /*languageServiceEnabled*/ false);
@@ -143180,9 +151425,7 @@ var ts;
                         this.detachScriptInfoIfNotRoot(f.fileName);
                     }
                     this.program.forEachResolvedProjectReference(function (ref) {
-                        if (ref) {
-                            _this.detachScriptInfoFromProject(ref.sourceFile.fileName);
-                        }
+                        return _this.detachScriptInfoFromProject(ref.sourceFile.fileName);
                     });
                 }
                 // Release external files
@@ -143207,11 +151450,12 @@ var ts;
                     ts.clearMap(this.missingFilesMap, ts.closeFileWatcher);
                     this.missingFilesMap = undefined;
                 }
-                if (this.packageJsonFilesMap) {
-                    ts.clearMap(this.packageJsonFilesMap, ts.closeFileWatcher);
-                    this.packageJsonFilesMap = undefined;
-                }
                 this.clearGeneratedFileWatch();
+                this.clearInvalidateResolutionOfFailedLookupTimer();
+                if (this.autoImportProviderHost) {
+                    this.autoImportProviderHost.close();
+                }
+                this.autoImportProviderHost = undefined;
                 // signal language service to release source files acquired from document registry
                 this.languageService.dispose();
                 this.languageService = undefined;
@@ -143374,13 +151618,13 @@ var ts;
                 this.markAsDirty();
             };
             Project.prototype.registerFileUpdate = function (fileName) {
-                (this.updatedFileNames || (this.updatedFileNames = ts.createMap())).set(fileName, true);
+                (this.updatedFileNames || (this.updatedFileNames = new ts.Set())).add(fileName);
             };
             /*@internal*/
             Project.prototype.markFileAsDirty = function (changedFile) {
                 this.markAsDirty();
                 if (!this.importSuggestionsCache.isEmpty()) {
-                    (this.dirtyFilesForSuggestions || (this.dirtyFilesForSuggestions = ts.createMap())).set(changedFile, true);
+                    (this.dirtyFilesForSuggestions || (this.dirtyFilesForSuggestions = new ts.Set())).add(changedFile);
                 }
             };
             Project.prototype.markAsDirty = function () {
@@ -143389,6 +151633,15 @@ var ts;
                     this.dirty = true;
                 }
             };
+            /*@internal*/
+            Project.prototype.markAutoImportProviderAsDirty = function () {
+                var _a;
+                if (this.autoImportProviderHost === false) {
+                    this.autoImportProviderHost = undefined;
+                }
+                (_a = this.autoImportProviderHost) === null || _a === void 0 ? void 0 : _a.markAsDirty();
+                this.importSuggestionsCache.clear();
+            };
             /* @internal */
             Project.prototype.onFileAddedOrRemoved = function () {
                 this.hasAddedorRemovedFiles = true;
@@ -143398,6 +151651,7 @@ var ts;
              * @returns: true if set of files in the project stays the same and false - otherwise.
              */
             Project.prototype.updateGraph = function () {
+                var _a;
                 ts.perfLogger.logStartUpdateGraph();
                 this.resolutionCache.startRecordingFilesWithChangedResolutions();
                 var hasNewProgram = this.updateGraphWorker();
@@ -143411,7 +151665,7 @@ var ts;
                 }
                 // update builder only if language service is enabled
                 // otherwise tell it to drop its internal state
-                if (this.languageServiceEnabled) {
+                if (this.languageServiceEnabled && this.projectService.serverMode === ts.LanguageServiceMode.Semantic) {
                     // 1. no changes in structure, no changes in unresolved imports - do nothing
                     // 2. no changes in structure, unresolved imports were changed - collect unresolved imports for all files
                     // (can reuse cached imports for files that were not changed)
@@ -143426,9 +151680,19 @@ var ts;
                 else {
                     this.lastCachedUnresolvedImportsList = undefined;
                 }
+                var isFirstLoad = this.projectProgramVersion === 0;
                 if (hasNewProgram) {
                     this.projectProgramVersion++;
                 }
+                if (hasAddedorRemovedFiles) {
+                    if (!this.autoImportProviderHost)
+                        this.autoImportProviderHost = undefined;
+                    (_a = this.autoImportProviderHost) === null || _a === void 0 ? void 0 : _a.markAsDirty();
+                }
+                if (isFirstLoad) {
+                    // Preload auto import provider so it's not created during completions request
+                    this.getPackageJsonAutoImportProvider();
+                }
                 ts.perfLogger.logStopUpdateGraph();
                 return !hasNewProgram;
             };
@@ -143464,8 +151728,7 @@ var ts;
                 // bump up the version if
                 // - oldProgram is not set - this is a first time updateGraph is called
                 // - newProgram is different from the old program and structure of the old program was not reused.
-                var hasNewProgram = this.program && (!oldProgram || (this.program !== oldProgram && !(oldProgram.structureIsReused & 2 /* Completely */)));
-                this.hasChangedAutomaticTypeDirectiveNames = false;
+                var hasNewProgram = this.program && (!oldProgram || (this.program !== oldProgram && !(this.program.structureIsReused & 2 /* Completely */)));
                 if (hasNewProgram) {
                     if (oldProgram) {
                         for (var _i = 0, _a = oldProgram.getSourceFiles(); _i < _a.length; _i++) {
@@ -143477,18 +151740,18 @@ var ts;
                                 this.detachScriptInfoFromProject(f.fileName, !!this.program.getSourceFileByPath(f.path));
                             }
                         }
-                        oldProgram.forEachResolvedProjectReference(function (resolvedProjectReference, resolvedProjectReferencePath) {
-                            if (resolvedProjectReference && !_this.program.getResolvedProjectReferenceByPath(resolvedProjectReferencePath)) {
+                        oldProgram.forEachResolvedProjectReference(function (resolvedProjectReference) {
+                            if (!_this.program.getResolvedProjectReferenceByPath(resolvedProjectReference.sourceFile.path)) {
                                 _this.detachScriptInfoFromProject(resolvedProjectReference.sourceFile.fileName);
                             }
                         });
                     }
                     // Update the missing file paths watcher
-                    ts.updateMissingFilePathsWatch(this.program, this.missingFilesMap || (this.missingFilesMap = ts.createMap()), 
+                    ts.updateMissingFilePathsWatch(this.program, this.missingFilesMap || (this.missingFilesMap = new ts.Map()), 
                     // Watch the missing files
                     function (missingFilePath) { return _this.addMissingFileWatcher(missingFilePath); });
                     if (this.generatedFilesMap) {
-                        var outPath = this.compilerOptions.outFile && this.compilerOptions.out;
+                        var outPath = ts.outFile(this.compilerOptions);
                         if (isGeneratedFileWatcher(this.generatedFilesMap)) {
                             // --out
                             if (!outPath || !this.isValidGeneratedFileWatcher(ts.removeFileExtension(outPath) + ".d.ts" /* Dts */, this.generatedFilesMap)) {
@@ -143514,12 +151777,12 @@ var ts;
                         }
                     }
                     // Watch the type locations that would be added to program as part of automatic type resolutions
-                    if (this.languageServiceEnabled) {
+                    if (this.languageServiceEnabled && this.projectService.serverMode === ts.LanguageServiceMode.Semantic) {
                         this.resolutionCache.updateTypeRootsWatch();
                     }
                 }
                 if (!this.importSuggestionsCache.isEmpty()) {
-                    if (this.hasAddedorRemovedFiles || oldProgram && !oldProgram.structureIsReused) {
+                    if (this.hasAddedorRemovedFiles || oldProgram && !this.program.structureIsReused) {
                         this.importSuggestionsCache.clear();
                     }
                     else if (this.dirtyFilesForSuggestions && oldProgram && this.program) {
@@ -143550,16 +151813,20 @@ var ts;
                     scriptInfo.attachToProject(_this);
                 }, function (removed) { return _this.detachScriptInfoFromProject(removed); });
                 var elapsed = ts.timestamp() - start;
-                this.projectService.sendUpdateGraphPerformanceEvent(elapsed);
+                this.sendPerformanceEvent("UpdateGraph", elapsed);
                 this.writeLog("Finishing updateGraphWorker: Project: " + this.getProjectName() + " Version: " + this.getProjectVersion() + " structureChanged: " + hasNewProgram + " Elapsed: " + elapsed + "ms");
                 if (this.hasAddedorRemovedFiles) {
                     this.print(/*writeProjectFileNames*/ true);
                 }
                 else if (this.program !== oldProgram) {
-                    this.writeLog("Different program with same set of files:: oldProgram.structureIsReused:: " + (oldProgram && oldProgram.structureIsReused));
+                    this.writeLog("Different program with same set of files:: structureIsReused:: " + this.program.structureIsReused);
                 }
                 return hasNewProgram;
             };
+            /* @internal */
+            Project.prototype.sendPerformanceEvent = function (kind, durationMs) {
+                this.projectService.sendPerformanceEvent(kind, durationMs);
+            };
             /*@internal*/
             Project.prototype.sourceFileHasChangedOwnImportSuggestions = function (oldSourceFile, newSourceFile) {
                 if (!oldSourceFile && !newSourceFile) {
@@ -143637,7 +151904,7 @@ var ts;
             };
             /* @internal */
             Project.prototype.addGeneratedFileWatch = function (generatedFile, sourceFile) {
-                if (this.compilerOptions.outFile || this.compilerOptions.out) {
+                if (ts.outFile(this.compilerOptions)) {
                     // Single watcher
                     if (!this.generatedFilesMap) {
                         this.generatedFilesMap = this.createGeneratedFileWatcher(generatedFile);
@@ -143655,7 +151922,7 @@ var ts;
                             return;
                     }
                     else {
-                        this.generatedFilesMap = ts.createMap();
+                        this.generatedFilesMap = new ts.Map();
                     }
                     this.generatedFilesMap.set(path, this.createGeneratedFileWatcher(generatedFile));
                 }
@@ -143714,6 +151981,9 @@ var ts;
                 this.writeLog("Project '" + this.projectName + "' (" + ProjectKind[this.projectKind] + ")");
                 this.writeLog(this.filesToString(writeProjectFileNames && this.projectService.logger.hasLevel(server.LogLevel.verbose)));
                 this.writeLog("-----------------------------------------------");
+                if (this.autoImportProviderHost) {
+                    this.autoImportProviderHost.print(/*writeProjectFileNames*/ false);
+                }
             };
             Project.prototype.setCompilerOptions = function (compilerOptions) {
                 if (compilerOptions) {
@@ -143738,6 +152008,14 @@ var ts;
             Project.prototype.getWatchOptions = function () {
                 return this.watchOptions;
             };
+            Project.prototype.setTypeAcquisition = function (newTypeAcquisition) {
+                if (newTypeAcquisition) {
+                    this.typeAcquisition = this.removeLocalTypingsFromTypeAcquisition(newTypeAcquisition);
+                }
+            };
+            Project.prototype.getTypeAcquisition = function () {
+                return this.typeAcquisition || {};
+            };
             /* @internal */
             Project.prototype.getChangesSinceVersion = function (lastKnownVersion, includeProjectReferenceRedirectInfo) {
                 var _this = this;
@@ -143941,60 +152219,79 @@ var ts;
             };
             /*@internal*/
             Project.prototype.getPackageJsonsVisibleToFile = function (fileName, rootDir) {
-                var packageJsonCache = this.packageJsonCache;
-                var watchPackageJsonFile = this.watchPackageJsonFile.bind(this);
-                var toPath = this.toPath.bind(this);
-                var rootPath = rootDir && toPath(rootDir);
-                var filePath = toPath(fileName);
-                var result = [];
-                ts.forEachAncestorDirectory(ts.getDirectoryPath(filePath), function processDirectory(directory) {
-                    switch (packageJsonCache.directoryHasPackageJson(directory)) {
-                        // Sync and check same directory again
-                        case 1 /* Maybe */:
-                            packageJsonCache.searchDirectoryAndAncestors(directory);
-                            return processDirectory(directory);
-                        // Check package.json
-                        case -1 /* True */:
-                            var packageJsonFileName = ts.combinePaths(directory, "package.json");
-                            watchPackageJsonFile(packageJsonFileName);
-                            var info = packageJsonCache.getInDirectory(directory);
-                            if (info)
-                                result.push(info);
-                    }
-                    if (rootPath && rootPath === toPath(directory)) {
-                        return true;
-                    }
-                });
-                return result;
+                if (this.projectService.serverMode !== ts.LanguageServiceMode.Semantic)
+                    return server.emptyArray;
+                return this.projectService.getPackageJsonsVisibleToFile(fileName, rootDir);
             };
             /*@internal*/
-            Project.prototype.onAddPackageJson = function (path) {
-                this.packageJsonCache.addOrUpdate(path);
-                this.watchPackageJsonFile(path);
+            Project.prototype.getPackageJsonsForAutoImport = function (rootDir) {
+                var packageJsons = this.getPackageJsonsVisibleToFile(ts.combinePaths(this.currentDirectory, ts.inferredTypesContainingFile), rootDir);
+                this.packageJsonsForAutoImport = new ts.Set(packageJsons.map(function (p) { return p.fileName; }));
+                return packageJsons;
             };
             /*@internal*/
             Project.prototype.getImportSuggestionsCache = function () {
                 return this.importSuggestionsCache;
             };
-            Project.prototype.watchPackageJsonFile = function (path) {
-                var _this = this;
-                var watchers = this.packageJsonFilesMap || (this.packageJsonFilesMap = ts.createMap());
-                if (!watchers.has(path)) {
-                    watchers.set(path, this.projectService.watchFactory.watchFile(this.projectService.host, path, function (fileName, eventKind) {
-                        var path = _this.toPath(fileName);
-                        switch (eventKind) {
-                            case ts.FileWatcherEventKind.Created:
-                                return ts.Debug.fail();
-                            case ts.FileWatcherEventKind.Changed:
-                                _this.packageJsonCache.addOrUpdate(path);
-                                break;
-                            case ts.FileWatcherEventKind.Deleted:
-                                _this.packageJsonCache.delete(path);
-                                watchers.get(path).close();
-                                watchers.delete(path);
-                        }
-                    }, ts.PollingInterval.Low, this.projectService.getWatchOptions(this), ts.WatchType.PackageJsonFile));
+            /*@internal*/
+            Project.prototype.includePackageJsonAutoImports = function () {
+                if (this.projectService.includePackageJsonAutoImports() === 0 /* Off */ ||
+                    !this.languageServiceEnabled ||
+                    ts.isInsideNodeModules(this.currentDirectory) ||
+                    !this.isDefaultProjectForOpenFiles()) {
+                    return 0 /* Off */;
                 }
+                return this.projectService.includePackageJsonAutoImports();
+            };
+            /*@internal*/
+            Project.prototype.getModuleResolutionHostForAutoImportProvider = function () {
+                var _a, _b;
+                if (this.program) {
+                    return {
+                        fileExists: this.program.fileExists,
+                        directoryExists: this.program.directoryExists,
+                        realpath: this.program.realpath || ((_a = this.projectService.host.realpath) === null || _a === void 0 ? void 0 : _a.bind(this.projectService.host)),
+                        getCurrentDirectory: this.getCurrentDirectory.bind(this),
+                        readFile: this.projectService.host.readFile.bind(this.projectService.host),
+                        getDirectories: this.projectService.host.getDirectories.bind(this.projectService.host),
+                        trace: (_b = this.projectService.host.trace) === null || _b === void 0 ? void 0 : _b.bind(this.projectService.host),
+                    };
+                }
+                return this.projectService.host;
+            };
+            /*@internal*/
+            Project.prototype.getPackageJsonAutoImportProvider = function () {
+                if (this.autoImportProviderHost === false) {
+                    return undefined;
+                }
+                if (this.projectService.serverMode !== ts.LanguageServiceMode.Semantic) {
+                    this.autoImportProviderHost = false;
+                    return undefined;
+                }
+                if (this.autoImportProviderHost) {
+                    server.updateProjectIfDirty(this.autoImportProviderHost);
+                    if (this.autoImportProviderHost.isEmpty()) {
+                        this.autoImportProviderHost.close();
+                        this.autoImportProviderHost = undefined;
+                        return undefined;
+                    }
+                    return this.autoImportProviderHost.getCurrentProgram();
+                }
+                var dependencySelection = this.includePackageJsonAutoImports();
+                if (dependencySelection) {
+                    var start = ts.timestamp();
+                    this.autoImportProviderHost = AutoImportProviderProject.create(dependencySelection, this, this.getModuleResolutionHostForAutoImportProvider(), this.documentRegistry);
+                    if (this.autoImportProviderHost) {
+                        server.updateProjectIfDirty(this.autoImportProviderHost);
+                        this.sendPerformanceEvent("CreatePackageJsonAutoImportProvider", ts.timestamp() - start);
+                        return this.autoImportProviderHost.getCurrentProgram();
+                    }
+                }
+            };
+            /*@internal*/
+            Project.prototype.isDefaultProjectForOpenFiles = function () {
+                var _this = this;
+                return !!ts.forEachEntry(this.projectService.openFiles, function (_, fileName) { return _this.projectService.tryGetDefaultProjectForFile(server.toNormalizedPath(fileName)) === _this; });
             };
             return Project;
         }());
@@ -144021,6 +152318,10 @@ var ts;
                 return unresolvedImports || server.emptyArray;
             });
         }
+        function createProjectNameFactoryWithCounter(nameFactory) {
+            var nextId = 1;
+            return function () { return nameFactory(nextId++); };
+        }
         /**
          * If a file is opened and no tsconfig (or jsconfig) is found,
          * the file and its imports/references are put into an InferredProject.
@@ -144028,13 +152329,14 @@ var ts;
         var InferredProject = /** @class */ (function (_super) {
             __extends(InferredProject, _super);
             /*@internal*/
-            function InferredProject(projectService, documentRegistry, compilerOptions, watchOptions, projectRootPath, currentDirectory, pluginConfigOverrides) {
+            function InferredProject(projectService, documentRegistry, compilerOptions, watchOptions, projectRootPath, currentDirectory, pluginConfigOverrides, typeAcquisition) {
                 var _this = _super.call(this, InferredProject.newName(), ProjectKind.Inferred, projectService, documentRegistry, 
                 // TODO: GH#18217
                 /*files*/ undefined, 
                 /*lastFileExceededProgramSize*/ undefined, compilerOptions, 
                 /*compileOnSaveEnabled*/ false, watchOptions, projectService.host, currentDirectory) || this;
                 _this._isJsInferredProject = false;
+                _this.typeAcquisition = typeAcquisition;
                 _this.projectRootPath = projectRootPath && projectService.toCanonicalFileName(projectRootPath);
                 if (!projectRootPath && !projectService.useSingleInferredProject) {
                     _this.canonicalCurrentDirectory = projectService.toCanonicalFileName(_this.currentDirectory);
@@ -144097,23 +152399,135 @@ var ts;
                 _super.prototype.close.call(this);
             };
             InferredProject.prototype.getTypeAcquisition = function () {
-                return {
+                return this.typeAcquisition || {
                     enable: allRootFilesAreJsOrDts(this),
-                    include: [],
-                    exclude: []
+                    include: ts.emptyArray,
+                    exclude: ts.emptyArray
                 };
             };
-            InferredProject.newName = (function () {
-                var nextId = 1;
-                return function () {
-                    var id = nextId;
-                    nextId++;
-                    return server.makeInferredProjectName(id);
-                };
-            })();
+            InferredProject.newName = createProjectNameFactoryWithCounter(server.makeInferredProjectName);
             return InferredProject;
         }(Project));
         server.InferredProject = InferredProject;
+        var AutoImportProviderProject = /** @class */ (function (_super) {
+            __extends(AutoImportProviderProject, _super);
+            /*@internal*/
+            function AutoImportProviderProject(hostProject, initialRootNames, documentRegistry, compilerOptions) {
+                var _this = _super.call(this, AutoImportProviderProject.newName(), ProjectKind.AutoImportProvider, hostProject.projectService, documentRegistry, 
+                /*hasExplicitListOfFiles*/ false, 
+                /*lastFileExceededProgramSize*/ undefined, compilerOptions, 
+                /*compileOnSaveEnabled*/ false, hostProject.getWatchOptions(), hostProject.projectService.host, hostProject.currentDirectory) || this;
+                _this.hostProject = hostProject;
+                _this.rootFileNames = initialRootNames;
+                return _this;
+            }
+            /*@internal*/
+            AutoImportProviderProject.getRootFileNames = function (dependencySelection, hostProject, moduleResolutionHost, compilerOptions) {
+                var _a, _b, _c, _d;
+                if (!dependencySelection) {
+                    return ts.emptyArray;
+                }
+                var dependencyNames;
+                var rootNames;
+                var rootFileName = ts.combinePaths(hostProject.currentDirectory, ts.inferredTypesContainingFile);
+                var packageJsons = hostProject.getPackageJsonsForAutoImport(ts.combinePaths(hostProject.currentDirectory, rootFileName));
+                for (var _i = 0, packageJsons_1 = packageJsons; _i < packageJsons_1.length; _i++) {
+                    var packageJson = packageJsons_1[_i];
+                    (_a = packageJson.dependencies) === null || _a === void 0 ? void 0 : _a.forEach(function (_, dependenyName) { return addDependency(dependenyName); });
+                    (_b = packageJson.peerDependencies) === null || _b === void 0 ? void 0 : _b.forEach(function (_, dependencyName) { return addDependency(dependencyName); });
+                }
+                if (dependencyNames) {
+                    var resolutions = ts.map(ts.arrayFrom(dependencyNames.keys()), function (name) { return ts.resolveTypeReferenceDirective(name, rootFileName, compilerOptions, moduleResolutionHost); });
+                    for (var _e = 0, resolutions_1 = resolutions; _e < resolutions_1.length; _e++) {
+                        var resolution = resolutions_1[_e];
+                        if (!((_c = resolution.resolvedTypeReferenceDirective) === null || _c === void 0 ? void 0 : _c.resolvedFileName))
+                            continue;
+                        var resolvedFileName = resolution.resolvedTypeReferenceDirective.resolvedFileName;
+                        var fileName = ((_d = moduleResolutionHost.realpath) === null || _d === void 0 ? void 0 : _d.call(moduleResolutionHost, resolvedFileName)) || resolvedFileName;
+                        if (!hostProject.getCurrentProgram().getSourceFile(fileName) && !hostProject.getCurrentProgram().getSourceFile(resolvedFileName)) {
+                            rootNames = ts.append(rootNames, fileName);
+                            // Avoid creating a large project that would significantly slow down time to editor interactivity
+                            if (dependencySelection === 2 /* Auto */ && rootNames.length > this.maxDependencies) {
+                                return ts.emptyArray;
+                            }
+                        }
+                    }
+                }
+                return rootNames || ts.emptyArray;
+                function addDependency(dependency) {
+                    if (!ts.startsWith(dependency, "@types/")) {
+                        (dependencyNames || (dependencyNames = new ts.Set())).add(dependency);
+                    }
+                }
+            };
+            /*@internal*/
+            AutoImportProviderProject.create = function (dependencySelection, hostProject, moduleResolutionHost, documentRegistry) {
+                if (dependencySelection === 0 /* Off */) {
+                    return undefined;
+                }
+                var compilerOptions = __assign(__assign({}, hostProject.getCompilerOptions()), { noLib: true, diagnostics: false, skipLibCheck: true, types: ts.emptyArray, lib: ts.emptyArray, sourceMap: false });
+                var rootNames = this.getRootFileNames(dependencySelection, hostProject, moduleResolutionHost, compilerOptions);
+                if (!rootNames.length) {
+                    return undefined;
+                }
+                return new AutoImportProviderProject(hostProject, rootNames, documentRegistry, compilerOptions);
+            };
+            /*@internal*/
+            AutoImportProviderProject.prototype.isEmpty = function () {
+                return !ts.some(this.rootFileNames);
+            };
+            AutoImportProviderProject.prototype.isOrphan = function () {
+                return true;
+            };
+            AutoImportProviderProject.prototype.updateGraph = function () {
+                var rootFileNames = this.rootFileNames;
+                if (!rootFileNames) {
+                    rootFileNames = AutoImportProviderProject.getRootFileNames(this.hostProject.includePackageJsonAutoImports(), this.hostProject, this.hostProject.getModuleResolutionHostForAutoImportProvider(), this.getCompilationSettings());
+                }
+                this.projectService.setFileNamesOfAutoImportProviderProject(this, rootFileNames);
+                this.rootFileNames = rootFileNames;
+                this.hostProject.getImportSuggestionsCache().clear();
+                return _super.prototype.updateGraph.call(this);
+            };
+            AutoImportProviderProject.prototype.markAsDirty = function () {
+                this.rootFileNames = undefined;
+                _super.prototype.markAsDirty.call(this);
+            };
+            AutoImportProviderProject.prototype.getScriptFileNames = function () {
+                return this.rootFileNames || ts.emptyArray;
+            };
+            AutoImportProviderProject.prototype.getLanguageService = function () {
+                throw new Error("AutoImportProviderProject language service should never be used. To get the program, use `project.getCurrentProgram()`.");
+            };
+            AutoImportProviderProject.prototype.markAutoImportProviderAsDirty = function () {
+                throw new Error("AutoImportProviderProject is an auto import provider; use `markAsDirty()` instead.");
+            };
+            AutoImportProviderProject.prototype.getModuleResolutionHostForAutoImportProvider = function () {
+                throw new Error("AutoImportProviderProject cannot provide its own host; use `hostProject.getModuleResolutionHostForAutomImportProvider()` instead.");
+            };
+            AutoImportProviderProject.prototype.getProjectReferences = function () {
+                return this.hostProject.getProjectReferences();
+            };
+            AutoImportProviderProject.prototype.useSourceOfProjectReferenceRedirect = function () {
+                return true;
+            };
+            /*@internal*/
+            AutoImportProviderProject.prototype.includePackageJsonAutoImports = function () {
+                return 0 /* Off */;
+            };
+            AutoImportProviderProject.prototype.getTypeAcquisition = function () {
+                return { enable: false };
+            };
+            /*@internal*/
+            AutoImportProviderProject.prototype.getSymlinkCache = function () {
+                return this.hostProject.getSymlinkCache();
+            };
+            AutoImportProviderProject.newName = createProjectNameFactoryWithCounter(server.makeAutoImportProviderProjectName);
+            /*@internal*/
+            AutoImportProviderProject.maxDependencies = 10;
+            return AutoImportProviderProject;
+        }(Project));
+        server.AutoImportProviderProject = AutoImportProviderProject;
         /**
          * If a file is opened, the server will look for a tsconfig (or jsconfig)
          * and if successful create a ConfiguredProject for it.
@@ -144130,7 +152544,7 @@ var ts;
                 /*compileOnSaveEnabled*/ false, 
                 /*watchOptions*/ undefined, cachedDirectoryStructureHost, ts.getDirectoryPath(configFileName)) || this;
                 /* @internal */
-                _this.openFileWatchTriggered = ts.createMap();
+                _this.openFileWatchTriggered = new ts.Map();
                 /*@internal*/
                 _this.canConfigFileJsonReportNoInputFiles = false;
                 /** Ref count to the project when opened from external project */
@@ -144177,6 +152591,7 @@ var ts;
              * @returns: true if set of files in the project stays the same and false - otherwise.
              */
             ConfiguredProject.prototype.updateGraph = function () {
+                var isInitialLoad = this.isInitialLoadPending();
                 this.isInitialLoadPending = ts.returnFalse;
                 var reloadLevel = this.pendingReload;
                 this.pendingReload = ts.ConfigFileProgramReloadLevel.None;
@@ -144190,7 +152605,7 @@ var ts;
                         this.openFileWatchTriggered.clear();
                         var reason = ts.Debug.checkDefined(this.pendingReloadReason);
                         this.pendingReloadReason = undefined;
-                        this.projectService.reloadConfiguredProject(this, reason);
+                        this.projectService.reloadConfiguredProject(this, reason, isInitialLoad);
                         result = true;
                         break;
                     default:
@@ -144218,7 +152633,7 @@ var ts;
             /*@internal*/
             ConfiguredProject.prototype.setPotentialProjectReference = function (canonicalConfigPath) {
                 ts.Debug.assert(this.isInitialLoadPending());
-                (this.potentialProjectReferences || (this.potentialProjectReferences = ts.createMap())).set(canonicalConfigPath, true);
+                (this.potentialProjectReferences || (this.potentialProjectReferences = new ts.Set())).add(canonicalConfigPath);
             };
             /*@internal*/
             ConfiguredProject.prototype.getResolvedProjectReferenceToRedirect = function (fileName) {
@@ -144226,6 +152641,11 @@ var ts;
                 return program && program.getResolvedProjectReferenceToRedirect(fileName);
             };
             /*@internal*/
+            ConfiguredProject.prototype.forEachResolvedProjectReference = function (cb) {
+                var _a;
+                return (_a = this.getCurrentProgram()) === null || _a === void 0 ? void 0 : _a.forEachResolvedProjectReference(cb);
+            };
+            /*@internal*/
             ConfiguredProject.prototype.enablePluginsWithOptions = function (options, pluginConfigOverrides) {
                 var host = this.projectService.host;
                 if (!host.require) {
@@ -144264,16 +152684,10 @@ var ts;
             ConfiguredProject.prototype.setProjectErrors = function (projectErrors) {
                 this.projectErrors = projectErrors;
             };
-            ConfiguredProject.prototype.setTypeAcquisition = function (newTypeAcquisition) {
-                this.typeAcquisition = this.removeLocalTypingsFromTypeAcquisition(newTypeAcquisition);
-            };
-            ConfiguredProject.prototype.getTypeAcquisition = function () {
-                return this.typeAcquisition;
-            };
             /*@internal*/
             ConfiguredProject.prototype.watchWildcards = function (wildcardDirectories) {
                 var _this = this;
-                ts.updateWatchingWildcardDirectories(this.directoriesWatchedForWildcards || (this.directoriesWatchedForWildcards = ts.createMap()), wildcardDirectories, 
+                ts.updateWatchingWildcardDirectories(this.directoriesWatchedForWildcards || (this.directoriesWatchedForWildcards = new ts.Map()), wildcardDirectories, 
                 // Create new directory watcher
                 function (directory, flags) { return _this.projectService.watchWildcardDirectory(directory, flags, _this); });
             };
@@ -144310,10 +152724,9 @@ var ts;
                     !this.canConfigFileJsonReportNoInputFiles;
             };
             /* @internal */
-            /** Find the configured project from the project references in this solution which contains the info directly */
-            ConfiguredProject.prototype.getDefaultChildProjectFromSolution = function (info) {
-                ts.Debug.assert(this.isSolution());
-                return server.forEachResolvedProjectReferenceProject(this, function (child) { return server.projectContainsInfoDirectly(child, info) ?
+            /** Find the configured project from the project references in project which contains the info directly */
+            ConfiguredProject.prototype.getDefaultChildProjectFromProjectWithReferences = function (info) {
+                return server.forEachResolvedProjectReferenceProject(this, info.path, function (child) { return server.projectContainsInfoDirectly(child, info) ?
                     child :
                     undefined; }, server.ProjectReferenceProjectLoadKind.Find);
             };
@@ -144335,13 +152748,14 @@ var ts;
                     // In that case keep the project alive if there are open files impacted by this project
                     return !!configFileExistenceInfo.openFilesImpactedByConfigFile.size;
                 }
-                var isSolution = this.isSolution();
                 // If there is no pending update for this project,
                 // We know exact set of open files that get impacted by this configured project as the files in the project
                 // The project is referenced only if open files impacted by this project are present in this project
-                return ts.forEachEntry(configFileExistenceInfo.openFilesImpactedByConfigFile, function (_value, infoPath) { return isSolution ?
-                    !!_this.getDefaultChildProjectFromSolution(_this.projectService.getScriptInfoForPath(infoPath)) :
-                    _this.containsScriptInfo(_this.projectService.getScriptInfoForPath(infoPath)); }) || false;
+                return ts.forEachEntry(configFileExistenceInfo.openFilesImpactedByConfigFile, function (_value, infoPath) {
+                    var info = _this.projectService.getScriptInfoForPath(infoPath);
+                    return _this.containsScriptInfo(info) ||
+                        !!server.forEachResolvedProjectReferenceProject(_this, info.path, function (child) { return child.containsScriptInfo(info); }, server.ProjectReferenceProjectLoadKind.Find);
+                }) || false;
             };
             /*@internal*/
             ConfiguredProject.prototype.hasExternalProjectRef = function () {
@@ -144381,16 +152795,6 @@ var ts;
             ExternalProject.prototype.getExcludedFiles = function () {
                 return this.excludedFiles;
             };
-            ExternalProject.prototype.getTypeAcquisition = function () {
-                return this.typeAcquisition;
-            };
-            ExternalProject.prototype.setTypeAcquisition = function (newTypeAcquisition) {
-                ts.Debug.assert(!!newTypeAcquisition, "newTypeAcquisition may not be null/undefined");
-                ts.Debug.assert(!!newTypeAcquisition.include, "newTypeAcquisition.include may not be null/undefined");
-                ts.Debug.assert(!!newTypeAcquisition.exclude, "newTypeAcquisition.exclude may not be null/undefined");
-                ts.Debug.assert(typeof newTypeAcquisition.enable === "boolean", "newTypeAcquisition.enable may not be null/undefined");
-                this.typeAcquisition = this.removeLocalTypingsFromTypeAcquisition(newTypeAcquisition);
-            };
             return ExternalProject;
         }(Project));
         server.ExternalProject = ExternalProject;
@@ -144427,7 +152831,7 @@ var ts;
         server.ProjectInfoTelemetryEvent = "projectInfo";
         server.OpenFileInfoTelemetryEvent = "openFileInfo";
         function prepareConvertersForEnumLikeCompilerOptions(commandLineOptions) {
-            var map = ts.createMap();
+            var map = new ts.Map();
             for (var _i = 0, commandLineOptions_1 = commandLineOptions; _i < commandLineOptions_1.length; _i++) {
                 var option = commandLineOptions_1[_i];
                 if (typeof option.type === "object") {
@@ -144443,11 +152847,11 @@ var ts;
         }
         var compilerOptionConverters = prepareConvertersForEnumLikeCompilerOptions(ts.optionDeclarations);
         var watchOptionsConverters = prepareConvertersForEnumLikeCompilerOptions(ts.optionsForWatch);
-        var indentStyle = ts.createMapFromTemplate({
+        var indentStyle = new ts.Map(ts.getEntries({
             none: ts.IndentStyle.None,
             block: ts.IndentStyle.Block,
             smart: ts.IndentStyle.Smart
-        });
+        }));
         /**
          * How to understand this block:
          *  * The 'match' property is a regexp that matches a filename.
@@ -144526,6 +152930,17 @@ var ts;
             return result;
         }
         server.convertWatchOptions = convertWatchOptions;
+        function convertTypeAcquisition(protocolOptions) {
+            var result;
+            ts.typeAcquisitionDeclarations.forEach(function (option) {
+                var propertyValue = protocolOptions[option.name];
+                if (propertyValue === undefined)
+                    return;
+                (result || (result = {}))[option.name] = propertyValue;
+            });
+            return result;
+        }
+        server.convertTypeAcquisition = convertTypeAcquisition;
         function tryConvertScriptKindName(scriptKindName) {
             return ts.isString(scriptKindName) ? convertScriptKindName(scriptKindName) : scriptKindName;
         }
@@ -144611,33 +153026,72 @@ var ts;
             /** Find existing project or create and load it for the project reference */
             ProjectReferenceProjectLoadKind[ProjectReferenceProjectLoadKind["FindCreateLoad"] = 2] = "FindCreateLoad";
         })(ProjectReferenceProjectLoadKind = server.ProjectReferenceProjectLoadKind || (server.ProjectReferenceProjectLoadKind = {}));
-        function forEachResolvedProjectReferenceProject(project, cb, projectReferenceProjectLoadKind, reason) {
-            return forEachResolvedProjectReference(project, function (ref) {
+        function forEachResolvedProjectReferenceProject(project, fileName, cb, projectReferenceProjectLoadKind, reason) {
+            var _a;
+            var resolvedRefs = (_a = project.getCurrentProgram()) === null || _a === void 0 ? void 0 : _a.getResolvedProjectReferences();
+            if (!resolvedRefs)
+                return undefined;
+            var seenResolvedRefs;
+            var possibleDefaultRef = fileName ? project.getResolvedProjectReferenceToRedirect(fileName) : undefined;
+            if (possibleDefaultRef) {
+                // Try to find the name of the file directly through resolved project references
+                var configFileName = server.toNormalizedPath(possibleDefaultRef.sourceFile.fileName);
+                var child = project.projectService.findConfiguredProjectByProjectName(configFileName);
+                if (child) {
+                    var result = cb(child);
+                    if (result)
+                        return result;
+                }
+                else if (projectReferenceProjectLoadKind !== ProjectReferenceProjectLoadKind.Find) {
+                    seenResolvedRefs = new ts.Map();
+                    // Try to see if this project can be loaded
+                    var result = forEachResolvedProjectReferenceProjectWorker(resolvedRefs, project.getCompilerOptions(), function (ref, loadKind) { return possibleDefaultRef === ref ? callback(ref, loadKind) : undefined; }, projectReferenceProjectLoadKind, project.projectService, seenResolvedRefs);
+                    if (result)
+                        return result;
+                    // Cleanup seenResolvedRefs
+                    seenResolvedRefs.clear();
+                }
+            }
+            return forEachResolvedProjectReferenceProjectWorker(resolvedRefs, project.getCompilerOptions(), function (ref, loadKind) { return possibleDefaultRef !== ref ? callback(ref, loadKind) : undefined; }, projectReferenceProjectLoadKind, project.projectService, seenResolvedRefs);
+            function callback(ref, loadKind) {
+                var configFileName = server.toNormalizedPath(ref.sourceFile.fileName);
+                var child = project.projectService.findConfiguredProjectByProjectName(configFileName) || (loadKind === ProjectReferenceProjectLoadKind.Find ?
+                    undefined :
+                    loadKind === ProjectReferenceProjectLoadKind.FindCreate ?
+                        project.projectService.createConfiguredProject(configFileName) :
+                        loadKind === ProjectReferenceProjectLoadKind.FindCreateLoad ?
+                            project.projectService.createAndLoadConfiguredProject(configFileName, reason) :
+                            ts.Debug.assertNever(loadKind));
+                return child && cb(child);
+            }
+        }
+        server.forEachResolvedProjectReferenceProject = forEachResolvedProjectReferenceProject;
+        function forEachResolvedProjectReferenceProjectWorker(resolvedProjectReferences, parentOptions, cb, projectReferenceProjectLoadKind, projectService, seenResolvedRefs) {
+            var loadKind = parentOptions.disableReferencedProjectLoad ? ProjectReferenceProjectLoadKind.Find : projectReferenceProjectLoadKind;
+            return ts.forEach(resolvedProjectReferences, function (ref) {
                 if (!ref)
                     return undefined;
                 var configFileName = server.toNormalizedPath(ref.sourceFile.fileName);
-                var child = project.projectService.findConfiguredProjectByProjectName(configFileName) || (projectReferenceProjectLoadKind === ProjectReferenceProjectLoadKind.FindCreate ?
-                    project.projectService.createConfiguredProject(configFileName) :
-                    projectReferenceProjectLoadKind === ProjectReferenceProjectLoadKind.FindCreateLoad ?
-                        project.projectService.createAndLoadConfiguredProject(configFileName, reason) :
-                        undefined);
-                return child && cb(child, configFileName);
+                var canonicalPath = projectService.toCanonicalFileName(configFileName);
+                var seenValue = seenResolvedRefs === null || seenResolvedRefs === void 0 ? void 0 : seenResolvedRefs.get(canonicalPath);
+                if (seenValue !== undefined && seenValue >= loadKind) {
+                    return undefined;
+                }
+                var result = cb(ref, loadKind);
+                if (result) {
+                    return result;
+                }
+                (seenResolvedRefs || (seenResolvedRefs = new ts.Map())).set(canonicalPath, loadKind);
+                return ref.references && forEachResolvedProjectReferenceProjectWorker(ref.references, ref.commandLine.options, cb, loadKind, projectService, seenResolvedRefs);
             });
         }
-        server.forEachResolvedProjectReferenceProject = forEachResolvedProjectReferenceProject;
-        /*@internal*/
-        function forEachResolvedProjectReference(project, cb) {
-            var program = project.getCurrentProgram();
-            return program && program.forEachResolvedProjectReference(cb);
-        }
-        server.forEachResolvedProjectReference = forEachResolvedProjectReference;
         function forEachPotentialProjectReference(project, cb) {
             return project.potentialProjectReferences &&
                 ts.forEachKey(project.potentialProjectReferences, cb);
         }
         function forEachAnyProjectReferenceKind(project, cb, cbProjectRef, cbPotentialProjectRef) {
             return project.getCurrentProgram() ?
-                forEachResolvedProjectReference(project, cb) :
+                project.forEachResolvedProjectReference(cb) :
                 project.isInitialLoadPending() ?
                     forEachPotentialProjectReference(project, cbPotentialProjectRef) :
                     ts.forEach(project.getProjectReferences(), cbProjectRef);
@@ -144647,7 +153101,7 @@ var ts;
             return refProject && cb(refProject);
         }
         function forEachReferencedProject(project, cb) {
-            return forEachAnyProjectReferenceKind(project, function (resolvedRef) { return callbackRefProject(project, cb, resolvedRef && resolvedRef.sourceFile.path); }, function (projectRef) { return callbackRefProject(project, cb, project.toPath(projectRef.path)); }, function (potentialProjectRef) { return callbackRefProject(project, cb, potentialProjectRef); });
+            return forEachAnyProjectReferenceKind(project, function (resolvedRef) { return callbackRefProject(project, cb, resolvedRef.sourceFile.path); }, function (projectRef) { return callbackRefProject(project, cb, project.toPath(ts.resolveProjectReferencePath(projectRef))); }, function (potentialProjectRef) { return callbackRefProject(project, cb, potentialProjectRef); });
         }
         function getDetailWatchInfo(watchType, project) {
             return "Project: " + (project ? project.getProjectName() : "") + " WatchType: " + watchType;
@@ -144664,6 +153118,7 @@ var ts;
         server.projectContainsInfoDirectly = projectContainsInfoDirectly;
         /*@internal*/
         function updateProjectIfDirty(project) {
+            project.invalidateResolutionsOfFailedLookupLocations();
             return project.dirty && project.updateGraph();
         }
         server.updateProjectIfDirty = updateProjectIfDirty;
@@ -144679,20 +153134,20 @@ var ts;
                  * Container of all known scripts
                  */
                 /*@internal*/
-                this.filenameToScriptInfo = ts.createMap();
-                this.scriptInfoInNodeModulesWatchers = ts.createMap();
+                this.filenameToScriptInfo = new ts.Map();
+                this.scriptInfoInNodeModulesWatchers = new ts.Map();
                 /**
                  * Contains all the deleted script info's version information so that
                  * it does not reset when creating script info again
                  * (and could have potentially collided with version where contents mismatch)
                  */
-                this.filenameToScriptInfoVersion = ts.createMap();
+                this.filenameToScriptInfoVersion = new ts.Map();
                 // Set of all '.js' files ever opened.
-                this.allJsFilesForOpenFileTelemetry = ts.createMap();
+                this.allJsFilesForOpenFileTelemetry = new ts.Map();
                 /**
                  * maps external project file name to list of config files that were the part of this project
                  */
-                this.externalProjectToConfiguredProjectMap = ts.createMap();
+                this.externalProjectToConfiguredProjectMap = new ts.Map();
                 /**
                  * external projects (configuration and list of root files is not controlled by tsserver)
                  */
@@ -144704,21 +153159,24 @@ var ts;
                 /**
                  * projects specified by a tsconfig.json file
                  */
-                this.configuredProjects = ts.createMap();
+                this.configuredProjects = new ts.Map();
                 /**
                  * Open files: with value being project root path, and key being Path of the file that is open
                  */
-                this.openFiles = ts.createMap();
+                this.openFiles = new ts.Map();
+                /* @internal */
+                this.configFileForOpenFiles = new ts.Map();
                 /**
                  * Map of open files that are opened without complete path but have projectRoot as current directory
                  */
-                this.openFilesWithNonRootedDiskPath = ts.createMap();
-                this.compilerOptionsForInferredProjectsPerProjectRoot = ts.createMap();
-                this.watchOptionsForInferredProjectsPerProjectRoot = ts.createMap();
+                this.openFilesWithNonRootedDiskPath = new ts.Map();
+                this.compilerOptionsForInferredProjectsPerProjectRoot = new ts.Map();
+                this.watchOptionsForInferredProjectsPerProjectRoot = new ts.Map();
+                this.typeAcquisitionForInferredProjectsPerProjectRoot = new ts.Map();
                 /**
                  * Project size for configured or external projects
                  */
-                this.projectToSizeMap = ts.createMap();
+                this.projectToSizeMap = new ts.Map();
                 /**
                  * This is a map of config file paths existence that doesnt need query to disk
                  * - The entry can be present because there is inferred project that needs to watch addition of config file to directory
@@ -144726,14 +153184,14 @@ var ts;
                  * - Or it is present if we have configured project open with config file at that location
                  *   In this case the exists property is always true
                  */
-                this.configFileExistenceInfoCache = ts.createMap();
+                this.configFileExistenceInfoCache = new ts.Map();
                 this.safelist = defaultTypeSafeList;
-                this.legacySafelist = ts.createMap();
-                this.pendingProjectUpdates = ts.createMap();
+                this.legacySafelist = new ts.Map();
+                this.pendingProjectUpdates = new ts.Map();
                 /* @internal */
                 this.pendingEnsureProjectForOpenFiles = false;
                 /** Tracks projects that we have already sent telemetry for. */
-                this.seenProjects = ts.createMap();
+                this.seenProjects = new ts.Map();
                 this.host = opts.host;
                 this.logger = opts.logger;
                 this.cancellationToken = opts.cancellationToken;
@@ -144747,8 +153205,18 @@ var ts;
                 this.pluginProbeLocations = opts.pluginProbeLocations || server.emptyArray;
                 this.allowLocalPluginLoads = !!opts.allowLocalPluginLoads;
                 this.typesMapLocation = (opts.typesMapLocation === undefined) ? ts.combinePaths(ts.getDirectoryPath(this.getExecutingFilePath()), "typesMap.json") : opts.typesMapLocation;
-                this.syntaxOnly = opts.syntaxOnly;
-                ts.Debug.assert(!!this.host.createHash, "'ServerHost.createHash' is required for ProjectService");
+                if (opts.serverMode !== undefined) {
+                    this.serverMode = opts.serverMode;
+                    this.syntaxOnly = this.serverMode === ts.LanguageServiceMode.Syntactic;
+                }
+                else if (opts.syntaxOnly) {
+                    this.serverMode = ts.LanguageServiceMode.Syntactic;
+                    this.syntaxOnly = true;
+                }
+                else {
+                    this.serverMode = ts.LanguageServiceMode.Semantic;
+                    this.syntaxOnly = false;
+                }
                 if (this.host.realpath) {
                     this.realpathToScriptInfos = ts.createMultiMap();
                 }
@@ -144776,7 +153244,14 @@ var ts;
                 var watchLogLevel = this.logger.hasLevel(server.LogLevel.verbose) ? ts.WatchLogLevel.Verbose :
                     this.logger.loggingEnabled() ? ts.WatchLogLevel.TriggerOnly : ts.WatchLogLevel.None;
                 var log = watchLogLevel !== ts.WatchLogLevel.None ? (function (s) { return _this.logger.info(s); }) : ts.noop;
-                this.watchFactory = ts.getWatchFactory(watchLogLevel, log, getDetailWatchInfo);
+                this.packageJsonCache = server.createPackageJsonCache(this);
+                this.watchFactory = this.serverMode !== ts.LanguageServiceMode.Semantic ?
+                    {
+                        watchFile: ts.returnNoopFileWatcher,
+                        watchFilePath: ts.returnNoopFileWatcher,
+                        watchDirectory: ts.returnNoopFileWatcher,
+                    } :
+                    ts.getWatchFactory(watchLogLevel, log, getDetailWatchInfo);
             }
             ProjectService.prototype.toPath = function (fileName) {
                 return ts.toPath(fileName, this.currentDirectory, this.toCanonicalFileName);
@@ -144862,6 +153337,7 @@ var ts;
                 }
                 this.delayUpdateProjectGraphAndEnsureProjectStructureForOpenFiles(project);
             };
+            /*@internal*/
             ProjectService.prototype.delayEnsureProjectForOpenFiles = function () {
                 var _this = this;
                 this.pendingEnsureProjectForOpenFiles = true;
@@ -144944,9 +153420,9 @@ var ts;
                 this.eventHandler(event);
             };
             /* @internal */
-            ProjectService.prototype.sendUpdateGraphPerformanceEvent = function (durationMs) {
+            ProjectService.prototype.sendPerformanceEvent = function (kind, durationMs) {
                 if (this.performanceEventHandler) {
-                    this.performanceEventHandler({ kind: "UpdateGraph", durationMs: durationMs });
+                    this.performanceEventHandler({ kind: kind, durationMs: durationMs });
                 }
             };
             /* @internal */
@@ -144970,6 +153446,7 @@ var ts;
                 ts.Debug.assert(projectRootPath === undefined || this.useInferredProjectPerProjectRoot, "Setting compiler options per project root path is only supported when useInferredProjectPerProjectRoot is enabled");
                 var compilerOptions = convertCompilerOptions(projectCompilerOptions);
                 var watchOptions = convertWatchOptions(projectCompilerOptions);
+                var typeAcquisition = convertTypeAcquisition(projectCompilerOptions);
                 // always set 'allowNonTsExtensions' for inferred projects since user cannot configure it from the outside
                 // previously we did not expose a way for user to change these settings and this option was enabled by default
                 compilerOptions.allowNonTsExtensions = true;
@@ -144977,10 +153454,12 @@ var ts;
                 if (canonicalProjectRootPath) {
                     this.compilerOptionsForInferredProjectsPerProjectRoot.set(canonicalProjectRootPath, compilerOptions);
                     this.watchOptionsForInferredProjectsPerProjectRoot.set(canonicalProjectRootPath, watchOptions || false);
+                    this.typeAcquisitionForInferredProjectsPerProjectRoot.set(canonicalProjectRootPath, typeAcquisition);
                 }
                 else {
                     this.compilerOptionsForInferredProjects = compilerOptions;
                     this.watchOptionsForInferredProjects = watchOptions;
+                    this.typeAcquisitionForInferredProjects = typeAcquisition;
                 }
                 for (var _i = 0, _a = this.inferredProjects; _i < _a.length; _i++) {
                     var project = _a[_i];
@@ -144997,6 +153476,7 @@ var ts;
                         !project.projectRootPath || !this.compilerOptionsForInferredProjectsPerProjectRoot.has(project.projectRootPath)) {
                         project.setCompilerOptions(compilerOptions);
                         project.setWatchOptions(watchOptions);
+                        project.setTypeAcquisition(typeAcquisition);
                         project.compileOnSaveEnabled = compilerOptions.compileOnSave;
                         project.markAsDirty();
                         this.delayUpdateProjectGraph(project);
@@ -145159,9 +153639,30 @@ var ts;
             /*@internal*/
             ProjectService.prototype.watchWildcardDirectory = function (directory, flags, project) {
                 var _this = this;
+                var watchOptions = this.getWatchOptions(project);
                 return this.watchFactory.watchDirectory(this.host, directory, function (fileOrDirectory) {
                     var fileOrDirectoryPath = _this.toPath(fileOrDirectory);
                     var fsResult = project.getCachedDirectoryStructureHost().addOrDeleteFileOrDirectory(fileOrDirectory, fileOrDirectoryPath);
+                    var configFileName = project.getConfigFilePath();
+                    if (ts.getBaseFileName(fileOrDirectoryPath) === "package.json" && !ts.isInsideNodeModules(fileOrDirectoryPath) &&
+                        (fsResult && fsResult.fileExists || !fsResult && _this.host.fileExists(fileOrDirectoryPath))) {
+                        _this.logger.info("Project: " + configFileName + " Detected new package.json: " + fileOrDirectory);
+                        _this.onAddPackageJson(fileOrDirectoryPath);
+                    }
+                    if (ts.isIgnoredFileFromWildCardWatching({
+                        watchedDirPath: directory,
+                        fileOrDirectory: fileOrDirectory,
+                        fileOrDirectoryPath: fileOrDirectoryPath,
+                        configFileName: configFileName,
+                        configFileSpecs: project.configFileSpecs,
+                        extraFileExtensions: _this.hostConfiguration.extraFileExtensions,
+                        currentDirectory: _this.currentDirectory,
+                        options: project.getCompilationSettings(),
+                        program: project.getCurrentProgram(),
+                        useCaseSensitiveFileNames: _this.host.useCaseSensitiveFileNames,
+                        writeLog: function (s) { return _this.logger.info(s); }
+                    }))
+                        return;
                     // don't trigger callback on open, existing files
                     if (project.fileIsOpen(fileOrDirectoryPath)) {
                         if (project.pendingReload !== ts.ConfigFileProgramReloadLevel.Full) {
@@ -145176,27 +153677,12 @@ var ts;
                         }
                         return;
                     }
-                    fileOrDirectoryPath = ts.removeIgnoredPath(fileOrDirectoryPath);
-                    if (!fileOrDirectoryPath)
-                        return;
-                    var configFilename = project.getConfigFilePath();
-                    if (ts.getBaseFileName(fileOrDirectoryPath) === "package.json" && !ts.isInsideNodeModules(fileOrDirectoryPath) &&
-                        (fsResult && fsResult.fileExists || !fsResult && _this.host.fileExists(fileOrDirectoryPath))) {
-                        _this.logger.info("Project: " + configFilename + " Detected new package.json: " + fileOrDirectory);
-                        project.onAddPackageJson(fileOrDirectoryPath);
-                    }
-                    // If the the added or created file or directory is not supported file name, ignore the file
-                    // But when watched directory is added/removed, we need to reload the file list
-                    if (fileOrDirectoryPath !== directory && ts.hasExtension(fileOrDirectoryPath) && !ts.isSupportedSourceFileName(fileOrDirectory, project.getCompilationSettings(), _this.hostConfiguration.extraFileExtensions)) {
-                        _this.logger.info("Project: " + configFilename + " Detected file add/remove of non supported extension: " + fileOrDirectory);
-                        return;
-                    }
                     // Reload is pending, do the reload
                     if (project.pendingReload !== ts.ConfigFileProgramReloadLevel.Full) {
                         project.pendingReload = ts.ConfigFileProgramReloadLevel.Partial;
                         _this.delayUpdateProjectGraphAndEnsureProjectStructureForOpenFiles(project);
                     }
-                }, flags, this.getWatchOptions(project), ts.WatchType.WildcardDirectory, project);
+                }, flags, watchOptions, ts.WatchType.WildcardDirectory, project);
             };
             /** Gets the config file existence info for the configured project */
             /*@internal*/
@@ -145387,6 +153873,7 @@ var ts;
                     }
                 }
                 this.openFiles.delete(info.path);
+                this.configFileForOpenFiles.delete(info.path);
                 if (!skipAssignOrphanScriptInfosToInferredProject && ensureProjectsForOpenFiles) {
                     this.assignOrphanScriptInfosToInferredProject();
                 }
@@ -145431,7 +153918,7 @@ var ts;
                 // Or the whole chain of config files for the roots of the inferred projects
                 // Cache the host value of file exists and add the info to map of open files impacted by this config file
                 var exists = this.host.fileExists(configFileName);
-                var openFilesImpactedByConfigFile = ts.createMap();
+                var openFilesImpactedByConfigFile = new ts.Map();
                 if (isOpenScriptInfo(info)) {
                     openFilesImpactedByConfigFile.set(info.path, false);
                 }
@@ -145458,7 +153945,7 @@ var ts;
                     // Since that route doesnt check if the config file is present or not
                     this.configFileExistenceInfoCache.set(project.canonicalConfigFilePath, {
                         exists: true,
-                        openFilesImpactedByConfigFile: ts.createMap()
+                        openFilesImpactedByConfigFile: new ts.Map()
                     });
                 }
             };
@@ -145574,7 +154061,7 @@ var ts;
                         // Create the cache
                         configFileExistenceInfo = {
                             exists: _this.host.fileExists(configFileName),
-                            openFilesImpactedByConfigFile: ts.createMap()
+                            openFilesImpactedByConfigFile: new ts.Map()
                         };
                         _this.configFileExistenceInfoCache.set(canonicalConfigFilePath, configFileExistenceInfo);
                     }
@@ -145616,7 +154103,7 @@ var ts;
              */
             ProjectService.prototype.forEachConfigFileLocation = function (info, action) {
                 var _this = this;
-                if (this.syntaxOnly) {
+                if (this.serverMode !== ts.LanguageServiceMode.Semantic) {
                     return undefined;
                 }
                 ts.Debug.assert(!isOpenScriptInfo(info) || this.openFiles.has(info.path));
@@ -145663,11 +154150,8 @@ var ts;
                 var configFileName = this.getConfigFileNameForFile(info);
                 var project = configFileName &&
                     this.findConfiguredProjectByProjectName(configFileName);
-                return (project === null || project === void 0 ? void 0 : project.isSolution()) ?
-                    project.getDefaultChildProjectFromSolution(info) :
-                    project && projectContainsInfoDirectly(project, info) ?
-                        project :
-                        undefined;
+                return project && projectContainsInfoDirectly(project, info) ?
+                    project : project === null || project === void 0 ? void 0 : project.getDefaultChildProjectFromProjectWithReferences(info);
             };
             /**
              * This function tries to search for a tsconfig.json for the given file.
@@ -145681,8 +154165,12 @@ var ts;
              */
             ProjectService.prototype.getConfigFileNameForFile = function (info) {
                 var _this = this;
-                if (isOpenScriptInfo(info))
+                if (isOpenScriptInfo(info)) {
                     ts.Debug.assert(info.isScriptOpen());
+                    var result = this.configFileForOpenFiles.get(info.path);
+                    if (result !== undefined)
+                        return result || undefined;
+                }
                 this.logger.info("Search path: " + ts.getDirectoryPath(info.fileName));
                 var configFileName = this.forEachConfigFileLocation(info, function (configFileName, canonicalConfigFilePath) {
                     return _this.configFileExists(configFileName, canonicalConfigFilePath, info);
@@ -145693,6 +154181,9 @@ var ts;
                 else {
                     this.logger.info("For info: " + info.fileName + " :: No config files found.");
                 }
+                if (isOpenScriptInfo(info)) {
+                    this.configFileForOpenFiles.set(info.path, configFileName || false);
+                }
                 return configFileName;
             };
             ProjectService.prototype.printProjects = function () {
@@ -145861,9 +154352,6 @@ var ts;
                 var configFilename = ts.normalizePath(project.getConfigFilePath());
                 var configFileContent = ts.tryReadFile(configFilename, function (fileName) { return _this.host.readFile(fileName); });
                 var result = ts.parseJsonText(configFilename, ts.isString(configFileContent) ? configFileContent : "");
-                if (!result.endOfFileToken) {
-                    result.endOfFileToken = { kind: 1 /* EndOfFileToken */ };
-                }
                 var configFileErrors = result.parseDiagnostics;
                 if (!ts.isString(configFileContent))
                     configFileErrors.push(configFileContent);
@@ -145891,7 +154379,7 @@ var ts;
                     };
                 }
                 project.configFileSpecs = parsedCommandLine.configFileSpecs;
-                project.canConfigFileJsonReportNoInputFiles = ts.canJsonReportNoInutFiles(parsedCommandLine.raw);
+                project.canConfigFileJsonReportNoInputFiles = ts.canJsonReportNoInputFiles(parsedCommandLine.raw);
                 project.setProjectErrors(configFileErrors);
                 project.updateReferences(parsedCommandLine.projectReferences);
                 var lastFileExceededProgramSize = this.getFilenameForExceededTotalSizeLimitForNonTsFiles(project.canonicalConfigFilePath, compilerOptions, parsedCommandLine.fileNames, fileNamePropertyReader);
@@ -145903,7 +154391,7 @@ var ts;
                     project.setCompilerOptions(compilerOptions);
                     project.setWatchOptions(parsedCommandLine.watchOptions);
                     project.enableLanguageService();
-                    project.watchWildcards(ts.createMapFromTemplate(parsedCommandLine.wildcardDirectories)); // TODO: GH#18217
+                    project.watchWildcards(new ts.Map(ts.getEntries(parsedCommandLine.wildcardDirectories))); // TODO: GH#18217
                 }
                 project.enablePluginsWithOptions(compilerOptions, this.currentPluginConfigOverrides);
                 var filesToAdd = parsedCommandLine.fileNames.concat(project.getExternalFiles());
@@ -145911,7 +154399,7 @@ var ts;
             };
             ProjectService.prototype.updateNonInferredProjectFiles = function (project, files, propertyReader) {
                 var projectRootFilesMap = project.getRootFilesMap();
-                var newRootScriptInfoMap = ts.createMap();
+                var newRootScriptInfoMap = new ts.Map();
                 for (var _i = 0, files_1 = files; _i < files_1.length; _i++) {
                     var f = files_1[_i];
                     var newRootFile = propertyReader.getFileName(f);
@@ -145995,17 +154483,21 @@ var ts;
                 this.updateNonInferredProjectFiles(project, fileNamesResult.fileNames.concat(project.getExternalFiles()), fileNamePropertyReader);
                 return project.updateGraph();
             };
+            /*@internal*/
+            ProjectService.prototype.setFileNamesOfAutoImportProviderProject = function (project, fileNames) {
+                this.updateNonInferredProjectFiles(project, fileNames, fileNamePropertyReader);
+            };
             /**
              * Read the config file of the project again by clearing the cache and update the project graph
              */
             /* @internal */
-            ProjectService.prototype.reloadConfiguredProject = function (project, reason) {
+            ProjectService.prototype.reloadConfiguredProject = function (project, reason, isInitialLoad) {
                 // At this point, there is no reason to not have configFile in the host
                 var host = project.getCachedDirectoryStructureHost();
                 // Clear the cache since we are reloading the project from disk
                 host.clearCache();
                 var configFileName = project.getConfigFilePath();
-                this.logger.info("Reloading configured project " + configFileName);
+                this.logger.info((isInitialLoad ? "Loading" : "Reloading") + " configured project " + configFileName);
                 // Load project from the disk
                 this.loadConfiguredProject(project, reason);
                 project.updateGraph();
@@ -146093,13 +154585,18 @@ var ts;
             ProjectService.prototype.createInferredProject = function (currentDirectory, isSingleInferredProject, projectRootPath) {
                 var compilerOptions = projectRootPath && this.compilerOptionsForInferredProjectsPerProjectRoot.get(projectRootPath) || this.compilerOptionsForInferredProjects; // TODO: GH#18217
                 var watchOptions;
+                var typeAcquisition;
                 if (projectRootPath) {
                     watchOptions = this.watchOptionsForInferredProjectsPerProjectRoot.get(projectRootPath);
+                    typeAcquisition = this.typeAcquisitionForInferredProjectsPerProjectRoot.get(projectRootPath);
                 }
                 if (watchOptions === undefined) {
                     watchOptions = this.watchOptionsForInferredProjects;
                 }
-                var project = new server.InferredProject(this, this.documentRegistry, compilerOptions, watchOptions || undefined, projectRootPath, currentDirectory, this.currentPluginConfigOverrides);
+                if (typeAcquisition === undefined) {
+                    typeAcquisition = this.typeAcquisitionForInferredProjects;
+                }
+                var project = new server.InferredProject(this, this.documentRegistry, compilerOptions, watchOptions || undefined, projectRootPath, currentDirectory, this.currentPluginConfigOverrides, typeAcquisition);
                 if (isSingleInferredProject) {
                     this.inferredProjects.unshift(project);
                 }
@@ -146403,7 +154900,7 @@ var ts;
                 if (sourceFileName) {
                     // Attach as source
                     var sourceInfo = this.getOrCreateScriptInfoNotOpenedByClient(sourceFileName, project.currentDirectory, project.directoryStructureHost);
-                    (sourceInfos || (sourceInfos = ts.createMap())).set(sourceInfo.path, true);
+                    (sourceInfos || (sourceInfos = new ts.Set())).add(sourceInfo.path);
                 }
                 return sourceInfos;
             };
@@ -146437,7 +154934,7 @@ var ts;
                 if (declarationInfo && ts.isString(declarationInfo.sourceMapFilePath) && info !== declarationInfo) {
                     var sourceMapInfo = this.getScriptInfoForPath(declarationInfo.sourceMapFilePath);
                     if (sourceMapInfo) {
-                        (sourceMapInfo.sourceInfos || (sourceMapInfo.sourceInfos = ts.createMap())).set(info.path, true);
+                        (sourceMapInfo.sourceInfos || (sourceMapInfo.sourceInfos = new ts.Set())).add(info.path);
                     }
                 }
                 // Key doesnt matter since its only for text and lines
@@ -146482,7 +154979,7 @@ var ts;
                         this.logger.info("Format host information updated");
                     }
                     if (args.preferences) {
-                        var lazyConfiguredProjectsFromExternalProject = this.hostConfiguration.preferences.lazyConfiguredProjectsFromExternalProject;
+                        var _a = this.hostConfiguration.preferences, lazyConfiguredProjectsFromExternalProject = _a.lazyConfiguredProjectsFromExternalProject, includePackageJsonAutoImports = _a.includePackageJsonAutoImports;
                         this.hostConfiguration.preferences = __assign(__assign({}, this.hostConfiguration.preferences), args.preferences);
                         if (lazyConfiguredProjectsFromExternalProject && !this.hostConfiguration.preferences.lazyConfiguredProjectsFromExternalProject) {
                             // Load configured projects for external projects that are pending reload
@@ -146494,6 +154991,9 @@ var ts;
                                 }
                             });
                         }
+                        if (includePackageJsonAutoImports !== args.preferences.includePackageJsonAutoImports) {
+                            this.invalidateProjectAutoImports(/*packageJsonPath*/ undefined);
+                        }
                     }
                     if (args.extraFileExtensions) {
                         this.hostConfiguration.extraFileExtensions = args.extraFileExtensions;
@@ -146549,9 +155049,17 @@ var ts;
              */
             ProjectService.prototype.reloadConfiguredProjectForFiles = function (openFiles, delayReload, shouldReloadProjectFor, reason) {
                 var _this = this;
-                var updatedProjects = ts.createMap();
+                var updatedProjects = new ts.Map();
+                var reloadChildProject = function (child) {
+                    if (!updatedProjects.has(child.canonicalConfigFilePath)) {
+                        updatedProjects.set(child.canonicalConfigFilePath, true);
+                        _this.reloadConfiguredProject(child, reason, /*isInitialLoad*/ false);
+                    }
+                };
                 // try to reload config file for all open files
                 openFiles.forEach(function (openFileValue, path) {
+                    // Invalidate default config file name for open file
+                    _this.configFileForOpenFiles.delete(path);
                     // Filter out the files that need to be ignored
                     if (!shouldReloadProjectFor(openFileValue)) {
                         return;
@@ -146574,16 +155082,18 @@ var ts;
                             }
                             else {
                                 // reload from the disk
-                                _this.reloadConfiguredProject(project, reason);
-                                // If this is solution, reload the project till the reloaded project contains the script info directly
-                                if (!project.containsScriptInfo(info) && project.isSolution()) {
-                                    forEachResolvedProjectReferenceProject(project, function (child) {
-                                        if (!updatedProjects.has(child.canonicalConfigFilePath)) {
-                                            updatedProjects.set(child.canonicalConfigFilePath, true);
-                                            _this.reloadConfiguredProject(child, reason);
-                                        }
+                                _this.reloadConfiguredProject(project, reason, /*isInitialLoad*/ false);
+                                // If this project does not contain this file directly, reload the project till the reloaded project contains the script info directly
+                                if (!projectContainsInfoDirectly(project, info)) {
+                                    var referencedProject = forEachResolvedProjectReferenceProject(project, info.path, function (child) {
+                                        reloadChildProject(child);
                                         return projectContainsInfoDirectly(child, info);
                                     }, ProjectReferenceProjectLoadKind.FindCreate);
+                                    if (referencedProject) {
+                                        // Reload the project's tree that is already present
+                                        forEachResolvedProjectReferenceProject(project, 
+                                        /*fileName*/ undefined, reloadChildProject, ProjectReferenceProjectLoadKind.Find);
+                                    }
                                 }
                             }
                         }
@@ -146669,12 +155179,15 @@ var ts;
                 var configuredProject = this.findConfiguredProjectByProjectName(configFileName) ||
                     this.createAndLoadConfiguredProject(configFileName, "Creating project for original file: " + originalFileInfo.fileName + (location !== originalLocation ? " for location: " + location.fileName : ""));
                 updateProjectIfDirty(configuredProject);
-                if (configuredProject.isSolution()) {
+                var projectContainsOriginalInfo = function (project) {
+                    var info = _this.getScriptInfo(fileName);
+                    return info && projectContainsInfoDirectly(project, info);
+                };
+                if (configuredProject.isSolution() || !projectContainsOriginalInfo(configuredProject)) {
                     // Find the project that is referenced from this solution that contains the script info directly
-                    configuredProject = forEachResolvedProjectReferenceProject(configuredProject, function (child) {
+                    configuredProject = forEachResolvedProjectReferenceProject(configuredProject, fileName, function (child) {
                         updateProjectIfDirty(child);
-                        var info = _this.getScriptInfo(fileName);
-                        return info && projectContainsInfoDirectly(child, info) ? child : undefined;
+                        return projectContainsOriginalInfo(child) ? child : undefined;
                     }, ProjectReferenceProjectLoadKind.FindCreateLoad, "Creating project referenced in solution " + configuredProject.projectName + " to find possible configured project for original file: " + originalFileInfo.fileName + (location !== originalLocation ? " for location: " + location.fileName : ""));
                     if (!configuredProject)
                         return undefined;
@@ -146695,9 +155208,9 @@ var ts;
                 return originalLocation;
                 function addOriginalConfiguredProject(originalProject) {
                     if (!project.originalConfiguredProjects) {
-                        project.originalConfiguredProjects = ts.createMap();
+                        project.originalConfiguredProjects = new ts.Set();
                     }
-                    project.originalConfiguredProjects.set(originalProject.canonicalConfigFilePath, true);
+                    project.originalConfiguredProjects.add(originalProject.canonicalConfigFilePath);
                 }
             };
             /** @internal */
@@ -146721,34 +155234,27 @@ var ts;
                 var configFileName;
                 var configFileErrors;
                 var project = this.findExternalProjectContainingOpenScriptInfo(info);
-                var defaultConfigProject;
                 var retainProjects;
-                if (!project && !this.syntaxOnly) { // Checking syntaxOnly is an optimization
+                var projectForConfigFileDiag;
+                var defaultConfigProjectIsCreated = false;
+                if (!project && this.serverMode === ts.LanguageServiceMode.Semantic) { // Checking semantic mode is an optimization
                     configFileName = this.getConfigFileNameForFile(info);
                     if (configFileName) {
                         project = this.findConfiguredProjectByProjectName(configFileName);
                         if (!project) {
                             project = this.createLoadAndUpdateConfiguredProject(configFileName, "Creating possible configured project for " + info.fileName + " to open");
-                            // Send the event only if the project got created as part of this open request and info is part of the project
-                            if (!project.containsScriptInfo(info)) {
-                                // Since the file isnt part of configured project, do not send config file info
-                                configFileName = undefined;
-                            }
-                            else {
-                                configFileErrors = project.getAllProjectErrors();
-                                this.sendConfigFileDiagEvent(project, info.fileName);
-                            }
+                            defaultConfigProjectIsCreated = true;
                         }
                         else {
                             // Ensure project is ready to check if it contains opened script info
                             updateProjectIfDirty(project);
                         }
-                        defaultConfigProject = project;
-                        retainProjects = defaultConfigProject;
+                        projectForConfigFileDiag = project.containsScriptInfo(info) ? project : undefined;
+                        retainProjects = project;
                         // If this configured project doesnt contain script info but
                         // it is solution with project references, try those project references
-                        if (!project.containsScriptInfo(info) && project.isSolution()) {
-                            forEachResolvedProjectReferenceProject(project, function (child, childConfigFileName) {
+                        if (!projectContainsInfoDirectly(project, info)) {
+                            forEachResolvedProjectReferenceProject(project, info.path, function (child) {
                                 updateProjectIfDirty(child);
                                 // Retain these projects
                                 if (!ts.isArray(retainProjects)) {
@@ -146759,17 +155265,29 @@ var ts;
                                 }
                                 // If script info belongs to this child project, use this as default config project
                                 if (projectContainsInfoDirectly(child, info)) {
-                                    configFileName = childConfigFileName;
-                                    configFileErrors = child.getAllProjectErrors();
-                                    _this.sendConfigFileDiagEvent(child, info.fileName);
+                                    projectForConfigFileDiag = child;
                                     return child;
                                 }
+                                // If this project uses the script info (even through project reference), if default project is not found, use this for configFileDiag
+                                if (!projectForConfigFileDiag && child.containsScriptInfo(info)) {
+                                    projectForConfigFileDiag = child;
+                                }
                             }, ProjectReferenceProjectLoadKind.FindCreateLoad, "Creating project referenced in solution " + project.projectName + " to find possible configured project for " + info.fileName + " to open");
                         }
+                        // Send the event only if the project got created as part of this open request and info is part of the project
+                        if (projectForConfigFileDiag) {
+                            configFileName = projectForConfigFileDiag.getConfigFilePath();
+                            if (projectForConfigFileDiag !== project || defaultConfigProjectIsCreated) {
+                                configFileErrors = projectForConfigFileDiag.getAllProjectErrors();
+                                this.sendConfigFileDiagEvent(projectForConfigFileDiag, info.fileName);
+                            }
+                        }
                         else {
-                            // Create ancestor configured project
-                            this.createAncestorProjects(info, defaultConfigProject || project);
+                            // Since the file isnt part of configured project, do not send config file info
+                            configFileName = undefined;
                         }
+                        // Create ancestor configured project
+                        this.createAncestorProjects(info, project);
                     }
                 }
                 // Project we have at this point is going to be updated since its either found through
@@ -146826,27 +155344,44 @@ var ts;
             };
             /*@internal*/
             ProjectService.prototype.loadAncestorProjectTree = function (forProjects) {
-                forProjects = forProjects || ts.mapDefinedMap(this.configuredProjects, function (project) { return !project.isInitialLoadPending() || undefined; });
-                var seenProjects = ts.createMap();
+                forProjects = forProjects || ts.mapDefinedEntries(this.configuredProjects, function (key, project) { return !project.isInitialLoadPending() ? [key, true] : undefined; });
+                var seenProjects = new ts.Set();
                 // Work on array copy as we could add more projects as part of callback
                 for (var _i = 0, _a = ts.arrayFrom(this.configuredProjects.values()); _i < _a.length; _i++) {
                     var project = _a[_i];
                     // If this project has potential project reference for any of the project we are loading ancestor tree for
-                    // we need to load this project tree
-                    if (forEachPotentialProjectReference(project, function (potentialRefPath) { return forProjects.has(potentialRefPath); }) || (project.isSolution() && forEachResolvedProjectReference(project, function (_ref, resolvedPath) { return forProjects.has(resolvedPath); }))) {
-                        // Load children
-                        this.ensureProjectChildren(project, seenProjects);
+                    // load this project first
+                    if (forEachPotentialProjectReference(project, function (potentialRefPath) { return forProjects.has(potentialRefPath); })) {
+                        updateProjectIfDirty(project);
                     }
+                    this.ensureProjectChildren(project, forProjects, seenProjects);
                 }
             };
-            ProjectService.prototype.ensureProjectChildren = function (project, seenProjects) {
-                var _this = this;
-                if (!ts.addToSeen(seenProjects, project.canonicalConfigFilePath))
+            ProjectService.prototype.ensureProjectChildren = function (project, forProjects, seenProjects) {
+                var _a;
+                if (!ts.tryAddToSet(seenProjects, project.canonicalConfigFilePath))
                     return;
-                // Update the project
-                updateProjectIfDirty(project);
-                // Create tree because project is uptodate we only care of resolved references
-                forEachResolvedProjectReferenceProject(project, function (child) { return _this.ensureProjectChildren(child, seenProjects); }, ProjectReferenceProjectLoadKind.FindCreateLoad, "Creating project for reference of project: " + project.projectName);
+                // If this project disables child load ignore it
+                if (project.getCompilerOptions().disableReferencedProjectLoad)
+                    return;
+                var children = (_a = project.getCurrentProgram()) === null || _a === void 0 ? void 0 : _a.getResolvedProjectReferences();
+                if (!children)
+                    return;
+                for (var _i = 0, children_1 = children; _i < children_1.length; _i++) {
+                    var child = children_1[_i];
+                    if (!child)
+                        continue;
+                    var referencedProject = ts.forEachResolvedProjectReference(child.references, function (ref) { return forProjects.has(ref.sourceFile.path) ? ref : undefined; });
+                    if (!referencedProject)
+                        continue;
+                    // Load this project,
+                    var configFileName = server.toNormalizedPath(child.sourceFile.fileName);
+                    var childProject = project.projectService.findConfiguredProjectByProjectName(configFileName) ||
+                        project.projectService.createAndLoadConfiguredProject(configFileName, "Creating project referenced by : " + project.projectName + " as it references project " + referencedProject.sourceFile.fileName);
+                    updateProjectIfDirty(childProject);
+                    // Ensure children for this project
+                    this.ensureProjectChildren(childProject, forProjects, seenProjects);
+                }
             };
             ProjectService.prototype.cleanupAfterOpeningFile = function (toRetainConfigProjects) {
                 // This was postponed from closeOpenFile to after opening next file,
@@ -146876,7 +155411,7 @@ var ts;
             };
             ProjectService.prototype.removeOrphanConfiguredProjects = function (toRetainConfiguredProjects) {
                 var _this = this;
-                var toRemoveConfiguredProjects = ts.cloneMap(this.configuredProjects);
+                var toRemoveConfiguredProjects = new ts.Map(this.configuredProjects);
                 var markOriginalProjectsAsUsed = function (project) {
                     if (!project.isOrphan() && project.originalConfiguredProjects) {
                         project.originalConfiguredProjects.forEach(function (_value, configuredProjectPath) {
@@ -146922,10 +155457,10 @@ var ts;
             };
             ProjectService.prototype.removeOrphanScriptInfos = function () {
                 var _this = this;
-                var toRemoveScriptInfos = ts.cloneMap(this.filenameToScriptInfo);
+                var toRemoveScriptInfos = new ts.Map(this.filenameToScriptInfo);
                 this.filenameToScriptInfo.forEach(function (info) {
                     // If script info is open or orphan, retain it and its dependencies
-                    if (!info.isScriptOpen() && info.isOrphan()) {
+                    if (!info.isScriptOpen() && info.isOrphan() && !info.isContainedByAutoImportProvider()) {
                         // Otherwise if there is any source info that is alive, this alive too
                         if (!info.sourceMapFilePath)
                             return;
@@ -146972,7 +155507,7 @@ var ts;
                 });
             };
             ProjectService.prototype.telemetryOnOpenFile = function (scriptInfo) {
-                if (this.syntaxOnly || !this.eventHandler || !scriptInfo.isJavaScript() || !ts.addToSeen(this.allJsFilesForOpenFileTelemetry, scriptInfo.path)) {
+                if (this.serverMode !== ts.LanguageServiceMode.Semantic || !this.eventHandler || !scriptInfo.isJavaScript() || !ts.addToSeen(this.allJsFilesForOpenFileTelemetry, scriptInfo.path)) {
                     return;
                 }
                 var project = scriptInfo.getDefaultProject();
@@ -147131,8 +155666,7 @@ var ts;
                 var rootFiles = proj.rootFiles;
                 var typeAcquisition = proj.typeAcquisition;
                 ts.Debug.assert(!!typeAcquisition, "proj.typeAcquisition should be set by now");
-                // If type acquisition has been explicitly disabled, do not exclude anything from the project
-                if (typeAcquisition.enable === false) {
+                if (typeAcquisition.enable === false || typeAcquisition.disableFilenameBasedTypeAcquisition) {
                     return [];
                 }
                 var typeAcqInclude = typeAcquisition.include || (typeAcquisition.include = []);
@@ -147267,7 +155801,7 @@ var ts;
                     var file = _a[_i];
                     var normalized = server.toNormalizedPath(file.fileName);
                     if (server.getBaseConfigFileName(normalized)) {
-                        if (!this.syntaxOnly && this.host.fileExists(normalized)) {
+                        if (this.serverMode === ts.LanguageServiceMode.Semantic && this.host.fileExists(normalized)) {
                             (tsConfigFiles || (tsConfigFiles = [])).push(normalized);
                         }
                     }
@@ -147378,9 +155912,88 @@ var ts;
                 this.forEachEnabledProject(function (project) { return project.onPluginConfigurationChanged(args.pluginName, args.configuration); });
                 // Also save the current configuration to pass on to any projects that are yet to be loaded.
                 // If a plugin is configured twice, only the latest configuration will be remembered.
-                this.currentPluginConfigOverrides = this.currentPluginConfigOverrides || ts.createMap();
+                this.currentPluginConfigOverrides = this.currentPluginConfigOverrides || new ts.Map();
                 this.currentPluginConfigOverrides.set(args.pluginName, args.configuration);
             };
+            /*@internal*/
+            ProjectService.prototype.getPackageJsonsVisibleToFile = function (fileName, rootDir) {
+                var _this = this;
+                var packageJsonCache = this.packageJsonCache;
+                var rootPath = rootDir && this.toPath(rootDir);
+                var filePath = this.toPath(fileName);
+                var result = [];
+                var processDirectory = function (directory) {
+                    switch (packageJsonCache.directoryHasPackageJson(directory)) {
+                        // Sync and check same directory again
+                        case 3 /* Maybe */:
+                            packageJsonCache.searchDirectoryAndAncestors(directory);
+                            return processDirectory(directory);
+                        // Check package.json
+                        case -1 /* True */:
+                            var packageJsonFileName = ts.combinePaths(directory, "package.json");
+                            _this.watchPackageJsonFile(packageJsonFileName);
+                            var info = packageJsonCache.getInDirectory(directory);
+                            if (info)
+                                result.push(info);
+                    }
+                    if (rootPath && rootPath === _this.toPath(directory)) {
+                        return true;
+                    }
+                };
+                ts.forEachAncestorDirectory(ts.getDirectoryPath(filePath), processDirectory);
+                return result;
+            };
+            /*@internal*/
+            ProjectService.prototype.watchPackageJsonFile = function (path) {
+                var _this = this;
+                var watchers = this.packageJsonFilesMap || (this.packageJsonFilesMap = new ts.Map());
+                if (!watchers.has(path)) {
+                    this.invalidateProjectAutoImports(path);
+                    watchers.set(path, this.watchFactory.watchFile(this.host, path, function (fileName, eventKind) {
+                        var path = _this.toPath(fileName);
+                        switch (eventKind) {
+                            case ts.FileWatcherEventKind.Created:
+                                return ts.Debug.fail();
+                            case ts.FileWatcherEventKind.Changed:
+                                _this.packageJsonCache.addOrUpdate(path);
+                                _this.invalidateProjectAutoImports(path);
+                                break;
+                            case ts.FileWatcherEventKind.Deleted:
+                                _this.packageJsonCache.delete(path);
+                                _this.invalidateProjectAutoImports(path);
+                                watchers.get(path).close();
+                                watchers.delete(path);
+                        }
+                    }, ts.PollingInterval.Low, this.hostConfiguration.watchOptions, ts.WatchType.PackageJsonFile));
+                }
+            };
+            /*@internal*/
+            ProjectService.prototype.onAddPackageJson = function (path) {
+                this.packageJsonCache.addOrUpdate(path);
+                this.watchPackageJsonFile(path);
+            };
+            /*@internal*/
+            ProjectService.prototype.includePackageJsonAutoImports = function () {
+                switch (this.hostConfiguration.preferences.includePackageJsonAutoImports) {
+                    case "on": return 1 /* On */;
+                    case "off": return 0 /* Off */;
+                    default: return 2 /* Auto */;
+                }
+            };
+            /*@internal*/
+            ProjectService.prototype.invalidateProjectAutoImports = function (packageJsonPath) {
+                if (this.includePackageJsonAutoImports()) {
+                    this.configuredProjects.forEach(invalidate);
+                    this.inferredProjects.forEach(invalidate);
+                    this.externalProjects.forEach(invalidate);
+                }
+                function invalidate(project) {
+                    var _a;
+                    if (!packageJsonPath || ((_a = project.packageJsonsForAutoImport) === null || _a === void 0 ? void 0 : _a.has(packageJsonPath))) {
+                        project.markAutoImportProviderAsDirty();
+                    }
+                }
+            };
             /** Makes a filename safe to insert in a RegExp */
             ProjectService.filenameEscapeRegexp = /[-\/\\^$*+?.()|[\]{}]/g;
             return ProjectService;
@@ -147401,11 +156014,13 @@ var ts;
 (function (ts) {
     var server;
     (function (server) {
-        function createPackageJsonCache(project) {
-            var packageJsons = ts.createMap();
-            var directoriesWithoutPackageJson = ts.createMap();
+        function createPackageJsonCache(host) {
+            var packageJsons = new ts.Map();
+            var directoriesWithoutPackageJson = new ts.Map();
             return {
                 addOrUpdate: addOrUpdate,
+                forEach: packageJsons.forEach.bind(packageJsons),
+                get: packageJsons.get.bind(packageJsons),
                 delete: function (fileName) {
                     packageJsons.delete(fileName);
                     directoriesWithoutPackageJson.set(ts.getDirectoryPath(fileName), true);
@@ -147416,11 +156031,11 @@ var ts;
                 directoryHasPackageJson: directoryHasPackageJson,
                 searchDirectoryAndAncestors: function (directory) {
                     ts.forEachAncestorDirectory(directory, function (ancestor) {
-                        if (directoryHasPackageJson(ancestor) !== 1 /* Maybe */) {
+                        if (directoryHasPackageJson(ancestor) !== 3 /* Maybe */) {
                             return true;
                         }
-                        var packageJsonFileName = project.toPath(ts.combinePaths(ancestor, "package.json"));
-                        if (ts.tryFileExists(project, packageJsonFileName)) {
+                        var packageJsonFileName = host.toPath(ts.combinePaths(ancestor, "package.json"));
+                        if (ts.tryFileExists(host, packageJsonFileName)) {
                             addOrUpdate(packageJsonFileName);
                         }
                         else {
@@ -147430,16 +156045,14 @@ var ts;
                 },
             };
             function addOrUpdate(fileName) {
-                var packageJsonInfo = ts.createPackageJsonInfo(fileName, project);
-                if (packageJsonInfo !== undefined) {
-                    packageJsons.set(fileName, packageJsonInfo);
-                    directoriesWithoutPackageJson.delete(ts.getDirectoryPath(fileName));
-                }
+                var packageJsonInfo = ts.Debug.checkDefined(ts.createPackageJsonInfo(fileName, host.host));
+                packageJsons.set(fileName, packageJsonInfo);
+                directoriesWithoutPackageJson.delete(ts.getDirectoryPath(fileName));
             }
             function directoryHasPackageJson(directory) {
                 return packageJsons.has(ts.combinePaths(directory, "package.json")) ? -1 /* True */ :
                     directoriesWithoutPackageJson.has(directory) ? 0 /* False */ :
-                        1 /* Maybe */;
+                        3 /* Maybe */;
             }
         }
         server.createPackageJsonCache = createPackageJsonCache;
@@ -147491,6 +156104,7 @@ var ts;
                 code: diag.code,
                 category: ts.diagnosticCategoryName(diag),
                 reportsUnnecessary: diag.reportsUnnecessary,
+                reportsDeprecated: diag.reportsDeprecated,
                 source: diag.source,
                 relatedInformation: ts.map(diag.relatedInformation, formatRelatedInformation),
             };
@@ -147530,6 +156144,7 @@ var ts;
                 code: code,
                 category: category,
                 reportsUnnecessary: diag.reportsUnnecessary,
+                reportsDeprecated: diag.reportsDeprecated,
                 source: source,
                 relatedInformation: ts.map(diag.relatedInformation, formatRelatedInformation),
             };
@@ -147650,25 +156265,37 @@ var ts;
             }
             return ts.deduplicate(outputs, ts.equateValues);
         }
+        function combineOutputResultContains(outputs, output, areEqual) {
+            return outputs.some(function (_a) {
+                var result = _a.result;
+                return ts.contains(result, output, areEqual);
+            });
+        }
+        function addToCombineOutputResult(outputs, project, result) {
+            if (result.length)
+                outputs.push({ project: project, result: result });
+        }
         function combineProjectOutputFromEveryProject(projectService, action, areEqual) {
             var outputs = [];
             projectService.loadAncestorProjectTree();
             projectService.forEachEnabledProject(function (project) {
                 var theseOutputs = action(project);
-                outputs.push.apply(outputs, theseOutputs.filter(function (output) { return !outputs.some(function (o) { return areEqual(o, output); }); }));
+                addToCombineOutputResult(outputs, project, ts.filter(theseOutputs, function (output) { return !combineOutputResultContains(outputs, output, areEqual); }));
             });
             return outputs;
         }
+        function flattenCombineOutputResult(outputs) {
+            return ts.flatMap(outputs, function (_a) {
+                var result = _a.result;
+                return result;
+            });
+        }
         function combineProjectOutputWhileOpeningReferencedProjects(projects, defaultProject, action, getLocation, resultsEqual) {
             var outputs = [];
             combineProjectOutputWorker(projects, defaultProject, 
             /*initialLocation*/ undefined, function (project, _, tryAddToTodo) {
-                for (var _i = 0, _a = action(project); _i < _a.length; _i++) {
-                    var output = _a[_i];
-                    if (!ts.contains(outputs, output, resultsEqual) && !tryAddToTodo(project, getLocation(output))) {
-                        outputs.push(output);
-                    }
-                }
+                var theseOutputs = action(project);
+                addToCombineOutputResult(outputs, project, ts.filter(theseOutputs, function (output) { return !combineOutputResultContains(outputs, output, resultsEqual) && !tryAddToTodo(project, getLocation(output)); }));
             });
             return outputs;
         }
@@ -147733,7 +156360,7 @@ var ts;
         function combineProjectOutputWorker(projects, defaultProject, initialLocation, cb) {
             var projectService = defaultProject.projectService;
             var toDo;
-            var seenProjects = ts.createMap();
+            var seenProjects = new ts.Set();
             forEachProjectInProjects(projects, initialLocation && initialLocation.fileName, function (project, path) {
                 // TLocation should be either `DocumentPosition` or `undefined`. Since `initialLocation` is `TLocation` this cast should be valid.
                 var location = (initialLocation ? { fileName: path, pos: initialLocation.pos } : undefined);
@@ -147829,7 +156456,7 @@ var ts;
                 toDo.push({ project: project, location: location });
         }
         function addToSeen(seenProjects, project) {
-            return ts.addToSeen(seenProjects, getProjectKey(project));
+            return ts.tryAddToSet(seenProjects, getProjectKey(project));
         }
         function getProjectKey(project) {
             return server.isConfiguredProject(project) ? project.canonicalConfigFilePath : project.getProjectName();
@@ -147865,12 +156492,75 @@ var ts;
                 { start: contextSpanStart.pos, length: contextSpanEnd.pos - contextSpanStart.pos } :
                 undefined;
         }
+        var invalidPartialSemanticModeCommands = [
+            server.CommandNames.OpenExternalProject,
+            server.CommandNames.OpenExternalProjects,
+            server.CommandNames.CloseExternalProject,
+            server.CommandNames.SynchronizeProjectList,
+            server.CommandNames.EmitOutput,
+            server.CommandNames.CompileOnSaveAffectedFileList,
+            server.CommandNames.CompileOnSaveEmitFile,
+            server.CommandNames.CompilerOptionsDiagnosticsFull,
+            server.CommandNames.EncodedSemanticClassificationsFull,
+            server.CommandNames.SemanticDiagnosticsSync,
+            server.CommandNames.SyntacticDiagnosticsSync,
+            server.CommandNames.SuggestionDiagnosticsSync,
+            server.CommandNames.Geterr,
+            server.CommandNames.GeterrForProject,
+            server.CommandNames.Reload,
+            server.CommandNames.ReloadProjects,
+            server.CommandNames.GetCodeFixes,
+            server.CommandNames.GetCodeFixesFull,
+            server.CommandNames.GetCombinedCodeFix,
+            server.CommandNames.GetCombinedCodeFixFull,
+            server.CommandNames.ApplyCodeActionCommand,
+            server.CommandNames.GetSupportedCodeFixes,
+            server.CommandNames.GetApplicableRefactors,
+            server.CommandNames.GetEditsForRefactor,
+            server.CommandNames.GetEditsForRefactorFull,
+            server.CommandNames.OrganizeImports,
+            server.CommandNames.OrganizeImportsFull,
+            server.CommandNames.GetEditsForFileRename,
+            server.CommandNames.GetEditsForFileRenameFull,
+            server.CommandNames.ConfigurePlugin,
+            server.CommandNames.PrepareCallHierarchy,
+            server.CommandNames.ProvideCallHierarchyIncomingCalls,
+            server.CommandNames.ProvideCallHierarchyOutgoingCalls,
+        ];
+        var invalidSyntacticModeCommands = __spreadArrays(invalidPartialSemanticModeCommands, [
+            server.CommandNames.Definition,
+            server.CommandNames.DefinitionFull,
+            server.CommandNames.DefinitionAndBoundSpan,
+            server.CommandNames.DefinitionAndBoundSpanFull,
+            server.CommandNames.TypeDefinition,
+            server.CommandNames.Implementation,
+            server.CommandNames.ImplementationFull,
+            server.CommandNames.References,
+            server.CommandNames.ReferencesFull,
+            server.CommandNames.Rename,
+            server.CommandNames.RenameLocationsFull,
+            server.CommandNames.RenameInfoFull,
+            server.CommandNames.Quickinfo,
+            server.CommandNames.QuickinfoFull,
+            server.CommandNames.CompletionInfo,
+            server.CommandNames.Completions,
+            server.CommandNames.CompletionsFull,
+            server.CommandNames.CompletionDetails,
+            server.CommandNames.CompletionDetailsFull,
+            server.CommandNames.SignatureHelp,
+            server.CommandNames.SignatureHelpFull,
+            server.CommandNames.Navto,
+            server.CommandNames.NavtoFull,
+            server.CommandNames.Occurrences,
+            server.CommandNames.DocumentHighlights,
+            server.CommandNames.DocumentHighlightsFull,
+        ]);
         var Session = /** @class */ (function () {
             function Session(opts) {
                 var _a;
                 var _this = this;
                 this.changeSeq = 0;
-                this.handlers = ts.createMapFromTemplate((_a = {},
+                this.handlers = new ts.Map(ts.getEntries((_a = {},
                     _a[server.CommandNames.Status] = function () {
                         var response = { version: ts.version }; // eslint-disable-line @typescript-eslint/no-unnecessary-qualifier
                         return _this.requiredResponse(response);
@@ -148218,7 +156908,31 @@ var ts;
                     _a[server.CommandNames.ProvideCallHierarchyOutgoingCalls] = function (request) {
                         return _this.requiredResponse(_this.provideCallHierarchyOutgoingCalls(request.arguments));
                     },
-                    _a));
+                    _a[server.CommandNames.ToggleLineComment] = function (request) {
+                        return _this.requiredResponse(_this.toggleLineComment(request.arguments, /*simplifiedResult*/ true));
+                    },
+                    _a[server.CommandNames.ToggleLineCommentFull] = function (request) {
+                        return _this.requiredResponse(_this.toggleLineComment(request.arguments, /*simplifiedResult*/ false));
+                    },
+                    _a[server.CommandNames.ToggleMultilineComment] = function (request) {
+                        return _this.requiredResponse(_this.toggleMultilineComment(request.arguments, /*simplifiedResult*/ true));
+                    },
+                    _a[server.CommandNames.ToggleMultilineCommentFull] = function (request) {
+                        return _this.requiredResponse(_this.toggleMultilineComment(request.arguments, /*simplifiedResult*/ false));
+                    },
+                    _a[server.CommandNames.CommentSelection] = function (request) {
+                        return _this.requiredResponse(_this.commentSelection(request.arguments, /*simplifiedResult*/ true));
+                    },
+                    _a[server.CommandNames.CommentSelectionFull] = function (request) {
+                        return _this.requiredResponse(_this.commentSelection(request.arguments, /*simplifiedResult*/ false));
+                    },
+                    _a[server.CommandNames.UncommentSelection] = function (request) {
+                        return _this.requiredResponse(_this.uncommentSelection(request.arguments, /*simplifiedResult*/ true));
+                    },
+                    _a[server.CommandNames.UncommentSelectionFull] = function (request) {
+                        return _this.requiredResponse(_this.uncommentSelection(request.arguments, /*simplifiedResult*/ false));
+                    },
+                    _a)));
                 this.host = opts.host;
                 this.cancellationToken = opts.cancellationToken;
                 this.typingsInstaller = opts.typingsInstaller;
@@ -148256,20 +156970,51 @@ var ts;
                     allowLocalPluginLoads: opts.allowLocalPluginLoads,
                     typesMapLocation: opts.typesMapLocation,
                     syntaxOnly: opts.syntaxOnly,
+                    serverMode: opts.serverMode,
                 };
                 this.projectService = new server.ProjectService(settings);
                 this.projectService.setPerformanceEventHandler(this.performanceEventHandler.bind(this));
                 this.gcTimer = new server.GcTimer(this.host, /*delay*/ 7000, this.logger);
+                // Make sure to setup handlers to throw error for not allowed commands on syntax server
+                switch (this.projectService.serverMode) {
+                    case ts.LanguageServiceMode.Semantic:
+                        break;
+                    case ts.LanguageServiceMode.PartialSemantic:
+                        invalidPartialSemanticModeCommands.forEach(function (commandName) {
+                            return _this.handlers.set(commandName, function (request) {
+                                throw new Error("Request: " + request.command + " not allowed in LanguageServiceMode.PartialSemantic");
+                            });
+                        });
+                        break;
+                    case ts.LanguageServiceMode.Syntactic:
+                        invalidSyntacticModeCommands.forEach(function (commandName) {
+                            return _this.handlers.set(commandName, function (request) {
+                                throw new Error("Request: " + request.command + " not allowed in LanguageServiceMode.Syntactic");
+                            });
+                        });
+                        break;
+                    default:
+                        ts.Debug.assertNever(this.projectService.serverMode);
+                }
             }
             Session.prototype.sendRequestCompletedEvent = function (requestId) {
                 this.event({ request_seq: requestId }, "requestCompleted");
             };
+            Session.prototype.addPerformanceData = function (key, value) {
+                var _a;
+                if (!this.performanceData) {
+                    this.performanceData = {};
+                }
+                this.performanceData[key] = ((_a = this.performanceData[key]) !== null && _a !== void 0 ? _a : 0) + value;
+            };
             Session.prototype.performanceEventHandler = function (event) {
                 switch (event.kind) {
-                    case "UpdateGraph": {
-                        this.updateGraphDurationMs = (this.updateGraphDurationMs || 0) + event.durationMs;
+                    case "UpdateGraph":
+                        this.addPerformanceData("updateGraphDurationMs", event.durationMs);
+                        break;
+                    case "CreatePackageJsonAutoImportProvider":
+                        this.addPerformanceData("createAutoImportProviderProgramDurationMs", event.durationMs);
                         break;
-                    }
                 }
             };
             Session.prototype.defaultEventHandler = function (event) {
@@ -148397,11 +157142,7 @@ var ts;
                     command: cmdName,
                     request_seq: reqSeq,
                     success: success,
-                    performanceData: !this.updateGraphDurationMs
-                        ? undefined
-                        : {
-                            updateGraphDurationMs: this.updateGraphDurationMs,
-                        },
+                    performanceData: this.performanceData
                 };
                 if (success) {
                     var metadata = void 0;
@@ -148537,7 +157278,7 @@ var ts;
             };
             Session.prototype.getEncodedSemanticClassifications = function (args) {
                 var _a = this.getFileAndProject(args), file = _a.file, project = _a.project;
-                return project.getLanguageService().getEncodedSemanticClassifications(file, args);
+                return project.getLanguageService().getEncodedSemanticClassifications(file, args, args.format);
             };
             Session.prototype.getProject = function (projectFileName) {
                 return projectFileName === undefined ? undefined : this.projectService.findProject(projectFileName);
@@ -148570,6 +157311,7 @@ var ts;
                     startLocation: (d.file && convertToLocation(ts.getLineAndCharacterOfPosition(d.file, d.start))),
                     endLocation: (d.file && convertToLocation(ts.getLineAndCharacterOfPosition(d.file, d.start + d.length))),
                     reportsUnnecessary: d.reportsUnnecessary,
+                    reportsDeprecated: d.reportsDeprecated,
                     relatedInformation: ts.map(d.relatedInformation, formatRelatedInformation)
                 }); });
             };
@@ -148593,6 +157335,7 @@ var ts;
                     startLocation: scriptInfo && scriptInfo.positionToLineOffset(d.start),
                     endLocation: scriptInfo && scriptInfo.positionToLineOffset(d.start + d.length),
                     reportsUnnecessary: d.reportsUnnecessary,
+                    reportsDeprecated: d.reportsDeprecated,
                     relatedInformation: ts.map(d.relatedInformation, formatRelatedInformation),
                 }); });
             };
@@ -148749,9 +157492,9 @@ var ts;
                 return this.getDiagnosticsWorker(args, /*isSemantic*/ true, function (project, file) { return project.getLanguageService().getSuggestionDiagnostics(file); }, !!args.includeLinePosition);
             };
             Session.prototype.getJsxClosingTag = function (args) {
-                var _a = this.getFileAndProject(args), file = _a.file, project = _a.project;
+                var _a = this.getFileAndLanguageServiceForSyntacticOperation(args), file = _a.file, languageService = _a.languageService;
                 var position = this.getPositionInFile(args, file);
-                var tag = project.getLanguageService().getJsxClosingTagAtPosition(file, position);
+                var tag = languageService.getJsxClosingTagAtPosition(file, position);
                 return tag === undefined ? undefined : { newText: tag.newText, caretOffset: 0 };
             };
             Session.prototype.getDocumentHighlights = function (args, simplifiedResult) {
@@ -148860,7 +157603,7 @@ var ts;
                 }
             };
             Session.prototype.toSpanGroups = function (locations) {
-                var map = ts.createMap();
+                var map = new ts.Map();
                 for (var _i = 0, locations_1 = locations; _i < locations_1.length; _i++) {
                     var _a = locations_1[_i];
                     var fileName = _a.fileName, textSpan = _a.textSpan, contextSpan = _a.contextSpan, _2 = _a.originalContextSpan, _ = _a.originalTextSpan, _1 = _a.originalFileName, prefixSuffixText = __rest(_a, ["fileName", "textSpan", "contextSpan", "originalContextSpan", "originalTextSpan", "originalFileName"]);
@@ -149104,10 +157847,10 @@ var ts;
                 var prefix = args.prefix || "";
                 var entries = ts.mapDefined(completions.entries, function (entry) {
                     if (completions.isMemberCompletion || ts.startsWith(entry.name.toLowerCase(), prefix.toLowerCase())) {
-                        var name = entry.name, kind_1 = entry.kind, kindModifiers = entry.kindModifiers, sortText = entry.sortText, insertText = entry.insertText, replacementSpan = entry.replacementSpan, hasAction = entry.hasAction, source = entry.source, isRecommended = entry.isRecommended;
+                        var name = entry.name, kind_1 = entry.kind, kindModifiers = entry.kindModifiers, sortText = entry.sortText, insertText = entry.insertText, replacementSpan = entry.replacementSpan, hasAction = entry.hasAction, source = entry.source, isRecommended = entry.isRecommended, isPackageJsonImport = entry.isPackageJsonImport;
                         var convertedSpan = replacementSpan ? toProtocolTextSpan(replacementSpan, scriptInfo) : undefined;
                         // Use `hasAction || undefined` to avoid serializing `false`.
-                        return { name: name, kind: kind_1, kindModifiers: kindModifiers, sortText: sortText, insertText: insertText, replacementSpan: convertedSpan, hasAction: hasAction || undefined, source: source, isRecommended: isRecommended };
+                        return { name: name, kind: kind_1, kindModifiers: kindModifiers, sortText: sortText, insertText: insertText, replacementSpan: convertedSpan, hasAction: hasAction || undefined, source: source, isRecommended: isRecommended, isPackageJsonImport: isPackageJsonImport };
                     }
                 }).sort(function (a, b) { return ts.compareStringsCaseSensitiveUI(a.name, b.name); });
                 if (kind === "completions" /* Completions */) {
@@ -149115,7 +157858,7 @@ var ts;
                         entries.metadata = completions.metadata;
                     return entries;
                 }
-                var res = __assign(__assign({}, completions), { entries: entries });
+                var res = __assign(__assign({}, completions), { optionalReplacementSpan: completions.optionalReplacementSpan && toProtocolTextSpan(completions.optionalReplacementSpan, scriptInfo), entries: entries });
                 return res;
             };
             Session.prototype.getCompletionEntryDetails = function (args, simplifiedResult) {
@@ -149151,7 +157894,7 @@ var ts;
                     return {
                         projectFileName: project.getProjectName(),
                         fileNames: project.getCompileOnSaveAffectedFileList(info),
-                        projectUsesOutFile: !!compilationSettings.outFile || !!compilationSettings.out
+                        projectUsesOutFile: !!ts.outFile(compilationSettings)
                     };
                 });
             };
@@ -149292,38 +158035,42 @@ var ts;
                         : tree;
             };
             Session.prototype.getNavigateToItems = function (args, simplifiedResult) {
-                var _this = this;
                 var full = this.getFullNavigateToItems(args);
-                return !simplifiedResult ? full : full.map(function (navItem) {
-                    var _a = _this.getFileAndProject({ file: navItem.fileName }), file = _a.file, project = _a.project;
-                    var scriptInfo = project.getScriptInfo(file);
-                    var bakedItem = {
-                        name: navItem.name,
-                        kind: navItem.kind,
-                        isCaseSensitive: navItem.isCaseSensitive,
-                        matchKind: navItem.matchKind,
-                        file: navItem.fileName,
-                        start: scriptInfo.positionToLineOffset(navItem.textSpan.start),
-                        end: scriptInfo.positionToLineOffset(ts.textSpanEnd(navItem.textSpan))
-                    };
-                    if (navItem.kindModifiers && (navItem.kindModifiers !== "")) {
-                        bakedItem.kindModifiers = navItem.kindModifiers;
-                    }
-                    if (navItem.containerName && (navItem.containerName.length > 0)) {
-                        bakedItem.containerName = navItem.containerName;
-                    }
-                    if (navItem.containerKind && (navItem.containerKind.length > 0)) {
-                        bakedItem.containerKind = navItem.containerKind;
-                    }
-                    return bakedItem;
-                });
+                return !simplifiedResult ?
+                    flattenCombineOutputResult(full) :
+                    ts.flatMap(full, function (_a) {
+                        var project = _a.project, result = _a.result;
+                        return result.map(function (navItem) {
+                            var scriptInfo = project.getScriptInfo(navItem.fileName);
+                            var bakedItem = {
+                                name: navItem.name,
+                                kind: navItem.kind,
+                                kindModifiers: navItem.kindModifiers,
+                                isCaseSensitive: navItem.isCaseSensitive,
+                                matchKind: navItem.matchKind,
+                                file: navItem.fileName,
+                                start: scriptInfo.positionToLineOffset(navItem.textSpan.start),
+                                end: scriptInfo.positionToLineOffset(ts.textSpanEnd(navItem.textSpan))
+                            };
+                            if (navItem.kindModifiers && (navItem.kindModifiers !== "")) {
+                                bakedItem.kindModifiers = navItem.kindModifiers;
+                            }
+                            if (navItem.containerName && (navItem.containerName.length > 0)) {
+                                bakedItem.containerName = navItem.containerName;
+                            }
+                            if (navItem.containerKind && (navItem.containerKind.length > 0)) {
+                                bakedItem.containerKind = navItem.containerKind;
+                            }
+                            return bakedItem;
+                        });
+                    });
             };
             Session.prototype.getFullNavigateToItems = function (args) {
                 var currentFileOnly = args.currentFileOnly, searchValue = args.searchValue, maxResultCount = args.maxResultCount, projectFileName = args.projectFileName;
                 if (currentFileOnly) {
                     ts.Debug.assertDefined(args.file);
                     var _a = this.getFileAndProject(args), file = _a.file, project = _a.project;
-                    return project.getLanguageService().getNavigateToItems(searchValue, maxResultCount, file);
+                    return [{ project: project, result: project.getLanguageService().getNavigateToItems(searchValue, maxResultCount, file) }];
                 }
                 else if (!args.file && !projectFileName) {
                     return combineProjectOutputFromEveryProject(this.projectService, function (project) { return project.getLanguageService().getNavigateToItems(searchValue, maxResultCount, /*filename*/ undefined, /*excludeDts*/ project.isNonTsProject()); }, navigateToItemIsEqualTo);
@@ -149362,18 +158109,21 @@ var ts;
                     position = getPosition(args);
                 }
                 else {
-                    var _a = this.getStartAndEndPosition(args, scriptInfo), startPosition = _a.startPosition, endPosition = _a.endPosition;
-                    textRange = { pos: startPosition, end: endPosition };
+                    textRange = this.getRange(args, scriptInfo);
                 }
                 return ts.Debug.checkDefined(position === undefined ? textRange : position);
                 function getPosition(loc) {
                     return loc.position !== undefined ? loc.position : scriptInfo.lineOffsetToPosition(loc.line, loc.offset);
                 }
             };
+            Session.prototype.getRange = function (args, scriptInfo) {
+                var _a = this.getStartAndEndPosition(args, scriptInfo), startPosition = _a.startPosition, endPosition = _a.endPosition;
+                return { pos: startPosition, end: endPosition };
+            };
             Session.prototype.getApplicableRefactors = function (args) {
                 var _a = this.getFileAndProject(args), file = _a.file, project = _a.project;
                 var scriptInfo = project.getScriptInfoForNormalizedPath(file);
-                return project.getLanguageService().getApplicableRefactors(file, this.extractPositionOrRange(args, scriptInfo), this.getPreferences(file));
+                return project.getLanguageService().getApplicableRefactors(file, this.extractPositionOrRange(args, scriptInfo), this.getPreferences(file), args.triggerReason);
             };
             Session.prototype.getEditsForRefactor = function (args, simplifiedResult) {
                 var _a = this.getFileAndProject(args), file = _a.file, project = _a.project;
@@ -149415,7 +158165,7 @@ var ts;
                 var newPath = server.toNormalizedPath(args.newFilePath);
                 var formatOptions = this.getHostFormatOptions();
                 var preferences = this.getHostPreferences();
-                var changes = combineProjectOutputFromEveryProject(this.projectService, function (project) { return project.getLanguageService().getEditsForFileRename(oldPath, newPath, formatOptions, preferences); }, function (a, b) { return a.fileName === b.fileName; });
+                var changes = flattenCombineOutputResult(combineProjectOutputFromEveryProject(this.projectService, function (project) { return project.getLanguageService().getEditsForFileRename(oldPath, newPath, formatOptions, preferences); }, function (a, b) { return a.fileName === b.fileName; }));
                 return simplifiedResult ? changes.map(function (c) { return _this.mapTextChangeToCodeEdit(c); }) : changes;
             };
             Session.prototype.getCodeFixes = function (args, simplifiedResult) {
@@ -149568,6 +158318,54 @@ var ts;
                     return simplifiedResult ? _this.mapSelectionRange(selectionRange, scriptInfo) : selectionRange;
                 });
             };
+            Session.prototype.toggleLineComment = function (args, simplifiedResult) {
+                var _this = this;
+                var _a = this.getFileAndLanguageServiceForSyntacticOperation(args), file = _a.file, languageService = _a.languageService;
+                var scriptInfo = this.projectService.getScriptInfo(file);
+                var textRange = this.getRange(args, scriptInfo);
+                var textChanges = languageService.toggleLineComment(file, textRange);
+                if (simplifiedResult) {
+                    var scriptInfo_2 = this.projectService.getScriptInfoForNormalizedPath(file);
+                    return textChanges.map(function (textChange) { return _this.convertTextChangeToCodeEdit(textChange, scriptInfo_2); });
+                }
+                return textChanges;
+            };
+            Session.prototype.toggleMultilineComment = function (args, simplifiedResult) {
+                var _this = this;
+                var _a = this.getFileAndLanguageServiceForSyntacticOperation(args), file = _a.file, languageService = _a.languageService;
+                var scriptInfo = this.projectService.getScriptInfoForNormalizedPath(file);
+                var textRange = this.getRange(args, scriptInfo);
+                var textChanges = languageService.toggleMultilineComment(file, textRange);
+                if (simplifiedResult) {
+                    var scriptInfo_3 = this.projectService.getScriptInfoForNormalizedPath(file);
+                    return textChanges.map(function (textChange) { return _this.convertTextChangeToCodeEdit(textChange, scriptInfo_3); });
+                }
+                return textChanges;
+            };
+            Session.prototype.commentSelection = function (args, simplifiedResult) {
+                var _this = this;
+                var _a = this.getFileAndLanguageServiceForSyntacticOperation(args), file = _a.file, languageService = _a.languageService;
+                var scriptInfo = this.projectService.getScriptInfoForNormalizedPath(file);
+                var textRange = this.getRange(args, scriptInfo);
+                var textChanges = languageService.commentSelection(file, textRange);
+                if (simplifiedResult) {
+                    var scriptInfo_4 = this.projectService.getScriptInfoForNormalizedPath(file);
+                    return textChanges.map(function (textChange) { return _this.convertTextChangeToCodeEdit(textChange, scriptInfo_4); });
+                }
+                return textChanges;
+            };
+            Session.prototype.uncommentSelection = function (args, simplifiedResult) {
+                var _this = this;
+                var _a = this.getFileAndLanguageServiceForSyntacticOperation(args), file = _a.file, languageService = _a.languageService;
+                var scriptInfo = this.projectService.getScriptInfoForNormalizedPath(file);
+                var textRange = this.getRange(args, scriptInfo);
+                var textChanges = languageService.uncommentSelection(file, textRange);
+                if (simplifiedResult) {
+                    var scriptInfo_5 = this.projectService.getScriptInfoForNormalizedPath(file);
+                    return textChanges.map(function (textChange) { return _this.convertTextChangeToCodeEdit(textChange, scriptInfo_5); });
+                }
+                return textChanges;
+            };
             Session.prototype.mapSelectionRange = function (selectionRange, scriptInfo) {
                 var result = {
                     textSpan: toProtocolTextSpan(selectionRange.textSpan, scriptInfo),
@@ -149591,7 +158389,9 @@ var ts;
                 return {
                     name: item.name,
                     kind: item.kind,
+                    kindModifiers: item.kindModifiers,
                     file: item.file,
+                    containerName: item.containerName,
                     span: toProtocolTextSpan(item.span, scriptInfo),
                     selectionSpan: toProtocolTextSpan(item.selectionSpan, scriptInfo)
                 };
@@ -149683,7 +158483,7 @@ var ts;
             };
             Session.prototype.onMessage = function (message) {
                 this.gcTimer.scheduleCollect();
-                this.updateGraphDurationMs = undefined;
+                this.performanceData = undefined;
                 var start;
                 if (this.logger.hasLevel(server.LogLevel.requestTime)) {
                     start = this.hrtime();
@@ -150555,5 +159355,904 @@ var ts;
         }());
     })(server = ts.server || (ts.server = {}));
 })(ts || (ts = {}));
+var ts;
+(function (ts) {
+    // The following are deprecations for the public API. Deprecated exports are removed from the compiler itself
+    // and compatible implementations are added here, along with an appropriate deprecation warning using
+    // the `@deprecated` JSDoc tag as well as the `Debug.deprecate` API.
+    //
+    // Deprecations fall into one of three categories:
+    //
+    //   * "soft" - Soft deprecations are indicated with the `@deprecated` JSDoc Tag.
+    //   * "warn" - Warning deprecations are indicated with the `@deprecated` JSDoc Tag and a diagnostic message (assuming a compatible host)
+    //   * "error" - Error deprecations are indicated with the `@deprecated` JSDoc tag and will throw a `TypeError` when invoked.
+    // DEPRECATION: Node factory top-level exports
+    // DEPRECATION PLAN:
+    //     - soft: 4.0
+    //     - warn: 4.1
+    //     - error: TBD
+    // #region Node factory top-level exports
+    // NOTE: These exports are deprecated in favor of using a `NodeFactory` instance and exist here purely for backwards compatibility reasons.
+    var factoryDeprecation = { since: "4.0", warnAfter: "4.1", message: "Use the appropriate method on 'ts.factory' or the 'factory' supplied by your transformation context instead." };
+    /** @deprecated Use `factory.createNodeArray` or the factory supplied by your transformation context instead. */
+    ts.createNodeArray = ts.Debug.deprecate(ts.factory.createNodeArray, factoryDeprecation);
+    /** @deprecated Use `factory.createNumericLiteral` or the factory supplied by your transformation context instead. */
+    ts.createNumericLiteral = ts.Debug.deprecate(ts.factory.createNumericLiteral, factoryDeprecation);
+    /** @deprecated Use `factory.createBigIntLiteral` or the factory supplied by your transformation context instead. */
+    ts.createBigIntLiteral = ts.Debug.deprecate(ts.factory.createBigIntLiteral, factoryDeprecation);
+    /** @deprecated Use `factory.createStringLiteral` or the factory supplied by your transformation context instead. */
+    ts.createStringLiteral = ts.Debug.deprecate(ts.factory.createStringLiteral, factoryDeprecation);
+    /** @deprecated Use `factory.createStringLiteralFromNode` or the factory supplied by your transformation context instead. */
+    ts.createStringLiteralFromNode = ts.Debug.deprecate(ts.factory.createStringLiteralFromNode, factoryDeprecation);
+    /** @deprecated Use `factory.createRegularExpressionLiteral` or the factory supplied by your transformation context instead. */
+    ts.createRegularExpressionLiteral = ts.Debug.deprecate(ts.factory.createRegularExpressionLiteral, factoryDeprecation);
+    /** @deprecated Use `factory.createLoopVariable` or the factory supplied by your transformation context instead. */
+    ts.createLoopVariable = ts.Debug.deprecate(ts.factory.createLoopVariable, factoryDeprecation);
+    /** @deprecated Use `factory.createUniqueName` or the factory supplied by your transformation context instead. */
+    ts.createUniqueName = ts.Debug.deprecate(ts.factory.createUniqueName, factoryDeprecation);
+    /** @deprecated Use `factory.createPrivateIdentifier` or the factory supplied by your transformation context instead. */
+    ts.createPrivateIdentifier = ts.Debug.deprecate(ts.factory.createPrivateIdentifier, factoryDeprecation);
+    /** @deprecated Use `factory.createSuper` or the factory supplied by your transformation context instead. */
+    ts.createSuper = ts.Debug.deprecate(ts.factory.createSuper, factoryDeprecation);
+    /** @deprecated Use `factory.createThis` or the factory supplied by your transformation context instead. */
+    ts.createThis = ts.Debug.deprecate(ts.factory.createThis, factoryDeprecation);
+    /** @deprecated Use `factory.createNull` or the factory supplied by your transformation context instead. */
+    ts.createNull = ts.Debug.deprecate(ts.factory.createNull, factoryDeprecation);
+    /** @deprecated Use `factory.createTrue` or the factory supplied by your transformation context instead. */
+    ts.createTrue = ts.Debug.deprecate(ts.factory.createTrue, factoryDeprecation);
+    /** @deprecated Use `factory.createFalse` or the factory supplied by your transformation context instead. */
+    ts.createFalse = ts.Debug.deprecate(ts.factory.createFalse, factoryDeprecation);
+    /** @deprecated Use `factory.createModifier` or the factory supplied by your transformation context instead. */
+    ts.createModifier = ts.Debug.deprecate(ts.factory.createModifier, factoryDeprecation);
+    /** @deprecated Use `factory.createModifiersFromModifierFlags` or the factory supplied by your transformation context instead. */
+    ts.createModifiersFromModifierFlags = ts.Debug.deprecate(ts.factory.createModifiersFromModifierFlags, factoryDeprecation);
+    /** @deprecated Use `factory.createQualifiedName` or the factory supplied by your transformation context instead. */
+    ts.createQualifiedName = ts.Debug.deprecate(ts.factory.createQualifiedName, factoryDeprecation);
+    /** @deprecated Use `factory.updateQualifiedName` or the factory supplied by your transformation context instead. */
+    ts.updateQualifiedName = ts.Debug.deprecate(ts.factory.updateQualifiedName, factoryDeprecation);
+    /** @deprecated Use `factory.createComputedPropertyName` or the factory supplied by your transformation context instead. */
+    ts.createComputedPropertyName = ts.Debug.deprecate(ts.factory.createComputedPropertyName, factoryDeprecation);
+    /** @deprecated Use `factory.updateComputedPropertyName` or the factory supplied by your transformation context instead. */
+    ts.updateComputedPropertyName = ts.Debug.deprecate(ts.factory.updateComputedPropertyName, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeParameterDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createTypeParameterDeclaration = ts.Debug.deprecate(ts.factory.createTypeParameterDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeParameterDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateTypeParameterDeclaration = ts.Debug.deprecate(ts.factory.updateTypeParameterDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createParameterDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createParameter = ts.Debug.deprecate(ts.factory.createParameterDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateParameterDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateParameter = ts.Debug.deprecate(ts.factory.updateParameterDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createDecorator` or the factory supplied by your transformation context instead. */
+    ts.createDecorator = ts.Debug.deprecate(ts.factory.createDecorator, factoryDeprecation);
+    /** @deprecated Use `factory.updateDecorator` or the factory supplied by your transformation context instead. */
+    ts.updateDecorator = ts.Debug.deprecate(ts.factory.updateDecorator, factoryDeprecation);
+    /** @deprecated Use `factory.createPropertyDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createProperty = ts.Debug.deprecate(ts.factory.createPropertyDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updatePropertyDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateProperty = ts.Debug.deprecate(ts.factory.updatePropertyDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createMethodDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createMethod = ts.Debug.deprecate(ts.factory.createMethodDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateMethodDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateMethod = ts.Debug.deprecate(ts.factory.updateMethodDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createConstructorDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createConstructor = ts.Debug.deprecate(ts.factory.createConstructorDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateConstructorDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateConstructor = ts.Debug.deprecate(ts.factory.updateConstructorDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createGetAccessorDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createGetAccessor = ts.Debug.deprecate(ts.factory.createGetAccessorDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateGetAccessorDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateGetAccessor = ts.Debug.deprecate(ts.factory.updateGetAccessorDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createSetAccessorDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createSetAccessor = ts.Debug.deprecate(ts.factory.createSetAccessorDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateSetAccessorDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateSetAccessor = ts.Debug.deprecate(ts.factory.updateSetAccessorDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createCallSignature` or the factory supplied by your transformation context instead. */
+    ts.createCallSignature = ts.Debug.deprecate(ts.factory.createCallSignature, factoryDeprecation);
+    /** @deprecated Use `factory.updateCallSignature` or the factory supplied by your transformation context instead. */
+    ts.updateCallSignature = ts.Debug.deprecate(ts.factory.updateCallSignature, factoryDeprecation);
+    /** @deprecated Use `factory.createConstructSignature` or the factory supplied by your transformation context instead. */
+    ts.createConstructSignature = ts.Debug.deprecate(ts.factory.createConstructSignature, factoryDeprecation);
+    /** @deprecated Use `factory.updateConstructSignature` or the factory supplied by your transformation context instead. */
+    ts.updateConstructSignature = ts.Debug.deprecate(ts.factory.updateConstructSignature, factoryDeprecation);
+    /** @deprecated Use `factory.updateIndexSignature` or the factory supplied by your transformation context instead. */
+    ts.updateIndexSignature = ts.Debug.deprecate(ts.factory.updateIndexSignature, factoryDeprecation);
+    /** @deprecated Use `factory.createKeywordTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createKeywordTypeNode = ts.Debug.deprecate(ts.factory.createKeywordTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createTypePredicateNode` or the factory supplied by your transformation context instead. */
+    ts.createTypePredicateNodeWithModifier = ts.Debug.deprecate(ts.factory.createTypePredicateNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypePredicateNode` or the factory supplied by your transformation context instead. */
+    ts.updateTypePredicateNodeWithModifier = ts.Debug.deprecate(ts.factory.updateTypePredicateNode, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeReferenceNode` or the factory supplied by your transformation context instead. */
+    ts.createTypeReferenceNode = ts.Debug.deprecate(ts.factory.createTypeReferenceNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeReferenceNode` or the factory supplied by your transformation context instead. */
+    ts.updateTypeReferenceNode = ts.Debug.deprecate(ts.factory.updateTypeReferenceNode, factoryDeprecation);
+    /** @deprecated Use `factory.createFunctionTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createFunctionTypeNode = ts.Debug.deprecate(ts.factory.createFunctionTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateFunctionTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateFunctionTypeNode = ts.Debug.deprecate(ts.factory.updateFunctionTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createConstructorTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createConstructorTypeNode = ts.Debug.deprecate(ts.factory.createConstructorTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateConstructorTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateConstructorTypeNode = ts.Debug.deprecate(ts.factory.updateConstructorTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeQueryNode` or the factory supplied by your transformation context instead. */
+    ts.createTypeQueryNode = ts.Debug.deprecate(ts.factory.createTypeQueryNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeQueryNode` or the factory supplied by your transformation context instead. */
+    ts.updateTypeQueryNode = ts.Debug.deprecate(ts.factory.updateTypeQueryNode, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeLiteralNode` or the factory supplied by your transformation context instead. */
+    ts.createTypeLiteralNode = ts.Debug.deprecate(ts.factory.createTypeLiteralNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeLiteralNode` or the factory supplied by your transformation context instead. */
+    ts.updateTypeLiteralNode = ts.Debug.deprecate(ts.factory.updateTypeLiteralNode, factoryDeprecation);
+    /** @deprecated Use `factory.createArrayTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createArrayTypeNode = ts.Debug.deprecate(ts.factory.createArrayTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateArrayTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateArrayTypeNode = ts.Debug.deprecate(ts.factory.updateArrayTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createTupleTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createTupleTypeNode = ts.Debug.deprecate(ts.factory.createTupleTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateTupleTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateTupleTypeNode = ts.Debug.deprecate(ts.factory.updateTupleTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createOptionalTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createOptionalTypeNode = ts.Debug.deprecate(ts.factory.createOptionalTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateOptionalTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateOptionalTypeNode = ts.Debug.deprecate(ts.factory.updateOptionalTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createRestTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createRestTypeNode = ts.Debug.deprecate(ts.factory.createRestTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateRestTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateRestTypeNode = ts.Debug.deprecate(ts.factory.updateRestTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createUnionTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createUnionTypeNode = ts.Debug.deprecate(ts.factory.createUnionTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateUnionTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateUnionTypeNode = ts.Debug.deprecate(ts.factory.updateUnionTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createIntersectionTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createIntersectionTypeNode = ts.Debug.deprecate(ts.factory.createIntersectionTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateIntersectionTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateIntersectionTypeNode = ts.Debug.deprecate(ts.factory.updateIntersectionTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createConditionalTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createConditionalTypeNode = ts.Debug.deprecate(ts.factory.createConditionalTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateConditionalTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateConditionalTypeNode = ts.Debug.deprecate(ts.factory.updateConditionalTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createInferTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createInferTypeNode = ts.Debug.deprecate(ts.factory.createInferTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateInferTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateInferTypeNode = ts.Debug.deprecate(ts.factory.updateInferTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createImportTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createImportTypeNode = ts.Debug.deprecate(ts.factory.createImportTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateImportTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateImportTypeNode = ts.Debug.deprecate(ts.factory.updateImportTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createParenthesizedType` or the factory supplied by your transformation context instead. */
+    ts.createParenthesizedType = ts.Debug.deprecate(ts.factory.createParenthesizedType, factoryDeprecation);
+    /** @deprecated Use `factory.updateParenthesizedType` or the factory supplied by your transformation context instead. */
+    ts.updateParenthesizedType = ts.Debug.deprecate(ts.factory.updateParenthesizedType, factoryDeprecation);
+    /** @deprecated Use `factory.createThisTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createThisTypeNode = ts.Debug.deprecate(ts.factory.createThisTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeOperatorNode` or the factory supplied by your transformation context instead. */
+    ts.updateTypeOperatorNode = ts.Debug.deprecate(ts.factory.updateTypeOperatorNode, factoryDeprecation);
+    /** @deprecated Use `factory.createIndexedAccessTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createIndexedAccessTypeNode = ts.Debug.deprecate(ts.factory.createIndexedAccessTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateIndexedAccessTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateIndexedAccessTypeNode = ts.Debug.deprecate(ts.factory.updateIndexedAccessTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createMappedTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createMappedTypeNode = ts.Debug.deprecate(ts.factory.createMappedTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateMappedTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateMappedTypeNode = ts.Debug.deprecate(ts.factory.updateMappedTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createLiteralTypeNode` or the factory supplied by your transformation context instead. */
+    ts.createLiteralTypeNode = ts.Debug.deprecate(ts.factory.createLiteralTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.updateLiteralTypeNode` or the factory supplied by your transformation context instead. */
+    ts.updateLiteralTypeNode = ts.Debug.deprecate(ts.factory.updateLiteralTypeNode, factoryDeprecation);
+    /** @deprecated Use `factory.createObjectBindingPattern` or the factory supplied by your transformation context instead. */
+    ts.createObjectBindingPattern = ts.Debug.deprecate(ts.factory.createObjectBindingPattern, factoryDeprecation);
+    /** @deprecated Use `factory.updateObjectBindingPattern` or the factory supplied by your transformation context instead. */
+    ts.updateObjectBindingPattern = ts.Debug.deprecate(ts.factory.updateObjectBindingPattern, factoryDeprecation);
+    /** @deprecated Use `factory.createArrayBindingPattern` or the factory supplied by your transformation context instead. */
+    ts.createArrayBindingPattern = ts.Debug.deprecate(ts.factory.createArrayBindingPattern, factoryDeprecation);
+    /** @deprecated Use `factory.updateArrayBindingPattern` or the factory supplied by your transformation context instead. */
+    ts.updateArrayBindingPattern = ts.Debug.deprecate(ts.factory.updateArrayBindingPattern, factoryDeprecation);
+    /** @deprecated Use `factory.createBindingElement` or the factory supplied by your transformation context instead. */
+    ts.createBindingElement = ts.Debug.deprecate(ts.factory.createBindingElement, factoryDeprecation);
+    /** @deprecated Use `factory.updateBindingElement` or the factory supplied by your transformation context instead. */
+    ts.updateBindingElement = ts.Debug.deprecate(ts.factory.updateBindingElement, factoryDeprecation);
+    /** @deprecated Use `factory.createArrayLiteral` or the factory supplied by your transformation context instead. */
+    ts.createArrayLiteral = ts.Debug.deprecate(ts.factory.createArrayLiteralExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateArrayLiteral` or the factory supplied by your transformation context instead. */
+    ts.updateArrayLiteral = ts.Debug.deprecate(ts.factory.updateArrayLiteralExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createObjectLiteral` or the factory supplied by your transformation context instead. */
+    ts.createObjectLiteral = ts.Debug.deprecate(ts.factory.createObjectLiteralExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateObjectLiteral` or the factory supplied by your transformation context instead. */
+    ts.updateObjectLiteral = ts.Debug.deprecate(ts.factory.updateObjectLiteralExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createPropertyAccess` or the factory supplied by your transformation context instead. */
+    ts.createPropertyAccess = ts.Debug.deprecate(ts.factory.createPropertyAccessExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updatePropertyAccess` or the factory supplied by your transformation context instead. */
+    ts.updatePropertyAccess = ts.Debug.deprecate(ts.factory.updatePropertyAccessExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createPropertyAccessChain` or the factory supplied by your transformation context instead. */
+    ts.createPropertyAccessChain = ts.Debug.deprecate(ts.factory.createPropertyAccessChain, factoryDeprecation);
+    /** @deprecated Use `factory.updatePropertyAccessChain` or the factory supplied by your transformation context instead. */
+    ts.updatePropertyAccessChain = ts.Debug.deprecate(ts.factory.updatePropertyAccessChain, factoryDeprecation);
+    /** @deprecated Use `factory.createElementAccess` or the factory supplied by your transformation context instead. */
+    ts.createElementAccess = ts.Debug.deprecate(ts.factory.createElementAccessExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateElementAccess` or the factory supplied by your transformation context instead. */
+    ts.updateElementAccess = ts.Debug.deprecate(ts.factory.updateElementAccessExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createElementAccessChain` or the factory supplied by your transformation context instead. */
+    ts.createElementAccessChain = ts.Debug.deprecate(ts.factory.createElementAccessChain, factoryDeprecation);
+    /** @deprecated Use `factory.updateElementAccessChain` or the factory supplied by your transformation context instead. */
+    ts.updateElementAccessChain = ts.Debug.deprecate(ts.factory.updateElementAccessChain, factoryDeprecation);
+    /** @deprecated Use `factory.createCall` or the factory supplied by your transformation context instead. */
+    ts.createCall = ts.Debug.deprecate(ts.factory.createCallExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateCall` or the factory supplied by your transformation context instead. */
+    ts.updateCall = ts.Debug.deprecate(ts.factory.updateCallExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createCallChain` or the factory supplied by your transformation context instead. */
+    ts.createCallChain = ts.Debug.deprecate(ts.factory.createCallChain, factoryDeprecation);
+    /** @deprecated Use `factory.updateCallChain` or the factory supplied by your transformation context instead. */
+    ts.updateCallChain = ts.Debug.deprecate(ts.factory.updateCallChain, factoryDeprecation);
+    /** @deprecated Use `factory.createNew` or the factory supplied by your transformation context instead. */
+    ts.createNew = ts.Debug.deprecate(ts.factory.createNewExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateNew` or the factory supplied by your transformation context instead. */
+    ts.updateNew = ts.Debug.deprecate(ts.factory.updateNewExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeAssertion` or the factory supplied by your transformation context instead. */
+    ts.createTypeAssertion = ts.Debug.deprecate(ts.factory.createTypeAssertion, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeAssertion` or the factory supplied by your transformation context instead. */
+    ts.updateTypeAssertion = ts.Debug.deprecate(ts.factory.updateTypeAssertion, factoryDeprecation);
+    /** @deprecated Use `factory.createParen` or the factory supplied by your transformation context instead. */
+    ts.createParen = ts.Debug.deprecate(ts.factory.createParenthesizedExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateParen` or the factory supplied by your transformation context instead. */
+    ts.updateParen = ts.Debug.deprecate(ts.factory.updateParenthesizedExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createFunctionExpression` or the factory supplied by your transformation context instead. */
+    ts.createFunctionExpression = ts.Debug.deprecate(ts.factory.createFunctionExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateFunctionExpression` or the factory supplied by your transformation context instead. */
+    ts.updateFunctionExpression = ts.Debug.deprecate(ts.factory.updateFunctionExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createDelete` or the factory supplied by your transformation context instead. */
+    ts.createDelete = ts.Debug.deprecate(ts.factory.createDeleteExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateDelete` or the factory supplied by your transformation context instead. */
+    ts.updateDelete = ts.Debug.deprecate(ts.factory.updateDeleteExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeOf` or the factory supplied by your transformation context instead. */
+    ts.createTypeOf = ts.Debug.deprecate(ts.factory.createTypeOfExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeOf` or the factory supplied by your transformation context instead. */
+    ts.updateTypeOf = ts.Debug.deprecate(ts.factory.updateTypeOfExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createVoid` or the factory supplied by your transformation context instead. */
+    ts.createVoid = ts.Debug.deprecate(ts.factory.createVoidExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateVoid` or the factory supplied by your transformation context instead. */
+    ts.updateVoid = ts.Debug.deprecate(ts.factory.updateVoidExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createAwait` or the factory supplied by your transformation context instead. */
+    ts.createAwait = ts.Debug.deprecate(ts.factory.createAwaitExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateAwait` or the factory supplied by your transformation context instead. */
+    ts.updateAwait = ts.Debug.deprecate(ts.factory.updateAwaitExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createPrefix` or the factory supplied by your transformation context instead. */
+    ts.createPrefix = ts.Debug.deprecate(ts.factory.createPrefixUnaryExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updatePrefix` or the factory supplied by your transformation context instead. */
+    ts.updatePrefix = ts.Debug.deprecate(ts.factory.updatePrefixUnaryExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createPostfix` or the factory supplied by your transformation context instead. */
+    ts.createPostfix = ts.Debug.deprecate(ts.factory.createPostfixUnaryExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updatePostfix` or the factory supplied by your transformation context instead. */
+    ts.updatePostfix = ts.Debug.deprecate(ts.factory.updatePostfixUnaryExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createBinary` or the factory supplied by your transformation context instead. */
+    ts.createBinary = ts.Debug.deprecate(ts.factory.createBinaryExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateConditional` or the factory supplied by your transformation context instead. */
+    ts.updateConditional = ts.Debug.deprecate(ts.factory.updateConditionalExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createTemplateExpression` or the factory supplied by your transformation context instead. */
+    ts.createTemplateExpression = ts.Debug.deprecate(ts.factory.createTemplateExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateTemplateExpression` or the factory supplied by your transformation context instead. */
+    ts.updateTemplateExpression = ts.Debug.deprecate(ts.factory.updateTemplateExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createTemplateHead` or the factory supplied by your transformation context instead. */
+    ts.createTemplateHead = ts.Debug.deprecate(ts.factory.createTemplateHead, factoryDeprecation);
+    /** @deprecated Use `factory.createTemplateMiddle` or the factory supplied by your transformation context instead. */
+    ts.createTemplateMiddle = ts.Debug.deprecate(ts.factory.createTemplateMiddle, factoryDeprecation);
+    /** @deprecated Use `factory.createTemplateTail` or the factory supplied by your transformation context instead. */
+    ts.createTemplateTail = ts.Debug.deprecate(ts.factory.createTemplateTail, factoryDeprecation);
+    /** @deprecated Use `factory.createNoSubstitutionTemplateLiteral` or the factory supplied by your transformation context instead. */
+    ts.createNoSubstitutionTemplateLiteral = ts.Debug.deprecate(ts.factory.createNoSubstitutionTemplateLiteral, factoryDeprecation);
+    /** @deprecated Use `factory.updateYield` or the factory supplied by your transformation context instead. */
+    ts.updateYield = ts.Debug.deprecate(ts.factory.updateYieldExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createSpread` or the factory supplied by your transformation context instead. */
+    ts.createSpread = ts.Debug.deprecate(ts.factory.createSpreadElement, factoryDeprecation);
+    /** @deprecated Use `factory.updateSpread` or the factory supplied by your transformation context instead. */
+    ts.updateSpread = ts.Debug.deprecate(ts.factory.updateSpreadElement, factoryDeprecation);
+    /** @deprecated Use `factory.createOmittedExpression` or the factory supplied by your transformation context instead. */
+    ts.createOmittedExpression = ts.Debug.deprecate(ts.factory.createOmittedExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createAsExpression` or the factory supplied by your transformation context instead. */
+    ts.createAsExpression = ts.Debug.deprecate(ts.factory.createAsExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateAsExpression` or the factory supplied by your transformation context instead. */
+    ts.updateAsExpression = ts.Debug.deprecate(ts.factory.updateAsExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createNonNullExpression` or the factory supplied by your transformation context instead. */
+    ts.createNonNullExpression = ts.Debug.deprecate(ts.factory.createNonNullExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateNonNullExpression` or the factory supplied by your transformation context instead. */
+    ts.updateNonNullExpression = ts.Debug.deprecate(ts.factory.updateNonNullExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createNonNullChain` or the factory supplied by your transformation context instead. */
+    ts.createNonNullChain = ts.Debug.deprecate(ts.factory.createNonNullChain, factoryDeprecation);
+    /** @deprecated Use `factory.updateNonNullChain` or the factory supplied by your transformation context instead. */
+    ts.updateNonNullChain = ts.Debug.deprecate(ts.factory.updateNonNullChain, factoryDeprecation);
+    /** @deprecated Use `factory.createMetaProperty` or the factory supplied by your transformation context instead. */
+    ts.createMetaProperty = ts.Debug.deprecate(ts.factory.createMetaProperty, factoryDeprecation);
+    /** @deprecated Use `factory.updateMetaProperty` or the factory supplied by your transformation context instead. */
+    ts.updateMetaProperty = ts.Debug.deprecate(ts.factory.updateMetaProperty, factoryDeprecation);
+    /** @deprecated Use `factory.createTemplateSpan` or the factory supplied by your transformation context instead. */
+    ts.createTemplateSpan = ts.Debug.deprecate(ts.factory.createTemplateSpan, factoryDeprecation);
+    /** @deprecated Use `factory.updateTemplateSpan` or the factory supplied by your transformation context instead. */
+    ts.updateTemplateSpan = ts.Debug.deprecate(ts.factory.updateTemplateSpan, factoryDeprecation);
+    /** @deprecated Use `factory.createSemicolonClassElement` or the factory supplied by your transformation context instead. */
+    ts.createSemicolonClassElement = ts.Debug.deprecate(ts.factory.createSemicolonClassElement, factoryDeprecation);
+    /** @deprecated Use `factory.createBlock` or the factory supplied by your transformation context instead. */
+    ts.createBlock = ts.Debug.deprecate(ts.factory.createBlock, factoryDeprecation);
+    /** @deprecated Use `factory.updateBlock` or the factory supplied by your transformation context instead. */
+    ts.updateBlock = ts.Debug.deprecate(ts.factory.updateBlock, factoryDeprecation);
+    /** @deprecated Use `factory.createVariableStatement` or the factory supplied by your transformation context instead. */
+    ts.createVariableStatement = ts.Debug.deprecate(ts.factory.createVariableStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateVariableStatement` or the factory supplied by your transformation context instead. */
+    ts.updateVariableStatement = ts.Debug.deprecate(ts.factory.updateVariableStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createEmptyStatement` or the factory supplied by your transformation context instead. */
+    ts.createEmptyStatement = ts.Debug.deprecate(ts.factory.createEmptyStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createExpressionStatement` or the factory supplied by your transformation context instead. */
+    ts.createExpressionStatement = ts.Debug.deprecate(ts.factory.createExpressionStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateExpressionStatement` or the factory supplied by your transformation context instead. */
+    ts.updateExpressionStatement = ts.Debug.deprecate(ts.factory.updateExpressionStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createExpressionStatement` or the factory supplied by your transformation context instead. */
+    ts.createStatement = ts.Debug.deprecate(ts.factory.createExpressionStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateExpressionStatement` or the factory supplied by your transformation context instead. */
+    ts.updateStatement = ts.Debug.deprecate(ts.factory.updateExpressionStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createIf` or the factory supplied by your transformation context instead. */
+    ts.createIf = ts.Debug.deprecate(ts.factory.createIfStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateIf` or the factory supplied by your transformation context instead. */
+    ts.updateIf = ts.Debug.deprecate(ts.factory.updateIfStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createDo` or the factory supplied by your transformation context instead. */
+    ts.createDo = ts.Debug.deprecate(ts.factory.createDoStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateDo` or the factory supplied by your transformation context instead. */
+    ts.updateDo = ts.Debug.deprecate(ts.factory.updateDoStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createWhile` or the factory supplied by your transformation context instead. */
+    ts.createWhile = ts.Debug.deprecate(ts.factory.createWhileStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateWhile` or the factory supplied by your transformation context instead. */
+    ts.updateWhile = ts.Debug.deprecate(ts.factory.updateWhileStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createFor` or the factory supplied by your transformation context instead. */
+    ts.createFor = ts.Debug.deprecate(ts.factory.createForStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateFor` or the factory supplied by your transformation context instead. */
+    ts.updateFor = ts.Debug.deprecate(ts.factory.updateForStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createForIn` or the factory supplied by your transformation context instead. */
+    ts.createForIn = ts.Debug.deprecate(ts.factory.createForInStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateForIn` or the factory supplied by your transformation context instead. */
+    ts.updateForIn = ts.Debug.deprecate(ts.factory.updateForInStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createForOf` or the factory supplied by your transformation context instead. */
+    ts.createForOf = ts.Debug.deprecate(ts.factory.createForOfStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateForOf` or the factory supplied by your transformation context instead. */
+    ts.updateForOf = ts.Debug.deprecate(ts.factory.updateForOfStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createContinue` or the factory supplied by your transformation context instead. */
+    ts.createContinue = ts.Debug.deprecate(ts.factory.createContinueStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateContinue` or the factory supplied by your transformation context instead. */
+    ts.updateContinue = ts.Debug.deprecate(ts.factory.updateContinueStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createBreak` or the factory supplied by your transformation context instead. */
+    ts.createBreak = ts.Debug.deprecate(ts.factory.createBreakStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateBreak` or the factory supplied by your transformation context instead. */
+    ts.updateBreak = ts.Debug.deprecate(ts.factory.updateBreakStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createReturn` or the factory supplied by your transformation context instead. */
+    ts.createReturn = ts.Debug.deprecate(ts.factory.createReturnStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateReturn` or the factory supplied by your transformation context instead. */
+    ts.updateReturn = ts.Debug.deprecate(ts.factory.updateReturnStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createWith` or the factory supplied by your transformation context instead. */
+    ts.createWith = ts.Debug.deprecate(ts.factory.createWithStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateWith` or the factory supplied by your transformation context instead. */
+    ts.updateWith = ts.Debug.deprecate(ts.factory.updateWithStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createSwitch` or the factory supplied by your transformation context instead. */
+    ts.createSwitch = ts.Debug.deprecate(ts.factory.createSwitchStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateSwitch` or the factory supplied by your transformation context instead. */
+    ts.updateSwitch = ts.Debug.deprecate(ts.factory.updateSwitchStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createLabel` or the factory supplied by your transformation context instead. */
+    ts.createLabel = ts.Debug.deprecate(ts.factory.createLabeledStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateLabel` or the factory supplied by your transformation context instead. */
+    ts.updateLabel = ts.Debug.deprecate(ts.factory.updateLabeledStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createThrow` or the factory supplied by your transformation context instead. */
+    ts.createThrow = ts.Debug.deprecate(ts.factory.createThrowStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateThrow` or the factory supplied by your transformation context instead. */
+    ts.updateThrow = ts.Debug.deprecate(ts.factory.updateThrowStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createTry` or the factory supplied by your transformation context instead. */
+    ts.createTry = ts.Debug.deprecate(ts.factory.createTryStatement, factoryDeprecation);
+    /** @deprecated Use `factory.updateTry` or the factory supplied by your transformation context instead. */
+    ts.updateTry = ts.Debug.deprecate(ts.factory.updateTryStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createDebuggerStatement` or the factory supplied by your transformation context instead. */
+    ts.createDebuggerStatement = ts.Debug.deprecate(ts.factory.createDebuggerStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createVariableDeclarationList` or the factory supplied by your transformation context instead. */
+    ts.createVariableDeclarationList = ts.Debug.deprecate(ts.factory.createVariableDeclarationList, factoryDeprecation);
+    /** @deprecated Use `factory.updateVariableDeclarationList` or the factory supplied by your transformation context instead. */
+    ts.updateVariableDeclarationList = ts.Debug.deprecate(ts.factory.updateVariableDeclarationList, factoryDeprecation);
+    /** @deprecated Use `factory.createFunctionDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createFunctionDeclaration = ts.Debug.deprecate(ts.factory.createFunctionDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateFunctionDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateFunctionDeclaration = ts.Debug.deprecate(ts.factory.updateFunctionDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createClassDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createClassDeclaration = ts.Debug.deprecate(ts.factory.createClassDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateClassDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateClassDeclaration = ts.Debug.deprecate(ts.factory.updateClassDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createInterfaceDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createInterfaceDeclaration = ts.Debug.deprecate(ts.factory.createInterfaceDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateInterfaceDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateInterfaceDeclaration = ts.Debug.deprecate(ts.factory.updateInterfaceDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeAliasDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createTypeAliasDeclaration = ts.Debug.deprecate(ts.factory.createTypeAliasDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypeAliasDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateTypeAliasDeclaration = ts.Debug.deprecate(ts.factory.updateTypeAliasDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createEnumDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createEnumDeclaration = ts.Debug.deprecate(ts.factory.createEnumDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateEnumDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateEnumDeclaration = ts.Debug.deprecate(ts.factory.updateEnumDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createModuleDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createModuleDeclaration = ts.Debug.deprecate(ts.factory.createModuleDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateModuleDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateModuleDeclaration = ts.Debug.deprecate(ts.factory.updateModuleDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createModuleBlock` or the factory supplied by your transformation context instead. */
+    ts.createModuleBlock = ts.Debug.deprecate(ts.factory.createModuleBlock, factoryDeprecation);
+    /** @deprecated Use `factory.updateModuleBlock` or the factory supplied by your transformation context instead. */
+    ts.updateModuleBlock = ts.Debug.deprecate(ts.factory.updateModuleBlock, factoryDeprecation);
+    /** @deprecated Use `factory.createCaseBlock` or the factory supplied by your transformation context instead. */
+    ts.createCaseBlock = ts.Debug.deprecate(ts.factory.createCaseBlock, factoryDeprecation);
+    /** @deprecated Use `factory.updateCaseBlock` or the factory supplied by your transformation context instead. */
+    ts.updateCaseBlock = ts.Debug.deprecate(ts.factory.updateCaseBlock, factoryDeprecation);
+    /** @deprecated Use `factory.createNamespaceExportDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createNamespaceExportDeclaration = ts.Debug.deprecate(ts.factory.createNamespaceExportDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateNamespaceExportDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateNamespaceExportDeclaration = ts.Debug.deprecate(ts.factory.updateNamespaceExportDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createImportEqualsDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createImportEqualsDeclaration = ts.Debug.deprecate(ts.factory.createImportEqualsDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateImportEqualsDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateImportEqualsDeclaration = ts.Debug.deprecate(ts.factory.updateImportEqualsDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createImportDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createImportDeclaration = ts.Debug.deprecate(ts.factory.createImportDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.updateImportDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateImportDeclaration = ts.Debug.deprecate(ts.factory.updateImportDeclaration, factoryDeprecation);
+    /** @deprecated Use `factory.createNamespaceImport` or the factory supplied by your transformation context instead. */
+    ts.createNamespaceImport = ts.Debug.deprecate(ts.factory.createNamespaceImport, factoryDeprecation);
+    /** @deprecated Use `factory.updateNamespaceImport` or the factory supplied by your transformation context instead. */
+    ts.updateNamespaceImport = ts.Debug.deprecate(ts.factory.updateNamespaceImport, factoryDeprecation);
+    /** @deprecated Use `factory.createNamedImports` or the factory supplied by your transformation context instead. */
+    ts.createNamedImports = ts.Debug.deprecate(ts.factory.createNamedImports, factoryDeprecation);
+    /** @deprecated Use `factory.updateNamedImports` or the factory supplied by your transformation context instead. */
+    ts.updateNamedImports = ts.Debug.deprecate(ts.factory.updateNamedImports, factoryDeprecation);
+    /** @deprecated Use `factory.createImportSpecifier` or the factory supplied by your transformation context instead. */
+    ts.createImportSpecifier = ts.Debug.deprecate(ts.factory.createImportSpecifier, factoryDeprecation);
+    /** @deprecated Use `factory.updateImportSpecifier` or the factory supplied by your transformation context instead. */
+    ts.updateImportSpecifier = ts.Debug.deprecate(ts.factory.updateImportSpecifier, factoryDeprecation);
+    /** @deprecated Use `factory.createExportAssignment` or the factory supplied by your transformation context instead. */
+    ts.createExportAssignment = ts.Debug.deprecate(ts.factory.createExportAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.updateExportAssignment` or the factory supplied by your transformation context instead. */
+    ts.updateExportAssignment = ts.Debug.deprecate(ts.factory.updateExportAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.createNamedExports` or the factory supplied by your transformation context instead. */
+    ts.createNamedExports = ts.Debug.deprecate(ts.factory.createNamedExports, factoryDeprecation);
+    /** @deprecated Use `factory.updateNamedExports` or the factory supplied by your transformation context instead. */
+    ts.updateNamedExports = ts.Debug.deprecate(ts.factory.updateNamedExports, factoryDeprecation);
+    /** @deprecated Use `factory.createExportSpecifier` or the factory supplied by your transformation context instead. */
+    ts.createExportSpecifier = ts.Debug.deprecate(ts.factory.createExportSpecifier, factoryDeprecation);
+    /** @deprecated Use `factory.updateExportSpecifier` or the factory supplied by your transformation context instead. */
+    ts.updateExportSpecifier = ts.Debug.deprecate(ts.factory.updateExportSpecifier, factoryDeprecation);
+    /** @deprecated Use `factory.createExternalModuleReference` or the factory supplied by your transformation context instead. */
+    ts.createExternalModuleReference = ts.Debug.deprecate(ts.factory.createExternalModuleReference, factoryDeprecation);
+    /** @deprecated Use `factory.updateExternalModuleReference` or the factory supplied by your transformation context instead. */
+    ts.updateExternalModuleReference = ts.Debug.deprecate(ts.factory.updateExternalModuleReference, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocTypeExpression` or the factory supplied by your transformation context instead. */
+    ts.createJSDocTypeExpression = ts.Debug.deprecate(ts.factory.createJSDocTypeExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocTypeTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocTypeTag = ts.Debug.deprecate(ts.factory.createJSDocTypeTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocReturnTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocReturnTag = ts.Debug.deprecate(ts.factory.createJSDocReturnTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocThisTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocThisTag = ts.Debug.deprecate(ts.factory.createJSDocThisTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocComment` or the factory supplied by your transformation context instead. */
+    ts.createJSDocComment = ts.Debug.deprecate(ts.factory.createJSDocComment, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocParameterTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocParameterTag = ts.Debug.deprecate(ts.factory.createJSDocParameterTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocClassTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocClassTag = ts.Debug.deprecate(ts.factory.createJSDocClassTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocAugmentsTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocAugmentsTag = ts.Debug.deprecate(ts.factory.createJSDocAugmentsTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocEnumTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocEnumTag = ts.Debug.deprecate(ts.factory.createJSDocEnumTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocTemplateTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocTemplateTag = ts.Debug.deprecate(ts.factory.createJSDocTemplateTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocTypedefTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocTypedefTag = ts.Debug.deprecate(ts.factory.createJSDocTypedefTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocCallbackTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocCallbackTag = ts.Debug.deprecate(ts.factory.createJSDocCallbackTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocSignature` or the factory supplied by your transformation context instead. */
+    ts.createJSDocSignature = ts.Debug.deprecate(ts.factory.createJSDocSignature, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocPropertyTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocPropertyTag = ts.Debug.deprecate(ts.factory.createJSDocPropertyTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocTypeLiteral` or the factory supplied by your transformation context instead. */
+    ts.createJSDocTypeLiteral = ts.Debug.deprecate(ts.factory.createJSDocTypeLiteral, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocImplementsTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocImplementsTag = ts.Debug.deprecate(ts.factory.createJSDocImplementsTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocAuthorTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocAuthorTag = ts.Debug.deprecate(ts.factory.createJSDocAuthorTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocPublicTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocPublicTag = ts.Debug.deprecate(ts.factory.createJSDocPublicTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocPrivateTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocPrivateTag = ts.Debug.deprecate(ts.factory.createJSDocPrivateTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocProtectedTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocProtectedTag = ts.Debug.deprecate(ts.factory.createJSDocProtectedTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocReadonlyTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocReadonlyTag = ts.Debug.deprecate(ts.factory.createJSDocReadonlyTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocUnknownTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocTag = ts.Debug.deprecate(ts.factory.createJSDocUnknownTag, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxElement` or the factory supplied by your transformation context instead. */
+    ts.createJsxElement = ts.Debug.deprecate(ts.factory.createJsxElement, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxElement` or the factory supplied by your transformation context instead. */
+    ts.updateJsxElement = ts.Debug.deprecate(ts.factory.updateJsxElement, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxSelfClosingElement` or the factory supplied by your transformation context instead. */
+    ts.createJsxSelfClosingElement = ts.Debug.deprecate(ts.factory.createJsxSelfClosingElement, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxSelfClosingElement` or the factory supplied by your transformation context instead. */
+    ts.updateJsxSelfClosingElement = ts.Debug.deprecate(ts.factory.updateJsxSelfClosingElement, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxOpeningElement` or the factory supplied by your transformation context instead. */
+    ts.createJsxOpeningElement = ts.Debug.deprecate(ts.factory.createJsxOpeningElement, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxOpeningElement` or the factory supplied by your transformation context instead. */
+    ts.updateJsxOpeningElement = ts.Debug.deprecate(ts.factory.updateJsxOpeningElement, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxClosingElement` or the factory supplied by your transformation context instead. */
+    ts.createJsxClosingElement = ts.Debug.deprecate(ts.factory.createJsxClosingElement, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxClosingElement` or the factory supplied by your transformation context instead. */
+    ts.updateJsxClosingElement = ts.Debug.deprecate(ts.factory.updateJsxClosingElement, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxFragment` or the factory supplied by your transformation context instead. */
+    ts.createJsxFragment = ts.Debug.deprecate(ts.factory.createJsxFragment, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxText` or the factory supplied by your transformation context instead. */
+    ts.createJsxText = ts.Debug.deprecate(ts.factory.createJsxText, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxText` or the factory supplied by your transformation context instead. */
+    ts.updateJsxText = ts.Debug.deprecate(ts.factory.updateJsxText, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxOpeningFragment` or the factory supplied by your transformation context instead. */
+    ts.createJsxOpeningFragment = ts.Debug.deprecate(ts.factory.createJsxOpeningFragment, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxJsxClosingFragment` or the factory supplied by your transformation context instead. */
+    ts.createJsxJsxClosingFragment = ts.Debug.deprecate(ts.factory.createJsxJsxClosingFragment, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxFragment` or the factory supplied by your transformation context instead. */
+    ts.updateJsxFragment = ts.Debug.deprecate(ts.factory.updateJsxFragment, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxAttribute` or the factory supplied by your transformation context instead. */
+    ts.createJsxAttribute = ts.Debug.deprecate(ts.factory.createJsxAttribute, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxAttribute` or the factory supplied by your transformation context instead. */
+    ts.updateJsxAttribute = ts.Debug.deprecate(ts.factory.updateJsxAttribute, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxAttributes` or the factory supplied by your transformation context instead. */
+    ts.createJsxAttributes = ts.Debug.deprecate(ts.factory.createJsxAttributes, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxAttributes` or the factory supplied by your transformation context instead. */
+    ts.updateJsxAttributes = ts.Debug.deprecate(ts.factory.updateJsxAttributes, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxSpreadAttribute` or the factory supplied by your transformation context instead. */
+    ts.createJsxSpreadAttribute = ts.Debug.deprecate(ts.factory.createJsxSpreadAttribute, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxSpreadAttribute` or the factory supplied by your transformation context instead. */
+    ts.updateJsxSpreadAttribute = ts.Debug.deprecate(ts.factory.updateJsxSpreadAttribute, factoryDeprecation);
+    /** @deprecated Use `factory.createJsxExpression` or the factory supplied by your transformation context instead. */
+    ts.createJsxExpression = ts.Debug.deprecate(ts.factory.createJsxExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateJsxExpression` or the factory supplied by your transformation context instead. */
+    ts.updateJsxExpression = ts.Debug.deprecate(ts.factory.updateJsxExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createCaseClause` or the factory supplied by your transformation context instead. */
+    ts.createCaseClause = ts.Debug.deprecate(ts.factory.createCaseClause, factoryDeprecation);
+    /** @deprecated Use `factory.updateCaseClause` or the factory supplied by your transformation context instead. */
+    ts.updateCaseClause = ts.Debug.deprecate(ts.factory.updateCaseClause, factoryDeprecation);
+    /** @deprecated Use `factory.createDefaultClause` or the factory supplied by your transformation context instead. */
+    ts.createDefaultClause = ts.Debug.deprecate(ts.factory.createDefaultClause, factoryDeprecation);
+    /** @deprecated Use `factory.updateDefaultClause` or the factory supplied by your transformation context instead. */
+    ts.updateDefaultClause = ts.Debug.deprecate(ts.factory.updateDefaultClause, factoryDeprecation);
+    /** @deprecated Use `factory.createHeritageClause` or the factory supplied by your transformation context instead. */
+    ts.createHeritageClause = ts.Debug.deprecate(ts.factory.createHeritageClause, factoryDeprecation);
+    /** @deprecated Use `factory.updateHeritageClause` or the factory supplied by your transformation context instead. */
+    ts.updateHeritageClause = ts.Debug.deprecate(ts.factory.updateHeritageClause, factoryDeprecation);
+    /** @deprecated Use `factory.createCatchClause` or the factory supplied by your transformation context instead. */
+    ts.createCatchClause = ts.Debug.deprecate(ts.factory.createCatchClause, factoryDeprecation);
+    /** @deprecated Use `factory.updateCatchClause` or the factory supplied by your transformation context instead. */
+    ts.updateCatchClause = ts.Debug.deprecate(ts.factory.updateCatchClause, factoryDeprecation);
+    /** @deprecated Use `factory.createPropertyAssignment` or the factory supplied by your transformation context instead. */
+    ts.createPropertyAssignment = ts.Debug.deprecate(ts.factory.createPropertyAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.updatePropertyAssignment` or the factory supplied by your transformation context instead. */
+    ts.updatePropertyAssignment = ts.Debug.deprecate(ts.factory.updatePropertyAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.createShorthandPropertyAssignment` or the factory supplied by your transformation context instead. */
+    ts.createShorthandPropertyAssignment = ts.Debug.deprecate(ts.factory.createShorthandPropertyAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.updateShorthandPropertyAssignment` or the factory supplied by your transformation context instead. */
+    ts.updateShorthandPropertyAssignment = ts.Debug.deprecate(ts.factory.updateShorthandPropertyAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.createSpreadAssignment` or the factory supplied by your transformation context instead. */
+    ts.createSpreadAssignment = ts.Debug.deprecate(ts.factory.createSpreadAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.updateSpreadAssignment` or the factory supplied by your transformation context instead. */
+    ts.updateSpreadAssignment = ts.Debug.deprecate(ts.factory.updateSpreadAssignment, factoryDeprecation);
+    /** @deprecated Use `factory.createEnumMember` or the factory supplied by your transformation context instead. */
+    ts.createEnumMember = ts.Debug.deprecate(ts.factory.createEnumMember, factoryDeprecation);
+    /** @deprecated Use `factory.updateEnumMember` or the factory supplied by your transformation context instead. */
+    ts.updateEnumMember = ts.Debug.deprecate(ts.factory.updateEnumMember, factoryDeprecation);
+    /** @deprecated Use `factory.updateSourceFile` or the factory supplied by your transformation context instead. */
+    ts.updateSourceFileNode = ts.Debug.deprecate(ts.factory.updateSourceFile, factoryDeprecation);
+    /** @deprecated Use `factory.createNotEmittedStatement` or the factory supplied by your transformation context instead. */
+    ts.createNotEmittedStatement = ts.Debug.deprecate(ts.factory.createNotEmittedStatement, factoryDeprecation);
+    /** @deprecated Use `factory.createPartiallyEmittedExpression` or the factory supplied by your transformation context instead. */
+    ts.createPartiallyEmittedExpression = ts.Debug.deprecate(ts.factory.createPartiallyEmittedExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updatePartiallyEmittedExpression` or the factory supplied by your transformation context instead. */
+    ts.updatePartiallyEmittedExpression = ts.Debug.deprecate(ts.factory.updatePartiallyEmittedExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createCommaList` or the factory supplied by your transformation context instead. */
+    ts.createCommaList = ts.Debug.deprecate(ts.factory.createCommaListExpression, factoryDeprecation);
+    /** @deprecated Use `factory.updateCommaList` or the factory supplied by your transformation context instead. */
+    ts.updateCommaList = ts.Debug.deprecate(ts.factory.updateCommaListExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createBundle` or the factory supplied by your transformation context instead. */
+    ts.createBundle = ts.Debug.deprecate(ts.factory.createBundle, factoryDeprecation);
+    /** @deprecated Use `factory.updateBundle` or the factory supplied by your transformation context instead. */
+    ts.updateBundle = ts.Debug.deprecate(ts.factory.updateBundle, factoryDeprecation);
+    /** @deprecated Use `factory.createImmediatelyInvokedFunctionExpression` or the factory supplied by your transformation context instead. */
+    ts.createImmediatelyInvokedFunctionExpression = ts.Debug.deprecate(ts.factory.createImmediatelyInvokedFunctionExpression, factoryDeprecation);
+    /** @deprecated Use `factory.createImmediatelyInvokedArrowFunction` or the factory supplied by your transformation context instead. */
+    ts.createImmediatelyInvokedArrowFunction = ts.Debug.deprecate(ts.factory.createImmediatelyInvokedArrowFunction, factoryDeprecation);
+    /** @deprecated Use `factory.createVoidZero` or the factory supplied by your transformation context instead. */
+    ts.createVoidZero = ts.Debug.deprecate(ts.factory.createVoidZero, factoryDeprecation);
+    /** @deprecated Use `factory.createExportDefault` or the factory supplied by your transformation context instead. */
+    ts.createExportDefault = ts.Debug.deprecate(ts.factory.createExportDefault, factoryDeprecation);
+    /** @deprecated Use `factory.createExternalModuleExport` or the factory supplied by your transformation context instead. */
+    ts.createExternalModuleExport = ts.Debug.deprecate(ts.factory.createExternalModuleExport, factoryDeprecation);
+    /** @deprecated Use `factory.createNamespaceExport` or the factory supplied by your transformation context instead. */
+    ts.createNamespaceExport = ts.Debug.deprecate(ts.factory.createNamespaceExport, factoryDeprecation);
+    /** @deprecated Use `factory.updateNamespaceExport` or the factory supplied by your transformation context instead. */
+    ts.updateNamespaceExport = ts.Debug.deprecate(ts.factory.updateNamespaceExport, factoryDeprecation);
+    /** @deprecated Use `factory.createToken` or the factory supplied by your transformation context instead. */
+    ts.createToken = ts.Debug.deprecate(function createToken(kind) {
+        return ts.factory.createToken(kind);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createIdentifier` or the factory supplied by your transformation context instead. */
+    ts.createIdentifier = ts.Debug.deprecate(function createIdentifier(text) {
+        return ts.factory.createIdentifier(text, /*typeArguments*/ undefined, /*originalKeywordKind*/ undefined);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createTempVariable` or the factory supplied by your transformation context instead. */
+    ts.createTempVariable = ts.Debug.deprecate(function createTempVariable(recordTempVariable) {
+        return ts.factory.createTempVariable(recordTempVariable, /*reserveInNestedScopes*/ undefined);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.getGeneratedNameForNode` or the factory supplied by your transformation context instead. */
+    ts.getGeneratedNameForNode = ts.Debug.deprecate(function getGeneratedNameForNode(node) {
+        return ts.factory.getGeneratedNameForNode(node, /*flags*/ undefined);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createUniqueName(text, GeneratedIdentifierFlags.Optimistic)` or the factory supplied by your transformation context instead. */
+    ts.createOptimisticUniqueName = ts.Debug.deprecate(function createOptimisticUniqueName(text) {
+        return ts.factory.createUniqueName(text, 16 /* Optimistic */);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createUniqueName(text, GeneratedIdentifierFlags.Optimistic | GeneratedIdentifierFlags.FileLevel)` or the factory supplied by your transformation context instead. */
+    ts.createFileLevelUniqueName = ts.Debug.deprecate(function createFileLevelUniqueName(text) {
+        return ts.factory.createUniqueName(text, 16 /* Optimistic */ | 32 /* FileLevel */);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createIndexSignature` or the factory supplied by your transformation context instead. */
+    ts.createIndexSignature = ts.Debug.deprecate(function createIndexSignature(decorators, modifiers, parameters, type) {
+        return ts.factory.createIndexSignature(decorators, modifiers, parameters, type);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createTypePredicateNode` or the factory supplied by your transformation context instead. */
+    ts.createTypePredicateNode = ts.Debug.deprecate(function createTypePredicateNode(parameterName, type) {
+        return ts.factory.createTypePredicateNode(/*assertsModifier*/ undefined, parameterName, type);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateTypePredicateNode` or the factory supplied by your transformation context instead. */
+    ts.updateTypePredicateNode = ts.Debug.deprecate(function updateTypePredicateNode(node, parameterName, type) {
+        return ts.factory.updateTypePredicateNode(node, /*assertsModifier*/ undefined, parameterName, type);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createStringLiteral`, `factory.createStringLiteralFromNode`, `factory.createNumericLiteral`, `factory.createBigIntLiteral`, `factory.createTrue`, `factory.createFalse`, or the factory supplied by your transformation context instead. */
+    ts.createLiteral = ts.Debug.deprecate(function createLiteral(value) {
+        if (typeof value === "number") {
+            return ts.factory.createNumericLiteral(value);
+        }
+        // eslint-disable-next-line no-in-operator
+        if (typeof value === "object" && "base10Value" in value) { // PseudoBigInt
+            return ts.factory.createBigIntLiteral(value);
+        }
+        if (typeof value === "boolean") {
+            return value ? ts.factory.createTrue() : ts.factory.createFalse();
+        }
+        if (typeof value === "string") {
+            return ts.factory.createStringLiteral(value, /*isSingleQuote*/ undefined);
+        }
+        return ts.factory.createStringLiteralFromNode(value);
+    }, { since: "4.0", warnAfter: "4.1", message: "Use `factory.createStringLiteral`, `factory.createStringLiteralFromNode`, `factory.createNumericLiteral`, `factory.createBigIntLiteral`, `factory.createTrue`, `factory.createFalse`, or the factory supplied by your transformation context instead." });
+    /** @deprecated Use `factory.createMethodSignature` or the factory supplied by your transformation context instead. */
+    ts.createMethodSignature = ts.Debug.deprecate(function createMethodSignature(typeParameters, parameters, type, name, questionToken) {
+        return ts.factory.createMethodSignature(/*modifiers*/ undefined, name, questionToken, typeParameters, parameters, type);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateMethodSignature` or the factory supplied by your transformation context instead. */
+    ts.updateMethodSignature = ts.Debug.deprecate(function updateMethodSignature(node, typeParameters, parameters, type, name, questionToken) {
+        return ts.factory.updateMethodSignature(node, node.modifiers, name, questionToken, typeParameters, parameters, type);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createTypeOperatorNode` or the factory supplied by your transformation context instead. */
+    ts.createTypeOperatorNode = ts.Debug.deprecate(function createTypeOperatorNode(operatorOrType, type) {
+        var operator;
+        if (type) {
+            operator = operatorOrType;
+        }
+        else {
+            type = operatorOrType;
+            operator = 138 /* KeyOfKeyword */;
+        }
+        return ts.factory.createTypeOperatorNode(operator, type);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createTaggedTemplate` or the factory supplied by your transformation context instead. */
+    ts.createTaggedTemplate = ts.Debug.deprecate(function createTaggedTemplate(tag, typeArgumentsOrTemplate, template) {
+        var typeArguments;
+        if (template) {
+            typeArguments = typeArgumentsOrTemplate;
+        }
+        else {
+            template = typeArgumentsOrTemplate;
+        }
+        return ts.factory.createTaggedTemplateExpression(tag, typeArguments, template);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateTaggedTemplate` or the factory supplied by your transformation context instead. */
+    ts.updateTaggedTemplate = ts.Debug.deprecate(function updateTaggedTemplate(node, tag, typeArgumentsOrTemplate, template) {
+        var typeArguments;
+        if (template) {
+            typeArguments = typeArgumentsOrTemplate;
+        }
+        else {
+            template = typeArgumentsOrTemplate;
+        }
+        return ts.factory.updateTaggedTemplateExpression(node, tag, typeArguments, template);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateBinary` or the factory supplied by your transformation context instead. */
+    ts.updateBinary = ts.Debug.deprecate(function updateBinary(node, left, right, operator) {
+        if (operator === void 0) { operator = node.operatorToken; }
+        if (typeof operator === "number") {
+            operator = operator === node.operatorToken.kind ? node.operatorToken : ts.factory.createToken(operator);
+        }
+        return ts.factory.updateBinaryExpression(node, left, operator, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createConditional` or the factory supplied by your transformation context instead. */
+    ts.createConditional = ts.Debug.deprecate(function createConditional(condition, questionTokenOrWhenTrue, whenTrueOrWhenFalse, colonToken, whenFalse) {
+        return arguments.length === 5 ? ts.factory.createConditionalExpression(condition, questionTokenOrWhenTrue, whenTrueOrWhenFalse, colonToken, whenFalse) :
+            arguments.length === 3 ? ts.factory.createConditionalExpression(condition, ts.factory.createToken(57 /* QuestionToken */), questionTokenOrWhenTrue, ts.factory.createToken(58 /* ColonToken */), whenTrueOrWhenFalse) :
+                ts.Debug.fail("Argument count mismatch");
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createYield` or the factory supplied by your transformation context instead. */
+    ts.createYield = ts.Debug.deprecate(function createYield(asteriskTokenOrExpression, expression) {
+        var asteriskToken;
+        if (expression) {
+            asteriskToken = asteriskTokenOrExpression;
+        }
+        else {
+            expression = asteriskTokenOrExpression;
+        }
+        return ts.factory.createYieldExpression(asteriskToken, expression);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createClassExpression` or the factory supplied by your transformation context instead. */
+    ts.createClassExpression = ts.Debug.deprecate(function createClassExpression(modifiers, name, typeParameters, heritageClauses, members) {
+        return ts.factory.createClassExpression(/*decorators*/ undefined, modifiers, name, typeParameters, heritageClauses, members);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateClassExpression` or the factory supplied by your transformation context instead. */
+    ts.updateClassExpression = ts.Debug.deprecate(function updateClassExpression(node, modifiers, name, typeParameters, heritageClauses, members) {
+        return ts.factory.updateClassExpression(node, /*decorators*/ undefined, modifiers, name, typeParameters, heritageClauses, members);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createPropertySignature` or the factory supplied by your transformation context instead. */
+    ts.createPropertySignature = ts.Debug.deprecate(function createPropertySignature(modifiers, name, questionToken, type, initializer) {
+        var node = ts.factory.createPropertySignature(modifiers, name, questionToken, type);
+        node.initializer = initializer;
+        return node;
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updatePropertySignature` or the factory supplied by your transformation context instead. */
+    ts.updatePropertySignature = ts.Debug.deprecate(function updatePropertySignature(node, modifiers, name, questionToken, type, initializer) {
+        var updated = ts.factory.updatePropertySignature(node, modifiers, name, questionToken, type);
+        if (node.initializer !== initializer) {
+            if (updated === node) {
+                updated = ts.factory.cloneNode(node);
+            }
+            updated.initializer = initializer;
+        }
+        return updated;
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createExpressionWithTypeArguments` or the factory supplied by your transformation context instead. */
+    ts.createExpressionWithTypeArguments = ts.Debug.deprecate(function createExpressionWithTypeArguments(typeArguments, expression) {
+        return ts.factory.createExpressionWithTypeArguments(expression, typeArguments);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateExpressionWithTypeArguments` or the factory supplied by your transformation context instead. */
+    ts.updateExpressionWithTypeArguments = ts.Debug.deprecate(function updateExpressionWithTypeArguments(node, typeArguments, expression) {
+        return ts.factory.updateExpressionWithTypeArguments(node, expression, typeArguments);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createArrowFunction` or the factory supplied by your transformation context instead. */
+    ts.createArrowFunction = ts.Debug.deprecate(function createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanTokenOrBody, body) {
+        return arguments.length === 6 ? ts.factory.createArrowFunction(modifiers, typeParameters, parameters, type, equalsGreaterThanTokenOrBody, body) :
+            arguments.length === 5 ? ts.factory.createArrowFunction(modifiers, typeParameters, parameters, type, /*equalsGreaterThanToken*/ undefined, equalsGreaterThanTokenOrBody) :
+                ts.Debug.fail("Argument count mismatch");
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateArrowFunction` or the factory supplied by your transformation context instead. */
+    ts.updateArrowFunction = ts.Debug.deprecate(function updateArrowFunction(node, modifiers, typeParameters, parameters, type, equalsGreaterThanTokenOrBody, body) {
+        return arguments.length === 7 ? ts.factory.updateArrowFunction(node, modifiers, typeParameters, parameters, type, equalsGreaterThanTokenOrBody, body) :
+            arguments.length === 6 ? ts.factory.updateArrowFunction(node, modifiers, typeParameters, parameters, type, node.equalsGreaterThanToken, equalsGreaterThanTokenOrBody) :
+                ts.Debug.fail("Argument count mismatch");
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createVariableDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createVariableDeclaration = ts.Debug.deprecate(function createVariableDeclaration(name, exclamationTokenOrType, typeOrInitializer, initializer) {
+        return arguments.length === 4 ? ts.factory.createVariableDeclaration(name, exclamationTokenOrType, typeOrInitializer, initializer) :
+            arguments.length >= 1 && arguments.length <= 3 ? ts.factory.createVariableDeclaration(name, /*exclamationToken*/ undefined, exclamationTokenOrType, typeOrInitializer) :
+                ts.Debug.fail("Argument count mismatch");
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateVariableDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateVariableDeclaration = ts.Debug.deprecate(function updateVariableDeclaration(node, name, exclamationTokenOrType, typeOrInitializer, initializer) {
+        return arguments.length === 5 ? ts.factory.updateVariableDeclaration(node, name, exclamationTokenOrType, typeOrInitializer, initializer) :
+            arguments.length === 4 ? ts.factory.updateVariableDeclaration(node, name, node.exclamationToken, exclamationTokenOrType, typeOrInitializer) :
+                ts.Debug.fail("Argument count mismatch");
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createImportClause` or the factory supplied by your transformation context instead. */
+    ts.createImportClause = ts.Debug.deprecate(function createImportClause(name, namedBindings, isTypeOnly) {
+        if (isTypeOnly === void 0) { isTypeOnly = false; }
+        return ts.factory.createImportClause(isTypeOnly, name, namedBindings);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateImportClause` or the factory supplied by your transformation context instead. */
+    ts.updateImportClause = ts.Debug.deprecate(function updateImportClause(node, name, namedBindings, isTypeOnly) {
+        return ts.factory.updateImportClause(node, isTypeOnly, name, namedBindings);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createExportDeclaration` or the factory supplied by your transformation context instead. */
+    ts.createExportDeclaration = ts.Debug.deprecate(function createExportDeclaration(decorators, modifiers, exportClause, moduleSpecifier, isTypeOnly) {
+        if (isTypeOnly === void 0) { isTypeOnly = false; }
+        return ts.factory.createExportDeclaration(decorators, modifiers, isTypeOnly, exportClause, moduleSpecifier);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.updateExportDeclaration` or the factory supplied by your transformation context instead. */
+    ts.updateExportDeclaration = ts.Debug.deprecate(function updateExportDeclaration(node, decorators, modifiers, exportClause, moduleSpecifier, isTypeOnly) {
+        return ts.factory.updateExportDeclaration(node, decorators, modifiers, isTypeOnly, exportClause, moduleSpecifier);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createJSDocParameterTag` or the factory supplied by your transformation context instead. */
+    ts.createJSDocParamTag = ts.Debug.deprecate(function createJSDocParamTag(name, isBracketed, typeExpression, comment) {
+        return ts.factory.createJSDocParameterTag(/*tagName*/ undefined, name, isBracketed, typeExpression, /*isNameFirst*/ false, comment);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createComma` or the factory supplied by your transformation context instead. */
+    ts.createComma = ts.Debug.deprecate(function createComma(left, right) {
+        return ts.factory.createComma(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createLessThan` or the factory supplied by your transformation context instead. */
+    ts.createLessThan = ts.Debug.deprecate(function createLessThan(left, right) {
+        return ts.factory.createLessThan(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createAssignment` or the factory supplied by your transformation context instead. */
+    ts.createAssignment = ts.Debug.deprecate(function createAssignment(left, right) {
+        return ts.factory.createAssignment(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createStrictEquality` or the factory supplied by your transformation context instead. */
+    ts.createStrictEquality = ts.Debug.deprecate(function createStrictEquality(left, right) {
+        return ts.factory.createStrictEquality(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createStrictInequality` or the factory supplied by your transformation context instead. */
+    ts.createStrictInequality = ts.Debug.deprecate(function createStrictInequality(left, right) {
+        return ts.factory.createStrictInequality(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createAdd` or the factory supplied by your transformation context instead. */
+    ts.createAdd = ts.Debug.deprecate(function createAdd(left, right) {
+        return ts.factory.createAdd(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createSubtract` or the factory supplied by your transformation context instead. */
+    ts.createSubtract = ts.Debug.deprecate(function createSubtract(left, right) {
+        return ts.factory.createSubtract(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createLogicalAnd` or the factory supplied by your transformation context instead. */
+    ts.createLogicalAnd = ts.Debug.deprecate(function createLogicalAnd(left, right) {
+        return ts.factory.createLogicalAnd(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createLogicalOr` or the factory supplied by your transformation context instead. */
+    ts.createLogicalOr = ts.Debug.deprecate(function createLogicalOr(left, right) {
+        return ts.factory.createLogicalOr(left, right);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createPostfixIncrement` or the factory supplied by your transformation context instead. */
+    ts.createPostfixIncrement = ts.Debug.deprecate(function createPostfixIncrement(operand) {
+        return ts.factory.createPostfixIncrement(operand);
+    }, factoryDeprecation);
+    /** @deprecated Use `factory.createLogicalNot` or the factory supplied by your transformation context instead. */
+    ts.createLogicalNot = ts.Debug.deprecate(function createLogicalNot(operand) {
+        return ts.factory.createLogicalNot(operand);
+    }, factoryDeprecation);
+    /** @deprecated Use an appropriate `factory` method instead. */
+    ts.createNode = ts.Debug.deprecate(function createNode(kind, pos, end) {
+        if (pos === void 0) { pos = 0; }
+        if (end === void 0) { end = 0; }
+        return ts.setTextRangePosEnd(kind === 297 /* SourceFile */ ? ts.parseBaseNodeFactory.createBaseSourceFileNode(kind) :
+            kind === 78 /* Identifier */ ? ts.parseBaseNodeFactory.createBaseIdentifierNode(kind) :
+                kind === 79 /* PrivateIdentifier */ ? ts.parseBaseNodeFactory.createBasePrivateIdentifierNode(kind) :
+                    !ts.isNodeKind(kind) ? ts.parseBaseNodeFactory.createBaseTokenNode(kind) :
+                        ts.parseBaseNodeFactory.createBaseNode(kind), pos, end);
+    }, { since: "4.0", warnAfter: "4.1", message: "Use an appropriate `factory` method instead." });
+    /**
+     * Creates a shallow, memberwise clone of a node ~for mutation~ with its `pos`, `end`, and `parent` set.
+     *
+     * NOTE: It is unsafe to change any properties of a `Node` that relate to its AST children, as those changes won't be
+     * captured with respect to transformations.
+     *
+     * @deprecated Use `factory.cloneNode` instead and use `setCommentRange` or `setSourceMapRange` and avoid setting `parent`.
+     */
+    ts.getMutableClone = ts.Debug.deprecate(function getMutableClone(node) {
+        var clone = ts.factory.cloneNode(node);
+        ts.setTextRange(clone, node);
+        ts.setParent(clone, node.parent);
+        return clone;
+    }, { since: "4.0", warnAfter: "4.1", message: "Use `factory.cloneNode` instead and use `setCommentRange` or `setSourceMapRange` and avoid setting `parent`." });
+    // #endregion Node Factory top-level exports
+    // DEPRECATION: Renamed node tests
+    // DEPRECATION PLAN:
+    //     - soft: 4.0
+    //     - warn: 4.1
+    //     - error: TBD
+    // #region Renamed node Tests
+    /** @deprecated Use `isTypeAssertionExpression` instead. */
+    ts.isTypeAssertion = ts.Debug.deprecate(function isTypeAssertion(node) {
+        return node.kind === 206 /* TypeAssertionExpression */;
+    }, {
+        since: "4.0",
+        warnAfter: "4.1",
+        message: "Use `isTypeAssertionExpression` instead."
+    });
+    // #endregion
+})(ts || (ts = {}));
 
 //# sourceMappingURL=tsserverlibrary.js.map